]> jfr.im git - irc/quakenet/qwebirc.git/blame_incremental - js/ui/baseui.js
Fixes issue #127, seems I never implemented this!
[irc/quakenet/qwebirc.git] / js / ui / baseui.js
... / ...
CommitLineData
1qwebirc.ui.WINDOW_STATUS = 0x01;
2qwebirc.ui.WINDOW_QUERY = 0x02;
3qwebirc.ui.WINDOW_CHANNEL = 0x04;
4qwebirc.ui.WINDOW_CUSTOM = 0x08;
5qwebirc.ui.WINDOW_CONNECT = 0x10;
6qwebirc.ui.WINDOW_MESSAGES = 0x20;
7
8qwebirc.ui.CUSTOM_CLIENT = "custom";
9
10qwebirc.ui.BaseUI = new Class({
11 Implements: [Events],
12 initialize: function(parentElement, windowClass, uiName, options) {
13 this.options = options;
14
15 this.windows = {};
16 this.clients = {};
17 this.windows[qwebirc.ui.CUSTOM_CLIENT] = {};
18 this.windowArray = [];
19 this.windowClass = windowClass;
20 this.parentElement = parentElement;
21 this.parentElement.addClass("qwebirc");
22 this.parentElement.addClass("qwebirc-" + uiName);
23 this.firstClient = false;
24 this.commandhistory = new qwebirc.irc.CommandHistory();
25 this.clientId = 0;
26
27 this.windowFocused = true;
28
29 if(Browser.Engine.trident) {
30 var checkFocus = function() {
31 var hasFocus = document.hasFocus();
32 if(hasFocus != this.windowFocused) {
33 this.windowFocused = hasFocus;
34 this.focusChange(hasFocus);
35 }
36 }
37
38 checkFocus.periodical(100, this);
39 } else {
40 var blur = function() { if(this.windowFocused) { this.windowFocused = false; this.focusChange(false); } }.bind(this);
41 var focus = function() { if(!this.windowFocused) { this.windowFocused = true; this.focusChange(true); } }.bind(this);
42
43 /* firefox requires both */
44
45 document.addEvent("blur", blur);
46 window.addEvent("blur", blur);
47 document.addEvent("focus", focus);
48 window.addEvent("focus", focus);
49 }
50 },
51 newClient: function(client) {
52 client.id = this.clientId++;
53 client.hilightController = new qwebirc.ui.HilightController(client);
54
55 this.windows[client.id] = {}
56 this.clients[client.id] = client;
57 var w = this.newWindow(client, qwebirc.ui.WINDOW_STATUS, "Status");
58 this.selectWindow(w);
59 if(!this.firstClient) {
60 this.firstClient = true;
61 w.addLine("", "qwebirc v" + qwebirc.VERSION);
62 w.addLine("", "Copyright (C) 2008-2010 Chris Porter and the qwebirc project.");
63 w.addLine("", "http://www.qwebirc.org");
64 w.addLine("", "Licensed under the GNU General Public License, Version 2.");
65 }
66 return w;
67 },
68 getClientId: function(client) {
69 if(client == qwebirc.ui.CUSTOM_CLIENT) {
70 return qwebirc.ui.CUSTOM_CLIENT;
71 } else {
72 return client.id;
73 }
74 },
75 getWindowIdentifier: function(client, type, name) {
76 if(type == qwebirc.ui.WINDOW_MESSAGES)
77 return "-M";
78 if(type == qwebirc.ui.WINDOW_STATUS)
79 return "";
80
81 if(client == qwebirc.ui.CUSTOM_CLIENT) /* HACK */
82 return "_" + name;
83
84 return "_" + client.toIRCLower(name);
85 },
86 newWindow: function(client, type, name) {
87 var w = this.getWindow(client, type, name);
88 if($defined(w))
89 return w;
90
91 var wId = this.getWindowIdentifier(client, type, name);
92 var w = this.windows[this.getClientId(client)][wId] = new this.windowClass(this, client, type, name, wId);
93 this.windowArray.push(w);
94
95 return w;
96 },
97 getWindow: function(client, type, name) {
98 var c = this.windows[this.getClientId(client)];
99 if(!$defined(c))
100 return null;
101
102 return c[this.getWindowIdentifier(client, type, name)];
103 },
104 getActiveWindow: function() {
105 return this.active;
106 },
107 getActiveIRCWindow: function(client) {
108 if(!this.active || this.active.type == qwebirc.ui.WINDOW_CUSTOM) {
109 return this.windows[this.getClientId(client)][this.getWindowIdentifier(client, qwebirc.ui.WINDOW_STATUS)];
110 } else {
111 return this.active;
112 }
113 },
114 __setActiveWindow: function(window) {
115 this.active = window;
116 },
117 selectWindow: function(window) {
118 if(this.active)
119 this.active.deselect();
120 window.select(); /* calls setActiveWindow */
121 this.updateTitle(window.name + " - " + this.options.appTitle);
122 },
123 updateTitle: function(text) {
124 document.title = text;
125 },
126 nextWindow: function(direction) {
127 if(this.windowArray.length == 0 || !this.active)
128 return;
129
130 if(!direction)
131 direction = 1;
132
133 var index = this.windowArray.indexOf(this.active);
134 if(index == -1)
135 return;
136
137 index = index + direction;
138 if(index < 0) {
139 index = this.windowArray.length - 1;
140 } else if(index >= this.windowArray.length) {
141 index = 0;
142 }
143
144 this.selectWindow(this.windowArray[index]);
145 },
146 prevWindow: function() {
147 this.nextWindow(-1);
148 },
149 __closed: function(window) {
150 if(window.active) {
151 this.active = undefined;
152 if(this.windowArray.length == 1) {
153 this.windowArray = [];
154 } else {
155 var index = this.windowArray.indexOf(window);
156 if(index == -1) {
157 return;
158 } else if(index == 0) {
159 this.selectWindow(this.windowArray[1]);
160 } else {
161 this.selectWindow(this.windowArray[index - 1]);
162 }
163 }
164 }
165
166 this.windowArray = this.windowArray.erase(window);
167 delete this.windows[this.getClientId(window.client)][window.identifier];
168 },
169 /*
170 this shouldn't be called by overriding classes!
171 they should implement their own!
172 some form of user input MUST be received before an
173 IRC connection is made, else users are going to get
174 tricked into getting themselves glined
175 */
176 loginBox: function(callback, initialNickname, initialChannels, autoConnect, autoNick) {
177 qwebirc.ui.GenericLoginBox(this.parentElement, callback, initialNickname, initialChannels, autoConnect, autoNick, this.options.networkName);
178 },
179 focusChange: function(newValue) {
180 var window_ = this.getActiveWindow();
181 if($defined(window_))
182 window_.focusChange(newValue);
183 }
184});
185
186qwebirc.ui.StandardUI = new Class({
187 Extends: qwebirc.ui.BaseUI,
188 UICommands: qwebirc.ui.UI_COMMANDS,
189 initialize: function(parentElement, windowClass, uiName, options) {
190 this.parent(parentElement, windowClass, uiName, options);
191
192 this.tabCompleter = new qwebirc.ui.TabCompleterFactory(this);
193 this.uiOptions = new qwebirc.ui.DefaultOptionsClass(this);
194 this.customWindows = {};
195
196 var ev;
197 if(Browser.Engine.trident) {
198 ev = "keydown";
199 } else {
200 ev = "keypress";
201 }
202 document.addEvent(ev, this.__handleHotkey.bind(this));
203 },
204 __handleHotkey: function(x) {
205 if(!x.alt || x.control) {
206 if(x.key == "backspace" || x.key == "/")
207 if(!this.getInputFocused(x))
208 new Event(x).stop();
209 return;
210 }
211 var success = false;
212 if(x.key == "a" || x.key == "A") {
213 var highestNum = 0;
214 var highestIndex = -1;
215 success = true;
216
217 new Event(x).stop();
218 for(var i=0;i<this.windowArray.length;i++) {
219 var h = this.windowArray[i].hilighted;
220 if(h > highestNum) {
221 highestIndex = i;
222 highestNum = h;
223 }
224 }
225 if(highestIndex > -1)
226 this.selectWindow(this.windowArray[highestIndex]);
227 } else if(x.key >= '0' && x.key <= '9') {
228 success = true;
229
230 number = x.key - '0';
231 if(number == 0)
232 number = 10
233
234 number = number - 1;
235
236 if(number >= this.windowArray.length)
237 return;
238
239 this.selectWindow(this.windowArray[number]);
240 } else if(x.key == "left") {
241 this.prevWindow();
242 success = true;
243 } else if(x.key == "right") {
244 this.nextWindow();
245 success = true;
246 }
247 if(success)
248 new Event(x).stop();
249 },
250 getInputFocused: function(x) {
251 if($$("input").indexOf(x.target) == -1 && $$("textarea").indexOf(x.target) == -1)
252 return false;
253 return true;
254 },
255 newCustomWindow: function(name, select, type) {
256 if(!type)
257 type = qwebirc.ui.WINDOW_CUSTOM;
258
259 var w = this.newWindow(qwebirc.ui.CUSTOM_CLIENT, type, name);
260 w.addEvent("close", function(w) {
261 delete this.windows[qwebirc.ui.CUSTOM_CLIENT][w.identifier];
262 }.bind(this));
263
264 if(select)
265 this.selectWindow(w);
266
267 return w;
268 },
269 addCustomWindow: function(windowName, class_, cssClass, options) {
270 if(!$defined(options))
271 options = {};
272
273 if(this.customWindows[windowName]) {
274 this.selectWindow(this.customWindows[windowName]);
275 return;
276 }
277
278 var d = this.newCustomWindow(windowName, true);
279 this.customWindows[windowName] = d;
280
281 d.addEvent("close", function() {
282 this.customWindows[windowName] = null;
283 }.bind(this));
284
285 if(cssClass)
286 d.lines.addClass("qwebirc-" + cssClass);
287
288 var ew = new class_(d.lines, options);
289 ew.addEvent("close", function() {
290 d.close();
291 }.bind(this));
292
293 d.setSubWindow(ew);
294 },
295 embeddedWindow: function() {
296 this.addCustomWindow("Embedding wizard", qwebirc.ui.EmbedWizard, "embeddedwizard", {baseURL: this.options.baseURL});
297 },
298 optionsWindow: function() {
299 this.addCustomWindow("Options", qwebirc.ui.OptionsPane, "optionspane", this.uiOptions);
300 },
301 aboutWindow: function() {
302 this.addCustomWindow("About", qwebirc.ui.AboutPane, "aboutpane", this.uiOptions);
303 },
304 privacyWindow: function() {
305 this.addCustomWindow("Privacy policy", qwebirc.ui.PrivacyPolicyPane, "privacypolicypane", this.uiOptions);
306 },
307 feedbackWindow: function() {
308 this.addCustomWindow("Feedback", qwebirc.ui.FeedbackPane, "feedbackpane", this.uiOptions);
309 },
310 faqWindow: function() {
311 this.addCustomWindow("FAQ", qwebirc.ui.FAQPane, "faqpane", this.uiOptions);
312 },
313 urlDispatcher: function(name, window) {
314 if(name == "embedded")
315 return ["a", this.embeddedWindow.bind(this)];
316
317 if(name == "options")
318 return ["a", this.optionsWindow.bind(this)];
319
320 /* doesn't really belong here */
321 if(name == "whois") {
322 return ["span", function(nick) {
323 this.client.exec("/WHOIS " + nick);
324 }.bind(window)];
325 }
326
327 return null;
328 },
329 tabComplete: function(element) {
330 this.tabCompleter.tabComplete(element);
331 },
332 resetTabComplete: function() {
333 this.tabCompleter.reset();
334 },
335 setModifiableStylesheet: function(name) {
336 this.__styleSheet = new qwebirc.ui.style.ModifiableStylesheet(qwebirc.global.staticBaseURL + "css/" + name + qwebirc.FILE_SUFFIX + ".mcss");
337
338 if($defined(this.options.hue)) {
339 this.setModifiableStylesheetValues(this.options.hue, 0, 0);
340 } else {
341 this.setModifiableStylesheetValues(this.uiOptions.STYLE_HUE, 0, 0);
342 }
343 },
344 setModifiableStylesheetValues: function(hue, saturation, lightness) {
345 if(!$defined(this.__styleSheet))
346 return;
347 this.__styleSheet.set(function(x) {
348 return x.setHue(hue).setSaturation(x.hsb[1] + saturation).setBrightness(x.hsb[2] + lightness);
349 });
350 }
351});
352
353qwebirc.ui.NotificationUI = new Class({
354 Extends: qwebirc.ui.StandardUI,
355 initialize: function(parentElement, windowClass, uiName, options) {
356 this.parent(parentElement, windowClass, uiName, options);
357
358 this.__beeper = new qwebirc.ui.Beeper(this.uiOptions);
359 this.__flasher = new qwebirc.ui.Flasher(this.uiOptions);
360
361 this.beep = this.__beeper.beep.bind(this.__beeper);
362
363 this.flash = this.__flasher.flash.bind(this.__flasher);
364 this.cancelFlash = this.__flasher.cancelFlash.bind(this.__flasher);
365 },
366 setBeepOnMention: function(value) {
367 if(value)
368 this.__beeper.soundInit();
369 },
370 updateTitle: function(text) {
371 if(this.__flasher.updateTitle(text))
372 this.parent(text);
373 },
374 focusChange: function(value) {
375 this.parent(value);
376 this.__flasher.focusChange(value);
377 }
378});
379
380qwebirc.ui.NewLoginUI = new Class({
381 Extends: qwebirc.ui.NotificationUI,
382 loginBox: function(callbackfn, initialNickname, initialChannels, autoConnect, autoNick) {
383 this.postInitialize();
384
385 /* I'd prefer something shorter and snappier! */
386 var w = this.newCustomWindow("Connection details", true, qwebirc.ui.WINDOW_CONNECT);
387 var callback = function(args) {
388 w.close();
389 callbackfn(args);
390 };
391
392 qwebirc.ui.GenericLoginBox(w.lines, callback, initialNickname, initialChannels, autoConnect, autoNick, this.options.networkName);
393 }
394});
395
396qwebirc.ui.QuakeNetUI = new Class({
397 Extends: qwebirc.ui.NewLoginUI,
398 urlDispatcher: function(name, window) {
399 if(name == "qwhois") {
400 return ["span", function(auth) {
401 this.client.exec("/MSG Q whois #" + auth);
402 }.bind(window)];
403 }
404 return this.parent(name, window);
405 },
406 logout: function() {
407 if(!qwebirc.auth.loggedin())
408 return;
409 if(confirm("Log out?")) {
410 for(var client in this.clients) {
411 this.clients[client].quit("Logged out");
412 };
413
414 /* HACK */
415 var foo = function() { document.location = qwebirc.global.dynamicBaseURL + "auth?logout=1"; };
416 foo.delay(500);
417 }
418 }
419});
420
421qwebirc.ui.RootUI = qwebirc.ui.QuakeNetUI;
422
423qwebirc.ui.RequestTransformHTML = function(options) {
424 var HREF_ELEMENTS = {
425 "IMG": 1
426 };
427
428 var update = options.update;
429 var onSuccess = options.onSuccess;
430
431 var fixUp = function(node) {
432 if(node.nodeType != 1)
433 return;
434
435 var tagName = node.nodeName.toUpperCase();
436 if(HREF_ELEMENTS[tagName]) {
437 var attr = node.getAttribute("transform_attr");
438 var value = node.getAttribute("transform_value");
439 if($defined(attr) && $defined(value)) {
440 node.removeAttribute("transform_attr");
441 node.removeAttribute("transform_value");
442 node.setAttribute(attr, qwebirc.global.staticBaseURL + value);
443 }
444 }
445
446 for(var i=0;i<node.childNodes.length;i++)
447 fixUp(node.childNodes[i]);
448 };
449
450 delete options["update"];
451 options.onSuccess = function(tree, elements, html, js) {
452 var container = new Element("div");
453 container.set("html", html);
454 fixUp(container);
455 update.empty();
456
457 while(container.childNodes.length > 0) {
458 var x = container.firstChild;
459 container.removeChild(x);
460 update.appendChild(x);
461 }
462 onSuccess();
463 };
464
465 return new Request.HTML(options);
466};
467