var DOKU_BASE = '/dokuwiki/';var DOKU_TPL = '/dokuwiki/lib/tpl/glfusion/';var DOKU_UHN = 0;var DOKU_UHC = 0;LANG = {"searchmedia":"Search for files","keepopen":"Keep window open on selection","hidedetails":"Hide Details","nosmblinks":"Linking to Windows shares only works in Microsoft Internet Explorer.\nYou still can copy and paste the link.","linkwiz":"Link Wizard","linkto":"Link to:","del_confirm":"Really delete selected item(s)?","mu_btn":"Upload multiple files at once","plugins":[]}; var toolbar = [{"type":"format","title":"Bold Text","icon":"bold.png","key":"b","open":"**","close":"**"},{"type":"format","title":"Italic Text","icon":"italic.png","key":"i","open":"\/\/","close":"\/\/"},{"type":"format","title":"Underlined Text","icon":"underline.png","key":"u","open":"__","close":"__"},{"type":"format","title":"Code Text","icon":"mono.png","key":"c","open":"''","close":"''"},{"type":"format","title":"Strike-through Text","icon":"strike.png","key":"d","open":"","close":"<\/del>"},{"type":"autohead","title":"Same Level Headline","icon":"hequal.png","key":"8","text":"Headline","mod":0},{"type":"autohead","title":"Lower Headline","icon":"hminus.png","key":"9","text":"Headline","mod":1},{"type":"autohead","title":"Higher Headline","icon":"hplus.png","key":"0","text":"Headline","mod":-1},{"type":"picker","title":"Select Headline","icon":"h.png","class":"pk_hl","list":[{"type":"format","title":"Level 1 Headline","icon":"h1.png","key":"1","open":"====== ","close":" ======\\n"},{"type":"format","title":"Level 2 Headline","icon":"h2.png","key":"2","open":"===== ","close":" =====\\n"},{"type":"format","title":"Level 3 Headline","icon":"h3.png","key":"3","open":"==== ","close":" ====\\n"},{"type":"format","title":"Level 4 Headline","icon":"h4.png","key":"4","open":"=== ","close":" ===\\n"},{"type":"format","title":"Level 5 Headline","icon":"h5.png","key":"5","open":"== ","close":" ==\\n"}]},{"type":"linkwiz","title":"Internal Link","icon":"link.png","key":"l","open":"[[","close":"]]"},{"type":"format","title":"External Link","icon":"linkextern.png","open":"[[","close":"]]","sample":"http:\/\/example.com|External Link"},{"type":"formatln","title":"Ordered List Item","icon":"ol.png","open":" - ","close":"","key":"-"},{"type":"formatln","title":"Unordered List Item","icon":"ul.png","open":" * ","close":"","key":"."},{"type":"insert","title":"Horizontal Rule","icon":"hr.png","insert":"\\n----\\n"},{"type":"mediapopup","title":"Add Images and other files","icon":"image.png","url":"lib\/exe\/mediamanager.php?ns=","name":"mediaselect","options":"width=750,height=500,left=20,top=20,scrollbars=yes,resizable=yes"},{"type":"picker","title":"Smileys","icon":"smiley.png","list":{"8-)":"icon_cool.gif","8-O":"icon_eek.gif","8-o":"icon_eek.gif",":-(":"icon_sad.gif",":-)":"icon_smile.gif","=)":"icon_smile2.gif",":-\/":"icon_doubt.gif",":-\\":"icon_doubt2.gif",":-?":"icon_confused.gif",":-D":"icon_biggrin.gif",":-P":"icon_razz.gif",":-o":"icon_surprised.gif",":-O":"icon_surprised.gif",":-x":"icon_silenced.gif",":-X":"icon_silenced.gif",":-|":"icon_neutral.gif",";-)":"icon_wink.gif","^_^":"icon_fun.gif",":?:":"icon_question.gif",":!:":"icon_exclaim.gif","LOL":"icon_lol.gif","FIXME":"fixme.gif","DELETEME":"delete.gif"},"icobase":"smileys"},{"type":"picker","title":"Special Chars","icon":"chars.png","list":["\u00c0","\u00e0","\u00c1","\u00e1","\u00c2","\u00e2","\u00c3","\u00e3","\u00c4","\u00e4","\u01cd","\u01ce","\u0102","\u0103","\u00c5","\u00e5","\u0100","\u0101","\u0104","\u0105","\u00c6","\u00e6","\u0106","\u0107","\u00c7","\u00e7","\u010c","\u010d","\u0108","\u0109","\u010a","\u010b","\u00d0","\u0111","\u00f0","\u010e","\u010f","\u00c8","\u00e8","\u00c9","\u00e9","\u00ca","\u00ea","\u00cb","\u00eb","\u011a","\u011b","\u0112","\u0113","\u0116","\u0117","\u0118","\u0119","\u0122","\u0123","\u011c","\u011d","\u011e","\u011f","\u0120","\u0121","\u0124","\u0125","\u00cc","\u00ec","\u00cd","\u00ed","\u00ce","\u00ee","\u00cf","\u00ef","\u01cf","\u01d0","\u012a","\u012b","\u0130","\u0131","\u012e","\u012f","\u0134","\u0135","\u0136","\u0137","\u0139","\u013a","\u013b","\u013c","\u013d","\u013e","\u0141","\u0142","\u013f","\u0140","\u0143","\u0144","\u00d1","\u00f1","\u0145","\u0146","\u0147","\u0148","\u00d2","\u00f2","\u00d3","\u00f3","\u00d4","\u00f4","\u00d5","\u00f5","\u00d6","\u00f6","\u01d1","\u01d2","\u014c","\u014d","\u0150","\u0151","\u0152","\u0153","\u00d8","\u00f8","\u0154","\u0155","\u0156","\u0157","\u0158","\u0159","\u015a","\u015b","\u015e","\u015f","\u0160","\u0161","\u015c","\u015d","\u0162","\u0163","\u0164","\u0165","\u00d9","\u00f9","\u00da","\u00fa","\u00db","\u00fb","\u00dc","\u00fc","\u01d3","\u01d4","\u016c","\u016d","\u016a","\u016b","\u016e","\u016f","\u01d6","\u01d8","\u01da","\u01dc","\u0172","\u0173","\u0170","\u0171","\u0174","\u0175","\u00dd","\u00fd","\u0178","\u00ff","\u0176","\u0177","\u0179","\u017a","\u017d","\u017e","\u017b","\u017c","\u00de","\u00fe","\u00df","\u0126","\u0127","\u00bf","\u00a1","\u00a2","\u00a3","\u00a4","\u00a5","\u20ac","\u00a6","\u00a7","\u00aa","\u00ac","\u00af","\u00b0","\u00b1","\u00f7","\u2030","\u00bc","\u00bd","\u00be","\u00b9","\u00b2","\u00b3","\u00b5","\u00b6","\u2020","\u2021","\u00b7","\u2022","\u00ba","\u2200","\u2202","\u2203","\u018f","\u0259","\u2205","\u2207","\u2208","\u2209","\u220b","\u220f","\u2211","\u203e","\u2212","\u2217","\u221a","\u221d","\u221e","\u2220","\u2227","\u2228","\u2229","\u222a","\u222b","\u2234","\u223c","\u2245","\u2248","\u2260","\u2261","\u2264","\u2265","\u2282","\u2283","\u2284","\u2286","\u2287","\u2295","\u2297","\u22a5","\u22c5","\u25ca","\u2118","\u2111","\u211c","\u2135","\u2660","\u2663","\u2665","\u2666","\u03b1","\u03b2","\u0393","\u03b3","\u0394","\u03b4","\u03b5","\u03b6","\u03b7","\u0398","\u03b8","\u03b9","\u03ba","\u039b","\u03bb","\u03bc","\u039e","\u03be","\u03a0","\u03c0","\u03c1","\u03a3","\u03c3","\u03a4","\u03c4","\u03c5","\u03a6","\u03c6","\u03c7","\u03a8","\u03c8","\u03a9","\u03c9","\u2605","\u2606","\u260e","\u261a","\u261b","\u261c","\u261d","\u261e","\u261f","\u2639","\u263a","\u2714","\u2718","\u00d7","\u201e","\u201c","\u201d","\u201a","\u2018","\u2019","\u00ab","\u00bb","\u2039","\u203a","\u2014","\u2013","\u2026","\u2190","\u2191","\u2192","\u2193","\u2194","\u21d0","\u21d1","\u21d2","\u21d3","\u21d4","\u00a9","\u2122","\u00ae","\u2032","\u2033","[","]","{","}","~","(",")","%","\u00a7","$","#","|","@"]},{"type":"signature","title":"Insert Signature","icon":"sig.png","key":"y"}]; /* XXXXXXXXXX begin of lib/scripts/helpers.js XXXXXXXXXX */ /** * Differrent helper functions * * @author Ilya Lebedev * @license LGPL */ //----------------------------------------------------------------------------- // Variable/property checks //----------------------------------------------------------------------------- /** * Checks if property is undefined * * @param {Object} prop value to check * @return {Boolean} true if matched * @scope public */ function isUndefined (prop /* :Object */) /* :Boolean */ { return (typeof prop == 'undefined'); } /** * Checks if property is function * * @param {Object} prop value to check * @return {Boolean} true if matched * @scope public */ function isFunction (prop /* :Object */) /* :Boolean */ { return (typeof prop == 'function'); } /** * Checks if property is string * * @param {Object} prop value to check * @return {Boolean} true if matched * @scope public */ function isString (prop /* :Object */) /* :Boolean */ { return (typeof prop == 'string'); } /** * Checks if property is number * * @param {Object} prop value to check * @return {Boolean} true if matched * @scope public */ function isNumber (prop /* :Object */) /* :Boolean */ { return (typeof prop == 'number'); } /** * Checks if property is the calculable number * * @param {Object} prop value to check * @return {Boolean} true if matched * @scope public */ function isNumeric (prop /* :Object */) /* :Boolean */ { return isNumber(prop)&&!isNaN(prop)&&isFinite(prop); } /** * Checks if property is array * * @param {Object} prop value to check * @return {Boolean} true if matched * @scope public */ function isArray (prop /* :Object */) /* :Boolean */ { return (prop instanceof Array); } /** * Checks if property is regexp * * @param {Object} prop value to check * @return {Boolean} true if matched * @scope public */ function isRegExp (prop /* :Object */) /* :Boolean */ { return (prop instanceof RegExp); } /** * Checks if property is a boolean value * * @param {Object} prop value to check * @return {Boolean} true if matched * @scope public */ function isBoolean (prop /* :Object */) /* :Boolean */ { return ('boolean' == typeof prop); } /** * Checks if property is a scalar value (value that could be used as the hash key) * * @param {Object} prop value to check * @return {Boolean} true if matched * @scope public */ function isScalar (prop /* :Object */) /* :Boolean */ { return isNumeric(prop)||isString(prop); } /** * Checks if property is empty * * @param {Object} prop value to check * @return {Boolean} true if matched * @scope public */ function isEmpty (prop /* :Object */) /* :Boolean */ { if (isBoolean(prop)) return false; if (isRegExp(prop) && new RegExp("").toString() == prop.toString()) return true; if (isString(prop) || isNumber(prop)) return !prop; if (Boolean(prop)&&false != prop) { for (var i in prop) if(prop.hasOwnProperty(i)) return false } return true; } /** * Checks if property is derived from prototype, applies method if it is not exists * * @param string property name * @return bool true if prototyped * @access public */ if ('undefined' == typeof Object.hasOwnProperty) { Object.prototype.hasOwnProperty = function (prop) { return !('undefined' == typeof this[prop] || this.constructor && this.constructor.prototype[prop] && this[prop] === this.constructor.prototype[prop]); } } /** * Very simplistic Flash plugin check, probably works for Flash 8 and higher only */ function hasFlash(version){ var ver = 0; try{ if(navigator.plugins != null && navigator.plugins.length > 0){ ver = navigator.plugins["Shockwave Flash"].description.split(' ')[2].split('.')[0]; }else{ var axo = new ActiveXObject("ShockwaveFlash.ShockwaveFlash"); ver = axo.GetVariable("$version").split(' ')[1].split(',')[0]; } }catch(e){ } if(ver >= version) return true; return false; } /* XXXXXXXXXX end of lib/scripts/helpers.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/scripts/events.js XXXXXXXXXX */ function removeEvent(element, type, handler) { // delete the event handler from the hash table if (element.events && element.events[type]) { delete element.events[type][handler.$$guid]; } }; function handleEvent(event) { var returnValue = true; // grab the event object (IE uses a global event object) event = event || fixEvent(window.event); // get a reference to the hash table of event handlers var handlers = this.events[event.type]; // execute each event handler for (var i in handlers) { if (!handlers.hasOwnProperty(i)) continue; this.$$handleEvent = handlers[i]; if (this.$$handleEvent(event) === false) { returnValue = false; } } return returnValue; }; function fixEvent(event) { // add W3C standard event methods event.preventDefault = fixEvent.preventDefault; event.stopPropagation = fixEvent.stopPropagation; // fix target event.target = event.srcElement; return event; }; fixEvent.preventDefault = function() { this.returnValue = false; }; fixEvent.stopPropagation = function() { this.cancelBubble = true; }; /** * Pseudo event handler to be fired after the DOM was parsed or * on window load at last. * * @author based upon some code by Dean Edwards * @author Dean Edwards * @link http://dean.edwards.name/weblog/2006/06/again/ */ window.fireoninit = function() { // quit if this function has already been called if (arguments.callee.done) return; // flag this function so we don't do the same thing twice arguments.callee.done = true; // kill the timer if (_timer) { clearInterval(_timer); _timer = null; } if (typeof window.oninit == 'function') { window.oninit(); } }; /** * Run the fireoninit function as soon as possible after * the DOM was loaded, using different methods for different * Browsers * * @author Dean Edwards * @link http://dean.edwards.name/weblog/2006/06/again/ */ // for Mozilla if (document.addEventListener) { document.addEventListener("DOMContentLoaded", window.fireoninit, null); } // for Internet Explorer (using conditional comments) /*@cc_on @*/ /*@if (@_win32) document.write("<\/script>"); var script = document.getElementById("__ie_init"); script.onreadystatechange = function() { if (this.readyState == "complete") { window.fireoninit(); // call the onload handler } }; /*@end @*/ // for Safari if (/WebKit/i.test(navigator.userAgent)) { // sniff var _timer = setInterval(function() { if (/loaded|complete/.test(document.readyState)) { window.fireoninit(); // call the onload handler } }, 10); } // for other browsers window.onload = window.fireoninit; /** * This is a pseudo Event that will be fired by the fireoninit * function above. * * Use addInitEvent to bind to this event! * * @author Andreas Gohr * @see fireoninit() */ window.oninit = function(){}; /** * Bind a function to the window.init pseudo event * * @author Simon Willison * @see http://simon.incutio.com/archive/2004/05/26/addLoadEvent */ function addInitEvent(func) { var oldoninit = window.oninit; if (typeof window.oninit != 'function') { window.oninit = func; } else { window.oninit = function() { oldoninit(); func(); }; } } /** * Bind variables to a function call creating a closure * * Use this to circumvent variable scope problems when creating closures * inside a loop * * @author Adrian Lang * @link http://www.cosmocode.de/en/blog/gohr/2009-10/15-javascript-fixing-the-closure-scope-in-loops * @param functionref fnc - the function to be called * @param mixed - any arguments to be passed to the function * @returns functionref */ function bind (fnc) { var args = Array.prototype.slice.call(arguments, 1); return function() { return fnc.apply(this, args); } } /* XXXXXXXXXX end of lib/scripts/events.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/scripts/cookie.js XXXXXXXXXX */ /** * Handles the cookie used by several JavaScript functions * * Only a single cookie is written and read. You may only save * sime name-value pairs - no complex types! * * You should only use the getValue and setValue methods * * @author Andreas Gohr */ DokuCookie = { data: Array(), name: 'DOKU_PREFS', /** * Save a value to the cookie * * @author Andreas Gohr */ setValue: function(key,val){ DokuCookie.init(); DokuCookie.data[key] = val; // prepare expire date var now = new Date(); DokuCookie.fixDate(now); now.setTime(now.getTime() + 365 * 24 * 60 * 60 * 1000); //expire in a year //save the whole data array var text = ''; for(var key in DokuCookie.data){ if (!DokuCookie.data.hasOwnProperty(key)) continue; text += '#'+escape(key)+'#'+DokuCookie.data[key]; } DokuCookie.setCookie(DokuCookie.name,text.substr(1),now,DOKU_BASE); }, /** * Get a Value from the Cookie * * @author Andreas Gohr */ getValue: function(key){ DokuCookie.init(); return DokuCookie.data[key]; }, /** * Loads the current set cookie * * @author Andreas Gohr */ init: function(){ if(DokuCookie.data.length) return; var text = DokuCookie.getCookie(DokuCookie.name); if(text){ var parts = text.split('#'); for(var i=0; i 0){ date.setTime(date.getTime() - skew); } } }; /* XXXXXXXXXX end of lib/scripts/cookie.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/scripts/script.js XXXXXXXXXX */ /** * Some of these scripts were taken from wikipedia.org and were modified for DokuWiki */ /** * Some browser detection */ var clientPC = navigator.userAgent.toLowerCase(); // Get client info var is_macos = navigator.appVersion.indexOf('Mac') != -1; var is_gecko = ((clientPC.indexOf('gecko')!=-1) && (clientPC.indexOf('spoofer')==-1) && (clientPC.indexOf('khtml') == -1) && (clientPC.indexOf('netscape/7.0')==-1)); var is_safari = ((clientPC.indexOf('AppleWebKit')!=-1) && (clientPC.indexOf('spoofer')==-1)); var is_khtml = (navigator.vendor == 'KDE' || ( document.childNodes && !document.all && !navigator.taintEnabled )); if (clientPC.indexOf('opera')!=-1) { var is_opera = true; var is_opera_preseven = (window.opera && !document.childNodes); var is_opera_seven = (window.opera && document.childNodes); } /** * Handy shortcut to document.getElementById * * This function was taken from the prototype library * * @link http://prototype.conio.net/ */ function $XXX() { var elements = new Array(); for (var i = 0; i < arguments.length; i++) { var element = arguments[i]; if (typeof element == 'string') element = document.getElementById(element); if (arguments.length == 1) return element; elements.push(element); } return elements; } /** * Simple function to check if a global var is defined * * @author Kae Verens * @link http://verens.com/archives/2005/07/25/isset-for-javascript/#comment-2835 */ function isset(varname){ return(typeof(window[varname])!='undefined'); } /** * Select elements by their class name * * @author Dustin Diaz * @link http://www.dustindiaz.com/getelementsbyclass/ */ function getElementsByClass(searchClass,node,tag) { var classElements = new Array(); if ( node == null ) node = document; if ( tag == null ) tag = '*'; var els = node.getElementsByTagName(tag); var elsLen = els.length; var pattern = new RegExp("(^|\\s)"+searchClass+"(\\s|$)"); for (var i = 0, j = 0; i < elsLen; i++) { if ( pattern.test(els[i].className) ) { classElements[j] = els[i]; j++; } } return classElements; } /** * Get the X offset of the top left corner of the given object * * @link http://www.quirksmode.org/index.html?/js/findpos.html */ function findPosX(object){ var curleft = 0; var obj = $(object); if (obj.offsetParent){ while (obj.offsetParent){ curleft += obj.offsetLeft; obj = obj.offsetParent; } } else if (obj.x){ curleft += obj.x; } return curleft; } //end findPosX function /** * Get the Y offset of the top left corner of the given object * * @link http://www.quirksmode.org/index.html?/js/findpos.html */ function findPosY(object){ var curtop = 0; var obj = $(object); if (obj.offsetParent){ while (obj.offsetParent){ curtop += obj.offsetTop; obj = obj.offsetParent; } } else if (obj.y){ curtop += obj.y; } return curtop; } //end findPosY function /** * Escape special chars in JavaScript * * @author Andreas Gohr */ function jsEscape(text){ var re=new RegExp("\\\\","g"); text=text.replace(re,"\\\\"); re=new RegExp("'","g"); text=text.replace(re,"\\'"); re=new RegExp('"',"g"); text=text.replace(re,'"'); re=new RegExp("\\\\\\\\n","g"); text=text.replace(re,"\\n"); return text; } /** * This function escapes some special chars * @deprecated by above function */ function escapeQuotes(text) { var re=new RegExp("'","g"); text=text.replace(re,"\\'"); re=new RegExp('"',"g"); text=text.replace(re,'"'); re=new RegExp("\\n","g"); text=text.replace(re,"\\n"); return text; } /** * Adds a node as the first childenode to the given parent * * @see appendChild() */ function prependChild(parent,element) { if(!parent.firstChild){ parent.appendChild(element); }else{ parent.insertBefore(element,parent.firstChild); } } /** * Prints a animated gif to show the search is performed * * Because we need to modify the DOM here before the document is loaded * and parsed completely we have to rely on document.write() * * @author Andreas Gohr */ function showLoadBar(){ document.write('...'); /* this does not work reliable in IE obj = $(id); if(obj){ obj.innerHTML = '...'; obj.style.display="block"; } */ } /** * Disables the animated gif to show the search is done * * @author Andreas Gohr */ function hideLoadBar(id){ obj = $(id); if(obj) obj.style.display="none"; } /** * Adds the toggle switch to the TOC */ function addTocToggle() { if(!document.getElementById) return; var header = $('toc__header'); if(!header) return; var toc = $('toc__inside'); var obj = document.createElement('span'); obj.id = 'toc__toggle'; obj.style.cursor = 'pointer'; if (toc && toc.style.display == 'none') { obj.innerHTML = '+'; obj.className = 'toc_open'; } else { obj.innerHTML = ''; obj.className = 'toc_close'; } prependChild(header,obj); obj.parentNode.onclick = toggleToc; try { obj.parentNode.style.cursor = 'pointer'; obj.parentNode.style.cursor = 'hand'; }catch(e){} } /** * This toggles the visibility of the Table of Contents */ function toggleToc() { var toc = $('toc__inside'); var obj = $('toc__toggle'); if(toc.style.display == 'none') { toc.style.display = ''; obj.innerHTML = ''; obj.className = 'toc_close'; } else { toc.style.display = 'none'; obj.innerHTML = '+'; obj.className = 'toc_open'; } } /** * Display an insitu footnote popup * * @author Andreas Gohr * @author Chris Smith */ function footnote(e){ var obj = e.target; var id = obj.id.substr(5); // get or create the footnote popup div var fndiv = $('insitu__fn'); if(!fndiv){ fndiv = document.createElement('div'); fndiv.id = 'insitu__fn'; fndiv.className = 'insitu-footnote JSpopup dokuwiki'; // autoclose on mouseout - ignoring bubbled up events adddwEvent(fndiv,'mouseout',function(e){ if(e.target != fndiv){ e.stopPropagation(); return; } // check if the element was really left if(e.pageX){ // Mozilla var bx1 = findPosX(fndiv); var bx2 = bx1 + fndiv.offsetWidth; var by1 = findPosY(fndiv); var by2 = by1 + fndiv.offsetHeight; var x = e.pageX; var y = e.pageY; if(x > bx1 && x < bx2 && y > by1 && y < by2){ // we're still inside boundaries e.stopPropagation(); return; } }else{ // IE if(e.offsetX > 0 && e.offsetX < fndiv.offsetWidth-1 && e.offsetY > 0 && e.offsetY < fndiv.offsetHeight-1){ // we're still inside boundaries e.stopPropagation(); return; } } // okay, hide it fndiv.style.display='none'; }); document.body.appendChild(fndiv); } // locate the footnote anchor element var a = $( "fn__"+id ); if (!a){ return; } // anchor parent is the footnote container, get its innerHTML var content = new String (a.parentNode.parentNode.innerHTML); // strip the leading content anchors and their comma separators content = content.replace(/.*<\/sup>/gi, ''); content = content.replace(/^\s+(,\s+)+/,''); // prefix ids on any elements with "insitu__" to ensure they remain unique content = content.replace(/\bid=\"(.*?)\"/gi,'id="insitu__$1'); // now put the content into the wrapper fndiv.innerHTML = content; // position the div and make it visible var x; var y; if(e.pageX){ // Mozilla x = e.pageX; y = e.pageY; }else{ // IE x = e.offsetX; y = e.offsetY; } fndiv.style.position = 'absolute'; fndiv.style.left = (x+2)+'px'; fndiv.style.top = (y+2)+'px'; fndiv.style.display = ''; } /** * Add the event handlers to footnotes * * @author Andreas Gohr */ addInitEvent(function(){ var elems = getElementsByClass('fn_top',null,'a'); for(var i=0; i * @author * @link http://news.hping.org/comp.lang.javascript.archive/12265.html * @link https://bugzilla.mozilla.org/show_bug.cgi?id=41464 */ function setWrap(textarea, wrapAttrValue){ textarea.setAttribute('wrap', wrapAttrValue); // Fix display for mozilla var parNod = textarea.parentNode; var nxtSib = textarea.nextSibling; parNod.removeChild(textarea); parNod.insertBefore(textarea, nxtSib); } /** * Handler to close all open Popups */ function closePopups(){ if(!document.getElementById){ return; } var divs = document.getElementsByTagName('div'); for(var i=0; i < divs.length; i++){ if(divs[i].className.indexOf('JSpopup') != -1){ divs[i].style.display = 'none'; } } } /** * Looks for an element with the ID scroll__here at scrolls to it */ function scrollToMarker(){ var obj = $('scroll__here'); if(obj) obj.scrollIntoView(); } /** * Looks for an element with the ID focus__this at sets focus to it */ function focusMarker(){ var obj = $('focus__this'); if(obj) obj.focus(); } /** * Remove messages */ function cleanMsgArea(){ var elems = getElementsByClass('(success|info|error)',document,'div'); if(elems){ for(var i=0; i */ addInitEvent(function(){ var revForm = $('page__revisions'); if (!revForm) return; var elems = revForm.elements; var countTicks = 0; for (var i=0; i= 2) input2.disabled = (input2.type=='checkbox' && !input2.checked); else input2.disabled = (input2.type!='checkbox'); } }); input1.checked = false; // chrome reselects on back button which messes up the logic } else if(input1.type=='submit'){ input1.disabled = true; } } }); /** * Add the event handler to the actiondropdown * * @author Andreas Gohr */ addInitEvent(function(){ var selector = $('action__selector'); if(!selector) return; adddwEvent(selector,'change',function(e){ this.form.submit(); }); $('action__selectorbtn').style.display = 'none'; }); /** * Display error for Windows Shares on browsers other than IE * * @author Michael Klier */ function checkWindowsShares() { if(!LANG['nosmblinks']) return true; var elems = getElementsByClass('windows',document,'a'); if(elems){ for(var i=0; i */ addInitEvent(function(){ checkWindowsShares(); }); /** * Highlight the section when hovering over the appropriate section edit button * * @author Andreas Gohr */ addInitEvent(function(){ var break_classes = new RegExp('secedit|toc|page'); var btns = getElementsByClass('btn_secedit',document,'form'); for(var i=0; i */ //prepare class function ajax_qsearch_class(){ this.sack = null; this.inObj = null; this.outObj = null; this.timer = null; } //create global object and add functions var ajax_qsearch = new ajax_qsearch_class(); ajax_qsearch.sack = new sack(DOKU_BASE + 'lib/exe/ajax.php'); ajax_qsearch.sack.AjaxFailedAlert = ''; ajax_qsearch.sack.encodeURIString = false; ajax_qsearch.init = function(inID,outID){ ajax_qsearch.inObj = document.getElementById(inID); ajax_qsearch.outObj = document.getElementById(outID); // objects found? if(ajax_qsearch.inObj === null){ return; } if(ajax_qsearch.outObj === null){ return; } // attach eventhandler to search field adddwEvent(ajax_qsearch.inObj,'keyup',ajax_qsearch.call); // attach eventhandler to output field adddwEvent(ajax_qsearch.outObj,'click',function(){ ajax_qsearch.outObj.style.display='none'; }); }; ajax_qsearch.clear = function(){ ajax_qsearch.outObj.style.display = 'none'; ajax_qsearch.outObj.innerHTML = ''; if(ajax_qsearch.timer !== null){ window.clearTimeout(ajax_qsearch.timer); ajax_qsearch.timer = null; } }; ajax_qsearch.exec = function(){ ajax_qsearch.clear(); var value = ajax_qsearch.inObj.value; if(value === ''){ return; } ajax_qsearch.sack.runAJAX('call=qsearch&q='+encodeURI(value)); }; ajax_qsearch.sack.onCompletion = function(){ var data = ajax_qsearch.sack.response; if(data === ''){ return; } ajax_qsearch.outObj.innerHTML = data; ajax_qsearch.outObj.style.display = 'block'; }; ajax_qsearch.call = function(){ ajax_qsearch.clear(); ajax_qsearch.timer = window.setTimeout("ajax_qsearch.exec()",500); }; /* XXXXXXXXXX end of lib/scripts/ajax.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/scripts/index.js XXXXXXXXXX */ /** * Javascript for index view * * @author Andreas Gohr */ var index = { /** * Delay in ms before showing the throbber. * Used to skip the throbber for fast AJAX calls. */ throbber_delay: 500, /** * Attach event handlers to all "folders" below the given element * * @author Andreas Gohr */ treeattach: function(obj){ if(!obj) return; var items = getElementsByClass('idx_dir',obj,'a'); for(var i=0; i * @author Ben Coburn */ toggle: function(e,clicky){ var listitem = clicky.parentNode.parentNode; listitem.open = !listitem.open; // listitem.open represents now the action to be done // if already open, close by removing the sublist var sublists = listitem.getElementsByTagName('ul'); if(!listitem.open){ if (sublists.length) { sublists[0].style.display='none'; } listitem.className='closed'; e.preventDefault(); return false; } // just show if already loaded if(sublists.length && listitem.open){ sublists[0].style.display=''; listitem.className='open'; e.preventDefault(); return false; } // prepare an AJAX call to fetch the subtree var ajax = new sack(DOKU_BASE + 'lib/exe/ajax.php'); ajax.AjaxFailedAlert = ''; ajax.encodeURIString = false; if(ajax.failed) return true; //prepare the new ul var ul = document.createElement('ul'); ul.className = 'idx'; timeout = window.setTimeout(function(){ // show the throbber as needed if (listitem.open) { ul.innerHTML = '
  • loading...
  • '; listitem.appendChild(ul); listitem.className='open'; } }, this.throbber_delay); ajax.elementObj = ul; ajax.afterCompletion = function(){ window.clearTimeout(timeout); index.treeattach(ul); if (listitem.className!='open') { if (!listitem.open) { ul.style.display='none'; } listitem.appendChild(ul); if (listitem.open) { listitem.className='open'; } } }; ajax.runAJAX(clicky.search.substr(1)+'&call=index'); e.preventDefault(); return false; } }; addInitEvent(function(){ index.treeattach($('index__tree')); }); /* XXXXXXXXXX end of lib/scripts/index.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/scripts/drag.js XXXXXXXXXX */ /** * Makes a DOM object draggable * * This is currently for movable DOM dialogs only. If needed it could be * extended to execute callbacks on special events... * * @link http://nofunc.org/Drag_Drop/ */ var drag = { obj: null, handle: null, oX: 0, // object X position oY: 0, // object Y position eX: 0, // event X delta eY: 0, // event Y delta /** * Attaches the needed handlers to the given object * * This can be called for multiple objects, the right one is later * determined from the event itself. The handle is optional * * @param DOMObject obj The object that should be draggable * @param DOMObject handle A handle on which the obj can be dragged */ attach: function (obj,handle) { if(handle){ handle.dragobject = obj; adddwEvent($(handle),'mousedown',drag.start); }else{ adddwEvent($(obj),'mousedown',drag.start); } }, /** * Starts the dragging operation */ start: function (e){ drag.handle = e.target; if(drag.handle.dragobject){ drag.obj = drag.handle.dragobject; }else{ drag.obj = drag.handle; } drag.handle.className += ' ondrag'; drag.obj.className += ' ondrag'; drag.oX = parseInt(drag.obj.style.left); drag.oY = parseInt(drag.obj.style.top); drag.eX = drag.evX(e); drag.eY = drag.evY(e); adddwEvent(document,'mousemove',drag.drag); adddwEvent(document,'mouseup',drag.stop); e.preventDefault(); e.stopPropagation(); return false; }, /** * Ends the dragging operation */ stop: function(){ drag.handle.className = drag.handle.className.replace(/ ?ondrag/,''); drag.obj.className = drag.obj.className.replace(/ ?ondrag/,''); removeEvent(document,'mousemove',drag.drag); removeEvent(document,'mouseup',drag.stop); drag.obj = null; drag.handle = null; }, /** * Moves the object during the dragging operation */ drag: function(e) { if(drag.obj) { drag.obj.style.top = (drag.evY(e)+drag.oY-drag.eY+'px'); drag.obj.style.left = (drag.evX(e)+drag.oX-drag.eX+'px'); } }, /** * Returns the X position of the given event. */ evX: function(e){ return (e.pageX) ? e.pageX : e.clientX + document.body.scrollTop; //fixme shouldn't this be scrollLeft? }, /** * Returns the Y position of the given event. */ evY: function(e){ return (e.pageY) ? e.pageY : e.clientY + document.body.scrollTop; } }; /* XXXXXXXXXX end of lib/scripts/drag.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/scripts/textselection.js XXXXXXXXXX */ /** * Text selection related functions. */ /** * selection prototype * * Object that capsulates the selection in a textarea. Returned by getSelection. * * @author Andreas Gohr */ function selection_class(){ this.start = 0; this.end = 0; this.obj = null; this.rangeCopy = null; this.scroll = 0; this.fix = 0; this.getLength = function(){ return this.end - this.start; }; this.getText = function(){ if(!this.obj) return ''; return this.obj.value.substring(this.start,this.end); } } /** * Get current selection/cursor position in a given textArea * * @link http://groups.drupal.org/node/1210 * @author Andreas Gohr * @link http://linebyline.blogspot.com/2006/11/textarea-cursor-position-in-internet.html * @returns object - a selection object */ function getSelection(textArea) { var sel = new selection_class(); sel.obj = textArea; sel.start = textArea.value.length; sel.end = textArea.value.length; textArea.focus(); if(document.getSelection) { // Mozilla et al. sel.start = textArea.selectionStart; sel.end = textArea.selectionEnd; sel.scroll = textArea.scrollTop; } else if(document.selection) { // MSIE /* * This huge lump of code is neccessary to work around two MSIE bugs: * * 1. Selections trim newlines at the end of the code * 2. Selections count newlines as two characters */ // The current selection sel.rangeCopy = document.selection.createRange().duplicate(); var before_range = document.body.createTextRange(); before_range.moveToElementText(textArea); // Selects all the text before_range.setEndPoint("EndToStart", sel.rangeCopy); // Moves the end where we need it var before_finished = false, selection_finished = false; var before_text, selection_text; // Load the text values we need to compare before_text = before_range.text; selection_text = sel.rangeCopy.text; sel.start = before_text.length; sel.end = sel.start + selection_text.length; // Check each range for trimmed newlines by shrinking the range by 1 character and seeing // if the text property has changed. If it has not changed then we know that IE has trimmed // a \r\n from the end. do { if (!before_finished) { if (before_range.compareEndPoints("StartToEnd", before_range) == 0) { before_finished = true; } else { before_range.moveEnd("character", -1); if (before_range.text == before_text) { sel.start += 2; sel.end += 2; } else { before_finished = true; } } } if (!selection_finished) { if (sel.rangeCopy.compareEndPoints("StartToEnd", sel.rangeCopy) == 0) { selection_finished = true; } else { sel.rangeCopy.moveEnd("character", -1); if (sel.rangeCopy.text == selection_text) { sel.end += 2; } else { selection_finished = true; } } } } while ((!before_finished || !selection_finished)); // count number of newlines in str to work around stupid IE selection bug var countNL = function(str) { var m = str.split("\r\n"); if (!m || !m.length) return 0; return m.length-1; }; sel.fix = countNL(sel.obj.value.substring(0,sel.start)); } return sel; } /** * Set the selection * * You need to get a selection object via getSelection() first, then modify the * start and end properties and pass it back to this function. * * @link http://groups.drupal.org/node/1210 * @author Andreas Gohr * @param object selection - a selection object as returned by getSelection() */ function setSelection(selection){ if(document.getSelection){ // FF // what a pleasure in FF ;) selection.obj.setSelectionRange(selection.start,selection.end); if(selection.scroll) selection.obj.scrollTop = selection.scroll; } else if(document.selection) { // IE selection.rangeCopy.collapse(true); selection.rangeCopy.moveStart('character',selection.start - selection.fix); selection.rangeCopy.moveEnd('character',selection.end - selection.start); selection.rangeCopy.select(); } } /** * Inserts the given text at the current cursor position or replaces the current * selection * * @author Andreas Gohr * @param string text - the new text to be pasted * @param objct selecttion - selection object returned by getSelection * @param int opts.startofs - number of charcters at the start to skip from new selection * @param int opts.endofs - number of characters at the end to skip from new selection * @param bool opts.nosel - set true if new text should not be selected */ function pasteText(selection,text,opts){ if(!opts) opts = {}; // replace the content selection.obj.value = selection.obj.value.substring(0, selection.start) + text + selection.obj.value.substring(selection.end, selection.obj.value.length); // set new selection selection.end = selection.start + text.length; // modify the new selection if wanted if(opts.startofs) selection.start += opts.startofs; if(opts.endofs) selection.end -= opts.endofs; // no selection wanted? set cursor to end position if(opts.nosel) selection.start = selection.end; setSelection(selection); } /** * Format selection * * Apply tagOpen/tagClose to selection in textarea, use sampleText instead * of selection if there is none. * * @author Andreas Gohr */ function insertTags(textAreaID, tagOpen, tagClose, sampleText){ var txtarea = $(textAreaID); var selection = getSelection(txtarea); var text = selection.getText(); var opts; // don't include trailing space in selection if(text.charAt(text.length - 1) == ' '){ selection.end--; text = selection.getText(); } if(!text){ // nothing selected, use the sample text and select it text = sampleText; opts = { startofs: tagOpen.length, endofs: tagClose.length }; }else{ // place cursor at the end opts = { nosel: true }; } // surround with tags text = tagOpen + text + tagClose; // do it pasteText(selection,text,opts); } /** * Wraps around pasteText() for backward compatibility * * @author Andreas Gohr */ function insertAtCarret(textAreaID, text){ var txtarea = $(textAreaID); var selection = getSelection(txtarea); pasteText(selection,text,{nosel: true}); } /* XXXXXXXXXX end of lib/scripts/textselection.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/scripts/toolbar.js XXXXXXXXXX */ // used to identify pickers var pickercounter=0; /** * Create a toolbar * * @param string tbid ID of the element where to insert the toolbar * @param string edid ID of the editor textarea * @param array tb Associative array defining the buttons * @author Andreas Gohr */ function initToolbar(tbid,edid,tb){ var toolbar = $(tbid); if(!toolbar) return; var edit = $(edid); if(!edit) return; if(edit.readOnly) return; //empty the toolbar area: toolbar.innerHTML=''; var cnt = tb.length; for(var i=0; i assign it as onclick actionFunc = 'tb_'+tb[i]['type']; if( isFunction(window[actionFunc]) ){ adddwEvent(btn,'click', bind(window[actionFunc],btn,tb[i],edid)); toolbar.appendChild(btn); continue; } // type is a init function -> execute it actionFunc = 'addBtnAction'+tb[i]['type'].charAt(0).toUpperCase()+tb[i]['type'].substring(1); if( isFunction(window[actionFunc]) ){ if(window[actionFunc](btn, tb[i], edid)){ toolbar.appendChild(btn); } continue; } alert('unknown toolbar type: '+tb[i]['type']+' '+actionFunc); } // end for } /** * Button action for format buttons * * @param DOMElement btn Button element to add the action to * @param array props Associative array of button properties * @param string edid ID of the editor textarea * @author Gabriel Birke * @author Andreas Gohr */ function tb_format(btn, props, edid) { var sample = props['title']; if(props['sample']){ sample = props['sample']; } insertTags(edid, fixtxt(props['open']), fixtxt(props['close']), fixtxt(sample)); pickerClose(); return false; } /** * Button action for format buttons * * This works exactly as tb_format() except that, if multiple lines * are selected, each line will be formatted seperately * * @param DOMElement btn Button element to add the action to * @param array props Associative array of button properties * @param string edid ID of the editor textarea * @author Gabriel Birke * @author Andreas Gohr */ function tb_formatln(btn, props, edid) { var sample = props['title']; if(props['sample']){ sample = props['sample']; } sample = fixtxt(sample); props['open'] = fixtxt(props['open']); props['close'] = fixtxt(props['close']); // is something selected? var opts; var selection = getSelection($(edid)); if(selection.getLength()){ sample = selection.getText(); opts = {nosel: true}; }else{ opts = { startofs: props['open'].length, endofs: props['close'].length }; } sample = sample.split("\n").join(props['close']+"\n"+props['open']); sample = props['open']+sample+props['close']; pasteText(selection,sample,opts); pickerClose(); return false; } /** * Button action for insert buttons * * @param DOMElement btn Button element to add the action to * @param array props Associative array of button properties * @param string edid ID of the editor textarea * @author Gabriel Birke * @author Andreas Gohr */ function tb_insert(btn, props, edid) { insertAtCarret(edid,fixtxt(props['insert'])); pickerClose(); return false; } /** * Button action for the media popup * * @param DOMElement btn Button element to add the action to * @param array props Associative array of button properties * @param string edid ID of the editor textarea * @author Andreas Gohr */ function tb_mediapopup(btn, props, edid) { window.open( DOKU_BASE+props['url']+encodeURIComponent(NS), props['name'], props['options']); return false; } /** * Button action for automatic headlines * * Insert a new headline based on the current section level * * @param DOMElement btn Button element to add the action to * @param array props Associative array of button properties * @param string edid ID of the editor textarea * @author Andreas Gohr */ function tb_autohead(btn, props, edid){ var lvl = currentHeadlineLevel(edid); // determine new level lvl += props['mod']; if(lvl < 1) lvl = 1; if(lvl > 5) lvl = 5; var tags = '='; for(var i=0; i<=5-lvl; i++) tags += '='; insertTags(edid, tags+' ', ' '+tags+"\n", props['text']); pickerClose(); return false; } /** * Add button action for picker buttons and create picker element * * @param DOMElement btn Button element to add the action to * @param array props Associative array of button properties * @param string edid ID of the editor textarea * @return boolean If button should be appended * @author Gabriel Birke */ function addBtnActionPicker(btn, props, edid) { var pickerid = 'picker'+(pickercounter++); createPicker(pickerid, props, edid); adddwEvent(btn,'click',function(){ pickerToggle(pickerid,btn); return false; }); return true; } /** * Add button action for the link wizard button * * @param DOMElement btn Button element to add the action to * @param array props Associative array of button properties * @param string edid ID of the editor textarea * @return boolean If button should be appended * @author Andreas Gohr */ function addBtnActionLinkwiz(btn, props, edid) { linkwiz.init($(edid)); adddwEvent(btn,'click',function(){ linkwiz.toggle(); return false; }); return true; } /** * Show/Hide a previosly created picker window * * @author Andreas Gohr */ function pickerToggle(pickerid,btn){ var picker = $(pickerid); if(picker.style.marginLeft == '-10000px'){ var x = findPosX(btn); var y = findPosY(btn); picker.style.left = (x+3)+'px'; picker.style.top = (y+btn.offsetHeight+3)+'px'; picker.style.marginLeft = '0px'; }else{ picker.style.marginLeft = '-10000px'; } } /** * Close all open pickers * * @author Andreas Gohr */ function pickerClose(){ var pobjs = getElementsByClass('picker'); for(var i=0; i */ /** * Creates a toolbar button through the DOM * * Style the buttons through the toolbutton class * * @author Andreas Gohr */ function createToolButton(icon,label,key,id){ var btn = document.createElement('button'); var ico = document.createElement('img'); // preapare the basic button stuff btn.className = 'toolbutton'; btn.title = label; if(key){ btn.title += ' ['+key.toUpperCase()+']'; btn.accessKey = key; } // set IDs if given if(id){ btn.id = id; ico.id = id+'_ico'; } // create the icon and add it to the button if(icon.substr(0,1) == '/'){ ico.src = icon; }else{ ico.src = DOKU_BASE+'lib/images/toolbar/'+icon; } btn.appendChild(ico); return btn; } /** * Creates a picker window for inserting text * * The given list can be an associative array with text,icon pairs * or a simple list of text. Style the picker window through the picker * class or the picker buttons with the pickerbutton class. Picker * windows are appended to the body and created invisible. * * @param string id the ID to assign to the picker * @param array props the properties for the picker * @param string edid the ID of the textarea * @rteurn DOMobject the created picker * @author Andreas Gohr */ function createPicker(id,props,edid){ var icobase = props['icobase']; var list = props['list']; // create the wrapping div var picker = document.createElement('div'); picker.className = 'picker'; if(props['class']){ picker.className += ' '+props['class']; } picker.id = id; picker.style.position = 'absolute'; picker.style.marginLeft = '-10000px'; // no display:none, to keep access keys working for(var key in list){ if (!list.hasOwnProperty(key)) continue; if(isNaN(key)){ // associative array -> treat as image/value pairs var btn = document.createElement('button'); btn.className = 'pickerbutton'; var ico = document.createElement('img'); if(list[key].substr(0,1) == '/'){ ico.src = list[key]; }else{ ico.src = DOKU_BASE+'lib/images/'+icobase+'/'+list[key]; } btn.title = key; btn.appendChild(ico); adddwEvent(btn,'click',bind(pickerInsert,key,edid)); picker.appendChild(btn); }else if(isString(list[key])){ // a list of text -> treat as text picker var btn = document.createElement('button'); btn.className = 'pickerbutton'; var txt = document.createTextNode(list[key]); btn.title = list[key]; btn.appendChild(txt); adddwEvent(btn,'click',bind(pickerInsert,list[key],edid)); picker.appendChild(btn); }else{ // a list of lists -> treat it as subtoolbar initToolbar(picker,edid,list); break; // all buttons handled already } } var body = document.getElementsByTagName('body')[0]; body.appendChild(picker); return picker; } /** * Called by picker buttons to insert Text and close the picker again * * @author Andreas Gohr */ function pickerInsert(text,edid){ insertAtCarret(edid,text); pickerClose(); } /** * Add button action for signature button * * @param DOMElement btn Button element to add the action to * @param array props Associative array of button properties * @param string edid ID of the editor textarea * @return boolean If button should be appended * @author Gabriel Birke */ function addBtnActionSignature(btn, props, edid) { if(typeof(SIG) != 'undefined' && SIG != ''){ adddwEvent(btn,'click',bind(insertAtCarret,edid,SIG)); return true; } return false; } /** * Make intended formattings easier to handle * * Listens to all key inputs and handle indentions * of lists and code blocks * * Currently handles space, backspce and enter presses * * @author Andreas Gohr * @fixme handle tabs */ function keyHandler(e){ if(e.keyCode != 13 && e.keyCode != 8 && e.keyCode != 32) return; var field = e.target; var selection = getSelection(field); var search = "\n"+field.value.substr(0,selection.start); var linestart = Math.max(search.lastIndexOf("\n"), search.lastIndexOf("\r")); //IE workaround search = search.substr(linestart); if(e.keyCode == 13){ // Enter // keep current indention for lists and code var match = search.match(/(\n +([\*-] ?)?)/); if(match){ var scroll = field.scrollHeight; insertAtCarret(field.id,match[1]); field.scrollTop += (field.scrollHeight - scroll); e.preventDefault(); // prevent enter key return false; } }else if(e.keyCode == 8){ // Backspace // unindent lists var match = search.match(/(\n +)([*-] ?)$/); if(match){ var spaces = match[1].length-1; if(spaces > 3){ // unindent one level field.value = field.value.substr(0,linestart)+ field.value.substr(linestart+2); selection.start = selection.start - 2; selection.end = selection.start; }else{ // delete list point field.value = field.value.substr(0,linestart)+ field.value.substr(selection.start); selection.start = linestart; selection.end = linestart; } setSelection(selection); e.preventDefault(); // prevent backspace return false; } }else if(e.keyCode == 32){ // Space // intend list item var match = search.match(/(\n +)([*-] )$/); if(match){ field.value = field.value.substr(0,linestart)+' '+ field.value.substr(linestart); selection.start = selection.start + 2; selection.end = selection.start; setSelection(selection); e.preventDefault(); // prevent space return false; } } } //FIXME consolidate somewhere else addInitEvent(function(){ var field = $('wiki__text'); if(!field) return; adddwEvent(field,'keydown',keyHandler); }); /** * Determine the current section level while editing * * @author Andreas Gohr */ function currentHeadlineLevel(textboxId){ var field = $(textboxId); var selection = getSelection(field); var search = "\n"+field.value.substr(0,selection.start); var lasthl = search.lastIndexOf("\n=="); if(lasthl == -1 && field.form.prefix){ // we need to look in prefix context search = field.form.prefix.value; lasthl = search.lastIndexOf("\n=="); } search = search.substr(lasthl+1,6); if(search == '======') return 1; if(search.substr(0,5) == '=====') return 2; if(search.substr(0,4) == '====') return 3; if(search.substr(0,3) == '===') return 4; if(search.substr(0,2) == '==') return 5; return 0; } /** * global var used for not saved yet warning */ var textChanged = false; /** * Check for changes before leaving the page */ function changeCheck(msg){ if(textChanged){ var ok = confirm(msg); if(ok){ // remove a possibly saved draft using ajax var dwform = $('dw__editform'); if(dwform){ var params = 'call=draftdel'; params += '&id='+encodeURIComponent(dwform.elements.id.value); var sackobj = new sack(DOKU_BASE + 'lib/exe/ajax.php'); sackobj.AjaxFailedAlert = ''; sackobj.encodeURIString = false; sackobj.runAJAX(params); // we send this request blind without waiting for // and handling the returned data } } return ok; }else{ return true; } } /** * Add changeCheck to all Links and Forms (except those with a * JSnocheck class), add handlers to monitor changes * * Sets focus to the editbox as well * * @fixme this is old and crappy code. needs to be redone */ function initChangeCheck(msg){ var edit_text = document.getElementById('wiki__text'); if(!edit_text) return; if(edit_text.readOnly) return; if(!$('dw__editform')) return; // add change check for links var links = document.getElementsByTagName('a'); for(var i=0; i < links.length; i++){ if(links[i].className.indexOf('JSnocheck') == -1){ links[i].onclick = function(){ var rc = changeCheck(msg); if(window.event) window.event.returnValue = rc; return rc; }; } } // add change check for forms var forms = document.forms; for(i=0; i < forms.length; i++){ if(forms[i].className.indexOf('JSnocheck') == -1){ forms[i].onsubmit = function(){ var rc = changeCheck(msg); if(window.event) window.event.returnValue = rc; return rc; }; } } // reset change memory var on submit var btn_save = document.getElementById('edbtn__save'); btn_save.onclick = function(){ textChanged = false; }; var btn_prev = document.getElementById('edbtn__preview'); btn_prev.onclick = function(){ textChanged = false; }; // add change memory setter edit_text.onchange = function(){ textChanged = true; //global var summaryCheck(); }; var summary = document.getElementById('edit__summary'); adddwEvent(summary, 'change', summaryCheck); adddwEvent(summary, 'keyup', summaryCheck); if (textChanged) summaryCheck(); // set focus edit_text.focus(); } /** * Checks if a summary was entered - if not the style is changed * * @author Andreas Gohr */ function summaryCheck(){ var sum = document.getElementById('edit__summary'); if(sum.value === ''){ sum.className='missing'; }else{ sum.className='edit'; } } /** * Class managing the timer to display a warning on a expiring lock */ function locktimer_class(){ this.sack = null; this.timeout = 0; this.timerID = null; this.lasttime = null; this.msg = ''; this.pageid = ''; }; var locktimer = new locktimer_class(); locktimer.init = function(timeout,msg,draft){ // init values locktimer.timeout = timeout*1000; locktimer.msg = msg; locktimer.draft = draft; locktimer.lasttime = new Date(); if(!$('dw__editform')) return; locktimer.pageid = $('dw__editform').elements.id.value; if(!locktimer.pageid) return; // init ajax component locktimer.sack = new sack(DOKU_BASE + 'lib/exe/ajax.php'); locktimer.sack.AjaxFailedAlert = ''; locktimer.sack.encodeURIString = false; locktimer.sack.onCompletion = locktimer.refreshed; // register refresh event adddwEvent($('dw__editform').elements.wikitext,'keypress',function(){locktimer.refresh();}); // start timer locktimer.reset(); }; /** * (Re)start the warning timer */ locktimer.reset = function(){ locktimer.clear(); locktimer.timerID = window.setTimeout("locktimer.warning()", locktimer.timeout); }; /** * Display the warning about the expiring lock */ locktimer.warning = function(){ locktimer.clear(); alert(locktimer.msg); }; /** * Remove the current warning timer */ locktimer.clear = function(){ if(locktimer.timerID !== null){ window.clearTimeout(locktimer.timerID); locktimer.timerID = null; } }; /** * Refresh the lock via AJAX * * Called on keypresses in the edit area */ locktimer.refresh = function(){ var now = new Date(); // refresh every minute only if(now.getTime() - locktimer.lasttime.getTime() > 30*1000){ //FIXME decide on time var params = 'call=lock&id='+encodeURIComponent(locktimer.pageid); if(locktimer.draft){ var dwform = $('dw__editform'); params += '&prefix='+encodeURIComponent(dwform.elements.prefix.value); params += '&wikitext='+encodeURIComponent(dwform.elements.wikitext.value); params += '&suffix='+encodeURIComponent(dwform.elements.suffix.value); params += '&date='+encodeURIComponent(dwform.elements.date.value); } locktimer.sack.runAJAX(params); locktimer.lasttime = now; } }; /** * Callback. Resets the warning timer */ locktimer.refreshed = function(){ var data = this.response; var error = data.charAt(0); data = data.substring(1); $('draft__status').innerHTML=data; if(error != '1') return; // locking failed locktimer.reset(); }; // end of locktimer class functions /* XXXXXXXXXX end of lib/scripts/edit.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/scripts/linkwiz.js XXXXXXXXXX */ /** * The Link Wizard * * @author Andreas Gohr */ var linkwiz = { wiz: null, entry: null, result: null, timer: null, sack: null, textArea: null, selected: -1, selection: null, /** * Initialize the linkwizard by creating the needed HTML * and attaching the eventhandlers */ init: function(textArea){ // prepare AJAX object linkwiz.sack = new sack(DOKU_BASE + 'lib/exe/ajax.php'); linkwiz.sack.AjaxFailedAlert = ''; linkwiz.sack.encodeURIString = false; // create HTML Structure linkwiz.wiz = document.createElement('div'); linkwiz.wiz.id = 'link__wiz'; linkwiz.wiz.className = 'picker'; linkwiz.wiz.style.top = (findPosY(textArea)+20)+'px'; linkwiz.wiz.style.left = (findPosX(textArea)+80)+'px'; linkwiz.wiz.style.marginLeft = '-10000px'; linkwiz.wiz.innerHTML = ''+ '
    '+LANG['linkto']+'
    '+ ''; textArea.form.parentNode.appendChild(linkwiz.wiz); linkwiz.textArea = textArea; linkwiz.result = $('link__wiz_result'); linkwiz.entry = $('link__wiz_entry'); // attach event handlers var obj; obj = $('link__wiz_close'); obj.onclick = linkwiz.hide; linkwiz.sack.elementObj = linkwiz.result; adddwEvent(linkwiz.entry,'keyup',linkwiz.onEntry); adddwEvent(linkwiz.result,'click',linkwiz.onResultClick); drag.attach(linkwiz.wiz,$('link__wiz_header')); }, /** * handle all keyup events in the entry field */ onEntry: function(e){ if(e.keyCode == 37 || e.keyCode == 39){ //left/right return true; //ignore } if(e.keyCode == 27){ linkwiz.hide(); e.preventDefault(); e.stopPropagation(); return false; } if(e.keyCode == 38){ //Up linkwiz.select(linkwiz.selected -1); e.preventDefault(); e.stopPropagation(); return false; } if(e.keyCode == 40){ //Down linkwiz.select(linkwiz.selected +1); e.preventDefault(); e.stopPropagation(); return false; } if(e.keyCode == 13){ //Enter if(linkwiz.selected > -1){ var obj = linkwiz.getResult(linkwiz.selected); if(obj){ var a = obj.getElementsByTagName('A')[0]; linkwiz.resultClick(a); } }else if(linkwiz.entry.value){ linkwiz.insertLink(linkwiz.entry.value); } e.preventDefault(); e.stopPropagation(); return false; } linkwiz.autocomplete(); }, /** * Get one of the result by index * * @param int result div to return * @returns DOMObject or null */ getResult: function(num){ var obj; var childs = linkwiz.result.getElementsByTagName('DIV'); obj = childs[num]; if(obj){ return obj; }else{ return null; } }, /** * Select the given result */ select: function(num){ if(num < 0){ linkwiz.deselect(); return; } var obj = linkwiz.getResult(num); if(obj){ linkwiz.deselect(); obj.className += ' selected'; // make sure the item is viewable in the scroll view // FIXME check IE compatibility if(obj.offsetTop > linkwiz.result.scrollTop + linkwiz.result.clientHeight){ linkwiz.result.scrollTop += obj.clientHeight; }else if(obj.offsetTop - linkwiz.result.clientHeight < linkwiz.result.scrollTop){ // this works but isn't quite right, fixes welcome linkwiz.result.scrollTop -= obj.clientHeight; } // now recheck - if still not in view, the user used the mouse to scroll if( (obj.offsetTop > linkwiz.result.scrollTop + linkwiz.result.clientHeight) || (obj.offsetTop < linkwiz.result.scrollTop) ){ obj.scrollIntoView(); } linkwiz.selected = num; } }, /** * deselect a result if any is selected */ deselect: function(){ if(linkwiz.selected > -1){ var obj = linkwiz.getResult(linkwiz.selected); if(obj){ obj.className = obj.className.replace(/ ?selected/,''); } } linkwiz.selected = -1; }, /** * Handle clicks in the result set an dispatch them to * resultClick() */ onResultClick: function(e){ if(e.target.tagName != 'A') return; e.stopPropagation(); e.preventDefault(); linkwiz.resultClick(e.target); return false; }, /** * Handles the "click" on a given result anchor */ resultClick: function(a){ var id = a.title; if(id == '' || id.substr(id.length-1) == ':'){ linkwiz.entry.value = id; linkwiz.autocomplete_exec(); }else{ linkwiz.entry.value = id; if(a.nextSibling && a.nextSibling.tagName == 'SPAN'){ linkwiz.insertLink(a.nextSibling.innerHTML); }else{ linkwiz.insertLink(''); } } }, /** * Insert the id currently in the entry box to the textarea, * replacing the current selection or at the curso postion. * When no selection is available the given title will be used * as link title instead */ insertLink: function(title){ if(!linkwiz.entry.value) return; var sel = getSelection(linkwiz.textArea); if(sel.start == 0 && sel.end == 0) sel = linkwiz.selection; var stxt = sel.getText(); if(!stxt && !DOKU_UHC) stxt=title; // prepend colon inside namespaces for non namespace pages if(linkwiz.textArea.form['id'].value.indexOf(':') != -1 && linkwiz.entry.value.indexOf(':') == -1){ linkwiz.entry.value = ':'+linkwiz.entry.value; } var link = '[['+linkwiz.entry.value+'|'; if(stxt) link += stxt; link += ']]'; var so = linkwiz.entry.value.length+3; var eo = 2; pasteText(sel,link,{startofs: so, endofs: eo}); linkwiz.hide(); }, /** * Start the page/namespace lookup timer * * Calls autocomplete_exec when the timer runs out */ autocomplete: function(){ if(linkwiz.timer !== null){ window.clearTimeout(linkwiz.timer); linkwiz.timer = null; } linkwiz.timer = window.setTimeout(linkwiz.autocomplete_exec,350); }, /** * Executes the AJAX call for the page/namespace lookup */ autocomplete_exec: function(){ linkwiz.deselect(); linkwiz.result.innerHTML = ''; linkwiz.sack.runAJAX('call=linkwiz&q='+encodeURI(linkwiz.entry.value)); }, /** * Clears the result area */ clear: function(){ linkwiz.result.innerHTML = 'Search for a matching page name above, or browse through the pages on the right'; linkwiz.entry.value = ''; }, /** * Show the linkwizard */ show: function(){ linkwiz.selection = getSelection(linkwiz.textArea); linkwiz.wiz.style.marginLeft = '0px'; linkwiz.entry.focus(); linkwiz.autocomplete(); }, /** * Hide the link wizard */ hide: function(){ linkwiz.wiz.style.marginLeft = '-10000px'; linkwiz.textArea.focus(); }, /** * Toggle the link wizard */ toggle: function(){ if(linkwiz.wiz.style.marginLeft == '-10000px'){ linkwiz.show(); }else{ linkwiz.hide(); } } }; /* XXXXXXXXXX end of lib/scripts/linkwiz.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/scripts/media.js XXXXXXXXXX */ /** * JavaScript functionalitiy for the media management popup * * @author Andreas Gohr */ var media_manager = { keepopen: false, hide: false, /** * Attach event handlers to all "folders" below the given element * * @author Andreas Gohr */ treeattach: function(obj){ if(!obj) return; var items = obj.getElementsByTagName('li'); for(var i=0; i */ selectorattach: function(obj){ if(!obj) return; var items = getElementsByClass('select',obj,'a'); for(var i=0; i */ confirmattach: function(obj){ if(!obj) return; items = getElementsByClass('btn_media_delete',obj,'a'); for(var i=0; i */ attachoptions: function(obj){ if(!obj) return; // keep open if(opener){ var kobox = document.createElement('input'); kobox.type = 'checkbox'; kobox.id = 'media__keepopen'; if(DokuCookie.getValue('keepopen')){ kobox.checked = true; kobox.defaultChecked = true; //IE wants this media_manager.keepopen = true; } adddwEvent(kobox,'click',function(event){ return media_manager.togglekeepopen(event,this); }); var kolbl = document.createElement('label'); kolbl.htmlFor = 'media__keepopen'; kolbl.innerHTML = LANG['keepopen']; var kobr = document.createElement('br'); obj.appendChild(kobox); obj.appendChild(kolbl); obj.appendChild(kobr); } // hide details var hdbox = document.createElement('input'); hdbox.type = 'checkbox'; hdbox.id = 'media__hide'; if(DokuCookie.getValue('hide')){ hdbox.checked = true; hdbox.defaultChecked = true; //IE wants this media_manager.hide = true; } adddwEvent(hdbox,'click',function(event){ return media_manager.togglehide(event,this); }); var hdlbl = document.createElement('label'); hdlbl.htmlFor = 'media__hide'; hdlbl.innerHTML = LANG['hidedetails']; var hdbr = document.createElement('br'); obj.appendChild(hdbox); obj.appendChild(hdlbl); obj.appendChild(hdbr); media_manager.updatehide(); }, /** * Opens the searchfield * * @author Tobias Sarnowski */ showsearchfield: function(event,link){ // prepare an AJAX call to fetch the search var ajax = new sack(DOKU_BASE + 'lib/exe/ajax.php'); ajax.AjaxFailedAlert = ''; ajax.encodeURIString = false; if(ajax.failed) return true; cleanMsgArea(); var content = $('media__content'); content.innerHTML = '...'; ajax.elementObj = content; ajax.afterCompletion = function(){ media_manager.selectorattach(content); media_manager.confirmattach(content); media_manager.updatehide(); }; ajax.runAJAX(link.search.substr(1)+'&call=mediasearchlist'); return false; }, /** * Toggles the keep open state * * @author Andreas Gohr */ togglekeepopen: function(event,cb){ if(cb.checked){ DokuCookie.setValue('keepopen',1); media_manager.keepopen = true; }else{ DokuCookie.setValue('keepopen',''); media_manager.keepopen = false; } }, /** * Toggles the hide details state * * @author Andreas Gohr */ togglehide: function(event,cb){ if(cb.checked){ DokuCookie.setValue('hide',1); media_manager.hide = true; }else{ DokuCookie.setValue('hide',''); media_manager.hide = false; } media_manager.updatehide(); }, /** * Sets the visibility of the image details accordingly to the * chosen hide state * * @author Andreas Gohr */ updatehide: function(){ var obj = $('media__content'); if(!obj) return; var details = getElementsByClass('detail',obj,'div'); for(var i=0; i */ select: function(event,link){ var id = link.name.substr(2); if(!opener){ // if we don't run in popup display example var ex = $('ex_'+id.replace(/:/g,'_')); if(ex.style.display == ''){ ex.style.display = 'none'; }else{ ex.style.display = ''; } return false; } opener.insertTags('wiki__text','{{'+id+'|','}}',''); if(!media_manager.keepopen) window.close(); opener.focus(); return false; }, /** * list the content of a namespace using AJAX * * @author Andreas Gohr */ list: function(event,link){ // prepare an AJAX call to fetch the subtree var ajax = new sack(DOKU_BASE + 'lib/exe/ajax.php'); ajax.AjaxFailedAlert = ''; ajax.encodeURIString = false; if(ajax.failed) return true; cleanMsgArea(); var content = $('media__content'); content.innerHTML = '...'; ajax.elementObj = content; ajax.afterCompletion = function(){ media_manager.selectorattach(content); media_manager.confirmattach(content); media_manager.updatehide(); media_manager.initFlashUpload(); }; ajax.runAJAX(link.search.substr(1)+'&call=medialist'); return false; }, /** * Open or close a subtree using AJAX * * @author Andreas Gohr */ toggle: function(event,clicky){ var listitem = clicky.parentNode; // if already open, close by removing the sublist var sublists = listitem.getElementsByTagName('ul'); if(sublists.length){ listitem.removeChild(sublists[0]); clicky.src = DOKU_BASE+'lib/images/plus.gif'; return false; } // get the enclosed link (is always the first one) var link = listitem.getElementsByTagName('a')[0]; // prepare an AJAX call to fetch the subtree var ajax = new sack(DOKU_BASE + 'lib/exe/ajax.php'); ajax.AjaxFailedAlert = ''; ajax.encodeURIString = false; if(ajax.failed) return true; //prepare the new ul var ul = document.createElement('ul'); //fixme add classname here listitem.appendChild(ul); ajax.elementObj = ul; ajax.afterCompletion = function(){ media_manager.treeattach(ul); }; ajax.runAJAX(link.search.substr(1)+'&call=medians'); clicky.src = DOKU_BASE+'lib/images/minus.gif'; return false; }, /** * Prefills the wikiname. * * @author Andreas Gohr */ suggest: function(){ var file = $('upload__file'); var name = $('upload__name'); if(!file || !name) return; var text = file.value; text = text.substr(text.lastIndexOf('/')+1); text = text.substr(text.lastIndexOf('\\')+1); name.value = text; }, initFlashUpload: function(){ if(!hasFlash(8)) return; var oform = $('dw__upload'); var oflash = $('dw__flashupload'); if(!oform || !oflash) return; var clicky = document.createElement('img'); clicky.src = DOKU_BASE+'lib/images/multiupload.png'; clicky.title = LANG['mu_btn']; clicky.alt = LANG['mu_btn']; clicky.style.cursor = 'pointer'; clicky.onclick = function(){ oform.style.display = 'none'; oflash.style.display = ''; }; oform.appendChild(clicky); } }; addInitEvent(function(){ media_manager.treeattach($('media__tree')); media_manager.selectorattach($('media__content')); media_manager.confirmattach($('media__content')); media_manager.attachoptions($('media__opts')); media_manager.initFlashUpload(); }); /* XXXXXXXXXX end of lib/scripts/media.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/tpl/glfusion/script.js XXXXXXXXXX */ /* XXXXXXXXXX end of lib/tpl/glfusion/script.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/plugins/glfusion/script.js XXXXXXXXXX */ /* XXXXXXXXXX end of lib/plugins/glfusion/script.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/plugins/popularity/script.js XXXXXXXXXX */ /* XXXXXXXXXX end of lib/plugins/popularity/script.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/plugins/revert/script.js XXXXXXXXXX */ /* XXXXXXXXXX end of lib/plugins/revert/script.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/plugins/autotags/script.js XXXXXXXXXX */ /* XXXXXXXXXX end of lib/plugins/autotags/script.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/plugins/plugin/script.js XXXXXXXXXX */ /* XXXXXXXXXX end of lib/plugins/plugin/script.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/plugins/note/script.js XXXXXXXXXX */ /* XXXXXXXXXX end of lib/plugins/note/script.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/plugins/info/script.js XXXXXXXXXX */ /* XXXXXXXXXX end of lib/plugins/info/script.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/plugins/config/script.js XXXXXXXXXX */ /* XXXXXXXXXX end of lib/plugins/config/script.js XXXXXXXXXX */ /* XXXXXXXXXX begin of lib/plugins/acl/script.js XXXXXXXXXX */ acl = { init: function(){ this.ctl = $('acl_manager'); if(!this.ctl) return; var sel = $('acl__user').getElementsByTagName('select')[0]; addEvent(sel,'change',acl.userselhandler); addEvent($('acl__tree'),'click',acl.treehandler); addEvent($('acl__user').getElementsByTagName('input')[1],'click',acl.loadinfo); }, /** * Handle user dropdown */ userselhandler: function(e){ // make entry field visible/invisible if(this.value == '__g__' || this.value == '__u__'){ $('acl__user').getElementsByTagName('input')[0].style.display = ''; //acl_w $('acl__user').getElementsByTagName('input')[1].style.display = ''; //submit }else{ $('acl__user').getElementsByTagName('input')[0].style.display = 'none'; $('acl__user').getElementsByTagName('input')[1].style.display = 'none'; } acl.loadinfo(); }, /** * Load the current permission info and edit form * * @param frm - Form element with needed data */ loadinfo: function(){ // get form var frm = $('acl__detail').getElementsByTagName('form')[0]; // prepare an AJAX call var ajax = new sack(DOKU_BASE + 'lib/plugins/acl/ajax.php'); ajax.AjaxFailedAlert = ''; ajax.encodeURIString = false; if(ajax.failed) return true; // prepare data var data = Array(); data[0] = ajax.encVar('ns',frm.elements['ns'].value); data[1] = ajax.encVar('id',frm.elements['id'].value); data[2] = ajax.encVar('acl_t',frm.elements['acl_t'].value); data[3] = ajax.encVar('acl_w',frm.elements['acl_w'].value); data[4] = ajax.encVar('sectok',frm.elements['sectok'].value); data[5] = ajax.encVar('ajax','info'); ajax.elementObj = $('acl__info'); ajax.runAJAX(data.join('&')); return false; }, /** * parse URL attributes into a associative array * * @todo put into global script lib? */ parseatt: function(str){ if(str[0] == '?') str = str.substr(1); var attributes = {}; var all = str.split('&'); for(var i=0; i/g,">"); return str; }, /** * Open or close a subtree using AJAX * * @author Andreas Gohr */ treetoggle: function(clicky){ var listitem = clicky.parentNode.parentNode; // if already open, close by removing the sublist var sublists = listitem.getElementsByTagName('ul'); if(sublists.length){ listitem.removeChild(sublists[0]); clicky.src = DOKU_BASE+'lib/images/plus.gif'; clicky.alt = '+'; return false; } // get the enclosed link (is always the first one) var link = listitem.getElementsByTagName('a')[0]; // prepare an AJAX call to fetch the subtree var ajax = new sack(DOKU_BASE + 'lib/plugins/acl/ajax.php'); ajax.AjaxFailedAlert = ''; ajax.encodeURIString = false; if(ajax.failed) return true; //prepare the new ul var ul = document.createElement('ul'); listitem.appendChild(ul); ajax.elementObj = ul; ajax.runAJAX(link.search.substr(1)+'&ajax=tree'); clicky.src = DOKU_BASE+'lib/images/minus.gif'; return false; }, /** * Handles all clicks in the tree, dispatching the right action based on the * clicked element */ treehandler: function(e){ if(e.target.src){ // is it an image? acl.treetoggle(e.target); } else if(e.target.href){ // is it a link? // remove highlighting var obj = getElementsByClass('cur',$('acl__tree'),'a'); for(var i=0; i -1){ frm.elements['ns'].value = ''; frm.elements['id'].value = acl.hsc(acl.parseatt(e.target.search)['id']); }else if(e.target.className.search(/idx_dir/) > -1){ frm.elements['ns'].value = acl.hsc(acl.parseatt(e.target.search)['ns']); frm.elements['id'].value = ''; } acl.loadinfo(); } e.stopPropagation(); e.preventDefault(); return false; } }; addInitEvent(acl.init); /* XXXXXXXXXX end of lib/plugins/acl/script.js XXXXXXXXXX */ /* XXXXXXXXXX begin of conf/userscript.js XXXXXXXXXX */ /* XXXXXXXXXX end of conf/userscript.js XXXXXXXXXX */ addInitEvent(function(){ ajax_qsearch.init('qsearch__in','qsearch__out'); }); addInitEvent(function(){ adddwEvent(document,'click',closePopups); }); addInitEvent(function(){ addTocToggle(); }); addInitEvent(function(){ initSizeCtl('size__ctl','wiki__text'); }); addInitEvent(function(){ initToolbar('tool__bar','wiki__text',toolbar); }); addInitEvent(function(){ initChangeCheck('Unsaved changes will be lost.\nReally continue?'); }); addInitEvent(function(){ locktimer.init(840,'Your lock for editing this page is about to expire in a minute.\nTo avoid conflicts use the preview button to reset the locktimer.',1); }); addInitEvent(function(){ scrollToMarker(); }); addInitEvent(function(){ focusMarker(); });