/*! jQuery v1.7.1 jquery.com | jquery.org/license */ /* Pathed by MJ with workaround for https://connect.microsoft.com/IE/feedback/details/774078 see http://bugs.jquery.com/ticket/12577 */ (function(a,b){function cy(a){return f.isWindow(a)?a:a.nodeType===9?a.defaultView||a.parentWindow:!1}function cv(a){if(!ck[a]){var b=c.body,d=f("<"+a+">").appendTo(b),e=d.css("display");d.remove();if(e==="none"||e===""){cl||(cl=c.createElement("iframe"),cl.frameBorder=cl.width=cl.height=0),b.appendChild(cl);if(!cm||!cl.createElement)cm=(cl.contentWindow||cl.contentDocument).document,cm.write((c.compatMode==="CSS1Compat"?"":"")+""),cm.close();d=cm.createElement(a),cm.body.appendChild(d),e=f.css(d,"display"),b.removeChild(cl)}ck[a]=e}return ck[a]}function cu(a,b){var c={};f.each(cq.concat.apply([],cq.slice(0,b)),function(){c[this]=a});return c}function ct(){cr=b}function cs(){setTimeout(ct,0);return cr=f.now()}function cj(){try{return new a.ActiveXObject("Microsoft.XMLHTTP")}catch(b){}}function ci(){try{return new a.XMLHttpRequest}catch(b){}}function cc(a,c){a.dataFilter&&(c=a.dataFilter(c,a.dataType));var d=a.dataTypes,e={},g,h,i=d.length,j,k=d[0],l,m,n,o,p;for(g=1;g0){if(c!=="border")for(;g=0===c})}function S(a){return!a||!a.parentNode||a.parentNode.nodeType===11}function K(){return!0}function J(){return!1}function n(a,b,c){var d=b+"defer",e=b+"queue",g=b+"mark",h=f._data(a,d);h&&(c==="queue"||!f._data(a,e))&&(c==="mark"||!f._data(a,g))&&setTimeout(function(){!f._data(a,e)&&!f._data(a,g)&&(f.removeData(a,d,!0),h.fire())},0)}function m(a){for(var b in a){if(b==="data"&&f.isEmptyObject(a[b]))continue;if(b!=="toJSON")return!1}return!0}function l(a,c,d){if(d===b&&a.nodeType===1){var e="data-"+c.replace(k,"-$1").toLowerCase();d=a.getAttribute(e);if(typeof d=="string"){try{d=d==="true"?!0:d==="false"?!1:d==="null"?null:f.isNumeric(d)?parseFloat(d):j.test(d)?f.parseJSON(d):d}catch(g){}f.data(a,c,d)}else d=b}return d}function h(a){var b=g[a]={},c,d;a=a.split(/\s+/);for(c=0,d=a.length;c)[^>]*$|#([\w\-]*)$)/,j=/\S/,k=/^\s+/,l=/\s+$/,m=/^<(\w+)\s*\/?>(?:<\/\1>)?$/,n=/^[\],:{}\s]*$/,o=/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g,p=/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g,q=/(?:^|:|,)(?:\s*\[)+/g,r=/(webkit)[ \/]([\w.]+)/,s=/(opera)(?:.*version)?[ \/]([\w.]+)/,t=/(msie) ([\w.]+)/,u=/(mozilla)(?:.*? rv:([\w.]+))?/,v=/-([a-z]|[0-9])/ig,w=/^-ms-/,x=function(a,b){return(b+"").toUpperCase()},y=d.userAgent,z,A,B,C=Object.prototype.toString,D=Object.prototype.hasOwnProperty,E=Array.prototype.push,F=Array.prototype.slice,G=String.prototype.trim,H=Array.prototype.indexOf,I={};e.fn=e.prototype={constructor:e,init:function(a,d,f){var g,h,j,k;if(!a)return this;if(a.nodeType){this.context=this[0]=a,this.length=1;return this}if(a==="body"&&!d&&c.body){this.context=c,this[0]=c.body,this.selector=a,this.length=1;return this}if(typeof a=="string"){a.charAt(0)!=="<"||a.charAt(a.length-1)!==">"||a.length<3?g=i.exec(a):g=[null,a,null];if(g&&(g[1]||!d)){if(g[1]){d=d instanceof e?d[0]:d,k=d?d.ownerDocument||d:c,j=m.exec(a),j?e.isPlainObject(d)?(a=[c.createElement(j[1])],e.fn.attr.call(a,d,!0)):a=[k.createElement(j[1])]:(j=e.buildFragment([g[1]],[k]),a=(j.cacheable?e.clone(j.fragment):j.fragment).childNodes);return e.merge(this,a)}h=c.getElementById(g[2]);if(h&&h.parentNode){if(h.id!==g[2])return f.find(a);this.length=1,this[0]=h}this.context=c,this.selector=a;return this}return!d||d.jquery?(d||f).find(a):this.constructor(d).find(a)}if(e.isFunction(a))return f.ready(a);a.selector!==b&&(this.selector=a.selector,this.context=a.context);return e.makeArray(a,this)},selector:"",jquery:"1.7.1",length:0,size:function(){return this.length},toArray:function(){return F.call(this,0)},get:function(a){return a==null?this.toArray():a<0?this[this.length+a]:this[a]},pushStack:function(a,b,c){var d=this.constructor();e.isArray(a)?E.apply(d,a):e.merge(d,a),d.prevObject=this,d.context=this.context,b==="find"?d.selector=this.selector+(this.selector?" ":"")+c:b&&(d.selector=this.selector+"."+b+"("+c+")");return d},each:function(a,b){return e.each(this,a,b)},ready:function(a){e.bindReady(),A.add(a);return this},eq:function(a){a=+a;return a===-1?this.slice(a):this.slice(a,a+1)},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},slice:function(){return this.pushStack(F.apply(this,arguments),"slice",F.call(arguments).join(","))},map:function(a){return this.pushStack(e.map(this,function(b,c){return a.call(b,c,b)}))},end:function(){return this.prevObject||this.constructor(null)},push:E,sort:[].sort,splice:[].splice},e.fn.init.prototype=e.fn,e.extend=e.fn.extend=function(){var a,c,d,f,g,h,i=arguments[0]||{},j=1,k=arguments.length,l=!1;typeof i=="boolean"&&(l=i,i=arguments[1]||{},j=2),typeof i!="object"&&!e.isFunction(i)&&(i={}),k===j&&(i=this,--j);for(;j0)return;A.fireWith(c,[e]),e.fn.trigger&&e(c).trigger("ready").off("ready")}},bindReady:function(){if(!A){A=e.Callbacks("once memory");if(c.readyState==="complete")return setTimeout(e.ready,1);if(c.addEventListener)c.addEventListener("DOMContentLoaded",B,!1),a.addEventListener("load",e.ready,!1);else if(c.attachEvent){c.attachEvent("onreadystatechange",B),a.attachEvent("onload",e.ready);var b=!1;try{b=a.frameElement==null}catch(d){}c.documentElement.doScroll&&b&&J()}}},isFunction:function(a){return e.type(a)==="function"},isArray:Array.isArray||function(a){return e.type(a)==="array"},isWindow:function(a){return a&&typeof a=="object"&&"setInterval"in a},isNumeric:function(a){return!isNaN(parseFloat(a))&&isFinite(a)},type:function(a){return a==null?String(a):I[C.call(a)]||"object"},isPlainObject:function(a){if(!a||e.type(a)!=="object"||a.nodeType||e.isWindow(a))return!1;try{if(a.constructor&&!D.call(a,"constructor")&&!D.call(a.constructor.prototype,"isPrototypeOf"))return!1}catch(c){return!1}var d;for(d in a);return d===b||D.call(a,d)},isEmptyObject:function(a){for(var b in a)return!1;return!0},error:function(a){throw new Error(a)},parseJSON:function(b){if(typeof b!="string"||!b)return null;b=e.trim(b);if(a.JSON&&a.JSON.parse)return a.JSON.parse(b);if(n.test(b.replace(o,"@").replace(p,"]").replace(q,"")))return(new Function("return "+b))();e.error("Invalid JSON: "+b)},parseXML:function(c){var d,f;try{a.DOMParser?(f=new DOMParser,d=f.parseFromString(c,"text/xml")):(d=new ActiveXObject("Microsoft.XMLDOM"),d.async="false",d.loadXML(c))}catch(g){d=b}(!d||!d.documentElement||d.getElementsByTagName("parsererror").length)&&e.error("Invalid XML: "+c);return d},noop:function(){},globalEval:function(b){b&&j.test(b)&&(a.execScript||function(b){a.eval.call(a,b)})(b)},camelCase:function(a){return a.replace(w,"ms-").replace(v,x)},nodeName:function(a,b){return a.nodeName&&a.nodeName.toUpperCase()===b.toUpperCase()},each:function(a,c,d){var f,g=0,h=a.length,i=h===b||e.isFunction(a);if(d){if(i){for(f in a)if(c.apply(a[f],d)===!1)break}else for(;g0&&a[0]&&a[j-1]||j===0||e.isArray(a));if(k)for(;i1?i.call(arguments,0):b,j.notifyWith(k,e)}}function l(a){return function(c){b[a]=arguments.length>1?i.call(arguments,0):c,--g||j.resolveWith(j,b)}}var b=i.call(arguments,0),c=0,d=b.length,e=Array(d),g=d,h=d,j=d<=1&&a&&f.isFunction(a.promise)?a:f.Deferred(),k=j.promise();if(d>1){for(;c
a",d=q.getElementsByTagName("*"),e=q.getElementsByTagName("a")[0];if(!d||!d.length||!e)return{};g=c.createElement("select"),h=g.appendChild(c.createElement("option")),i=q.getElementsByTagName("input")[0],b={leadingWhitespace:q.firstChild.nodeType===3,tbody:!q.getElementsByTagName("tbody").length,htmlSerialize:!!q.getElementsByTagName("link").length,style:/top/.test(e.getAttribute("style")),hrefNormalized:e.getAttribute("href")==="/a",opacity:/^0.55/.test(e.style.opacity),cssFloat:!!e.style.cssFloat,checkOn:i.value==="on",optSelected:h.selected,getSetAttribute:q.className!=="t",enctype:!!c.createElement("form").enctype,html5Clone:c.createElement("nav").cloneNode(!0).outerHTML!=="<:nav>",submitBubbles:!0,changeBubbles:!0,focusinBubbles:!1,deleteExpando:!0,noCloneEvent:!0,inlineBlockNeedsLayout:!1,shrinkWrapBlocks:!1,reliableMarginRight:!0},i.checked=!0,b.noCloneChecked=i.cloneNode(!0).checked,g.disabled=!0,b.optDisabled=!h.disabled;try{delete q.test}catch(s){b.deleteExpando=!1}!q.addEventListener&&q.attachEvent&&q.fireEvent&&(q.attachEvent("onclick",function(){b.noCloneEvent=!1}),q.cloneNode(!0).fireEvent("onclick")),i=c.createElement("input"),i.value="t",i.setAttribute("type","radio"),b.radioValue=i.value==="t",i.setAttribute("checked","checked"),q.appendChild(i),k=c.createDocumentFragment(),k.appendChild(q.lastChild),b.checkClone=k.cloneNode(!0).cloneNode(!0).lastChild.checked,b.appendChecked=i.checked,k.removeChild(i),k.appendChild(q),q.innerHTML="",a.getComputedStyle&&(j=c.createElement("div"),j.style.width="0",j.style.marginRight="0",q.style.width="2px",q.appendChild(j),b.reliableMarginRight=(parseInt((a.getComputedStyle(j,null)||{marginRight:0}).marginRight,10)||0)===0);if(q.attachEvent)for(o in{submit:1,change:1,focusin:1})n="on"+o,p=n in q,p||(q.setAttribute(n,"return;"),p=typeof q[n]=="function"),b[o+"Bubbles"]=p;k.removeChild(q),k=g=h=j=q=i=null,f(function(){var a,d,e,g,h,i,j,k,m,n,o,r=c.getElementsByTagName("body")[0];!r||(j=1,k="position:absolute;top:0;left:0;width:1px;height:1px;margin:0;",m="visibility:hidden;border:0;",n="style='"+k+"border:5px solid #000;padding:0;'",o="
"+""+"
",a=c.createElement("div"),a.style.cssText=m+"width:0;height:0;position:static;top:0;margin-top:"+j+"px",r.insertBefore(a,r.firstChild),q=c.createElement("div"),a.appendChild(q),q.innerHTML="
t
",l=q.getElementsByTagName("td"),p=l[0].offsetHeight===0,l[0].style.display="",l[1].style.display="none",b.reliableHiddenOffsets=p&&l[0].offsetHeight===0,q.innerHTML="",q.style.width=q.style.paddingLeft="1px",f.boxModel=b.boxModel=q.offsetWidth===2,typeof q.style.zoom!="undefined"&&(q.style.display="inline",q.style.zoom=1,b.inlineBlockNeedsLayout=q.offsetWidth===2,q.style.display="",q.innerHTML="
",b.shrinkWrapBlocks=q.offsetWidth!==2),q.style.cssText=k+m,q.innerHTML=o,d=q.firstChild,e=d.firstChild,h=d.nextSibling.firstChild.firstChild,i={doesNotAddBorder:e.offsetTop!==5,doesAddBorderForTableAndCells:h.offsetTop===5},e.style.position="fixed",e.style.top="20px",i.fixedPosition=e.offsetTop===20||e.offsetTop===15,e.style.position=e.style.top="",d.style.overflow="hidden",d.style.position="relative",i.subtractsBorderForOverflowNotVisible=e.offsetTop===-5,i.doesNotIncludeMarginInBodyOffset=r.offsetTop!==j,r.removeChild(a),q=a=null,f.extend(b,i))});return b}();var j=/^(?:\{.*\}|\[.*\])$/,k=/([A-Z])/g;f.extend({cache:{},uuid:0,expando:"jQuery"+(f.fn.jquery+Math.random()).replace(/\D/g,""),noData:{embed:!0,object:"clsid:D27CDB6E-AE6D-11cf-96B8-444553540000",applet:!0},hasData:function(a){a=a.nodeType?f.cache[a[f.expando]]:a[f.expando];return!!a&&!m(a)},data:function(a,c,d,e){if(!!f.acceptData(a)){var g,h,i,j=f.expando,k=typeof c=="string",l=a.nodeType,m=l?f.cache:a,n=l?a[j]:a[j]&&j,o=c==="events";if((!n||!m[n]||!o&&!e&&!m[n].data)&&k&&d===b)return;n||(l?a[j]=n=++f.uuid:n=j),m[n]||(m[n]={},l||(m[n].toJSON=f.noop));if(typeof c=="object"||typeof c=="function")e?m[n]=f.extend(m[n],c):m[n].data=f.extend(m[n].data,c);g=h=m[n],e||(h.data||(h.data={}),h=h.data),d!==b&&(h[f.camelCase(c)]=d);if(o&&!h[c])return g.events;k?(i=h[c],i==null&&(i=h[f.camelCase(c)])):i=h;return i}},removeData:function(a,b,c){if(!!f.acceptData(a)){var d,e,g,h=f.expando,i=a.nodeType,j=i?f.cache:a,k=i?a[h]:h;if(!j[k])return;if(b){d=c?j[k]:j[k].data;if(d){f.isArray(b)||(b in d?b=[b]:(b=f.camelCase(b),b in d?b=[b]:b=b.split(" ")));for(e=0,g=b.length;e-1)return!0;return!1},val:function(a){var c,d,e,g=this[0];{if(!!arguments.length){e=f.isFunction(a);return this.each(function(d){var g=f(this),h;if(this.nodeType===1){e?h=a.call(this,d,g.val()):h=a,h==null?h="":typeof h=="number"?h+="":f.isArray(h)&&(h=f.map(h,function(a){return a==null?"":a+""})),c=f.valHooks[this.nodeName.toLowerCase()]||f.valHooks[this.type];if(!c||!("set"in c)||c.set(this,h,"value")===b)this.value=h}})}if(g){c=f.valHooks[g.nodeName.toLowerCase()]||f.valHooks[g.type];if(c&&"get"in c&&(d=c.get(g,"value"))!==b)return d;d=g.value;return typeof d=="string"?d.replace(q,""):d==null?"":d}}}}),f.extend({valHooks:{option:{get:function(a){var b=a.attributes.value;return!b||b.specified?a.value:a.text}},select:{get:function(a){var b,c,d,e,g=a.selectedIndex,h=[],i=a.options,j=a.type==="select-one";if(g<0)return null;c=j?g:0,d=j?g+1:i.length;for(;c=0}),c.length||(a.selectedIndex=-1);return c}}},attrFn:{val:!0,css:!0,html:!0,text:!0,data:!0,width:!0,height:!0,offset:!0},attr:function(a,c,d,e){var g,h,i,j=a.nodeType;if(!!a&&j!==3&&j!==8&&j!==2){if(e&&c in f.attrFn)return f(a)[c](d);if(typeof a.getAttribute=="undefined")return f.prop(a,c,d);i=j!==1||!f.isXMLDoc(a),i&&(c=c.toLowerCase(),h=f.attrHooks[c]||(u.test(c)?x:w));if(d!==b){if(d===null){f.removeAttr(a,c);return}if(h&&"set"in h&&i&&(g=h.set(a,d,c))!==b)return g;a.setAttribute(c,""+d);return d}if(h&&"get"in h&&i&&(g=h.get(a,c))!==null)return g;g=a.getAttribute(c);return g===null?b:g}},removeAttr:function(a,b){var c,d,e,g,h=0;if(b&&a.nodeType===1){d=b.toLowerCase().split(p),g=d.length;for(;h=0}})});var z=/^(?:textarea|input|select)$/i,A=/^([^\.]*)?(?:\.(.+))?$/,B=/\bhover(\.\S+)?\b/,C=/^key/,D=/^(?:mouse|contextmenu)|click/,E=/^(?:focusinfocus|focusoutblur)$/,F=/^(\w*)(?:#([\w\-]+))?(?:\.([\w\-]+))?$/,G=function(a){var b=F.exec(a);b&&(b[1]=(b[1]||"").toLowerCase(),b[3]=b[3]&&new RegExp("(?:^|\\s)"+b[3]+"(?:\\s|$)"));return b},H=function(a,b){var c=a.attributes||{};return(!b[1]||a.nodeName.toLowerCase()===b[1])&&(!b[2]||(c.id||{}).value===b[2])&&(!b[3]||b[3].test((c["class"]||{}).value))},I=function(a){return f.event.special.hover?a:a.replace(B,"mouseenter$1 mouseleave$1")}; f.event={add:function(a,c,d,e,g){var h,i,j,k,l,m,n,o,p,q,r,s;if(!(a.nodeType===3||a.nodeType===8||!c||!d||!(h=f._data(a)))){d.handler&&(p=d,d=p.handler),d.guid||(d.guid=f.guid++),j=h.events,j||(h.events=j={}),i=h.handle,i||(h.handle=i=function(a){return typeof f!="undefined"&&(!a||f.event.triggered!==a.type)?f.event.dispatch.apply(i.elem,arguments):b},i.elem=a),c=f.trim(I(c)).split(" ");for(k=0;k=0&&(h=h.slice(0,-1),k=!0),h.indexOf(".")>=0&&(i=h.split("."),h=i.shift(),i.sort());if((!e||f.event.customEvent[h])&&!f.event.global[h])return;c=typeof c=="object"?c[f.expando]?c:new f.Event(h,c):new f.Event(h),c.type=h,c.isTrigger=!0,c.exclusive=k,c.namespace=i.join("."),c.namespace_re=c.namespace?new RegExp("(^|\\.)"+i.join("\\.(?:.*\\.)?")+"(\\.|$)"):null,o=h.indexOf(":")<0?"on"+h:"";if(!e){j=f.cache;for(l in j)j[l].events&&j[l].events[h]&&f.event.trigger(c,d,j[l].handle.elem,!0);return}c.result=b,c.target||(c.target=e),d=d!=null?f.makeArray(d):[],d.unshift(c),p=f.event.special[h]||{};if(p.trigger&&p.trigger.apply(e,d)===!1)return;r=[[e,p.bindType||h]];if(!g&&!p.noBubble&&!f.isWindow(e)){s=p.delegateType||h,m=E.test(s+h)?e:e.parentNode,n=null;for(;m;m=m.parentNode)r.push([m,s]),n=m;n&&n===e.ownerDocument&&r.push([n.defaultView||n.parentWindow||a,s])}for(l=0;le&&i.push({elem:this,matches:d.slice(e)});for(j=0;j0?this.on(b,null,a,c):this.trigger(b)},f.attrFn&&(f.attrFn[b]=!0),C.test(b)&&(f.event.fixHooks[b]=f.event.keyHooks),D.test(b)&&(f.event.fixHooks[b]=f.event.mouseHooks)}),function(){function x(a,b,c,e,f,g){for(var h=0,i=e.length;h0){k=j;break}}j=j[a]}e[h]=k}}}function w(a,b,c,e,f,g){for(var h=0,i=e.length;h+~,(\[\\]+)+|[>+~])(\s*,\s*)?((?:.|\r|\n)*)/g,d="sizcache"+(Math.random()+"").replace(".",""),e=0,g=Object.prototype.toString,h=!1,i=!0,j=/\\/g,k=/\r\n/g,l=/\W/;[0,0].sort(function(){i=!1;return 0});var m=function(b,d,e,f){e=e||[],d=d||c;var h=d;if(d.nodeType!==1&&d.nodeType!==9)return[];if(!b||typeof b!="string")return e;var i,j,k,l,n,q,r,t,u=!0,v=m.isXML(d),w=[],x=b;do{a.exec(""),i=a.exec(x);if(i){x=i[3],w.push(i[1]);if(i[2]){l=i[3];break}}}while(i);if(w.length>1&&p.exec(b))if(w.length===2&&o.relative[w[0]])j=y(w[0]+w[1],d,f);else{j=o.relative[w[0]]?[d]:m(w.shift(),d);while(w.length)b=w.shift(),o.relative[b]&&(b+=w.shift()),j=y(b,j,f)}else{!f&&w.length>1&&d.nodeType===9&&!v&&o.match.ID.test(w[0])&&!o.match.ID.test(w[w.length-1])&&(n=m.find(w.shift(),d,v),d=n.expr?m.filter(n.expr,n.set)[0]:n.set[0]);if(d){n=f?{expr:w.pop(),set:s(f)}:m.find(w.pop(),w.length===1&&(w[0]==="~"||w[0]==="+")&&d.parentNode?d.parentNode:d,v),j=n.expr?m.filter(n.expr,n.set):n.set,w.length>0?k=s(j):u=!1;while(w.length)q=w.pop(),r=q,o.relative[q]?r=w.pop():q="",r==null&&(r=d),o.relative[q](k,r,v)}else k=w=[]}k||(k=j),k||m.error(q||b);if(g.call(k)==="[object Array]")if(!u)e.push.apply(e,k);else if(d&&d.nodeType===1)for(t=0;k[t]!=null;t++)k[t]&&(k[t]===!0||k[t].nodeType===1&&m.contains(d,k[t]))&&e.push(j[t]);else for(t=0;k[t]!=null;t++)k[t]&&k[t].nodeType===1&&e.push(j[t]);else s(k,e);l&&(m(l,h,e,f),m.uniqueSort(e));return e};m.uniqueSort=function(a){if(u){h=i,a.sort(u);if(h)for(var b=1;b0},m.find=function(a,b,c){var d,e,f,g,h,i;if(!a)return[];for(e=0,f=o.order.length;e":function(a,b){var c,d=typeof b=="string",e=0,f=a.length;if(d&&!l.test(b)){b=b.toLowerCase();for(;e=0)?c||d.push(h):c&&(b[g]=!1));return!1},ID:function(a){return a[1].replace(j,"")},TAG:function(a,b){return a[1].replace(j,"").toLowerCase()},CHILD:function(a){if(a[1]==="nth"){a[2]||m.error(a[0]),a[2]=a[2].replace(/^\+|\s*/g,"");var b=/(-?)(\d*)(?:n([+\-]?\d*))?/.exec(a[2]==="even"&&"2n"||a[2]==="odd"&&"2n+1"||!/\D/.test(a[2])&&"0n+"+a[2]||a[2]);a[2]=b[1]+(b[2]||1)-0,a[3]=b[3]-0}else a[2]&&m.error(a[0]);a[0]=e++;return a},ATTR:function(a,b,c,d,e,f){var g=a[1]=a[1].replace(j,"");!f&&o.attrMap[g]&&(a[1]=o.attrMap[g]),a[4]=(a[4]||a[5]||"").replace(j,""),a[2]==="~="&&(a[4]=" "+a[4]+" ");return a},PSEUDO:function(b,c,d,e,f){if(b[1]==="not")if((a.exec(b[3])||"").length>1||/^\w/.test(b[3]))b[3]=m(b[3],null,null,c);else{var g=m.filter(b[3],c,d,!0^f);d||e.push.apply(e,g);return!1}else if(o.match.POS.test(b[0])||o.match.CHILD.test(b[0]))return!0;return b},POS:function(a){a.unshift(!0);return a}},filters:{enabled:function(a){return a.disabled===!1&&a.type!=="hidden"},disabled:function(a){return a.disabled===!0},checked:function(a){return a.checked===!0},selected:function(a){a.parentNode&&a.parentNode.selectedIndex;return a.selected===!0},parent:function(a){return!!a.firstChild},empty:function(a){return!a.firstChild},has:function(a,b,c){return!!m(c[3],a).length},header:function(a){return/h\d/i.test(a.nodeName)},text:function(a){var b=a.getAttribute("type"),c=a.type;return a.nodeName.toLowerCase()==="input"&&"text"===c&&(b===c||b===null)},radio:function(a){return a.nodeName.toLowerCase()==="input"&&"radio"===a.type},checkbox:function(a){return a.nodeName.toLowerCase()==="input"&&"checkbox"===a.type},file:function(a){return a.nodeName.toLowerCase()==="input"&&"file"===a.type},password:function(a){return a.nodeName.toLowerCase()==="input"&&"password"===a.type},submit:function(a){var b=a.nodeName.toLowerCase();return(b==="input"||b==="button")&&"submit"===a.type},image:function(a){return a.nodeName.toLowerCase()==="input"&&"image"===a.type},reset:function(a){var b=a.nodeName.toLowerCase();return(b==="input"||b==="button")&&"reset"===a.type},button:function(a){var b=a.nodeName.toLowerCase();return b==="input"&&"button"===a.type||b==="button"},input:function(a){return/input|select|textarea|button/i.test(a.nodeName)},focus:function(a){return a===a.ownerDocument.activeElement}},setFilters:{first:function(a,b){return b===0},last:function(a,b,c,d){return b===d.length-1},even:function(a,b){return b%2===0},odd:function(a,b){return b%2===1},lt:function(a,b,c){return bc[3]-0},nth:function(a,b,c){return c[3]-0===b},eq:function(a,b,c){return c[3]-0===b}},filter:{PSEUDO:function(a,b,c,d){var e=b[1],f=o.filters[e];if(f)return f(a,c,b,d);if(e==="contains")return(a.textContent||a.innerText||n([a])||"").indexOf(b[3])>=0;if(e==="not"){var g=b[3];for(var h=0,i=g.length;h=0}},ID:function(a,b){return a.nodeType===1&&a.getAttribute("id")===b},TAG:function(a,b){return b==="*"&&a.nodeType===1||!!a.nodeName&&a.nodeName.toLowerCase()===b},CLASS:function(a,b){return(" "+(a.className||a.getAttribute("class"))+" ").indexOf(b)>-1},ATTR:function(a,b){var c=b[1],d=m.attr?m.attr(a,c):o.attrHandle[c]?o.attrHandle[c](a):a[c]!=null?a[c]:a.getAttribute(c),e=d+"",f=b[2],g=b[4];return d==null?f==="!=":!f&&m.attr?d!=null:f==="="?e===g:f==="*="?e.indexOf(g)>=0:f==="~="?(" "+e+" ").indexOf(g)>=0:g?f==="!="?e!==g:f==="^="?e.indexOf(g)===0:f==="$="?e.substr(e.length-g.length)===g:f==="|="?e===g||e.substr(0,g.length+1)===g+"-":!1:e&&d!==!1},POS:function(a,b,c,d){var e=b[2],f=o.setFilters[e];if(f)return f(a,c,b,d)}}},p=o.match.POS,q=function(a,b){return"\\"+(b-0+1)};for(var r in o.match)o.match[r]=new RegExp(o.match[r].source+/(?![^\[]*\])(?![^\(]*\))/.source),o.leftMatch[r]=new RegExp(/(^(?:.|\r|\n)*?)/.source+o.match[r].source.replace(/\\(\d+)/g,q));var s=function(a,b){a=Array.prototype.slice.call(a,0);if(b){b.push.apply(b,a);return b}return a};try{Array.prototype.slice.call(c.documentElement.childNodes,0)[0].nodeType}catch(t){s=function(a,b){var c=0,d=b||[];if(g.call(a)==="[object Array]")Array.prototype.push.apply(d,a);else if(typeof a.length=="number")for(var e=a.length;c",e.insertBefore(a,e.firstChild),c.getElementById(d)&&(o.find.ID=function(a,c,d){if(typeof c.getElementById!="undefined"&&!d){var e=c.getElementById(a[1]);return e?e.id===a[1]||typeof e.getAttributeNode!="undefined"&&e.getAttributeNode("id").nodeValue===a[1]?[e]:b:[]}},o.filter.ID=function(a,b){var c=typeof a.getAttributeNode!="undefined"&&a.getAttributeNode("id");return a.nodeType===1&&c&&c.nodeValue===b}),e.removeChild(a),e=a=null}(),function(){var a=c.createElement("div");a.appendChild(c.createComment("")),a.getElementsByTagName("*").length>0&&(o.find.TAG=function(a,b){var c=b.getElementsByTagName(a[1]);if(a[1]==="*"){var d=[];for(var e=0;c[e];e++)c[e].nodeType===1&&d.push(c[e]);c=d}return c}),a.innerHTML="",a.firstChild&&typeof a.firstChild.getAttribute!="undefined"&&a.firstChild.getAttribute("href")!=="#"&&(o.attrHandle.href=function(a){return a.getAttribute("href",2)}),a=null}(),c.querySelectorAll&&function(){var a=m,b=c.createElement("div"),d="__sizzle__";b.innerHTML="

";if(!b.querySelectorAll||b.querySelectorAll(".TEST").length!==0){m=function(b,e,f,g){e=e||c;if(!g&&!m.isXML(e)){var h=/^(\w+$)|^\.([\w\-]+$)|^#([\w\-]+$)/.exec(b);if(h&&(e.nodeType===1||e.nodeType===9)){if(h[1])return s(e.getElementsByTagName(b),f);if(h[2]&&o.find.CLASS&&e.getElementsByClassName)return s(e.getElementsByClassName(h[2]),f)}if(e.nodeType===9){if(b==="body"&&e.body)return s([e.body],f);if(h&&h[3]){var i=e.getElementById(h[3]);if(!i||!i.parentNode)return s([],f);if(i.id===h[3])return s([i],f)}try{return s(e.querySelectorAll(b),f)}catch(j){}}else if(e.nodeType===1&&e.nodeName.toLowerCase()!=="object"){var k=e,l=e.getAttribute("id"),n=l||d,p=e.parentNode,q=/^\s*[+~]/.test(b);l?n=n.replace(/'/g,"\$&"):e.setAttribute("id",n),q&&p&&(e=e.parentNode);try{if(!q||p)return s(e.querySelectorAll("[id='"+n+"'] "+b),f)}catch(r){}finally{l||k.removeAttribute("id")}}}return a(b,e,f,g)};for(var e in a)m[e]=a[e];b=null}}(),function(){var a=c.documentElement,b=a.matchesSelector||a.mozMatchesSelector||a.webkitMatchesSelector||a.msMatchesSelector;if(b){var d=!b.call(c.createElement("div"),"div"),e=!1;try{b.call(c.documentElement,"[test!='']:sizzle")}catch(f){e=!0}m.matchesSelector=function(a,c){c=c.replace(/\=\s*([^'"\]]*)\s*\]/g,"='$1']");if(!m.isXML(a))try{if(e||!o.match.PSEUDO.test(c)&&!/!=/.test(c)){var f=b.call(a,c);if(f||!d||a.document&&a.document.nodeType!==11)return f}}catch(g){}return m(c,null,null,[a]).length>0}}}(),function(){var a=c.createElement("div");a.innerHTML="
";if(!!a.getElementsByClassName&&a.getElementsByClassName("e").length!==0){a.lastChild.className="e";if(a.getElementsByClassName("e").length===1)return;o.order.splice(1,0,"CLASS"),o.find.CLASS=function(a,b,c){if(typeof b.getElementsByClassName!="undefined"&&!c)return b.getElementsByClassName(a[1])},a=null}}(),c.documentElement.contains?m.contains=function(a,b){return a!==b&&(a.contains?a.contains(b):!0)}:c.documentElement.compareDocumentPosition?m.contains=function(a,b){return!!(a.compareDocumentPosition(b)&16)}:m.contains=function(){return!1},m.isXML=function(a){var b=(a?a.ownerDocument||a:0).documentElement;return b?b.nodeName!=="HTML":!1};var y=function(a,b,c){var d,e=[],f="",g=b.nodeType?[b]:b;while(d=o.match.PSEUDO.exec(a))f+=d[0],a=a.replace(o.match.PSEUDO,"");a=o.relative[a]?a+"*":a;for(var h=0,i=g.length;h0)for(h=g;h=0:f.filter(a,this).length>0:this.filter(a).length>0)},closest:function(a,b){var c=[],d,e,g=this[0];if(f.isArray(a)){var h=1;while(g&&g.ownerDocument&&g!==b){for(d=0;d-1:f.find.matchesSelector(g,a)){c.push(g);break}g=g.parentNode;if(!g||!g.ownerDocument||g===b||g.nodeType===11)break}}c=c.length>1?f.unique(c):c;return this.pushStack(c,"closest",a)},index:function(a){if(!a)return this[0]&&this[0].parentNode?this.prevAll().length:-1;if(typeof a=="string")return f.inArray(this[0],f(a));return f.inArray(a.jquery?a[0]:a,this)},add:function(a,b){var c=typeof a=="string"?f(a,b):f.makeArray(a&&a.nodeType?[a]:a),d=f.merge(this.get(),c);return this.pushStack(S(c[0])||S(d[0])?d:f.unique(d))},andSelf:function(){return this.add(this.prevObject)}}),f.each({parent:function(a){var b=a.parentNode;return b&&b.nodeType!==11?b:null},parents:function(a){return f.dir(a,"parentNode")},parentsUntil:function(a,b,c){return f.dir(a,"parentNode",c)},next:function(a){return f.nth(a,2,"nextSibling")},prev:function(a){return f.nth(a,2,"previousSibling")},nextAll:function(a){return f.dir(a,"nextSibling")},prevAll:function(a){return f.dir(a,"previousSibling")},nextUntil:function(a,b,c){return f.dir(a,"nextSibling",c)},prevUntil:function(a,b,c){return f.dir(a,"previousSibling",c)},siblings:function(a){return f.sibling(a.parentNode.firstChild,a)},children:function(a){return f.sibling(a.firstChild)},contents:function(a){return f.nodeName(a,"iframe")?a.contentDocument||a.contentWindow.document:f.makeArray(a.childNodes)}},function(a,b){f.fn[a]=function(c,d){var e=f.map(this,b,c);L.test(a)||(d=c),d&&typeof d=="string"&&(e=f.filter(d,e)),e=this.length>1&&!R[a]?f.unique(e):e,(this.length>1||N.test(d))&&M.test(a)&&(e=e.reverse());return this.pushStack(e,a,P.call(arguments).join(","))}}),f.extend({filter:function(a,b,c){c&&(a=":not("+a+")");return b.length===1?f.find.matchesSelector(b[0],a)?[b[0]]:[]:f.find.matches(a,b)},dir:function(a,c,d){var e=[],g=a[c];while(g&&g.nodeType!==9&&(d===b||g.nodeType!==1||!f(g).is(d)))g.nodeType===1&&e.push(g),g=g[c];return e},nth:function(a,b,c,d){b=b||1;var e=0;for(;a;a=a[c])if(a.nodeType===1&&++e===b)break;return a},sibling:function(a,b){var c=[];for(;a;a=a.nextSibling)a.nodeType===1&&a!==b&&c.push(a);return c}});var V="abbr|article|aside|audio|canvas|datalist|details|figcaption|figure|footer|header|hgroup|mark|meter|nav|output|progress|section|summary|time|video",W=/ jQuery\d+="(?:\d+|null)"/g,X=/^\s+/,Y=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/ig,Z=/<([\w:]+)/,$=/",""],legend:[1,"
","
"],thead:[1,"","
"],tr:[2,"","
"],td:[3,"","
"],col:[2,"","
"],area:[1,"",""],_default:[0,"",""]},bh=U(c);bg.optgroup=bg.option,bg.tbody=bg.tfoot=bg.colgroup=bg.caption=bg.thead,bg.th=bg.td,f.support.htmlSerialize||(bg._default=[1,"div
","
"]),f.fn.extend({text:function(a){if(f.isFunction(a))return this.each(function(b){var c=f(this);c.text(a.call(this,b,c.text()))});if(typeof a!="object"&&a!==b)return this.empty().append((this[0]&&this[0].ownerDocument||c).createTextNode(a));return f.text(this)},wrapAll:function(a){if(f.isFunction(a))return this.each(function(b){f(this).wrapAll(a.call(this,b))});if(this[0]){var b=f(a,this[0].ownerDocument).eq(0).clone(!0);this[0].parentNode&&b.insertBefore(this[0]),b.map(function(){var a=this;while(a.firstChild&&a.firstChild.nodeType===1)a=a.firstChild;return a}).append(this)}return this},wrapInner:function(a){if(f.isFunction(a))return this.each(function(b){f(this).wrapInner(a.call(this,b))});return this.each(function(){var b=f(this),c=b.contents();c.length?c.wrapAll(a):b.append(a)})},wrap:function(a){var b=f.isFunction(a);return this.each(function(c){f(this).wrapAll(b?a.call(this,c):a)})},unwrap:function(){return this.parent().each(function(){f.nodeName(this,"body")||f(this).replaceWith(this.childNodes)}).end()},append:function(){return this.domManip(arguments,!0,function(a){this.nodeType===1&&this.appendChild(a)})},prepend:function(){return this.domManip(arguments,!0,function(a){this.nodeType===1&&this.insertBefore(a,this.firstChild)})},before:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this)});if(arguments.length){var a=f.clean(arguments);a.push.apply(a,this.toArray());return this.pushStack(a,"before",arguments)}},after:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,!1,function(a){this.parentNode.insertBefore(a,this.nextSibling)});if(arguments.length){var a=this.pushStack(this,"after",arguments);a.push.apply(a,f.clean(arguments));return a}},remove:function(a,b){for(var c=0,d;(d=this[c])!=null;c++)if(!a||f.filter(a,[d]).length)!b&&d.nodeType===1&&(f.cleanData(d.getElementsByTagName("*")),f.cleanData([d])),d.parentNode&&d.parentNode.removeChild(d);return this},empty:function() {for(var a=0,b;(b=this[a])!=null;a++){b.nodeType===1&&f.cleanData(b.getElementsByTagName("*"));while(b.firstChild)b.removeChild(b.firstChild)}return this},clone:function(a,b){a=a==null?!1:a,b=b==null?a:b;return this.map(function(){return f.clone(this,a,b)})},html:function(a){if(a===b)return this[0]&&this[0].nodeType===1?this[0].innerHTML.replace(W,""):null;if(typeof a=="string"&&!ba.test(a)&&(f.support.leadingWhitespace||!X.test(a))&&!bg[(Z.exec(a)||["",""])[1].toLowerCase()]){a=a.replace(Y,"<$1>");try{for(var c=0,d=this.length;c1&&l0?this.clone(!0):this).get();f(e[h])[b](j),d=d.concat(j)}return this.pushStack(d,a,e.selector)}}),f.extend({clone:function(a,b,c){var d,e,g,h=f.support.html5Clone||!bc.test("<"+a.nodeName)?a.cloneNode(!0):bo(a);if((!f.support.noCloneEvent||!f.support.noCloneChecked)&&(a.nodeType===1||a.nodeType===11)&&!f.isXMLDoc(a)){bk(a,h),d=bl(a),e=bl(h);for(g=0;d[g];++g)e[g]&&bk(d[g],e[g])}if(b){bj(a,h);if(c){d=bl(a),e=bl(h);for(g=0;d[g];++g)bj(d[g],e[g])}}d=e=null;return h},clean:function(a,b,d,e){var g;b=b||c,typeof b.createElement=="undefined"&&(b=b.ownerDocument||b[0]&&b[0].ownerDocument||c);var h=[],i;for(var j=0,k;(k=a[j])!=null;j++){typeof k=="number"&&(k+="");if(!k)continue;if(typeof k=="string")if(!_.test(k))k=b.createTextNode(k);else{k=k.replace(Y,"<$1>");var l=(Z.exec(k)||["",""])[1].toLowerCase(),m=bg[l]||bg._default,n=m[0],o=b.createElement("div");b===c?bh.appendChild(o):U(b).appendChild(o),o.innerHTML=m[1]+k+m[2];while(n--)o=o.lastChild;if(!f.support.tbody){var p=$.test(k),q=l==="table"&&!p?o.firstChild&&o.firstChild.childNodes:m[1]===""&&!p?o.childNodes:[];for(i=q.length-1;i>=0;--i)f.nodeName(q[i],"tbody")&&!q[i].childNodes.length&&q[i].parentNode.removeChild(q[i])}!f.support.leadingWhitespace&&X.test(k)&&o.insertBefore(b.createTextNode(X.exec(k)[0]),o.firstChild),k=o.childNodes}var r;if(!f.support.appendChecked)if(k[0]&&typeof (r=k.length)=="number")for(i=0;i=0)return b+"px"}}}),f.support.opacity||(f.cssHooks.opacity={get:function(a,b){return br.test((b&&a.currentStyle?a.currentStyle.filter:a.style.filter)||"")?parseFloat(RegExp.$1)/100+"":b?"1":""},set:function(a,b){var c=a.style,d=a.currentStyle,e=f.isNumeric(b)?"alpha(opacity="+b*100+")":"",g=d&&d.filter||c.filter||"";c.zoom=1;if(b>=1&&f.trim(g.replace(bq,""))===""){c.removeAttribute("filter");if(d&&!d.filter)return}c.filter=bq.test(g)?g.replace(bq,e):g+" "+e}}),f(function(){f.support.reliableMarginRight||(f.cssHooks.marginRight={get:function(a,b){var c;f.swap(a,{display:"inline-block"},function(){b?c=bz(a,"margin-right","marginRight"):c=a.style.marginRight});return c}})}),c.defaultView&&c.defaultView.getComputedStyle&&(bA=function(a,b){var c,d,e;b=b.replace(bs,"-$1").toLowerCase(),(d=a.ownerDocument.defaultView)&&(e=d.getComputedStyle(a,null))&&(c=e.getPropertyValue(b),c===""&&!f.contains(a.ownerDocument.documentElement,a)&&(c=f.style(a,b)));return c}),c.documentElement.currentStyle&&(bB=function(a,b){var c,d,e,f=a.currentStyle&&a.currentStyle[b],g=a.style;f===null&&g&&(e=g[b])&&(f=e),!bt.test(f)&&bu.test(f)&&(c=g.left,d=a.runtimeStyle&&a.runtimeStyle.left,d&&(a.runtimeStyle.left=a.currentStyle.left),g.left=b==="fontSize"?"1em":f||0,f=g.pixelLeft+"px",g.left=c,d&&(a.runtimeStyle.left=d));return f===""?"auto":f}),bz=bA||bB,f.expr&&f.expr.filters&&(f.expr.filters.hidden=function(a){var b=a.offsetWidth,c=a.offsetHeight;return b===0&&c===0||!f.support.reliableHiddenOffsets&&(a.style&&a.style.display||f.css(a,"display"))==="none"},f.expr.filters.visible=function(a){return!f.expr.filters.hidden(a)});var bD=/%20/g,bE=/\[\]$/,bF=/\r?\n/g,bG=/#.*$/,bH=/^(.*?):[ \t]*([^\r\n]*)\r?$/mg,bI=/^(?:color|date|datetime|datetime-local|email|hidden|month|number|password|range|search|tel|text|time|url|week)$/i,bJ=/^(?:about|app|app\-storage|.+\-extension|file|res|widget):$/,bK=/^(?:GET|HEAD)$/,bL=/^\/\//,bM=/\?/,bN=/)<[^<]*)*<\/script>/gi,bO=/^(?:select|textarea)/i,bP=/\s+/,bQ=/([?&])_=[^&]*/,bR=/^([\w\+\.\-]+:)(?:\/\/([^\/?#:]*)(?::(\d+))?)?/,bS=f.fn.load,bT={},bU={},bV,bW,bX=["*/"]+["*"];try{bV=e.href}catch(bY){bV=c.createElement("a"),bV.href="",bV=bV.href}bW=bR.exec(bV.toLowerCase())||[],f.fn.extend({load:function(a,c,d){if(typeof a!="string"&&bS)return bS.apply(this,arguments);if(!this.length)return this;var e=a.indexOf(" ");if(e>=0){var g=a.slice(e,a.length);a=a.slice(0,e)}var h="GET";c&&(f.isFunction(c)?(d=c,c=b):typeof c=="object"&&(c=f.param(c,f.ajaxSettings.traditional),h="POST"));var i=this;f.ajax({url:a,type:h,dataType:"html",data:c,complete:function(a,b,c){c=a.responseText,a.isResolved()&&(a.done(function(a){c=a}),i.html(g?f("
").append(c.replace(bN,"")).find(g):c)),d&&i.each(d,[c,b,a])}});return this},serialize:function(){return f.param(this.serializeArray())},serializeArray:function(){return this.map(function(){return this.elements?f.makeArray(this.elements):this}).filter(function(){return this.name&&!this.disabled&&(this.checked||bO.test(this.nodeName)||bI.test(this.type))}).map(function(a,b){var c=f(this).val();return c==null?null:f.isArray(c)?f.map(c,function(a,c){return{name:b.name,value:a.replace(bF,"\r\n")}}):{name:b.name,value:c.replace(bF,"\r\n")}}).get()}}),f.each("ajaxStart ajaxStop ajaxComplete ajaxError ajaxSuccess ajaxSend".split(" "),function(a,b){f.fn[b]=function(a){return this.on(b,a)}}),f.each(["get","post"],function(a,c){f[c]=function(a,d,e,g){f.isFunction(d)&&(g=g||e,e=d,d=b);return f.ajax({type:c,url:a,data:d,success:e,dataType:g})}}),f.extend({getScript:function(a,c){return f.get(a,b,c,"script")},getJSON:function(a,b,c){return f.get(a,b,c,"json")},ajaxSetup:function(a,b){b?b_(a,f.ajaxSettings):(b=a,a=f.ajaxSettings),b_(a,b);return a},ajaxSettings:{url:bV,isLocal:bJ.test(bW[1]),global:!0,type:"GET",contentType:"application/x-www-form-urlencoded",processData:!0,async:!0,accepts:{xml:"application/xml, text/xml",html:"text/html",text:"text/plain",json:"application/json, text/javascript","*":bX},contents:{xml:/xml/,html:/html/,json:/json/},responseFields:{xml:"responseXML",text:"responseText"},converters:{"* text":a.String,"text html":!0,"text json":f.parseJSON,"text xml":f.parseXML},flatOptions:{context:!0,url:!0}},ajaxPrefilter:bZ(bT),ajaxTransport:bZ(bU),ajax:function(a,c){function w(a,c,l,m){if(s!==2){s=2,q&&clearTimeout(q),p=b,n=m||"",v.readyState=a>0?4:0;var o,r,u,w=c,x=l?cb(d,v,l):b,y,z;if(a>=200&&a<300||a===304){if(d.ifModified){if(y=v.getResponseHeader("Last-Modified"))f.lastModified[k]=y;if(z=v.getResponseHeader("Etag"))f.etag[k]=z}if(a===304)w="notmodified",o=!0;else try{r=cc(d,x),w="success",o=!0}catch(A){w="parsererror",u=A}}else{u=w;if(!w||a)w="error",a<0&&(a=0)}v.status=a,v.statusText=""+(c||w),o?h.resolveWith(e,[r,w,v]):h.rejectWith(e,[v,w,u]),v.statusCode(j),j=b,t&&g.trigger("ajax"+(o?"Success":"Error"),[v,d,o?r:u]),i.fireWith(e,[v,w]),t&&(g.trigger("ajaxComplete",[v,d]),--f.active||f.event.trigger("ajaxStop"))}}typeof a=="object"&&(c=a,a=b),c=c||{};var d=f.ajaxSetup({},c),e=d.context||d,g=e!==d&&(e.nodeType||e instanceof f)?f(e):f.event,h=f.Deferred(),i=f.Callbacks("once memory"),j=d.statusCode||{},k,l={},m={},n,o,p,q,r,s=0,t,u,v={readyState:0,setRequestHeader:function(a,b){if(!s){var c=a.toLowerCase();a=m[c]=m[c]||a,l[a]=b}return this},getAllResponseHeaders:function(){return s===2?n:null},getResponseHeader:function(a){var c;if(s===2){if(!o){o={};while(c=bH.exec(n))o[c[1].toLowerCase()]=c[2]}c=o[a.toLowerCase()]}return c===b?null:c},overrideMimeType:function(a){s||(d.mimeType=a);return this},abort:function(a){a=a||"abort",p&&p.abort(a),w(0,a);return this}};h.promise(v),v.success=v.done,v.error=v.fail,v.complete=i.add,v.statusCode=function(a){if(a){var b;if(s<2)for(b in a)j[b]=[j[b],a[b]];else b=a[v.status],v.then(b,b)}return this},d.url=((a||d.url)+"").replace(bG,"").replace(bL,bW[1]+"//"),d.dataTypes=f.trim(d.dataType||"*").toLowerCase().split(bP),d.crossDomain==null&&(r=bR.exec(d.url.toLowerCase()),d.crossDomain=!(!r||r[1]==bW[1]&&r[2]==bW[2]&&(r[3]||(r[1]==="http:"?80:443))==(bW[3]||(bW[1]==="http:"?80:443)))),d.data&&d.processData&&typeof d.data!="string"&&(d.data=f.param(d.data,d.traditional)),b$(bT,d,c,v);if(s===2)return!1;t=d.global,d.type=d.type.toUpperCase(),d.hasContent=!bK.test(d.type),t&&f.active++===0&&f.event.trigger("ajaxStart");if(!d.hasContent){d.data&&(d.url+=(bM.test(d.url)?"&":"?")+d.data,delete d.data),k=d.url;if(d.cache===!1){var x=f.now(),y=d.url.replace(bQ,"$1_="+x);d.url=y+(y===d.url?(bM.test(d.url)?"&":"?")+"_="+x:"")}}(d.data&&d.hasContent&&d.contentType!==!1||c.contentType)&&v.setRequestHeader("Content-Type",d.contentType),d.ifModified&&(k=k||d.url,f.lastModified[k]&&v.setRequestHeader("If-Modified-Since",f.lastModified[k]),f.etag[k]&&v.setRequestHeader("If-None-Match",f.etag[k])),v.setRequestHeader("Accept",d.dataTypes[0]&&d.accepts[d.dataTypes[0]]?d.accepts[d.dataTypes[0]]+(d.dataTypes[0]!=="*"?", "+bX+"; q=0.01":""):d.accepts["*"]);for(u in d.headers)v.setRequestHeader(u,d.headers[u]);if(d.beforeSend&&(d.beforeSend.call(e,v,d)===!1||s===2)){v.abort();return!1}for(u in{success:1,error:1,complete:1})v[u](d[u]);p=b$(bU,d,c,v);if(!p)w(-1,"No Transport");else{v.readyState=1,t&&g.trigger("ajaxSend",[v,d]),d.async&&d.timeout>0&&(q=setTimeout(function(){v.abort("timeout")},d.timeout));try{s=1,p.send(l,w)}catch(z){if(s<2)w(-1,z);else throw z}}return v},param:function(a,c){var d=[],e=function(a,b){b=f.isFunction(b)?b():b,d[d.length]=encodeURIComponent(a)+"="+encodeURIComponent(b)};c===b&&(c=f.ajaxSettings.traditional);if(f.isArray(a)||a.jquery&&!f.isPlainObject(a))f.each(a,function(){e(this.name,this.value)});else for(var g in a)ca(g,a[g],c,e);return d.join("&").replace(bD,"+")}}),f.extend({active:0,lastModified:{},etag:{}});var cd=f.now(),ce=/(\=)\?(&|$)|\?\?/i;f.ajaxSetup({jsonp:"callback",jsonpCallback:function(){return f.expando+"_"+cd++}}),f.ajaxPrefilter("json jsonp",function(b,c,d){var e=b.contentType==="application/x-www-form-urlencoded"&&typeof b.data=="string";if(b.dataTypes[0]==="jsonp"||b.jsonp!==!1&&(ce.test(b.url)||e&&ce.test(b.data))){var g,h=b.jsonpCallback=f.isFunction(b.jsonpCallback)?b.jsonpCallback():b.jsonpCallback,i=a[h],j=b.url,k=b.data,l="$1"+h+"$2";b.jsonp!==!1&&(j=j.replace(ce,l),b.url===j&&(e&&(k=k.replace(ce,l)),b.data===k&&(j+=(/\?/.test(j)?"&":"?")+b.jsonp+"="+h))),b.url=j,b.data=k,a[h]=function(a){g=[a]},d.always(function(){a[h]=i,g&&f.isFunction(i)&&a[h](g[0])}),b.converters["script json"]=function(){g||f.error(h+" was not called");return g[0]},b.dataTypes[0]="json";return"script"}}),f.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/javascript|ecmascript/},converters:{"text script":function(a){f.globalEval(a);return a}}}),f.ajaxPrefilter("script",function(a){a.cache===b&&(a.cache=!1),a.crossDomain&&(a.type="GET",a.global=!1)}),f.ajaxTransport("script",function(a){if(a.crossDomain){var d,e=c.head||c.getElementsByTagName("head")[0]||c.documentElement;return{send:function(f,g){d=c.createElement("script"),d.async="async",a.scriptCharset&&(d.charset=a.scriptCharset),d.src=a.url,d.onload=d.onreadystatechange=function(a,c){if(c||!d.readyState||/loaded|complete/.test(d.readyState))d.onload=d.onreadystatechange=null,e&&d.parentNode&&e.removeChild(d),d=b,c||g(200,"success")},e.insertBefore(d,e.firstChild)},abort:function(){d&&d.onload(0,1)}}}});var cf=a.ActiveXObject?function(){for(var a in ch)ch[a](0,1)}:!1,cg=0,ch;f.ajaxSettings.xhr=a.ActiveXObject?function(){return!this.isLocal&&ci()||cj()}:ci,function(a){f.extend(f.support,{ajax:!!a,cors:!!a&&"withCredentials"in a})}(f.ajaxSettings.xhr()),f.support.ajax&&f.ajaxTransport(function(c){if(!c.crossDomain||f.support.cors){var d;return{send:function(e,g){var h=c.xhr(),i,j;c.username?h.open(c.type,c.url,c.async,c.username,c.password):h.open(c.type,c.url,c.async);if(c.xhrFields)for(j in c.xhrFields)h[j]=c.xhrFields[j];c.mimeType&&h.overrideMimeType&&h.overrideMimeType(c.mimeType),!c.crossDomain&&!e["X-Requested-With"]&&(e["X-Requested-With"]="XMLHttpRequest");try{for(j in e)h.setRequestHeader(j,e[j])}catch(k){}h.send(c.hasContent&&c.data||null),d=function(a,e){var j,k,l,m,n;try{if(d&&(e||h.readyState===4)){d=b,i&&(h.onreadystatechange=f.noop,cf&&delete ch[i]);if(e)h.readyState!==4&&h.abort();else{j=h.status,l=h.getAllResponseHeaders(),m={},n=h.responseXML,n&&n.documentElement&&(m.xml=n),m.text=h.responseText;try{k=h.statusText}catch(o){k=""}!j&&c.isLocal&&!c.crossDomain?j=m.text?200:404:j===1223&&(j=204)}}}catch(p){e||g(-1,p)}m&&g(j,k,m,l)},!c.async||h.readyState===4?d():(i=++cg,cf&&(ch||(ch={},f(a).unload(cf)),ch[i]=d),h.onreadystatechange=d)},abort:function(){d&&d(0,1)}}}});var ck={},cl,cm,cn=/^(?:toggle|show|hide)$/,co=/^([+\-]=)?([\d+.\-]+)([a-z%]*)$/i,cp,cq=[["height","marginTop","marginBottom","paddingTop","paddingBottom"],["width","marginLeft","marginRight","paddingLeft","paddingRight"],["opacity"]],cr;f.fn.extend({show:function(a,b,c){var d,e;if(a||a===0)return this.animate(cu("show",3),a,b,c);for(var g=0,h=this.length;g=i.duration+this.startTime){this.now=this.end,this.pos=this.state=1,this.update(),i.animatedProperties[this.prop]=!0;for(b in i.animatedProperties)i.animatedProperties[b]!==!0&&(g=!1);if(g){i.overflow!=null&&!f.support.shrinkWrapBlocks&&f.each(["","X","Y"],function(a,b){h.style["overflow"+b]=i.overflow[a]}),i.hide&&f(h).hide();if(i.hide||i.show)for(b in i.animatedProperties)f.style(h,b,i.orig[b]),f.removeData(h,"fxshow"+b,!0),f.removeData(h,"toggle"+b,!0);d=i.complete,d&&(i.complete=!1,d.call(h))}return!1}i.duration==Infinity?this.now=e:(c=e-this.startTime,this.state=c/i.duration,this.pos=f.easing[i.animatedProperties[this.prop]](this.state,c,0,1,i.duration),this.now=this.start+(this.end-this.start)*this.pos),this.update();return!0}},f.extend(f.fx,{tick:function(){var a,b=f.timers,c=0;for(;c-1,k={},l={},m,n;j?(l=e.position(),m=l.top,n=l.left):(m=parseFloat(h)||0,n=parseFloat(i)||0),f.isFunction(b)&&(b=b.call(a,c,g)),b.top!=null&&(k.top=b.top-g.top+m),b.left!=null&&(k.left=b.left-g.left+n),"using"in b?b.using.call(a,k):e.css(k)}},f.fn.extend({position:function(){if(!this[0])return null;var a=this[0],b=this.offsetParent(),c=this.offset(),d=cx.test(b[0].nodeName)?{top:0,left:0}:b.offset();c.top-=parseFloat(f.css(a,"marginTop"))||0,c.left-=parseFloat(f.css(a,"marginLeft"))||0,d.top+=parseFloat(f.css(b[0],"borderTopWidth"))||0,d.left+=parseFloat(f.css(b[0],"borderLeftWidth"))||0;return{top:c.top-d.top,left:c.left-d.left}},offsetParent:function(){return this.map(function(){var a=this.offsetParent||c.body;while(a&&!cx.test(a.nodeName)&&f.css(a,"position")==="static")a=a.offsetParent;return a})}}),f.each(["Left","Top"],function(a,c){var d="scroll"+c;f.fn[d]=function(c){var e,g;if(c===b){e=this[0];if(!e)return null;g=cy(e);return g?"pageXOffset"in g?g[a?"pageYOffset":"pageXOffset"]:f.support.boxModel&&g.document.documentElement[d]||g.document.body[d]:e[d]}return this.each(function(){g=cy(this),g?g.scrollTo(a?f(g).scrollLeft():c,a?c:f(g).scrollTop()):this[d]=c})}}),f.each(["Height","Width"],function(a,c){var d=c.toLowerCase();f.fn["inner"+c]=function(){var a=this[0];return a?a.style?parseFloat(f.css(a,d,"padding")):this[d]():null},f.fn["outer"+c]=function(a){var b=this[0];return b?b.style?parseFloat(f.css(b,d,a?"margin":"border")):this[d]():null},f.fn[d]=function(a){var e=this[0];if(!e)return a==null?null:this;if(f.isFunction(a))return this.each(function(b){var c=f(this);c[d](a.call(this,b,c[d]()))});if(f.isWindow(e)){var g=e.document.documentElement["client"+c],h=e.document.body;return e.document.compatMode==="CSS1Compat"&&g||h&&h["client"+c]||g}if(e.nodeType===9)return Math.max(e.documentElement["client"+c],e.body["scroll"+c],e.documentElement["scroll"+c],e.body["offset"+c],e.documentElement["offset"+c]);if(a===b){var i=f.css(e,d),j=parseFloat(i);return f.isNumeric(j)?j:i}return this.css(d,typeof a=="string"?a:a+"px")}}),a.jQuery=a.$=f,typeof define=="function"&&define.amd&&define.amd.jQuery&&define("jquery",[],function(){return f})})(window); var jq = jQuery.noConflict(); // Reclaim the $() function from jQuery /* * jQuery doTimeout: Like setTimeout, but better! - v1.0 - 3/3/2010 * http://benalman.com/projects/jquery-dotimeout-plugin/ * * Copyright (c) 2010 "Cowboy" Ben Alman * Dual licensed under the MIT and GPL licenses. * http://benalman.com/about/license/ */ (function($){var a={},c="doTimeout",d=Array.prototype.slice;$[c]=function(){return b.apply(window,[0].concat(d.call(arguments)))};$.fn[c]=function(){var f=d.call(arguments),e=b.apply(this,[c+f[0]].concat(f));return typeof f[0]==="number"||typeof f[1]==="number"?this:e};function b(l){var m=this,h,k={},g=l?$.fn:$,n=arguments,i=4,f=n[1],j=n[2],p=n[3];if(typeof f!=="string"){i--;f=l=0;j=n[1];p=n[2]}if(l){h=m.eq(0);h.data(l,k=h.data(l)||{})}else{if(f){k=a[f]||(a[f]={})}}k.id&&clearTimeout(k.id);delete k.id;function e(){if(l){h.removeData(l)}else{if(f){delete a[f]}}}function o(){k.id=setTimeout(function(){k.fn()},j)}if(p){k.fn=function(q){if(typeof p==="string"){p=g[p]}p.apply(m,d.call(n,i))===true&&!q?o():e()};o()}else{if(k.fn){j===undefined?e():k.fn(j===false);return true}else{e()}}}})(jQuery); /* Polyfill to avoid errors accessing the console object in older (IE) browsers. Part of https://html5boilerplate.com/ licensed under MIT license */ // Avoid `console` errors in browsers that lack a console. (function() { var method; var noop = function () {}; var methods = [ 'assert', 'clear', 'count', 'debug', 'dir', 'dirxml', 'error', 'exception', 'group', 'groupCollapsed', 'groupEnd', 'info', 'log', 'markTimeline', 'profile', 'profileEnd', 'table', 'time', 'timeEnd', 'timeline', 'timelineEnd', 'timeStamp', 'trace', 'warn' ]; var length = methods.length; var console = (window.console = window.console || {}); while (length--) { method = methods[length]; // Only stub undefined methods. if (!console[method]) { console[method] = noop; } } }()); // To help debugging
// Constants ///////////////////////////////////////////////// var SECOND_MS = 1000; // Milliseconds in a second var MINUTE_MS = 60 * SECOND_MS; // Milliseconds in a minute var HOUR_MS = 60 * MINUTE_MS; // Milliseconds in an hour var DAY_MS = 24 * HOUR_MS; // Milliseconds in a day var TO = "_to"; // _to part of an interval field var FROM = "_from"; // _from part of an interval field var KEY_Q = 81; // Q var ARROW_DOWN = 40; //arrow-down var FIELD_ID = "field_id"; var NAME = "name"; PartnerType=new Object(); PartnerType.BUYER="BY"; ///////////////////////////////////////////////// // Utility methods ///////////////////////////////////////////////// function $(fieldId, rowNumber) { if(rowNumber == null) return document.getElementById(fieldId); else return document.getElementById(fieldId + "_" + rowNumber); } /** If prameter is string, look up element with that ID, otherwise return parameter (=element) */ function element(elemOrId) { return (typeof elemOrId == 'string')? $(elemOrId) : elemOrId; } function findParentTag(node, tagName) { node = node.parentNode; while(node != null && node.tagName != tagName) node = node.parentNode; return node; } /** * Sets focus to field * Include fixes to work in all browsers */ function setFocus(field, select){ if(!field.readOnly && !field.disabled){ // readOnly and disabled fields should never be focused window.focusField = field; // Must use global variable for setTimeout to work //field.tabIndex = 0; // fix for focus() problem in Firefox if(field.id != "user_id" && field.id != "password") // Don't break password autocomplete field.setAttribute('autocomplete','off'); // fix for focus() problem in Firefox if(showTabWithField) // Ensure that this funtion is loaded showTabWithField(field); window.setTimeout('doSetFocus(' + select + ')',0); // Set focus to the element. setTimeout needed to avoid bugs in both IE and Firefox } } function doSetFocus(select) { try { if(window.focusField.focus) // Ensure that the focus method exists for the element window.focusField.focus(); if(window.focusField.select && // Ensure that the focus method exists for the element (select == null || select)) window.focusField.select(); } catch (e) { // Do nothing } } ///////////////////////////////////////////////// // Escaping data for URLS ///////////////////////////////////////////////// /** * Escape all data with ISO-8859-1 plus escapes '+' to '%2B' * If encodeURI() instead, change resin.conf + TextUtils.ENCODING */ function urlEncodeData(data) { var output = new String( escape(data) ); var re = new RegExp('\\+', 'g'); return output.replace(re,'%2B'); // Replace + with %2B } ///////////////////////////////////////////////// // Trimming strings ///////////////////////////////////////////////// /** * Trims a string * @param str The string to trim * @return The trimmed string */ function trimString (str) { str = this != window? this : str; return str.replace(/^\s+/g, '').replace(/\s+$/g, ''); } // Make trimString the trim method of strings String.prototype.trim = trimString; /** * Trim the contents of a field. * @param theField The field to trim */ function trimField(theField) { theField.value = theField.value.trim(); return theField.value; } String.prototype.startsWith = function(s) { return this.indexOf(s) == 0; }; ///////////////////////////////////////////////// // Replacing variables ///////////////////////////////////////////////// /** * Replace variables in a string */ String.prototype.replaceVar = function(variable, value) { var re = new RegExp("{" + variable + "}","g"); return this.replace(re , value); } /** * Replace indexed (numeric) variables in a string */ String.prototype.replaceIndex = function(variable, value) { var re = new RegExp("\\{" + variable + "\\}","g"); return this.replace(re , value); } ///////////////////////////////////////////////// // Filter digits ///////////////////////////////////////////////// function filterDigits(s) { return s.replace(/[^0-9]/g,''); // Remove all non digit characters } function filterLetterOrDigit(s) { return s.replace(/[^0-9a-zA-Z]/g,''); // Remove all non digit characters } /** * Replace commas with dots */ function replaceDecimalChar(s){ return s.replace(/,/g , "."); } ///////////////////////////////////////////////// // Filter starting zeroes ///////////////////////////////////////////////// /** * Delete starting zeroes in string * @param str The string * @return The trimmed string */ function deleteStartingZeroes(str) { return str.replace(/^0+/, ''); } ///////////////////////////////////////////////// // Array methods ///////////////////////////////////////////////// Array.prototype.contains = function (element) { for(var i = 0; i < this.length; i++) { if(this[i] == element) return true; } return false; }; Array.prototype.remove = function (element) { for(var i = 0; i < this.length; ) { if(this[i] == element) { this.splice(i, 1); } else i++; } }; ///////////////////////////////////////////////// // Utility functions for dates ///////////////////////////////////////////////// /** A regular expression that matches dates */ var dateFormatRegExp = new RegExp("[0-9]{2,4}-?[0-9]{2}-?[0-9]{2}"); var dateExtractRegExp = new RegExp("([0-9]{2,4})-?([0-9]{2})-?([0-9]{2})"); /** DD-MM-CCYY */ var danishDateFormatRegExp = new RegExp("[0-9]{2}[-.]?[0-9]{2}[-.]?[0-9]{2,4}"); var danishDateExtractRegExp = new RegExp("([0-9]{2})[-.]?([0-9]{2})[-.]?([0-9]{2,4})"); function stringToDate(dateString) { // Make sure not blank if(dateString == null || dateString == "" ) // If empty return null; // No date // Make sure the format is correct CCYYMMDD or CCYY-MM-DD if( dateString == dateString.match(dateFormatRegExp) ) { // Lastly check to make sure the date itself is valid // Extract year, month and day. Create a date object from that. dateExtractRegExp.exec(dateString); var matchedYear = RegExp.$1; if(matchedYear.length == 2) matchedYear = "20" + matchedYear; var createdDate = new Date(matchedYear, RegExp.$2 - 1, RegExp.$3) // See if the date in the object matches the string // (to make sure the string contained a valid date) var validYear = createdDate.getFullYear(); // Date.getYear() gets you in trouble with years like 9999 if( validYear < 1000 ) validYear += 1900; if( matchedYear != validYear || RegExp.$2 != (createdDate.getMonth()+1) || RegExp.$3 != createdDate.getDate() ) return null; else return createdDate; } return null; // Not matching allowed dates } Date.prototype.toString = function() { var year = this.getFullYear(); var month = this.getMonth() + 1; var date = this.getDate(); if("DA" == "SV" || "NO" == "SV") { return (date < 10 ? "0" : "") + date + "-" + (month < 10 ? "0" : "") + month + "-" + year; } else return year + "-" + (month < 10 ? "0" : "") + month + "-" + (date < 10 ? "0" : "") + date; } Date.prototype.equals = function(that) { return that != null && this.getTime() == that.getTime(); } /** Get a Date object representing today, without any time information */ function today() { var now = new Date(); return new Date(now.getFullYear(), now.getMonth(), now.getDate()); } /** A regular expression that matches dates */ var timeFormatRegExp = new RegExp("[0-9]{2}:?[0-9]{2}:?[0-9]{0,2}"); var timeExtractRegExp = new RegExp("([0-9]{2}):?([0-9]{2}):?([0-9]{0,2})"); function stringToTime(timeString) { // Make sure not blank if( timeString == "" ) // If empty return null; // No date // Make sure the format is correct CCYYMMDD or CCYY-MM-DD if( timeString !=timeString.match(timeFormatRegExp) ) return null; // Lastly check to make sure the date itself is valid // Extract year, month and day. Create a date object from that. timeExtractRegExp.exec(timeString); var createdTime = new Date(2001, 1 , 1, RegExp.$1, RegExp.$2, RegExp.$3) // See if the date in the object matches the string // (to make sure the string contained a valid date) if( RegExp.$1 != createdTime.getHours() || RegExp.$2 != createdTime.getMinutes() || RegExp.$3 != createdTime.getSeconds()) return null; else return createdTime; } /** * Format time to a string * @param noSeconds Do not include seconds in resulting string */ Date.prototype.toTimeString = function(noSeconds) { var hours = this.getHours(); var minutes = this.getMinutes(); var seconds = this.getSeconds(); var secondString = ""; if( seconds > 0 && ! noSeconds) secondString = ":" + (seconds < 10 ? "0" : "") + seconds; return (hours < 10 ? "0" : "") + hours + ":" + (minutes < 10 ? "0" : "") + minutes + secondString; } ///////////////////////////////////////////////// // Check a form before posting ///////////////////////////////////////////////// /** * If second parameter is false, the event will be cancelled. * See http://www.xulplanet.com/references/objref/Event.html */ function eventReturn(event, retVal) { if(event != null) { event.returnValue = retVal; event.cancel = ! retVal; if(! retVal) { // retVal == false if(event.preventDefault) event.preventDefault(); } } return retVal; } function alreadyCancelled(event) { if(event != null) { if(event.getPreventDefault) // Firefox return event.getPreventDefault(); if(event.returnValue != null) // Opera and IE return ! event.returnValue; // returnValue == false --> Cancelled } return false; // No event to cancel... } function checkForm(theForm, event) { if(navigator.appName == "HttpUnit") // Skip if HttpUnit. No support for event. return true; if(event != null && event.returnValue == false) // If already problems return false; if(theForm == null) // If no form... return true; // ... no problem var inputFields = theForm.elements; // To get selects also var errorField = null; for(var i = 0; i < inputFields.length; i++) { var currentField = inputFields[i]; if( ( isRequired(currentField) ) && ( ! currentField.readOnly )){ checkNotBlankNew(currentField); // An error is reported if the field is required and blank or 0 } if(errorField == null && isFieldMarked(currentField)) errorField = currentField; } if(markErrors()){ // If any error was found return eventReturn(event, false); } else if(errorField != null) { // Error not registered in error handler alert("Sidan inneh\u00E5ller fel som du m\u00E5ste \u00E5tg\u00E4rda f\u00F6rst!"); setFocus(errorField); return eventReturn(event, false); } return eventReturn(event, true); } /** * Flag to tell if the user has already posted the form so that the * page must first be reloaded. */ var alreadyPosted = false; /** * This function checks if the button has already been clicked */ function preventDoubleClick(event) { if(navigator.appName == "HttpUnit") // Skip if HttpUnit. No support for event. return true; if(alreadyCancelled(event)) // If something (form check) has already gone wrong return false; // For 5.0 only: if(! alreadyPosted) // If page not already posted { alreadyPosted = true; // Now, we have clicked already return eventReturn(event, true); } else // If we have clicked already { alert('Bearbetning p\u00E5g\u00E5r, v\u00E4nligen v\u00E4nta.'); return eventReturn(event, false); } } /** * This function should be set as onClick on links, and prevents the * link på be followed if the page has already been posted (as defined * above). */ function stopIfSubmitted(event) { // if(alreadyCancelled(event)) // If something has already gone wrong // return; if(alreadyPosted) { return eventReturn(event, false); } } /** Allow user to click again in x seconds */ function allowClickAgain(seconds) { window.setTimeout('alreadyPosted = false', seconds * 1000); } // Include tabs scripts ///////////////////////////////////////////////// // Tabs ///////////////////////////////////////////////// /** Global variable containing ID of current tab */ var currentTabId = null; /** * This function sets the tab to be used on posting or reloading * @param tab The name of the tab to set. */ function setTab(tab) { if(typeof tab == 'string') tab = $(tab); // 1. Remember this tab in the form posting // MJ 2002-04-25: Return to first tab after post // var theForm = document.forms(0); // if( theForm.baseAction == null ) // theForm.baseAction = theForm.action; // theForm.action = theForm.baseAction + "?TAB=" + tab.id; // 2. Remeber tab in document in case of reload currentTabId = tab.id; // 3. Mark the current tab var tabLinks = document.getElementsByName("tabLink"); // Find all items with id=tabLink for(var i = 0; tabLinks != null && i < tabLinks.length; i++) { var currentTabLink = tabLinks[i]; if( currentTabLink.getAttribute('tab') == tab.id) // If this was the clicked link currentTabLink.style.color = "black"; else currentTabLink.style.color = "#154A8A"; } } /** * Switch tab. Hides all other tabs (divs) and shows the new one. * @param tab The name of the tab to show. (The ID of the DIV). */ function showTab(tab) { if(typeof tab == 'string') tab = $(tab); // 1. Remeber the tab (and mark correct current) setTab(tab); // 2. Hide all other tabs var allDivs = document.getElementsByTagName("div"); if( allDivs != null ) { for(var i = 0; i < allDivs.length; i++) { if(allDivs[i].getAttribute('tab') != null) // If this this div is a tab. allDivs[i].className = "hidden"; } } // 3. Show the selected tab tab.className = "shown"; } /** * Switch tab to the one that contains the specified field */ function showTabWithField(field){ // Try to find the corresponding tab var containingTab = findParentTag(field, "DIV"); // If we found a div that is tab, switch to it if( containingTab && containingTab.getAttribute('tab')) showTab( containingTab ); } ///////////////////////////////////////////////// // Key validation -> page reload ///////////////////////////////////////////////// /** * Reloads the page with the value of the key field as the new parameter. * @param The fields (one or many) with the key */ function reload() { // location.search = keyField.name + "=" + keyField.value; var searchString = ""; for ( var i = 0; i < arguments.length; i++) { var currentField = element(arguments[i]); if(fieldHasAnyError(currentField)) // Do not reload if key is invalid return; // MJ 2002-05-29: searchString += currentField.name + "=" + urlEncodeData(currentField.value) + "&"; // searchString += currentField.name + "=" + encodeURIComponent(currentField.value) + "&"; } if( currentTabId != null ) searchString += "TAB=" + currentTabId; waitForReload(); window.location.search = searchString; } /** * Stop the user from interaction until page has reloaded */ function waitForReload() { alreadyPosted = true; // Forbid post or click before reload // Clear the page, so no other clicks (like submit) is possible // MJ 2002-06-19: 5.0 still crashes... if(navigator.appVersion.indexOf('MSIE 5.0') == -1) // If not IE 5.0 document.body.innerHTML = "Wait while reloading..."; // MJ 2002-07-16: Crashes 5.0 //document.body.removeNode(true); // Remove body and all inside } ///////////////////////////////////////////////// // Confirmation of deletes and obliterations ///////////////////////////////////////////////// function confirmDelete(event) { if( ! confirm("\u00C4r du s\u00E4ker p\u00E5 att du vill radera?") ) { return eventReturn(event, false); } } function confirmObliterate(theField, event) { var dataType = theField.getAttribute('unlockTime'); var textInfo = "\u00C4r du s\u00E4ker p\u00E5 att du vill makulera? "; if (dataType !== null && dataType !== "null") { textInfo = "Artikelinformation \u00E4r l\u00E5st tills " + dataType; if (!alert(textInfo)) { return eventReturn(event, false); } } if (!confirm(textInfo)) { return eventReturn(event, false); } } ///////////////////////////////////////////////// // Input correction //////////////////////////////////////////////// /** * Filter the input and cancel unallowed keys */ function filterInput(theField, event) { var keyCode = event.keyCode ? event.keyCode : // IE (event.charCode ? event.charCode : event.which); // Firefox // First do some common filtering if( event.ctrlKey ) // If control-key is pressed (this may always be true in Firefox) return; // Always allow (for Ctrl+C and Ctrl+V) if(keyCode == 9 || keyCode == 32 || keyCode == 27) // If TAB or SPACE (allow space for onChange(???)) or ESC return; if(keyCode == 8 || keyCode == 46) // If BACKSPACE or DELETE return; if(keyCode >= 33 && keyCode <= 40) // If PAGEUP/DOWN or END or HOME or ARROWS return; if(theField.tagName.toUpperCase() == "TEXTAREA") { // If maxlength not used or not yet reached if(theField.maxlength == null || theField.value.length < theField.maxlength) return; else { return eventReturn(event, false); } } // Numeric values are allowed everywhere if( (keyCode >= 48 && keyCode <= 57) || // >= 0 && <= 9 (keyCode >= 96 && keyCode <= 105) ) // numeric keyboard >= 0 && <= 9 return; // Now let's look at the type and do specific filtering var dataType = theField.getAttribute('datatype'); // alert("dataType " + dataType); if(dataType == null || dataType == 6 || dataType == 7) { return; // For text, any character is allowed } // Negative numbers and dates if(dataType == 1 || dataType == 4 || dataType == 8) { if(keyCode == 109 || keyCode == 189) // If MINUS/DASH '-' (normal or numeric keyboard) return; } // FLOAT if(dataType == 4 || dataType == 5) { var maxDecimals = parseInt(theField.getAttribute('decimals')); if(theField.value.indexOf('.') == -1 && theField.value.indexOf(',') == -1 && // If not already punctuation maxDecimals != 0 && // And decimals are allowed event.shiftKey == false && // Not shifted (keyCode == 110 || keyCode == 188 || keyCode == 190) ) // If punctuation (, or .) return; } // TIME if(dataType == 9) { if(event.shiftKey == true && keyCode == 190) // If colon (:) return; } return eventReturn(event, false); } /** * Verify that there is input on this required field. * The function checks if the field is required, since it * might be set onChange for fields toBeRequired * @return true if the input is ok, false if it is blank */ function checkNotBlank(theField) { trimField(theField); // We make sure the field is required, since the field might be "toBeRequired" if(isRequired(theField) && stringIsBlank(theField.value) ) // If required and blank { alert("Obligatoriskt f\u00E4lt! V\u00E4rde m\u00E5ste anges"); markField(theField); return false; // Check was NOT OK } else { unmarkField(theField); return true; // Check was OK } } /** * Verify that there is input on this required field. * The function checks if the field is required, since it * might be set onChange for fields toBeRequired * @return true if the input is ok, false if it is blank */ function checkNotBlankNew(theField) { trimField(theField); errorhandler.removeErrorForField(theField, "required_field"); if(isRequired(theField) && ! theField.readOnly){ if( theField.type == "checkbox" && ! theField.checked){ // If non checked checkbox errorhandler.addErrorForField(theField, "required_field"); return false; } else if(stringIsBlank(theField.value)){ errorhandler.addErrorForField(theField, "required_field"); return false; } else { var dataType = theField.getAttribute('datatype'); // Required numeric fields should not have value 0 if( ( dataType >= 1 && dataType <= 5 ) && parseFloat(replaceDecimalChar(theField.value)) == 0 ){ errorhandler.addErrorForField(theField, "required_field"); return false; } } } else if(stringIsBlank(theField.value)){ // Blank but not requried return false; } return true; // Not blank } /** * Check if there is a valid entry in the field * Valid values depend on the field attribute 'datatype' * This method reports errors to the gloabalErrorHandler */ function checkInput(theField) { //debugger; // This method takes care of all erorrs on level 10 so start by removing those errors from the field //errorhandler.removeErrorsForFieldByLevel(theField, 10); // This method is always(?) called first in validation routines so remove all errors from field errorhandler.removeAllErrorsForField(theField); if( ! checkNotBlankNew(theField) ) // If the field was blank, error has been noted return; // we don't need to check the value var theValue = theField.value; var dataType = theField.getAttribute('datatype'); if(typeof dataType == 'undefined') dataType = jq(theField).data('datatype'); // Backed-up up by auto complete var numericField = false; var errorType = null; // No error type means the field is valid // INTEGER if(dataType >= 1 && dataType <= 3) { numericField = true; var parsedValue = parseInt(theValue, 10); // Parse as decimal var i = 0; for(i = 0; i < theValue.length && theValue.charAt(i) == '0'; i++) // Loop over all leading zeros ; if( theValue != "" && theValue.substring(i) != parsedValue ) // (Remove leading 0s) errorType = "invalid_integer"; else if( dataType == 2 && parsedValue < 0 ) errorType = "invalid_positive"; else if( dataType == 3 && parsedValue < 1 ) // STÄMMER 1??? errorType = "invalid_key"; } // FLOAT else if(dataType == 4 || dataType == 5) { numericField = true; // Convert all , to . theField.value = replaceDecimalChar(theField.value); theValue = theField.value; var parsedValue = parseFloat(theValue); if( theValue != "" && theValue != parsedValue ) errorType = "invalid_float"; else if( dataType == 5 && parsedValue < 0 ) errorType = "invalid_positive"; else if(theField.getAttribute('decimals') != null) theField.value = formatDecimal(parsedValue, parseInt(theField.getAttribute('decimals'))); } // DATE else if(dataType == 8) { var dateInField = stringToDate(theValue); if( dateInField == null ) errorType = "invalid_date"; else theField.value = dateInField.toString(); } // TIME else if(dataType == 9) { var timeInField = stringToTime(theValue); if( timeInField == null ) errorType = "invalid_time"; else { var noSeconds = theField.getAttribute("noSeconds"); theField.value = timeInField.toTimeString(noSeconds); } } // Check required field if( isRequired(theField) && numericField && parseFloat(theValue) == 0) errorType = "must_not_be_zero"; // Mark the field and display error OR unmark if( errorType ){ errorhandler.addErrorForField(theField, errorType); return false; } else{ return true; } } /** * Change the letters in a field to upper case. * @param field The field to change to upper case */ function upperCase(theField) { theField.value = theField.value.toUpperCase(); } /** * Get the hidden value element for the specified checkbox * */ function getHiddenCheckboxValueElement(checkboxElement){ if( checkboxElement.getAttribute("valueField") == null ){ // If the value field has not been set, set it var CHECKBOX_PREFIX = "CHECKBOX_"; checkboxElement.setAttribute("valueField", checkboxElement.id.substr(CHECKBOX_PREFIX.length)); } return $(checkboxElement.getAttribute("valueField")); } /** * Set the hidden value field of a checkbox withou calling * the onClick method. For example used by checkboxes on new rows. */ function setHiddenCheckboxValue(checkboxElem) { var valueElem = getHiddenCheckboxValueElement(checkboxElem); var checked = checkboxElem.checked; if(checked) valueElem.value = 't'; else valueElem.value = 'f'; unmarkField(checkboxElem); } ////////////////////////////////////////////////////////////// // Find fields ////////////////////////////////////////////////////////////// function getElementsByAttribute(tagName, attributeName, attributeValue) { var output = new Array(); var foundCount = 0; var allTags = document.getElementsByTagName(tagName); for(i = 0; i < allTags.length; i++) { var currentElement = allTags[i]; if(currentElement.attributes != null && // If element has attributes currentElement.getAttribute(attributeName) == attributeValue) // And attribute matches value output[foundCount++] = currentElement; } return output; } /** * Find a field. First try to find it on the row and, if not found there, * then try to find it on the "head" (no row). */ function getFieldOnRowOrHead(fieldName, rowNumber) { if(rowNumber == null) // If recursed to "head" level return document.getElementById(fieldName); // Return from "head" var fieldOnRow = document.getElementById(fieldName + "_" + rowNumber); if(fieldOnRow != null) // If found on row return fieldOnRow; else // If not found on this level, look at parent level recursively return getFieldOnRowOrHead(fieldName, getParentRowNumber(rowNumber) ); } /** Get the rowNumber, given a field */ function getRowNumber(field) { return field.getAttribute('rowNumber'); } /** Get the rowNumber with suffix, given a field */ function getRowNumberWithPrefix(field) { var rowNumber = getRowNumber(field); return rowNumber ? "_" + rowNumber : ""; }; /** * Find parent rowNumber of a rowNumber */ function getParentRowNumber(rowNumber) { rowNumber = new String(rowNumber); // Make sure rowNumber is a String (can be number) if(rowNumber == null) return null; var indexOfUnderscore = rowNumber.lastIndexOf('_'); if(indexOfUnderscore == -1) // If the rownumber is not a child row return null; else return rowNumber.substring(0,indexOfUnderscore); } /* * Extract rowNumber part of an id. Example a_b_c_1_2 -> 1_2. * If the ID does not contain a rowNumber, null will be returned. */ function extractRowNumber(id) { var startUnderscore = id.lastIndexOf('_'); if(startUnderscore == -1) // If id does not contain underscore return null; var endUnderscore = id.length; // Initially, look until end while(startUnderscore != -1 && ! isNaN(parseInt(id.substring(startUnderscore + 1,endUnderscore)))) { endUnderscore = startUnderscore; // Move back one underscore startUnderscore = id.lastIndexOf('_', endUnderscore - 1); } if(endUnderscore < (id.length)) // If there were rownumbers return id.substring(endUnderscore + 1); else return null; } function isTemplate(idOrElem) { var elem = element(idOrElem); var templateLevel = elem.getAttribute("templateLevel"); return (templateLevel != null) && templateLevel > 0; } ////////////////////////////////////////////////////////////// // Change field state ////////////////////////////////////////////////////////////// /** * Mark a field containing an error */ function markField(theField) { // For future use: // if(theField.className != null && ( ! fieldHasAnyError(theField) ) ) // theField.oldClassName = theField.className; theField.className = 'error'; // Mark the field } /** * Clear mark from an error free field */ function unmarkField(theField) { if( theField.readOnly ) theField.className = "readonly"; else if( isRequired(theField) ) theField.className = "required"; // Unmark a required field else theField.className = null; // Unmark the field } /** * Clear mark from an error free field if it's not marked as required */ function unmarkUnrequiredField(field){ if( ! isRequired(field) ) unmarkField(field); } /** Is the field marketd (CSS style) with error? */ function isFieldMarked(field) { return (field.className == "error"); } /** * Set whether field is open/editable or closed/uneditable by providing a flag * @param editable true if the field should be editable, false if it should not * @param clearOnClose If the field is closed, should it also be cleared? */ function setEditable(elemOrId, editable, clearOnClose) { if(editable) openField(elemOrId); else { if(clearOnClose) closeAndClear(elemOrId); else closeField(elemOrId); } } /** * Make a field uneditable */ function closeField(elemOrId) { if(navigator.appName == "HttpUnit") // Skip if HttpUnit return; var theField = element(elemOrId); theField.className = "readonly"; theField.readOnly = true; if(theField.type == "checkbox") theField.disabled = true; theField.tabIndex = -1; var checkBox = $('CHECKBOX_' + theField.id); if(checkBox != null) checkBox.disabled = true; if(theField.tagName == "SELECT") theField.disabled = true; } /** * Closes an interval of fields, specifying only the underlying field id * Ex: article_id -> article_id_from, article_id_to * @param id the field id to close */ function closeIntervalFields(id) { closeField(id + "_from"); closeField(id + "_to"); } /** * Make a field editable */ function openField(elemOrId) { if(navigator.appName == "HttpUnit") // Skip if HttpUnit return; var theField = element(elemOrId); theField.readOnly = false; theField.disabled = false; theField.tabIndex = 0; unmarkField(theField); // Set class to required or null // moved from first line by mu 2002-06-13 var checkBox = $('CHECKBOX_' + theField.id); if(checkBox != null) checkBox.disabled = false; if(theField.tagName == "SELECT") theField.disabled = false; } /** * Opens an interval of fields, specifying only the underlying field id * Ex: article_id -> article_id_from, article_id_to * @param id the field id to open */ function openIntervalFields(id) { openField(id + "_from"); openField(id + "_to"); } /** * Close given field (=make uneditable) and clear data */ function closeAndClear(elemOrId) { var theField = element(elemOrId); closeField(theField); // Close theField.value = ''; // Clear closeField(theField); theField.value = ''; } /** * Check if given field is required */ function isRequired(theField) { return theField.getAttribute('required'); } /** * Make a field required */ function makeRequired(fieldOrId) { var theField = element(fieldOrId); if(isTemplate(theField)) { theField.setAttribute('toBeRequired', true); // This may be a template field } else theField.setAttribute('required', true); if(! theField.readOnly) theField.className = "required"; // Unmark a required field } /** * Make a field no longer required */ function makeNotRequired(fieldOrId) { var theField = element(fieldOrId); if(isTemplate(theField)) { theField.removeAttribute('toBeRequired'); // This may be a template field } else theField.removeAttribute('required'); errorhandler.removeErrorForField(theField, "required_field"); if(! theField.readOnly) theField.className = null; // Unmark a required field } /** * Toggle wether the the fields of a row are required or not, depending on the * delete checkbox. */ function toggleRequiredRow(checkboxElem) { checkboxElem = getCheckBox(checkboxElem); // Make sure we have the checkbox (and not the hidden field) var deleteRowsString = checkboxElem.getAttribute('deleteRows'); var rowsToCheck = new Array(); if(deleteRowsString ){ // The deleteRows attribute consists of a comma separated list of elements var rowNumber = extractRowNumber(checkboxElem.id); var rowsIdsToCheck = deleteRowsString.split(","); // find the rows to check using id and rownumber for(var i = 0; i < rowsIdsToCheck.length; i++ ){ rowsToCheck.push( $(trimString(rowsIdsToCheck[i]), rowNumber) ); } } else{ var deleteLevels = checkboxElem.getAttribute('deleteLevels'); if(! deleteLevels) deleteLevels = 1; // Default only 1 level = only current TR var row = findParentTag(checkboxElem, "TR"); for(var j = 1; j < deleteLevels; j++) { row = findParentTag(row, "TR"); // Step up to next TR (in case of nested table) } // If no table was found, alert and exit if( row == null ) { alert("No row"); return; } rowsToCheck.push(row); } // Locate the hidden value field for the checkbox var valueElem = getHiddenCheckboxValueElement(checkboxElem); // Find all input tags // It is tempting to handle tag 'SELECT' the same way but don´t, // because a readonly SELECT is not among the parameters. for(var ix = 0; ix < rowsToCheck.length; ix++ ){ var inputTags = rowsToCheck[ix].getElementsByTagName("INPUT"); _handleRequiredFromArray(inputTags, checkboxElem, valueElem); var textAreaTags = rowsToCheck[ix].getElementsByTagName("TEXTAREA"); _handleRequiredFromArray(textAreaTags, checkboxElem, valueElem); var selectTags = rowsToCheck[ix].getElementsByTagName("SELECT"); _handleRequiredFromArraySelect(selectTags, checkboxElem, valueElem); } } /* * "Private method" to handle readOnly and required, depending on checkbox */ function _handleRequiredFromArray(labelTags, checkboxElem, valueElem) { for(i = 0; i < labelTags.length; i++) { var currentTag = labelTags[i]; var checked = checkboxElem.checked; if(currentTag != checkboxElem && currentTag != valueElem){ // Don't do anything with either the delete checkbox itself or it's complimentary value element if(checked){ if(!currentTag.readOnly){// Fields that are originally marked readOnly should not be toggled currentTag.setAttribute("toggleReadOnly", true); } if(currentTag.getAttribute("toggleReadOnly")){ // closeField(currentTag); } if(isRequired(currentTag)){// If delete and this one is required currentTag.setAttribute("wasRequired", true); // Remeber it was required if unchecked currentTag.removeAttribute('required'); //errorhandler.removeAllErrorsForField(currentTag); // Added this call to enable removal of empy row that has been marked for errors if( currentTag.className == "required" ) // If marked required (not "error") currentTag.className = ""; // view as not required (was = null) } } else{ // If not delete if(currentTag.getAttribute("wasRequired")){ // If this one was required currentTag.setAttribute('required', true); currentTag.removeAttribute("wasRequired"); if( currentTag.className == "") // If not "error" (was currentTag.className == "null" ) currentTag.className = "required"; // view as required } if(currentTag.getAttribute("toggleReadOnly")){ openField(currentTag); fieldHasAnyError(currentTag); // Mark reopened field if it has an error } } } } } function _handleRequiredFromArraySelect(labelTags, checkboxElem, valueElem) { for(i = 0; i < labelTags.length; i++) { var currentTag = labelTags[i]; var checked = checkboxElem.checked; if(currentTag != checkboxElem && currentTag != valueElem){ // Don't do anything with either the delete checkbox itself or it's complimentary value element if(checked){ if(currentTag.getAttribute("readonly") !== "readonly"){// Fields that are originally marked readOnly should not be toggled currentTag.setAttribute("toggleReadOnly", true); } if(currentTag.getAttribute("toggleReadOnly")){ // var select = document.getElementById(currentTag.id); var selectId = select.id; var input = document.querySelector('input#' + selectId); if(!input) { input = document.createElement("INPUT"); input.setAttribute("type", "hidden"); input.setAttribute("id", selectId); input.setAttribute("name", selectId); input.setAttribute("field_id", selectId); input.setAttribute("value", select.value); select.parentNode.insertBefore(input, select.nextSibling); } closeField(currentTag); } if(isRequired(currentTag)){// If delete and this one is required currentTag.setAttribute("wasRequired", true); // Remeber it was required if unchecked currentTag.removeAttribute('required'); if( currentTag.className == "required" ) // If marked required (not "error") currentTag.className = ""; // view as not required (was = null) } } else{ // If not delete if(currentTag.getAttribute("wasRequired")){ // If this one was required currentTag.setAttribute('required', true); currentTag.removeAttribute("wasRequired"); if( currentTag.className == "") // If not "error" (was currentTag.className == "null" ) currentTag.className = "required"; // view as required } if(currentTag.getAttribute("toggleReadOnly")){ var select = document.getElementById(currentTag.id); var selectId = select.id; var input = document.querySelector('input#' + selectId); if (input) { input.remove(); } openField(currentTag); fieldHasAnyError(currentTag); // Mark reopened field if it has an error } } } } } function toggleHidden(elem) { elem = element(elem); elem.style.display = (elem.style.display == 'none') ? 'block' : 'none'; } /** * Used to show by class property * @param elem */ function shown(elem) { elem = element(elem); // firefox cant handle table structures if doing this, read below... //elem.className = "shown"; elem.className = ""; elem.style.display = ""; // to make it work for both ie and ff we instead set the class to ""; and... // set style.display to blank, to use browser default, this way you can still use css class "hidden" on // objects that you want to have hidden from start } /** * Used to hide by class property * @param elem */ function hidden(elem) { elem = element(elem); elem.className = "hidden"; } /** * Used to show or hide by class property * @param elem * @param shownOrHidden */ function setShownOrHidden(elem, shownOrHidden){ if(shownOrHidden) shown(elem); else hidden(elem); } function show(elem) { elem = element(elem); // elem.style.visibility = "visible"; elem.style.display = ""; } function hide(elem) { elem = element(elem); // elem.style.visibility = "hidden"; elem.style.display = "none"; } /** Show or hide element (possibly ID) depending on the visible flag */ function setVisible(elemOrId, visible) { if(visible) show(elemOrId); else hide(elemOrId); } ////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////// /** Given a field ID, a hidden field or a checkbox, return the checkbox*/ function getCheckBox(elemOrId) { var field = element(elemOrId); // String or field return (field.type == "checkbox") ? field : $('CHECKBOX_' + field.id); // Look for checkbox } function setCheckBox(field, value) { // find visible checkbox var checkbox = getCheckBox(field); // Look for checkbox checkbox.checked = value; setHiddenCheckboxValue(checkbox); } function isCheckboxChecked(field) { return getCheckBox(field).checked; /* if(field.type == "checkbox") // If this is the actual checkbox return field.checked; else // If this is the hidden field containing T or F return field.value == "t"; */ } ////////////////////////////////////////////////////////////// // Scripts for value checks ////////////////////////////////////////////////////////////// /** * Make sure the value of a from field is less than the value of * the to field. * @param from The from field * @param to The to field */ function checkFromTo(fromElemOrId, toElemOrId, event, customErrorMessage) { var from = element(fromElemOrId); var to = element(toElemOrId); // Determine error type to send right error message var dataType = from.getAttribute('datatype'); var errorType = null; if(dataType == 8){ errorType = "end_date_earlier_start_date"; } else errorType = "error_from_to"; errorhandler.removeErrorForField(from, errorType); errorhandler.removeErrorForField(to, errorType); // IE and W3C (Firefox) have different ways to access events // Opera seems to support both W3C's and IE's ways of event handling var eventField = window.event ? window.event.srcElement : event ? event.target : null; var fromIsValid = checkInput(from); // Special case // if from-field is entered and the user haven't reached the to-field, no error message should be sent // even if to-field is required if(eventField && (eventField == from) && (to.value == "") && fromIsValid){ return; } var toIsValid = checkInput(to); if( fromIsValid && toIsValid){ // If both fields are valid by themselves var fromValue = fieldToObject(from); var toValue = fieldToObject(to); if( fromValue == null || toValue == null ){ // Is this check really necessary after checkInput() ? return; } else if( fromValue > toValue ){ errorhandler.addErrorForField(from, errorType, customErrorMessage, to); errorhandler.addErrorForField(to, errorType, customErrorMessage, from); } } } function timeOnlyIfDate(dateFieldId, timeFieldId) { var dateField = element(dateFieldId); var timeField = element(timeFieldId); if(fieldIsBlank(dateField)) { closeField(timeField); timeField.value = ''; } else openField(timeField); } /** * Make sure the fromDate is before toDate */ function validateFromToDates(startDateFieldId, endDateFieldId, event, rowNumber){ var startDateField = $(startDateFieldId, rowNumber); var endDateField = $(endDateFieldId, rowNumber); checkFromTo(startDateField, endDateField, event); } function checkFromToTimes(dateFieldId, timeFieldId, event) { var fromDateField = $(dateFieldId + '_from'); var toDateField = $(dateFieldId + '_to'); var fromTimeField = $(timeFieldId + '_from'); var toTimeField = $(timeFieldId + '_to'); var fromDate = fieldToObject(fromDateField); if(fromDate != null && fromDate.equals(fieldToObject(toDateField))) { // Same day checkFromTo(fromTimeField, toTimeField, event); // Check time intervall } else { // Dates are not same, allow any time intervall errorhandler.removeErrorForField(fromTimeField, "error_from_to"); errorhandler.removeErrorForField(toTimeField, "error_from_to"); } } function checkEmailAddress(emailAddressField) { if(fieldHasAnyError(emailAddressField)) // If already erronous return false; // Skip check var emailAddress = emailAddressField.value; if(emailAddress.length == 0) // If no mail entered return true; // Skip check (required check is done elsewhere) if( ! emailCheck(emailAddress)) // If error when checking e-mail { markField(emailAddressField); alert("Felaktig e-postadress"); return false; // Check was NOT OK } else return true; } function checkEmailAddresses(emailAddressField) { if(fieldHasAnyError(emailAddressField)) // If already erronous return false; // Skip check var emailAddresses = emailAddressField.value.split(new RegExp("[ ;,]")); if(! emailAddresses) // Only delimiters return false; for(var i =0; i < emailAddresses.length; i++ ){ var emailAddress = emailAddresses[i]; if(! stringIsBlank(emailAddress) && ! emailCheck(emailAddress)) // If error when checking e-mail { markField(emailAddressField); alert("Felaktig e-postadress"); return false; // Check was NOT OK } } return true; // All valid } /* Mattias Jiderhamn found this script at http://javascript.internet.com/forms/check-email.html Modified by Mattias Jiderhamn: All alerts removed Changes: Sandeep V. Tamhankar (stamhankar@hotmail.com) 1.1.2: Fixed a bug where trailing . in e-mail address was passing (the bug is actually in the weak regexp engine of the browser; I simplified the regexps to make it work). 1.1.1: Removed restriction that countries must be preceded by a domain, so abc@host.uk is now legal. However, there's still the restriction that an address must end in a two or three letter word. 1.1: Rewrote most of the function to conform more closely to RFC 822. 1.0: Original This script and many more are available free online at The JavaScript Source!! http://javascript.internet.com */ function emailCheck(emailStr) { /* The following pattern is used to check if the entered e-mail address fits the user@domain format. It also is used to separate the username from the domain. */ var emailPat=/^(.+)@(.+)$/ /* The following string represents the pattern for matching all special characters. We don't want to allow special characters in the address. These characters include ( ) < > @ , ; : \ " . [ ] */ var specialChars="\\(\\)<>@,;:\\\\\\\"\\.\\[\\]" /* The following string represents the range of characters allowed in a username or domainname. It really states which chars aren't allowed. */ var validChars="\[^\\s" + specialChars + "\]" /* The following pattern applies if the "user" is a quoted string (in which case, there are no rules about which characters are allowed and which aren't; anything goes). E.g. "jiminy cricket"@disney.com is a legal e-mail address. */ var quotedUser="(\"[^\"]*\")" /* The following pattern applies for domains that are IP addresses, rather than symbolic names. E.g. joe@[123.124.233.4] is a legal e-mail address. NOTE: The square brackets are required. */ var ipDomainPat=/^\[(\d{1,3})\.(\d{1,3})\.(\d{1,3})\.(\d{1,3})\]$/ /* The following string represents an atom (basically a series of non-special characters.) */ var atom=validChars + '+' /* The following string represents one word in the typical username. For example, in john.doe@somewhere.com, john and doe are words. Basically, a word is either an atom or quoted string. */ var word="(" + atom + "|" + quotedUser + ")" // The following pattern describes the structure of the user var userPat=new RegExp("^" + word + "(\\." + word + ")*$") /* The following pattern describes the structure of a normal symbolic domain, as opposed to ipDomainPat, shown above. */ var domainPat=new RegExp("^" + atom + "(\\." + atom +")*$") /* Finally, let's start trying to figure out if the supplied address is valid. */ /* Begin with the coarse pattern to simply break up user@domain into different pieces that are easy to analyze. */ var matchArray=emailStr.match(emailPat) if (matchArray==null) { /* Too many/few @'s or something; basically, this address doesn't even fit the general mould of a valid e-mail address. */ return false } var user=matchArray[1] var domain=matchArray[2] // See if "user" is valid if (user.match(userPat)==null) { // user is not valid return false } /* if the e-mail address is at an IP address (as opposed to a symbolic host name) make sure the IP address is valid. */ var IPArray=domain.match(ipDomainPat) if (IPArray!=null) { // this is an IP address for (var i=1;i<=4;i++) { if (IPArray[i]>255) { return false } } return true } // Domain is symbolic name var domainArray=domain.match(domainPat) if (domainArray==null) { return false } // Make sure there's a host name preceding the domain. if (len<2) { var errStr="This address is missing a hostname!" return false } // If we've gotten this far, everything's valid! return true; } /** * Validate that the the field complies to a regular expression, unless it is blank * (since blank should be caught by required check). If no error message is provided a * general "Invalid value" will be used. */ function validateRegExpIfNotBlank(field, re, errorMessage) { var value = trimField(field); if(value.length != 0) // If not blank (required will catch that) { if(value.match(re) == null) { markField(field); if(errorMessage != null) alert(errorMessage); else alert('Ogiltigt v\u00E4rde'); } else unmarkField(field); } } /** * Get the value of a field with the corrent object type */ // MJ 2002-06-11: function fieldToObject(theField) { if(theField == null) return null; trimField(theField); var theValue = theField.value; var dataType = theField.getAttribute('datatype'); if( theValue == "" ) // If blank return null; // INTEGER if(dataType >= 1 && dataType <= 3) return parseInt(theValue); // FLOAT else if(dataType == 4 || dataType == 5) { // Convert all , to . theField.value = replaceDecimalChar(theField.value); theValue = theField.value; return parseFloat(theValue); } // DATE else if(dataType == 8) { return stringToDate(theValue); } // TIME else if(dataType == 9) { return stringToTime(theValue); } return new String( theValue.trim() ); } function stringIsBlank(s) { return s == null || s.trim().length == 0; } function stringIsTrue(s) { return s == "true" || s == "t"; } function fieldIsBlank(elemOrId) { var field = element(elemOrId); return field == null || stringIsBlank(field.value); } /** * Get the valid value of the field, provided by field or ID. * Will return null if field is blank or has any errors */ function getValidValue(elemOrId) { var field = element(elemOrId); if(field == null || fieldHasAnyError(field) || fieldIsBlank(field)) { return null; } else return field.value; } /** Get text content (i.e. inner text) of object */ function getTextContent(obj) { if(obj.tagName == "INPUT" || obj.tagName == "SELECT") return obj.value; else if(document.all) return obj.innerText; else return obj.textContent; } /** Set text content (i.e. inner text) of object */ function setTextContent(obj, value) { if(obj.tagName == "INPUT") obj.value = value; else if(document.all) obj.innerText = value; else obj.textContent = value; } ///////////////////////////////////////////////// // Pagination of dirty pages ///////////////////////////////////////////////// var pageDirty = false; // Initially page is not dirty function warnIfDirty(event) { if(pageDirty) { if( ! confirm("\u00C4ndringar har gjorts. Vill du forts\u00E4tta utan att spara?") ) { return eventReturn(event, false); // Do not move to next page } } } ///////////////////////////////////////////////// // Tooltip scripts and utility functions for // both tooltips and menus ///////////////////////////////////////////////// /** * Hide the SELECT tags that are covered by the parameter object, * since they would otherwise be seen through the object. * If dimensions of object cannot be determined, hide all SELECT tags. * @param doc The document on which to hide all the SELECTs * @param obj The HTML element whose dimensions should determine what SELECTs to hide */ function hideSelectsBehindObject(doc, obj) { var objDims = obj.getBoundingClientRect ? obj.getBoundingClientRect() : null; var selectTags = doc.getElementsByTagName("SELECT"); for(i = 0; i < selectTags.length; i++) // Go through all elements { var selectTag = selectTags[i]; if(selectTag.getBoundingClientRect) // If IE, hide only what needs to be hidden { var currentSelectCords = selectTag.getBoundingClientRect(); // If above or below if(currentSelectCords.bottom < objDims.top || currentSelectCords.top > objDims.bottom) continue; // If above or below if(currentSelectCords.right < objDims.left || currentSelectCords.left > objDims.right) continue; } selectTags[i].style.visibility = "hidden"; } } /** * Show all the SELECT tags hidden by hideSelectsBehindObject above. * @param doc The document on which to show all the SELECTs. */ function showAllSelects(doc) { var selectTags = doc.getElementsByTagName("SELECT"); for(i = 0; i < selectTags.length; i++) // Go through all elements selectTags[i].style.visibility = "visible"; } // Include tooltip scripts ///////////////////////////////////////////////////////////////////////////////// // TOOLTIPS ///////////////////////////////////////////////////////////////////////////////// function showTooltip(evt,txt) { if($('tipCell') == null) return; $('tipCell').innerHTML = txt; $('tooltipDiv').style.left = document.body.scrollLeft + evt.clientX + 5; // Was offsetX and no scroll $('tooltipDiv').style.top = document.body.scrollTop + evt.clientY + 5; // Was offsetY and no scroll $('tooltipDiv').timeout = window.setTimeout('performShowTooltip()', 600); } function performShowTooltip() { hideSelectsBehindObject(document, $('tipCell')); // Position the tooltip where the mouse is at now // document.tooltipDiv.style.pixelLeft = document.body.scrollLeft + window.event.clientX + 3; // Was offsetX and no scroll // document.tooltipDiv.style.pixelTop = document.body.scrollTop + window.event.clientY + 3; // Was offsetY and no scroll // Show the tooltip $('tooltipDiv').style.visibility = "visible"; } function hideTip() { if($('tooltipDiv') == null) return; // Make sure any shown tooptip is hidden $('tooltipDiv').style.visibility = "hidden"; // Clear any "queued" tooltips window.clearTimeout( $('tooltipDiv').timeout ); if(menuActive) // If any menu is active return; // Do not reenable selects // Reenable selects showAllSelects(document); } ///////////////////////////////////////////////// // Keep-alive pings ///////////////////////////////////////////////// function ping() { var req; if (window.ActiveXObject) { // IE req = new ActiveXObject("Microsoft.XMLHTTP"); } else { // Firefox and Opera req = new XMLHttpRequest(); } req.open("GET", "https://oden.exder.net/html/ping.jsp", false); // false = Synch req.send(null); schedulePing(); // Schedule next ping } function schedulePing() { window.setTimeout('ping()', 10 * 60 * 1000); // Ping in 10 minutes } schedulePing(); // Schedule first ping ///////////////////////////////////////////////// // Scripts for uploaded files ///////////////////////////////////////////////// function checkFileExtension(caller) { var fileName = caller.value.trim(); if(stringIsBlank(fileName)) { unmarkField(caller); return; } var extension = ""; var lastDot = fileName.lastIndexOf('.'); if(lastDot > 0 && lastDot < (fileName.length - 1)) extension = fileName.substring(lastDot + 1).toLowerCase(); var extensionList = caller.getAttribute('extensions').toLowerCase(); var allowedExtensions = extensionList.split(","); for(var i = 0; i < allowedExtensions.length; i++) { if(extension == allowedExtensions[i]) { unmarkField(caller); return; } } markField(caller); alert("Otill\u00E5ten fil\u00E4ndelse. Bara f\u00F6ljande filtyper \u00E4r till\u00E5tna: \n" + extensionList); } ///////////////////////////////////////////////// // Make phone calls with 3Switch ///////////////////////////////////////////////// // Popup informing the user that the call is being placed var dialingPopup; /** JavaScript function for placing call using 3Switch*/ function placePhoneCall(number) { if(! dialingPopup) { // Popup has not been initialized dialingPopup = jq('
Ringer ...
').getBubbleContainer(); } var dialingDots = jq('#dialingDots'); var numberInDialPopup = jq('#numberInDialPopup'); // Show "Dialing 123456", without dots numberInDialPopup.removeClass('warning').html(number); dialingDots.css('opacity', 0); dialingPopup.center().fadeIn(); // Show popup, centered on page dialingPopup.doTimeout('dialing', 2000, 'fadeOut'); // Fade out in 2 seconds, if nothing else happens ThreeSwitchDWR.call(number, { callback: function(result) { if(result) { dialingDots.css('opacity', 1); // Show dots dialingPopup.doTimeout('dialing', 2000, 'fadeOut'); // Fade out after 2 seconds } else { dialingDots.css('opacity', 0); // Hide dots numberInDialPopup.addClass('warning').html('Ett fel uppstod'); // Show error dialingPopup.doTimeout('dialing', 4000, 'fadeOut'); // Fade out after 4 seconds } }, errorHandler: function(errorString, exception) { // Handle error dialingDots.css('opacity', 0); // Hide dots numberInDialPopup.addClass('warning').html(errorString); // Show error dialingPopup.doTimeout('dialing', 4000, 'fadeOut'); // Fade out after 4 seconds } }); } ///////////////////////////////////////////////// // Rounding and number formatting ///////////////////////////////////////////////// /** * This function looks up the correct rounding principle from the page or setting and rounds the given value accordingly */ function round(value) { var roundingPrincipleFromEntityField = $('rounding_principle'); var roundingPrinciple = ''; if(!fieldIsBlank(roundingPrincipleFromEntityField)) roundingPrinciple = roundingPrincipleFromEntityField.value; // first use given rounding principle on page else roundingPrinciple = 'HU'; // secondly from setting if(stringIsBlank(roundingPrinciple)) roundingPrinciple = 'HU'; // shouldnt happend, but thirdly use default if(roundingPrinciple == 'HE') { return roundHalfEven(value); } else if(roundingPrinciple == 'HU') { return roundHalfUp(value); } else { // should never happend !!! alert("Rounding principle " + roundingPrinciple + " not recognized!") return value; } } /** * Round the value using HALF_EVEN principle (bankers rounding) * Mattias Jiderhamn, 2003-05-14 * @param value */ function roundHalfEven(value) { var floored = Math.floor(value); if(value - floored == 0.5) // If exactly half { if(floored % 2 == 0) // If even return floored; else return Math.ceil(value); } else // If not half... return Math.round(value); // ... use normal rounding } /** * Round the value using HALF_UP principle (javascript default round) * @param value */ function roundHalfUp(value) { return Math.round(value); } /** * Format a decimal number to always have two decimals. * The rounding principle used is determined in round() */ function formatAmount(number) { var rounded = round(number * 100) / 100.0; var numberString = new String(rounded); var decimalPosition = numberString.indexOf('.'); if( decimalPosition < 0) return numberString + '.00'; else if(decimalPosition == numberString.length - 2) return numberString + '0'; else return numberString; } /** * Format a decimal number to always have three decimals. * The rounding principle used is determined in round() */ function formatQuantity(number) { var rounded = round(number * 1000) / 1000.0; var numberString = new String(rounded); var decimalPosition = numberString.indexOf('.'); if( decimalPosition < 0) return numberString + '.000'; else if(decimalPosition == numberString.length - 2) return numberString + '00'; else if(decimalPosition == numberString.length - 3) return numberString + '0'; else return numberString; } function formatDecimal(number, noOfDecimals) { var multiplier = Math.pow(10, noOfDecimals); var rounded = round(number * multiplier) / multiplier; var numberString = new String(rounded); if(noOfDecimals > 0) { var currentDecimals = getNoOfDecimals(numberString); if(currentDecimals == 0) numberString += '.'; for( ; currentDecimals < noOfDecimals ; currentDecimals++) numberString += '0'; } return numberString; } /** * Rounds value to either .00 or .50 * Gerry Askefalk, 2010-07-01 */ function currencyRounding(value) { var floored = Math.floor(value) if ( value - floored < 0.25 ) return floored; else { if (value -floored > 0.75) return floored +1.0; else return floored + 0.5; } } function getNoOfDecimals(number) { var val = new String(number); var decimalPosition = val.indexOf('.'); // Look for decimal dot if( decimalPosition < 0) // If no decimal dot ... decimalPosition = val.indexOf(','); // ... look for decimal comma if(decimalPosition >= 0) // Decimal delimiter found return val.length - decimalPosition - 1; else return 0; } /** * See http://www.gs1.org/productssolutions/barcodes/support/check_digit_calculator.html **/ function getEANChecksum(input) { var length = input.length; var factor = 3; var sum = 0; for(var i = length; i > 0; --i) { var current = input.charAt(i-1); sum += current * factor; factor = 4 - factor; // 3,1,3,1,3,1... } return (1000 - sum) % 10; } /** Check if string is a valid EAN code (GTIN, GLN etc) */ function isEANChecksumValid(input) { if(stringIsBlank(input)) return false; else return getEANChecksum(input.substring(0,input.length-1)) == input.charAt(input.length-1); } /** * Perform an action for each line in a table. * @param table The table element or the ID of the table. * @param rowDependentFieldId The ID if a field that must exist on the row to perform the action * @param callbackFunction The funtion that will be called for each row containing the field, given the rowNunber as parameter. */ function forEachTableLineContaining(table, rowDependentFieldId, callbackFunction) { forEachChildTableLineContaining(table, rowDependentFieldId, null, callbackFunction); } /** * Perform an action for each line in a table. * @param table The table element or the ID of the table. * @param rowDependentFieldId The ID if a field that must exist on the row to perform the action * @param parentRowNumber The row number of the parent of the table, if any * @param callbackFunction The funtion that will be called for each row containing the field, given the rowNunber as parameter. */ function forEachChildTableLineContaining(table, rowDependentFieldId, parentRowNumber, callbackFunction) { if(navigator.appName == "HttpUnit") // Skip if HttpUnit. No support for table rows. return; if(typeof table == 'string') // ID provided table = $(table, parentRowNumber); // Look up actual table var maxRowNo = table.rows.length; // Does not work with HttpUnit for(var childRowNumber = 1; childRowNumber <= maxRowNo; childRowNumber++) { var rowNumber = (parentRowNumber != null) ? (parentRowNumber + '_' + childRowNumber) : childRowNumber; var rowDependentField = $(rowDependentFieldId, rowNumber); if(rowDependentField != null) { callbackFunction(rowNumber, rowDependentField); } } } /** * Perform an action for each field with a given ID; looping rows. * @param fieldId The ID if a field that must exist on the row to perform the action. When no field exist, loop will end. * @param callbackFunction The funtion that will be called for each row containing the field, given the rowNunber as parameter. */ function forEachField(fieldId, callbackFunction) { for(var rowNumber = 1; true; rowNumber++) { var field = $(fieldId, rowNumber); if(field == null) break; else { callbackFunction(rowNumber, field); } } } /** * Performs a test to see if the browser is blocking popup-windows * */ function isPopupBlocked(){ var popup = window.open('', 'testWindow', 'scrollbar=no, width=1, height=1, outerHeight=1, outerWidth=1, menubar=no, location=no'); if (popup) { popup.close(); return false; } else { return true; } } function openRequestHelpPopup() { var page; try{ page = "/logic/requestHelp/edit?function_id=" + functionId; }catch(ex){ // If functionId doesn't exist we end up here. This means the current page doesn't have a // text:function-tag or doesn't print requestHelp-tag page = "/logic/requestHelp/edit?function_id=nonFunctionPage"; } openSearchPopup(page, 600, 520); } function zeroPad(orig,len) { var result = orig; if (result == null) return; while (result.length < len) { result = "0" + result; } return result; } /** * IS currentBrowser IE ? */ function isInternetExplorer() { return (navigator.userAgent.indexOf("MSIE") > 1); } // To help debugging // To help debugging
//
//
// To help debugging
////////////////////////////////////////////////////////// // Resizing main frame // This script is always accessible, even if not logged in ////////////////////////////////////////////////////////// /** * Resize the main (middle) frame * http://msdn.microsoft.com/library/default.asp?url=/workshop/author/om/measuring.asp * http://www.softcomplex.com/docs/get_window_size_and_scrollbar_position.html * http://www.quirksmode.org/viewport/compatibility.html */ function resizeMainFrame() { if(navigator.appName == "HttpUnit") // Skip if HttpUnit return; var mainFrame = document.getElementById('mainFrame'); // Try to find mainFram if(mainFrame != null) // If mainFrame found { var temp = jQuery("
"); jQuery("body").append(temp); /* alert( //"document.body.clientTop: " + document.body.clientTop + "\n" + // 0 //"document.body.clientHeight: " + document.body.clientHeight + "\n" + // Window "document.body.offsetHeight: " + document.body.offsetHeight + "\n" + // OK in Firefox "document.body.scrollHeight: " + document.body.scrollHeight + "\n" + // OK in older IE // "document.body.scrollHeight - document.body.offsetHeight + document.body.clientTop: " + (document.body.scrollHeight - document.body.offsetHeight + document.body.clientTop) + "\n" + "temp.offset().top: " + temp.offset().top + "\n" + "jQuery(document.body).height(): " + jQuery(document.body).height() + "\n" + // Only Firefox "jQuery(document.documentElement).height(): " + jQuery(document.documentElement).height() + "\n" + // Only Firefox - same as temp.top "jQuery(document.body).outerHeight(true): " + jQuery(document.body).outerHeight(true) + "\n" // Only Firefox // Only works in Firefox, and they are all the same as jQuery(document.documentElement).height() //(document.documentElement ? // "document.documentElement.offsetHeight: " + document.documentElement.offsetHeight + "\n" + // "document.documentElement.scrollHeight: " + document.documentElement.scrollHeight + "\n" + // "document.documentElement.clientHeight: " + document.documentElement.clientHeight // : "-") // "document.body.getBoundingClientRect().bottom: " + document.body.getBoundingClientRect().bottom + "\n" + // Only Firefox ); */ var tempTop = temp.offset(); temp.remove(); if(window.innerHeight) { // var clientTop = document.body.clientTop ? document.body.clientTop : 0; // var excess = document.body.offsetHeight - window.innerHeight; // + clientTop; // offsetHeight only works in Firefox, not Chrome nor (newer) IE var excess = tempTop.top - window.innerHeight; // + clientTop; /* window.status = "window.innerWidth: " + window.innerWidth + " / document.body.scrollWidth: " + document.body.scrollWidth + " / document.body.offsetWidth: " + document.body.offsetWidth; */ if(document.body.scrollWidth > window.innerWidth) // If horisontal scrollbars present excess += 15; var calculatedHeight = mainFrame.offsetHeight - excess - 34; // Save space for horizontal scroll bar if(mainFrame.getAttribute('minHeight') && calculatedHeight < mainFrame.getAttribute('minHeight')) calculatedHeight = mainFrame.getAttribute('minHeight'); else if (calculatedHeight < 0) calculatedHeight = 0; /* window.status = "window.innerHeight: " + window.innerHeight + " / document.body.scrollHeight: " + document.body.scrollHeight + " / document.body.offsetHeight: " + document.body.offsetHeight + " / mainFrame.offsetHeight: " + mainFrame.offsetHeight + " / excess: " + excess + " / calculatedHeight: " + calculatedHeight; */ mainFrame.style.height = calculatedHeight + "px"; // Change height of all divs (they have the overflow-y: auto) var tabBodies = document.getElementsByTagName("div"); for(var i = 0; i < tabBodies.length; i++) { if(tabBodies[i].getAttribute('tab') && tabBodies[i].style.overflowY == "auto") tabBodies[i].style.height = calculatedHeight + "px"; } } } if(navigator.appVersion.indexOf('MSIE 5.0') != -1) // If IE 5.0 { var activeTab = document.getElementById(currentTabId); if(activeTab != null) showTab(activeTab); // Fix for IE 5.0 } } /** * Show the full main frame. Used onbeforeprint */ function showFullMainFrame() { if(currentTabId != null) // If a tab has been selected { var activeTab = document.getElementById(currentTabId); // Find active tab if(activeTab.style.overflowY == 'auto') { activeTab.style.overflowY = 'visible'; activeTab.setAttribute('wasScrolling', true); } } //Hide menu before printing hideMenus(); } function hideMenus() { var allMenus = document.getElementById('allMenus'); if(allMenus != null) allMenus.style.display = "none"; } /** * Make the main frame scrollable again after printing. Used onafterprint */ function makeMainFrameScrollable() { if(currentTabId != null) // If a tab has been selected { var activeTab = document.getElementById(currentTabId); // Find active tab if(activeTab.getAttribute('wasScrolling')) activeTab.style.overflowY = 'auto'; } //Show the hidden menus again! var allMenus = document.getElementById('allMenus'); if(allMenus != null) allMenus.style.display = ""; } //
//A note to the reader! // // In this lib do not get confused by the reference to this.className It IS in fact // the Instance (AND the Class.) function DatePickerControl() { } /** * Creates the calendar's div element and the button into the input-texts with * attibute datepicker="true" */ DatePickerControl.init = function() { // try to create the DatePickerControl.container: if (!document.getElementById("CalendarPickerControl")){ this.calBG = null; if (this.useTrickyBG){ // Creates a tricky bg to hide the select controls (IE bug). // We use a iframe element, because is one of the elements that can // stay on top of select controls. // I don't like this solution, but IE is a pseudo-browser for developers this.calBG = document.createElement("iframe"); this.calBG.id = "CalendarPickerControlBG"; // Style properties set here on purpose due to IE behaviour this.calBG.style.zIndex = "49999"; // below calcontainer this.calBG.style.position = "absolute"; this.calBG.style.display = "none"; this.calBG.style.border = "0px solid transparent"; document.body.appendChild(this.calBG); } this.calContainer = document.createElement("div"); this.calContainer.id = "CalendarPickerControl"; // Style properties set here on purpose due to IE behaviour this.calContainer.style.zIndex = "50000"; this.calContainer.style.position = "absolute"; this.calContainer.style.display = "none"; this.calContainer.style.width = this.datePickerControlWidth; document.body.appendChild(this.calContainer); if (this.calContainer.addEventListener){ this.calContainer.addEventListener("click", DPC_onContainerClick, false); window.addEventListener("resize", DPC_onWindowResize, false); } else if (this.calContainer.attachEvent){ this.calContainer.attachEvent("onclick", DPC_onContainerClick); window.attachEvent("onresize", DPC_onWindowResize); } } /** looking for input controls that will be transformed into DatePickerControls. */ jq('input[datepicker="true"]').each(function(index,domEle) { var templateLevel = domEle.getAttribute('templateLevel'); if(templateLevel == null) { doAttachDatePickerControlToInput(domEle) } }); /** When new rows are created, check for items to add DatePickerControl to*/ addNewRowEventHandler(function(event, newRowNumber, myNewRow) { attachDatePickerControlToInput(myNewRow); // Attach DatePickerControl }); } /** * Helpfunction to attach DatepickerControl to an input */ function attachDatePickerControlToInput(context) { jq('input[datepicker="true"]', context).each(function () { doAttachDatePickerControlToInput(this); })}; /** * Helperfunction that attaches eventListener to the Control */ function doAttachDatePickerControlToInput(editctrl) { if (! editctrl.id){ alert("Attribute 'id' is mandatory for DatePickerControl."); return false; } if (!DatePickerControl.createButton(editctrl)) return false; // Will there be a button displayed? editctrl.setAttribute("isdatepicker", "true"); if(editctrl.addEventListener){ editctrl.addEventListener("keyup", DPC_onEditControlKeyUp, false); editctrl.addEventListener("blur", DPC_onEditControlBlur, false); editctrl.addEventListener("focus", DPC_onEditControlFocus, false); editctrl.addEventListener("keydown", DPC_onEditControlKeyDown, false); } else if (editctrl.attachEvent){ editctrl.attachEvent("onkeyup", DPC_onEditControlKeyUp); editctrl.attachEvent("onblur", DPC_onEditControlBlur); editctrl.attachEvent("onfocus", DPC_onEditControlFocus); editctrl.attachEvent("onkeydown", DPC_onEditControlKeyDown); } return true; } /** * Wrapper for init() */ function DPC_autoInit() { DatePickerControl.init(); } if (window.addEventListener){ window.addEventListener("load", DPC_autoInit, false); } else if (window.attachEvent){ window.attachEvent("onload", DPC_autoInit); } /** * Creates the calendar button for a text-input control and assign some attributes. * @param input The associated text-input to create the button. * @param useId Specify if you want to use the Id of input control to obtain the format * @return true is the control has been created, otherwise false */ DatePickerControl.createButton = function(input) { var newid = this.buttonIdPrefix + input.id; if (document.getElementById(newid)) return false; // if exists previously.... // set the date format fmt = input.getAttribute("datepicker_format"); if (!fmt) fmt = this.defaultFormat; input.setAttribute("datepicker_format", fmt); input.setAttribute("maxlength", fmt.length); // Get min/max used for bracketing input.setMinDate = function(d){this.setAttribute("datepicker_min", d);} input.setMaxDate = function(d){this.setAttribute("datepicker_max", d);} // Create the button. ie an img var calButton = document.createElement('img'); calButton.id = newid; calButton.title = this.buttonTitle; // Set some attributes to remember the text-input associated // with this button and its format: calButton.setAttribute("datepicker_inputid", input.id); calButton.setAttribute("datepicker_format", fmt); // Add the event listeners: if (calButton.addEventListener){ calButton.addEventListener("click", DPC_onButtonClick, false); } else if (calButton.attachEvent){ calButton.attachEvent("onclick", DPC_onButtonClick); } // Set the style and position: calButton.className = "calendarbutton"; calButton.src = "/html/images/buttons/calendar.gif"; calButton.style.left = "-" + calButton.offsetWidth; jq(input).wrap(' '); jq(input).after(' ').after(jq(calButton)).after(" "); // everything is ok return true; } /** * Show the calendar */ DatePickerControl.show = function() { if (!this.displayed){ var input = this.inputControl; if (input == null) return; // Do NOT allow change for inputs that are readonly if (input.readOnly) { input.value = this.originalValue; return; } if (input.disabled) return; // just in case ;) var calframe = document.getElementById(this.calFrameId); this.setCalContainerPosition(input); this.calContainer.style.visibility = "visible"; this.calContainer.style.display = "block"; this.calContainer.style.height = calframe.offsetHeight; if (this.calBG){ // the ugly patch for IE this.calBG.style.top = this.calContainer.style.top; this.calBG.style.left = this.calContainer.style.left; this.calBG.style.visibility = "visible"; this.calBG.style.display = "block"; this.calBG.style.width = this.calContainer.offsetWidth; this.calBG.style.height = calframe.offsetHeight; } this.displayed = true; // setFocus(input); // jq(this).focus(); } } //----------------------------------------------------------------------------- /** * Hide the calendar */ DatePickerControl.hide = function() { if (this.displayed){ this.calContainer.style.visibility = "hidden"; this.calContainer.style.left = -1000; // some problems with overlaped controls this.calContainer.style.top = -1000; if (this.calBG){ // the ugly patch for IE this.calBG.style.visibility = "hidden"; this.calBG.style.left = -1000; this.calBG.style.top = -1000; } this.inputControl.value = this.originalValue; this.displayed = false; } } /** * Gets the name of a numbered month */ DatePickerControl.getMonthName = function(monthNumber) { return this.Months[monthNumber]; } /** * Obtains the days of a given month and year */ DatePickerControl.getDaysOfMonth = function(monthNo, p_year) { if (this.isLeapYear(p_year)){ return this.lDOMonth[monthNo]; } else { return this.DOMonth[monthNo]; } } /** * Will return an 1-D array with 1st element being the calculated month * and second being the calculated year after applying the month increment/decrement * as specified by 'incr' parameter. 'incr' will normally have 1/-1 to navigate thru * the months. */ DatePickerControl.calcMonthYear = function(p_Month, p_Year, incr) { var ret_arr = new Array(); if (incr == -1) { if (p_Month == 0) { ret_arr[0] = 11; ret_arr[1] = parseInt(p_Year) - 1; } else { ret_arr[0] = parseInt(p_Month) - 1; ret_arr[1] = parseInt(p_Year); } } else if (incr == 1) { if (p_Month == 11) { ret_arr[0] = 0; ret_arr[1] = parseInt(p_Year) + 1; } else { ret_arr[0] = parseInt(p_Month) + 1; ret_arr[1] = parseInt(p_Year); } } return ret_arr; } /** * Gets the DatePickerControl HTML code */ DatePickerControl.getAllCode = function() { var vCode = ""; vCode += "
"; vCode += this.getHeaderCode(); vCode += this.getDaysHeaderCode(); vCode += this.getDaysCode(); vCode += "
"; return vCode; } //----------------------------------------------------------------------------- /** * The title and nav buttons */ DatePickerControl.getHeaderCode = function() { var prevMMYYYY = this.calcMonthYear(this.month, this.year, -1); var prevMM = prevMMYYYY[0]; var prevYYYY = prevMMYYYY[1]; var nextMMYYYY = this.calcMonthYear(this.month, this.year, 1); var nextMM = nextMMYYYY[0]; var nextYYYY = nextMMYYYY[1]; var gNow = new Date(); var vCode = ""; var numberCols = this.weekNumber ? 8 : 7; vCode += ""; vCode += this.monthName + "  "; vCode += "«"; vCode += " " + this.year + " "; vCode += "»"; vCode += ""; vCode += ""; vCode += ""; vCode += ""; vCode += ""; vCode += ""; vCode += ""; vCode += ""; return vCode; } /** * The days' name headers */ DatePickerControl.getDaysHeaderCode = function(){ var vCode = ""; vCode = vCode + ""; if (this.weekNumber){ vCode += " " } for (i=this.firstWeekDay; i"; } vCode = vCode + ""; return vCode; } /** * The days numbers code */ DatePickerControl.getDaysCode = function() { var vDate = new Date(); vDate.setDate(1); vDate.setMonth(this.month); vDate.setFullYear(this.year); var vFirstDay = vDate.getDay(); // Get day of week var vDay = 1; // Constant for the value 1 var vLastDay = this.getDaysOfMonth(this.month, this.year); var vOnLastDay = 0; var vCode = "";// Constant for the value "" this.dayOfWeek = vFirstDay; var prevm = this.month == 0 ? 11 : this.month-1; var prevy = this.prevm == 11 ? this.year - 1 : this.year; prevmontdays = this.getDaysOfMonth(prevm, prevy); // Get the number of days in PREVIOUS Month vFirstDay = (vFirstDay == 0 && this.firstWeekDay) ? 7 : vFirstDay; if (this.weekNumber){ var week = this.getWeekNumberSV(this.year, this.month, 1); var yearsLastWeekNumber = this.getWeekNumberSV(this.year, 11, 31); if (yearsLastWeekNumber == 1) yearsLastWeekNumber = 52; } vCode += ""; if (this.weekNumber){ vCode += "" + week + ""; } // Write the last days of previous month for (i=this.firstWeekDay; i" + (prevmontdays-vFirstDay+i+1) + ""; } // Write rest of the 1st week for (j=vFirstDay-this.firstWeekDay; j<7; j++) { if (this.isInRange(vDay)){ classname = this.getDayClass(vDay, j); vCode += "" + vDay + ""; } else{ vCode += "" + vDay + ""; } vDay++; } vCode += ""; // Write the rest of the weeks hence beginning on k=2 for (k=2; k<7; k++){ vCode = vCode + ""; if (this.weekNumber){ week++; if (week > yearsLastWeekNumber) week = 1; //Roll over weeknumber vCode += "" + week + ""; } for (j=0; j<7; j++){ if (this.isInRange(vDay)){ classname = this.getDayClass(vDay, j); vCode += "" + vDay + ""; } else{ vCode += "" + vDay + ""; } vDay++; if (vDay > vLastDay){ vOnLastDay = 1; break; } } // If week is full (j is then 7, ) if (j == 7) { vCode += ""; } if (vOnLastDay == 1) break; } // Fill up the rest of last week for (m=1; m<(7-j); m++){ vCode += "" + m + ""; } // Add a closing for last row vCode += ""; return vCode; } /** * Get the class according if is 'today', the 'current' date at the control, * a 'weekend' day, or a 'normal' day. * @param vday The number of the day in the current month and year * @param dayofweek The number of the day within the week (0..6) */ DatePickerControl.getDayClass = function(vday, dayofweek) { var gNow = new Date(); var vNowDay = gNow.getDate(); var vNowMonth = gNow.getMonth(); var vNowYear = gNow.getFullYear(); if (vday == vNowDay && this.month == vNowMonth && this.year == vNowYear){ return "today"; } else{ // transform the day acording the specified first day of week var realdayofweek = (7 + dayofweek + this.firstWeekDay) % 7; for (i=0; i= 30){ year += 1900; } else{ year += 2000; } } return year; } /** * Writes the specified date in the control and close the calendar. */ DatePickerControl.writeDate = function(day) { var d = this.formatData(day); this.inputControl.value = d; this.originalValue = d; this.hide(); if (DatePickerControl.onSelect) DatePickerControl.onSelect(this.inputControl.id); this.firstFocused = true; if ( typeof this.inputControl.onchange != 'undefined') { this.inputControl.onchange(); } // setFocus(this.inputControl); jq(this.inputControl).focus(); } /** * Writes the current date in the control */ DatePickerControl.writeCurrentDate = function() { var d = this.formatData(this.currentDay); this.inputControl.value = d; } /** * Creates and write the calendar's code * @param m The month to build * @param y The year to build */ DatePickerControl.build = function(m, y) { var currentMonth = this.month; var currentYear = this.year; var calframe = document.getElementById(this.calFrameId); if (m==null){ var now = new Date(); this.month = now.getMonth(); this.year = now.getFullYear(); } else{ this.month = m; this.year = y; } // validate range if (!this.isInRange(null)){ this.month = currentMonth; this.year = currentYear; } if (!this.isInRange(this.currentDay)){ if (this.minDate && this.currentDay < this.minDate.getDate()) this.currentDay = this.minDate.getDate(); if (this.maxDate && this.currentDay > this.maxDate.getDate()) this.currentDay = this.maxDate.getDate(); } this.monthName = this.Months[this.month]; var code = this.getAllCode(); writeLayer(this.calContainer.id, null, code); if (this.calContainer && calframe) this.calContainer.style.height = calframe.offsetHeight; this.firstFocused = true; // setFocus(this.inputControl); jq(this.inputControl).focus(); this.selectDay(this.currentDay); } /** * Build the prev month calendar */ DatePickerControl.buildPrev = function() { if (!this.displayed) return; var prevMMYYYY = this.calcMonthYear(this.month, this.year, -1); var prevMM = prevMMYYYY[0]; var prevYYYY = prevMMYYYY[1]; this.build(prevMM, prevYYYY); } /** * Build the next month calendar */ DatePickerControl.buildNext = function() { if (!this.displayed) return; var nextMMYYYY = this.calcMonthYear(this.month, this.year, 1); var nextMM = nextMMYYYY[0]; var nextYYYY = nextMMYYYY[1]; this.build(nextMM, nextYYYY); } /** * Today button action */ DatePickerControl.selectToday = function() { var now = new Date(); var today = now.getDate(); if (!this.isInRange(today)) return; // If the day not is within min/max bracket if (this.closeOnTodayBtn){ this.currentDay = today; this.writeDate(this.currentDay); } else{ this.selectDay(today); } } //----------------------------------------------------------------------------- /** * Select a specific day */ DatePickerControl.selectDay = function(day) { if (!this.displayed) return; if (!this.isInRange(day)){ return; } var max = this.getDaysOfMonth(this.month, this.year); if (day > max) return; var newDayObject = document.getElementById(this.dayIdPrefix+day); var currentDayObject = document.getElementById(this.dayIdPrefix+this.currentDay); // TODO: Kolla om en skal in i andra if (newDayObject){ if (currentDayObject){ currentDayObject.className = currentDayObject.getAttribute("class_orig"); } newDayObject.className = "current"; this.currentDay = day; this.writeCurrentDate(); } } //----------------------------------------------------------------------------- /** * Select the prev week day * @param decr Use 1 for yesterday or 7 for prev week */ DatePickerControl.selectPrevDay = function(decr) { if (!this.displayed) return; var currentDay = this.currentDay; var max = this.getDaysOfMonth(this.month, this.year); var prev = currentDay - decr; if ( prev <= 0 ){ if (decr == 7){ currentDay += 27; prev = currentDay > max ? currentDay - 7 : currentDay; } else{ prev = max; } } this.selectDay(prev); } /** * Select the next week day * @param decr Use 1 for tomorrow or 7 for next week */ DatePickerControl.selectNextDay = function(incr) { if (!this.displayed) return; var currentDay = this.currentDay; var max = this.getDaysOfMonth(this.month, this.year); var nextDay = currentDay + incr; if ( nextDay > max ){ if (incr == 7){ currentDay = ((currentDay + this.dayOfWeek) % 7) - this.dayOfWeek; nextDay = currentDay < 0 ? currentDay+7 : currentDay; nextDay++; } else{ nextDay = 1; } } this.selectDay(nextDay); } /** * Show the calendar for an edit control */ DatePickerControl.showForEdit = function(edit) { if (this.displayed) return; if (edit == null) return; if (edit.disabled) return; this.inputControl = edit; this.originalValue = edit.value; this.setupRange(); // the format var format = this.inputControl.getAttribute("datepicker_format"); if (format == null) format = this.defaultFormat; this.format = format; // build with the current date in the control? if (this.validateDate(edit.value, format)){ var date = this.getDateFromControl(); this.currentDate = date; var year = date.getFullYear(); if (year < 1000) { this.currentDay = new Date().getDate(); this.build(null ,null); } else { this.build(date.getMonth(), date.getFullYear()); this.currentDay = date.getDate(); } } else{ this.currentDay = new Date().getDate(); this.build(null, null); } var currentDayObject = document.getElementById(this.dayIdPrefix+this.currentDay); if (currentDayObject) currentDayObject.className = "current"; this.writeCurrentDate(); // and finally this.show(); } /** * Determine if a given day (with current month and year) is in range * according to the min and max limit. * @param day The number of the day; if null then the current month is validated. */ DatePickerControl.isInRange = function(day) { if (!this.minDate && !this.maxDate) return true; if (day){ var aDate = new Date(); aDate.setFullYear(this.year); aDate.setMonth(this.month); aDate.setDate(day); if (this.minDate){ if (this.compareDates(aDate, this.minDate) < 0) return false; } if (this.maxDate){ if (this.compareDates(aDate, this.maxDate) > 0) return false; } } else{ // validate only the month. var currentym = parseInt(this.year.toString() + zeroPad(this.month.toString(),2)); var month; if (this.minDate){ month = this.minDate.getMonth(); var minym = parseInt(this.minDate.getFullYear().toString() + zeroPad(month.toString(),2)); if (currentym < minym) return false; } if (this.maxDate){ month = this.maxDate.getMonth(); var maxym = parseInt(this.maxDate.getFullYear().toString() + zeroPad(month.toString(),2)); if (currentym > maxym) return false; } } return true; } /** * Set up the Range ie. date between min and max */ DatePickerControl.setupRange = function() { // get the date range var edit = this.inputControl; var format = edit.getAttribute("datepicker_format"); var min = edit.getAttribute("datepicker_min"); this.minDate = min ? this.getDateFromString(min, format) : null; var max = edit.getAttribute("datepicker_max"); this.maxDate = max ? this.getDateFromString(max, format) : null; if (this.maxDate && this.minDate){ if (this.maxDate.getTime() < this.minDate.getTime()){ var tmp = this.maxDate; this.maxDate = this.minDate; this.minDate = tmp; } } } /** * Compare two dates. We cannot use Date getTime() mehtod in some specific cases, * so we use a trick with the string year+month+day transformed in a number. * @return '<0' if d1 < d2 '0' if d1 = d2 '>0' if d1 > d2 */ DatePickerControl.compareDates = function(d1, d2) { var month = d1.getMonth(); var date = d1.getDate(); var date1AsString = d1.getFullYear().toString() + zeroPad(month.toString(),2) +zeroPad (d.toString(),2); month = d2.getMonth(); d = d2.getDate(); var date1AsString = d2.getFullYear().toString() + zeroPad(month.toString(),2) + zeroPad (d.toString(),2); var n1 = parseInt(date1AsString); var n2 = parseInt(date2AsString); return n1-n2; } /** * Validate a string according to a date format. * Portions of code from: http://www.rgagnon.com/jsdetails/js-0063.html * @param strdate The string with the date. * @param format The format to validate. * @return true if succesfull or false otherwise */ DatePickerControl.validateDate = function(strdate, format) { var dateRegExp; var separator; var d, m, y; var currentDay = this.currentDay var currentMonth = this.month var currentYear = this.year; if (strdate == "") return false; // use the correct regular expresion... if (format.substring(0,1).toUpperCase() == "D"){ dateRegExp = /^\d{1,2}(\-|\/|\.)\d{1,2}\1\d{2,4}$/ } else if (format.substring(0,1).toUpperCase() == "Y"){ dateRegExp = /^\d{4}(\-|\/|\.)\d{1,2}\1\d{1,2}$/ // Only allow for years with 4 digits.. } else if (format.substring(0,1).toUpperCase() == "M"){ dateRegExp = /^\d{1,2}(\-|\/|\.)\d{1,2}\1\d{2,4}$/ } // is ok at least with the format? if (!dateRegExp.test(strdate)){ return false; } // check for a valid day month day combination separator = (strdate.indexOf("/") > 1) ? "/" : ((strdate.indexOf("-") > 1) ? "-" : "."); var datearray = strdate.split(separator); // get the number of date elements if (format.substring(0,1).toUpperCase() == "D"){ d = parseFloat(datearray[0]); m = parseFloat(datearray[1]); y = parseFloat(datearray[2]); } else if (format.substring(0,1).toUpperCase() == "Y"){ d = parseFloat(datearray[2]); m = parseFloat(datearray[1]); y = parseFloat(datearray[0]); } else if (format.substring(0,1).toUpperCase() == "M"){ d = parseFloat(datearray[1]); m = parseFloat(datearray[0]); y = parseFloat(datearray[2]); } // is a valid month? if (m<1 || m>12) return false; //check if month value and day value agree if (d > this.getDaysOfMonth(m-1, y)) return false; if (y <= 1900) return false; // Conform to Exder / // ok, date is valid... but is it in range? try { this.month = m; this.year = y; var res = this.isInRange(d); } catch (e) { } this.month = currentMonth; this.year = currentYear; return res; } /** * Check if a year is leap: * 1.Years evenly divisible by four are normally leap years, except for... * 2.Years also evenly divisible by 100 are not leap years, except for... * 3.Years also evenly divisible by 400 are leap years. * @return true if the year is leap or false otherwise. */ DatePickerControl.isLeapYear = function(year) { if ((year % 4) == 0){ if ((year % 100) == 0 && (year % 400) != 0){ return false; } return true; } return false; } /** * Click event for calendar button */ function DPC_onButtonClick(event){DatePickerControl.onButtonClick(event);} DatePickerControl.onButtonClick = function(event) { if (!this.displayed){ // get the button if (event == null) event = window.event; var button = (event.srcElement) ? event.srcElement : event.originalTarget; // gets the associated input: var input = document.getElementById(button.getAttribute("datepicker_inputid")); this.showForEdit(input); } else{ this.hide(); } } /** * Click event for calendar layer. */ function DPC_onContainerClick(event){DatePickerControl.onContainerClick(event);} DatePickerControl.onContainerClick = function(event) { if (event == null) event = window.event; if (this.hideTimeout){ clearTimeout(this.hideTimeout); this.hideTimeout = null; } // setFocus(this.inputControl); jq(this.inputControl).focus(); return false; } function DPC_onEditControlKeyDown(event){DatePickerControl.onEditControlKeyDown(event);} DatePickerControl.onEditControlKeyDown = function(event) { if (event == null) event = window.event; var kc = event.charCode ? event.charCode : event.which ? event.which : event.keyCode; if (this.displayed && kc == 9 ){ if (event.stopPropagation) event.stopPropagation(); if (event.preventDefault) event.preventDefault(); event.returnValue = false; event.cancelBubble = true; return false; } } /** * Key-up event for edit controls as date-pickers */ function DPC_onEditControlKeyUp(event){DatePickerControl.onEditControlKeyUp(event);} DatePickerControl.onEditControlKeyUp = function(event) { if (event == null) event = window.event; var edit = event.srcElement ? event.srcElement : event.originalTarget; var kc = event.charCode ? event.charCode : event.which ? event.which : event.keyCode; switch (kc){ case 37: // left arrow key if (this.displayed){ this.selectPrevDay(1); } break; case 38: // up arrow key this.selectPrevDay(7); break; case 39: // right arrow key this.selectNextDay(1); break; case 40: // down arrow key if (!this.displayed){ this.showForEdit(edit); } else{ this.selectNextDay(7); break; } break; case 27: // escape key this.hide(); break; case 33: // PageUp if (event.shiftKey){ this.build(this.month, parseInt(this.year)-1); } else{ this.buildPrev(); } break; case 34: // PageDown if (event.shiftKey){ this.build(this.month, parseInt(this.year)+1); } else{ this.buildNext(); } break; case 13: // enter-key (forms without submit buttons) if (this.displayed && this.currentDay > 0){ this.writeDate(this.currentDay); } break; } return false; } /** * Blur event for edit controls as date-pickers */ function DPC_onEditControlBlur(event){DatePickerControl.onEditControlBlur(event);} DatePickerControl.onEditControlBlur = function(event) { if (event == null) event = window.event; if (!this.hideTimeout){ this.hideTimeout = setTimeout("DatePickerControl.hide()", this.HIDE_TIMEOUT); } this.firstFocused = false; this.hideCauseBlur = true; } /** * Focus event for edit controls as date-pickers */ function DPC_onEditControlFocus(event){DatePickerControl.onEditControlFocus(event);} DatePickerControl.onEditControlFocus = function(event) { if (event == null) event = window.event; var edit = (event.srcElement) ? event.srcElement : event.originalTarget; this.inputControl = edit; // regarding 41701 / 41643 / 41779 / 41881 // commenting this line SHOULD fix a problem where a date might show on the field but the "onchange" doesnt trigger // and makes the field populated even if it shouldnt... // this.originalValue = edit.value; // when we are here edit.value has in some cases already been written to, uncorrectly saving the original value here this.setupRange(); if ((!this.displayed || this.hideCauseBlur) && this.autoShow && !this.firstFocused){ clearTimeout(this.hideTimeout); this.hideTimeout = null; this.firstFocused = true; if (this.hideCauseBlur){ this.hideCauseBlur = false; this.hide(); } this.showForEdit(edit); } else if (this.inputControl && this.inputControl.id != edit.id) { this.hide(); } else if (this.hideTimeout){ clearTimeout(this.hideTimeout); this.hideTimeout = null; } } //----------------------------------------------------------------------------- /** * Window resize event. */ function DPC_onWindowResize(event){DatePickerControl.onWindowResize(event);} DatePickerControl.onWindowResize = function(event) { this.relocate(); } /** * Relocate the calendar's frame */ DatePickerControl.relocate = function() { if (this.displayed){ var input = this.inputControl; if (input == null) return; this.setCalContainerPosition(input); if (this.calBG){ // the ugly patch for IE this.calBG.style.top = this.calContainer.style.top; this.calBG.style.left = this.calContainer.style.left; } } } /** * Helperfunction to set ContainerPosition */ DatePickerControl.setCalContainerPosition = function(input) { var top = getSize("offsetTop", input); var left = getSize("offsetLeft", input); this.calContainer.style.top = top + input.offsetHeight + this.offsetY + "px"; // find client browser width var windowWidth = jQuery(window).width(); // if calculated left position + offset setting + width of picker exceeds window width, adjust left position, // but only do if we found a windowWidth (which we always should) if(windowWidth != null && ( (left + this.offsetX + this.datePickerControlWidth) > windowWidth) ) { // set left property to window width - the width of calendar - some margin this.calContainer.style.left = document.body.clientWidth - this.datePickerControlWidth - 10 + "px"; } else // if not reaching out of window, set left property normally this.calContainer.style.left = left + this.offsetX + "px"; } /** * Gets the number of the week on the year for the given year, month, day. * ¨For SWEDISH Standard. */ DatePickerControl.getWeekNumberSV = function (year, month, day) { var when = new Date(year,month,day); var yearsFirstDay = new Date(year,0,1); var dayNumberYearsFirstDay = yearsFirstDay.getDay(); var yearsLastDay = new Date(year,11,31); var dayNumberYearsLastDay = yearsLastDay.getDay(); var dayNumberCurrentDay = ((Date.UTC(y2k(year),when.getMonth(),when.getDate(),0,0,0) - Date.UTC(y2k(year),0,1,0,0,0)) /1000/60/60/24) + 1; var weekNumber = Math.floor((dayNumberCurrentDay + 5 + dayNumberYearsFirstDay)/7); if (dayNumberYearsFirstDay > 4) weekNumber = weekNumber - 1; if ((weekNumber == 53 && dayNumberYearsLastDay < 4 )) weekNumber = 1; if (weekNumber == 0) { weekNumber = 52 if (dayNumberYearsFirstDay == 5) { weekNumber = 53; } else if ( (dayNumberYearsFirstDay == 4) && ( (when.getYear() - 1) % 4 == 0)) { weekNumber = 53; } } return weekNumber; } function y2k(number) { return (number < 1000) ? number + 1900 : number; } /** * Used to calculate the size when relocating * Based on code by: Mircho Mirev * Adjusted to take (nested) divs scroll positions into account * @param sParam * @param hLayer */ function getSize(sParam, hLayer) { nPos = 0; var scrollAmount = 0; if (sParam == 'offsetTop') { // Only get scroll when looking for vertical needs to be done before loop scrollAmount = getFrameScrollAmount(hLayer); } while ((hLayer.tagName) && !( /(body|html)/i.test(hLayer.tagName))){ nPos += eval('hLayer.' + sParam); if (sParam == 'offsetTop'){ if (hLayer.clientTop){ nPos += hLayer.clientTop; } } if (sParam == 'offsetLeft'){ if (hLayer.clientLeft){ nPos += hLayer.clientLeft; } } hLayer = hLayer.offsetParent; } nPos -= scrollAmount; // ONLY for vertical ie.offsetTop return nPos; } /** * Get Accumulated Scrollamount of all parentDivs for a given element * @param input */ function getFrameScrollAmount(input) { var scrollAmount = 0; jq(input).parents('div').each(function() { scrollAmount += this.scrollTop; }); return scrollAmount; } /** * Based on code by: Peter Todorov */ function writeLayer(ID, parentID, sText) { if (document.layers){ var oLayer; if(parentID){ oLayer = eval('document.' + parentID + '.document.' + ID + '.document'); } else{ oLayer = document.layers[ID].document; } oLayer.open(); oLayer.write(sText); oLayer.close(); } else if(document.all){ document.all[ID].innerHTML = sText; } else{ document.getElementById(ID).innerHTML = sText; } } // To help debugging
/** * * User: Gerry Askefalk * Date: 2011-sep-27, 2011-Nov-17 * Added handling of intervals with push-copy and delay of validation in from/to fields * when moving between those */ var lastFocus = []; // holds the last focused field /** * Attach events (browser-transparent via jQuery) for keydown. */ function attachKeyDown(context) { jq("input",context).each(function() { if( this.getAttribute('templateLevel') != null) // Avoid to add events to templateRows return; if( this.getAttribute('datepicker')) // Avoid keyDown event for datepickerfields return; jq( this).keydown(function(event) { if( (jq(this).hasClass('active'))) // Avoid auto-complete fields return; if (event.which == ARROW_DOWN) { // arrow-down see http://api.jquery.com/event.which/ event.preventDefault(); var searchButton = jq( this).next('a'); // Find the next element, given it is a link. if(searchButton) { var href = searchButton.attr('href'); if ( typeof searchButton.attr('href') != 'undefined') { if (href.length > 0) { searchButton.click(); // Trigger onClick to prepare search window.location.href = searchButton.attr('href'); // Follow the link, trick to get native click-action } } } }; }); }); }; jq(document).ready(function() { addNewRowEventHandler(function(event, newRowNumber, myNewRow) { attachKeyDown(myNewRow); // Attach handler to new row }); attachKeyDown(); /** * Bind a KeyUp handler for ALL interval fields (from-to), listen for Alt-Q, once found do a push-copy * bi-directional,works for either field in the interval. */ jq('input[isIntervalField="true"],select[isIntervalField="true"]').keyup(function(event) { if(event.altKey) if (event.which == KEY_Q) { var currentBaseField = jq(this).attr(FIELD_ID); // the base_name of the field i.e with _to/from excluded var currentFieldId = jq(this).attr(NAME); var targetFieldId = currentBaseField + (jq.isSubstring(currentFieldId,FROM) ? TO : FROM); if (jq(this).val() != "") jq("#" + targetFieldId).val(jq(this).val()); else jq(this).val(jq("#" + targetFieldId).val()); errorhandler.removeAllErrorsForField($(currentFieldId)); errorhandler.removeAllErrorsForField($(targetFieldId)); checkFromTo($(currentBaseField + FROM),$(currentBaseField + TO),event); errorhandler.markFieldWithError(this, true, true, event); } }); /** * Bind a Focus handler for ALL fields that executes interval sanity checking on interval fields (from-to) */ jq('input,select,radio,checkbox,textarea,a').focus(function(event) { if (lastFocus != null) { // If we already show an error skip further check to avoid never-ending loop over alert <> focus if( errorhandler.fieldHasAnyError(lastFocus) == false ) { if (jq(lastFocus).isFromToField()) { var fieldId = jq(lastFocus).attr(FIELD_ID); var currentFieldId = jq(this).attr(FIELD_ID); if (fieldId != currentFieldId) { // We have changed to another pair (..or another field) errorhandler.removeAllErrorsForField(lastFocus); checkFromTo($(fieldId + FROM),$(fieldId + TO),event); errorhandler.markFieldWithError(lastFocus, true, true, event); } else { // We have moved between from and to or vice versa errorhandler.removeAllErrorsForField(lastFocus); errorhandler.removeAllErrorsForField(this); } } } } lastFocus = this; //set lastFocused field to current field }); });