X-Git-Url: https://jfr.im/git/irc/quakenet/qwebirc.git/blobdiff_plain/6dd4f8229c38b597d02908d2c5181d8437ca2c7e..ea29e3d77bb6b6f11545dd02a43883def07ea869:/js/ui/baseui.js diff --git a/js/ui/baseui.js b/js/ui/baseui.js index 02a1d9a..582a490 100644 --- a/js/ui/baseui.js +++ b/js/ui/baseui.js @@ -1,93 +1,529 @@ -WINDOW_STATUS = 1; -WINDOW_QUERY = 2; -WINDOW_CHANNEL = 3; - -var UIWindow = new Class({ - Implements: [Events], - initialize: function(parentObject, client, type, name, identifier) { - this.parentObject = parentObject; - this.type = type; - this.name = name; - this.active = false; - this.client = client; - this.identifier = identifier; - }, - updateNickList: function(nicks) { - }, - updateTopic: function(topic) { - }, - close: function() { - this.parentObject.__closed(this); - this.fireEvent("close", this); - }, - select: function() { - this.active = true; - this.parentObject.__setActiveWindow(this); - }, - deselect: function() { - this.active = false; - }, - addLine: function(type, line, colour) { - }, - errorMessage: function(message) { - this.addLine("", message, "red"); - } -}); - -var UI = new Class({ - initialize: function(parentElement, windowClass, uiName) { - this.windows = {}; - this.windowArray = []; - this.windowClass = windowClass; - this.parentElement = parentElement; - this.parentElement.addClass("qwebirc"); - this.parentElement.addClass("qwebirc-" + uiName); - }, - newClient: function(client) { - this.windows[client] = {} - var w = this.newWindow(client, WINDOW_STATUS, "Status"); - this.selectWindow(w); - - return w; - }, - newWindow: function(client, type, name) { - var identifier = name; - if(type == WINDOW_STATUS) - identifier = ""; - - var w = this.windows[client][identifier] = new this.windowClass(this, client, type, name, identifier); - this.windowArray.push(w); - - return w; - }, - getActiveWindow: function() { - return this.active; - }, - __setActiveWindow: function(window) { - this.active = window; - }, - selectWindow: function(window) { - if(this.active) - this.active.deselect(); - window.select(); /* calls setActiveWindow */ - }, - __closed: function(window) { - if(window.active) { - this.active = undefined; - if(this.windowArray.length == 1) { - this.windowArray = []; - } else { - var index = this.windowArray.indexOf(window); - if(index == 0) { - this.selectWindow(this.windowArray[1]); - } else { - this.selectWindow(this.windowArray[index - 1]); - } - - this.windowArray = this.windowArray.erase(window); - } - } - - delete this.windows[window.client][window.identifier]; - } -}); +qwebirc.ui.WINDOW_STATUS = 0x01; +qwebirc.ui.WINDOW_QUERY = 0x02; +qwebirc.ui.WINDOW_CHANNEL = 0x04; +qwebirc.ui.WINDOW_CUSTOM = 0x08; +qwebirc.ui.WINDOW_CONNECT = 0x10; +qwebirc.ui.WINDOW_MESSAGES = 0x20; + +qwebirc.ui.CUSTOM_CLIENT = "custom"; + +qwebirc.ui.BaseUI = new Class({ + Implements: [Events], + initialize: function(parentElement, windowClass, uiName, options) { + this.options = options; + + this.windows = new QHash(); + this.clients = new QHash(); + this.windows.put(qwebirc.ui.CUSTOM_CLIENT, new QHash()); + this.windowArray = []; + this.windowClass = windowClass; + this.parentElement = parentElement; + this.parentElement.addClass("qwebirc"); + this.parentElement.addClass("qwebirc-" + uiName); + this.firstClient = false; + this.commandhistory = new qwebirc.irc.CommandHistory(); + this.clientId = 0; + + this.windowFocused = true; + + if(Browser.Engine.trident) { + var checkFocus = function() { + var hasFocus = document.hasFocus(); + if(hasFocus != this.windowFocused) { + this.windowFocused = hasFocus; + this.focusChange(hasFocus); + } + } + + checkFocus.periodical(100, this); + } else { + var blur = function() { if(this.windowFocused) { this.windowFocused = false; this.focusChange(false); } }.bind(this); + var focus = function() { if(!this.windowFocused) { this.windowFocused = true; this.focusChange(true); } }.bind(this); + + /* firefox requires both */ + + document.addEvent("blur", blur); + window.addEvent("blur", blur); + document.addEvent("focus", focus); + window.addEvent("focus", focus); + } + }, + newClient: function(client) { + client.id = String(this.clientId++); + client.hilightController = new qwebirc.ui.HilightController(client); + client.addEvent("signedOn", function() { + this.fireEvent("signedOn", client); + }.bind(this)); + this.windows.put(client.id, new QHash()); + this.clients.put(client.id, client); + var w = this.newWindow(client, qwebirc.ui.WINDOW_STATUS, "Status"); + this.selectWindow(w); + if(!this.firstClient) { + this.firstClient = true; + w.addLine("", "qwebirc v" + qwebirc.VERSION); + w.addLine("", "Copyright (C) 2008-2014 Chris Porter and the qwebirc project."); + w.addLine("", "http://www.qwebirc.org"); + w.addLine("", "Licensed under the GNU General Public License, Version 2."); + } + return w; + }, + getClientId: function(client) { + if(client == qwebirc.ui.CUSTOM_CLIENT) { + return qwebirc.ui.CUSTOM_CLIENT; + } else { + return client.id; + } + }, + getWindowIdentifier: function(client, type, name) { + if(type == qwebirc.ui.WINDOW_MESSAGES) + return "-M"; + if(type == qwebirc.ui.WINDOW_STATUS) + return ""; + + if(client == qwebirc.ui.CUSTOM_CLIENT) /* HACK */ + return "_" + name; + + return "_" + client.toIRCLower(name); + }, + newWindow: function(client, type, name) { + var w = this.getWindow(client, type, name); + if($defined(w)) + return w; + + var wId = this.getWindowIdentifier(client, type, name); + var w = new this.windowClass(this, client, type, name, wId); + this.windows.get(this.getClientId(client)).put(wId, w); + this.windowArray.push(w); + + return w; + }, + getWindow: function(client, type, name) { + var c = this.windows.get(this.getClientId(client)); + if(!$defined(c)) + return null; + + return c[this.getWindowIdentifier(client, type, name)]; + }, + getActiveWindow: function() { + return this.active; + }, + getActiveIRCWindow: function(client) { + if(!this.active || this.active.type == qwebirc.ui.WINDOW_CUSTOM) { + return this.windows.get(this.getClientId(client)).get(this.getWindowIdentifier(client, qwebirc.ui.WINDOW_STATUS)); + } else { + return this.active; + } + }, + __setActiveWindow: function(window) { + this.active = window; + }, + renameWindow: function(window, name) { + if(this.getWindow(window.client, window.type, name)) + return null; + + var clientId = this.getClientId(window.client); + var index = this.windowArray.indexOf(window); + if(index == -1) + return null; + + this.windows.get(clientId).remove(window.identifier); + + var window = this.windowArray[index]; + window.name = name; + window.identifier = this.getWindowIdentifier(window.client, window.type, window.name); + + this.windows.get(clientId).put(window.identifier, this.windowArray[index]); + + if(window.active) + this.updateTitle(window.name + " - " + this.options.appTitle); + + window.rename(window.name); + return window; + }, + selectWindow: function(window) { + if(this.active) + this.active.deselect(); + window.select(); /* calls setActiveWindow */ + this.updateTitle(window.name + " - " + this.options.appTitle); + }, + updateTitle: function(text) { + document.title = text; + }, + nextWindow: function(direction) { + if(this.windowArray.length == 0 || !this.active) + return; + + if(!direction) + direction = 1; + + var index = this.windowArray.indexOf(this.active); + if(index == -1) + return; + + index = index + direction; + if(index < 0) { + index = this.windowArray.length - 1; + } else if(index >= this.windowArray.length) { + index = 0; + } + + this.selectWindow(this.windowArray[index]); + }, + prevWindow: function() { + this.nextWindow(-1); + }, + __closed: function(window) { + if(window.active) { + this.active = undefined; + if(this.windowArray.length == 1) { + this.windowArray = []; + } else { + var index = this.windowArray.indexOf(window); + if(index == -1) { + return; + } else if(index == 0) { + this.selectWindow(this.windowArray[1]); + } else { + this.selectWindow(this.windowArray[index - 1]); + } + } + } + + this.windowArray = this.windowArray.erase(window); + this.windows.get(this.getClientId(window.client)).remove(window.identifier); + }, + /* + this shouldn't be called by overriding classes! + they should implement their own! + some form of user input MUST be received before an + IRC connection is made, else users are going to get + tricked into getting themselves glined + */ + loginBox: function(callback, initialNickname, initialChannels, autoConnect, autoNick) { + qwebirc.ui.GenericLoginBox(this.parentElement, callback, initialNickname, initialChannels, autoConnect, autoNick, this.options.networkName); + }, + focusChange: function(newValue) { + var window_ = this.getActiveWindow(); + if($defined(window_)) + window_.focusChange(newValue); + } +}); + +qwebirc.ui.StandardUI = new Class({ + Extends: qwebirc.ui.BaseUI, + UICommands: qwebirc.ui.UI_COMMANDS, + initialize: function(parentElement, windowClass, uiName, options) { + this.parent(parentElement, windowClass, uiName, options); + + this.tabCompleter = new qwebirc.ui.TabCompleterFactory(this); + this.uiOptions = new qwebirc.ui.DefaultOptionsClass(this, options.uiOptionsArg); + this.customWindows = new QHash(); + + this.__styleValues = {hue: this.uiOptions.STYLE_HUE, saturation: 0, lightness: 0}; + if($defined(this.options.hue)) this.__styleValues.hue = this.options.hue; + if($defined(this.options.saturation)) this.__styleValues.saturation = this.options.saturation; + if($defined(this.options.lightness)) this.__styleValues.lightness = this.options.lightness; + + if(this.options.thue !== null) this.__styleValues.textHue = this.options.thue; + if(this.options.tsaturation !== null) this.__styleValues.textSaturation = this.options.tsaturation; + if(this.options.tlightness !== null) this.__styleValues.textLightness = this.options.tlightness; + + var ev; + if(Browser.Engine.trident) { + ev = "keydown"; + } else { + ev = "keypress"; + } + document.addEvent(ev, this.__handleHotkey.bind(this)); + }, + __handleHotkey: function(x) { + if(!x.alt || x.control) { + if(x.key == "backspace" || x.key == "/") + if(!this.getInputFocused(x)) + new Event(x).stop(); + return; + } + var success = false; + if(x.key == "a" || x.key == "A") { + var highestNum = 0; + var highestIndex = -1; + success = true; + + new Event(x).stop(); + for(var i=0;i highestNum) { + highestIndex = i; + highestNum = h; + } + } + if(highestIndex > -1) + this.selectWindow(this.windowArray[highestIndex]); + } else if(x.key >= '0' && x.key <= '9') { + success = true; + + number = x.key - '0'; + if(number == 0) + number = 10 + + number = number - 1; + + if(number >= this.windowArray.length) + return; + + this.selectWindow(this.windowArray[number]); + } else if(x.key == "left") { + this.prevWindow(); + success = true; + } else if(x.key == "right") { + this.nextWindow(); + success = true; + } + if(success) + new Event(x).stop(); + }, + getInputFocused: function(x) { + if($$("input").indexOf(x.target) == -1 && $$("textarea").indexOf(x.target) == -1) + return false; + return true; + }, + newCustomWindow: function(name, select, type) { + if(!type) + type = qwebirc.ui.WINDOW_CUSTOM; + + var w = this.newWindow(qwebirc.ui.CUSTOM_CLIENT, type, name); + w.addEvent("close", function(w) { + this.windows.get(qwebirc.ui.CUSTOM_CLIENT).remove(w.identifier); + }.bind(this)); + + if(select) + this.selectWindow(w); + + return w; + }, + addCustomWindow: function(windowName, class_, cssClass, options) { + if(!$defined(options)) + options = {}; + + if(this.customWindows.contains(windowName)) { + this.selectWindow(this.customWindows.get(windowName)); + return; + } + + var d = this.newCustomWindow(windowName, true); + this.customWindows.put(windowName, d); + + d.addEvent("close", function() { + this.customWindows.remove(windowName); + }.bind(this)); + + if(cssClass) + d.lines.addClass("qwebirc-" + cssClass); + + var ew = new class_(d.lines, options); + ew.addEvent("close", function() { + d.close(); + }.bind(this)); + + d.setSubWindow(ew); + }, + embeddedWindow: function() { + this.addCustomWindow("Add webchat to your site", qwebirc.ui.EmbedWizard, "embeddedwizard", {baseURL: this.options.baseURL, uiOptions: this.uiOptions, optionsCallback: function() { + this.optionsWindow(); + }.bind(this)}); + }, + optionsWindow: function() { + this.addCustomWindow("Options", qwebirc.ui.OptionsPane, "optionspane", this.uiOptions); + }, + aboutWindow: function() { + this.addCustomWindow("About", qwebirc.ui.AboutPane, "aboutpane", this.uiOptions); + }, + privacyWindow: function() { + this.addCustomWindow("Privacy policy", qwebirc.ui.PrivacyPolicyPane, "privacypolicypane", this.uiOptions); + }, + feedbackWindow: function() { + this.addCustomWindow("Feedback", qwebirc.ui.FeedbackPane, "feedbackpane", this.uiOptions); + }, + helpWindow: function() { + this.addCustomWindow("Help!", qwebirc.ui.HelpPane, "helppane", this.uiOptions); + }, + urlDispatcher: function(name, window) { + if(name == "embedded") + return ["a", this.embeddedWindow.bind(this)]; + + if(name == "options") + return ["a", this.optionsWindow.bind(this)]; + + /* doesn't really belong here */ + if(name == "whois") { + return ["span", function(nick) { + if(this.uiOptions.QUERY_ON_NICK_CLICK) { + window.client.exec("/QUERY " + nick); + } else { + window.client.exec("/WHOIS " + nick); + } + }.bind(this)]; + } + + return null; + }, + tabComplete: function(element) { + this.tabCompleter.tabComplete(element); + }, + resetTabComplete: function() { + this.tabCompleter.reset(); + }, + setModifiableStylesheet: function(name) { + this.__styleSheet = new qwebirc.ui.style.ModifiableStylesheet(qwebirc.global.staticBaseURL + "css/" + name + qwebirc.FILE_SUFFIX + ".mcss"); + this.setModifiableStylesheetValues({}); + }, + setModifiableStylesheetValues: function(values) { + for(var k in values) + this.__styleValues[k] = values[k]; + + if(!$defined(this.__styleSheet)) + return; + + var back = {hue: this.__styleValues.hue, lightness: this.__styleValues.lightness, saturation: this.__styleValues.saturation}; + var front = {hue: this.__styleValues.textHue, lightness: this.__styleValues.textLightness, saturation: this.__styleValues.textSaturation}; + + if(!this.__styleValues.textHue && !this.__styleValues.textLightness && !this.__styleValues.textSaturation) + front = back; + + var colours = { + back: back, + front: front + }; + + this.__styleSheet.set(function() { + var mode = arguments[0]; + if(mode == "c") { + var t = colours[arguments[2]]; + var x = new Color(arguments[1]); + var c = x.setHue(t.hue).setSaturation(x.hsb[1] + t.saturation).setBrightness(x.hsb[2] + t.lightness); + if(c == "255,255,255") /* IE confuses white with transparent... */ + c = "255,255,254"; + + return "rgb(" + c + ")"; + } else if(mode == "o") { + return this.uiOptions[arguments[1]] ? arguments[2] : arguments[3]; + } + }.bind(this)); + } +}); + +qwebirc.ui.NotificationUI = new Class({ + Extends: qwebirc.ui.StandardUI, + initialize: function(parentElement, windowClass, uiName, options) { + this.parent(parentElement, windowClass, uiName, options); + + this.__beeper = new qwebirc.ui.Beeper(this.uiOptions); + this.__flasher = new qwebirc.ui.Flasher(this.uiOptions); + + this.beep = this.__beeper.beep.bind(this.__beeper); + + this.flash = this.__flasher.flash.bind(this.__flasher); + this.cancelFlash = this.__flasher.cancelFlash.bind(this.__flasher); + }, + setBeepOnMention: function(value) { + if(value) + this.__beeper.soundInit(); + }, + updateTitle: function(text) { + if(this.__flasher.updateTitle(text)) + this.parent(text); + }, + focusChange: function(value) { + this.parent(value); + this.__flasher.focusChange(value); + } +}); + +qwebirc.ui.NewLoginUI = new Class({ + Extends: qwebirc.ui.NotificationUI, + loginBox: function(callbackfn, initialNickname, initialChannels, autoConnect, autoNick) { + this.postInitialize(); + + /* I'd prefer something shorter and snappier! */ + var w = this.newCustomWindow("Connection details", true, qwebirc.ui.WINDOW_CONNECT); + var callback = function(args) { + w.close(); + callbackfn(args); + }; + + qwebirc.ui.GenericLoginBox(w.lines, callback, initialNickname, initialChannels, autoConnect, autoNick, this.options.networkName); + } +}); + +qwebirc.ui.QuakeNetUI = new Class({ + Extends: qwebirc.ui.NewLoginUI, + urlDispatcher: function(name, window) { + if(name == "qwhois") { + return ["span", function(auth) { + this.client.exec("/MSG Q whois #" + auth); + }.bind(window)]; + } + return this.parent(name, window); + }, + logout: function() { + if(!qwebirc.auth.loggedin()) + return; + if(confirm("Log out?")) { + this.clients.each(function(k, v) { + v.quit("Logged out"); + }, this); + + /* HACK */ + var foo = function() { document.location = qwebirc.global.dynamicBaseURL + "auth?logout=1"; }; + foo.delay(500); + } + } +}); + +qwebirc.ui.RootUI = qwebirc.ui.QuakeNetUI; + +qwebirc.ui.RequestTransformHTML = function(options) { + var HREF_ELEMENTS = { + "IMG": 1 + }; + + var update = options.update; + var onSuccess = options.onSuccess; + + var fixUp = function(node) { + if(node.nodeType != 1) + return; + + var tagName = node.nodeName.toUpperCase(); + if(HREF_ELEMENTS[tagName]) { + var attr = node.getAttribute("transform_attr"); + var value = node.getAttribute("transform_value"); + if($defined(attr) && $defined(value)) { + node.removeAttribute("transform_attr"); + node.removeAttribute("transform_value"); + node.setAttribute(attr, qwebirc.global.staticBaseURL + value); + } + } + + for(var i=0;i 0) { + var x = container.firstChild; + container.removeChild(x); + update.appendChild(x); + } + onSuccess(); + }; + + return new Request.HTML(options); +}; +