/** * @preserve This file is part of Mibew Messenger project. * http://mibew.org * * Copyright (c) 2005-2011 Mibew Messenger Community * License: http://mibew.org/license.php */ var FrameUtils = { getDocument: function(frm) { if (frm.contentDocument) { return frm.contentDocument; } else if (frm.contentWindow) { return frm.contentWindow.document; } else if (frm.document) { return frm.document; } else { return null; } }, initFrame: function(frm) { var doc = this.getDocument(frm); doc.open(); doc.write(""); doc.write(""); doc.write(""); doc.write("
"); doc.write(""); doc.close(); frm.onload = function() { if( frm.myHtml ) { FrameUtils.getDocument(frm).getElementById('content').innerHTML += frm.myHtml; FrameUtils.scrollDown(frm); } }; }, insertIntoFrame: function(frm, htmlcontent) { var vcontent = this.getDocument(frm).getElementById('content'); if( vcontent == null ) { if( !frm.myHtml ) frm.myHtml = ""; frm.myHtml += htmlcontent; } else { vcontent.innerHTML += htmlcontent; } }, scrollDown: function(frm) { var vbottom = this.getDocument(frm).getElementById('bottom'); if( myAgent == 'opera' ) { try { frm.contentWindow.scrollTo(0,this.getDocument(frm).getElementById('content').clientHeight); } catch(e) {} } if( vbottom ) { vbottom.scrollIntoView(false); } } }; ChatServer = Class.create(); /** * @todo Think about error handling */ ChatServer.prototype = { /** * @constructor */ initialize: function(options) { var chatServer = this; /** * Update timer */ this.updateTimer = null; /** * Options for the ChatServer object * @private * @todo Check onResponseError handler */ this.options = { // Server gateway URL servl: "", // Frequency for automatic updater requestsFrequency: 2, // Call on request timeout onTimeout: function() {}, // Call when transport error was caught onTransportError: function(e) {}, // Call when callFunctions related error was caught onCallError: function(e) {}, // Call when update related error was caught onUpdateError: function(e) {}, // Call when response related error was caught onResponseError: function(e) {} }.extend(options); /** * Binds request's token and callback function * @type Object * @private */ this.callbacks = {}; /** * Array of periodically called functions * @type Array * @private */ this.callPeriodically = []; /** * Options for an Ajax.Request object * @type Array * @private */ this.ajaxOptions = { _method: 'post', asynchronous: true, timeout: 5000, onComplete: chatServer.receiveResponse.bind(chatServer), onException: chatServer.onTransportError.bind(chatServer), onTimeout: chatServer.onTimeout.bind(chatServer) } /** * An object of the Ajax.Request class * @type Ajax.Request * @private */ this.ajaxRequest = null; /** * This buffer store requests and responses between sending packages * @private */ this.buffer = []; /** * Contains object of registered functions handlers * @private */ this.functions = {} /** * An instance of the MibewAPI class * @type MibewAPI * @private */ this.mibewAPI = new MibewAPI(new MibewAPICoreInteraction()); }, /** * Make call to the chat server * * @param {Oblect[]} functionsList List of the function objects. See Mibew API * for details. * @param {Function} callbackFunction * @param {Boolean} forceSend Force requests buffer send right after call * @returns {Boolean} boolean true on success and false on failure */ callFunctions: function(functionsList, callbackFunction, forceSend) { try { // Check function objects if (!(functionsList instanceof Array)) { throw new Error("The first arguments must be an array"); } for (var i in functionsList) { // Filter 'Prototype' properties if (! functionsList.hasOwnProperty(i)) { continue; } this.mibewAPI.checkFunction(functionsList[i], false); } // Generate request token var token = this.generateToken(); // Store callback function this.callbacks[token] = callbackFunction; // Add request to buffer this.buffer.push({ 'token': token, 'functions': functionsList }); if (forceSend) { // Force update this.update(); } } catch (e) { // Handle errors this.options.onCallError(e); return false; } return true; }, /** * Call function at every request to build functions list * * @param {Function} functionsListBuilder Call before every request to build a * list of functions that must be called * @param {Function} callbackFunction Call after response received */ callFunctionsPeriodically: function(functionsListBuilder, callbackFunction) { this.callPeriodically.push({ functionsListBuilder: functionsListBuilder, callbackFunction: callbackFunction }); }, /** * Generates unique request token * * @private * @returns {String} Request token */ generateToken: function() { var token; do { // Create token token = "wnd" + (new Date()).getTime().toString() + (Math.round(Math.random() * 50)).toString(); // Check token uniqueness } while(token in this.callbacks); return token; }, /** * Process request * * @param {Object} requestObject Request object. See Mibew API for details. * @private */ processRequest: function(requestObject) { var context = new MibewAPIExecutionContext(); // Get result function var resultFunction = this.mibewAPI.getResultFunction( requestObject.functions, this.callbacks.hasOwnProperty(requestObject.token) ); if (resultFunction === null) { // Result function not found for (var i in requestObject.functions) { if (! requestObject.functions.hasOwnProperty(i)) { continue; } // Execute functions this.processFunction(requestObject.functions[i], context); // Build and store result this.buffer.push(this.mibewAPI.buildResult( context.getResults(), requestObject.token )); } } else { // Result function found if (this.callbacks.hasOwnProperty(requestObject.token)) { // Invoke callback this.callbacks[requestObject.token](resultFunction.arguments); // Remove callback delete this.callbacks[requestObject.token]; } } }, /** * Process function * * @param {Object} functionObject Function object. See Mibew API for details * @param {MibewAPIExecutionContext} context Execution context * @private */ processFunction: function(functionObject, context) { if (! this.functions.hasOwnProperty(functionObject["function"])) { return; } // Get function arguments with replaced refences var functionArguments = context.getArgumentsList(functionObject); var results = {}; for (var i in this.functions[functionObject["function"]]) { if (! this.functions[functionObject["function"]].hasOwnProperty(i)) { continue; } // Get results results.extend(this.functions[functionObject["function"]][i]( functionArguments )); } // Add function results to the execution context context.storeFunctionResults(functionObject, results); }, /** * Send the request to the chat server * * @param {Object[]} requestsList Array of requests that must be sent to the * chat server * @private */ sendRequests: function(requestsList) { // Create new AJAX request this.ajaxRequest = new Ajax.Request( this.options.servl, this.ajaxOptions.extend({ parameters: 'data=' + this.mibewAPI.encodePackage(requestsList) }) ); }, /** * Sets up next automatic updater iteration */ runUpdater: function() { if (this.updateTimer == null) { this.update(); } this.updateTimer = setTimeout( this.update.bind(this), this.options.requestsFrequency * 1000 ); }, /** * Restarts the automatic updater */ restartUpdater: function() { // Clear timeout if (this.updateTimer) { clearTimeout(this.updateTimer); } // Clear request onComplete callback if (this.ajaxRequest._options) { this.ajaxRequest._options.onComplete = undefined; } // Update thread this.update(); // Restart updater. Try to reconnect after a while this.updateTimer = setTimeout( this.update.bind(this), 1000 ); }, /** * Send request for update thread and client code's requests * @private */ update: function() { if (this.updateTimer) { clearTimeout(this.updateTimer); } for (var i = 0; i < this.callPeriodically.length; i++) { this.callFunctions( this.callPeriodically[i].functionsListBuilder(), this.callPeriodically[i].callbackFunction ); } // Check buffer length if (this.buffer.length == 0) { // Rerun updater later this.runUpdater(); return; } try { // Send requests this.sendRequests(this.buffer); // Clear requests buffer this.buffer = []; } catch (e) { // Handle errors this.options.onUpdateError(e); } }, /** * Process response from the Core * * @param {String} responseObject The response object provided by * Ajax.Request class * @private */ receiveResponse: function(responseObject) { // Do not parse empty responses if (responseObject.response == '') { this.runUpdater(); } try { var packageObject = this.mibewAPI.decodePackage(responseObject.response); for (var i in packageObject.requests) { this.processRequest(packageObject.requests[i]); } } catch (e) { this.options.onResponseError(e); } finally { this.runUpdater(); } }, /** * Add function that can be called by the Core * * @param {String} functionName Name of the function * @param {Function} handler Provided function */ registerFunction: function(functionName, handler) { if (!(functionName in this.functions)) { this.functions[functionName] = []; } this.functions[functionName].push(handler); }, /** * Call on all AJAX transport errors * @param {Ajax.Request} transport AJAX Transport object * @param {Error} e Error object */ onTransportError: function (transport, e) { this.restartUpdater(); this.options.onTransportError(e); }, /** * Call on all timeouts */ onTimeout: function(transport) { this.restartUpdater(); this.options.onTimeout() } } ChatThreadUpdater = Class.create(); ChatThreadUpdater.prototype = { /** * @constructor * @todo Add error handlers to chatServer * @todo Think about code format */ initialize: function(chatServer, thread, options) { /** * Array of options * @type Array * @private */ this._options = options; /** * An instance of the Thread class * @type ChatThread */ this.thread = { threadid: 0, token: 0, lastid: 0, user: false }.extend(thread || {}); /** * An instance of the ChatServer class * @type ChatServer */ this.chatServer = chatServer; /** * Indicates if user can post messages * @type Boolean */ this.cansend = true; /** * Indicates if next message's sound must be skipped * @type Boolean */ this.skipNextsound = true; /** * Indicates if message input area ihn focus * @type Boolean */ this.focused = true; /** * Indicates the thread belong to this operator * @type Boolean */ this.ownThread = this._options.message != null; FrameUtils.initFrame(this._options.container); if (this._options.message) { this._options.message.onkeydown = this.handleKeyDown.bind(this); this._options.message.onfocus = (function() {this.focused = true;}).bind(this); this._options.message.onblur = (function() {this.focused = false;}).bind(this); } // Add periodic functions this.chatServer.callFunctionsPeriodically( this.updateFunctionBuilder.bind(this), this.updateChatState.bind(this) ); // Register functions this.chatServer.registerFunction( 'updateMessages', this.updateMessages.bind(this) ); this.chatServer.registerFunction( 'setupAvatar', this.setupAvatar.bind(this) ); this.chatServer.runUpdater(); }, /** * Exception handler. Updates status message */ handleException: function(e) { this.setStatus("offline, reconnecting"); this.enableInput(true); }, /** * Timeout handler. Updates status message */ handleTimeout: function() { this.setStatus("timeout, reconnecting"); this.enableInput(true); }, /** * Enables or disables input field * @param {Boolean} val Use boolean true for enable input and false otherwise */ enableInput: function(val) { if( this._options.message ) { this._options.message.disabled = !val; } }, /** * Load new messages by restarting thread updater. */ refresh: function() { this.chatServer.restartUpdater(); }, /** * Sends message to the chat server * @param {String} msg Message for send */ postMessage: function(msg) { // Check if message can be sent if(msg == "" || !this.cansend) { return; } // Disable message sending this.cansend = false; // Disable next sound this.skipNextsound = true; // Disable input if(myRealAgent != 'opera') { this.enableInput(false); } // Post message this.chatServer.callFunctions( [{ "function": "post", "arguments": { "references": {}, "return": {}, "message": msg, "threadId": this.thread.threadid, "token": this.thread.token, "user": this.thread.user } }], (function(){ this.enableInput(true); this.cansend = true; this.skipNextsound = false; if(this._options.message) { this._options.message.value = ''; this._options.message.focus(); } }).bind(this), true ); }, /** * Change user name * @param {String} newname A new user name */ changeName: function(newname) { this.skipNextsound = true; this.chatServer.callFunctions( [{ "function": "rename", "arguments": { "references": {}, "return": {}, "threadId": this.thread.threadid, "token": this.thread.token, "name": newname } }], (function(args){ if (args.errorCode) { this.handleError(args, 'cannot rename'); } }).bind(this), true ); }, /** * Send request for close chat to the core */ closeThread: function() { // Show confirmation message if can if(this._options.localizedStrings.closeConfirmation){ if(! confirm(this._options.localizedStrings.closeConfirmation)){ return; } } // Send request this.chatServer.callFunctions( [{ "function": "close", "arguments": { "references": {}, "return": {"closed": "closed"}, "threadId": this.thread.threadid, "token": this.thread.token, "lastId": this.thread.lastid, "user": this.thread.user } }], this.onThreadClosed.bind(this), true ); }, /** * Callback function for close chat request. * * Close chat window if closing success or warn on fail */ onThreadClosed: function(args) { if (args.closed) { window.close(); } else { this.handleError(args, 'cannot close'); } }, /** * Add message to the message window * @param {Object} _target Target DOM element * @param {String} message HTML message to insert */ processMessage: function(_target, message) { FrameUtils.insertIntoFrame(_target, message); }, /** * Displays typing status * @param {Boolean} istyping Indicates the other side of conversation is * typing a message or not */ showTyping: function(istyping) { if( $("typingdiv") ) { $("typingdiv").style.display=istyping ? 'inline' : 'none'; } }, /** * Update operator's avatar * @param {Array} args Array of arguments passed from the core */ setupAvatar: function(args) { if (this._options.avatar && this.thread.user) { this._options.avatar.innerHTML = args.imageLink != "" ? "\"\"\"\"/" : ""; } }, /** * Add new messages to chat window * @param {Object} args object of function arguments passed from the server * @todo Fix skipNextSound */ updateMessages: function(args){ // Update last message id if (args.lastId) { this.thread.lastid = args.lastId; } // Add messages for (var i = 0; i < args.messages.length; i++) { // TODO: Add template engine this.processMessage(this._options.container, args.messages[i]); } // Clear status string this.clearStatus(); // There are some new messages if (args.messages.length > 0) { FrameUtils.scrollDown(this._options.container); if (!this.skipNextsound) { var tsound = $('soundimg'); if (tsound == null || tsound.className.match(new RegExp("\\bisound\\b"))) { playSound(this._options.webimRoot+'/sounds/new_message.wav'); } } if (!this.focused) { window.focus(); } } this.skipNextsound = false; }, /** * Build update function to call at the core */ updateFunctionBuilder: function() { return [ { "function": "update", "arguments": { "return": {'typing': 'typing', 'canPost': 'canPost'}, "references": {}, "threadId": this.thread.threadid, "token": this.thread.token, "lastId": this.thread.lastid, "typed": (this._options.message && this._options.message.value != ''), "user": this.thread.user } } ]; }, /** * Set current chat state message * @param {Array} args Array of arguments passed from the core */ updateChatState: function(args) { if (args.errorCode) { // Something went wrong this.handleError(args, 'refresh failed'); return; } // Update typing indicator if (typeof args.typing != 'undefined') { this.showTyping(args.typing); } // Check if user can post messages if (typeof args.canPost != 'undefined') { if ((args.canPost && !this.ownThread) || (this.ownThread && ! args.canPost)) { // Refresh the page window.location.href = window.location.href; } } }, /** * Check if send key (Enter or Ctrl+Enter) pressed * @param {Boolean} ctrlpressed Indicates ctrl key is pressed or not * @param {Number} key Key code */ isSendkey: function(ctrlpressed, key) { return ((key==13 && (ctrlpressed || this._options.ignorectrl)) || (key==10)); }, /** * Key down handler * * @param {Object} k Event object */ handleKeyDown: function(k) { if( k ){ctrl=k.ctrlKey;k=k.which;} else {k=event.keyCode;ctrl=event.ctrlKey;} if( this._options.message && this.isSendkey(ctrl, k) ) { var mmsg = this._options.message.value; if( this._options.ignorectrl ) { mmsg = mmsg.replace(/[\r\n]+$/,''); } this.postMessage( mmsg ); return false; } return true; }, /** * Update status message * * @param {Array} args Array of arguments. Must contain 'errorCode' and * 'errorMessage' keys * @param {String} descr Error description */ handleError: function(args, descr) { if (args.errorCode) { this.setStatus(args.errorMessage); } else { this.setStatus('reconnecting'); } }, /** * Displays status div and sets the status string into it * * @param {String} k Status string */ showStatusDiv: function(k) { if( $("engineinfo") ) { $("engineinfo").style.display='inline'; $("engineinfo").innerHTML = k; } }, /** * Sets the status * * @param {String} k Status string */ setStatus: function(k) { if( this.statusTimeout ) clearTimeout(this.statusTimeout); this.showStatusDiv(k); this.statusTimeout = setTimeout(this.clearStatus.bind(this), 4000); }, /** * Hide the status string */ clearStatus: function() { $("engineinfo").style.display='none'; } } var Chat = { threadUpdater : {}, applyName: function() { Chat.threadUpdater.changeName($('uname').value); $('changename1').style.display='none'; $('changename2').style.display='inline'; $('unamelink').innerHTML = htmlescape($('uname').value); }, showNameField: function() { $('changename1').style.display='inline'; $('changename2').style.display='none'; } }; Behaviour.register({ '#postmessage a' : function(el) { el.onclick = function() { var message = $('msgwnd'); if( message ) Chat.threadUpdater.postMessage(message.value); }; }, 'select#predefined' : function(el) { el.onchange = function() { var message = $('msgwnd'); if(this.selectedIndex!=0) { message.value = Chat.predefinedAnswers[this.selectedIndex-1]; } this.selectedIndex = 0; message.focus(); }; }, 'div#changename2 a' : function(el) { el.onclick = function() { Chat.showNameField(); return false; }; }, 'div#changename1 a' : function(el) { el.onclick = function() { Chat.applyName(); return false; }; }, 'div#changename1 input#uname' : function(el) { el.onkeydown = function(e) { var ev = e || event; if( ev.keyCode == 13 ) { Chat.applyName(); } }; }, 'a#refresh' : function(el) { el.onclick = function() { Chat.threadUpdater.refresh(); }; }, 'a#togglesound' : function(el) { el.onclick = function() { var tsound = $('soundimg'); if(!tsound) { return; } if(tsound.className.match(new RegExp("\\bisound\\b"))) { tsound.className = "tplimage inosound"; } else { tsound.className = "tplimage isound"; } var messagePane = $('msgwnd'); if(messagePane) messagePane.focus(); }; }, 'a.closethread' : function(el) { el.onclick = function() { Chat.threadUpdater.closeThread(); }; } }); EventHelper.register(window, 'onload', function(){ Chat.cssfile = chatParams.cssfile; Chat.predefinedAnswers = chatParams.predefinedAnswers || []; Chat.localizedStrings = chatParams.localizedStrings; Chat.threadUpdater = new ChatThreadUpdater( new ChatServer(chatParams.serverParams), chatParams.threadParams, { ignorectrl: -1, container: myRealAgent=='safari'?self.frames[0]:$("chatwnd"), avatar: $("avatarwnd"), message: $("msgwnd") }.extend(chatParams.threadUpdaterParams || {}) ); });