X-Git-Url: http://git.polytechnique.org/?a=blobdiff_plain;ds=sidebyside;f=htdocs%2Fjavascript%2Fxorg.js;h=47bf78b14dd064e7e24e97b5af849848cd231a9d;hb=cb6e7db980fcc5fcdd2dad5aa645cfdf98057cd4;hp=94d8a1eabebf19781aceb4e1d622d49e29a4126d;hpb=5ff39ae112406d05b4af08234d5b5dae5ac17863;p=platal.git diff --git a/htdocs/javascript/xorg.js b/htdocs/javascript/xorg.js index 94d8a1e..47bf78b 100644 --- a/htdocs/javascript/xorg.js +++ b/htdocs/javascript/xorg.js @@ -18,12 +18,10 @@ * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * ***************************************************************************/ -var is_IE = $.browser.msie; - // {{{ function getNow() var days = ['Dimanche', 'Lundi', 'Mardi', 'Mercredi', 'Jeudi', 'Vendredi', 'Samedi']; var months = ['janvier', 'février', 'mars', 'avril', 'mai', 'juin', 'juillet', - 'août', 'septembre', 'octobre', 'novembre', 'décembre'] + 'août', 'septembre', 'octobre', 'novembre', 'décembre']; function getNow() { var dt = new Date(); @@ -31,13 +29,16 @@ function getNow() { var mh = dt.getMonth(); var wd = dt.getDate(); var yr = dt.getYear(); - if (yr<1000) yr += 1900; var hr = dt.getHours(); var mi = dt.getMinutes(); + var se = dt.getSeconds(); + + if (yr<1000) { + yr += 1900; + } if (mi < 10) { mi = '0' + mi; } - var se = dt.getSeconds(); if (se < 10) { se = '0' + se; } @@ -51,8 +52,8 @@ function getNow() { function canAddSearchEngine() { - if (((typeof window.sidebar == "object") && $.isFunction(window.sidebar.addSearchEngine)) - || ((typeof window.external == "object") && $.isFunction(window.external.AddSearchProvider))) { + if (((typeof window.sidebar === "object") && $.isFunction(window.sidebar.addSearchEngine)) + || ((typeof window.external === "object") && $.isFunction(window.external.AddSearchProvider))) { return true; } return false; @@ -61,7 +62,7 @@ function canAddSearchEngine() function addSearchEngine() { var searchURI = "http://www.polytechnique.org/xorg.opensearch.xml"; - if ((typeof window.sidebar == "object") && $.isFunction(window.sidebar.addSearchEngine)) { + if ((typeof window.sidebar === "object") && $.isFunction(window.sidebar.addSearchEngine)) { window.sidebar.addSearchEngine( searchURI, "http://www.polytechnique.org/images/xorg.png", @@ -84,72 +85,77 @@ function addSearchEngine() // {{{ function goodiesPopup() -var __goodies_active = true; - -var __goodies_ical = { - default_title: 'Calendrier iCal', - sites: [ - {'url_prefix': '', - 'img': 'images/icons/calendar_view_day.gif', - 'title': 'Calendrier iCal'}, - {'url_prefix': 'http://www.google.com/calendar/render?cid=', - 'img': 'images/goodies/add-google-calendar.gif', - 'title': 'Ajouter à Google Calendar'}, - {'url_prefix': 'https://www.google.com/calendar/hosted/polytechnique.org/render?cid=', - 'img': 'images/goodies/add-google-calendar.gif', - 'title': 'Ajouter à Google Apps / Calendar'} - ] -}; - -var __goodies_rss = { - default_title: 'Fils RSS', - sites: [ - {'url_prefix': '', - 'img': 'images/icons/feed.gif', - 'title': 'Fil rss'}, - {'url_prefix': 'http://fusion.google.com/add?feedurl=', - 'img': 'images/goodies/add-google.gif', - 'alt': 'Add to Google', - 'title': 'Ajouter à iGoogle/Google Reader'}, - {'url_prefix': 'http://www.netvibes.com/subscribe.php?url=', - 'img': 'images/goodies/add-netvibes.gif', - 'title': 'Ajouter à Netvibes'}, - {'url_prefix': 'http://add.my.yahoo.com/content?.intl=fr&url=', - 'img': 'images/goodies/add-yahoo.gif', - 'alt': 'Add to My Yahoo!', - 'title': 'Ajouter à My Yahoo!'} - ] -}; - -function disableGoodiesPopups() { - __goodies_active = false; -} - -function goodiesPopup(node, goodies) { - var text = '
'; - for (var site in goodies.sites) { - var entry = goodies.sites[site]; - var s_alt = entry["alt"] ? entry["alt"] : ""; - var s_img = entry["img"]; - var s_title = entry["title"] ? entry["title"] : ""; - var s_url = entry["url_prefix"].length > 0 ? entry["url_prefix"] + escape(this.href) : this.href; - - text += '' + s_alt + '
'; - } - text += 'Plus de bonus ...
'; - - var title = node.title ? node.title : goodies.default_title; - - $(node) - .mouseover( - function() { - if (__goodies_active) { - return overlib(text, CAPTION, title, CLOSETEXT, 'Fermer', DELAY, 800, STICKY, WIDTH, 150); - } +(function($) { + var goodies = { + ical: { + default_title: 'Calendrier iCal', + sites: [ + {url_prefix: '', + img: 'images/icons/calendar_view_day.gif', + title: 'Calendrier iCal'}, + {url_prefix: 'http://www.google.com/calendar/render?cid=', + img: 'images/goodies/add-google-calendar.gif', + title: 'Ajouter à Google Calendar'}, + {url_prefix: 'https://www.google.com/calendar/hosted/polytechnique.org/render?cid=', + img: 'images/goodies/add-google-calendar.gif', + title: 'Ajouter à Google Apps / Calendar'} + ] + }, + + rss: { + default_title: 'Fils RSS', + sites: [ + {url_prefix: '', + img: 'images/icons/feed.gif', + title: 'Fil rss'}, + {url_prefix: 'http://fusion.google.com/add?feedurl=', + img: 'images/goodies/add-google.gif', + alt: 'Add to Google', + title: 'Ajouter à iGoogle/Google Reader'}, + {url_prefix: 'http://www.netvibes.com/subscribe.php?url=', + img: 'images/goodies/add-netvibes.gif', + title: 'Ajouter à Netvibes'}, + {url_prefix: 'http://add.my.yahoo.com/content?.intl=fr&url=', + img: 'images/goodies/add-yahoo.gif', + alt: 'Add to My Yahoo!', + title: 'Ajouter à My Yahoo!'} + ] + } + }; + + $.fn.extend({ + goodiesPopup: function goodiesPopup(type) { + var text = '
'; + var site; + var entry; + var s_alt; + var s_img; + var s_title; + var s_url; + var href = this.attr('href'); + + for (site in goodies[type].sites) { + entry = goodies[type].sites[site]; + s_alt = entry.alt || ""; + s_img = entry.img; + s_title = entry.title || ""; + s_url = entry.url_prefix.length > 0 ? entry.url_prefix + escape(href) : href; + + text += '' + s_alt + '
'; } - ) - .mouseout(nd); -} + text += 'Plus de bonus ...
'; + + return this.overlib({ + text: text, + caption: this.attr('title') || goodies.default_title, + close_text: 'Fermer', + delay: 800, + sticky: true, + width: 150 + }); + } + }); +}(jQuery)); // }}} // {{{ function auto_links() @@ -157,43 +163,47 @@ function goodiesPopup(node, goodies) { function auto_links() { var url = document.URL; var fqdn = url.replace(/^https?:\/\/([^\/]*)\/.*$/,'$1'); - var light = (url.indexOf('display=light') > url.indexOf('?')); - var resource_page = (url.indexOf('rss') > -1 || url.indexOf('ical') > -1); + var light = url.indexOf('display=light') > url.indexOf('?'); + var resource_page = url.contains('rss') || url.contains('ical'); $("a,link").each(function(i) { var node = $(this); var href = this.href; + var matches; + var rss; + var ical; + if(!href || node.hasClass('xdx') - || href.indexOf('mailto:') > -1 || href.indexOf('javascript:') > -1) { + || href.startsWith('mailto:') || href.startsWith('javascript:')) { return; } - if ((href.indexOf(fqdn) < 0 && this.className.indexOf('popup') < 0) || node.hasClass('popup')) { + if ((!href.contains(fqdn) && !this.className.contains('popup')) || node.hasClass('popup')) { node.click(function () { - window.open(href); + window.open(this.href); return false; }); } - if (href.indexOf(fqdn) > -1 && light) { + if (href.contains(fqdn) && light) { href = href.replace(/([^\#\?]*)\??([^\#]*)(\#.*)?/, "$1?display=light&$2$3"); this.href = href; } - var rss = href.indexOf('rss') > -1; - var ical = href.indexOf('ical') > -1; + rss = href.contains('rss'); + ical = href.contains('ical'); if (rss || ical) { - if (href.indexOf('http') < 0) { + if (!href.startsWith('http')) { href = 'http://' + fqdn + '/' + href; } } - if (this.nodeName.toLowerCase() == 'a' && !resource_page) { - if (rss && href.indexOf('prefs/rss') < 0 && (href.indexOf('xml') > -1 || href.indexOf('hash'))) { - goodiesPopup(this, __goodies_rss); + if (this.nodeName.toLowerCase() === 'a' && !resource_page) { + if (rss && !href.contains('prefs/rss') && (href.contains('xml') || href.contains('hash'))) { + node.goodiesPopup('rss'); } else if (ical) { - goodiesPopup(this, __goodies_ical); + node.goodiesPopup('ical'); } } - if(matches = (/^popup_([0-9]*)x([0-9]*)$/).exec(this.className)) { - var w = matches[1], h = matches[2]; - node.popWin(w, h); + matches = /^popup_([0-9]*)x([0-9]*)$/.exec(this.className); + if (matches) { + node.popWin(matches[1], matches[2]); } }); $('.popup2').popWin(840, 600); @@ -244,7 +254,7 @@ function str_hmac_sha1(key, data){ return binb2str(core_hmac_sha1(key, data));} */ function sha1_vm_test() { - return hex_sha1("abc") == "a9993e364706816aba3e25717850c26c9cd0d89d"; + return hex_sha1("abc") === "a9993e364706816aba3e25717850c26c9cd0d89d"; } /* @@ -252,30 +262,34 @@ function sha1_vm_test() */ function core_sha1(x, len) { + var w, a, b, c, d, e; + var olda, oldb, oldc, oldd, olde; + var i, j, t; + /* append padding */ x[len >> 5] |= 0x80 << (24 - len % 32); x[((len + 64 >> 9) << 4) + 15] = len; - var w = Array(80); - var a = 1732584193; - var b = -271733879; - var c = -1732584194; - var d = 271733878; - var e = -1009589776; + w = Array(80); + a = 1732584193; + b = -271733879; + c = -1732584194; + d = 271733878; + e = -1009589776; - for(var i = 0; i < x.length; i += 16) + for(i = 0; i < x.length; i += 16) { - var olda = a; - var oldb = b; - var oldc = c; - var oldd = d; - var olde = e; + olda = a; + oldb = b; + oldc = c; + oldd = d; + olde = e; - for(var j = 0; j < 80; j++) + for(j = 0; j < 80; j++) { if(j < 16) w[j] = x[i + j]; else w[j] = rol(w[j-3] ^ w[j-8] ^ w[j-14] ^ w[j-16], 1); - var t = safe_add(safe_add(rol(a, 5), sha1_ft(j, b, c, d)), + t = safe_add(safe_add(rol(a, 5), sha1_ft(j, b, c, d)), safe_add(safe_add(e, w[j]), sha1_kt(j))); e = d; d = c; @@ -321,16 +335,20 @@ function sha1_kt(t) function core_hmac_sha1(key, data) { var bkey = str2binb(key); + var i, ipad, opad; + var hash; + if(bkey.length > 16) bkey = core_sha1(bkey, key.length * chrsz); - var ipad = Array(16), opad = Array(16); - for(var i = 0; i < 16; i++) + ipad = Array(16); + opad = Array(16); + for(i = 0; i < 16; i++) { ipad[i] = bkey[i] ^ 0x36363636; opad[i] = bkey[i] ^ 0x5C5C5C5C; } - var hash = core_sha1(ipad.concat(str2binb(data)), 512 + data.length * chrsz); + hash = core_sha1(ipad.concat(str2binb(data)), 512 + data.length * chrsz); return core_sha1(opad.concat(hash), 512 + 160); } @@ -361,7 +379,8 @@ function str2binb(str) { var bin = Array(); var mask = (1 << chrsz) - 1; - for(var i = 0; i < str.length * chrsz; i += chrsz) + var i; + for(i = 0; i < str.length * chrsz; i += chrsz) bin[i>>5] |= (str.charCodeAt(i / chrsz) & mask) << (32 - chrsz - i%32); return bin; } @@ -373,7 +392,8 @@ function binb2str(bin) { var str = ""; var mask = (1 << chrsz) - 1; - for(var i = 0; i < bin.length * 32; i += chrsz) + var i; + for(i = 0; i < bin.length * 32; i += chrsz) str += String.fromCharCode((bin[i>>5] >>> (32 - chrsz - i%32)) & mask); return str; } @@ -385,7 +405,8 @@ function binb2hex(binarray) { var hex_tab = hexcase ? "0123456789ABCDEF" : "0123456789abcdef"; var str = ""; - for(var i = 0; i < binarray.length * 4; i++) + var i; + for(i = 0; i < binarray.length * 4; i++) { str += hex_tab.charAt((binarray[i>>2] >> ((3 - i%4)*8+4)) & 0xF) + hex_tab.charAt((binarray[i>>2] >> ((3 - i%4)*8 )) & 0xF); @@ -400,12 +421,13 @@ function binb2b64(binarray) { var tab = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; var str = ""; - for(var i = 0; i < binarray.length * 4; i += 3) + var i, j, triplet; + for(i = 0; i < binarray.length * 4; i += 3) { - var triplet = (((binarray[i >> 2] >> 8 * (3 - i %4)) & 0xFF) << 16) + triplet = (((binarray[i >> 2] >> 8 * (3 - i %4)) & 0xFF) << 16) | (((binarray[i+1 >> 2] >> 8 * (3 - (i+1)%4)) & 0xFF) << 8 ) | ((binarray[i+2 >> 2] >> 8 * (3 - (i+2)%4)) & 0xFF); - for(var j = 0; j < 4; j++) + for(j = 0; j < 4; j++) { if(i * 8 + j * 6 > binarray.length * 32) str += b64pad; else str += tab.charAt((triplet >> 6*(3-j)) & 0x3F); @@ -431,23 +453,23 @@ function hexdec(a) { } function hash_xor(a, b) { - var c,i,j,k; + var c,i,j,k,d; c = ""; i = a.length; j = b.length; if (i < j) { - var d; d = a; a = b; b = d; k = i; i = j; j = k; } - for (k = 0; k < j; k++) + for (k = 0; k < j; k++) { c += dechex(hexdec(a.charAt(k)) ^ hexdec(b.charAt(k))); - for (; k < i; k++) + } + for (; k < i; k++) { c += a.charAt(k); + } return c; } - function getType(c) { if (c >= 'a' && c <= 'z') { return 1; @@ -462,10 +484,11 @@ function getType(c) { function differentTypes(password) { var prev = 0; + var type; for (i = 0 ; i < password.length ; ++i) { - var type = getType(password.charAt(i)); - if (prev != 0 && prev != type) { + type = getType(password.charAt(i)); + if (prev !== 0 && prev !== type) { return true; } prev = type; @@ -478,15 +501,16 @@ function passwordStrength(password) { var prev = 0; var firstType = true; var types = Array(0, 0, 0, 0, 0); + var type; for (i = 0 ; i < password.length ; ++i) { - var type = getType(password.charAt(i)); - if (prev != 0 && prev != type) { + type = getType(password.charAt(i)); + if (prev !== 0 && prev !== type) { prop += 5; firstType = false; } prop += i; - if (types[type] == 0 && !firstType) { + if (types[type] === 0 && !firstType) { prop += 15; } types[type]++; @@ -510,6 +534,7 @@ function passwordStrength(password) { function checkPassword(box, okLabel) { var password = box.value; var prop = passwordStrength(password); + var submitButton; if (prop >= 60) { color = "#4f4"; @@ -531,7 +556,7 @@ function checkPassword(box, okLabel) { }, 750) .parent().stop() .animate({ backgroundColor: bgcolor }, 750); - var submitButton = $(":submit[name='" + passwordprompt_submit + "']"); + submitButton = $(":submit[name='" + passwordprompt_submit + "']"); if (ok && password.length >= 6 && differentTypes(password)) { submitButton.attr("value", okLabel); submitButton.removeAttr("disabled"); @@ -541,17 +566,18 @@ function checkPassword(box, okLabel) { } } -function hashResponse(password1, password2, hasConfirmation) { - pw1 = $('[name=' + password1 + ']').val(); +function hashResponse(password1, password2, hasConfirmation, doAuth) { + var pw1 = $('[name=' + password1 + ']').val(); + var pw2; if (hasConfirmation) { pw2 = $('[name=' + password2 + ']').val(); - if (pw1 != pw2) { + if (pw1 !== pw2) { alert("\nErreur : les deux champs ne sont pas identiques !"); return false; } $('[name=' + password2 + ']').val(''); - } else if (pw1 == '********') { + } else if (pw1 === '********') { return true; } @@ -564,27 +590,52 @@ function hashResponse(password1, password2, hasConfirmation) { return false; } - alert("Le mot de passe que tu as rentré va être chiffré avant de nous parvenir par Internet ! Ainsi il ne circulera pas en clair."); + alert("Le mot de passe va être chiffré avant de nous parvenir par Internet ! Ainsi il ne circulera pas en clair."); $('[name=' + password1 + ']').val(''); $('[name=pwhash]').val(hash_encrypt(pw1)); + + if (doAuth) { + $('[name=password]').val(pw1); + doChallengeResponse(); + } + return true; } function correctUserName() { var u = document.forms.login.username; + var mots; + // login with no space - if (u.value.indexOf(' ') < 0) return true; - var mots = u.value.split(' '); + if (!u.value.contains(' ')) { + return true; + } + mots = u.value.split(' '); // jean paul.du pont -> jean-paul.du-pont - if (u.value.indexOf('.') > 0) { u.value = mots.join('-'); return true; } + if (u.value.contains('.')) { + u.value = mots.join('-'); + return true; + } // jean dupont -> jean.dupont - if (mots.length == 2) { u.value = mots[0]+"."+mots[1]; return true; } + if (mots.length === 2) { + u.value = mots[0] + "." + mots[1]; + return true; + } // jean dupont 2001 -> jean.dupont.2001 - if (mots.length == 3 && mots[2] > 1920 && mots[2] < 3000) { u.value = mots.join('.'); return true; } + if (mots.length === 3 && mots[2] > 1920 && mots[2] < 3000) { + u.value = mots.join('.'); + return true; + } // jean de la vallee -> jean.de-la-vallee - if (mots[1].toUpperCase() == 'DE') { u.value = mots[0]+"."+mots.join('-').substr(mots[0].length+1); return true; } + if (mots[1].toUpperCase() === 'DE') { + u.value = mots[0] + "." + mots.join('-').substr(mots[0].length+1); + return true; + } // jean paul dupont -> jean-paul.dupont - if (mots.length == 3 && mots[0].toUpperCase() == 'JEAN') { u.value = mots[0]+"-"+mots[1]+"."+mots[2]; return true; } + if (mots.length === 3 && mots[0].toUpperCase() === 'JEAN') { + u.value = mots[0] + "-" + mots[1] + "." + mots[2]; + return true; + } alert('Ton email ne doit pas contenir de blanc.\nLe format standard est\n\nprenom.nom.promotion\n\nSi ton nom ou ton prenom est composé,\nsépare les mots par des -'); @@ -592,18 +643,21 @@ function correctUserName() { } function doChallengeResponse() { + var new_pass, old_pass, str; - if (!correctUserName()) return false; + if (!correctUserName()) { + return false; + } - var new_pass = hash_encrypt(document.forms.login.password.value); - var old_pass = hash_encrypt(document.forms.login.password.value.substr(0, 10)); + new_pass = hash_encrypt(document.forms.login.password.value); + old_pass = hash_encrypt(document.forms.login.password.value.substr(0, 10)); str = document.forms.login.username.value + ":" + new_pass + ":" + document.forms.loginsub.challenge.value; document.forms.loginsub.response.value = hash_encrypt(str); - if (new_pass != old_pass) { + if (new_pass !== old_pass) { document.forms.loginsub.xorpass.value = hash_xor(new_pass, old_pass); } document.forms.loginsub.username.value = document.forms.login.username.value; @@ -614,9 +668,7 @@ function doChallengeResponse() { } function doChallengeResponseLogged() { - var new_pass = hash_encrypt(document.forms.login.password.value); - - str = document.forms.loginsub.username.value + ":" + + var str = document.forms.loginsub.username.value + ":" + hash_encrypt(document.forms.login.password.value) + ":" + document.forms.loginsub.challenge.value; @@ -633,6 +685,166 @@ function sendTestEmail(token, hruid) { var url = 'emails/test'; var msg = "Un email a été envoyé avec succès"; + if (hruid) { + url += '/' + hruid; + msg += " sur l'adresse de " + hruid + "."; + } else { + msg += " sur ton addresse."; + } + $('#mail_sent').successMessage(url + '?token=' + token, msg); + return false; +} + +// }}} +// {{{ jQuery object extension + +(function($) { + /* Add new functions to jQuery namesapce */ + $.extend({ + /* The goal of the following functions is to provide an AJAX API that + * take a different callback in case of HTTP success code (2XX) and in + * other cases. + */ + + xajax: function(source, method, data, onSuccess, onError, type) { + /* Shift argument */ + if ($.isFunction(data)) { + type = type || onError; + onError = onSuccess; + onSuccess = data; + data = null; + } + if (onError != null && !$.isFunction(onError)) { + type = type || onError; + onError = null; + } + + function ajaxHandler(data, textStatus, xhr) { + if (textStatus == 'success') { + if (onSuccess) { + onSuccess(data, textStatus, xhr); + } + } else if (textStatus == 'error') { + if (onError) { + onError(data, textStatus, xhr); + } else { + alert("Une error s'est produite lors du traitement de la requête.\n" + + "Ta session a peut-être expiré"); + } + } + } + return $.ajax({ + url: source, + type: method, + success: ajaxHandler, + data : data, + dataType: type + }); + }, + + xget: function(source, data, onSuccess, onError, type) { + return $.xajax(source, 'GET', data, onSuccess, onError, type); + }, + + xgetJSON: function(source, data, onSuccess, onError) { + return $.xget(source, data, onSuccess, onError, 'json'); + }, + + xgetScript: function(source, onSuccess, onError) { + return $.xget(source, null, onSuccess, onError, 'script'); + }, + + xgetText: function(source, data, onSuccess, onError) { + return $.xget(source, data, onSuccess, onError, 'text'); + }, + + xpost: function(source, data, onSuccess, onError, type) { + return $.xajax(source, 'POST', data, onSuccess, onError, type); + } + }); + + /* Add new functions to jQuery objects */ + $.fn.extend({ + tmpMessage: function(message, success) { + if (success) { + this.html(" " + message) + .css('color', 'green'); + } else { + this.html(" " + message) + .css('color', 'red'); + } + return this.css('fontWeight', 'bold') + .show() + .delay(1000) + .fadeOut(500); + }, + + updateHtml: function(source, callback) { + var elements = this; + function handler(data) { + elements.html(data); + if (callback) { + callback(data); + } + } + $.xget(source, handler, 'text'); + return this; + }, + + successMessage: function(source, message) { + var elements = this; + $.xget(source, function() { + elements.tmpMessage(message, true); + }); + return this; + }, + + wiki: function(text, withTitle) { + if (text == '') { + return this.html(''); + } + var url = 'wiki_preview'; + if (!withTitle) { + url += '/notitile'; + } + var $this = this; + $.post(url, { text: text }, + function (data) { + $this.html(data); + }, 'text'); + return this; + }, + + popWin: function(w, h) { + return this.click(function() { + window.open(this.href, '_blank', + 'toolbar=0,location=0,directories=0,status=0,' + +'menubar=0,scrollbars=1,resizable=1,' + +'width='+w+',height='+h); + return false; + }); + } + }); +})(jQuery); + +// }}} +// {{{ preview wiki + +function previewWiki(idFrom, idTo, withTitle, idShow) +{ + $('#' + idTo).wiki($('#' + idFrom).val(), withTitle); + if (idShow != null) { + $('#' + idShow).show(); + } +} + +// }}} +// {{{ send test email + +function sendTestEmail(token, hruid) +{ + var url = 'emails/test'; + var msg = "Un email a été envoyé avec succès"; if (hruid != null) { url += '/' + hruid; msg += " sur l'adresse de " + hruid + "."; @@ -645,6 +857,373 @@ function sendTestEmail(token, hruid) // }}} +/*************************************************************************** + * Quick search + */ + +/* quick search {{{ */ +(function($) { + function findPos(obj) { + var curleft = obj.offsetLeft || 0; + var curtop = obj.offsetTop || 0; + while (obj = obj.offsetParent) { + curleft += obj.offsetLeft + curtop += obj.offsetTop + } + return {x:curleft,y:curtop}; + } + + $.template('quickMinifiche', + '
' + + '
' + + '
${directory_name}
' + + '
' + + '{{if is_female}}•{{/if}}${directory_name}' + + '
' + + '
${promo}
' + + '
' + + '
' + + '
' + + '
'); + + + function buildPopup(input, destination, linkBindFunction) + { + var $popup = destination; + var selected = null; + var hovered = 0; + + function updateSelection() + { + var sel = $popup.children('.contact').addClass('grayed'); + if (selected !== null) { + while (selected < 0) { + selected += sel.length; + } + if (selected >= sel.length) { + selected -= sel.length; + } + sel.eq(selected).removeClass('grayed'); + } + } + + function formatProfile(i, profile) { + var data = $.tmpl('quickMinifiche', profile) + .css('cursor', 'pointer') + .hover(function() { + selected = i; + updateSelection(); + hovered++; + }, function() { + if (selected === i) { + selected = null; + updateSelection(); + } + hovered--; + }).mouseup(function() { + var sel = $(this).find('a'); + if (!sel.attr('hovered')) { + sel.click(); + } + }); + data.find('a').hover(function() { $(this).attr('hovered', true) }, + function() { $(this).attr('hovered', false) }); + return data; + } + + if (!$popup) { + $popup = $('
').hide() + .addClass('contact-list') + .css({ + position: 'absolute', + width: '300px', + top: input.css('bottom'), + clear: 'both', + 'text-align': 'left' + }); + input.after($popup); + } + + return { + hide: function(ignoreIfHover) { + if (ignoreIfHover && hovered !== 0) { + return true; + } + selected = null; + updateSelection(); + $popup.hide(); + return true; + }, + + show: function() { + var pos = findPos(input.get(0)); + $popup.css('left', pos.x - 300 + input.width()).show(); + return true; + }, + + selected: function() { + return selected !== null; + }, + + unselect: function() { + selected = null; + updateSelection(); + }, + + selectNext: function() { + if (selected === null) { + selected = 0; + } else { + selected++; + } + updateSelection(); + return true; + }, + + selectPrev: function() { + if (selected === null) { + selected = -1; + } else { + selected--; + } + updateSelection(); + return true; + }, + + activeCurrent: function() { + var sel = $popup.children('.contact'); + if (selected !== null) { + sel.eq(selected).find('a').click(); + return false; + } + return true; + }, + + updateContent: function(profiles, extra) { + var profile; + var $this; + $popup.empty(); + for (var i = 0, len = profiles.length; i < len; i++) { + (function(elt) { + var profile = formatProfile(i, elt); + profile.find('a').each(function() { + linkBindFunction.call(this, elt, $this, extra); + }); + profile.appendTo($popup); + }(profiles[i])); + } + if (len === 1) { + selected = 0; + } else { + selected = null; + } + updateSelection(); + if (len > 0) { + this.show(); + } else { + this.hide(); + } + return true; + } + }; + } + + $.fn.extend({ + quickSearch: function(options) { + return this.each(function() { + var $this = $(this); + var $input = this; + var $popup; + var previous = null; + var pending = false; + var disabled = false; + var updatePopup; + + options = options || { }; + options = $.extend({ + destination: null, + minChars: 3, + shortChars: 5, + shortTimeout: 300, + longTimeout: 100, + queryParams: { + offset: 0, + count: 10, + allow_special: true + }, + loadingClassLeft: 'ac_loading', + loadingClassRight: 'ac_loading_left', + selectAction: function(profile, popup, extra) { + var type = extra.link_type || 'profile'; + switch (type) { + case 'profile': + $(this).attr('href', 'profile/' + profile.hrpid) + .popWin(840, 600) + .click(function() { $popup.hide(); return false; }); + break; + case 'admin': + $(this).attr('href', 'admin/user/' + profile.hrpid) + .click(function() { window.open($(this).attr('href')); return false }); + break; + } + } + }, options); + options.loadingClass = $this.css('text-align') === 'right' ? options.loadingClassRight + : options.loadingClassLeft; + $this.attr('autocomplete', 'off'); + $popup = buildPopup($this, options.destination, options.selectAction); + + function markPending() { + pending = true; + } + + function performUpdate(quick) + { + if (updatePopup === markPending) { + return true; + } + updatePopup = markPending; + $this.addClass(options.loadingClass); + $.xapi('search', $.extend({ 'quick': quick }, options.queryParams), function(data) { + if (data.profile_count > options.queryParams.count || data.profile_count < 0) { + return $popup.hide(); + } + $popup.updateContent(data.profiles, data); + previous = quick; + }, function(data, text) { + if (text !== 'abort') { + disabled = true; + } + }).complete(function() { + $this.removeClass(options.loadingClass); + updatePopup = doUpdatePopup; + if (pending) { + updatePopup(); + } + }); + return true; + } + + function doUpdatePopup(dontDelay) + { + var quick = $this.val(); + if ($.isFunction(quick.trim)) { + quick = quick.trim(); + } + pending = false; + if (disabled || quick.length < options.minChars) { + previous = quick; + return $popup.hide(); + } else if (!dontDelay) { + var timeout = quick.length < options.shortChars ? options.shortTimeout : options.longTimeout; + setTimeout(function() { + updatePopup(true); + }, timeout); + return true; + } else if (previous === quick) { + return $popup.show(); + } + return performUpdate(quick); + } + + updatePopup = doUpdatePopup; + + return $this.keyup(function(e) { + if (e.keyCode !== 27 /* escape */ && e.keyCode !== 13 /* enter */ + && e.keyCode !== 9 /* tab */ && e.keyCode !== 38 /* up */ + && e.keyCode !== 40 /* down */) { + return updatePopup(); + } + return true; + }) + .keydown(function(e) { + switch (e.keyCode) { + case 9: /* Tab */ + case 40: /* Down */ + $popup.selectNext(); + return false; + + case 38: + $popup.selectPrev(); + return false; + + case 13: /* Return */ + return $popup.activeCurrent(); + + case 27: /* Escape */ + if ($popup.selected()) { + $popup.unselect(); + } else { + $popup.hide(); + } + return true; + } + return true; + }) + .blur(function() { + return $popup.hide(true); + }) + .focus(updatePopup);}); + } + }); +}(jQuery)); + +/*************************************************************************** + * Overlib made simple + */ + +(function($) { + $.fn.extend({ + overlib: function(text, width, height) { + var args = [ ]; + var key; + + if (typeof text === 'string') { + args.push(text); + if (width) { + args.push(WIDTH, width); + } + if (height) { + args.push(HEIGHT, height); + } + } else { + for (key in text) { + switch (key) { + case 'text': + args.unshift(text[key]); + break; + case 'caption': + args.push(CAPTION, text[key]); + break; + case 'close_text': + args.push(CLOSETEXT, text[key]); + break; + case 'delay': + args.push(DELAY, text[key]); + break; + case 'sticky': + if (text[key]) { + args.push(STICKY); + } + break; + case 'width': + args.push(WIDTH, text[key]); + break; + case 'height': + args.push(HEIGHT, text[key]); + break; + } + } + } + return this + .mouseover(function () { + return overlib.apply(null, args); + }) + .mouseout(nd); + } + }); +}(jQuery)); +/* }}} */ /*************************************************************************** * The real OnLoad @@ -663,7 +1242,8 @@ $(function() { }) .blur(function() { $("#quick_button").hide(); - }); + }) + .quickSearch(); $("#quick_button").click(function() { if ($("#quick").val() === 'Recherche dans l\'annuaire' || $("#quick").val() === '') {