/* Last Modified: 28/04/06 16:28:09 AmiJs library A very small library with DOM and Ajax functions. For a much larger script look on http://www.mochikit.com/ AUTHOR 4mir Salihefendic (http://amix.dk) - amix@amix.dk LICENSE Copyright (c) 2006 Amir Salihefendic. All rights reserved. Copyright (c) 2005 Bob Ippolito. All rights reserved. http://www.opensource.org/licenses/mit-license.php VERSION 2.1 SITE http://amix.dk/amijs **/ var AJS = { //// // Accessor functions //// /** * @returns The element with the id */ getElement: function(id) { if(typeof(id) == "string") return document.getElementById(id); else return id; }, /** * @returns The elements with the ids */ getElements: function(/*id1, id2, id3*/) { var elements = new Array(); for (var i = 0; i < arguments.length; i++) { var element = this.getElement(arguments[i]); elements.push(element); } return elements; }, /** * @returns The GET query argument */ getQueryArgument: function(var_name) { var query = window.location.search.substring(1); var vars = query.split("&"); for (var i=0;i<vars.length;i++) { var pair = vars[i].split("="); if (pair[0] == var_name) { return pair[1]; } } return null; }, /** * @returns If the browser is Internet Explorer */ isIe: function() { return (navigator.userAgent.toLowerCase().indexOf("msie") != -1 && navigator.userAgent.toLowerCase().indexOf("opera") == -1); }, /** * @returns The document body */ getBody: function() { return this.getElementsByTagAndClassName('body')[0] }, /** * @returns All the elements that have a specific tag name or class name */ getElementsByTagAndClassName: function(tag_name, class_name, /*optional*/ parent) { var class_elements = new Array(); if(!this.isDefined(parent)) parent = document; if(!this.isDefined(tag_name)) tag_name = '*'; var els = parent.getElementsByTagName(tag_name); var els_len = els.length; var pattern = new RegExp("(^|\\s)" + class_name + "(\\s|$)"); for (i = 0, j = 0; i < els_len; i++) { if ( pattern.test(els[i].className) || class_name == null ) { class_elements[j] = els[i]; j++; } } return class_elements; }, //// // DOM manipulation //// /** * Appends some nodes to a node */ appendChildNodes: function(node/*, nodes...*/) { if(arguments.length >= 2) { for(var i=1; i < arguments.length; i++) { var n = arguments[i]; if(typeof(n) == "string") n = document.createTextNode(n); if(this.isDefined(n)) node.appendChild(n); } } return node; }, /** * Replaces a nodes children with another node(s) */ replaceChildNodes: function(node/*, nodes...*/) { var child; while ((child = node.firstChild)) { node.removeChild(child); } if (arguments.length < 2) { return node; } else { return this.appendChildNodes.apply(this, arguments); } }, /** * Insert a node after another node */ insertAfter: function(node, referenceNode) { referenceNode.parentNode.insertBefore(node, referenceNode.nextSibling); }, /** * Insert a node before another node */ insertBefore: function(node, referenceNode) { referenceNode.parentNode.insertBefore(node, referenceNode); }, /** * Shows the element */ showElement: function(elm) { elm.style.display = ''; }, /** * Hides the element */ hideElement: function(elm) { elm.style.display = 'none'; }, isElementHidden: function(elm) { return elm.style.visibility == "hidden"; }, /** * Swaps one element with another. To delete use swapDOM(elm, null) */ swapDOM: function(dest, src) { dest = this.getElement(dest); var parent = dest.parentNode; if (src) { src = this.getElement(src); parent.replaceChild(src, dest); } else { parent.removeChild(dest); } return src; }, /** * Removes an element from the world */ removeElement: function(elm) { this.swapDOM(elm, null); }, /** * @returns Is an object a dictionary? */ isDict: function(o) { var str_repr = String(o); return str_repr.indexOf(" Object") != -1; }, /** * Creates a DOM element * @param {String} name The elements DOM name * @param {Dict} attrs Attributes sent to the function */ createDOM: function(name, attrs) { var i=0; elm = document.createElement(name); if(this.isDict(attrs[i])) { for(k in attrs[0]) { if(k == "style") elm.style.cssText = attrs[0][k]; else if(k == "class") elm.className = attrs[0][k]; else elm.setAttribute(k, attrs[0][k]); } i++; } if(attrs[0] == null) i = 1; for(i; i < attrs.length; i++) { var n = attrs[i]; if(this.isDefined(n)) { if(typeof(n) == "string") n = document.createTextNode(n); elm.appendChild(n); } } return elm; }, UL: function() { return this.createDOM.apply(this, ["ul", arguments]); }, LI: function() { return this.createDOM.apply(this, ["li", arguments]); }, TD: function() { return this.createDOM.apply(this, ["td", arguments]); }, TR: function() { return this.createDOM.apply(this, ["tr", arguments]); }, TH: function() { return this.createDOM.apply(this, ["th", arguments]); }, TBODY: function() { return this.createDOM.apply(this, ["tbody", arguments]); }, TABLE: function() { return this.createDOM.apply(this, ["table", arguments]); }, INPUT: function() { return this.createDOM.apply(this, ["input", arguments]); }, SPAN: function() { return this.createDOM.apply(this, ["span", arguments]); }, B: function() { return this.createDOM.apply(this, ["b", arguments]); }, A: function() { return this.createDOM.apply(this, ["a", arguments]); }, DIV: function() { return this.createDOM.apply(this, ["div", arguments]); }, IMG: function() { return this.createDOM.apply(this, ["img", arguments]); }, BUTTON: function() { return this.createDOM.apply(this, ["button", arguments]); }, H1: function() { return this.createDOM.apply(this, ["h1", arguments]); }, H2: function() { return this.createDOM.apply(this, ["h2", arguments]); }, H3: function() { return this.createDOM.apply(this, ["h3", arguments]); }, BR: function() { return this.createDOM.apply(this, ["br", arguments]); }, TEXTAREA: function() { return this.createDOM.apply(this, ["textarea", arguments]); }, FORM: function() { return this.createDOM.apply(this, ["form", arguments]); }, P: function() { return this.createDOM.apply(this, ["p", arguments]); }, SELECT: function() { return this.createDOM.apply(this, ["select", arguments]); }, OPTION: function() { return this.createDOM.apply(this, ["option", arguments]); }, TN: function(text) { return document.createTextNode(text); }, IFRAME: function() { return this.createDOM.apply(this, ["iframe", arguments]); }, SCRIPT: function() { return this.createDOM.apply(this, ["script", arguments]); }, //// // Ajax functions //// /** * @returns A new XMLHttpRequest object */ getXMLHttpRequest: function() { var try_these = [ function () { return new XMLHttpRequest(); }, function () { return new ActiveXObject('Msxml2.XMLHTTP'); }, function () { return new ActiveXObject('Microsoft.XMLHTTP'); }, function () { return new ActiveXObject('Msxml2.XMLHTTP.4.0'); }, function () { throw "Browser does not support XMLHttpRequest"; } ]; for (var i = 0; i < try_these.length; i++) { var func = try_these[i]; try { return func(); } catch (e) { } } }, /** * Use this function to do a simple HTTP Request */ doSimpleXMLHttpRequest: function(url) { var req = this.getXMLHttpRequest(); req.open("GET", url, true); return this.sendXMLHttpRequest(req); }, getRequest: function(url, data) { var req = this.getXMLHttpRequest(); req.open("POST", url, true); req.setRequestHeader("Content-type", "application/x-www-form-urlencoded"); return this.sendXMLHttpRequest(req); }, /** * Send a XMLHttpRequest */ sendXMLHttpRequest: function(req, data) { var d = new AJSDeferred(req); var onreadystatechange = function () { if (req.readyState == 4) { try { var status = req.status; } catch(e) {}; if(status == 200 || status == 304 || req.responseText == null) { d.callback(req, data); } else { d.errback(); } } } req.onreadystatechange = onreadystatechange; return d; }, /** * Represent an object as a string */ reprString: function(o) { return ('"' + o.replace(/(["\\])/g, '\\$1') + '"' ).replace(/[\f]/g, "\\f" ).replace(/[\b]/g, "\\b" ).replace(/[\n]/g, "\\n" ).replace(/[\t]/g, "\\t" ).replace(/[\r]/g, "\\r"); }, /** * Serialize an object to JSON notation */ serializeJSON: function(o) { var objtype = typeof(o); if (objtype == "undefined") { return "undefined"; } else if (objtype == "number" || objtype == "boolean") { return o + ""; } else if (o === null) { return "null"; } if (objtype == "string") { return this.reprString(o); } var me = arguments.callee; var newObj; if (typeof(o.__json__) == "function") { newObj = o.__json__(); if (o !== newObj) { return me(newObj); } } if (typeof(o.json) == "function") { newObj = o.json(); if (o !== newObj) { return me(newObj); } } if (objtype != "function" && typeof(o.length) == "number") { var res = []; for (var i = 0; i < o.length; i++) { var val = me(o[i]); if (typeof(val) != "string") { val = "undefined"; } res.push(val); } return "[" + res.join(",") + "]"; } res = []; for (var k in o) { var useKey; if (typeof(k) == "number") { useKey = '"' + k + '"'; } else if (typeof(k) == "string") { useKey = this.reprString(k); } else { // skip non-string or number keys continue; } val = me(o[k]); if (typeof(val) != "string") { // skip non-serializable values continue; } res.push(useKey + ":" + val); } return "{" + res.join(",") + "}"; }, /** * Send and recive JSON using GET */ loadJSONDoc: function(url) { var d = this.getRequest(url); var eval_req = function(req) { var text = req.responseText; return eval('(' + text + ')'); }; d.addCallback(eval_req); return d; }, //// // Misc. //// /** * Alert the objects key attrs */ keys: function(obj) { var rval = []; for (var prop in obj) { rval.push(prop); } return rval; }, urlencode: function(str) { return encodeURIComponent(str.toString()); }, /** * @returns True if the object is defined, otherwise false */ isDefined: function(o) { return (o != "undefined" && o != null) }, /** * @returns True if an object is a array, false otherwise */ isArray: function(obj) { try { return (typeof(obj.length) == "undefined") ? false : true; } catch(e) { return false; } }, isObject: function(obj) { return (obj && typeof obj == 'object'); }, /** * Export DOM elements to the global namespace */ exportDOMElements: function() { UL = this.UL; LI = this.LI; TD = this.TD; TR = this.TR; TH = this.TH; TBODY = this.TBODY; TABLE = this.TABLE; INPUT = this.INPUT; SPAN = this.SPAN; B = this.B; A = this.A; DIV = this.DIV; IMG = this.IMG; BUTTON = this.BUTTON; H1 = this.H1; H2 = this.H2; H3 = this.H3; BR = this.BR; TEXTAREA = this.TEXTAREA; FORM = this.FORM; P = this.P; SELECT = this.SELECT; OPTION = this.OPTION; TN = this.TN; IFRAME = this.IFRAME; SCRIPT = this.SCRIPT; }, /** * Export AmiJS functions to the global namespace */ exportToGlobalScope: function() { getElement = this.getElement; getQueryArgument = this.getQueryArgument; isIe = this.isIe; $ = this.getElement; getElements = this.getElements; getBody = this.getBody; getElementsByTagAndClassName = this.getElementsByTagAndClassName; appendChildNodes = this.appendChildNodes; ACN = appendChildNodes; replaceChildNodes = this.replaceChildNodes; RCN = replaceChildNodes; insertAfter = this.insertAfter; insertBefore = this.insertBefore; showElement = this.showElement; hideElement = this.hideElement; isElementHidden = this.isElementHidden; swapDOM = this.swapDOM; removeElement = this.removeElement; isDict = this.isDict; createDOM = this.createDOM; this.exportDOMElements(); getXMLHttpRequest = this.getXMLHttpRequest; doSimpleXMLHttpRequest = this.doSimpleXMLHttpRequest; getRequest = this.getRequest; sendXMLHttpRequest = this.sendXMLHttpRequest; reprString = this.reprString; serializeJSON = this.serializeJSON; loadJSONDoc = this.loadJSONDoc; keys = this.keys; isDefined = this.isDefined; isArray = this.isArray; } } AJSDeferred = function(req) { this.callbacks = []; this.req = req; this.callback = function (res) { while (this.callbacks.length > 0) { var fn = this.callbacks.pop(); res = fn(res); } }; this.errback = function(e){ alert("Error encountered:\n" + e); }; this.addErrback = function(fn) { this.errback = fn; }; this.addCallback = function(fn) { this.callbacks.unshift(fn); }; this.addCallbacks = function(fn1, fn2) { this.addCallback(fn1); this.addErrback(fn2); }; this.sendReq = function(data) { if(AJS.isObject(data)) { var post_data = []; for(k in data) { post_data.push(k + "=" + AJS.urlencode(data[k])); } post_data = post_data.join("&"); this.req.send(post_data); } else if(AJS.isDefined(data)) this.req.send(data); else { this.req.send(""); } }; }; AJSDeferred.prototype = new AJSDeferred(); /**** Last Modified: 28/04/06 15:26:06 GoogieSpell Google spell checker for your own web-apps :) Copyright Amir Salihefendic 2006 LICENSE GPL (see gpl.txt for more information) This basically means that you can't use this script with/in proprietary software! There is another license that permits you to use this script with proprietary software. Check out:... for more info. AUTHOR 4mir Salihefendic (http://amix.dk) - amix@amix.dk VERSION 3.22 ****/ var GOOGIE_CUR_LANG = "en"; function GoogieSpell(img_dir, server_url) { var cookie_value; var lang; cookie_value = getCookie('language'); if(cookie_value != null) GOOGIE_CUR_LANG = cookie_value; this.img_dir = img_dir; this.server_url = server_url; this.lang_to_word = {"da": "Dansk", "de": "Deutsch", "en": "English", "es": "Español", "fr": "Français", "it": "Italiano", "nl": "Nederlands", "pl": "Polski", "pt": "Português", "fi": "Suomi", "sv": "Svenska"}; this.langlist_codes = AJS.keys(this.lang_to_word); this.show_change_lang_pic = true; this.lang_state_observer = null; this.spelling_state_observer = null; this.request = null; this.error_window = null; this.language_window = null; this.edit_layer = null; this.orginal_text = null; this.results = null; this.text_area = null; this.gselm = null; this.ta_scroll_top = 0; this.el_scroll_top = 0; this.lang_chck_spell = "Check spelling"; this.lang_rsm_edt = "Resume editing"; this.lang_close = "Close"; this.lang_no_error_found = "No spelling errors found"; this.lang_revert = "Revert to"; this.show_spell_img = false; // modified by roundcube } GoogieSpell.prototype.setStateChanged = function(current_state) { if(this.spelling_state_observer != null) this.spelling_state_observer(current_state); } GoogieSpell.item_onmouseover = function(e) { var elm = GoogieSpell.getEventElm(e); if(elm.className != "googie_list_close" && elm.className != "googie_list_revert") elm.className = "googie_list_onhover"; else elm.parentNode.className = "googie_list_onhover"; } GoogieSpell.item_onmouseout = function(e) { var elm = GoogieSpell.getEventElm(e); if(elm.className != "googie_list_close" && elm.className != "googie_list_revert") elm.className = "googie_list_onout"; else elm.parentNode.className = "googie_list_onout"; } GoogieSpell.prototype.getGoogleUrl = function() { return this.server_url + GOOGIE_CUR_LANG; } GoogieSpell.prototype.spellCheck = function(elm, name) { this.ta_scroll_top = this.text_area.scrollTop; this.appendIndicator(elm); try { this.hideLangWindow(); } catch(e) {} this.gselm = elm; this.createEditLayer(this.text_area.offsetWidth, this.text_area.offsetHeight); this.createErrorWindow(); AJS.getBody().appendChild(this.error_window); try { netscape.security.PrivilegeManager.enablePrivilege("UniversalBrowserRead"); } catch (e) { } this.gselm.onclick = null; this.orginal_text = this.text_area.value; var me = this; //Create request var d = AJS.getRequest(this.getGoogleUrl()); var reqdone = function(req) { var r_text = req.responseText; if(r_text.match(/<c.*>/) != null) { var results = GoogieSpell.parseResult(r_text); //Before parsing be sure that errors were found me.results = results; me.showErrorsInIframe(results); me.resumeEditingState(); } else { me.flashNoSpellingErrorState(); } me.removeIndicator(); }; var reqfailed = function(req) { alert("An error was encountered on the server. Please try again later."); AJS.removeElement(me.gselm); me.checkSpellingState(); me.removeIndicator(); }; d.addCallback(reqdone); d.addErrback(reqfailed); var req_text = GoogieSpell.escapeSepcial(this.orginal_text); d.sendReq(GoogieSpell.createXMLReq(req_text)); } GoogieSpell.escapeSepcial = function(val) { return val.replace(/&/g, "&").replace(/</g, "<").replace(/>/g, ">"); } GoogieSpell.createXMLReq = function (text) { return '<?xml version="1.0" encoding="utf-8" ?><spellrequest textalreadyclipped="0" ignoredups="0" ignoredigits="1" ignoreallcaps="1"><text>' + text + '</text></spellrequest>'; } //Retunrs an array //result[item] -> ['attrs'] // ['suggestions'] GoogieSpell.parseResult = function(r_text) { var re_split_attr_c = /\w="\d+"/g; var re_split_text = /\t/g; var matched_c = r_text.match(/<c[^>]*>[^<]*<\/c>/g); var results = new Array(); for(var i=0; i < matched_c.length; i++) { var item = new Array(); //Get attributes item['attrs'] = new Array(); var split_c = matched_c[i].match(re_split_attr_c); for(var j=0; j < split_c.length; j++) { var c_attr = split_c[j].split(/=/); item['attrs'][c_attr[0]] = parseInt(c_attr[1].replace('"', '')); } //Get suggestions item['suggestions'] = new Array(); var only_text = matched_c[i].replace(/<[^>]*>/g, ""); var split_t = only_text.split(re_split_text); for(var k=0; k < split_t.length; k++) { if(split_t[k] != "") item['suggestions'].push(split_t[k]); } results.push(item); } return results; } /**** Error window (the drop-down window) ****/ GoogieSpell.prototype.createErrorWindow = function() { this.error_window = AJS.DIV(); this.error_window.className = "googie_window"; } GoogieSpell.prototype.hideErrorWindow = function() { this.error_window.style.visibility = "hidden"; } GoogieSpell.prototype.updateOrginalText = function(offset, old_value, new_value, id) { var part_1 = this.orginal_text.substring(0, offset); var part_2 = this.orginal_text.substring(offset+old_value.length); this.orginal_text = part_1 + new_value + part_2; var add_2_offset = new_value.length - old_value.length; for(var j=0; j < this.results.length; j++) { //Don't edit the offset of the current item if(j != id && j > id){ this.results[j]['attrs']['o'] += add_2_offset; } } } GoogieSpell.prototype.saveOldValue = function (id, old_value) { this.results[id]['is_changed'] = true; this.results[id]['old_value'] = old_value; } GoogieSpell.prototype.showErrorWindow = function(elm, id) { var me = this; var abs_pos = GoogieSpell.absolutePosition(elm); abs_pos.y -= this.edit_layer.scrollTop; this.error_window.style.visibility = "visible"; this.error_window.style.top = (abs_pos.y+20) + "px"; this.error_window.style.left = (abs_pos.x) + "px"; this.error_window.innerHTML = ""; //Build up the result list var table = AJS.TABLE({'class': 'googie_list'}); var list = AJS.TBODY(); var suggestions = this.results[id]['suggestions']; var offset = this.results[id]['attrs']['o']; var len = this.results[id]['attrs']['l']; if(suggestions.length == 0) { var row = AJS.TR(); var item = AJS.TD(); var dummy = AJS.SPAN(); item.appendChild(AJS.TN("No suggestions :(")); row.appendChild(item); list.appendChild(row); } for(i=0; i < suggestions.length; i++) { var row = AJS.TR(); var item = AJS.TD(); var dummy = AJS.SPAN(); dummy.innerHTML = suggestions[i]; item.appendChild(AJS.TN(dummy.innerHTML)); item.onclick = function(e) { var l_elm = GoogieSpell.getEventElm(e); var old_value = elm.innerHTML; var new_value = l_elm.innerHTML; elm.style.color = "green"; elm.innerHTML = l_elm.innerHTML; me.hideErrorWindow(); me.updateOrginalText(offset, old_value, new_value, id); //Update to the new length me.results[id]['attrs']['l'] = new_value.length; me.saveOldValue(id, old_value); }; item.onmouseover = GoogieSpell.item_onmouseover; item.onmouseout = GoogieSpell.item_onmouseout; row.appendChild(item); list.appendChild(row); } //The element is changed, append the revert if(this.results[id]['is_changed']) { var old_value = this.results[id]['old_value']; var offset = this.results[id]['attrs']['o']; var revert_row = AJS.TR(); var revert = AJS.TD(); revert.onmouseover = GoogieSpell.item_onmouseover; revert.onmouseout = GoogieSpell.item_onmouseout; var rev_span = AJS.SPAN({'class': 'googie_list_revert'}); rev_span.innerHTML = this.lang_revert + " " + old_value; revert.appendChild(rev_span); revert.onclick = function(e) { me.updateOrginalText(offset, elm.innerHTML, old_value, id); elm.style.color = "#b91414"; elm.innerHTML = old_value; me.hideErrorWindow(); }; revert_row.appendChild(revert); list.appendChild(revert_row); } //Append the edit box var edit_row = AJS.TR(); var edit = AJS.TD(); var edit_input = AJS.INPUT({'style': 'width: 120px; margin:0; padding:0'}); var onsub = function () { if(edit_input.value != "") { me.saveOldValue(id, elm.innerHTML); me.updateOrginalText(offset, elm.innerHTML, edit_input.value, id); elm.style.color = "green" elm.innerHTML = edit_input.value; me.hideErrorWindow(); return false; } }; var ok_pic = AJS.IMG({'src': this.img_dir + "ok.gif", 'style': 'width: 32px; height: 16px; margin-left: 2px; margin-right: 2px;'}); var edit_form = AJS.FORM({'style': 'margin: 0; padding: 0'}, edit_input, ok_pic); ok_pic.onclick = onsub; edit_form.onsubmit = onsub; edit.appendChild(edit_form); edit_row.appendChild(edit); list.appendChild(edit_row); //Close button var close_row = AJS.TR(); var close = AJS.TD(); close.onmouseover = GoogieSpell.item_onmouseover; close.onmouseout = GoogieSpell.item_onmouseout; var spn_close = AJS.SPAN({'class': 'googie_list_close'}); spn_close.innerHTML = this.lang_close; close.appendChild(spn_close); close.onclick = function() { me.hideErrorWindow()}; close_row.appendChild(close); list.appendChild(close_row); table.appendChild(list); this.error_window.appendChild(table); } /**** Edit layer (the layer where the suggestions are stored) ****/ GoogieSpell.prototype.createEditLayer = function(width, height) { this.edit_layer = AJS.DIV({'class': 'googie_edit_layer'}); //Set the style so it looks like edit areas this.edit_layer.className = this.text_area.className; this.edit_layer.style.border = "1px solid #999"; this.edit_layer.style.overflow = "auto"; this.edit_layer.style.backgroundColor = "#F1EDFE"; this.edit_layer.style.padding = "3px"; this.edit_layer.style.width = (width-8) + "px"; this.edit_layer.style.height = height + "px"; } GoogieSpell.prototype.resumeEditing = function(e, me) { this.setStateChanged("check_spelling"); me.switch_lan_pic.style.display = "inline"; this.el_scroll_top = me.edit_layer.scrollTop; var elm = GoogieSpell.getEventElm(e); AJS.replaceChildNodes(elm, this.createSpellDiv()); elm.onclick = function(e) { me.spellCheck(elm, me.text_area.id); }; me.hideErrorWindow(); //Remove the EDIT_LAYER me.edit_layer.parentNode.removeChild(me.edit_layer); me.text_area.value = me.orginal_text; AJS.showElement(me.text_area); me.gselm.className = "googie_no_style"; me.text_area.scrollTop = this.el_scroll_top; elm.onmouseout = null; } GoogieSpell.prototype.createErrorLink = function(text, id) { var elm = AJS.SPAN({'class': 'googie_link'}); var me = this; elm.onclick = function () { me.showErrorWindow(elm, id); }; elm.innerHTML = text; return elm; } GoogieSpell.createPart = function(txt_part) { if(txt_part == " ") return AJS.TN(" "); var result = AJS.SPAN(); var is_first = true; var is_safari = (navigator.userAgent.toLowerCase().indexOf("safari") != -1); var part = AJS.SPAN(); txt_part = GoogieSpell.escapeSepcial(txt_part); txt_part = txt_part.replace(/\n/g, "<br>"); txt_part = txt_part.replace(/ /g, " "); txt_part = txt_part.replace(/^ /g, " "); txt_part = txt_part.replace(/ $/g, " "); part.innerHTML = txt_part; return part; } GoogieSpell.prototype.showErrorsInIframe = function(results) { var output = AJS.DIV(); output.style.textAlign = "left"; var pointer = 0; for(var i=0; i < results.length; i++) { var offset = results[i]['attrs']['o']; var len = results[i]['attrs']['l']; var part_1_text = this.orginal_text.substring(pointer, offset); var part_1 = GoogieSpell.createPart(part_1_text); output.appendChild(part_1); pointer += offset - pointer; //If the last child was an error, then insert some space output.appendChild(this.createErrorLink(this.orginal_text.substr(offset, len), i)); pointer += len; } //Insert the rest of the orginal text var part_2_text = this.orginal_text.substr(pointer, this.orginal_text.length); var part_2 = GoogieSpell.createPart(part_2_text); output.appendChild(part_2); this.edit_layer.appendChild(output); //Hide text area AJS.hideElement(this.text_area); this.text_area.parentNode.insertBefore(this.edit_layer, this.text_area.nextSibling); this.edit_layer.scrollTop = this.ta_scroll_top; } GoogieSpell.Position = function(x, y) { this.x = x; this.y = y; } //Get the absolute position of menu_slide GoogieSpell.absolutePosition = function(element) { //Create a new object that has elements y and x pos... var posObj = new GoogieSpell.Position(element.offsetLeft, element.offsetTop); //Check if the element has an offsetParent - if it has .. loop until it has not if(element.offsetParent) { var temp_pos = GoogieSpell.absolutePosition(element.offsetParent); posObj.x += temp_pos.x; posObj.y += temp_pos.y; } return posObj; } GoogieSpell.getEventElm = function(e) { var targ; if (!e) var e = window.event; if (e.target) targ = e.target; else if (e.srcElement) targ = e.srcElement; if (targ.nodeType == 3) // defeat Safari bug targ = targ.parentNode; return targ; } GoogieSpell.prototype.removeIndicator = function(elm) { // modified by roundcube if (window.rcube_webmail_client) rcube_webmail_client.set_busy(false); //AJS.removeElement(this.indicator); } GoogieSpell.prototype.appendIndicator = function(elm) { // modified by roundcube if (window.rcube_webmail_client) rcube_webmail_client.set_busy(true, 'checking'); /* var img = AJS.IMG({'src': this.img_dir + 'indicator.gif', 'style': 'margin-right: 5px;'}); img.style.width = "16px"; img.style.height = "16px"; this.indicator = img; img.style.textDecoration = "none"; AJS.insertBefore(img, elm); */ } /**** Choose language ****/ GoogieSpell.prototype.createLangWindow = function() { this.language_window = AJS.DIV({'class': 'googie_window'}); this.language_window.style.width = "130px"; //Build up the result list var table = AJS.TABLE({'class': 'googie_list'}); var list = AJS.TBODY(); this.lang_elms = new Array(); for(i=0; i < this.langlist_codes.length; i++) { var row = AJS.TR(); var item = AJS.TD(); item.googieId = this.langlist_codes[i]; this.lang_elms.push(item); var lang_span = AJS.SPAN(); lang_span.innerHTML = this.lang_to_word[this.langlist_codes[i]]; item.appendChild(AJS.TN(lang_span.innerHTML)); var me = this; item.onclick = function(e) { var elm = GoogieSpell.getEventElm(e); me.deHighlightCurSel(); me.setCurrentLanguage(elm.googieId); if(me.lang_state_observer != null) { me.lang_state_observer(); } me.highlightCurSel(); me.hideLangWindow(); }; item.onmouseover = function(e) { var i_it = GoogieSpell.getEventElm(e); if(i_it.className != "googie_list_selected") i_it.className = "googie_list_onhover"; }; item.onmouseout = function(e) { var i_it = GoogieSpell.getEventElm(e); if(i_it.className != "googie_list_selected") i_it.className = "googie_list_onout"; }; row.appendChild(item); list.appendChild(row); } this.highlightCurSel(); //Close button var close_row = AJS.TR(); var close = AJS.TD(); close.onmouseover = GoogieSpell.item_onmouseover; close.onmouseout = GoogieSpell.item_onmouseout; var spn_close = AJS.SPAN({'class': 'googie_list_close'}); spn_close.innerHTML = this.lang_close; close.appendChild(spn_close); var me = this; close.onclick = function(e) { me.hideLangWindow(); GoogieSpell.item_onmouseout(e); }; close_row.appendChild(close); list.appendChild(close_row); table.appendChild(list); this.language_window.appendChild(table); } GoogieSpell.prototype.setCurrentLanguage = function(lan_code) { GOOGIE_CUR_LANG = lan_code; //Set cookie var now = new Date(); now.setTime(now.getTime() + 365 * 24 * 60 * 60 * 1000); setCookie('language', lan_code, now); } GoogieSpell.prototype.hideLangWindow = function() { this.language_window.style.visibility = "hidden"; this.switch_lan_pic.className = "googie_lang_3d_on"; } GoogieSpell.prototype.deHighlightCurSel = function() { this.lang_cur_elm.className = "googie_list_onout"; } GoogieSpell.prototype.highlightCurSel = function() { for(var i=0; i < this.lang_elms.length; i++) { if(this.lang_elms[i].googieId == GOOGIE_CUR_LANG) { this.lang_elms[i].className = "googie_list_selected"; this.lang_cur_elm = this.lang_elms[i]; } else { this.lang_elms[i].className = "googie_list_onout"; } } } GoogieSpell.prototype.showLangWindow = function(elm, ofst_top, ofst_left) { if(!AJS.isDefined(ofst_top)) ofst_top = 20; if(!AJS.isDefined(ofst_left)) ofst_left = 50; this.createLangWindow(); AJS.getBody().appendChild(this.language_window); var abs_pos = GoogieSpell.absolutePosition(elm); AJS.showElement(this.language_window); this.language_window.style.top = (abs_pos.y+ofst_top) + "px"; this.language_window.style.left = (abs_pos.x+ofst_left-this.language_window.offsetWidth) + "px"; this.highlightCurSel(); this.language_window.style.visibility = "visible"; } GoogieSpell.prototype.flashNoSpellingErrorState = function() { this.setStateChanged("no_error_found"); var me = this; AJS.hideElement(this.switch_lan_pic); this.gselm.innerHTML = this.lang_no_error_found; this.gselm.className = "googie_check_spelling_ok"; this.gselm.style.textDecoration = "none"; this.gselm.style.cursor = "default"; var fu = function() { AJS.removeElement(me.gselm); me.checkSpellingState(); }; setTimeout(fu, 1000); } GoogieSpell.prototype.resumeEditingState = function() { this.setStateChanged("resume_editing"); var me = this; AJS.hideElement(me.switch_lan_pic); //Change link text to resume me.gselm.innerHTML = this.lang_rsm_edt; me.gselm.onclick = function(e) { me.resumeEditing(e, me); } me.gselm.className = "googie_check_spelling_ok"; me.edit_layer.scrollTop = me.ta_scroll_top; } GoogieSpell.prototype.createChangeLangPic = function() { var switch_lan = AJS.A({'class': 'googie_lang_3d_on', 'style': 'padding-left: 6px;'}, AJS.IMG({'src': this.img_dir + 'change_lang.gif', 'alt': "Change language"})); switch_lan.onmouseover = function() { if(this.className != "googie_lang_3d_click") this.className = "googie_lang_3d_on"; } var me = this; switch_lan.onclick = function() { if(this.className == "googie_lang_3d_click") { me.hideLangWindow(); } else { me.showLangWindow(switch_lan); this.className = "googie_lang_3d_click"; } } return switch_lan; } GoogieSpell.prototype.createSpellDiv = function() { var chk_spell = AJS.SPAN({'class': 'googie_check_spelling_link'}); chk_spell.innerHTML = this.lang_chck_spell; var spell_img = null; if(this.show_spell_img) spell_img = AJS.IMG({'src': this.img_dir + "spellc.gif"}); return AJS.SPAN(spell_img, " ", chk_spell); } GoogieSpell.prototype.checkSpellingState = function() { this.setStateChanged("check_spelling"); var me = this; if(this.show_change_lang_pic) this.switch_lan_pic = this.createChangeLangPic(); else this.switch_lan_pic = AJS.SPAN(); var span_chck = this.createSpellDiv(); span_chck.onclick = function() { me.spellCheck(span_chck); } AJS.appendChildNodes(this.spell_container, span_chck, " ", this.switch_lan_pic); // modified by roundcube this.check_link = span_chck; } GoogieSpell.prototype.setLanguages = function(lang_dict) { this.lang_to_word = lang_dict; this.langlist_codes = AJS.keys(lang_dict); } GoogieSpell.prototype.decorateTextarea = function(id, /*optional*/spell_container_id, force_width) { var me = this; if(typeof(id) == "string") this.text_area = AJS.getElement(id); else this.text_area = id; var r_width; if(this.text_area != null) { if(AJS.isDefined(spell_container_id)) { if(typeof(spell_container_id) == "string") this.spell_container = AJS.getElement(spell_container_id); else this.spell_container = spell_container_id; } else { var table = AJS.TABLE(); var tbody = AJS.TBODY(); var tr = AJS.TR(); if(AJS.isDefined(force_width)) { r_width = force_width; } else { r_width = this.text_area.offsetWidth + "px"; } var spell_container = AJS.TD(); this.spell_container = spell_container; tr.appendChild(spell_container); tbody.appendChild(tr); table.appendChild(tbody); AJS.insertBefore(table, this.text_area); //Set width table.style.width = '100%'; // modified by roundcube (old: r_width) spell_container.style.width = r_width; spell_container.style.textAlign = "right"; } this.checkSpellingState(); } else { alert("Text area not found"); } }