From df60ba6ada279e6cfd0b357dcf4537fe0147fac8 Mon Sep 17 00:00:00 2001 From: opi Date: Mon, 17 Feb 2014 13:07:28 +0100 Subject: [PATCH] YNH Panel. 1st draft. --- access.lua | 3 + portal/assets/css/ynhpanel.css | 172 ++++++++++++++++++++++++++++ portal/assets/js/ynhpanel.js | 200 +++++++++++++++++++++++++++++---- 3 files changed, 351 insertions(+), 24 deletions(-) create mode 100644 portal/assets/css/ynhpanel.css diff --git a/access.lua b/access.lua index 9495b0f..0505024 100644 --- a/access.lua +++ b/access.lua @@ -613,6 +613,9 @@ if is_logged_in() then if string.match(ngx.var.uri, "^/ynhpanel.js$") then serve("/ynhsso/assets/js/ynhpanel.js") end + if string.match(ngx.var.uri, "^/ynhpanel.css$") then + serve("/ynhsso/assets/css/ynhpanel.css") + end if string.match(ngx.var.uri, "^/ynhpanel.json$") then serve("/ynhsso/assets/js/ynhpanel.json") end diff --git a/portal/assets/css/ynhpanel.css b/portal/assets/css/ynhpanel.css new file mode 100644 index 0000000..8e4a637 --- /dev/null +++ b/portal/assets/css/ynhpanel.css @@ -0,0 +1,172 @@ +/* ****************************************************************** + Button +******************************************************************* */ +#ynhportal { + display: block; + position: fixed; + z-index: 1000; + bottom: 2%; + right: 2%; + + width: 60px; + height: 52px; + padding: 10px; + border-radius: 5px; + background: #000; + background-image: url(); + background-position: center center; + background-repeat: no-repeat; + + opacity: 0.5; + zoom: 1;filter: alpha(opacity=50); +} +#ynhportal:hover { + opacity: 1; + filter: alpha(opacity=100); +} + + + +/* ****************************************************************** + Overlay +******************************************************************* */ + +/* Background */ +#ynhoverlay { + display: block; + position: absolute; + top: -100%; + left: 0; + width: 100%; + height: 0; + z-index: 999; + text-align: left; + overflow-y: auto; + + opacity: 0; + zoom: 1;filter: alpha(opacity=0); + + color:#fff; + background-color: rgba(0,0,0, 0.7); + + font-family: Arial, sans-serif; + font-size: 16px; /* 100% */ + font-style: normal; + font-weight: normal; + font-variant: normal; + + -webkit-transition: all .25s ease-in-out; + -moz-transition: all .25s ease-in-out; + -ms-transition: all .25s ease-in-out; + -o-transition: all .25s ease-in-out; + transition: all .25s ease-in-out; +} +#ynhoverlay.visible { + top:0; + height:100%; + opacity: 1; + filter: alpha(opacity=100); +} + +/* reset inherited style */ +#ynhoverlay * { + margin: 0; + padding: 0; + border: 0; + vertical-align: baseline; + text-align: left; + line-height: 1; + + color: #fff; + font-family: Arial, sans-serif; + font-size: 16px; /* 100% */ + font-style: normal; + font-weight: normal; + font-variant: normal; + text-shadow: none; + + opacity: 1; + filter: alpha(opacity=100); +} + + +/* Close button */ +#ynhclose { + display: block; + float: right; + margin: 2%; + cursor: pointer; + z-index: 1002; + + font-weight: bold; + font-size: 1.5em; +} +#ynhclose:hover {color:pink;} + + +/* Header */ +#ynhoverlay .header { + display: block; + width:50%; + margin:1em auto; +} +#ynhoverlay h1 { + display: block; + margin:0 1em 0 0; + line-height: 1.4; + font-size: 3em; +} +#ynhoverlay .account-link { + display: inline-block; + text-decoration: none; + color: #ccc; +} +#ynhoverlay .account-link:hover { + text-decoration: underline; + color: #fff; +} + + + +/* Application list */ +#ynhoverlay ul { + margin: 10% auto; + display: block; + width: 50%; + list-style: none; +} + #ynhoverlay li {display: inline;} + + #ynhoverlay ul a { + display: inline-block; + vertical-align: top; + width: 110px; + margin: 3% 6% 3% 0; + + text-align: center; + text-decoration: none; + + border-radius: 5px; + } + #ynhoverlay ul a:before { + content: attr(data-first-letter); + display: block; + height: 110px; + line-height: 110px; + margin-bottom:5%; + + color: #000; + background: #fff; + text-align: center; + text-decoration: none; + font-size: 3em; + + border-radius: 5px; + } + + #ynhoverlay ul a:hover { + color:pink; + } + #ynhoverlay ul a:hover:before { + background-color: pink; + } diff --git a/portal/assets/js/ynhpanel.js b/portal/assets/js/ynhpanel.js index 9c1b53f..3b8366c 100644 --- a/portal/assets/js/ynhpanel.js +++ b/portal/assets/js/ynhpanel.js @@ -1,39 +1,191 @@ -// Smallest DOMReady -// http://dustindiaz.com/smallest-domready-ever +/* ---------------------------------------------------------- + Utilities +---------------------------------------------------------- */ + +/* Console log fix +-------------------------- */ +if (typeof(console) === 'undefined') { + var console = {}; + console.log = console.error = console.info = console.debug = console.warn = console.trace = console.dir = console.dirxml = console.group = console.groupEnd = console.time = console.timeEnd = console.assert = console.profile = function() {}; +} + + +/* Array utilities + https://github.com/Darklg/JavaScriptUtilities/blob/master/assets/js/vanilla-js/libs/vanilla-arrays.js +-------------------------- */ +Array.contains = function(needle, haystack) { + var i = 0, + length = haystack.length; + + for (; i < length; i++) { + if (haystack[i] === needle) return true; + } + return false; +}; +Array.each = function(arrayToParse, callback) { + var i = 0, + length = arrayToParse.length; + for (; i < length; i++) { + callback(arrayToParse[i]); + } +}; + + + +/* CSS classes utilities + https://github.com/Darklg/JavaScriptUtilities/blob/master/assets/js/vanilla-js/libs/vanilla-classes.js +-------------------------- */ +Element.getClassNames = function(element) { + var classNames = [], + elementClassName = element.className; + if (elementClassName !== '') { + elementClassName = elementClassName.replace(/\s+/g, ' '); + classNames = elementClassName.split(' '); + } + return classNames; +}; +Element.hasClass = function(element, className) { + if (element.classList) { + return element.classList.contains(className); + } + return Array.contains(className, Element.getClassNames(element)); +}; +Element.addClass = function(element, className) { + if (element.classList) { + element.classList.add(className); + return; + } + if (!Element.hasClass(element, className)) { + var elementClasses = Element.getClassNames(element); + elementClasses.push(className); + element.className = elementClasses.join(' '); + } +}; +Element.removeClass = function(element, className) { + if (element.classList) { + element.classList.remove(className); + return; + } + var elementClasses = Element.getClassNames(element); + var newElementClasses = []; + var i = 0, + arLength = elementClasses.length; + for (; i < arLength; i++) { + if (elementClasses[i] !== className) { + newElementClasses.push(elementClasses[i]); + } + } + element.className = newElementClasses.join(' '); +}; +Element.toggleClass = function(element, className) { + if (!Element.hasClass(element, className)) { + Element.addClass(element, className); + } + else { + Element.removeClass(element, className); + } +}; + + +/* Add Event + https://github.com/Darklg/JavaScriptUtilities/blob/master/assets/js/vanilla-js/libs/vanilla-events.js +-------------------------- */ +window.addEvent = function(el, eventName, callback) { + if (el.addEventListener) { + el.addEventListener(eventName, callback, false); + } + else if (el.attachEvent) { + el.attachEvent("on" + eventName, function(e) { + return callback.call(el, e); + }); + } +}; +window.eventPreventDefault = function(event) { + return (event.preventDefault) ? event.preventDefault() : event.returnValue = false; +}; + + +/* Smallest DOMReady + http://dustindiaz.com/smallest-domready-ever +-------------------------- */ function domReady(cb) { /in/.test(document.readyState) // in = loadINg ? setTimeout('domReady('+cb+')', 9) : cb(); } + +/* ---------------------------------------------------------- + Main +---------------------------------------------------------- */ domReady(function(){ - //console.log('DOM is ready !'); - //alert("YNH Portal"); + // Add portal stylesheet + var portalStyle = document.createElement("link"); + portalStyle.setAttribute("rel", "stylesheet"); + portalStyle.setAttribute("type", "text/css"); + portalStyle.setAttribute("href", '/ynhpanel.css'); + document.getElementsByTagName("head")[0].insertBefore(portalStyle, null); + + // Create portal link var portal = document.createElement('a'); + portal.setAttribute('id', 'ynhportal'); + portal.setAttribute('href', '/ynhsso/'); + document.body.insertBefore(portal, null); - portal.id = 'ynhportal'; - portal.href = '/ynhsso/'; + // Get user's app + var r = new XMLHttpRequest(); + r.open("GET", "/ynhpanel.json", true); + r.onreadystatechange = function () { + // Die if error + if (r.readyState != 4 || r.status != 200) return; - portal.style.display="block"; - portal.style.position="fixed"; - portal.style.zIndex="1000"; - portal.style.bottom="5%"; - portal.style.right="30px"; + // Response is JSON + response = JSON.parse(r.responseText); - portal.style.width="60px"; - portal.style.height="52px"; - portal.style.padding="10px"; - portal.style.borderRadius="5px"; - portal.style.background="#000"; - portal.style.backgroundImage = "url()"; - portal.style.backgroundPosition = "center center"; - portal.style.backgroundRepeat = "no-repeat"; + // Create overlay element + var overlay = document.createElement("div"); + overlay.setAttribute("id","ynhoverlay"); - portal.style.opacity = "0.35"; - portal.addEventListener('mouseover', function(){portal.style.opacity="1";}); - portal.addEventListener('mouseout', function(){portal.style.opacity="0.5";}); + // Append close button + var closeBtn = document.createElement("div"); + closeBtn.setAttribute("id","ynhclose"); + closeBtn.innerHTML = "X"; + overlay.insertBefore(closeBtn, null); + + // Add overlay header + overlay.innerHTML += '
' + + '

'+ response.user +'

' + + '' + + '
'; + + // Add application links + var links = []; + Array.each(response.app, function(app){ + links.push('
  • '+app.name+'
  • '); + }); + overlay.innerHTML += ''; + + // Add overlay to DOM + document.body.insertBefore(overlay, null); + + // Bind YNH Button + window.addEvent(portal, 'click', function(e){ + // Prevent default click + window.eventPreventDefault(e); + // Toggle overlay on YNHPortal button + Element.toggleClass(overlay, 'visible'); + }); + + // Bind close button + window.addEvent(document.getElementById('ynhclose'), 'click', function(e){ + // Prevent default click + window.eventPreventDefault(e); + // Hide overlay + Element.removeClass(overlay, 'visible'); + }); + + }; + r.send(); - document.body.insertBefore(portal); }); -