//MooTools, My Object Oriented Javascript Tools. Copyright (c) 2006 Valerio Proietti, <http://mad4milk.net>, MIT Style License.
var MooTools={version:"1.12"};function $defined(a){return(a!=undefined)}function $type(b){if(!$defined(b)){return false}if(b.htmlElement){return"element"}var a=typeof b;if(a=="object"&&b.nodeName){switch(b.nodeType){case 1:return"element";case 3:return(/\S/).test(b.nodeValue)?"textnode":"whitespace"}}if(a=="object"||a=="function"){switch(b.constructor){case Array:return"array";case RegExp:return"regexp";case Class:return"class"}if(typeof b.length=="number"){if(b.item){return"collection"}if(b.callee){return"arguments"}}}return a}function $merge(){var c={};for(var b=0;b<arguments.length;b++){for(var f in arguments[b]){var a=arguments[b][f];var d=c[f];if(d&&$type(a)=="object"&&$type(d)=="object"){c[f]=$merge(d,a)}else{c[f]=a}}}return c}var $extend=function(){var a=arguments;if(!a[1]){a=[this,a[0]]}for(var b in a[1]){a[0][b]=a[1][b]}return a[0]};var $native=function(){for(var b=0,a=arguments.length;b<a;b++){arguments[b].extend=function(c){for(var d in c){if(!this.prototype[d]){this.prototype[d]=c[d]}if(!this[d]){this[d]=$native.generic(d)}}}}};$native.generic=function(a){return function(b){return this.prototype[a].apply(b,Array.prototype.slice.call(arguments,1))}};$native(Function,Array,String,Number);function $chk(a){return !!(a||a===0)}function $pick(b,a){return $defined(b)?b:a}function $random(b,a){return Math.floor(Math.random()*(a-b+1)+b)}function $time(){return new Date().getTime()}function $clear(a){clearTimeout(a);clearInterval(a);return null}var Abstract=function(a){a=a||{};a.extend=$extend;return a};var Window=new Abstract(window);var Document=new Abstract(document);document.head=document.getElementsByTagName("head")[0];window.xpath=!!(document.evaluate);if(window.ActiveXObject){window.ie=window[window.XMLHttpRequest?"ie7":"ie6"]=true}else{if(document.childNodes&&!document.all&&!navigator.taintEnabled){window.webkit=window[window.xpath?"webkit420":"webkit419"]=true}else{if(document.getBoxObjectFor!=null||window.mozInnerScreenX!=null){window.gecko=true}}}window.khtml=window.webkit;Object.extend=$extend;if(typeof HTMLElement=="undefined"){var HTMLElement=function(){};if(window.webkit){document.createElement("iframe")}HTMLElement.prototype=(window.webkit)?window["[[DOMElement.prototype]]"]:{}}HTMLElement.prototype.htmlElement=function(){};if(window.ie6){try{document.execCommand("BackgroundImageCache",false,true)}catch(e){}}var Class=function(b){var a=function(){return(arguments[0]!==null&&this.initialize&&$type(this.initialize)=="function")?this.initialize.apply(this,arguments):this};$extend(a,this);a.prototype=b;a.constructor=Class;return a};Class.empty=function(){};Class.prototype={extend:function(b){var c=new this(null);for(var d in b){var a=c[d];c[d]=Class.Merge(a,b[d])}return new Class(c)},implement:function(){for(var b=0,a=arguments.length;b<a;b++){$extend(this.prototype,arguments[b])}}};Class.Merge=function(c,d){if(c&&c!=d){var b=$type(d);if(b!=$type(c)){return d}switch(b){case"function":var a=function(){this.parent=arguments.callee.parent;return d.apply(this,arguments)};a.parent=c;return a;case"object":return $merge(c,d)}}return d};var Chain=new Class({chain:function(a){this.chains=this.chains||[];this.chains.push(a);return this},callChain:function(){if(this.chains&&this.chains.length){this.chains.shift().delay(10,this)}},clearChain:function(){this.chains=[]}});var Events=new Class({addEvent:function(b,a){if(a!=Class.empty){this.$events=this.$events||{};this.$events[b]=this.$events[b]||[];this.$events[b].include(a)}return this},fireEvent:function(c,b,a){if(this.$events&&this.$events[c]){this.$events[c].each(function(d){d.create({bind:this,delay:a,"arguments":b})()},this)}return this},removeEvent:function(b,a){if(this.$events&&this.$events[b]){this.$events[b].remove(a)}return this}});var Options=new Class({setOptions:function(){this.options=$merge.apply(null,[this.options].extend(arguments));if(this.addEvent){for(var a in this.options){if($type(this.options[a]=="function")&&(/^on[A-Z]/).test(a)){this.addEvent(a,this.options[a])}}}return this}});Array.extend({forEach:function(c,d){for(var b=0,a=this.length;b<a;b++){c.call(d,this[b],b,this)}},filter:function(d,f){var c=[];for(var b=0,a=this.length;b<a;b++){if(d.call(f,this[b],b,this)){c.push(this[b])}}return c},map:function(d,f){var c=[];for(var b=0,a=this.length;b<a;b++){c[b]=d.call(f,this[b],b,this)}return c},every:function(c,d){for(var b=0,a=this.length;b<a;b++){if(!c.call(d,this[b],b,this)){return false}}return true},some:function(c,d){for(var b=0,a=this.length;b<a;b++){if(c.call(d,this[b],b,this)){return true}}return false},indexOf:function(c,d){var a=this.length;for(var b=(d<0)?Math.max(0,a+d):d||0;b<a;b++){if(this[b]===c){return b}}return -1},copy:function(d,c){d=d||0;if(d<0){d=this.length+d}c=c||(this.length-d);var a=[];for(var b=0;b<c;b++){a[b]=this[d++]}return a},remove:function(c){var b=0;var a=this.length;while(b<a){if(this[b]===c){this.splice(b,1);a--}else{b++}}return this},contains:function(a,b){return this.indexOf(a,b)!=-1},associate:function(c){var d={},b=Math.min(this.length,c.length);for(var a=0;a<b;a++){d[c[a]]=this[a]}return d},extend:function(c){for(var b=0,a=c.length;b<a;b++){this.push(c[b])}return this},merge:function(c){for(var b=0,a=c.length;b<a;b++){this.include(c[b])}return this},include:function(a){if(!this.contains(a)){this.push(a)}return this},getRandom:function(){return this[$random(0,this.length-1)]||null},getLast:function(){return this[this.length-1]||null}});Array.prototype.each=Array.prototype.forEach;Array.each=Array.forEach;function $A(a){return Array.copy(a)}function $each(c,b,d){if(c&&typeof c.length=="number"&&$type(c)!="object"){Array.forEach(c,b,d)}else{for(var a in c){b.call(d||c,c[a],a)}}}Array.prototype.test=Array.prototype.contains;String.extend({test:function(a,b){return(($type(a)=="string")?new RegExp(a,b):a).test(this)},toInt:function(){return parseInt(this,10)},toFloat:function(){return parseFloat(this)},camelCase:function(){return this.replace(/-\D/g,function(a){return a.charAt(1).toUpperCase()})},hyphenate:function(){return this.replace(/\w[A-Z]/g,function(a){return(a.charAt(0)+"-"+a.charAt(1).toLowerCase())})},capitalize:function(){return this.replace(/\b[a-z]/g,function(a){return a.toUpperCase()})},trim:function(){return this.replace(/^\s+|\s+$/g,"")},clean:function(){return this.replace(/\s{2,}/g," ").trim()},rgbToHex:function(b){var a=this.match(/\d{1,3}/g);return(a)?a.rgbToHex(b):false},hexToRgb:function(b){var a=this.match(/^#?(\w{1,2})(\w{1,2})(\w{1,2})$/);return(a)?a.slice(1).hexToRgb(b):false},contains:function(a,b){return(b)?(b+this+b).indexOf(b+a+b)>-1:this.indexOf(a)>-1},escapeRegExp:function(){return this.replace(/([.*+?^${}()|[\]\/\\])/g,"\\$1")}});Array.extend({rgbToHex:function(d){if(this.length<3){return false}if(this.length==4&&this[3]==0&&!d){return"transparent"}var b=[];for(var a=0;a<3;a++){var c=(this[a]-0).toString(16);b.push((c.length==1)?"0"+c:c)}return d?b:"#"+b.join("")},hexToRgb:function(c){if(this.length!=3){return false}var a=[];for(var b=0;b<3;b++){a.push(parseInt((this[b].length==1)?this[b]+this[b]:this[b],16))}return c?a:"rgb("+a.join(",")+")"}});Function.extend({create:function(a){var b=this;a=$merge({bind:b,event:false,"arguments":null,delay:false,periodical:false,attempt:false},a);if($chk(a.arguments)&&$type(a.arguments)!="array"){a.arguments=[a.arguments]}return function(f){var c;if(a.event){f=f||window.event;c=[(a.event===true)?f:new a.event(f)];if(a.arguments){c.extend(a.arguments)}}else{c=a.arguments||arguments}var g=function(){return b.apply($pick(a.bind,b),c)};if(a.delay){return setTimeout(g,a.delay)}if(a.periodical){return setInterval(g,a.periodical)}if(a.attempt){try{return g()}catch(d){return false}}return g()}},pass:function(a,b){return this.create({"arguments":a,bind:b})},attempt:function(a,b){return this.create({"arguments":a,bind:b,attempt:true})()},bind:function(b,a){return this.create({bind:b,"arguments":a})},bindAsEventListener:function(b,a){return this.create({bind:b,event:true,"arguments":a})},delay:function(b,c,a){return this.create({delay:b,bind:c,"arguments":a})()},periodical:function(a,c,b){return this.create({periodical:a,bind:c,"arguments":b})()}});Number.extend({toInt:function(){return parseInt(this)},toFloat:function(){return parseFloat(this)},limit:function(b,a){return Math.min(a,Math.max(b,this))},round:function(a){a=Math.pow(10,a||0);return Math.round(this*a)/a},times:function(b){for(var a=0;a<this;a++){b(a)}}});var Element=new Class({initialize:function(d,c){if($type(d)=="string"){if(window.ie&&c&&(c.name||c.type)){var a=(c.name)?' name="'+c.name+'"':"";var b=(c.type)?' type="'+c.type+'"':"";delete c.name;delete c.type;d="<"+d+a+b+">"}d=document.createElement(d)}d=$(d);return(!c||!d)?d:d.set(c)}});var Elements=new Class({initialize:function(a){return(a)?$extend(a,this):this}});Elements.extend=function(a){for(var b in a){this.prototype[b]=a[b];this[b]=$native.generic(b)}};function $(b){if(!b){return null}if(b.htmlElement){return Garbage.collect(b)}if([window,document].contains(b)){return b}var a=$type(b);if(a=="string"){b=document.getElementById(b);a=(b)?"element":false}if(a!="element"){return null}if(b.htmlElement){return Garbage.collect(b)}if(["object","embed"].contains(b.tagName.toLowerCase())){return b}$extend(b,Element.prototype);b.htmlElement=function(){};return Garbage.collect(b)}document.getElementsBySelector=document.getElementsByTagName;function $$(){var d=[];for(var c=0,b=arguments.length;c<b;c++){var a=arguments[c];switch($type(a)){case"element":d.push(a);case"boolean":break;case false:break;case"string":a=document.getElementsBySelector(a,true);default:d.extend(a)}}return $$.unique(d)}$$.unique=function(j){var f=[];for(var c=0,a=j.length;c<a;c++){if(j[c].$included){continue}var b=$(j[c]);if(b&&!b.$included){b.$included=true;f.push(b)}}for(var h=0,g=f.length;h<g;h++){f[h].$included=null}return new Elements(f)};Elements.Multi=function(a){return function(){var d=arguments;var b=[];var h=true;for(var f=0,c=this.length,g;f<c;f++){g=this[f][a].apply(this[f],d);if($type(g)!="element"){h=false}b.push(g)}return(h)?$$.unique(b):b}};Element.extend=function(a){for(var b in a){HTMLElement.prototype[b]=a[b];Element.prototype[b]=a[b];Element[b]=$native.generic(b);var c=(Array.prototype[b])?b+"Elements":b;Elements.prototype[c]=Elements.Multi(b)}};Element.extend({set:function(a){for(var c in a){var b=a[c];switch(c){case"styles":this.setStyles(b);break;case"events":if(this.addEvents){this.addEvents(b)}break;case"properties":this.setProperties(b);break;default:this.setProperty(c,b)}}return this},inject:function(c,a){c=$(c);switch(a){case"before":c.parentNode.insertBefore(this,c);break;case"after":var b=c.getNext();if(!b){c.parentNode.appendChild(this)}else{c.parentNode.insertBefore(this,b)}break;case"top":var d=c.firstChild;if(d){c.insertBefore(this,d);break}default:c.appendChild(this)}return this},injectBefore:function(a){return this.inject(a,"before")},injectAfter:function(a){return this.inject(a,"after")},injectInside:function(a){return this.inject(a,"bottom")},injectTop:function(a){return this.inject(a,"top")},adopt:function(){var a=[];$each(arguments,function(b){a=a.concat(b)});$$(a).inject(this);return this},remove:function(){return this.parentNode.removeChild(this)},clone:function(c){var b=$(this.cloneNode(c!==false));if(!b.$events){return b}b.$events={};for(var a in this.$events){b.$events[a]={keys:$A(this.$events[a].keys),values:$A(this.$events[a].values)}}return b.removeEvents()},replaceWith:function(a){a=$(a);this.parentNode.replaceChild(a,this);return a},appendText:function(a){this.appendChild(document.createTextNode(a));return this},hasClass:function(a){return this.className.contains(a," ")},addClass:function(a){if(!this.hasClass(a)){this.className=(this.className+" "+a).clean()}return this},removeClass:function(a){this.className=this.className.replace(new RegExp("(^|\\s)"+a+"(?:\\s|$)"),"$1").clean();return this},toggleClass:function(a){return this.hasClass(a)?this.removeClass(a):this.addClass(a)},setStyle:function(b,a){switch(b){case"opacity":return this.setOpacity(parseFloat(a));case"float":b=(window.ie)?"styleFloat":"cssFloat"}b=b.camelCase();switch($type(a)){case"number":if(!["zIndex","zoom"].contains(b)){a+="px"}break;case"array":a="rgb("+a.join(",")+")"}this.style[b]=a;return this},setStyles:function(a){switch($type(a)){case"object":Element.setMany(this,"setStyle",a);break;case"string":this.style.cssText=a}return this},setOpacity:function(a){if(a==0){if(this.style.visibility!="hidden"){this.style.visibility="hidden"}}else{if(this.style.visibility!="visible"){this.style.visibility="visible"}}if(!this.currentStyle||!this.currentStyle.hasLayout){this.style.zoom=1}if(window.ie){this.style.filter=(a==1)?"":"alpha(opacity="+a*100+")"}this.style.opacity=this.$tmp.opacity=a;return this},getStyle:function(c){c=c.camelCase();var a=this.style[c];if(!$chk(a)){if(c=="opacity"){return this.$tmp.opacity}a=[];for(var b in Element.Styles){if(c==b){Element.Styles[b].each(function(g){var f=this.getStyle(g);a.push(parseInt(f)?f:"0px")},this);if(c=="border"){var d=a.every(function(f){return(f==a[0])});return(d)?a[0]:false}return a.join(" ")}}if(c.contains("border")){if(Element.Styles.border.contains(c)){return["Width","Style","Color"].map(function(f){return this.getStyle(c+f)},this).join(" ")}else{if(Element.borderShort.contains(c)){return["Top","Right","Bottom","Left"].map(function(f){return this.getStyle("border"+f+c.replace("border",""))},this).join(" ")}}}if(document.defaultView){a=document.defaultView.getComputedStyle(this,null).getPropertyValue(c.hyphenate())}else{if(this.currentStyle){a=this.currentStyle[c]}}}if(window.ie){a=Element.fixStyle(c,a,this)}if(a&&c.test(/color/i)&&a.contains("rgb")){return a.split("rgb").splice(1,4).map(function(f){return f.rgbToHex()}).join(" ")}return a},getStyles:function(){return Element.getMany(this,"getStyle",arguments)},walk:function(a,c){a+="Sibling";var b=(c)?this[c]:this[a];while(b&&$type(b)!="element"){b=b[a]}return $(b)},getPrevious:function(){return this.walk("previous")},getNext:function(){return this.walk("next")},getFirst:function(){return this.walk("next","firstChild")},getLast:function(){return this.walk("previous","lastChild")},getParent:function(){return $(this.parentNode)},getChildren:function(){return $$(this.childNodes)},hasChild:function(a){return !!$A(this.getElementsByTagName("*")).contains(a)},getProperty:function(d){var b=Element.Properties[d];if(b){return this[b]}var a=Element.PropertiesIFlag[d]||0;if(!window.ie||a){return this.getAttribute(d,a)}var c=this.attributes[d];return(c)?c.nodeValue:null},removeProperty:function(b){var a=Element.Properties[b];if(a){this[a]=""}else{this.removeAttribute(b)}return this},getProperties:function(){return Element.getMany(this,"getProperty",arguments)},setProperty:function(c,b){var a=Element.Properties[c];if(a){this[a]=b}else{this.setAttribute(c,b)}return this},setProperties:function(a){return Element.setMany(this,"setProperty",a)},setHTML:function(){this.innerHTML=$A(arguments).join("");return this},setText:function(b){var a=this.getTag();if(["style","script"].contains(a)){if(window.ie){if(a=="style"){this.styleSheet.cssText=b}else{if(a=="script"){this.setProperty("text",b)}}return this}else{this.removeChild(this.firstChild);return this.appendText(b)}}this[$defined(this.innerText)?"innerText":"textContent"]=b;return this},getText:function(){var a=this.getTag();if(["style","script"].contains(a)){if(window.ie){if(a=="style"){return this.styleSheet.cssText}else{if(a=="script"){return this.getProperty("text")}}}else{return this.innerHTML}}return($pick(this.innerText,this.textContent))},getTag:function(){return this.tagName.toLowerCase()},empty:function(){Garbage.trash(this.getElementsByTagName("*"));return this.setHTML("")}});Element.fixStyle=function(f,a,d){if($chk(parseInt(a))){return a}if(["height","width"].contains(f)){var b=(f=="width")?["left","right"]:["top","bottom"];var c=0;b.each(function(g){c+=d.getStyle("border-"+g+"-width").toInt()+d.getStyle("padding-"+g).toInt()});return d["offset"+f.capitalize()]-c+"px"}else{if(f.test(/border(.+)Width|margin|padding/)){return"0px"}}return a};Element.Styles={border:[],padding:[],margin:[]};["Top","Right","Bottom","Left"].each(function(b){for(var a in Element.Styles){Element.Styles[a].push(a+b)}});Element.borderShort=["borderWidth","borderStyle","borderColor"];Element.getMany=function(b,d,c){var a={};$each(c,function(f){a[f]=b[d](f)});return a};Element.setMany=function(b,d,c){for(var a in c){b[d](a,c[a])}return b};Element.Properties=new Abstract({"class":"className","for":"htmlFor",colspan:"colSpan",rowspan:"rowSpan",accesskey:"accessKey",tabindex:"tabIndex",maxlength:"maxLength",readonly:"readOnly",frameborder:"frameBorder",value:"value",disabled:"disabled",checked:"checked",multiple:"multiple",selected:"selected"});Element.PropertiesIFlag={href:2,src:2};Element.Methods={Listeners:{addListener:function(b,a){if(this.addEventListener){this.addEventListener(b,a,false)}else{this.attachEvent("on"+b,a)}return this},removeListener:function(b,a){if(this.removeEventListener){this.removeEventListener(b,a,false)}else{this.detachEvent("on"+b,a)}return this}}};window.extend(Element.Methods.Listeners);document.extend(Element.Methods.Listeners);Element.extend(Element.Methods.Listeners);var Garbage={elements:[],collect:function(a){if(!a.$tmp){Garbage.elements.push(a);a.$tmp={opacity:1}}return a},trash:function(f){for(var b=0,a=f.length,c;b<a;b++){if(!(c=f[b])||!c.$tmp){continue}if(c.$events){c.fireEvent("trash").removeEvents()}for(var g in c.$tmp){c.$tmp[g]=null}for(var h in Element.prototype){c[h]=null}Garbage.elements[Garbage.elements.indexOf(c)]=null;c.htmlElement=c.$tmp=c=null}Garbage.elements.remove(null)},empty:function(){Garbage.collect(window);Garbage.collect(document);Garbage.trash(Garbage.elements)}};window.addListener("beforeunload",function(){window.addListener("unload",Garbage.empty);if(window.ie){window.addListener("unload",CollectGarbage)}});var Event=new Class({initialize:function(c){if(c&&c.$extended){return c}this.$extended=true;c=c||window.event;this.event=c;this.type=c.type;this.target=c.target||c.srcElement;if(this.target.nodeType==3){this.target=this.target.parentNode}this.shift=c.shiftKey;this.control=c.ctrlKey;this.alt=c.altKey;this.meta=c.metaKey;if(["DOMMouseScroll","mousewheel"].contains(this.type)){this.wheel=(c.wheelDelta)?c.wheelDelta/120:-(c.detail||0)/3}else{if(this.type.contains("key")){this.code=c.which||c.keyCode;for(var b in Event.keys){if(Event.keys[b]==this.code){this.key=b;break}}if(this.type=="keydown"){var a=this.code-111;if(a>0&&a<13){this.key="f"+a}}this.key=this.key||String.fromCharCode(this.code).toLowerCase()}else{if(this.type.test(/(click|mouse|menu)/)){this.page={x:c.pageX||c.clientX+document.documentElement.scrollLeft,y:c.pageY||c.clientY+document.documentElement.scrollTop};this.client={x:c.pageX?c.pageX-window.pageXOffset:c.clientX,y:c.pageY?c.pageY-window.pageYOffset:c.clientY};this.rightClick=(c.which==3)||(c.button==2);switch(this.type){case"mouseover":this.relatedTarget=c.relatedTarget||c.fromElement;break;case"mouseout":this.relatedTarget=c.relatedTarget||c.toElement}this.fixRelatedTarget()}}}return this},stop:function(){return this.stopPropagation().preventDefault()},stopPropagation:function(){if(this.event.stopPropagation){this.event.stopPropagation()}else{this.event.cancelBubble=true}return this},preventDefault:function(){if(this.event.preventDefault){this.event.preventDefault()}else{this.event.returnValue=false}return this}});Event.fix={relatedTarget:function(){if(this.relatedTarget&&this.relatedTarget.nodeType==3){this.relatedTarget=this.relatedTarget.parentNode}},relatedTargetGecko:function(){try{Event.fix.relatedTarget.call(this)}catch(a){this.relatedTarget=this.target}}};Event.prototype.fixRelatedTarget=(window.gecko)?Event.fix.relatedTargetGecko:Event.fix.relatedTarget;Event.keys=new Abstract({enter:13,up:38,down:40,left:37,right:39,esc:27,space:32,backspace:8,tab:9,"delete":46});Element.Methods.Events={addEvent:function(c,b){this.$events=this.$events||{};this.$events[c]=this.$events[c]||{keys:[],values:[]};if(this.$events[c].keys.contains(b)){return this}this.$events[c].keys.push(b);var a=c;var d=Element.Events[c];if(d){if(d.add){d.add.call(this,b)}if(d.map){b=d.map}if(d.type){a=d.type}}if(!this.addEventListener){b=b.create({bind:this,event:true})}this.$events[c].values.push(b);return(Element.NativeEvents.contains(a))?this.addListener(a,b):this},removeEvent:function(c,b){if(!this.$events||!this.$events[c]){return this}var g=this.$events[c].keys.indexOf(b);if(g==-1){return this}var a=this.$events[c].keys.splice(g,1)[0];var f=this.$events[c].values.splice(g,1)[0];var d=Element.Events[c];if(d){if(d.remove){d.remove.call(this,b)}if(d.type){c=d.type}}return(Element.NativeEvents.contains(c))?this.removeListener(c,f):this},addEvents:function(a){return Element.setMany(this,"addEvent",a)},removeEvents:function(a){if(!this.$events){return this}if(!a){for(var b in this.$events){this.removeEvents(b)}this.$events=null}else{if(this.$events[a]){this.$events[a].keys.each(function(c){this.removeEvent(a,c)},this);this.$events[a]=null}}return this},fireEvent:function(c,b,a){if(this.$events&&this.$events[c]){this.$events[c].keys.each(function(d){d.create({bind:this,delay:a,"arguments":b})()},this)}return this},cloneEvents:function(c,a){if(!c.$events){return this}if(!a){for(var b in c.$events){this.cloneEvents(c,b)}}else{if(c.$events[a]){c.$events[a].keys.each(function(d){this.addEvent(a,d)},this)}}return this}};window.extend(Element.Methods.Events);document.extend(Element.Methods.Events);Element.extend(Element.Methods.Events);Element.Events=new Abstract({mouseenter:{type:"mouseover",map:function(a){a=new Event(a);if(a.relatedTarget!=this&&!this.hasChild(a.relatedTarget)){this.fireEvent("mouseenter",a)}}},mouseleave:{type:"mouseout",map:function(a){a=new Event(a);if(a.relatedTarget!=this&&!this.hasChild(a.relatedTarget)){this.fireEvent("mouseleave",a)}}},mousewheel:{type:(window.gecko)?"DOMMouseScroll":"mousewheel"}});Element.NativeEvents=["click","dblclick","mouseup","mousedown","mousewheel","DOMMouseScroll","mouseover","mouseout","mousemove","keydown","keypress","keyup","load","unload","beforeunload","resize","move","focus","blur","change","submit","reset","select","error","abort","contextmenu","scroll"];Function.extend({bindWithEvent:function(b,a){return this.create({bind:b,"arguments":a,event:Event})}});Elements.extend({filterByTag:function(a){return new Elements(this.filter(function(b){return(Element.getTag(b)==a)}))},filterByClass:function(a,c){var b=this.filter(function(d){return(d.className&&d.className.contains(a," "))});return(c)?b:new Elements(b)},filterById:function(c,b){var a=this.filter(function(d){return(d.id==c)});return(b)?a:new Elements(a)},filterByAttribute:function(b,a,d,f){var c=this.filter(function(g){var h=Element.getProperty(g,b);if(!h){return false}if(!a){return true}switch(a){case"=":return(h==d);case"*=":return(h.contains(d));case"^=":return(h.substr(0,d.length)==d);case"$=":return(h.substr(h.length-d.length)==d);case"!=":return(h!=d);case"~=":return h.contains(d," ")}return false});return(f)?c:new Elements(c)}});function $E(a,b){return($(b)||document).getElement(a)}function $ES(a,b){return($(b)||document).getElementsBySelector(a)}$$.shared={regexp:/^(\w*|\*)(?:#([\w-]+)|\.([\w-]+))?(?:\[(\w+)(?:([!*^$]?=)["']?([^"'\]]*)["']?)?])?$/,xpath:{getParam:function(b,d,f,c){var a=[d.namespaceURI?"xhtml:":"",f[1]];if(f[2]){a.push('[@id="',f[2],'"]')}if(f[3]){a.push('[contains(concat(" ", @class, " "), " ',f[3],' ")]')}if(f[4]){if(f[5]&&f[6]){switch(f[5]){case"*=":a.push("[contains(@",f[4],', "',f[6],'")]');break;case"^=":a.push("[starts-with(@",f[4],', "',f[6],'")]');break;case"$=":a.push("[substring(@",f[4],", string-length(@",f[4],") - ",f[6].length,' + 1) = "',f[6],'"]');break;case"=":a.push("[@",f[4],'="',f[6],'"]');break;case"!=":a.push("[@",f[4],'!="',f[6],'"]')}}else{a.push("[@",f[4],"]")}}b.push(a.join(""));return b},getItems:function(b,f,h){var g=[];var a=document.evaluate(".//"+b.join("//"),f,$$.shared.resolver,XPathResult.UNORDERED_NODE_SNAPSHOT_TYPE,null);for(var d=0,c=a.snapshotLength;d<c;d++){g.push(a.snapshotItem(d))}return(h)?g:new Elements(g.map($))}},normal:{getParam:function(a,c,f,b){if(b==0){if(f[2]){var d=c.getElementById(f[2]);if(!d||((f[1]!="*")&&(Element.getTag(d)!=f[1]))){return false}a=[d]}else{a=$A(c.getElementsByTagName(f[1]))}}else{a=$$.shared.getElementsByTagName(a,f[1]);if(f[2]){a=Elements.filterById(a,f[2],true)}}if(f[3]){a=Elements.filterByClass(a,f[3],true)}if(f[4]){a=Elements.filterByAttribute(a,f[4],f[5],f[6],true)}return a},getItems:function(a,b,c){return(c)?a:$$.unique(a)}},resolver:function(a){return(a=="xhtml")?"http://www.w3.org/1999/xhtml":false},getElementsByTagName:function(d,c){var f=[];for(var b=0,a=d.length;b<a;b++){f.extend(d[b].getElementsByTagName(c))}return f}};$$.shared.method=(window.xpath)?"xpath":"normal";Element.Methods.Dom={getElements:function(a,k){var c=[];a=a.trim().split(" ");for(var f=0,d=a.length;f<d;f++){var g=a[f];var h=g.match($$.shared.regexp);if(!h){break}h[1]=h[1]||"*";var b=$$.shared[$$.shared.method].getParam(c,this,h,f);if(!b){break}c=b}return $$.shared[$$.shared.method].getItems(c,this,k)},getElement:function(a){return $(this.getElements(a,true)[0]||false)},getElementsBySelector:function(a,f){var d=[];a=a.split(",");for(var c=0,b=a.length;c<b;c++){d=d.concat(this.getElements(a[c],true))}return(f)?d:$$.unique(d)}};Element.extend({getElementById:function(c){var b=document.getElementById(c);if(!b){return false}for(var a=b.parentNode;a!=this;a=a.parentNode){if(!a){return false}}return b},getElementsByClassName:function(a){return this.getElements("."+a)}});document.extend(Element.Methods.Dom);Element.extend(Element.Methods.Dom);Element.extend({getValue:function(){switch(this.getTag()){case"select":var a=[];$each(this.options,function(b){if(b.selected){a.push($pick(b.value,b.text))}});return(this.multiple)?a:a[0];case"input":if(!(this.checked&&["checkbox","radio"].contains(this.type))&&!["hidden","text","password"].contains(this.type)){break}case"textarea":return this.value}return false},getFormElements:function(){return $$(this.getElementsByTagName("input"),this.getElementsByTagName("select"),this.getElementsByTagName("textarea"))},toQueryString:function(){var a=[];this.getFormElements().each(function(d){var c=d.name;var f=d.getValue();if(f===false||!c||d.disabled){return}var b=function(g){a.push(c+"="+encodeURIComponent(g))};if($type(f)=="array"){f.each(b)}else{b(f)}});return a.join("&")}});Element.extend({scrollTo:function(a,b){this.scrollLeft=a;this.scrollTop=b},getSize:function(){return{scroll:{x:this.scrollLeft,y:this.scrollTop},size:{x:this.offsetWidth,y:this.offsetHeight},scrollSize:{x:this.scrollWidth,y:this.scrollHeight}}},getPosition:function(a){a=a||[];var b=this,d=0,c=0;do{d+=b.offsetLeft||0;c+=b.offsetTop||0;b=b.offsetParent}while(b);a.each(function(f){d-=f.scrollLeft||0;c-=f.scrollTop||0});return{x:d,y:c}},getTop:function(a){return this.getPosition(a).y},getLeft:function(a){return this.getPosition(a).x},getCoordinates:function(b){var a=this.getPosition(b);var c={width:this.offsetWidth,height:this.offsetHeight,left:a.x,top:a.y};c.right=c.left+c.width;c.bottom=c.top+c.height;return c}});Element.Events.domready={add:function(b){if(window.loaded){b.call(this);return}var a=function(){if(window.loaded){return}window.loaded=true;window.timer=$clear(window.timer);this.fireEvent("domready")}.bind(this);if(document.readyState&&window.webkit){window.timer=function(){if(["loaded","complete"].contains(document.readyState)){a()}}.periodical(50)}else{if(document.readyState&&window.ie){if(!$("ie_ready")){var c=(window.location.protocol=="https:")?"://0":"javascript:void(0)";document.write('<script id="ie_ready" defer src="'+c+'"><\/script>');$("ie_ready").onreadystatechange=function(){if(this.readyState=="complete"){a()}}}}else{window.addListener("load",a);document.addListener("DOMContentLoaded",a)}}}};window.onDomReady=function(a){return this.addEvent("domready",a)};window.extend({getWidth:function(){if(this.webkit419){return this.innerWidth}if(this.opera){return document.body.clientWidth}return document.documentElement.clientWidth},getHeight:function(){if(this.webkit419){return this.innerHeight}if(this.opera){return document.body.clientHeight}return document.documentElement.clientHeight},getScrollWidth:function(){if(this.ie){return Math.max(document.documentElement.offsetWidth,document.documentElement.scrollWidth)}if(this.webkit){return document.body.scrollWidth}return document.documentElement.scrollWidth},getScrollHeight:function(){if(this.ie){return Math.max(document.documentElement.offsetHeight,document.documentElement.scrollHeight)}if(this.webkit){return document.body.scrollHeight}return document.documentElement.scrollHeight},getScrollLeft:function(){return this.pageXOffset||document.documentElement.scrollLeft},getScrollTop:function(){return this.pageYOffset||document.documentElement.scrollTop},getSize:function(){return{size:{x:this.getWidth(),y:this.getHeight()},scrollSize:{x:this.getScrollWidth(),y:this.getScrollHeight()},scroll:{x:this.getScrollLeft(),y:this.getScrollTop()}}},getPosition:function(){return{x:0,y:0}}});var Fx={};Fx.Base=new Class({options:{onStart:Class.empty,onComplete:Class.empty,onCancel:Class.empty,transition:function(a){return -(Math.cos(Math.PI*a)-1)/2},duration:500,unit:"px",wait:true,fps:50},initialize:function(a){this.element=this.element||null;this.setOptions(a);if(this.options.initialize){this.options.initialize.call(this)}},step:function(){var a=$time();if(a<this.time+this.options.duration){this.delta=this.options.transition((a-this.time)/this.options.duration);this.setNow();this.increase()}else{this.stop(true);this.set(this.to);this.fireEvent("onComplete",this.element,10);this.callChain()}},set:function(a){this.now=a;this.increase();return this},setNow:function(){this.now=this.compute(this.from,this.to)},compute:function(b,a){return(a-b)*this.delta+b},start:function(b,a){if(!this.options.wait){this.stop()}else{if(this.timer){return this}}this.from=b;this.to=a;this.change=this.to-this.from;this.time=$time();this.timer=this.step.periodical(Math.round(1000/this.options.fps),this);this.fireEvent("onStart",this.element);return this},stop:function(a){if(!this.timer){return this}this.timer=$clear(this.timer);if(!a){this.fireEvent("onCancel",this.element)}return this},custom:function(b,a){return this.start(b,a)},clearTimer:function(a){return this.stop(a)}});Fx.Base.implement(new Chain,new Events,new Options);Fx.CSS={select:function(b,c){if(b.test(/color/i)){return this.Color}var a=$type(c);if((a=="array")||(a=="string"&&c.contains(" "))){return this.Multi}return this.Single},parse:function(c,d,a){if(!a.push){a=[a]}var g=a[0],f=a[1];if(!$chk(f)){f=g;g=c.getStyle(d)}var b=this.select(d,f);return{from:b.parse(g),to:b.parse(f),css:b}}};Fx.CSS.Single={parse:function(a){return parseFloat(a)},getNow:function(c,b,a){return a.compute(c,b)},getValue:function(c,a,b){if(a=="px"&&b!="opacity"){c=Math.round(c)}return c+a}};Fx.CSS.Multi={parse:function(a){return a.push?a:a.split(" ").map(function(b){return parseFloat(b)})},getNow:function(f,d,c){var a=[];for(var b=0;b<f.length;b++){a[b]=c.compute(f[b],d[b])}return a},getValue:function(c,a,b){if(a=="px"&&b!="opacity"){c=c.map(Math.round)}return c.join(a+" ")+a}};Fx.CSS.Color={parse:function(a){return a.push?a:a.hexToRgb(true)},getNow:function(f,d,c){var a=[];for(var b=0;b<f.length;b++){a[b]=Math.round(c.compute(f[b],d[b]))}return a},getValue:function(a){return"rgb("+a.join(",")+")"}};Fx.Style=Fx.Base.extend({initialize:function(b,c,a){this.element=$(b);this.property=c;this.parent(a)},hide:function(){return this.set(0)},setNow:function(){this.now=this.css.getNow(this.from,this.to,this)},set:function(a){this.css=Fx.CSS.select(this.property,a);return this.parent(this.css.parse(a))},start:function(c,b){if(this.timer&&this.options.wait){return this}var a=Fx.CSS.parse(this.element,this.property,[c,b]);this.css=a.css;return this.parent(a.from,a.to)},increase:function(){this.element.setStyle(this.property,this.css.getValue(this.now,this.options.unit,this.property))}});Element.extend({effect:function(b,a){return new Fx.Style(this,b,a)}});Fx.Styles=Fx.Base.extend({initialize:function(b,a){this.element=$(b);this.parent(a)},setNow:function(){for(var a in this.from){this.now[a]=this.css[a].getNow(this.from[a],this.to[a],this)}},set:function(c){var a={};this.css={};for(var b in c){this.css[b]=Fx.CSS.select(b,c[b]);a[b]=this.css[b].parse(c[b])}return this.parent(a)},start:function(c){if(this.timer&&this.options.wait){return this}this.now={};this.css={};var f={},d={};for(var b in c){var a=Fx.CSS.parse(this.element,b,c[b]);f[b]=a.from;d[b]=a.to;this.css[b]=a.css}return this.parent(f,d)},increase:function(){for(var a in this.now){this.element.setStyle(a,this.css[a].getValue(this.now[a],this.options.unit,a))}}});Element.extend({effects:function(a){return new Fx.Styles(this,a)}});Fx.Elements=Fx.Base.extend({initialize:function(b,a){this.elements=$$(b);this.parent(a)},setNow:function(){for(var c in this.from){var g=this.from[c],f=this.to[c],b=this.css[c],a=this.now[c]={};for(var d in g){a[d]=b[d].getNow(g[d],f[d],this)}}},set:function(h){var b={};this.css={};for(var d in h){var g=h[d],c=this.css[d]={},a=b[d]={};for(var f in g){c[f]=Fx.CSS.select(f,g[f]);a[f]=c[f].parse(g[f])}}return this.parent(b)},start:function(d){if(this.timer&&this.options.wait){return this}this.now={};this.css={};var k={},l={};for(var f in d){var h=d[f],a=k[f]={},j=l[f]={},c=this.css[f]={};for(var b in h){var g=Fx.CSS.parse(this.elements[f],b,h[b]);a[b]=g.from;j[b]=g.to;c[b]=g.css}}return this.parent(k,l)},increase:function(){for(var c in this.now){var a=this.now[c],b=this.css[c];for(var d in a){this.elements[c].setStyle(d,b[d].getValue(a[d],this.options.unit,d))}}}});Fx.Scroll=Fx.Base.extend({options:{overflown:[],offset:{x:0,y:0},wheelStops:true},initialize:function(b,a){this.now=[];this.element=$(b);this.bound={stop:this.stop.bind(this,false)};this.parent(a);if(this.options.wheelStops){this.addEvent("onStart",function(){document.addEvent("mousewheel",this.bound.stop)}.bind(this));this.addEvent("onComplete",function(){document.removeEvent("mousewheel",this.bound.stop)}.bind(this))}},setNow:function(){for(var a=0;a<2;a++){this.now[a]=this.compute(this.from[a],this.to[a])}},scrollTo:function(b,g){if(this.timer&&this.options.wait){return this}var d=this.element.getSize();var c={x:b,y:g};for(var f in d.size){var a=d.scrollSize[f]-d.size[f];if($chk(c[f])){c[f]=($type(c[f])=="number")?c[f].limit(0,a):a}else{c[f]=d.scroll[f]}c[f]+=this.options.offset[f]}return this.start([d.scroll.x,d.scroll.y],[c.x,c.y])},toTop:function(){return this.scrollTo(false,0)},toBottom:function(){return this.scrollTo(false,"full")},toLeft:function(){return this.scrollTo(0,false)},toRight:function(){return this.scrollTo("full",false)},toElement:function(b){var a=this.element.getPosition(this.options.overflown);var c=$(b).getPosition(this.options.overflown);return this.scrollTo(c.x-a.x,c.y-a.y)},increase:function(){this.element.scrollTo(this.now[0],this.now[1])}});Fx.Slide=Fx.Base.extend({options:{mode:"vertical"},initialize:function(b,a){this.element=$(b);this.wrapper=new Element("div",{styles:$extend(this.element.getStyles("margin"),{overflow:"hidden"})}).injectAfter(this.element).adopt(this.element);this.element.setStyle("margin",0);this.setOptions(a);this.now=[];this.parent(this.options);this.open=true;this.addEvent("onComplete",function(){this.open=(this.now[0]===0)});if(window.webkit419){this.addEvent("onComplete",function(){if(this.open){this.element.remove().inject(this.wrapper)}})}},setNow:function(){for(var a=0;a<2;a++){this.now[a]=this.compute(this.from[a],this.to[a])}},vertical:function(){this.margin="margin-top";this.layout="height";this.offset=this.element.offsetHeight},horizontal:function(){this.margin="margin-left";this.layout="width";this.offset=this.element.offsetWidth},slideIn:function(a){this[a||this.options.mode]();return this.start([this.element.getStyle(this.margin).toInt(),this.wrapper.getStyle(this.layout).toInt()],[0,this.offset])},slideOut:function(a){this[a||this.options.mode]();return this.start([this.element.getStyle(this.margin).toInt(),this.wrapper.getStyle(this.layout).toInt()],[-this.offset,0])},hide:function(a){this[a||this.options.mode]();this.open=false;return this.set([-this.offset,0])},show:function(a){this[a||this.options.mode]();this.open=true;return this.set([0,this.offset])},toggle:function(a){if(this.wrapper.offsetHeight==0||this.wrapper.offsetWidth==0){return this.slideIn(a)}return this.slideOut(a)},increase:function(){this.element.setStyle(this.margin,this.now[0]+this.options.unit);this.wrapper.setStyle(this.layout,this.now[1]+this.options.unit)}});Fx.Transition=function(b,a){a=a||[];if($type(a)!="array"){a=[a]}return $extend(b,{easeIn:function(c){return b(c,a)},easeOut:function(c){return 1-b(1-c,a)},easeInOut:function(c){return(c<=0.5)?b(2*c,a)/2:(2-b(2*(1-c),a))/2}})};Fx.Transitions=new Abstract({linear:function(a){return a}});Fx.Transitions.extend=function(a){for(var b in a){Fx.Transitions[b]=new Fx.Transition(a[b]);Fx.Transitions.compat(b)}};Fx.Transitions.compat=function(a){["In","Out","InOut"].each(function(b){Fx.Transitions[a.toLowerCase()+b]=Fx.Transitions[a]["ease"+b]})};Fx.Transitions.extend({Pow:function(b,a){return Math.pow(b,a[0]||6)},Expo:function(a){return Math.pow(2,8*(a-1))},Circ:function(a){return 1-Math.sin(Math.acos(a))},Sine:function(a){return 1-Math.sin((1-a)*Math.PI/2)},Back:function(b,a){a=a[0]||1.618;return Math.pow(b,2)*((a+1)*b-a)},Bounce:function(g){var f;for(var d=0,c=1;1;d+=c,c/=2){if(g>=(7-4*d)/11){f=-Math.pow((11-6*d-11*g)/4,2)+c*c;break}}return f},Elastic:function(b,a){return Math.pow(2,10*--b)*Math.cos(20*b*Math.PI*(a[0]||1)/3)}});["Quad","Cubic","Quart","Quint"].each(function(b,a){Fx.Transitions[b]=new Fx.Transition(function(c){return Math.pow(c,[a+2])});Fx.Transitions.compat(b)});var Drag={};Drag.Base=new Class({options:{handle:false,unit:"px",onStart:Class.empty,onBeforeStart:Class.empty,onComplete:Class.empty,onSnap:Class.empty,onDrag:Class.empty,limit:false,modifiers:{x:"left",y:"top"},grid:false,snap:6},initialize:function(b,a){this.setOptions(a);this.element=$(b);this.handle=$(this.options.handle)||this.element;this.mouse={now:{},pos:{}};this.value={start:{},now:{}};this.bound={start:this.start.bindWithEvent(this),check:this.check.bindWithEvent(this),drag:this.drag.bindWithEvent(this),stop:this.stop.bind(this)};this.attach();if(this.options.initialize){this.options.initialize.call(this)}},attach:function(){this.handle.addEvent("mousedown",this.bound.start);return this},detach:function(){this.handle.removeEvent("mousedown",this.bound.start);return this},start:function(c){this.fireEvent("onBeforeStart",this.element);this.mouse.start=c.page;var a=this.options.limit;this.limit={x:[],y:[]};for(var d in this.options.modifiers){if(!this.options.modifiers[d]){continue}this.value.now[d]=this.element.getStyle(this.options.modifiers[d]).toInt();this.mouse.pos[d]=c.page[d]-this.value.now[d];if(a&&a[d]){for(var b=0;b<2;b++){if($chk(a[d][b])){this.limit[d][b]=($type(a[d][b])=="function")?a[d][b]():a[d][b]}}}}if($type(this.options.grid)=="number"){this.options.grid={x:this.options.grid,y:this.options.grid}}document.addListener("mousemove",this.bound.check);document.addListener("mouseup",this.bound.stop);this.fireEvent("onStart",this.element);c.stop()},check:function(a){var b=Math.round(Math.sqrt(Math.pow(a.page.x-this.mouse.start.x,2)+Math.pow(a.page.y-this.mouse.start.y,2)));if(b>this.options.snap){document.removeListener("mousemove",this.bound.check);document.addListener("mousemove",this.bound.drag);this.drag(a);this.fireEvent("onSnap",this.element)}a.stop()},drag:function(a){this.out=false;this.mouse.now=a.page;for(var b in this.options.modifiers){if(!this.options.modifiers[b]){continue}this.value.now[b]=this.mouse.now[b]-this.mouse.pos[b];if(this.limit[b]){if($chk(this.limit[b][1])&&(this.value.now[b]>this.limit[b][1])){this.value.now[b]=this.limit[b][1];this.out=true}else{if($chk(this.limit[b][0])&&(this.value.now[b]<this.limit[b][0])){this.value.now[b]=this.limit[b][0];this.out=true}}}if(this.options.grid[b]){this.value.now[b]-=(this.value.now[b]%this.options.grid[b])}this.element.setStyle(this.options.modifiers[b],this.value.now[b]+this.options.unit)}this.fireEvent("onDrag",this.element);a.stop()},stop:function(){document.removeListener("mousemove",this.bound.check);document.removeListener("mousemove",this.bound.drag);document.removeListener("mouseup",this.bound.stop);this.fireEvent("onComplete",this.element)}});Drag.Base.implement(new Events,new Options);Element.extend({makeResizable:function(a){return new Drag.Base(this,$merge({modifiers:{x:"width",y:"height"}},a))}});Drag.Move=Drag.Base.extend({options:{droppables:[],container:false,overflown:[]},initialize:function(b,a){this.setOptions(a);this.element=$(b);this.droppables=$$(this.options.droppables);this.container=$(this.options.container);this.position={element:this.element.getStyle("position"),container:false};if(this.container){this.position.container=this.container.getStyle("position")}if(!["relative","absolute","fixed"].contains(this.position.element)){this.position.element="absolute"}var d=this.element.getStyle("top").toInt();var c=this.element.getStyle("left").toInt();if(this.position.element=="absolute"&&!["relative","absolute","fixed"].contains(this.position.container)){d=$chk(d)?d:this.element.getTop(this.options.overflown);c=$chk(c)?c:this.element.getLeft(this.options.overflown)}else{d=$chk(d)?d:0;c=$chk(c)?c:0}this.element.setStyles({top:d,left:c,position:this.position.element});this.parent(this.element)},start:function(c){this.overed=null;if(this.container){var a=this.container.getCoordinates();var b=this.element.getCoordinates();if(this.position.element=="absolute"&&!["relative","absolute","fixed"].contains(this.position.container)){this.options.limit={x:[a.left,a.right-b.width],y:[a.top,a.bottom-b.height]}}else{this.options.limit={y:[0,a.height-b.height],x:[0,a.width-b.width]}}}this.parent(c)},drag:function(a){this.parent(a);var b=this.out?false:this.droppables.filter(this.checkAgainst,this).getLast();if(this.overed!=b){if(this.overed){this.overed.fireEvent("leave",[this.element,this])}this.overed=b?b.fireEvent("over",[this.element,this]):null}return this},checkAgainst:function(b){b=b.getCoordinates(this.options.overflown);var a=this.mouse.now;return(a.x>b.left&&a.x<b.right&&a.y<b.bottom&&a.y>b.top)},stop:function(){if(this.overed&&!this.out){this.overed.fireEvent("drop",[this.element,this])}else{this.element.fireEvent("emptydrop",this)}this.parent();return this}});Element.extend({makeDraggable:function(a){return new Drag.Move(this,a)}});var XHR=new Class({options:{method:"post",async:true,onRequest:Class.empty,onSuccess:Class.empty,onFailure:Class.empty,urlEncoded:true,encoding:"utf-8",autoCancel:false,headers:{}},setTransport:function(){this.transport=(window.XMLHttpRequest)?new XMLHttpRequest():(window.ie?new ActiveXObject("Microsoft.XMLHTTP"):false);return this},initialize:function(a){this.setTransport().setOptions(a);this.options.isSuccess=this.options.isSuccess||this.isSuccess;this.headers={};if(this.options.urlEncoded&&this.options.method=="post"){var b=(this.options.encoding)?"; charset="+this.options.encoding:"";this.setHeader("Content-type","application/x-www-form-urlencoded"+b)}if(this.options.initialize){this.options.initialize.call(this)}},onStateChange:function(){if(this.transport.readyState!=4||!this.running){return}this.running=false;var a=0;try{a=this.transport.status}catch(b){}if(this.options.isSuccess.call(this,a)){this.onSuccess()}else{this.onFailure()}this.transport.onreadystatechange=Class.empty},isSuccess:function(a){return((a>=200)&&(a<300))},onSuccess:function(){this.response={text:this.transport.responseText,xml:this.transport.responseXML};this.fireEvent("onSuccess",[this.response.text,this.response.xml]);this.callChain()},onFailure:function(){this.fireEvent("onFailure",this.transport)},setHeader:function(a,b){this.headers[a]=b;return this},send:function(a,c){if(this.options.autoCancel){this.cancel()}else{if(this.running){return this}}this.running=true;if(c&&this.options.method=="get"){a=a+(a.contains("?")?"&":"?")+c;c=null}this.transport.open(this.options.method.toUpperCase(),a,this.options.async);this.transport.onreadystatechange=this.onStateChange.bind(this);if((this.options.method=="post")&&this.transport.overrideMimeType){/*this.setHeader("Connection","close")*/}$extend(this.headers,this.options.headers);for(var b in this.headers){try{this.transport.setRequestHeader(b,this.headers[b])}catch(d){}}this.fireEvent("onRequest");this.transport.send($pick(c,null));return this},cancel:function(){if(!this.running){return this}this.running=false;this.transport.abort();this.transport.onreadystatechange=Class.empty;this.setTransport();this.fireEvent("onCancel");return this}});XHR.implement(new Chain,new Events,new Options);var Ajax=XHR.extend({options:{data:null,update:null,onComplete:Class.empty,evalScripts:false,evalResponse:false},initialize:function(b,a){this.addEvent("onSuccess",this.onComplete);this.setOptions(a);this.options.data=this.options.data||this.options.postBody;if(!["post","get"].contains(this.options.method)){this._method="_method="+this.options.method;this.options.method="post"}this.parent();this.setHeader("X-Requested-With","XMLHttpRequest");this.setHeader("Accept","text/javascript, text/html, application/xml, text/xml, */*");this.url=b},onComplete:function(){if(this.options.update){$(this.options.update).empty().setHTML(this.response.text)}if(this.options.evalScripts||this.options.evalResponse){this.evalScripts()}this.fireEvent("onComplete",[this.response.text,this.response.xml],20)},request:function(a){a=a||this.options.data;switch($type(a)){case"element":a=$(a).toQueryString();break;case"object":a=Object.toQueryString(a)}if(this._method){a=(a)?[this._method,a].join("&"):this._method}return this.send(this.url,a)},evalScripts:function(){var b,a;if(this.options.evalResponse||(/(ecma|java)script/).test(this.getHeader("Content-type"))){a=this.response.text}else{a=[];var c=/<script[^>]*>([\s\S]*?)<\/script>/gi;while((b=c.exec(this.response.text))){a.push(b[1])}a=a.join("\n")}if(a){(window.execScript)?window.execScript(a):window.setTimeout(a,0)}},getHeader:function(a){try{return this.transport.getResponseHeader(a)}catch(b){}return null}});Object.toQueryString=function(b){var c=[];for(var a in b){c.push(encodeURIComponent(a)+"="+encodeURIComponent(b[a]))}return c.join("&")};Element.extend({send:function(a){return new Ajax(this.getProperty("action"),$merge({data:this.toQueryString()},a,{method:"post"})).request()}});var Cookie=new Abstract({options:{domain:false,path:false,duration:false,secure:false},set:function(c,d,b){b=$merge(this.options,b);d=encodeURIComponent(d);if(b.domain){d+="; domain="+b.domain}if(b.path){d+="; path="+b.path}if(b.duration){var a=new Date();a.setTime(a.getTime()+b.duration*24*60*60*1000);d+="; expires="+a.toGMTString()}if(b.secure){d+="; secure"}document.cookie=c+"="+d;return $extend(b,{key:c,value:d})},get:function(a){var b=document.cookie.match("(?:^|;)\\s*"+a.escapeRegExp()+"=([^;]*)");return b?decodeURIComponent(b[1]):false},remove:function(b,a){if($type(b)=="object"){this.set(b.key,"",$merge(b,{duration:-1}))}else{this.set(b,"",$merge(a,{duration:-1}))}}});var Json={toString:function(c){switch($type(c)){case"string":return'"'+c.replace(/(["\\])/g,"\\$1")+'"';case"array":return"["+c.map(Json.toString).join(",")+"]";case"object":var a=[];for(var b in c){a.push(Json.toString(b)+":"+Json.toString(c[b]))}return"{"+a.join(",")+"}";case"number":if(isFinite(c)){break}case false:return"null"}return String(c)},evaluate:function(str,secure){return(($type(str)!="string")||(secure&&!str.test(/^("(\\.|[^"\\\n\r])*?"|[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t])+?$/)))?null:eval("("+str+")")}};Json.Remote=XHR.extend({initialize:function(b,a){this.url=b;this.addEvent("onSuccess",this.onComplete);this.parent(a);this.setHeader("X-Request","JSON")},send:function(a){return this.parent(this.url,"json="+Json.toString(a))},onComplete:function(){this.fireEvent("onComplete",[Json.evaluate(this.response.text,this.options.secure)])}});var Asset=new Abstract({javascript:function(c,b){b=$merge({onload:Class.empty},b);var a=new Element("script",{src:c}).addEvents({load:b.onload,readystatechange:function(){if(this.readyState=="complete"){this.fireEvent("load")}}});delete b.onload;return a.setProperties(b).inject(document.head)},css:function(b,a){return new Element("link",$merge({rel:"stylesheet",media:"screen",type:"text/css",href:b},a)).inject(document.head)},image:function(c,b){b=$merge({onload:Class.empty,onabort:Class.empty,onerror:Class.empty},b);var d=new Image();d.src=c;var a=new Element("img",{src:c});["load","abort","error"].each(function(f){var g=b["on"+f];delete b["on"+f];a.addEvent(f,function(){this.removeEvent(f,arguments.callee);g.call(this)})});if(d.width&&d.height){a.fireEvent("load",a,1)}return a.setProperties(b)},images:function(d,c){c=$merge({onComplete:Class.empty,onProgress:Class.empty},c);if(!d.push){d=[d]}var a=[];var b=0;d.each(function(g){var f=new Asset.image(g,{onload:function(){c.onProgress.call(this,b);b++;if(b==d.length){c.onComplete()}}});a.push(f)});return new Elements(a)}});var Hash=new Class({length:0,initialize:function(a){this.obj=a||{};this.setLength()},get:function(a){return(this.hasKey(a))?this.obj[a]:null},hasKey:function(a){return(a in this.obj)},set:function(a,b){if(!this.hasKey(a)){this.length++}this.obj[a]=b;return this},setLength:function(){this.length=0;for(var a in this.obj){this.length++}return this},remove:function(a){if(this.hasKey(a)){delete this.obj[a];this.length--}return this},each:function(a,b){$each(this.obj,a,b)},extend:function(a){$extend(this.obj,a);return this.setLength()},merge:function(){this.obj=$merge.apply(null,[this.obj].extend(arguments));return this.setLength()},empty:function(){this.obj={};this.length=0;return this},keys:function(){var a=[];for(var b in this.obj){a.push(b)}return a},values:function(){var a=[];for(var b in this.obj){a.push(this.obj[b])}return a}});function $H(a){return new Hash(a)}Hash.Cookie=Hash.extend({initialize:function(b,a){this.name=b;this.options=$extend({autoSave:true},a||{});this.load()},save:function(){if(this.length==0){Cookie.remove(this.name,this.options);return true}var a=Json.toString(this.obj);if(a.length>4096){return false}Cookie.set(this.name,a,this.options);return true},load:function(){this.obj=Json.evaluate(Cookie.get(this.name),true)||{};this.setLength()}});Hash.Cookie.Methods={};["extend","set","merge","empty","remove"].each(function(a){Hash.Cookie.Methods[a]=function(){Hash.prototype[a].apply(this,arguments);if(this.options.autoSave){this.save()}return this}});Hash.Cookie.implement(Hash.Cookie.Methods);var Color=new Class({initialize:function(b,d){d=d||(b.push?"rgb":"hex");var c,a;switch(d){case"rgb":c=b;a=c.rgbToHsb();break;case"hsb":c=b.hsbToRgb();a=b;break;default:c=b.hexToRgb(true);a=c.rgbToHsb()}c.hsb=a;c.hex=c.rgbToHex();return $extend(c,Color.prototype)},mix:function(){var a=$A(arguments);var c=($type(a[a.length-1])=="number")?a.pop():50;var b=this.copy();a.each(function(d){d=new Color(d);for(var f=0;f<3;f++){b[f]=Math.round((b[f]/100*(100-c))+(d[f]/100*c))}});return new Color(b,"rgb")},invert:function(){return new Color(this.map(function(a){return 255-a}))},setHue:function(a){return new Color([a,this.hsb[1],this.hsb[2]],"hsb")},setSaturation:function(a){return new Color([this.hsb[0],a,this.hsb[2]],"hsb")},setBrightness:function(a){return new Color([this.hsb[0],this.hsb[1],a],"hsb")}});function $RGB(d,c,a){return new Color([d,c,a],"rgb")}function $HSB(d,c,a){return new Color([d,c,a],"hsb")}Array.extend({rgbToHsb:function(){var b=this[0],c=this[1],k=this[2];var h,g,i;var j=Math.max(b,c,k),f=Math.min(b,c,k);var l=j-f;i=j/255;g=(j!=0)?l/j:0;if(g==0){h=0}else{var d=(j-b)/l;var a=(j-c)/l;var m=(j-k)/l;if(b==j){h=m-a}else{if(c==j){h=2+d-m}else{h=4+a-d}}h/=6;if(h<0){h++}}return[Math.round(h*360),Math.round(g*100),Math.round(i*100)]},hsbToRgb:function(){var c=Math.round(this[2]/100*255);if(this[1]==0){return[c,c,c]}else{var a=this[0]%360;var g=a%60;var h=Math.round((this[2]*(100-this[1]))/10000*255);var d=Math.round((this[2]*(6000-this[1]*g))/600000*255);var b=Math.round((this[2]*(6000-this[1]*(60-g)))/600000*255);switch(Math.floor(a/60)){case 0:return[c,b,h];case 1:return[d,c,h];case 2:return[h,c,b];case 3:return[h,d,c];case 4:return[b,h,c];case 5:return[c,h,d]}}return false}});var Scroller=new Class({options:{area:20,velocity:1,onChange:function(a,b){this.element.scrollTo(a,b)}},initialize:function(b,a){this.setOptions(a);this.element=$(b);this.mousemover=([window,document].contains(b))?$(document.body):this.element},start:function(){this.coord=this.getCoords.bindWithEvent(this);this.mousemover.addListener("mousemove",this.coord)},stop:function(){this.mousemover.removeListener("mousemove",this.coord);this.timer=$clear(this.timer)},getCoords:function(a){this.page=(this.element==window)?a.client:a.page;if(!this.timer){this.timer=this.scroll.periodical(50,this)}},scroll:function(){var a=this.element.getSize();var d=this.element.getPosition();var c={x:0,y:0};for(var b in this.page){if(this.page[b]<(this.options.area+d[b])&&a.scroll[b]!=0){c[b]=(this.page[b]-this.options.area-d[b])*this.options.velocity}else{if(this.page[b]+this.options.area>(a.size[b]+d[b])&&a.scroll[b]+a.size[b]!=a.scrollSize[b]){c[b]=(this.page[b]-a.size[b]+this.options.area-d[b])*this.options.velocity}}}if(c.y||c.x){this.fireEvent("onChange",[a.scroll.x+c.x,a.scroll.y+c.y])}}});Scroller.implement(new Events,new Options);var Slider=new Class({options:{onChange:Class.empty,onComplete:Class.empty,onTick:function(a){this.knob.setStyle(this.p,a)},mode:"horizontal",steps:100,offset:0},initialize:function(d,a,b){this.element=$(d);this.knob=$(a);this.setOptions(b);this.previousChange=-1;this.previousEnd=-1;this.step=-1;this.element.addEvent("mousedown",this.clickedElement.bindWithEvent(this));var c,g;switch(this.options.mode){case"horizontal":this.z="x";this.p="left";c={x:"left",y:false};g="offsetWidth";break;case"vertical":this.z="y";this.p="top";c={x:false,y:"top"};g="offsetHeight"}this.max=this.element[g]-this.knob[g]+(this.options.offset*2);this.half=this.knob[g]/2;this.getPos=this.element["get"+this.p.capitalize()].bind(this.element);this.knob.setStyle("position","relative").setStyle(this.p,-this.options.offset);var f={};f[this.z]=[-this.options.offset,this.max-this.options.offset];this.drag=new Drag.Base(this.knob,{limit:f,modifiers:c,snap:0,onStart:function(){this.draggedKnob()}.bind(this),onDrag:function(){this.draggedKnob()}.bind(this),onComplete:function(){this.draggedKnob();this.end()}.bind(this)});if(this.options.initialize){this.options.initialize.call(this)}},set:function(a){this.step=a.limit(0,this.options.steps);this.checkStep();this.end();this.fireEvent("onTick",this.toPosition(this.step));return this},clickedElement:function(b){var a=b.page[this.z]-this.getPos()-this.half;a=a.limit(-this.options.offset,this.max-this.options.offset);this.step=this.toStep(a);this.checkStep();this.end();this.fireEvent("onTick",a)},draggedKnob:function(){this.step=this.toStep(this.drag.value.now[this.z]);this.checkStep()},checkStep:function(){if(this.previousChange!=this.step){this.previousChange=this.step;this.fireEvent("onChange",this.step)}},end:function(){if(this.previousEnd!==this.step){this.previousEnd=this.step;this.fireEvent("onComplete",this.step+"")}},toStep:function(a){return Math.round((a+this.options.offset)/this.max*this.options.steps)},toPosition:function(a){return this.max*a/this.options.steps}});Slider.implement(new Events);Slider.implement(new Options);var SmoothScroll=Fx.Scroll.extend({initialize:function(b){this.parent(window,b);this.links=(this.options.links)?$$(this.options.links):$$(document.links);var a=window.location.href.match(/^[^#]*/)[0]+"#";this.links.each(function(d){if(d.href.indexOf(a)!=0){return}var c=d.href.substr(a.length);if(c&&$(c)){this.useLink(d,c)}},this);if(!window.webkit419){this.addEvent("onComplete",function(){window.location.hash=this.anchor})}},useLink:function(b,a){b.addEvent("click",function(c){this.anchor=a;this.toElement(a);c.stop()}.bindWithEvent(this))}});var Sortables=new Class({options:{handles:false,onStart:Class.empty,onComplete:Class.empty,ghost:true,snap:3,onDragStart:function(a,b){b.setStyle("opacity",0.7);a.setStyle("opacity",0.7)},onDragComplete:function(a,b){a.setStyle("opacity",1);b.remove();this.trash.remove()}},initialize:function(d,b){this.setOptions(b);this.list=$(d);this.elements=this.list.getChildren();this.handles=(this.options.handles)?$$(this.options.handles):this.elements;this.bound={start:[],moveGhost:this.moveGhost.bindWithEvent(this)};for(var c=0,a=this.handles.length;c<a;c++){this.bound.start[c]=this.start.bindWithEvent(this,this.elements[c])}this.attach();if(this.options.initialize){this.options.initialize.call(this)}this.bound.move=this.move.bindWithEvent(this);this.bound.end=this.end.bind(this)},attach:function(){this.handles.each(function(b,a){b.addEvent("mousedown",this.bound.start[a])},this)},detach:function(){this.handles.each(function(b,a){b.removeEvent("mousedown",this.bound.start[a])},this)},start:function(c,b){this.active=b;this.coordinates=this.list.getCoordinates();if(this.options.ghost){var a=b.getPosition();this.offset=c.page.y-a.y;this.trash=new Element("div").inject(document.body);this.ghost=b.clone().inject(this.trash).setStyles({position:"absolute",left:a.x,top:c.page.y-this.offset});document.addListener("mousemove",this.bound.moveGhost);this.fireEvent("onDragStart",[b,this.ghost])}document.addListener("mousemove",this.bound.move);document.addListener("mouseup",this.bound.end);this.fireEvent("onStart",b);c.stop()},moveGhost:function(a){var b=a.page.y-this.offset;b=b.limit(this.coordinates.top,this.coordinates.bottom-this.ghost.offsetHeight);this.ghost.setStyle("top",b);a.stop()},move:function(f){var b=f.page.y;this.previous=this.previous||b;var a=((this.previous-b)>0);var d=this.active.getPrevious();var c=this.active.getNext();if(d&&a&&b<d.getCoordinates().bottom){this.active.injectBefore(d)}if(c&&!a&&b>c.getCoordinates().top){this.active.injectAfter(c)}this.previous=b},serialize:function(a){return this.list.getChildren().map(a||function(b){return this.elements.indexOf(b)},this)},end:function(){this.previous=null;document.removeListener("mousemove",this.bound.move);document.removeListener("mouseup",this.bound.end);if(this.options.ghost){document.removeListener("mousemove",this.bound.moveGhost);this.fireEvent("onDragComplete",[this.active,this.ghost])}this.fireEvent("onComplete",this.active)}});Sortables.implement(new Events,new Options);var Tips=new Class({options:{onShow:function(a){a.setStyle("visibility","visible")},onHide:function(a){a.setStyle("visibility","hidden")},maxTitleChars:30,showDelay:100,hideDelay:100,className:"tool",offsets:{x:16,y:16},fixed:false},initialize:function(b,a){this.setOptions(a);this.toolTip=new Element("div",{"class":this.options.className+"-tip",styles:{position:"absolute",top:"0",left:"0",visibility:"hidden"}}).inject(document.body);this.wrapper=new Element("div").inject(this.toolTip);$$(b).each(this.build,this);if(this.options.initialize){this.options.initialize.call(this)}},build:function(b){b.$tmp.myTitle=(b.href&&b.getTag()=="a")?b.href.replace("http://",""):(b.rel||false);if(b.title){var c=b.title.split("::");if(c.length>1){b.$tmp.myTitle=c[0].trim();b.$tmp.myText=c[1].trim()}else{b.$tmp.myText=b.title}b.removeAttribute("title")}else{b.$tmp.myText=false}if(b.$tmp.myTitle&&b.$tmp.myTitle.length>this.options.maxTitleChars){b.$tmp.myTitle=b.$tmp.myTitle.substr(0,this.options.maxTitleChars-1)+"&hellip;"}b.addEvent("mouseenter",function(d){this.start(b);if(!this.options.fixed){this.locate(d)}else{this.position(b)}}.bind(this));if(!this.options.fixed){b.addEvent("mousemove",this.locate.bindWithEvent(this))}var a=this.end.bind(this);b.addEvent("mouseleave",a);b.addEvent("trash",a)},start:function(a){this.wrapper.empty();if(a.$tmp.myTitle){this.title=new Element("span").inject(new Element("div",{"class":this.options.className+"-title"}).inject(this.wrapper)).setHTML(a.$tmp.myTitle)}if(a.$tmp.myText){this.text=new Element("span").inject(new Element("div",{"class":this.options.className+"-text"}).inject(this.wrapper)).setHTML(a.$tmp.myText)}$clear(this.timer);this.timer=this.show.delay(this.options.showDelay,this)},end:function(a){$clear(this.timer);this.timer=this.hide.delay(this.options.hideDelay,this)},position:function(a){var b=a.getPosition();this.toolTip.setStyles({left:b.x+this.options.offsets.x,top:b.y+this.options.offsets.y})},locate:function(b){var d={x:window.getWidth(),y:window.getHeight()};var a={x:window.getScrollLeft(),y:window.getScrollTop()};var c={x:this.toolTip.offsetWidth,y:this.toolTip.offsetHeight};var h={x:"left",y:"top"};for(var f in h){var g=b.page[f]+this.options.offsets[f];if((g+c[f]-a[f])>d[f]){g=b.page[f]-this.options.offsets[f]-c[f]}this.toolTip.setStyle(h[f],g)}},show:function(){if(this.options.timeout){this.timer=this.hide.delay(this.options.timeout,this)}this.fireEvent("onShow",[this.toolTip])},hide:function(){this.fireEvent("onHide",[this.toolTip])}});Tips.implement(new Events,new Options);var Group=new Class({initialize:function(){this.instances=$A(arguments);this.events={};this.checker={}},addEvent:function(b,a){this.checker[b]=this.checker[b]||{};this.events[b]=this.events[b]||[];if(this.events[b].contains(a)){return false}else{this.events[b].push(a)}this.instances.each(function(c,d){c.addEvent(b,this.check.bind(this,[b,c,d]))},this);return this},check:function(c,a,b){this.checker[c][b]=true;var d=this.instances.every(function(g,f){return this.checker[c][f]||false},this);if(!d){return}this.checker[c]={};this.events[c].each(function(f){f.call(this,this.instances,a)},this)}});var Accordion=Fx.Elements.extend({options:{onActive:Class.empty,onBackground:Class.empty,display:0,show:false,height:true,width:false,opacity:true,fixedHeight:false,fixedWidth:false,wait:false,alwaysHide:false},initialize:function(){var c,f,g,b;$each(arguments,function(k,j){switch($type(k)){case"object":c=k;break;case"element":b=$(k);break;default:var h=$$(k);if(!f){f=h}else{g=h}}});this.togglers=f||[];this.elements=g||[];this.container=$(b);this.setOptions(c);this.previous=-1;if(this.options.alwaysHide){this.options.wait=true}if($chk(this.options.show)){this.options.display=false;this.previous=this.options.show}if(this.options.start){this.options.display=false;this.options.show=false}this.effects={};if(this.options.opacity){this.effects.opacity="fullOpacity"}if(this.options.width){this.effects.width=this.options.fixedWidth?"fullWidth":"offsetWidth"}if(this.options.height){this.effects.height=this.options.fixedHeight?"fullHeight":"scrollHeight"}for(var d=0,a=this.togglers.length;d<a;d++){this.addSection(this.togglers[d],this.elements[d])}this.elements.each(function(j,h){if(this.options.show===h){this.fireEvent("onActive",[this.togglers[h],j])}else{for(var k in this.effects){j.setStyle(k,0)}}},this);this.parent(this.elements);if($chk(this.options.display)){this.display(this.options.display)}},addSection:function(f,c,h){f=$(f);c=$(c);var g=this.togglers.contains(f);var b=this.togglers.length;this.togglers.include(f);this.elements.include(c);if(b&&(!g||h)){h=$pick(h,b-1);f.injectBefore(this.togglers[h]);c.injectAfter(f)}else{if(this.container&&!g){f.inject(this.container);c.inject(this.container)}}var a=this.togglers.indexOf(f);f.addEvent("click",this.display.bind(this,a));if(this.options.height){c.setStyles({"padding-top":0,"border-top":"none","padding-bottom":0,"border-bottom":"none"})}if(this.options.width){c.setStyles({"padding-left":0,"border-left":"none","padding-right":0,"border-right":"none"})}c.fullOpacity=1;if(this.options.fixedWidth){c.fullWidth=this.options.fixedWidth}if(this.options.fixedHeight){c.fullHeight=this.options.fixedHeight}c.setStyle("overflow","hidden");if(!g){for(var d in this.effects){c.setStyle(d,0)}}return this},display:function(a){a=($type(a)=="element")?this.elements.indexOf(a):a;if((this.timer&&this.options.wait)||(a===this.previous&&!this.options.alwaysHide)){return this}this.previous=a;var b={};this.elements.each(function(f,d){b[d]={};var c=(d!=a)||(this.options.alwaysHide&&(f.offsetHeight>0));this.fireEvent(c?"onBackground":"onActive",[this.togglers[d],f]);for(var g in this.effects){b[d][g]=c?0:f[this.effects[g]]}},this);return this.start(b)},showThisHideOpen:function(a){return this.display(a)}});Fx.Accordion=Accordion;
/**
* The synergee general javascript code.
* This file contains all the necessary code used by the form widget.
*
* Copyright (c) 2007 Pyrameed all right reserved (http://www.pyrameed.com)
*/
/**
* Add two usefull method to the Element objects in order to determine
* its synergee web page parent component name and id.
* Those informations are usefull when dealing with AJAX requests.
*/
Element.extend({
/**
* Return the first synergee web page component name
*
* @return string The first synergee web page component name
*/
getComponentName : function() {
var currentElement = this;
while(!(currentElement.className && currentElement.className.test('Synergee-Web-Page-Component-(.*)-Container', 'i'))) {
currentElement = currentElement.getParent();
}
var regex = new RegExp('Synergee-Web-Page-Component-(.*)-Container', 'i');
var values = regex.exec(currentElement.className);
if (values && values.length == 2) {
return values[1];
} else {
return null;
}
},
/**
* Return the first synergee web page component id
*
* @return string The first synergee web page component id
*/
getComponentId : function() {
var currentElement = this;
while(!(currentElement.className && currentElement.className.test('Synergee-Web-Page-Component-(.*)-Container', 'i'))) {
currentElement = currentElement.getParent();
}
var regex = new RegExp('Synergee-Web-Page-Component-(.*)-Container', 'i');
var values = regex.exec(currentElement.className);
if (values && values.length == 2) {
return currentElement.getParent().id;
} else {
return null;
}
}
});
/**
* The getUrl method is added to the window object in order to be able to get the current page url easily
*/
window.extend({
/**
* Bookmark the current web page
*
* @param string title The page title
* @param string url The web page url
*/
bookmark : function(title, url) {
if (document.all) {
window.external.AddFavorite(url, title);
} else if (window.sidebar) {
window.sidebar.addPanel(title, url, "");
}
},
/**
* Return the current web page url
*
* @return string The current web page url
*/
getCurrentPageUrl : function() {
var currentPageUrl = window.location.href;
if(currentPageUrl.indexOf('#') != -1){
currentPageUrl = currentPageUrl.slice(0, currentPageUrl.indexOf('#'));
}
if (currentPageUrl.indexOf('?') != -1) {
return currentPageUrl.substr(0, currentPageUrl.indexOf('?'));
}
return currentPageUrl;
},
/**
* Return the current JSON url.
* This url is used by all the JSON invokations
*
* @param Element element The component element
* @return string The current JSON url
*/
getCurrentJsonUrl : function(element) {
var htmlElement = $E('html');
var windowUrl = window.getCurrentPageUrl();
if (htmlElement.getAttribute('designmode') == 'on') {
var webPageId = /webPageId=([0-9]*)/.exec(window.location.href);
return windowUrl + '?webPageId=' + webPageId[1] + '&componentName=' + element.getComponentName() + '&componentId=' + element.getComponentId();
} else {
return windowUrl + '?componentName=' + element.getComponentName() + '&componentId=' + element.getComponentId();
}
},
/**
* Return the currrent page location (the url without the page name)
*
* @return string The current page location
*/
getCurrentPageLocation : function() {
return window.getCurrentPageUrl().substr(0, window.getCurrentPageUrl().lastIndexOf('/')) + '/';
}
});
/**
* Docucment is extended to add the isInDesignMode method.
*/
document.extend({
/**
* This method return true when the page is displayed in the designer
* @return boolean True if the page is displayed in the designer
*/
isInDesignMode : function() {
var htmlElement = $E('html');
return (htmlElement.getAttribute('designmode') == 'on');
}
});
/**
* Decode the html entities
*
* @param String sa
* @return String
*/
function decodeHtmlEntities(sa) {
var ta=document.createElement("textarea");
ta.innerHTML=sa.replace(/</g,"&lt;").replace(/>/g,"&gt;");
return ta.value;
};
/**
* Pad a number regarding the number passed to the pad. By default 2
* @param number valueToPad The value to pad
* @param number pad The pad value
* @return string the padded number
*/
Number.padTo = function(valueToPad, pad) {
pad = pad ? pad : 2;
pad = Math.pow(10, pad-1);
return (valueToPad < pad) ? ("0" + valueToPad) : valueToPad;
};
/**
* Populate the date from an iso formatted string that represent a date
*
* @param string The iso formatted date
*/
Date.prototype.fromIso = function(isoFormattedDate) {
var dateTimeRegex = new RegExp('^([0-9]{4})-([0-9]{2})-([0-9]{2})[T ]([0-9]{2}):([0-9]{2}):([0-9]{2})(([+-])([0-9]{2})(:([0-9]{2}))?)?$');
var dateRegex = new RegExp('^([0-9]{4})-([0-9]{2})-([0-9]{2})$');
var timeRegex = new RegExp('^([0-9]{2}):([0-9]{2}):([0-9]{2})$');
if (dateTimeRegex.test(isoFormattedDate)) {
var values = dateTimeRegex.exec(isoFormattedDate);
var tmpDate = new Date(values[1], values[2] - 1, values[3], values[4], values[5], values[6]);
this.setTime(tmpDate.getTime());
//this.setTime(Date.parse(values[1] + ' ' + values[2] + ' ' + values[3] + ' ' + values[4] + ':' + values[5] + ':' + values[6]));
// The time zone is calculated
if(values[9]){
var timeZoneOffset = -1 * (values[9]*60 + (values[11]?values[11]*1:0)) * (values[8]=='-'?-1:1);
var currentTimeZoneOffset = this.getTimezoneOffset();
this.setTime(this.getTime() + (timeZoneOffset - currentTimeZoneOffset) * 60000);
}
} else if (dateRegex.test(isoFormattedDate)) {
var values = dateRegex.exec(isoFormattedDate);
var tmpDate = new Date(values[1], values[2] - 1, values[3]);
this.setTime(tmpDate.getTime());
//this.setTime(Date.parse(values[1] + ' ' + values[2] + ' ' + values[3]));
} else if (timeRegex.test(isoFormattedDate)) {
var values = timeRegex.exec(isoFormattedDate);
this.setHours(values[1]);
this.setMinutes(values[2]);
this.setSeconds(values[3]);
}
return this;
};
/**
* Return the iso formatted representation of the date
*
* @return string The iso formatted date
*/
Date.prototype.toIso = function() {
var formattedDate = this.getFullYear() + '-' + Number.padTo(this.getMonth()+1) + '-' + Number.padTo(this.getDate());
formattedDate = formattedDate + 'T' + Number.padTo(this.getHours()) + ':' + Number.padTo(this.getMinutes()) + ':' + Number.padTo(this.getSeconds());
return formattedDate;
};
/**
* Return formatted date
*
* @param string format The date output format
* @optionalparam string[] months The translated months
* @return string The formatted date
*/
Date.prototype.format = function(format, months) {
// The date is formatted
format = format.replace('dddd','dd');
format = format.replace('ddd', 'dd');
format = format.replace('dd', Number.padTo(this.getDate()));
format = format.replace('d', this.getDate());
format = format.replace('yyyy', this.getFullYear());
format = format.replace('yy', (new String(this.getFullYear())).substr(2,2));
format = format.replace('y', (new String(this.getFullYear())).substr(2,2));
// The time is formatted
format = format.replace('HH', Number.padTo(this.getHours()));
format = format.replace('H', this.getHours());
format = format.replace('mm', Number.padTo(this.getMinutes()));
format = format.replace('m', this.getMinutes());
format = format.replace('ss', Number.padTo(this.getSeconds()));
format = format.replace('s', this.getSeconds());
if(format.indexOf('MMMM') != -1 && months && months.length && months[this.getMonth()]){
format = format.replace('MMMM', months[this.getMonth()]);
}else{
format = format.replace('MMMM','MM');
format = format.replace('MMM', 'MM');
format = format.replace('MM', Number.padTo(this.getMonth()+1));
format = format.replace('M', this.getMonth()+1);
}
format = format.replace(/['"]/g,'');
return format;
};
/**
* Returns the week number for this date.
* @return int the week number
*/
Date.prototype.getWeek = function() {
var onejan = new Date(this.getFullYear(),0,1);
return Math.ceil((((this - onejan) / 86400000) + onejan.getDay())/7);
};
/**
* The synergee form javascript code.
* This file contains all the necessary code used by the form widget.
*
* Copyright (c) 2007 Pyrameed all right reserved (http://www.pyrameed.com)
*/
/**
* The field validator.
* This class is used to validate a form field
*/
var SynergeeFieldValidator = new Class({
/**
* The validation method
*
* @param Element field The field element to validate
* @return boolean True if the field is valid
*/
isValid : function(field) {
return true;
},
/**
* Return the vaidator name
*
* @return string The validator name
*/
getValidatorName : function() {
throw Error('The method "getValidatorName" has to be defined for the field validator classes.');
}
});
/**
* The not empty field validator
*/
var SynergeeFieldValidatorNotEmpty = SynergeeFieldValidator.extend({
/**
* The validation method
*
* @param Element field The field element to validate
* @return boolean True if the field is valid
*/
isValid : function(field) {
return (field.tagName.toLowerCase() == 'div' || field.getValue() != '');
},
/**
* Return the vaidator name
*
* @return string The validator name
*/
getValidatorName : function() {
return 'SynergeeFieldValidatorNotEmpty';
}
});
/**
* The email field validator
*/
var SynergeeFieldValidatorEmail = SynergeeFieldValidator.extend({
_emailRegex : new RegExp(/^([\w-\._\+%]+@(?:[\w-]+\.)+[\w]{2,6})?$/),
/**
* The validation method
*
* @param Element field The field element to validate
* @return boolean True if the field is valid
*/
isValid : function(field) {
return this._emailRegex.test(field.getValue());
},
/**
* Return the vaidator name
*
* @return string The validator name
*/
getValidatorName : function() {
return 'SynergeeFieldValidatorEmail';
}
});
/**
* The list field validator
*/
var SynergeeFieldValidatorList = SynergeeFieldValidator.extend({
/**
* The validation method
*
* @param Element field The field element to validate
* @return boolean True if the field is valid
*/
isValid : function(field) {
var inputElements = field.getElementsByTagName('input');
for (var i = 0; i < inputElements.length; i++) {
if (inputElements[i].checked) {
return true;
}
}
return false;
},
/**
* Return the vaidator name
*
* @return string The validator name
*/
getValidatorName : function() {
return 'SynergeeFieldValidatorList';
}
});
/**
* The alpha field validator
*/
var SynergeeFieldValidatorAlpha = SynergeeFieldValidator.extend({
_alphaRegex : new RegExp("^([a-zA-Z])*?$"),
/**
* The validation method
*
* @param Element field The field element to validate
* @return boolean True if the field is valid
*/
isValid : function(field) {
return this._alphaRegex.test(field.getValue());
},
/**
* Return the vaidator name
*
* @return string The validator name
*/
getValidatorName : function() {
return 'SynergeeFieldValidatorAlpha';
}
});
/**
* The numeric field validator
*/
var SynergeeFieldValidatorNum = SynergeeFieldValidator.extend({
_numRegex : new RegExp("^([0-9])*?$"),
/**
* The validation method
*
* @param Element field The field element to validate
* @return boolean True if the field is valid
*/
isValid : function(field) {
return this._numRegex.test(field.getValue());
},
/**
* Return the vaidator name
*
* @return string The validator name
*/
getValidatorName : function() {
return 'SynergeeFieldValidatorNum';
}
});
/**
* The alphanumeric field validator
*/
var SynergeeFieldValidatorAlnum = SynergeeFieldValidator.extend({
_alnumRegex : new RegExp("^([a-zA-Z0-9])*?$"),
/**
* The validation method
*
* @param Element field The field element to validate
* @return boolean True if the field is valid
*/
isValid : function(field) {
return this._alnumRegex.test(field.getValue());
},
/**
* Return the vaidator name
*
* @return string The validator name
*/
getValidatorName : function() {
return 'SynergeeFieldValidatorAlnum';
}
});
/**
* The date field validator
*/
var SynergeeFieldValidatorDate = SynergeeFieldValidator.extend({
/**
* The validation method
*
* @param Element field The field element to validate
* @return boolean True if the field is valid
*/
isValid : function(field) {
return (field.tagName.toLowerCase() == 'div' || field.getValue() != '');
},
/**
* Return the vaidator name
*
* @return string The validator name
*/
getValidatorName : function() {
return 'SynergeeFieldValidatorDate';
}
});
/**
* The Birthdate field validator
*/
var SynergeeFieldValidatorBirthdate = SynergeeFieldValidator.extend({
initialize : function(minDate, maxDate, format) {
if(minDate){
this._minDate = (new Date()).fromIso(minDate);
}   else{
this._minDate = null;
}
if(maxDate){
this._maxDate = (new Date()).fromIso(maxDate);
}   else{
this._maxDate = null;
}
if(format) this._format = format;
},
/**
* The validation method
*
* @param Element field The field element to validate
* @return boolean True if the field is valid
*/
isValid : function(field) {
// always return true, the backend check the right dates
////
//        var fieldValue = field.getValue();
//        if(fieldValue != '' && (new Date()).fromIso(fieldValue)){
//            fieldValue = (new Date()).fromIso(fieldValue);
//            if(this._maxDate && this._maxDate.getTime() < fieldValue){
//                return false;
//
//            }
//            if(this._minDate && this._minDate.getTime() > fieldValue){
//                return false;
//            }
//        }
return true;
},
/**
* Return the vaidator name
*
* @return string The validator name
*/
getValidatorName : function() {
return 'SynergeeFieldValidatorBirthdate';
}
});
/**
* The Birthdate field validator
*/
var SynergeeFieldValidatorLevel = SynergeeFieldValidator.extend({
initialize : function(minimumLevel, maximumLevel) {
if(minimumLevel){
this._minimumLevel = minimumLevel;
}   else{
this._minimumLevel = 1;
}
if(maximumLevel){
this._maximumLevel = maximumLevel;
}   else{
this._maximumLevel = 7;
}
},
/**
* The validation method
*
* @param Element field The field element to validate
* @return boolean True if the field is valid
*/
isValid : function(field) {
var selIndex = field.selectedIndex;
var comboValue = field.options[selIndex].value;
if(comboValue){
if(comboValue >= this._minimumLevel && comboValue <= this._maximumLevel){
return true;
}
return false;
}
return true;
},
/**
* Return the validator name
*
* @return string The validator name
*/
getValidatorName : function() {
return 'SynergeeFieldValidatorLevel';
}
});
/**
* The form validator.
* This class is used to validate the fields of a specific form
*/
var SynergeeFormValidator = new Class({
_formElement : null,
_onComplete : null,
_fields : [],
_submitButton : null,
/**
* The constructor
*
* @param Element formElement The form element
* @param function onComplete The function called once the form has been submitted (this function is called by the ajax component)
*/
initialize: function(formElement, onComplete) {
this._fields = [];
this._formElement = formElement;
this._scroller = new Fx.Scroll(window, {offset: {'x': 0, 'y': -50}});
if (typeof(onComplete) == 'function') {
this._onComplete = onComplete;
} else {
this._onComplete = eval(onComplete);
}
this._formElement.addEvent('submit', function(e) {
new Event(e).stop();
});
// The error message functions are added to the form
this._formElement._errorMessage = new Fx.Slide(this._formElement.getElementsByTagName('div')[0]);
this._formElement.getElementsByTagName('div')[0].setStyle('clear', 'both');
this._formElement._errorMessage.hide();
this._formElement.displayErrorMessage = function() {
if (this._errorMessage) {
this._errorMessage.stop();
this._errorMessage.slideIn();
}
};
this._formElement.hideErrorMessage = function() {
if (this._errorMessage) {
this._errorMessage.stop();
this._errorMessage.slideOut();
}
};
},
/**
* This method is called when the response of the form has been received.
* The goal of this method is to analyse the response and if some fields were
* invalid the error message is displayed again.
* If no invalidate fields are returned, the user defined function (set on the onComplete
* attribute of the form tag) is called with the response as parameter.
*
* @param string response The response returned by the server
*/
onComplete : function(response) {
// The submit button is ungrayed
if (this._submitButton) {
this._submitButton.removeClass('WaitingButton');
this._submitButton.removeProperty('disabled');
}
// We check if the invalidateFields is empty or not
try {
var response = Json.evaluate(response);
if (response.invalidFields && response.invalidFields.length) {
try {
if ($(response.invalidFields[0])) {
$(response.invalidFields[0]).focus();
} else {
$('id_' + response.invalidFields[0]).focus();
}
}catch(ex){
// if the element is not focusable (it's hidden or something)
}
response.invalidFields.each(function(fieldName) {
if ($(fieldName)) {
$(fieldName).displayErrorMessage();
} else {
$('id_' + fieldName).displayErrorMessage();
}
});
try {
if(response.invalidFields[0]){
if ($(response.invalidFields[0])) {
if($(response.invalidFields[0]).getStyle('display') == 'none'){
var element = $(response.invalidFields[0]).getParent();
}else {
var element = $(response.invalidFields[0]);
}
this._scroller.toElement(element);
} else {
if($('id_' + response.invalidFields[0]).getStyle('display') == 'none'){
var element = $('id_' + response.invalidFields[0]).getParent();
}else {
var element = $('id_' + response.invalidFields[0]);
}
this._scroller.toElement(element);
}
}
}catch (ex){}
this._formElement.fireEvent('invalidfield');
} else if (response.isValid) {
// The user defined on complete method is called with the response as parameter
this._onComplete(response.response);
} else {
// The request wasn't valid, then a global message has to be send
this._formElement.displayErrorMessage();
}
} catch(ex) {
// The response is not a valid JSON encoded one
this._formElement.displayErrorMessage();
this._formElement.fireEvent('invalidfield');
}
},
/**
* Validate the fields of the form
*
* @param boolean ajaxCall True if an ajax call has to be done
*/
validate : function(ajaxCall) {
// The submit button is grayed
if (this._submitButton) {
this._submitButton.addClass('WaitingButton');
this._submitButton.setProperty('disabled', 'disabled');
}
// The error message of the form is hidden
this._formElement.hideErrorMessage();
var valid = true;
var invalidFields = new Array();
for (var i = 0; i < this._fields.length; i++) {
for (var j = 0; j < this._fields[i].getValidators().length; j++) {
if (!this._fields[i].getValidators()[j].isValid(this._fields[i])) {
invalidFields.push(this._fields[i]);
}
}
}
// The valid fields error message are hidden
for (var i = 0; i < this._fields.length; i++) {
if (!invalidFields.contains(this._fields[i])) {
this._fields[i].hideErrorMessage();
}
}
// The invalidate fields error message are displayed
if (invalidFields.length) {
invalidFields[0].focus();
// The page is scrolled to the first error message
this._scroller.toElement(invalidFields[0]);
for (var i = 0; i < invalidFields.length; i++) {
invalidFields[i].displayErrorMessage();
}
// The submit button is ungrayed
if (this._submitButton) {
this._submitButton.removeClass('WaitingButton');
this._submitButton.removeProperty('disabled');
}
return false;
} else {
if (ajaxCall) {
// The right page location is set to the action attribute of the form
this._formElement.setProperty('action', window.getCurrentPageUrl());
// The componentName and componentId fields are populated
if (!this._formElement.getChildren().contains($('id_componentName'))) {
if ($('id_componentName')) {
$('id_componentName').remove();
}
(new Element('input').setProperties({name:'componentName',id:'id_componentName',type:'hidden'})).injectBefore(this._formElement.getChildren()[0]);
}
$('id_componentName').setProperty('value', this._formElement.getComponentName());
if (!this._formElement.getChildren().contains($('id_componentId'))) {
if ($('id_componentId')) {
$('id_componentId').remove();
}
(new Element('input').setProperties({name:'componentId',id:'id_componentId',type:'hidden'})).injectBefore(this._formElement.getChildren()[1]);
}
$('id_componentId').setProperty('value', this._formElement.getComponentId());
// The validationInformation field is populated with all the validation information
if (!this._formElement.getChildren().contains($('id_validationInformation'))) {
if ($('id_validationInformation')) {
$('id_validationInformation').remove();
}
(new Element('input').setProperties({name:'validationInformation',id:'id_validationInformation',type:'hidden'})).injectBefore(this._formElement.getChildren()[2]);
}
$('id_validationInformation').setProperty('value', this.getValidationInformation());
// All the fields are ok, then the form is sent
this._formElement.fireEvent('ajaxcall');
this._formElement.send({onComplete:this.onComplete.bind(this)});
return false;
} else {
return true;
}
}
},
createTips : function(field) {
},
/**
* Add a field to the validator
*
* @param Element field The field to validate
* @optionalParam string[] validators The list of validators to apply to the field
*/
addField : function(field, validators) {
var validatorObjects = [];
if (validators && validators.length) {
for (var i = 0; i < validators.length; i++) {
validatorObjects.push(validators[i]);
}
if (field.getStyle('position') == "absolute") {
if (field.parentNode.getElementsByTagName('div').length) {
field._errorImage = field.getParent().getElementsByTagName('div')[0];
field._errorMessage = new Tips(field._errorImage);
}
} else {
if (field.parentNode.getElementsByTagName('div').length) {
if (field.parentNode.getElementsByTagName('div').length) {
// The error fx object is created
field._errorMessage = new Fx.Slide(field.parentNode.getElementsByTagName('div')[0]);
field.getParent().getElementsByTagName('div')[0].addClass('Synergee-Web-Page-Form-FieldErrorMessage-Container');
field._errorMessage.hide();
}
}
}
}
field._validators = validatorObjects;
field.getValidators = function() {
return this._validators;
};
field.displayErrorMessage = function() {
if (this._errorMessage) {
if (this.getStyle('position') == "absolute") {
this._errorImage.setStyle('display', 'block');
} else {
this.getParent().addClass('InvalidField')
this._errorMessage.stop();
//                    this.setStyle('margin-top', '2px');
this._errorMessage.slideIn();
}
}
};
field.hideErrorMessage = function() {
if (this._errorMessage) {
if (this.getStyle('position') == "absolute") {
this._errorImage.setStyle('display', 'none');
} else {
this.getParent().removeClass('InvalidField')
this._errorMessage.stop();
//                    this.setStyle('margin-top', '0px');
this._errorMessage.slideOut();
}
}
};
this._fields.push(field);
// A check to determine if this is the submit button
if (field && field.getProperty('type') && field.getProperty('type').toLowerCase() == 'submit') {
this._submitButton = field;
}
},
/**
* Return the validation information encoded in JSON.
* This is a hash table that have the field name as key and a coma separated validator name as values.
*
* @return string
*/
getValidationInformation : function() {
var validationInformation = {};
this._fields.each(function(field) {
var validators = new Array();
field.getValidators().each(function(validator) {
validators.push(validator.getValidatorName());
});
validationInformation[field.getProperty('name')] = validators.join(',');
});
return Json.toString(validationInformation);
}
});
/**
*  Create a waiting panel object
*
*/
var WaitingPanel = Events.extend({
initialize: function(el) {
if($(el).waitingPanel){
return $(el).waitingPanel;
}
this.el = $(el);
this.el.waitingPanel = this;
},
buildObjects : function() {
// the waiting panel layer and fx style are created
this._waitingPanelLayer = new Element('div', {'class': 'WaitingPanel-AjaxLoading','id':'waitingPanel'});
this._waitingFxStyle = new Fx.Style(this._waitingPanelLayer, 'opacity', {duration:100});
this._waitingPanelLayer.injectAfter(document.getElementsByTagName('div')[0]);
this._waitingPanelLayer.setStyle('position', 'absolute');
this._waitingPanelLayer.setStyles({
opacity : 0
});
this._waitingImageLayer = new Element('div', {'class': 'WaitingPanel-AjaxLoadingImage','id':'waitingPanel'});
this._waitingImageFxStyle = new Fx.Style(this._waitingImageLayer, 'opacity', {duration:100});
this._waitingImageLayer.injectAfter(this._waitingPanelLayer);
this._waitingImageLayer.setStyle('position', 'absolute');
this._waitingImageLayer.setStyles({
opacity : 0
});
},
/**
*  Display a waiting panel layer over the element
*/
display : function() {
if (!this._waitingPanelLayer) {
this.buildObjects();
}
// refresh the height of the waiting panel layer
var coordinates = this.el.getCoordinates();
this._waitingPanelLayer.elementTo = this.el;
this._waitingPanelLayer.setStyles({
height : (coordinates.height + 2),
width : (coordinates.width + 2),
left : (coordinates.left - 2),
top : (coordinates.top - 2)
});
this._waitingImageLayer.setStyles({
height : (coordinates.height + 2),
width : (coordinates.width + 2),
left : (coordinates.left - 2),
top : (coordinates.top - 2)
});
this._waitingFxStyle.start(0, 0.5);
this._waitingImageFxStyle.start(0, 1);
},
/**
* Hide the waiting panel
*/
hide : function() {
if (this._waitingPanelLayer.getStyle('opacity') != 0) {
this._waitingFxStyle.start(0.5, 0);
this._waitingImageFxStyle.start(1, 0);
}
}
});
var ResultHelper = Events.extend( {
initialize: function (resultsPerPage, countResults) {
this.resultsPerPage = resultsPerPage;
this.countResults = countResults;
if (this.resultsPerPage == 0)
this.resultsPerPage = 1;
this.countPages = Math.ceil(this.countResults / this.resultsPerPage);
this.currentPage = 0;
},
getCountPages : function () {
return this.countPages;
},
getCurrentPage : function () {
return this.currentPage;
},
injectNavBar : function(toInjectNavBarClass, tagName, navClassBaseName, separator) {
var s = "";
for (var i=0; i<this.countPages; i++) {
var className = navClassBaseName + i;
s += "<" + (tagName == 'button' ? tagName + ' type="button"' : tagName) + " class='" + className + "'>" + (i+1) + "</" + tagName + ">";
if (i != this.countPages - 1) {
s += separator;
}
}
$$('.'+toInjectNavBarClass).setHTML(s);
},
nextPage : function () {
if (this.currentPage < this.countPages - 1) {
this.fireEvent('onPageChangeBefore');
this.currentPage++;
this.fireEvent('onPageChangeAfter');
if (this.currentPage == this.countPages - 1) {
this.fireEvent('onPageChangeUpperLimit');
}
}
},
previousPage : function () {
if (this.currentPage > 0) {
this.fireEvent('onPageChangeBefore');
this.currentPage--;
this.fireEvent('onPageChangeAfter');
if (this.currentPage == 0) {
this.fireEvent('onPageChangeLowerLimit');
}
}
},
selectPage : function (page) {
if (page >=0 && page < this.countPages) {
this.fireEvent('onPageChangeBefore');
this.currentPage = page;
this.fireEvent('onPageChangeAfter');
if (this.currentPage == 0) {
this.fireEvent('onPageChangeLowerLimit');
}
if (this.currentPage == this.countPages - 1) {
this.fireEvent('onPageChangeUpperLimit');
}
}
},
firstPage : function () {
this.selectPage(0);
},
lastPage : function () {
this.selectPage(this.countPages - 1);
}
});
window.addEvent('domready', function () {
var searchForm = $('Home-SchoolSearch');
var searchPanelToggle = $('Home-SchoolSearchAdvancedToggle');
var searchPanel = $('Home-SchoolSearchAdvancedPanel');
var closedLabelNode;
var openLabelNode;
function ensureToggleLabelNodes(closedLabel, openLabel) {
closedLabelNode = searchPanelToggle.getElement('.mas-school-search-compact__toggle-label--closed');
openLabelNode = searchPanelToggle.getElement('.mas-school-search-compact__toggle-label--open');
if (closedLabelNode && openLabelNode) {
return;
}
searchPanelToggle.empty();
closedLabelNode = new Element('span', {
'class': 'mas-school-search-compact__toggle-label mas-school-search-compact__toggle-label--closed'
});
closedLabelNode.appendText(closedLabel);
openLabelNode = new Element('span', {
'class': 'mas-school-search-compact__toggle-label mas-school-search-compact__toggle-label--open'
});
openLabelNode.appendText(openLabel);
closedLabelNode.inject(searchPanelToggle);
openLabelNode.inject(searchPanelToggle);
}
function setToggleLabel(label) {
searchPanelToggle.setProperty('aria-label', label);
searchPanelToggle.setProperty('title', label);
}
if (!searchForm || !searchPanelToggle || !searchPanel || searchPanelToggle.getProperty('data-mas-panel-bound') == 'true') {
return;
}
var closedLabel = searchPanelToggle.getProperty('data-label-closed') || '+ de choix';
var openLabel = searchPanelToggle.getProperty('data-label-open') || 'Moins de choix';
searchPanelToggle.setProperty('data-mas-panel-bound', 'true');
ensureToggleLabelNodes(closedLabel, openLabel);
setToggleLabel(closedLabel);
searchPanelToggle.setProperty('aria-expanded', 'false');
searchPanel.setProperty('aria-hidden', 'true');
searchPanel.setProperty('hidden', 'hidden');
searchPanel.removeClass('is-open');
searchForm.removeClass('is-advanced-open');
searchPanelToggle.addEvent('click', function () {
var searchPanelOpen = !searchPanel.hasClass('is-open');
setToggleLabel(searchPanelOpen ? openLabel : closedLabel);
searchPanelToggle.setProperty('aria-expanded', searchPanelOpen ? 'true' : 'false');
searchPanel.setProperty('aria-hidden', searchPanelOpen ? 'false' : 'true');
if (searchPanelOpen) {
searchPanel.removeProperty('hidden');
searchPanel.addClass('is-open');
searchForm.addClass('is-advanced-open');
} else {
searchPanel.removeClass('is-open');
searchPanel.setProperty('hidden', 'hidden');
searchForm.removeClass('is-advanced-open');
}
});
});
window.addEvent('domready', function () {
if (!document.body || !document.body.classList.contains('MoveAndStudyCore-home')) {
return;
}
function normalizeDestinationLabel(rawLabel) {
var label = (rawLabel || '')
.replace(/\u00a0/g, ' ')
.replace(/\s+/g, ' ')
.trim();
label = label.replace(/^s[ée]jour linguistique\s+/i, '');
label = label.replace(/^s[ée]jour\s+/i, '');
label = label.replace(/^ados?\s+/i, '');
label = label.replace(/^(?:en|au|aux|a|à)\s+/i, '');
return label.trim();
}
function updateDestinationTitle(titleElement) {
var heading = titleElement.querySelector('h1, h2, h3, h4, p');
var link = titleElement.querySelector('a');
var rawLabel = (link && link.textContent) || (heading && heading.textContent) || titleElement.textContent;
var cleanLabel = normalizeDestinationLabel(rawLabel);
if (!cleanLabel) {
return;
}
if (link && heading && link.parentNode === heading) {
link.textContent = cleanLabel;
} else if (link && heading && heading.parentNode === link) {
heading.textContent = cleanLabel;
} else if (link) {
link.textContent = cleanLabel;
} else if (heading) {
heading.textContent = cleanLabel;
} else {
titleElement.textContent = cleanLabel;
}
titleElement.setAttribute('data-mas-destination-clean', 'true');
titleElement.setAttribute('data-mas-destination-label', cleanLabel);
}
Array.prototype.forEach.call(document.querySelectorAll('.mas-destinations .mas-destination-card__title'), function (titleElement) {
if (titleElement.getAttribute('data-mas-destination-clean') === 'true') {
return;
}
updateDestinationTitle(titleElement);
});
});
(function () {
var defaultProfile = "young";
function onReady(callback) {
if (document.readyState === "loading") {
document.addEventListener("DOMContentLoaded", callback);
return;
}
callback();
}
onReady(function () {
var stage = document.querySelector("[data-mas-home-hero-stage]");
var source = document.querySelector("[data-mas-home-hero-source]");
var image = document.querySelector("[data-mas-home-hero-image]");
var cards = Array.prototype.slice.call(document.querySelectorAll("[data-mas-home-profile-card]"));
var activeProfile = defaultProfile;
var lockedProfile = defaultProfile;
var hasUserLock = false;
var isRuntimeReady = document.readyState === "complete";
var supportsWebp = false;
var hasScheduledWarmup = false;
if (!stage || !source || !image || !cards.length) {
return;
}
supportsWebp = /\.webp(?:$|\?)/i.test(image.currentSrc || "");
function getCardAsset(card) {
var webp = card.getAttribute("data-mas-home-hero-webp");
var fallback = card.getAttribute("data-mas-home-hero-fallback");
return supportsWebp && webp ? webp : fallback;
}
function findCard(profile) {
return cards.filter(function (card) {
return card.getAttribute("data-mas-home-profile") === profile;
})[0] || null;
}
function updateCardState(profile) {
cards.forEach(function (card) {
var cardProfile = card.getAttribute("data-mas-home-profile");
card.classList.toggle("is-active", cardProfile === profile);
card.classList.toggle("is-locked", cardProfile === lockedProfile);
card.setAttribute("aria-pressed", cardProfile === lockedProfile ? "true" : "false");
});
}
function updateImage(card) {
var webp = card.getAttribute("data-mas-home-hero-webp");
var fallback = card.getAttribute("data-mas-home-hero-fallback");
var currentSrc = image.currentSrc || image.getAttribute("src") || "";
if (!isRuntimeReady) {
return;
}
if (!webp || !fallback || currentSrc.indexOf(webp) !== -1 || currentSrc.indexOf(fallback) !== -1) {
return;
}
stage.classList.add("is-switching");
window.requestAnimationFrame(function () {
source.setAttribute("srcset", webp);
image.setAttribute("src", fallback);
image.setAttribute("srcset", fallback + " 1672w");
stage.setAttribute("data-mas-home-hero-active", card.getAttribute("data-mas-home-profile"));
window.requestAnimationFrame(function () {
stage.classList.remove("is-switching");
});
});
}
function setProfile(profile) {
var card = findCard(profile);
if (!card) {
return;
}
activeProfile = profile;
updateCardState(profile);
updateImage(card);
}
function warmupSecondaryImages() {
cards.forEach(function (card) {
var asset = getCardAsset(card);
if (!asset || asset === image.currentSrc || asset === image.getAttribute("src")) {
return;
}
new Image().src = asset;
});
}
function scheduleWarmup() {
if (hasScheduledWarmup) {
return;
}
hasScheduledWarmup = true;
if (window.requestIdleCallback) {
window.requestIdleCallback(warmupSecondaryImages, { timeout: 1800 });
return;
}
window.setTimeout(warmupSecondaryImages, 240);
}
cards.forEach(function (card) {
var profile = card.getAttribute("data-mas-home-profile");
card.addEventListener("click", function (event) {
if (event.target && event.target.closest && event.target.closest(".mas-home-main-card-arrow")) {
return;
}
event.preventDefault();
lockedProfile = profile;
hasUserLock = true;
setProfile(profile);
stage.setAttribute("data-mas-home-hero-locked", lockedProfile);
});
card.addEventListener("keydown", function (event) {
if (event.key !== " " && event.key !== "Spacebar") {
return;
}
event.preventDefault();
card.click();
});
});
lockedProfile = defaultProfile;
stage.setAttribute("data-mas-home-hero-locked", lockedProfile);
setProfile(activeProfile);
if (!isRuntimeReady) {
window.addEventListener("load", function () {
isRuntimeReady = true;
supportsWebp = /\.webp(?:$|\?)/i.test(image.currentSrc || "") || supportsWebp;
setProfile(activeProfile);
scheduleWarmup();
}, { once: true });
return;
}
scheduleWarmup();
});
}());
(function () {
function onReady(callback) {
if (document.readyState === "loading") {
document.addEventListener("DOMContentLoaded", callback);
return;
}
callback();
}
function closest(element, selector) {
while (element && element.nodeType === 1) {
if (element.matches && element.matches(selector)) {
return element;
}
element = element.parentNode;
}
return null;
}
function bindMegaMenu(menu) {
var mobileToggle = menu.querySelector(".mas-mega-menu__toggle");
var items = Array.prototype.slice.call(menu.querySelectorAll("[data-mas-mega-item]"));
var triggers = Array.prototype.slice.call(menu.querySelectorAll("[data-mas-mega-trigger]"));
var languageButtons = Array.prototype.slice.call(menu.querySelectorAll("[data-mas-menu-lang]"));
var countryCards = Array.prototype.slice.call(menu.querySelectorAll("[data-mas-country-langs]"));
var closeTimer = null;
function setItemOpen(item, isOpen) {
var trigger = item.querySelector("[data-mas-mega-trigger]");
var panel = item.querySelector("[data-mas-mega-panel]");
if (!trigger || !panel) {
return;
}
item.classList.toggle("is-open", isOpen);
trigger.setAttribute("aria-expanded", isOpen ? "true" : "false");
panel.hidden = !isOpen;
}
function closeAll(exceptItem) {
items.forEach(function (item) {
if (item !== exceptItem) {
setItemOpen(item, false);
}
});
}
function openItem(item) {
window.clearTimeout(closeTimer);
closeAll(item);
setItemOpen(item, true);
}
function closeSoon(item) {
window.clearTimeout(closeTimer);
closeTimer = window.setTimeout(function () {
setItemOpen(item, false);
}, 140);
}
if (mobileToggle) {
mobileToggle.addEventListener("click", function () {
var isOpen = menu.classList.toggle("is-mobile-open");
mobileToggle.setAttribute("aria-expanded", isOpen ? "true" : "false");
if (!isOpen) {
closeAll();
}
});
}
triggers.forEach(function (trigger) {
trigger.addEventListener("click", function () {
var item = closest(trigger, "[data-mas-mega-item]");
var isOpen = item && item.classList.contains("is-open");
closeAll(item);
if (item) {
setItemOpen(item, !isOpen);
}
});
trigger.addEventListener("keydown", function (event) {
if (event.key === "Escape") {
closeAll();
trigger.focus();
}
});
});
items.forEach(function (item) {
item.addEventListener("mouseenter", function () {
if (window.matchMedia("(min-width: 901px)").matches) {
openItem(item);
}
});
item.addEventListener("mouseleave", function () {
if (window.matchMedia("(min-width: 901px)").matches) {
closeSoon(item);
}
});
item.addEventListener("focusin", function () {
openItem(item);
});
item.addEventListener("focusout", function (event) {
if (!item.contains(event.relatedTarget)) {
closeSoon(item);
}
});
});
document.addEventListener("click", function (event) {
if (!menu.contains(event.target)) {
closeAll();
menu.classList.remove("is-mobile-open");
if (mobileToggle) {
mobileToggle.setAttribute("aria-expanded", "false");
}
}
});
document.addEventListener("keydown", function (event) {
if (event.key === "Escape") {
closeAll();
menu.classList.remove("is-mobile-open");
if (mobileToggle) {
mobileToggle.setAttribute("aria-expanded", "false");
mobileToggle.focus();
}
}
});
languageButtons.forEach(function (button) {
button.addEventListener("click", function () {
var language = button.getAttribute("data-mas-menu-lang");
languageButtons.forEach(function (item) {
item.classList.toggle("is-active", item === button);
});
countryCards.forEach(function (card) {
var languages = card.getAttribute("data-mas-country-langs") || "";
var shouldShow = language === "all" || languages.split(" ").indexOf(language) !== -1;
card.classList.toggle("is-hidden", !shouldShow);
});
});
});
}
onReady(function () {
Array.prototype.slice.call(document.querySelectorAll("[data-mas-mega-menu]")).forEach(bindMegaMenu);
});
}());
(function () {
"use strict";
function toArray(nodeList) {
return Array.prototype.slice.call(nodeList || []);
}
function toggleHidden(element, shouldHide) {
if (!element) {
return;
}
if (shouldHide) {
element.setAttribute("hidden", "hidden");
} else {
element.removeAttribute("hidden");
}
}
function pluralize(count, singular, plural) {
return count + " " + (count > 1 ? plural : singular);
}
function buildMetaText(visibleCount, totalCount) {
if (totalCount <= 0) {
return "Aucune école disponible pour cette formule.";
}
if (visibleCount >= totalCount) {
return pluralize(totalCount, "école disponible", "écoles disponibles") + " pour cette formule.";
}
return "Aperçu de " + pluralize(visibleCount, "école", "écoles") + " sur " + totalCount + " disponibles pour cette formule.";
}
function initCourseTypeDirectory() {
var body = document.body;
if (!body || body.className.indexOf("MoveAndStudyCore-course-type") === -1) {
return;
}
var directory = document.querySelector("[data-course-type-directory]");
var explorer = document.querySelector("[data-course-type-explorer]");
if (!directory || !explorer) {
return;
}
var pageSize = parseInt(directory.getAttribute("data-page-size"), 10) || 4;
var countryCarousel = explorer.querySelector("[data-course-type-country-carousel]");
var countryPageSize = countryCarousel ? parseInt(countryCarousel.getAttribute("data-country-page-size"), 10) || 4 : 4;
var state = {
language: directory.getAttribute("data-initial-language") || "",
country: directory.getAttribute("data-initial-country") || "",
schoolPageIndex: 0,
countryPageIndex: 0
};
var meta = directory.querySelector("[data-course-type-meta]");
var nextButton = directory.querySelector("[data-course-type-next]");
var previousButton = directory.querySelector("[data-course-type-prev]");
var pagination = directory.querySelector("[data-course-type-pagination]");
var track = directory.querySelector("[data-course-type-track]");
var fallbackList = directory.querySelector("[data-course-type-directory-list]");
var countryTrack = countryCarousel ? countryCarousel.querySelector("[data-course-type-country-track]") : null;
var countryFallbackList = countryCarousel ? countryCarousel.querySelector("[data-course-type-country-list]") : null;
var countryNextButton = countryCarousel ? countryCarousel.querySelector("[data-course-type-country-next]") : null;
var countryPreviousButton = countryCarousel ? countryCarousel.querySelector("[data-course-type-country-prev]") : null;
var countryPagination = countryCarousel ? countryCarousel.querySelector("[data-course-type-country-pagination]") : null;
var schoolCards = toArray(directory.querySelectorAll("[data-course-type-school-card]"));
var languageButtons = toArray(explorer.querySelectorAll("[data-course-type-language]"));
var countryButtons = toArray(explorer.querySelectorAll(".mas-course-type-country-toolbar [data-course-type-country]"));
var countryCards = toArray(explorer.querySelectorAll("[data-course-type-country-card]"));
var renderedSchoolKey = "";
var renderedSchoolPages = [];
var renderedSchoolCards = [];
var renderedCountryKey = "";
var renderedCountryPages = [];
var renderedCountryCards = [];
if (!track || !fallbackList || !countryTrack || !countryFallbackList) {
return;
}
directory.classList.add("is-enhanced");
countryCarousel.classList.add("is-enhanced");
function getLanguageCode(element) {
return element.getAttribute("data-language-code") || "";
}
function getCountryCode(element) {
return element.getAttribute("data-country-code") || "";
}
function getMatchingSchoolCards() {
return schoolCards.filter(function (card) {
var cardLanguage = getLanguageCode(card);
var cardCountry = getCountryCode(card);
if (state.language && cardLanguage !== state.language) {
return false;
}
if (state.country && cardCountry !== state.country) {
return false;
}
return true;
});
}
function getCurrentStateKey() {
return [state.language, state.country].join("|");
}
function getCountryStateKey() {
return state.language || "";
}
function clearTrack(element) {
while (element.firstChild) {
element.removeChild(element.firstChild);
}
}
function clearElement(element) {
while (element && element.firstChild) {
element.removeChild(element.firstChild);
}
}
function createPage(cards, pageClass, gridClass) {
var page = document.createElement("section");
var list = document.createElement("ul");
page.className = pageClass;
list.className = gridClass;
cards.forEach(function (card) {
toggleHidden(card, false);
list.appendChild(card);
});
page.appendChild(list);
return page;
}
function buildSchoolPages(matchingCards) {
var index;
var pageCards;
var page;
clearTrack(track);
renderedSchoolPages = [];
renderedSchoolCards = matchingCards.slice();
schoolCards.forEach(function (card) {
toggleHidden(card, true);
});
for (index = 0; index < matchingCards.length; index += pageSize) {
pageCards = matchingCards.slice(index, index + pageSize);
page = createPage(pageCards, "mas-course-type-directory__page", "mas-course-type-directory__page-grid");
page.setAttribute("data-course-type-page", "");
renderedSchoolPages.push(page);
track.appendChild(page);
}
renderedSchoolKey = getCurrentStateKey();
}
function getMatchingCountryCards() {
return countryCards.filter(function (card) {
var cardLanguage = getLanguageCode(card);
if (state.language && cardLanguage !== state.language) {
return false;
}
return true;
});
}
function buildCountryPages(matchingCards) {
var index;
var pageCards;
var page;
clearTrack(countryTrack);
renderedCountryPages = [];
renderedCountryCards = matchingCards.slice();
countryCards.forEach(function (card) {
toggleHidden(card, true);
});
for (index = 0; index < matchingCards.length; index += countryPageSize) {
pageCards = matchingCards.slice(index, index + countryPageSize);
page = createPage(pageCards, "mas-course-type-country-carousel__page", "mas-course-type-country-carousel__page-grid");
page.setAttribute("data-course-type-country-page", "");
renderedCountryPages.push(page);
countryTrack.appendChild(page);
}
renderedCountryKey = getCountryStateKey();
}
function buildPagination(container, pageCount, activeIndex, labelPrefix, onSelect) {
var index;
var button;
var isActive;
if (!container) {
return;
}
clearElement(container);
if (pageCount <= 1) {
toggleHidden(container, true);
return;
}
for (index = 0; index < pageCount; index += 1) {
isActive = index === activeIndex;
button = document.createElement("button");
button.type = "button";
button.className = "mas-course-type-slider-pagination__dot" + (isActive ? " is-active" : "");
button.setAttribute("aria-label", labelPrefix + " " + (index + 1));
button.setAttribute("aria-pressed", isActive ? "true" : "false");
if (isActive) {
button.setAttribute("aria-current", "true");
}
button.title = labelPrefix + " " + (index + 1);
button.addEventListener("click", (function (targetIndex) {
return function () {
onSelect(targetIndex);
};
}(index)));
container.appendChild(button);
}
toggleHidden(container, false);
}
function updateSchoolPager() {
var pageCount = renderedSchoolPages.length;
var visibleCount = 0;
var isLastPage;
var isFirstPage;
if (pageCount && state.schoolPageIndex >= pageCount) {
state.schoolPageIndex = pageCount - 1;
}
isLastPage = !pageCount || state.schoolPageIndex >= pageCount - 1;
isFirstPage = state.schoolPageIndex <= 0;
renderedSchoolPages.forEach(function (page, index) {
page.classList.toggle("is-active", index === state.schoolPageIndex);
});
if (pageCount) {
visibleCount = Math.max(0, Math.min(pageSize, renderedSchoolCards.length - (state.schoolPageIndex * pageSize)));
}
track.style.transform = pageCount ? "translate3d(-" + (state.schoolPageIndex * 100) + "%, 0, 0)" : "";
if (meta) {
meta.textContent = buildMetaText(visibleCount, renderedSchoolCards.length);
}
if (nextButton) {
toggleHidden(nextButton, pageCount <= 1 || isLastPage);
}
if (previousButton) {
toggleHidden(previousButton, pageCount <= 1 || isFirstPage);
}
buildPagination(pagination, pageCount, state.schoolPageIndex, "Slide écoles", function (targetIndex) {
state.schoolPageIndex = targetIndex;
updateSchoolPager();
});
}
function updateCountryPager() {
var pageCount = renderedCountryPages.length;
var isLastPage;
var isFirstPage;
if (pageCount && state.countryPageIndex >= pageCount) {
state.countryPageIndex = pageCount - 1;
}
isLastPage = !pageCount || state.countryPageIndex >= pageCount - 1;
isFirstPage = state.countryPageIndex <= 0;
renderedCountryPages.forEach(function (page, index) {
page.classList.toggle("is-active", index === state.countryPageIndex);
});
countryTrack.style.transform = pageCount ? "translate3d(-" + (state.countryPageIndex * 100) + "%, 0, 0)" : "";
if (countryNextButton) {
toggleHidden(countryNextButton, pageCount <= 1 || isLastPage);
}
if (countryPreviousButton) {
toggleHidden(countryPreviousButton, pageCount <= 1 || isFirstPage);
}
buildPagination(countryPagination, pageCount, state.countryPageIndex, "Slide pays", function (targetIndex) {
state.countryPageIndex = targetIndex;
updateCountryPager();
});
}
function updateLanguageButtons() {
languageButtons.forEach(function (button) {
var isActive = (button.getAttribute("data-course-type-language") || "") === state.language;
button.classList.toggle("is-active", isActive);
button.setAttribute("aria-pressed", isActive ? "true" : "false");
});
}
function updateCountryControls() {
countryButtons.forEach(function (button) {
var buttonLanguage = getLanguageCode(button);
var isVisible = !state.language || buttonLanguage === state.language;
var isActive = isVisible && (button.getAttribute("data-course-type-country") || "") === state.country;
toggleHidden(button, !isVisible);
button.classList.toggle("is-active", isActive);
button.setAttribute("aria-pressed", isActive ? "true" : "false");
});
countryCards.forEach(function (card) {
var cardCountry = getCountryCode(card);
var cardButton = card.querySelector("[data-course-type-country]");
var isActive = cardCountry === state.country;
card.classList.toggle("is-active", isActive);
if (cardButton) {
cardButton.setAttribute("aria-pressed", isActive ? "true" : "false");
}
});
}
function updateSchoolCards() {
var matchingCards = getMatchingSchoolCards();
if (renderedSchoolKey !== getCurrentStateKey()) {
buildSchoolPages(matchingCards);
}
updateSchoolPager();
}
function updateCountryCards() {
var matchingCards = getMatchingCountryCards();
if (renderedCountryKey !== getCountryStateKey()) {
buildCountryPages(matchingCards);
}
updateCountryPager();
}
function update() {
updateLanguageButtons();
updateCountryControls();
updateCountryCards();
updateSchoolCards();
}
languageButtons.forEach(function (button) {
button.addEventListener("click", function () {
state.language = button.getAttribute("data-course-type-language") || "";
state.country = "";
state.schoolPageIndex = 0;
state.countryPageIndex = 0;
update();
});
});
countryButtons.forEach(function (button) {
button.addEventListener("click", function () {
state.language = getLanguageCode(button) || state.language;
state.country = button.getAttribute("data-course-type-country") || "";
state.schoolPageIndex = 0;
update();
});
});
countryCards.forEach(function (card) {
var button = card.querySelector("[data-course-type-country]");
if (!button) {
return;
}
button.addEventListener("click", function () {
state.language = getLanguageCode(card) || state.language;
state.country = getCountryCode(card);
state.schoolPageIndex = 0;
update();
});
});
if (countryNextButton) {
countryNextButton.addEventListener("click", function () {
state.countryPageIndex += 1;
updateCountryPager();
});
}
if (countryPreviousButton) {
countryPreviousButton.addEventListener("click", function () {
state.countryPageIndex = Math.max(0, state.countryPageIndex - 1);
updateCountryPager();
});
}
if (nextButton) {
nextButton.addEventListener("click", function () {
state.schoolPageIndex += 1;
updateSchoolPager();
});
}
if (previousButton) {
previousButton.addEventListener("click", function () {
state.schoolPageIndex = Math.max(0, state.schoolPageIndex - 1);
updateSchoolPager();
});
}
update();
}
if (document.readyState === "loading") {
document.addEventListener("DOMContentLoaded", initCourseTypeDirectory);
} else {
initCourseTypeDirectory();
}
})();
(function () {
function onReady(callback) {
if (document.readyState === "loading") {
document.addEventListener("DOMContentLoaded", callback);
return;
}
callback();
}
function bindFavouriteSchools(root) {
var images = Array.prototype.slice.call(root.querySelectorAll("[data-mas-favourite-slide]"));
var slides = Array.prototype.slice.call(root.querySelectorAll("[data-mas-favourite-info]"));
var dots = Array.prototype.slice.call(root.querySelectorAll("[data-mas-favourite-dot]"));
var previous = root.querySelector("[data-mas-favourite-prev]");
var next = root.querySelector("[data-mas-favourite-next]");
var current = 0;
var timer = null;
function setActive(index) {
if (!slides.length) {
return;
}
current = (index + slides.length) % slides.length;
images.forEach(function (image, imageIndex) {
image.classList.toggle("is-active", imageIndex === current);
});
slides.forEach(function (slide, slideIndex) {
slide.classList.toggle("is-active", slideIndex === current);
});
dots.forEach(function (dot, dotIndex) {
dot.classList.toggle("is-active", dotIndex === current);
});
}
function stop() {
window.clearInterval(timer);
timer = null;
}
function start() {
if (slides.length < 2 || timer) {
return;
}
timer = window.setInterval(function () {
setActive(current + 1);
}, 4500);
}
if (previous) {
previous.addEventListener("click", function () {
setActive(current - 1);
stop();
});
}
if (next) {
next.addEventListener("click", function () {
setActive(current + 1);
stop();
});
}
dots.forEach(function (dot) {
dot.addEventListener("click", function () {
setActive(parseInt(dot.getAttribute("data-mas-favourite-dot"), 10) || 0);
stop();
});
});
root.addEventListener("mouseenter", stop);
root.addEventListener("mouseleave", start);
setActive(0);
start();
}
onReady(function () {
Array.prototype.slice.call(document.querySelectorAll("[data-mas-favourite-schools]")).forEach(bindFavouriteSchools);
});
}());
(function () {
var apiBase = "/synergee/MoveAndStudyCountryExchange.php";
var cacheLifetime = 6 * 60 * 60 * 1000;
var supportedTargets = ["EUR", "USD", "GBP", "CHF", "CAD", "AUD", "NZD", "JPY"];
function onReady(callback) {
if (document.readyState === "loading") {
document.addEventListener("DOMContentLoaded", callback);
return;
}
callback();
}
function getCurrencyFromText(text) {
var normalized = (text || "").toLowerCase();
var currencies = [
{code: "AUD", pattern: /dollar australien|aud/},
{code: "CAD", pattern: /dollar canadien|cad/},
{code: "USD", pattern: /dollar americain|dollar américain|usd/},
{code: "NZD", pattern: /dollar neo zelandais|dollar néo zélandais|nzd/},
{code: "GBP", pattern: /livre sterling|gbp/},
{code: "JPY", pattern: /yen|jpy/},
{code: "KRW", pattern: /won|krw/},
{code: "MXN", pattern: /peso mexicain|mxn/},
{code: "ARS", pattern: /peso argentin|ars/},
{code: "ZAR", pattern: /rand|zar/},
{code: "EUR", pattern: /euro|eur/},
{code: "CHF", pattern: /franc suisse|chf/}
];
var index;
for (index = 0; index < currencies.length; index += 1) {
if (currencies[index].pattern.test(normalized)) {
return currencies[index].code;
}
}
return "";
}
function normalizeCurrency(value) {
var currency = (value || "").toUpperCase().replace(/[^A-Z]/g, "");
return /^[A-Z]{3}$/.test(currency) ? currency : "";
}
function isSupportedTarget(value) {
return supportedTargets.indexOf(normalizeCurrency(value)) !== -1;
}
function getStoredCurrency() {
var keys = [
"SynergeeCurrency",
"synergee.currency",
"currency",
"masCurrency",
"booking.currency"
];
var index;
var value;
try {
for (index = 0; index < keys.length; index += 1) {
value = normalizeCurrency(window.localStorage && window.localStorage.getItem(keys[index]));
if (isSupportedTarget(value)) {
return value;
}
}
for (index = 0; index < keys.length; index += 1) {
value = normalizeCurrency(window.sessionStorage && window.sessionStorage.getItem(keys[index]));
if (isSupportedTarget(value)) {
return value;
}
}
} catch (error) {
return "";
}
return "";
}
function getDomCurrency(card) {
var selector = [
"select[name='currency']",
"input[name='currency']",
"[data-visitor-currency]",
"[data-website-currency]",
"[data-currency]"
].join(",");
var nodes = Array.prototype.slice.call(document.querySelectorAll(selector));
var index;
var node;
var value;
for (index = 0; index < nodes.length; index += 1) {
node = nodes[index];
if (card && card.contains(node)) {
continue;
}
value = normalizeCurrency(node.value || node.getAttribute("data-visitor-currency") || node.getAttribute("data-website-currency") || node.getAttribute("data-currency"));
if (isSupportedTarget(value)) {
return value;
}
}
return "";
}
function getLocaleCurrency() {
var locale = (navigator.language || (navigator.languages && navigator.languages[0]) || "").toLowerCase();
if (!locale) {
return "";
}
if (/-(us|pr|gu|as|vi)$/.test(locale)) {
return "USD";
}
if (/-(gb|gg|im|je)$/.test(locale)) {
return "GBP";
}
if (/-ch$/.test(locale)) {
return "CHF";
}
if (/-ca$/.test(locale)) {
return "CAD";
}
if (/-au$/.test(locale)) {
return "AUD";
}
if (/-nz$/.test(locale)) {
return "NZD";
}
if (/-jp$/.test(locale)) {
return "JPY";
}
if (/^(fr|de|es|it|nl|pt|fi|sv|da|el|ga|et|lv|lt|mt|sk|sl|hr)(-|$)/.test(locale)) {
return "EUR";
}
return "";
}
function getTargetCurrency(card, sourceCurrency) {
var fallback = normalizeCurrency(card.getAttribute("data-fallback-currency")) || "EUR";
var candidates = [
normalizeCurrency(card.getAttribute("data-target-currency")),
getStoredCurrency(),
getDomCurrency(card),
getLocaleCurrency(),
fallback
];
var index;
var currency;
for (index = 0; index < candidates.length; index += 1) {
currency = normalizeCurrency(candidates[index]);
if (isSupportedTarget(currency) && currency != sourceCurrency) {
return currency;
}
}
return sourceCurrency == "EUR" ? "USD" : "EUR";
}
function formatDate(dateValue) {
try {
return new Intl.DateTimeFormat("fr-FR", {
day: "2-digit",
month: "2-digit",
year: "numeric"
}).format(new Date(dateValue + "T12:00:00Z"));
} catch (error) {
return dateValue;
}
}
function formatRate(value, currency) {
try {
return new Intl.NumberFormat("fr-FR", {
style: "currency",
currency: currency,
maximumFractionDigits: 4
}).format(value);
} catch (error) {
return Math.round(value * 10000) / 10000 + " " + currency;
}
}
function formatRoundedAmount(value, currency) {
try {
return new Intl.NumberFormat("fr-FR", {
style: "currency",
currency: currency,
minimumFractionDigits: 2,
maximumFractionDigits: 2
}).format(value);
} catch (error) {
return Math.round(value * 100) / 100 + " " + currency;
}
}
function formatUpdated(dateValue) {
var consultedAt = "";
try {
consultedAt = new Intl.DateTimeFormat("fr-FR", {
hour: "2-digit",
minute: "2-digit"
}).format(new Date());
} catch (error) {
consultedAt = "";
}
return "Dernière mise à jour : " + formatDate(dateValue) + (consultedAt ? " · consulté à " + consultedAt : "");
}
function isoDate(date) {
return date.toISOString().slice(0, 10);
}
function getCachedJson(cacheKey) {
var stored;
var parsed;
try {
stored = window.sessionStorage && window.sessionStorage.getItem(cacheKey);
if (!stored) {
return null;
}
parsed = JSON.parse(stored);
if (!parsed || !parsed.createdAt || (Date.now() - parsed.createdAt) > cacheLifetime) {
window.sessionStorage.removeItem(cacheKey);
return null;
}
return parsed.payload || null;
} catch (error) {
return null;
}
}
function setCachedJson(cacheKey, payload) {
try {
if (window.sessionStorage) {
window.sessionStorage.setItem(cacheKey, JSON.stringify({
createdAt: Date.now(),
payload: payload
}));
}
} catch (error) {
return;
}
}
function fetchJson(url, cacheKey) {
var cached = getCachedJson(cacheKey);
var controller;
var timeoutId;
var fetchOptions = {};
if (cached) {
return Promise.resolve(cached);
}
if (window.AbortController) {
controller = new window.AbortController();
fetchOptions.signal = controller.signal;
timeoutId = window.setTimeout(function () {
controller.abort();
}, 10000);
}
return window.fetch(url, fetchOptions)
.then(function (response) {
if (timeoutId) {
window.clearTimeout(timeoutId);
}
if (!response.ok) {
throw new Error("Exchange rate unavailable");
}
return response.json();
})
.then(function (data) {
setCachedJson(cacheKey, data);
return data;
})
.catch(function (error) {
if (timeoutId) {
window.clearTimeout(timeoutId);
}
throw error;
});
}
function getLatestRate(data, targetCurrency) {
var latest;
if (Array.isArray(data)) {
latest = data.filter(function (item) {
return item && item.quote == targetCurrency && typeof item.rate !== "undefined";
}).pop();
return latest ? parseFloat(latest.rate) : NaN;
}
if (data && typeof data.rate !== "undefined") {
return parseFloat(data.rate);
}
if (data && data.rates && typeof data.rates[targetCurrency] !== "undefined") {
return parseFloat(data.rates[targetCurrency]);
}
return NaN;
}
function getLatestDate(data) {
if (Array.isArray(data) && data.length && data[data.length - 1].date) {
return data[data.length - 1].date;
}
return data && data.date ? data.date : isoDate(new Date());
}
function getDateFromIso(value) {
var parts = String(value || "").split("-");
if (parts.length != 3) {
return new Date();
}
return new Date(parseInt(parts[0], 10), parseInt(parts[1], 10) - 1, parseInt(parts[2], 10));
}
function normalizeSeries(data, targetCurrency) {
var rows = [];
if (Array.isArray(data)) {
data.forEach(function (item) {
if (item && item.quote == targetCurrency && typeof item.rate !== "undefined") {
rows.push({
date: item.date,
rate: parseFloat(item.rate)
});
}
});
return rows;
}
Object.keys((data && data.rates) || {}).sort().forEach(function (date) {
if (data.rates[date] && typeof data.rates[date][targetCurrency] !== "undefined") {
rows.push({
date: date,
rate: parseFloat(data.rates[date][targetCurrency])
});
}
});
return rows;
}
function renderChart(polyline, series) {
var values = series.map(function (item) {
return item.rate;
}).filter(function (value) {
return !isNaN(value);
});
var min;
var max;
var range;
var width;
var height;
var points;
if (!values.length) {
return;
}
min = Math.min.apply(Math, values);
max = Math.max.apply(Math, values);
range = max - min || 1;
width = 180;
height = 44;
points = values.map(function (value, index) {
var x = values.length == 1 ? width : (index / (values.length - 1)) * width;
var y = height - ((value - min) / range) * (height - 6) - 3;
return Math.round(x * 10) / 10 + "," + Math.round(y * 10) / 10;
});
polyline.setAttribute("points", points.join(" "));
}
function setUnavailable(exchange, pair, rate, updated, sourceCurrency, targetCurrency) {
applyExchangeDisplay(null, exchange, pair, sourceCurrency, targetCurrency, "forward");
if (rate) {
rate.textContent = "Taux de change momentanément indisponible";
}
if (updated) {
updated.textContent = "La monnaie reste indiquée ci-contre.";
}
if (exchange) {
exchange.classList.add("is-unavailable");
exchange.classList.remove("is-ready");
}
}
function applyExchangeDisplay(card, exchange, pair, fromCurrency, toCurrency, direction) {
var resolvedDirection = direction == "reverse" ? "reverse" : "forward";
if (pair && fromCurrency) {
pair.textContent = fromCurrency + " \u2192 " + toCurrency;
}
if (card) {
card.setAttribute("data-mas-exchange-from-currency", fromCurrency || "");
card.setAttribute("data-mas-exchange-to-currency", toCurrency || "");
card.setAttribute("data-mas-exchange-direction", resolvedDirection);
}
if (exchange) {
exchange.setAttribute("data-mas-exchange-from-currency", fromCurrency || "");
exchange.setAttribute("data-mas-exchange-to-currency", toCurrency || "");
exchange.setAttribute("data-mas-exchange-direction", resolvedDirection);
}
}
function getDirectionalState(sourceCurrency, targetCurrency, latestRate, direction) {
var normalizedDirection = direction == "reverse" ? "reverse" : "forward";
var resolvedRate = parseFloat(latestRate);
if (isNaN(resolvedRate) || !isFinite(resolvedRate) || resolvedRate <= 0) {
return null;
}
if (normalizedDirection == "reverse") {
return {
direction: normalizedDirection,
fromCurrency: targetCurrency,
toCurrency: sourceCurrency,
rate: 1 / resolvedRate,
operator: "\u2248"
};
}
return {
direction: normalizedDirection,
fromCurrency: sourceCurrency,
toCurrency: targetCurrency,
rate: resolvedRate,
operator: "="
};
}
function renderExchangeState(card, exchange, pair, rate, updated, toggle, sourceCurrency, targetCurrency, latestRate, latestDate, direction) {
var state = getDirectionalState(sourceCurrency, targetCurrency, latestRate, direction);
if (!state) {
setUnavailable(exchange, pair, rate, updated, sourceCurrency, targetCurrency);
if (toggle) {
toggle.hidden = true;
toggle.disabled = true;
toggle.setAttribute("aria-pressed", "false");
}
return false;
}
applyExchangeDisplay(card, exchange, pair, state.fromCurrency, state.toCurrency, state.direction);
if (rate) {
rate.textContent = "1 " + state.fromCurrency + " " + state.operator + " " + formatRate(state.rate, state.toCurrency);
}
if (updated) {
updated.textContent = "100 " + state.fromCurrency + " \u2248 " + formatRoundedAmount(state.rate * 100, state.toCurrency) + " \u00b7 mis \u00e0 jour le " + formatDate(latestDate);
}
if (toggle) {
toggle.hidden = false;
toggle.disabled = false;
toggle.setAttribute("aria-pressed", state.direction == "reverse" ? "true" : "false");
}
if (exchange) {
exchange.classList.add("is-ready");
exchange.classList.remove("is-unavailable");
}
return true;
}
function getExchangeState(card) {
return card && card._masExchangeState ? card._masExchangeState : null;
}
function syncExchangeCard(card) {
var state = getExchangeState(card);
if (!state) {
return false;
}
return renderExchangeState(
card,
state.exchange,
state.pair,
state.rate,
state.updated,
state.toggle,
state.sourceCurrency,
state.targetCurrency,
state.latestRateValue,
state.latestDateValue,
state.currentDirection
);
}
function toggleExchangeCard(card) {
var state = getExchangeState(card);
if (!state || isNaN(state.latestRateValue) || !isFinite(state.latestRateValue) || state.latestRateValue <= 0) {
return false;
}
state.currentDirection = state.currentDirection == "forward" ? "reverse" : "forward";
return syncExchangeCard(card);
}
function getClosestExchangeNode(node, attributeName) {
var currentNode = node;
if (currentNode && currentNode.nodeType !== 1 && currentNode.parentNode && currentNode.parentNode.nodeType === 1) {
currentNode = currentNode.parentNode;
}
while (currentNode && currentNode.nodeType === 1) {
if (currentNode.hasAttribute(attributeName)) {
return currentNode;
}
currentNode = currentNode.parentNode;
}
return null;
}
function hydrateExchange(card) {
var copy = card.querySelector(".mas-country-info__currency-copy");
var exchange = card.querySelector(".mas-country-info__exchange");
var pair = card.querySelector("[data-mas-exchange-pair]");
var rate = card.querySelector("[data-mas-exchange-rate]");
var updated = card.querySelector("[data-mas-exchange-updated]");
var toggle = card.querySelector("[data-mas-exchange-toggle]");
var chart = card.querySelector("[data-mas-exchange-chart]");
var sourceCurrency = getCurrencyFromText(copy ? copy.textContent : "");
var targetCurrency = getTargetCurrency(card, sourceCurrency);
var now;
var start;
var latestUrl;
var historyUrl;
var cachePrefix;
var latestRateValue = NaN;
var latestDateValue = "";
var currentDirection = "forward";
card._masExchangeState = {
exchange: exchange,
pair: pair,
rate: rate,
updated: updated,
toggle: toggle,
sourceCurrency: sourceCurrency,
targetCurrency: targetCurrency,
latestRateValue: latestRateValue,
latestDateValue: latestDateValue,
currentDirection: currentDirection
};
if (!sourceCurrency || sourceCurrency == targetCurrency || !window.fetch) {
setUnavailable(exchange, pair, rate, updated, sourceCurrency, targetCurrency);
if (toggle) {
toggle.hidden = true;
toggle.disabled = true;
toggle.setAttribute("aria-pressed", "false");
}
return;
}
applyExchangeDisplay(card, exchange, pair, sourceCurrency, targetCurrency, currentDirection);
if (toggle) {
toggle.hidden = true;
toggle.disabled = true;
}
now = new Date();
latestUrl = apiBase + "?base=" + encodeURIComponent(sourceCurrency) + "&quotes=" + encodeURIComponent(targetCurrency);
cachePrefix = "mas.exchange." + sourceCurrency + "." + targetCurrency + "." + isoDate(now);
fetchJson(latestUrl, cachePrefix + ".latest")
.then(function (data) {
latestRateValue = getLatestRate(data, targetCurrency);
latestDateValue = getLatestDate(data);
card._masExchangeState.latestRateValue = latestRateValue;
card._masExchangeState.latestDateValue = latestDateValue;
if (isNaN(latestRateValue) || !isFinite(latestRateValue) || latestRateValue <= 0) {
throw new Error("Exchange rate missing");
}
syncExchangeCard(card);
if (!chart) {
if (chart) {
chart.setAttribute("points", "");
}
return;
}
start = getDateFromIso(latestDateValue);
start.setMonth(start.getMonth() - 3);
historyUrl = apiBase + "?from=" + encodeURIComponent(isoDate(start)) + "&to=" + encodeURIComponent(latestDateValue) + "&base=" + encodeURIComponent(sourceCurrency) + "&quotes=" + encodeURIComponent(targetCurrency);
return fetchJson(historyUrl, cachePrefix + ".history." + latestDateValue)
.then(function (historyData) {
var series = normalizeSeries(historyData, targetCurrency);
renderChart(chart, series);
if (exchange && series.length > 1) {
exchange.classList.add("has-chart");
}
})
.catch(function () {
chart.setAttribute("points", "");
if (exchange) {
exchange.classList.remove("has-chart");
}
});
})
.catch(function () {
setUnavailable(exchange, pair, rate, updated, sourceCurrency, targetCurrency);
});
}
onReady(function () {
if (!document._masCountryExchangeDelegated) {
document.addEventListener("click", function (event) {
var toggle = getClosestExchangeNode(event.target, "data-mas-exchange-toggle");
var card;
if (!toggle) {
return;
}
card = getClosestExchangeNode(toggle, "data-mas-country-exchange");
if (!card) {
return;
}
event.preventDefault();
toggleExchangeCard(card);
});
document._masCountryExchangeDelegated = true;
}
Array.prototype.slice.call(document.querySelectorAll("[data-mas-country-exchange]")).forEach(hydrateExchange);
});
}());
(function () {
var mobileMedia = window.matchMedia ? window.matchMedia("(max-width: 759px)") : null;
var collapsedHeight = 416;
function onReady(callback) {
if (document.readyState === "loading") {
document.addEventListener("DOMContentLoaded", callback);
return;
}
callback();
}
function setExpanded(root, toggle, expanded) {
root.classList.toggle("is-expanded", expanded);
toggle.setAttribute("aria-expanded", expanded ? "true" : "false");
toggle.textContent = expanded ? "Réduire" : "Continuer à lire";
}
function resetCountryIntroItems(content) {
Array.prototype.slice.call(content.children).forEach(function (item) {
item.classList.remove("mas-country-intro__item", "is-hidden-by-default");
});
}
function getCountryIntroItemsToHide(content, isMobile) {
var items = Array.prototype.slice.call(content.children);
var hiddenItems = [];
var firstHeadingSeen = false;
var firstTextCardSeen = false;
var afterFirstGroup = false;
items.forEach(function (item) {
var tagName = item.tagName ? item.tagName.toUpperCase() : "";
var isHeading = tagName === "H3";
var isTextCard = tagName === "P" || tagName === "UL" || tagName === "OL";
item.classList.add("mas-country-intro__item");
if (isMobile) {
if (!firstHeadingSeen && isHeading) {
firstHeadingSeen = true;
return;
}
if (!firstTextCardSeen && isTextCard) {
firstTextCardSeen = true;
return;
}
hiddenItems.push(item);
return;
}
if (isHeading) {
if (!firstHeadingSeen) {
firstHeadingSeen = true;
return;
}
afterFirstGroup = true;
}
if (afterFirstGroup) {
hiddenItems.push(item);
}
});
return hiddenItems;
}
function syncCountryIntro(root) {
var toggle = root.querySelector("[data-mas-country-intro-toggle]");
var content = root.querySelector("#Home-Txt2 .Synergee-Web-Page-Component-Text-Container");
var isMobile = mobileMedia ? mobileMedia.matches : window.innerWidth < 760;
var hiddenItems;
if (!toggle || !content) {
return;
}
resetCountryIntroItems(content);
hiddenItems = getCountryIntroItemsToHide(content, isMobile);
root.classList.toggle("is-collapsible", hiddenItems.length > 0);
toggle.hidden = hiddenItems.length === 0;
if (!hiddenItems.length) {
setExpanded(root, toggle, true);
return;
}
hiddenItems.forEach(function (item) {
item.classList.add("is-hidden-by-default");
});
if (!root.hasAttribute("data-mas-country-intro-initialized")) {
setExpanded(root, toggle, false);
} else {
toggle.setAttribute("aria-expanded", root.classList.contains("is-expanded") ? "true" : "false");
toggle.textContent = root.classList.contains("is-expanded") ? "Réduire" : "Continuer à lire";
}
}
function initCountryIntro(root) {
var toggle = root.querySelector("[data-mas-country-intro-toggle]");
var resizeTimer = null;
if (!toggle) {
return;
}
toggle.addEventListener("click", function () {
setExpanded(root, toggle, !root.classList.contains("is-expanded"));
});
function refresh() {
syncCountryIntro(root);
root.setAttribute("data-mas-country-intro-initialized", "true");
}
if (mobileMedia) {
if (mobileMedia.addEventListener) {
mobileMedia.addEventListener("change", refresh);
} else if (mobileMedia.addListener) {
mobileMedia.addListener(refresh);
}
}
window.addEventListener("resize", function () {
window.clearTimeout(resizeTimer);
resizeTimer = window.setTimeout(refresh, 120);
});
refresh();
}
onReady(function () {
Array.prototype.slice.call(document.querySelectorAll("[data-mas-country-intro]")).forEach(initCountryIntro);
});
}());
(function () {
var forecastUrl = "https://api.open-meteo.com/v1/forecast";
var cacheLifetime = 60 * 60 * 1000;
var clockTimer = null;
var clocks = [];
function onReady(callback) {
if (document.readyState === "loading") {
document.addEventListener("DOMContentLoaded", callback);
return;
}
callback();
}
function normalize(value) {
return (value || "")
.toLowerCase()
.replace(/[’']/g, " ")
.replace(/[^a-z0-9\u00c0-\u017f]+/g, " ")
.replace(/\s+/g, " ")
.trim();
}
function getCityName(card) {
var title = card.querySelector(".mas-destination-card__title");
var heading = title ? title.querySelector("h1, h2, h3, h4, p, a") : null;
return (heading ? heading.textContent : title ? title.textContent : "").replace(/\s+/g, " ").trim();
}
function getMarkers(root) {
var markers = {
byName: {},
list: []
};
Array.prototype.slice.call(root.querySelectorAll(".mas-country-map__markers [data-title][data-lat][data-lng]")).forEach(function (marker) {
var cityMarker = {
title: marker.getAttribute("data-title"),
latitude: marker.getAttribute("data-lat"),
longitude: marker.getAttribute("data-lng")
};
var key = normalize(cityMarker.title);
if (key) {
markers.byName[key] = cityMarker;
markers.list.push(cityMarker);
}
});
return markers;
}
function getCityOverviewMarker() {
var marker = document.querySelector(".MoveAndStudyCore.MoveAndStudyCore-city .mas-city-map__markers [data-city][data-lat][data-lng]");
if (!marker) {
return null;
}
return {
title: marker.getAttribute("data-city") || "",
latitude: marker.getAttribute("data-lat"),
longitude: marker.getAttribute("data-lng")
};
}
function findMarkerForCity(markers, cityName) {
var cityKey = normalize(cityName);
var candidates;
if (!cityKey || !markers) {
return null;
}
if (markers.byName[cityKey]) {
return markers.byName[cityKey];
}
candidates = markers.list.filter(function (marker) {
var markerKey = normalize(marker.title);
return markerKey && (markerKey.indexOf(cityKey) !== -1 || cityKey.indexOf(markerKey) !== -1);
});
return candidates.length == 1 ? candidates[0] : null;
}
function isFiniteNumber(value) {
return value !== null && value !== "" && isFinite(parseFloat(value));
}
function weatherIconSvg(iconKey) {
switch (iconKey) {
case "sun":
return '<svg viewBox="0 0 24 24" focusable="false"><circle cx="12" cy="12" r="3.2"></circle><path d="M12 2.8v2.3M12 18.9v2.3M21.2 12h-2.3M5.1 12H2.8M18.5 5.5l-1.6 1.6M7.1 16.9l-1.6 1.6M18.5 18.5l-1.6-1.6M7.1 7.1 5.5 5.5"></path></svg>';
case "sun-cloud":
return '<svg viewBox="0 0 24 24" focusable="false"><path d="M8.2 6.1a3.1 3.1 0 0 1 5.5-.7"></path><path d="M7.1 3.6v1.7M3.9 6.8H5.6M15.4 6.8h1.7M5.2 4.9l1.2 1.2"></path><path d="M7.7 17.5h8.1a3.2 3.2 0 0 0 .2-6.3 4.7 4.7 0 0 0-9.1 1 2.7 2.7 0 0 0 .8 5.3Z"></path></svg>';
case "cloud":
return '<svg viewBox="0 0 24 24" focusable="false"><path d="M7.4 17.5h8.4a3.3 3.3 0 0 0 .2-6.5 4.8 4.8 0 0 0-9.3 1 2.8 2.8 0 0 0 .7 5.5Z"></path></svg>';
case "rain":
return '<svg viewBox="0 0 24 24" focusable="false"><path d="M7.4 14.9h8.4a3.3 3.3 0 0 0 .2-6.5 4.8 4.8 0 0 0-9.3 1 2.8 2.8 0 0 0 .7 5.5Z"></path><path d="M9.2 17.7 8.4 20M12.2 17.7l-.8 2.3M15.2 17.7l-.8 2.3"></path></svg>';
case "snow":
return '<svg viewBox="0 0 24 24" focusable="false"><path d="M7.4 14.6h8.4A3.3 3.3 0 0 0 16 8.1a4.8 4.8 0 0 0-9.3 1 2.8 2.8 0 0 0 .7 5.5Z"></path><path d="m9.5 17.5 1.1 1.9m0-1.9-1.1 1.9m1.1-3v4.1"></path><path d="m14 17.5 1.1 1.9m0-1.9-1.1 1.9m1.1-3v4.1"></path></svg>';
case "storm":
return '<svg viewBox="0 0 24 24" focusable="false"><path d="M7.4 14.4h8.4A3.3 3.3 0 0 0 16 8a4.8 4.8 0 0 0-9.3 1 2.8 2.8 0 0 0 .7 5.4Z"></path><path d="m11.8 15.8-1.7 3.1h2l-.9 2.3 2.9-4h-2.1l1.1-1.4"></path></svg>';
default:
return '<svg viewBox="0 0 24 24" focusable="false"><path d="M8 15.8a4.2 4.2 0 1 0 0-7.6"></path><path d="M16 8.2a4.2 4.2 0 1 0 0 7.6"></path><path d="M7.6 12h8.8"></path></svg>';
}
}
function weatherSummary(code) {
var numericCode = parseInt(code, 10);
if ([0].indexOf(numericCode) !== -1) {
return {iconKey: "sun", label: "Ensoleillé"};
}
if ([1, 2].indexOf(numericCode) !== -1) {
return {iconKey: "sun-cloud", label: "Éclaircies"};
}
if ([3, 45, 48].indexOf(numericCode) !== -1) {
return {iconKey: "cloud", label: "Nuageux"};
}
if ([51, 53, 55, 56, 57, 61, 63, 65, 66, 67, 80, 81, 82].indexOf(numericCode) !== -1) {
return {iconKey: "rain", label: "Pluie"};
}
if ([71, 73, 75, 77, 85, 86].indexOf(numericCode) !== -1) {
return {iconKey: "snow", label: "Neige"};
}
if ([95, 96, 99].indexOf(numericCode) !== -1) {
return {iconKey: "storm", label: "Orage"};
}
return {iconKey: "variable", label: "Variable"};
}
function formatTemperature(value) {
if (value === null || typeof value === "undefined" || isNaN(parseFloat(value))) {
return "-";
}
return Math.round(parseFloat(value)) + "\u00b0";
}
function formatDay(value, index) {
if (index === 0) {
return "Auj.";
}
try {
return new Intl.DateTimeFormat("fr-FR", {weekday: "short"}).format(new Date(value + "T12:00:00"));
} catch (error) {
return value.slice(5);
}
}
function formatUpdated(value) {
if (!value) {
return "";
}
try {
return new Intl.DateTimeFormat("fr-FR", {
hour: "2-digit",
minute: "2-digit"
}).format(new Date(value));
} catch (error) {
return value.slice(11, 16);
}
}
function formatLocalTime(timezone) {
try {
return new Intl.DateTimeFormat("fr-FR", {
hour: "2-digit",
minute: "2-digit",
timeZone: timezone
}).format(new Date());
} catch (error) {
return "";
}
}
function padClockPart(value) {
return String(value).padStart(2, "0");
}
function getLocalTimeParts(timezone) {
var parts;
var values = {};
var hour;
var minute;
try {
parts = new Intl.DateTimeFormat("fr-FR", {
hour: "2-digit",
minute: "2-digit",
hour12: false,
timeZone: timezone
}).formatToParts(new Date());
parts.forEach(function (part) {
values[part.type] = part.value;
});
hour = parseInt(values.hour || "0", 10) % 24;
minute = parseInt(values.minute || "0", 10);
return {
label: padClockPart(hour) + ":" + padClockPart(minute),
hour: hour,
minute: minute
};
} catch (error) {
return null;
}
}
function getTimezoneOffsetMinutes(timezone, date) {
var parts;
var values = {};
var utcDate;
try {
parts = new Intl.DateTimeFormat("en-GB", {
year: "numeric",
month: "2-digit",
day: "2-digit",
hour: "2-digit",
minute: "2-digit",
second: "2-digit",
hour12: false,
timeZone: timezone
}).formatToParts(date);
parts.forEach(function (part) {
values[part.type] = part.value;
});
utcDate = Date.UTC(
parseInt(values.year, 10),
parseInt(values.month, 10) - 1,
parseInt(values.day, 10),
parseInt(values.hour, 10) % 24,
parseInt(values.minute, 10),
parseInt(values.second, 10)
);
return Math.round((utcDate - date.getTime()) / 60000);
} catch (error) {
return null;
}
}
function formatTimezoneLabel(timezone) {
var offset = getTimezoneOffsetMinutes(timezone, new Date());
var zoneName;
if (timezone == "Australia/Sydney") {
return offset >= 660 ? "AEDT" : "AEST";
}
if (timezone == "Europe/Zurich") {
return offset >= 120 ? "CEST" : "CET";
}
try {
zoneName = new Intl.DateTimeFormat("en-US", {
timeZone: timezone,
timeZoneName: "short"
}).formatToParts(new Date()).filter(function (part) {
return part.type == "timeZoneName";
}).map(function (part) {
return part.value;
}).join("");
return zoneName || timezone;
} catch (error) {
return timezone;
}
}
function updateClock(clock) {
var time = getLocalTimeParts(clock.timezone);
var hourDegrees;
var minuteDegrees;
if (!time || !time.label) {
clock.element.classList.add("is-unavailable");
return false;
}
hourDegrees = ((time.hour % 12) + (time.minute / 60)) * 30;
minuteDegrees = time.minute * 6;
clock.element.classList.remove("is-unavailable");
clock.timeElement.textContent = time.label;
clock.hourHand.style.transform = "translateX(-50%) rotate(" + hourDegrees + "deg)";
clock.minuteHand.style.transform = "translateX(-50%) rotate(" + minuteDegrees + "deg)";
if (clock.labelElement) {
clock.labelElement.textContent = clock.label;
}
if (clock.zoneElement) {
clock.zoneElement.textContent = formatTimezoneLabel(clock.timezone);
}
clock.element.setAttribute("aria-label", clock.label + " : " + time.label + " " + formatTimezoneLabel(clock.timezone));
return true;
}
function refreshClocks() {
clocks = clocks.filter(function (clock) {
if (!clock.element || !clock.element.parentNode) {
return false;
}
return updateClock(clock);
});
if (!clocks.length && clockTimer) {
window.clearInterval(clockTimer);
clockTimer = null;
}
}
function registerAnalogClock(element, timezone, label) {
var clock;
if (!element || !timezone) {
return;
}
clock = {
element: element,
timezone: timezone,
label: label || element.getAttribute("data-mas-clock-label") || "Heure locale",
timeElement: element.querySelector("[data-mas-clock-time]"),
labelElement: element.querySelector("[data-mas-clock-label]"),
zoneElement: element.querySelector("[data-mas-clock-zone]"),
hourHand: element.querySelector("[data-mas-clock-hour]"),
minuteHand: element.querySelector("[data-mas-clock-minute]")
};
if (!clock.timeElement || !clock.hourHand || !clock.minuteHand) {
return;
}
clocks = clocks.filter(function (registeredClock) {
return registeredClock.element !== element;
});
clocks.push(clock);
updateClock(clock);
if (!clockTimer) {
clockTimer = window.setInterval(refreshClocks, 60000);
}
}
function getPreferredClockCity(section) {
var cityNames = Array.prototype.slice.call(section.querySelectorAll(".mas-destination-card")).map(getCityName).filter(Boolean);
var preferred = cityNames.filter(function (cityName) {
return normalize(cityName) == "sydney";
});
return preferred[0] || cityNames[0] || "";
}
function getPreferredClockMarker(markers, preferredCity) {
var preferredKey = normalize(preferredCity);
if (!markers || !markers.list || !markers.list.length) {
return null;
}
if (preferredKey && markers.byName[preferredKey]) {
return markers.byName[preferredKey];
}
return markers.list.filter(function (marker) {
return normalize(marker.title) == "sydney";
})[0] || markers.list[0] || null;
}
function createCountryClockController(section) {
var root = document.querySelector("[data-mas-country-clocks]");
var destinationClock = root ? root.querySelector("[data-mas-country-clock='destination']") : null;
var referenceClock = root ? root.querySelector("[data-mas-country-clock='reference']") : null;
var preferredCity = getPreferredClockCity(section);
var destinationReady = false;
if (!root) {
return null;
}
if (referenceClock) {
registerAnalogClock(
referenceClock,
referenceClock.getAttribute("data-mas-clock-timezone"),
referenceClock.getAttribute("data-mas-clock-label") || "Genève"
);
}
return {
registerDestination: function (cityName, timezone) {
if (destinationReady || !destinationClock || !timezone || (preferredCity && normalize(cityName) != normalize(preferredCity))) {
return;
}
destinationReady = true;
registerAnalogClock(destinationClock, timezone, cityName);
}
};
}
function createCityClockController(section) {
var root = section ? section.querySelector("[data-mas-city-clocks]") : null;
var destinationClock = root ? root.querySelector("[data-mas-city-clock='destination']") : null;
var referenceClock = root ? root.querySelector("[data-mas-city-clock='reference']") : null;
var destinationReady = false;
if (!root) {
return null;
}
if (referenceClock) {
registerAnalogClock(
referenceClock,
referenceClock.getAttribute("data-mas-clock-timezone"),
referenceClock.getAttribute("data-mas-clock-label") || "Geneve"
);
}
return {
registerDestination: function (cityName, timezone) {
if (destinationReady || !destinationClock || !timezone) {
return;
}
destinationReady = true;
registerAnalogClock(destinationClock, timezone, cityName || "Destination");
}
};
}
function setFallback(panel, cityName) {
panel.classList.add("is-unavailable");
panel.classList.remove("is-ready");
panel.innerHTML = [
'<p class="mas-city-weather__eyebrow">Météo locale</p>',
'<p class="mas-city-weather__current"><span class="mas-city-weather__icon" aria-hidden="true">' + weatherIconSvg("variable") + '</span><span class="mas-city-weather__summary">Prévisions indisponibles</span></p>',
'<p class="mas-city-weather__fallback">Consultez la destination ' + escapeHtml(cityName || "sélectionnée") + ' pour préparer votre séjour.</p>'
].join("");
}
function escapeHtml(value) {
return String(value || "")
.replace(/&/g, "&amp;")
.replace(/</g, "&lt;")
.replace(/>/g, "&gt;")
.replace(/"/g, "&quot;");
}
function renderWeather(panel, cityName, data) {
var current = data.current || {};
var daily = data.daily || {};
var summary = weatherSummary(current.weather_code);
var dates = (daily.time || []).slice(0, 4);
var highs = daily.temperature_2m_max || [];
var lows = daily.temperature_2m_min || [];
var codes = daily.weather_code || [];
var forecast = dates.map(function (date, index) {
var daySummary = weatherSummary(codes[index]);
return [
'<div class="mas-city-weather__day">',
'<div class="mas-city-weather__day-meta">',
'<span class="mas-city-weather__day-label">' + escapeHtml(formatDay(date, index)) + '</span>',
'<span class="mas-city-weather__day-icon" aria-hidden="true">' + weatherIconSvg(daySummary.iconKey) + '</span>',
'</div>',
'<strong class="mas-city-weather__day-temp">' + escapeHtml(formatTemperature(lows[index])) + '/' + escapeHtml(formatTemperature(highs[index])) + '</strong>',
'</div>'
].join("");
}).join("");
var updated = formatUpdated(current.time);
panel.classList.add("is-ready");
panel.classList.remove("is-unavailable");
panel.innerHTML = [
'<p class="mas-city-weather__eyebrow">Météo à ' + escapeHtml(cityName) + '</p>',
'<p class="mas-city-weather__current">',
'<span class="mas-city-weather__icon" aria-hidden="true">' + weatherIconSvg(summary.iconKey) + '</span>',
'<span class="mas-city-weather__summary">' + escapeHtml(formatTemperature(current.temperature_2m)) + ' aujourd’hui, ' + escapeHtml(summary.label.toLowerCase()) + '</span>',
'</p>',
'<div class="mas-city-weather__forecast">' + forecast + '</div>',
updated ? '<p class="mas-city-weather__updated">Mise à jour à ' + escapeHtml(updated) + '</p>' : ''
].join("");
}
function getWeatherCacheKey(marker) {
var latitude = Math.round(parseFloat(marker.latitude) * 1000) / 1000;
var longitude = Math.round(parseFloat(marker.longitude) * 1000) / 1000;
return "mas.weather." + latitude + "." + longitude + "." + new Date().toISOString().slice(0, 13);
}
function getCachedJson(cacheKey) {
var stored;
var parsed;
try {
stored = window.sessionStorage && window.sessionStorage.getItem(cacheKey);
if (!stored) {
return null;
}
parsed = JSON.parse(stored);
if (!parsed || !parsed.createdAt || (Date.now() - parsed.createdAt) > cacheLifetime) {
window.sessionStorage.removeItem(cacheKey);
return null;
}
return parsed.payload || null;
} catch (error) {
return null;
}
}
function setCachedJson(cacheKey, payload) {
try {
if (window.sessionStorage) {
window.sessionStorage.setItem(cacheKey, JSON.stringify({
createdAt: Date.now(),
payload: payload
}));
}
} catch (error) {
return;
}
}
function fetchWeather(url, cacheKey) {
var cached = getCachedJson(cacheKey);
if (cached) {
return Promise.resolve(cached);
}
return window.fetch(url)
.then(function (response) {
if (!response.ok) {
throw new Error("Weather unavailable");
}
return response.json();
})
.then(function (data) {
setCachedJson(cacheKey, data);
return data;
});
}
function hydrateCard(card, markers, countryClockController) {
var panel = card.querySelector("[data-mas-city-weather]");
var cityName = getCityName(card);
var marker = findMarkerForCity(markers, cityName);
if (!panel || !cityName || !marker || !window.fetch || !isFiniteNumber(marker.latitude) || !isFiniteNumber(marker.longitude)) {
if (panel) {
setFallback(panel, cityName);
}
return;
}
var url = forecastUrl
+ "?latitude=" + encodeURIComponent(marker.latitude)
+ "&longitude=" + encodeURIComponent(marker.longitude)
+ "&current=temperature_2m,weather_code"
+ "&daily=weather_code,temperature_2m_max,temperature_2m_min"
+ "&forecast_days=4"
+ "&timezone=auto";
fetchWeather(url, getWeatherCacheKey(marker))
.then(function (data) {
if (countryClockController) {
countryClockController.registerDestination(cityName, data.timezone);
}
renderWeather(panel, cityName, data);
})
.catch(function () {
setFallback(panel, cityName);
});
}
function hydrateCityOverview(section) {
var panel = section ? section.querySelector("[data-mas-city-weather]") : null;
var marker = getCityOverviewMarker();
var cityClockController = createCityClockController(section);
var cityName = marker ? marker.title : "";
var url;
if (!panel || !marker || !window.fetch || !isFiniteNumber(marker.latitude) || !isFiniteNumber(marker.longitude)) {
if (panel) {
setFallback(panel, cityName);
}
return;
}
url = forecastUrl
+ "?latitude=" + encodeURIComponent(marker.latitude)
+ "&longitude=" + encodeURIComponent(marker.longitude)
+ "&current=temperature_2m,weather_code"
+ "&daily=weather_code,temperature_2m_max,temperature_2m_min"
+ "&forecast_days=4"
+ "&timezone=auto";
fetchWeather(url, getWeatherCacheKey(marker))
.then(function (data) {
if (cityClockController) {
cityClockController.registerDestination(cityName, data.timezone);
}
renderWeather(panel, cityName, data);
})
.catch(function () {
setFallback(panel, cityName);
});
}
onReady(function () {
var section = document.querySelector(".MoveAndStudyCore-country .mas-country-destinations");
var cards;
var markers;
var countryClockController;
var countryClockRoot;
var destinationClock;
var preferredMarker;
var url;
if (!section) {
return;
}
cards = Array.prototype.slice.call(section.querySelectorAll(".mas-destination-card"));
markers = getMarkers(section);
countryClockController = createCountryClockController(section);
countryClockRoot = document.querySelector("[data-mas-country-clocks]");
destinationClock = countryClockRoot ? countryClockRoot.querySelector("[data-mas-country-clock='destination']") : null;
cards.forEach(function (card) {
hydrateCard(card, markers, countryClockController);
});
if (cards.length || !countryClockController) {
return;
}
preferredMarker = getPreferredClockMarker(markers, getPreferredClockCity(section));
if (!preferredMarker || !window.fetch || !isFiniteNumber(preferredMarker.latitude) || !isFiniteNumber(preferredMarker.longitude)) {
return;
}
url = forecastUrl
+ "?latitude=" + encodeURIComponent(preferredMarker.latitude)
+ "&longitude=" + encodeURIComponent(preferredMarker.longitude)
+ "&current=temperature_2m,weather_code"
+ "&daily=weather_code,temperature_2m_max,temperature_2m_min"
+ "&forecast_days=4"
+ "&timezone=auto";
fetchWeather(url, getWeatherCacheKey(preferredMarker))
.then(function (data) {
if (!destinationClock || !data || !data.timezone) {
return;
}
registerAnalogClock(destinationClock, data.timezone, preferredMarker.title || "Destination");
})
.catch(function () {
return;
});
});
}());
(function () {
var LAYOUTS = {
COMPACT: "editorial-compact",
LANDSCAPE: "editorial-landscape",
MIXED: "editorial-mixed",
PANORAMA: "editorial-panorama"
};
function onReady(callback) {
if (document.readyState === "loading") {
document.addEventListener("DOMContentLoaded", callback);
return;
}
callback();
}
function qs(root, selector) {
return root ? root.querySelector(selector) : null;
}
function qsa(root, selector) {
return root ? Array.prototype.slice.call(root.querySelectorAll(selector)) : [];
}
function getBreakpoint() {
if (window.matchMedia("(min-width: 960px)").matches) {
return "desktop";
}
if (window.matchMedia("(min-width: 760px)").matches) {
return "tablet";
}
return "mobile";
}
function getPreviewLimit(layout, count) {
var breakpoint = getBreakpoint();
if (breakpoint === "mobile") {
return Math.min(count, 3);
}
if (breakpoint === "tablet") {
if (layout === LAYOUTS.COMPACT) {
return Math.min(count, 4);
}
if (layout === LAYOUTS.PANORAMA) {
return Math.min(count, 4);
}
return Math.min(count, 5);
}
if (layout === LAYOUTS.COMPACT) {
return Math.min(count, 4);
}
if (layout === LAYOUTS.MIXED) {
return Math.min(count, 6);
}
if (layout === LAYOUTS.PANORAMA) {
return Math.min(count, 5);
}
return Math.min(count, 7);
}
function getLegacyPreviewLayout(layout) {
if (layout === LAYOUTS.COMPACT) {
return "compact";
}
if (layout === LAYOUTS.MIXED) {
return "balanced";
}
return "editorial";
}
function classifyRatio(ratio) {
if (!isFinite(ratio) || ratio <= 0) {
return "landscape";
}
if (ratio < 0.9) {
return "portrait";
}
if (ratio <= 1.2) {
return "square";
}
if (ratio <= 1.9) {
return "landscape";
}
if (ratio <= 2.8) {
return "wide";
}
return "panorama";
}
function buildViewerUrl(url) {
try {
var absolute = new URL(url, window.location.href);
if (absolute.pathname.indexOf("image.php") === -1) {
return absolute.toString();
}
absolute.searchParams.set("destinationWidth", "1800");
absolute.searchParams.set("destinationHeight", "1200");
absolute.searchParams.set("keepRatio", "1");
return absolute.pathname + "?" + absolute.searchParams.toString();
} catch (error) {
return url;
}
}
function normalizeText(text) {
return (text || "").replace(/\s+/g, " ").trim();
}
function waitForImageMetrics(image, href) {
return new Promise(function (resolve) {
var source = image ? (image.currentSrc || image.getAttribute("src") || href) : href;
if (image && image.complete && image.naturalWidth > 0 && image.naturalHeight > 0) {
resolve({
naturalWidth: image.naturalWidth,
naturalHeight: image.naturalHeight
});
return;
}
if (!source) {
resolve(null);
return;
}
var probe = new Image();
var settled = false;
function finish(result) {
if (settled) {
return;
}
settled = true;
resolve(result);
}
probe.onload = function () {
finish({
naturalWidth: probe.naturalWidth,
naturalHeight: probe.naturalHeight
});
};
probe.onerror = function () {
finish(null);
};
window.setTimeout(function () {
finish(null);
}, 3000);
probe.src = source;
});
}
async function createGalleryModel(links) {
var models = await Promise.all(links.map(async function (link, index) {
var image = qs(link, ".mas-city-gallery-component__image");
var title = normalizeText(link.getAttribute("title") || (image ? image.getAttribute("alt") : ""));
var href = link.getAttribute("href") || "";
var metrics = await waitForImageMetrics(image, href);
var ratio = metrics ? metrics.naturalWidth / metrics.naturalHeight : 0;
return {
index: index,
link: link,
image: image,
href: href,
viewerHref: buildViewerUrl(href),
alt: image ? (image.getAttribute("alt") || "") : "",
title: title,
naturalWidth: metrics ? metrics.naturalWidth : 0,
naturalHeight: metrics ? metrics.naturalHeight : 0,
ratio: ratio,
ratioType: classifyRatio(ratio),
isValid: !!metrics
};
}));
return models;
}
function chooseLayout(items) {
var validItems = items.filter(function (item) {
return item.isValid;
});
var counts = {
portrait: 0,
square: 0,
landscape: 0,
wide: 0,
panorama: 0
};
var firstType;
validItems.forEach(function (item) {
counts[item.ratioType] += 1;
});
if (validItems.length <= 5) {
return LAYOUTS.COMPACT;
}
firstType = validItems[0] ? validItems[0].ratioType : "landscape";
if (firstType === "panorama" || (counts.panorama >= 1 && (counts.wide + counts.panorama) >= 2 && firstType !== "portrait")) {
return LAYOUTS.PANORAMA;
}
if (firstType === "portrait" || counts.portrait >= 1 || counts.square >= 2) {
return LAYOUTS.MIXED;
}
return LAYOUTS.LANDSCAPE;
}
function getVisiblePreviewIndexes(items, previewLimit) {
var visible = [];
items.forEach(function (item) {
if (!item.isValid || visible.length >= previewLimit) {
return;
}
visible.push(item.index);
});
return visible;
}
function setPreviewState(frame, items, actions) {
var validItems = items.filter(function (item) {
return item.isValid;
});
var layout = chooseLayout(items);
var previewLimit = getPreviewLimit(layout, validItems.length);
var visibleIndexes = getVisiblePreviewIndexes(items, previewLimit);
frame.setAttribute("data-gallery-state", "preview");
frame.setAttribute("data-city-gallery-layout", layout);
frame.setAttribute("data-image-count", String(items.length));
frame.setAttribute("data-city-gallery-ready", "true");
frame.setAttribute("data-preview-layout", getLegacyPreviewLayout(layout));
items.forEach(function (item) {
item.link.classList.toggle("is-preview-hidden", visibleIndexes.indexOf(item.index) === -1);
});
if (actions) {
actions.hidden = validItems.length <= previewLimit;
}
}
function bindGallery(frame) {
var links = qsa(frame, ".mas-city-gallery-component__link");
var actions = qs(frame, "[data-mas-city-gallery-actions]");
var openButton = qs(frame, "[data-mas-city-gallery-open]");
var viewer = qs(frame, "[data-mas-city-gallery-viewer]");
var viewerImage = qs(frame, "[data-mas-city-gallery-image]");
var viewerCaption = qs(frame, "[data-mas-city-gallery-caption]");
var closeButtons = qsa(frame, "[data-mas-city-gallery-close]");
var prevButton = qs(frame, "[data-mas-city-gallery-prev]");
var nextButton = qs(frame, "[data-mas-city-gallery-next]");
var items = [];
var viewerItems = [];
var currentIndex = 0;
if (!links.length || !viewer || !viewerImage) {
return;
}
frame.setAttribute("data-city-gallery-ready", "false");
frame.setAttribute("data-image-count", String(links.length));
function render(index) {
var item = viewerItems[index];
if (!item) {
return;
}
currentIndex = index;
viewerImage.src = item.viewerHref || item.href;
viewerImage.alt = item.alt || "";
if (viewerCaption) {
if (item.title) {
viewerCaption.textContent = item.title;
viewerCaption.hidden = false;
} else {
viewerCaption.textContent = "";
viewerCaption.hidden = true;
}
}
}
function open(index) {
render(index);
viewer.hidden = false;
document.body.classList.add("is-city-gallery-viewer-open");
}
function close() {
viewer.hidden = true;
document.body.classList.remove("is-city-gallery-viewer-open");
}
function prev() {
render((currentIndex - 1 + viewerItems.length) % viewerItems.length);
}
function next() {
render((currentIndex + 1) % viewerItems.length);
}
function updatePreview() {
setPreviewState(frame, items, actions);
}
createGalleryModel(links).then(function (models) {
items = models;
viewerItems = items.filter(function (item) {
return item.isValid;
});
if (!viewerItems.length) {
frame.setAttribute("data-city-gallery-ready", "true");
if (actions) {
actions.hidden = true;
}
return;
}
items.forEach(function (item) {
item.link.addEventListener("click", function (event) {
var viewerIndex;
event.preventDefault();
viewerIndex = viewerItems.findIndex(function (viewerItem) {
return viewerItem.index === item.index;
});
if (viewerIndex === -1) {
return;
}
open(viewerIndex);
});
});
if (openButton) {
openButton.addEventListener("click", function () {
open(0);
});
}
closeButtons.forEach(function (button) {
button.addEventListener("click", close);
});
if (prevButton) {
prevButton.addEventListener("click", prev);
}
if (nextButton) {
nextButton.addEventListener("click", next);
}
updatePreview();
window.addEventListener("resize", updatePreview);
});
document.addEventListener("keydown", function (event) {
if (viewer.hidden || !viewerItems.length) {
return;
}
if (event.key === "Escape") {
close();
} else if (event.key === "ArrowLeft") {
prev();
} else if (event.key === "ArrowRight") {
next();
}
});
}
onReady(function () {
qsa(document, "[data-mas-city-gallery]").forEach(bindGallery);
});
})();
(function () {
var messageTypeInit = "CityInteractiveMap:init";
var messageTypeReady = "CityInteractiveMap:ready";
var messageTypeUnavailable = "CityInteractiveMap:unavailable";
var iframeSource = "/synergee/Themes/MoveAndStudyCore/Components/InteractiveMap/Iframe/CityInteractiveMap.html";
function onReady(callback) {
if (document.readyState === "loading") {
document.addEventListener("DOMContentLoaded", callback);
} else {
callback();
}
}
function addClass(element, className) {
if (element && element.className.indexOf(className) === -1) {
element.className += " " + className;
}
}
function hideMap(component) {
if (!component) {
return;
}
addClass(component, "is-unavailable");
}
function getMarkers(component) {
var markerLinks = component.querySelectorAll(".mas-city-map__markers a");
var markers = [];
Array.prototype.forEach.call(markerLinks, function (markerLink) {
var latitude = parseFloat(markerLink.getAttribute("data-lat"));
var longitude = parseFloat(markerLink.getAttribute("data-lng"));
var zoom = parseFloat(markerLink.getAttribute("data-zoom")) || 14;
var url = markerLink.getAttribute("href");
if (!isFinite(latitude) || !isFinite(longitude) || latitude === 0 || longitude === 0 || !url) {
return;
}
markers.push({
title: markerLink.getAttribute("data-title") || markerLink.textContent || "",
city: markerLink.getAttribute("data-city") || "",
url: url,
zoom: zoom,
position: {
lat: latitude,
lng: longitude
}
});
});
return markers;
}
function initMap(component) {
var key = component.getAttribute("data-map-key");
var frame = component.querySelector(".mas-city-map__iframe");
var markers = getMarkers(component);
var timeout = null;
var settled = false;
if (!key || !frame || !markers.length) {
hideMap(component);
return;
}
function settleAsReady() {
if (settled) {
return;
}
settled = true;
clearTimeout(timeout);
addClass(component, "is-ready");
}
function settleAsUnavailable() {
if (settled) {
return;
}
settled = true;
clearTimeout(timeout);
hideMap(component);
}
function postInitMessage() {
if (!frame.contentWindow) {
settleAsUnavailable();
return;
}
frame.contentWindow.postMessage({
type: messageTypeInit,
key: key,
markers: markers
}, window.location.origin);
}
function onMessage(event) {
if (event.origin !== window.location.origin || event.source !== frame.contentWindow || !event.data) {
return;
}
if (event.data.type === messageTypeReady) {
window.removeEventListener("message", onMessage);
settleAsReady();
}
if (event.data.type === messageTypeUnavailable) {
window.removeEventListener("message", onMessage);
settleAsUnavailable();
}
}
window.addEventListener("message", onMessage);
frame.addEventListener("load", postInitMessage);
frame.setAttribute("src", iframeSource);
timeout = setTimeout(function () {
window.removeEventListener("message", onMessage);
settleAsUnavailable();
}, 12000);
}
onReady(function () {
var maps = document.querySelectorAll(".MoveAndStudyCore.MoveAndStudyCore-city [data-mas-city-interactive-map]");
Array.prototype.forEach.call(maps, function (component) {
if (component.getAttribute("data-map-bound") === "true") {
return;
}
component.setAttribute("data-map-bound", "true");
initMap(component);
});
});
}());
(function () {
var forecastUrl = "https://api.open-meteo.com/v1/forecast";
var cacheLifetime = 60 * 60 * 1000;
var clockTimer = null;
var clocks = [];
function onReady(callback) {
if (document.readyState === "loading") {
document.addEventListener("DOMContentLoaded", callback);
return;
}
callback();
}
function escapeHtml(value) {
return String(value || "")
.replace(/&/g, "&amp;")
.replace(/</g, "&lt;")
.replace(/>/g, "&gt;")
.replace(/"/g, "&quot;");
}
function isFiniteNumber(value) {
return value !== null && value !== "" && isFinite(parseFloat(value));
}
function weatherIconSvg(iconKey) {
switch (iconKey) {
case "sun":
return '<svg viewBox="0 0 24 24" focusable="false"><circle cx="12" cy="12" r="3.2"></circle><path d="M12 2.8v2.3M12 18.9v2.3M21.2 12h-2.3M5.1 12H2.8M18.5 5.5l-1.6 1.6M7.1 16.9l-1.6 1.6M18.5 18.5l-1.6-1.6M7.1 7.1 5.5 5.5"></path></svg>';
case "sun-cloud":
return '<svg viewBox="0 0 24 24" focusable="false"><path d="M8.2 6.1a3.1 3.1 0 0 1 5.5-.7"></path><path d="M7.1 3.6v1.7M3.9 6.8H5.6M15.4 6.8h1.7M5.2 4.9l1.2 1.2"></path><path d="M7.7 17.5h8.1a3.2 3.2 0 0 0 .2-6.3 4.7 4.7 0 0 0-9.1 1 2.7 2.7 0 0 0 .8 5.3Z"></path></svg>';
case "cloud":
return '<svg viewBox="0 0 24 24" focusable="false"><path d="M7.4 17.5h8.4a3.3 3.3 0 0 0 .2-6.5 4.8 4.8 0 0 0-9.3 1 2.8 2.8 0 0 0 .7 5.5Z"></path></svg>';
case "rain":
return '<svg viewBox="0 0 24 24" focusable="false"><path d="M7.4 14.9h8.4a3.3 3.3 0 0 0 .2-6.5 4.8 4.8 0 0 0-9.3 1 2.8 2.8 0 0 0 .7 5.5Z"></path><path d="M9.2 17.7 8.4 20M12.2 17.7l-.8 2.3M15.2 17.7l-.8 2.3"></path></svg>';
case "snow":
return '<svg viewBox="0 0 24 24" focusable="false"><path d="M7.4 14.6h8.4A3.3 3.3 0 0 0 16 8.1a4.8 4.8 0 0 0-9.3 1 2.8 2.8 0 0 0 .7 5.5Z"></path><path d="m9.5 17.5 1.1 1.9m0-1.9-1.1 1.9m1.1-3v4.1"></path><path d="m14 17.5 1.1 1.9m0-1.9-1.1 1.9m1.1-3v4.1"></path></svg>';
case "storm":
return '<svg viewBox="0 0 24 24" focusable="false"><path d="M7.4 14.4h8.4A3.3 3.3 0 0 0 16 8a4.8 4.8 0 0 0-9.3 1 2.8 2.8 0 0 0 .7 5.4Z"></path><path d="m11.8 15.8-1.7 3.1h2l-.9 2.3 2.9-4h-2.1l1.1-1.4"></path></svg>';
default:
return '<svg viewBox="0 0 24 24" focusable="false"><path d="M8 15.8a4.2 4.2 0 1 0 0-7.6"></path><path d="M16 8.2a4.2 4.2 0 1 0 0 7.6"></path><path d="M7.6 12h8.8"></path></svg>';
}
}
function weatherSummary(code) {
var numericCode = parseInt(code, 10);
if ([0].indexOf(numericCode) !== -1) {
return {iconKey: "sun", label: "Ensoleillé"};
}
if ([1, 2].indexOf(numericCode) !== -1) {
return {iconKey: "sun-cloud", label: "Éclaircies"};
}
if ([3, 45, 48].indexOf(numericCode) !== -1) {
return {iconKey: "cloud", label: "Nuageux"};
}
if ([51, 53, 55, 56, 57, 61, 63, 65, 66, 67, 80, 81, 82].indexOf(numericCode) !== -1) {
return {iconKey: "rain", label: "Pluie"};
}
if ([71, 73, 75, 77, 85, 86].indexOf(numericCode) !== -1) {
return {iconKey: "snow", label: "Neige"};
}
if ([95, 96, 99].indexOf(numericCode) !== -1) {
return {iconKey: "storm", label: "Orage"};
}
return {iconKey: "variable", label: "Variable"};
}
function formatTemperature(value) {
if (value === null || typeof value === "undefined" || isNaN(parseFloat(value))) {
return "-";
}
return Math.round(parseFloat(value)) + "\u00b0";
}
function formatDay(value, index) {
if (index === 0) {
return "Auj.";
}
try {
return new Intl.DateTimeFormat("fr-FR", {weekday: "short"}).format(new Date(value + "T12:00:00"));
} catch (error) {
return value.slice(5);
}
}
function formatUpdated(value) {
if (!value) {
return "";
}
try {
return new Intl.DateTimeFormat("fr-FR", {
hour: "2-digit",
minute: "2-digit"
}).format(new Date(value));
} catch (error) {
return value.slice(11, 16);
}
}
function getLocalTimeParts(timezone) {
var parts;
var values = {};
var hour;
var minute;
try {
parts = new Intl.DateTimeFormat("fr-FR", {
hour: "2-digit",
minute: "2-digit",
hour12: false,
timeZone: timezone
}).formatToParts(new Date());
parts.forEach(function (part) {
values[part.type] = part.value;
});
hour = parseInt(values.hour || "0", 10) % 24;
minute = parseInt(values.minute || "0", 10);
return {
label: String(hour).padStart(2, "0") + ":" + String(minute).padStart(2, "0"),
hour: hour,
minute: minute
};
} catch (error) {
return null;
}
}
function getTimezoneOffsetMinutes(timezone, date) {
var parts;
var values = {};
var utcDate;
try {
parts = new Intl.DateTimeFormat("en-GB", {
year: "numeric",
month: "2-digit",
day: "2-digit",
hour: "2-digit",
minute: "2-digit",
second: "2-digit",
hour12: false,
timeZone: timezone
}).formatToParts(date);
parts.forEach(function (part) {
values[part.type] = part.value;
});
utcDate = Date.UTC(
parseInt(values.year, 10),
parseInt(values.month, 10) - 1,
parseInt(values.day, 10),
parseInt(values.hour, 10) % 24,
parseInt(values.minute, 10),
parseInt(values.second, 10)
);
return Math.round((utcDate - date.getTime()) / 60000);
} catch (error) {
return null;
}
}
function formatTimezoneLabel(timezone) {
var offset = getTimezoneOffsetMinutes(timezone, new Date());
var zoneName;
if (timezone == "Australia/Sydney") {
return offset >= 660 ? "AEDT" : "AEST";
}
if (timezone == "Europe/Zurich") {
return offset >= 120 ? "CEST" : "CET";
}
try {
zoneName = new Intl.DateTimeFormat("en-US", {
timeZone: timezone,
timeZoneName: "short"
}).formatToParts(new Date()).filter(function (part) {
return part.type == "timeZoneName";
}).map(function (part) {
return part.value;
}).join("");
return zoneName || timezone;
} catch (error) {
return timezone;
}
}
function updateClock(clock) {
var time = getLocalTimeParts(clock.timezone);
var hourDegrees;
var minuteDegrees;
if (!time || !time.label) {
clock.element.classList.add("is-unavailable");
return false;
}
hourDegrees = ((time.hour % 12) + (time.minute / 60)) * 30;
minuteDegrees = time.minute * 6;
clock.element.classList.remove("is-unavailable");
clock.timeElement.textContent = time.label;
clock.hourHand.style.transform = "translateX(-50%) rotate(" + hourDegrees + "deg)";
clock.minuteHand.style.transform = "translateX(-50%) rotate(" + minuteDegrees + "deg)";
if (clock.labelElement) {
clock.labelElement.textContent = clock.label;
}
if (clock.zoneElement) {
clock.zoneElement.textContent = formatTimezoneLabel(clock.timezone);
}
clock.element.setAttribute("aria-label", clock.label + " : " + time.label + " " + formatTimezoneLabel(clock.timezone));
return true;
}
function refreshClocks() {
clocks = clocks.filter(function (clock) {
if (!clock.element || !clock.element.parentNode) {
return false;
}
return updateClock(clock);
});
if (!clocks.length && clockTimer) {
window.clearInterval(clockTimer);
clockTimer = null;
}
}
function registerAnalogClock(element, timezone, label) {
var clock;
if (!element || !timezone) {
return;
}
clock = {
element: element,
timezone: timezone,
label: label || element.getAttribute("data-mas-clock-label") || "Heure locale",
timeElement: element.querySelector("[data-mas-clock-time]"),
labelElement: element.querySelector("p[data-mas-clock-label]"),
zoneElement: element.querySelector("[data-mas-clock-zone]"),
hourHand: element.querySelector("[data-mas-clock-hour]"),
minuteHand: element.querySelector("[data-mas-clock-minute]")
};
if (!clock.timeElement || !clock.hourHand || !clock.minuteHand) {
return;
}
clocks = clocks.filter(function (registeredClock) {
return registeredClock.element !== element;
});
clocks.push(clock);
updateClock(clock);
if (!clockTimer) {
clockTimer = window.setInterval(refreshClocks, 60000);
}
}
function createCityClockController(section) {
var root = section ? section.querySelector("[data-mas-city-clocks]") : null;
var destinationClock = root ? root.querySelector("[data-mas-city-clock='destination']") : null;
var referenceClock = root ? root.querySelector("[data-mas-city-clock='reference']") : null;
var destinationReady = false;
if (!root) {
return null;
}
if (referenceClock) {
registerAnalogClock(
referenceClock,
referenceClock.getAttribute("data-mas-clock-timezone"),
referenceClock.getAttribute("data-mas-clock-label") || "Geneve"
);
}
return {
registerDestination: function (cityName, timezone) {
if (destinationReady || !destinationClock || !timezone) {
return;
}
destinationReady = true;
registerAnalogClock(destinationClock, timezone, cityName || "Destination");
}
};
}
function getCityOverviewMarker() {
var marker = document.querySelector(".MoveAndStudyCore.MoveAndStudyCore-city .mas-city-map__markers [data-city][data-lat][data-lng]");
if (!marker) {
return null;
}
return {
title: marker.getAttribute("data-city") || "",
latitude: marker.getAttribute("data-lat"),
longitude: marker.getAttribute("data-lng")
};
}
function setFallback(panel, cityName) {
panel.classList.add("is-unavailable");
panel.classList.remove("is-ready");
panel.innerHTML = [
'<p class="mas-city-weather__eyebrow">Météo locale</p>',
'<p class="mas-city-weather__current"><span class="mas-city-weather__icon" aria-hidden="true">' + weatherIconSvg("variable") + '</span><span class="mas-city-weather__summary">Prévisions indisponibles</span></p>',
'<p class="mas-city-weather__fallback">Consultez la destination ' + escapeHtml(cityName || "sélectionnée") + ' pour préparer votre séjour.</p>'
].join("");
}
function renderWeather(panel, cityName, data) {
var current = data.current || {};
var daily = data.daily || {};
var summary = weatherSummary(current.weather_code);
var dates = (daily.time || []).slice(0, 4);
var highs = daily.temperature_2m_max || [];
var lows = daily.temperature_2m_min || [];
var codes = daily.weather_code || [];
var forecast = dates.map(function (date, index) {
var daySummary = weatherSummary(codes[index]);
return [
'<div class="mas-city-weather__day">',
'<div class="mas-city-weather__day-meta">',
'<span class="mas-city-weather__day-label">' + escapeHtml(formatDay(date, index)) + '</span>',
'<span class="mas-city-weather__day-icon" aria-hidden="true">' + weatherIconSvg(daySummary.iconKey) + '</span>',
'</div>',
'<strong class="mas-city-weather__day-temp">' + escapeHtml(formatTemperature(lows[index])) + '/' + escapeHtml(formatTemperature(highs[index])) + '</strong>',
'</div>'
].join("");
}).join("");
var updated = formatUpdated(current.time);
panel.classList.add("is-ready");
panel.classList.remove("is-unavailable");
panel.innerHTML = [
'<p class="mas-city-weather__eyebrow">Météo à ' + escapeHtml(cityName) + '</p>',
'<p class="mas-city-weather__current">',
'<span class="mas-city-weather__icon" aria-hidden="true">' + weatherIconSvg(summary.iconKey) + '</span>',
'<span class="mas-city-weather__summary">' + escapeHtml(formatTemperature(current.temperature_2m)) + ' aujourd’hui, ' + escapeHtml(summary.label.toLowerCase()) + '</span>',
'</p>',
'<div class="mas-city-weather__forecast">' + forecast + '</div>',
updated ? '<p class="mas-city-weather__updated">Mise à jour à ' + escapeHtml(updated) + '</p>' : ''
].join("");
}
function getWeatherCacheKey(marker) {
var latitude = Math.round(parseFloat(marker.latitude) * 1000) / 1000;
var longitude = Math.round(parseFloat(marker.longitude) * 1000) / 1000;
return "mas.weather." + latitude + "." + longitude + "." + new Date().toISOString().slice(0, 13);
}
function getCachedJson(cacheKey) {
var stored;
var parsed;
try {
stored = window.sessionStorage && window.sessionStorage.getItem(cacheKey);
if (!stored) {
return null;
}
parsed = JSON.parse(stored);
if (!parsed || !parsed.createdAt || (Date.now() - parsed.createdAt) > cacheLifetime) {
window.sessionStorage.removeItem(cacheKey);
return null;
}
return parsed.payload || null;
} catch (error) {
return null;
}
}
function setCachedJson(cacheKey, payload) {
try {
if (window.sessionStorage) {
window.sessionStorage.setItem(cacheKey, JSON.stringify({
createdAt: Date.now(),
payload: payload
}));
}
} catch (error) {
return;
}
}
function fetchWeather(url, cacheKey) {
var cached = getCachedJson(cacheKey);
if (cached) {
return Promise.resolve(cached);
}
return window.fetch(url)
.then(function (response) {
if (!response.ok) {
throw new Error("Weather unavailable");
}
return response.json();
})
.then(function (data) {
setCachedJson(cacheKey, data);
return data;
});
}
function hydrateCityOverview(section) {
var panel = section ? section.querySelector("[data-mas-city-weather]") : null;
var marker = getCityOverviewMarker();
var cityClockController = createCityClockController(section);
var cityName = marker ? marker.title : "";
var url;
if (!panel || !marker || !window.fetch || !isFiniteNumber(marker.latitude) || !isFiniteNumber(marker.longitude)) {
if (panel) {
setFallback(panel, cityName);
}
return;
}
url = forecastUrl
+ "?latitude=" + encodeURIComponent(marker.latitude)
+ "&longitude=" + encodeURIComponent(marker.longitude)
+ "&current=temperature_2m,weather_code"
+ "&daily=weather_code,temperature_2m_max,temperature_2m_min"
+ "&forecast_days=4"
+ "&timezone=auto";
fetchWeather(url, getWeatherCacheKey(marker))
.then(function (data) {
if (cityClockController) {
cityClockController.registerDestination(cityName, data.timezone);
}
renderWeather(panel, cityName, data);
})
.catch(function () {
setFallback(panel, cityName);
});
}
onReady(function () {
var cityOverview = document.querySelector(".MoveAndStudyCore.MoveAndStudyCore-city .mas-city-overview");
if (!cityOverview) {
return;
}
hydrateCityOverview(cityOverview);
});
}());
(function () {
function splitCityHeroTitle() {
var heroLead = document.querySelector(".MoveAndStudyCore-city .mas-city-hero__lead");
if (!heroLead) {
return;
}
var titleNode = heroLead.querySelector("h1");
if (!titleNode || titleNode.querySelector(".mas-city-hero__title-main")) {
return;
}
var rawTitle = (titleNode.textContent || "").replace(/\s+/g, " ").trim();
if (!rawTitle) {
return;
}
var splitIndex = -1;
var splitPattern = /\s[àa]\s+/ig;
var match;
while ((match = splitPattern.exec(rawTitle)) !== null) {
splitIndex = match.index;
}
if (splitIndex <= 0) {
return;
}
var mainTitle = rawTitle.slice(0, splitIndex).trim();
var placeTitle = rawTitle.slice(splitIndex + 1).trim();
if (!mainTitle || !/^[àa]\s+\S+/i.test(placeTitle)) {
return;
}
var mainSpan = document.createElement("span");
mainSpan.className = "mas-city-hero__title-main";
mainSpan.textContent = mainTitle;
var placeSpan = document.createElement("span");
placeSpan.className = "mas-city-hero__title-place";
placeSpan.textContent = placeTitle;
titleNode.textContent = "";
titleNode.appendChild(mainSpan);
titleNode.appendChild(document.createElement("br"));
titleNode.appendChild(placeSpan);
}
if (document.readyState === "loading") {
document.addEventListener("DOMContentLoaded", splitCityHeroTitle);
} else {
splitCityHeroTitle();
}
})();
(function () {
function onReady(callback) {
if (document.readyState === "loading") {
document.addEventListener("DOMContentLoaded", callback);
return;
}
callback();
}
function hideEmptyAccreditations() {
var accreditations = document.getElementById("school-accreditations");
if (!accreditations) {
return;
}
var component = accreditations.querySelector(".Component-Container");
if (component && component.innerHTML.trim() === "") {
accreditations.style.display = "none";
}
}
function disableMapScroll() {
if (
typeof window.gmapstraction === "object" &&
window.gmapstraction.maps &&
window.gmapstraction.maps.google &&
typeof window.gmapstraction.maps.google.disableScrollWheelZoom === "function"
) {
window.gmapstraction.maps.google.disableScrollWheelZoom();
}
}
var googleMapsCallbacks = [];
var googleMapsFailureCallbacks = [];
var googleMapsLoading = false;
var googleMapsTimeoutId = null;
var googleMapsCallbackName = "__masSchoolPageGoogleMapsReady";
var holidayTooltipsReady = false;
function flushGoogleMapsCallbacks() {
var callbacks = googleMapsCallbacks.slice();
googleMapsCallbacks = [];
callbacks.forEach(function (readyCallback) {
readyCallback();
});
}
function flushGoogleMapsFailureCallbacks() {
var callbacks = googleMapsFailureCallbacks.slice();
googleMapsFailureCallbacks = [];
callbacks.forEach(function (failureCallback) {
failureCallback();
});
}
function resetGoogleMapsLoader() {
googleMapsLoading = false;
if (googleMapsTimeoutId) {
window.clearTimeout(googleMapsTimeoutId);
googleMapsTimeoutId = null;
}
}
function loadGoogleMaps(key, callback, failureCallback) {
if (window.google && window.google.maps && typeof window.google.maps.Map === "function") {
callback();
return;
}
googleMapsCallbacks.push(callback);
if (typeof failureCallback === "function") {
googleMapsFailureCallbacks.push(failureCallback);
}
if (googleMapsLoading || !key) {
return;
}
googleMapsLoading = true;
function fail() {
resetGoogleMapsLoader();
googleMapsCallbacks = [];
flushGoogleMapsFailureCallbacks();
}
window.gm_authFailure = fail;
window[googleMapsCallbackName] = function () {
if (!(window.google && window.google.maps && typeof window.google.maps.Map === "function")) {
fail();
return;
}
resetGoogleMapsLoader();
flushGoogleMapsCallbacks();
};
var script = document.createElement("script");
script.src = "https://maps.googleapis.com/maps/api/js?key=" + encodeURIComponent(key) +
"&callback=" + encodeURIComponent(googleMapsCallbackName) +
"&loading=async";
script.async = true;
script.defer = true;
script.onerror = fail;
document.head.appendChild(script);
googleMapsTimeoutId = window.setTimeout(fail, 10000);
}
function escapeHtml(value) {
return String(value || "")
.replace(/&/g, "&amp;")
.replace(/</g, "&lt;")
.replace(/>/g, "&gt;")
.replace(/"/g, "&quot;")
.replace(/'/g, "&#039;");
}
function getNumberAttribute(element, name, fallback) {
var value = parseFloat(element.getAttribute(name));
return isFinite(value) ? value : fallback;
}
function hydrateSchoolMapFrames() {
var components = Array.prototype.slice.call(document.querySelectorAll("[data-mas-school-map][data-map-key][data-map-lat][data-map-lng]"));
components.forEach(function (component) {
var frame = component.querySelector(".mas-school-map__iframe");
var key = component.getAttribute("data-map-key") || "";
var lat = component.getAttribute("data-map-lat") || "";
var lng = component.getAttribute("data-map-lng") || "";
var zoom = component.getAttribute("data-map-zoom") || "14";
var mapSrc = "";
if (!frame || !key || !lat || !lng) {
return;
}
mapSrc = "/synergee/Themes/MoveAndStudyCore/Components/InteractiveMap/Iframe/InteractiveMap.html" +
"?key=" + encodeURIComponent(key) +
"&lat=" + encodeURIComponent(lat) +
"&lng=" + encodeURIComponent(lng) +
"&zoom=" + encodeURIComponent(zoom);
if (frame.getAttribute("src") === mapSrc) {
return;
}
frame.setAttribute("src", mapSrc);
});
}
function renderSchoolStreetView(component) {
var canvas = component.querySelector("[data-mas-school-streetview-canvas]");
var lat = getNumberAttribute(component, "data-latitude", null);
var lng = getNumberAttribute(component, "data-longitude", null);
if (!canvas || component.getAttribute("data-mas-school-streetview-ready") === "1" || lat === null || lng === null) {
return;
}
new google.maps.StreetViewPanorama(canvas, {
position: { lat: lat, lng: lng },
pov: {
heading: getNumberAttribute(component, "data-heading", 0),
pitch: getNumberAttribute(component, "data-pitch", 0)
},
zoom: getNumberAttribute(component, "data-zoom", 1),
addressControl: false,
fullscreenControl: true,
linksControl: true,
panControl: true,
enableCloseButton: false
});
component.setAttribute("data-mas-school-streetview-ready", "1");
}
function initSchoolLocationComponents() {
var streetViewComponents = Array.prototype.slice.call(document.querySelectorAll("[data-mas-school-streetview]"));
var key = "";
hydrateSchoolMapFrames();
if (!streetViewComponents.length) {
return;
}
streetViewComponents.some(function (component) {
key = component.getAttribute("data-mas-google-key") || "";
return Boolean(key);
});
loadGoogleMaps(
key,
function () {
streetViewComponents.forEach(function (component) {
renderSchoolStreetView(component);
});
},
function () {}
);
}
function hideBookingCart() {
if (!window.jQuery) {
return;
}
window.jQuery(".Synergee-Web-Page-Component-Booking-Cart").addClass("hidden");
}
function initHolidayTooltips() {
var chips = Array.prototype.slice.call(document.querySelectorAll("[data-mas-holiday-chip]"));
if (!chips.length || holidayTooltipsReady) {
return;
}
holidayTooltipsReady = true;
var isCoarsePointer = typeof window.matchMedia === "function" &&
window.matchMedia("(hover: none), (pointer: coarse)").matches;
function setExpandedState(chip, expanded) {
var trigger = chip.querySelector("[data-mas-holiday-trigger]");
var tooltip = chip.querySelector("[data-mas-holiday-tooltip]");
chip.classList.toggle("is-active", expanded);
if (trigger) {
trigger.setAttribute("aria-expanded", expanded ? "true" : "false");
}
if (tooltip) {
tooltip.setAttribute("aria-hidden", expanded ? "false" : "true");
}
}
function closeAll(exceptChip) {
chips.forEach(function (chip) {
if (chip !== exceptChip) {
setExpandedState(chip, false);
}
});
}
chips.forEach(function (chip) {
var trigger = chip.querySelector("[data-mas-holiday-trigger]");
if (!trigger) {
return;
}
trigger.addEventListener("click", function (event) {
var isExpanded = chip.classList.contains("is-active");
event.preventDefault();
closeAll(chip);
setExpandedState(chip, !isExpanded);
});
chip.addEventListener("mouseenter", function () {
if (isCoarsePointer) {
return;
}
closeAll(chip);
setExpandedState(chip, true);
});
chip.addEventListener("mouseleave", function () {
if (isCoarsePointer) {
return;
}
if (!chip.contains(document.activeElement)) {
setExpandedState(chip, false);
}
});
chip.addEventListener("focusin", function () {
closeAll(chip);
setExpandedState(chip, true);
});
chip.addEventListener("focusout", function () {
window.setTimeout(function () {
if (!chip.contains(document.activeElement)) {
setExpandedState(chip, false);
}
}, 0);
});
});
document.addEventListener("click", function (event) {
var clickedInside = chips.some(function (chip) {
return chip.contains(event.target);
});
if (!clickedInside) {
closeAll();
}
});
document.addEventListener("keydown", function (event) {
if (event.key !== "Escape") {
return;
}
closeAll();
});
}
function revealBookingCartWhenReady() {
if (!window.jQuery || !window.bookingManager || !window.bookingManager.booking) {
return;
}
if (typeof window.bookingManager.booking.addEvent === "function") {
window.bookingManager.booking.addEvent("onPriceRefreshComplete", function () {
window.jQuery(".Synergee-Web-Page-Component-Booking-Cart").removeClass("hidden");
});
}
}
function initSchoolMediaSlider() {
var slider = document.querySelector("[data-mas-school-media-slider]");
if (!slider || slider.getAttribute("data-mas-school-media-ready") === "1") {
return;
}
var rawSlides = Array.prototype.slice.call(slider.querySelectorAll("[data-mas-school-media-item]"));
var slides = rawSlides.filter(function (slide) {
var image = slide.querySelector("img");
var video = slide.querySelector(".Synergee-Web-Page-Component-Video-PlayVideoButton[href]");
if (!image && !video) {
slide.classList.add("is-empty");
slide.setAttribute("aria-hidden", "true");
return false;
}
return true;
});
if (!slides.length) {
slider.classList.add("is-empty");
return;
}
slider.setAttribute("data-mas-school-media-ready", "1");
var previous = slider.querySelector("[data-mas-school-media-prev]");
var next = slider.querySelector("[data-mas-school-media-next]");
var controls = slider.querySelector("[data-mas-school-media-controls]");
var status = slider.querySelector("[data-mas-school-media-status]");
var dotsContainer = slider.querySelector("[data-mas-school-media-dots]");
var hideDots = slider.getAttribute("data-mas-school-media-hide-dots") === "1";
var dots = [];
var current = 0;
function getSlideLabel(slide, index) {
var videoText = slide.querySelector(".school-gallery-video-txt");
var image = slide.querySelector("img");
var text = "";
if (videoText) {
text = videoText.textContent || "";
}
if (!text && image) {
text = image.getAttribute("alt") || "";
}
text = text.replace(/\s+/g, " ").trim();
if (text) {
return text;
}
return (slide.getAttribute("data-mas-media-kind") === "video" ? "Video " : "Photo ") + (index + 1);
}
function setFocusable(slide, enabled) {
var focusables = slide.querySelectorAll("a, button, input, select, textarea, [tabindex]");
Array.prototype.forEach.call(focusables, function (element) {
element.setAttribute("tabindex", enabled ? "0" : "-1");
});
if (slide.matches("a")) {
slide.setAttribute("tabindex", enabled ? "0" : "-1");
}
}
function showSlide(index) {
current = (index + slides.length) % slides.length;
slides.forEach(function (slide, slideIndex) {
var active = slideIndex === current;
slide.classList.toggle("is-active", active);
slide.setAttribute("aria-hidden", active ? "false" : "true");
setFocusable(slide, active);
});
dots.forEach(function (dot, dotIndex) {
var active = dotIndex === current;
dot.classList.toggle("is-active", active);
dot.setAttribute("aria-current", active ? "true" : "false");
});
if (status) {
status.textContent = (current + 1) + " / " + slides.length;
}
}
if (dotsContainer && !hideDots) {
dotsContainer.innerHTML = "";
slides.forEach(function (slide, index) {
var dot = document.createElement("button");
var kind = slide.getAttribute("data-mas-media-kind");
dot.type = "button";
dot.className = "mas-school-media-slider__dot";
if (kind === "video") {
dot.className += " mas-school-media-slider__dot--video";
}
dot.textContent = String(index + 1);
dot.setAttribute("aria-label", "Afficher " + getSlideLabel(slide, index));
dot.addEventListener("click", function () {
showSlide(index);
});
dotsContainer.appendChild(dot);
dots.push(dot);
});
} else if (dotsContainer) {
dotsContainer.setAttribute("hidden", "hidden");
}
if (previous) {
previous.addEventListener("click", function () {
showSlide(current - 1);
});
}
if (next) {
next.addEventListener("click", function () {
showSlide(current + 1);
});
}
slider.addEventListener("keydown", function (event) {
if (event.key === "ArrowLeft") {
event.preventDefault();
showSlide(current - 1);
}
if (event.key === "ArrowRight") {
event.preventDefault();
showSlide(current + 1);
}
});
if (controls && slides.length > 1) {
controls.hidden = false;
}
slider.classList.add("is-ready");
showSlide(0);
}
onReady(function () {
var galleryToggle = document.querySelector("[data-mas-school-gallery-toggle]");
var galleryToggleContainer = document.getElementById("gallery-btn-container");
var gallery = document.getElementById("school-gallery");
hydrateSchoolMapFrames();
if (galleryToggle && gallery) {
galleryToggle.addEventListener("click", function () {
gallery.classList.toggle("clicked");
if (galleryToggleContainer) {
galleryToggleContainer.classList.toggle("active");
}
});
}
initSchoolMediaSlider();
initSchoolLocationComponents();
initHolidayTooltips();
hideEmptyAccreditations();
window.setTimeout(hideEmptyAccreditations, 400);
hideBookingCart();
revealBookingCartWhenReady();
});
if (typeof window.addEvent === "function") {
window.addEvent("domready", function () {
hydrateSchoolMapFrames();
initSchoolMediaSlider();
initSchoolLocationComponents();
initHolidayTooltips();
hideEmptyAccreditations();
hideBookingCart();
revealBookingCartWhenReady();
});
window.addEvent("mapInitialized", disableMapScroll);
}
}());
(function () {
var DEFAULT_CHAT_WIDGET_URL = "https://chat.selvix-test.ch/widget?sourceSite=moveandstudy&language=fr";
var CHAT_WIDGET_URL = "";
if (typeof window.CHAT_WIDGET_URL === "string") {
CHAT_WIDGET_URL = window.CHAT_WIDGET_URL.replace(/^\s+|\s+$/g, "");
}
if (!CHAT_WIDGET_URL) {
CHAT_WIDGET_URL = DEFAULT_CHAT_WIDGET_URL;
}
window.CHAT_WIDGET_URL = CHAT_WIDGET_URL;
function onReady(callback) {
if (document.readyState === "loading") {
document.addEventListener("DOMContentLoaded", callback);
return;
}
callback();
}
function detectPageType() {
var body = document.body;
var classes;
var i;
var className;
var prefix = "MoveAndStudyCore-";
if (!body) {
return "";
}
if (body.dataset && body.dataset.pageType) {
return body.dataset.pageType;
}
classes = (body.className || "").split(/\s+/);
for (i = 0; i < classes.length; i += 1) {
className = classes[i];
if (className.indexOf(prefix) === 0) {
return className.substring(prefix.length);
}
}
return "";
}
function buildWidgetUrl(source) {
var url;
try {
url = new URL(source, window.location.href);
} catch (error) {
return "";
}
url.searchParams.set("lang", document.documentElement.lang || "");
url.searchParams.set("pageUrl", window.location.href);
url.searchParams.set("pageType", detectPageType());
return url.toString();
}
function bindChatWidget(root) {
var toggle = root.querySelector("[data-mas-chat-toggle]");
var panel = root.querySelector("[data-mas-chat-panel]");
var closeButton = root.querySelector("[data-mas-chat-close]");
var iframe = root.querySelector("[data-mas-chat-iframe]");
var fallback = root.querySelector("[data-mas-chat-fallback]");
var source = window.CHAT_WIDGET_URL || "";
var loaded = false;
var loadAttempted = false;
var loadTimeout = 0;
var unavailable = false;
if (!toggle || !panel || !iframe) {
return;
}
function clearUnavailableState() {
unavailable = false;
iframe.hidden = false;
if (fallback) {
fallback.hidden = true;
}
}
function setUnavailableState() {
unavailable = true;
iframe.hidden = true;
if (fallback) {
fallback.hidden = false;
}
}
function clearLoadTimeout() {
if (loadTimeout) {
window.clearTimeout(loadTimeout);
loadTimeout = 0;
}
}
if (!source) {
toggle.setAttribute("disabled", "disabled");
toggle.setAttribute("title", "Chat non configure");
return;
}
iframe.addEventListener("load", function () {
loaded = true;
clearLoadTimeout();
clearUnavailableState();
});
iframe.addEventListener("error", function () {
clearLoadTimeout();
setUnavailableState();
});
function openPanel() {
if (!loadAttempted) {
var resolvedSource = buildWidgetUrl(source);
if (!resolvedSource) {
return;
}
clearUnavailableState();
iframe.src = resolvedSource;
loadAttempted = true;
loadTimeout = window.setTimeout(function () {
if (!loaded) {
setUnavailableState();
}
}, 10000);
} else if (!loaded && unavailable) {
setUnavailableState();
}
panel.hidden = false;
toggle.setAttribute("aria-expanded", "true");
}
function closePanel() {
panel.hidden = true;
toggle.setAttribute("aria-expanded", "false");
}
toggle.addEventListener("click", function () {
if (panel.hidden) {
openPanel();
return;
}
closePanel();
});
if (closeButton) {
closeButton.addEventListener("click", closePanel);
}
document.addEventListener("keydown", function (event) {
if (event.key === "Escape" && !panel.hidden) {
closePanel();
toggle.focus();
}
});
document.addEventListener("click", function (event) {
if (panel.hidden) {
return;
}
if (!root.contains(event.target)) {
closePanel();
}
});
}
onReady(function () {
var chatWidgets = document.querySelectorAll("[data-mas-chat-widget]");
var i;
for (i = 0; i < chatWidgets.length; i += 1) {
bindChatWidget(chatWidgets[i]);
}
});
}());
