]> jfr.im git - irc/quakenet/qwebirc.git/blame - js/ui/panes/options.js
Merge pull request #402 from retropc/reqs
[irc/quakenet/qwebirc.git] / js / ui / panes / options.js
CommitLineData
85449eee
CP
1qwebirc.ui.supportsFocus = function() {
2 var ua = navigator.userAgent;
3 if(!$defined(ua))
4 return [true];
5
6 if(Browser.Engine.ipod || ua.indexOf("Konqueror") != -1)
7 return [false, false];
8
9 return [true];
10}
11
c0f2f367
CP
12/**
13 * Note that options are settable by the uioptions url arg by default unless you specifiy
14 * settableByURL...
15 */
ebb21d2e 16qwebirc.config.DEFAULT_OPTIONS = [
9d0347c5 17 [1, "BEEP_ON_MENTION", "Beep on activity", true],
f63006ab
CP
18 [16, "NOTIFICATIONS", "Emit HTML5 notifications on activity", false, {
19 enabled: function() {
20 if(!("Notification" in window))
21 return [false, false]; /* [disabled, default_value] */
22 return [true];
23 },
24 applyChanges: function(value, ui) {
25 if(ui.setNotifications)
26 ui.setNotifications(value);
27 }
28 }],
2fb3b3b0 29 [7, "FLASH_ON_MENTION", "Flash titlebar when nick mentioned or on query activity", true, {
85449eee 30 enabled: qwebirc.ui.supportsFocus
2fb3b3b0 31 }],
7af2d0c1
CP
32 [2, "DEDICATED_MSG_WINDOW", "Send privmsgs to dedicated messages window", false],
33 [4, "DEDICATED_NOTICE_WINDOW", "Send notices to dedicated message window", false],
9aea28e8 34 [3, "NICK_OV_STATUS", "Show status (@/+) before nicknames in channel lines", true],
13afa38f 35 /* 5 and 6 are reserved */
85449eee
CP
36 [8, "LASTPOS_LINE", "Show a last position indicator for each window", true, {
37 enabled: qwebirc.ui.supportsFocus
38 }],
0a5b0904 39 [9, "NICK_COLOURS", "Automatically colour nicknames", false],
889ecb33 40 [10, "HIDE_JOINPARTS", "Hide JOINS/PARTS/QUITS", false],
ffb5ea9a
CP
41 [11, "STYLE_HUE", "Adjust user interface hue", function(ui) {
42 return {class_: qwebirc.config.HueOption, default_: ui.__styleValues.hue};
889ecb33 43 }, {
82cf3f26 44 applyChanges: function(value, ui) {
6f8a20df 45 ui.setModifiableStylesheetValues({hue: value});
889ecb33 46 }
cbef082a 47 }],
5f464ed5 48 [12, "QUERY_ON_NICK_CLICK", "Query on nickname click in channel", false],
22538fb7 49 [13, "SHOW_NICKLIST", "Show nickname list in channels", qwebirc.util.deviceHasKeyboard()],
a4a71818
CP
50 [14, "SHOW_TIMESTAMPS", "Show timestamps", true], /* we rely on the hue update */
51 [15, "SIDE_TABS", "Show tabs on the side", false, {
52 enabled: function() {
53 if(Browser.Engine.trident && Browser.Engine.version < 8)
54 return [false, false]; /* [disabled, default_value] */
55 return [true];
56 },
57 applyChanges: function(value, ui) {
58 ui.setSideTabs(value);
59 }
60 }]
ebb21d2e
CP
61];
62
13afa38f
CP
63qwebirc.config.QUAKENET_OPTIONS = [
64 [5, "ACCEPT_SERVICE_INVITES", "Automatically join channels when invited by Q", false, {
65 settableByURL: false
66 }],
67 [6, "USE_HIDDENHOST", "Hide your hostmask when authed to Q (+x)", true, {
68 settableByURL: false
69 }]
70];
71
ebb21d2e
CP
72qwebirc.config.DefaultOptions = null;
73
c38a0240 74qwebirc.config.Input = new Class({
fb71087a 75 initialize: function(parent, option, position, parentObject) {
c38a0240
CP
76 this.option = option;
77 this.value = option.value;
2fb3b3b0 78 this.enabled = this.option.enabled;
c38a0240 79 this.position = position;
fb71087a
CP
80 this.parentElement = parent;
81 this.parentObject = parentObject;
c38a0240
CP
82
83 this.render();
84 },
a9cbd00d 85 createInput: function(type, parent, name, selected, id) {
a1e826c7
CP
86 if(!$defined(parent))
87 parent = this.parentElement;
88
a9cbd00d 89 return qwebirc.util.createInput(type, parent, name, selected, this.option.id);
a1e826c7 90 },
c38a0240
CP
91 FE: function(element, parent) {
92 var n = new Element(element);
93 if(!$defined(parent))
fb71087a 94 parent = this.parentElement;
c38a0240
CP
95
96 parent.appendChild(n);
97 return n;
98 },
99 focus: function() {
100 this.mainElement.focus();
fb71087a
CP
101 },
102 render: function() {
103 this.event("render", this.mainElement);
104 },
82cf3f26
CP
105 applyChanges: function() {
106 this.event("applyChanges", [this.get(), this.parentObject.optionObject.ui]);
fb71087a
CP
107 },
108 event: function(name, x) {
109 if(!$defined(this.option.extras))
110 return;
111 var t = this.option.extras[name];
112 if(!$defined(t))
113 return;
114
115 t.pass(x, this)();
889ecb33
CP
116 },
117 cancel: function() {
c38a0240
CP
118 }
119});
120
121qwebirc.config.TextInput = new Class({
122 Extends: qwebirc.config.Input,
123 render: function() {
a1e826c7 124 var i = this.createInput("text");
c38a0240
CP
125 this.mainElement = i;
126
c38a0240 127 i.value = this.value;
2fb3b3b0
CP
128 i.disabled = !this.enabled;
129
fb71087a 130 this.parent();
c38a0240
CP
131 },
132 get: function() {
82cf3f26 133 return this.mainElement.value;
c38a0240
CP
134 }
135});
136
889ecb33
CP
137qwebirc.config.HueInput = new Class({
138 Extends: qwebirc.config.Input,
a1fe9cd7 139 wide: true,
889ecb33
CP
140 render: function() {
141 var i = new Element("div");
142 i.addClass("qwebirc-optionspane");
143 i.addClass("hue-slider");
889ecb33
CP
144 this.parentElement.appendChild(i);
145
146 var k = new Element("div");
147 k.addClass("knob");
1f7d5f9e
CP
148 if(Browser.Engine.trident) {
149 k.setStyle("top", "0px");
150 k.setStyle("background-color", "black");
151 }
152
889ecb33
CP
153 i.appendChild(k);
154
155 var slider = new Slider(i, k, {steps: 36, range: [0, 369], wheel: true});
156 slider.set(this.value);
157 this.startValue = this.value;
158
159 slider.addEvent("change", function(step) {
160 this.value = step;
82cf3f26 161 this.applyChanges();
889ecb33
CP
162 }.bind(this));
163 this.mainElement = i;
164
165 if(!this.enabled)
166 slider.detach();
167
168 this.parent();
169 },
170 get: function() {
82cf3f26 171 return this.value;
889ecb33
CP
172 },
173 cancel: function() {
174 this.value = this.startValue;
82cf3f26 175 this.applyChanges();
889ecb33
CP
176 }
177});
178
c38a0240
CP
179qwebirc.config.CheckInput = new Class({
180 Extends: qwebirc.config.Input,
181 render: function() {
a9cbd00d 182 var i = this.createInput("checkbox", null, null, null, this.id);
c38a0240
CP
183 this.mainElement = i;
184
c38a0240 185 i.checked = this.value;
2fb3b3b0
CP
186 i.disabled = !this.enabled;
187
fb71087a 188 this.parent();
c38a0240
CP
189 },
190 get: function() {
82cf3f26 191 return this.mainElement.checked;
c38a0240
CP
192 }
193});
194
195qwebirc.config.RadioInput = new Class({
196 Extends: qwebirc.config.Input,
197 render: function() {
198 var value = this.option.options;
199
200 this.elements = [];
2fb3b3b0 201
c38a0240
CP
202 for(var i=0;i<value.length;i++) {
203 var d = this.FE("div", this.parentObject);
a1e826c7 204 var e = this.createInput("radio", d, "options_radio" + this.position, i == this.option.position);
c38a0240 205 this.elements.push(e);
2fb3b3b0
CP
206 e.disabled = !this.enabled;
207
c38a0240
CP
208 if(i == 0)
209 this.mainElement = e;
210
211 d.appendChild(document.createTextNode(value[i][0]));
212 };
fb71087a 213 this.parent();
c38a0240
CP
214 },
215 get: function() {
216 for(var i=0;i<this.elements.length;i++) {
217 var x = this.elements[i];
218 if(x.checked) {
219 this.option.position = i;
82cf3f26 220 return this.option.options[i][1];
c38a0240
CP
221 }
222 }
223 }
224});
225
ebb21d2e 226qwebirc.config.Option = new Class({
fb71087a 227 initialize: function(optionId, prefix, label, default_, extras) {
c38a0240 228 this.prefix = prefix;
ebb21d2e 229 this.label = label;
c38a0240
CP
230 this.default_ = default_;
231 this.optionId = optionId;
fb71087a 232 this.extras = extras;
2fb3b3b0
CP
233
234 if($defined(extras) && $defined(extras.enabled)) {
235 var enabledResult = extras.enabled();
236 this.enabled = enabledResult[0];
237
238 if(!enabledResult[0] && enabledResult.length > 1)
239 this.default_ = enabledResult[1];
240 } else {
241 this.enabled = true;
c0f2f367
CP
242 }
243
244 if($defined(extras) && $defined(extras.settableByURL)) {
245 this.settableByURL = extras.settableByURL;
246 } else {
247 this.settableByURL = true;
248 }
c38a0240
CP
249 },
250 setSavedValue: function(x) {
01343497
CP
251 if(this.enabled)
252 this.value = x;
1f7d5f9e 253 }
ebb21d2e
CP
254});
255
c38a0240
CP
256qwebirc.config.RadioOption = new Class({
257 Extends: qwebirc.config.Option,
258 Element: qwebirc.config.RadioInput,
fb71087a 259 initialize: function(optionId, prefix, label, default_, extras, options) {
c38a0240
CP
260 this.options = options.map(function(x) {
261 if(typeof(x) == "string")
262 return [x, x];
263 return x;
264 });
265 this.defaultposition = default_;
266
fb71087a 267 this.parent(optionId, prefix, label, this.options[default_][1], extras);
c38a0240
CP
268 },
269 setSavedValue: function(x) {
270 for(var i=0;i<this.options.length;i++) {
271 var y = this.options[i][1];
272 if(x == y) {
273 this.position = i;
274 this.value = x;
275 return;
276 }
277 }
278 this.position = this.defaultposition;
279 this.value = this.default_;
280 }
281});
282
283qwebirc.config.TextOption = new Class({
284 Extends: qwebirc.config.Option,
285 Element: qwebirc.config.TextInput
286});
287
288qwebirc.config.CheckOption = new Class({
289 Extends: qwebirc.config.Option,
290 Element: qwebirc.config.CheckInput
291});
292
889ecb33
CP
293qwebirc.config.HueOption = new Class({
294 Extends: qwebirc.config.Option,
295 Element: qwebirc.config.HueInput
296});
297
ebb21d2e 298qwebirc.ui.Options = new Class({
fb71087a 299 initialize: function(ui) {
ffb5ea9a
CP
300 this.ui = ui;
301
ebb21d2e
CP
302 if(!$defined(qwebirc.config.DefaultOptions))
303 this.__configureDefaults();
304
c38a0240 305 this.optionList = qwebirc.config.DefaultOptions.slice();
ffb5ea9a
CP
306 this.optionHash = {};
307
c38a0240
CP
308 this._setup();
309 this.optionList.forEach(function(x) {
310 x.setSavedValue(this._get(x));
311 this.optionHash[x.prefix] = x;
312 this[x.prefix] = x.value;
ebb21d2e
CP
313 }.bind(this));
314 },
315 __configureDefaults: function() {
13afa38f
CP
316 var combined = qwebirc.config.DEFAULT_OPTIONS.slice(0);
317
318 var xo = null;
319 if(this.ui.options.networkName == "QuakeNet") /* HACK */
320 xo = qwebirc.config.QUAKENET_OPTIONS;
321
322 if(xo)
323 for(var i=0;i<xo.length;i++)
324 combined.push(xo[i]);
325
326 qwebirc.config.DefaultOptions = combined.map(function(x) {
c38a0240
CP
327 var optionId = x[0];
328 var prefix = x[1];
329 var label = x[2];
330 var default_ = x[3];
fb71087a
CP
331 var moreextras = x[4];
332 var extras = x[5];
ebb21d2e 333
c38a0240
CP
334 var stype = typeof(default_);
335 if(stype == "number") {
fb71087a 336 return new qwebirc.config.RadioOption(optionId, prefix, label, default_, moreextras, extra);
ebb21d2e 337 } else {
c38a0240
CP
338 var type;
339 if(stype == "boolean") {
340 type = qwebirc.config.CheckOption;
b076b3ab 341 } else if(stype == "function") {
ffb5ea9a 342 var options = default_.call(this, this.ui);
889ecb33 343 type = options.class_;
b076b3ab
CP
344 default_ = options.default_;
345 } else {
346 type = qwebirc.config.TextOption;
c38a0240 347 }
fb71087a 348 return new type(optionId, prefix, label, default_, moreextras);
ebb21d2e 349 }
ffb5ea9a 350 }, this);
c38a0240
CP
351 },
352 setValue: function(option, value) {
353 this.optionHash[option.prefix].value = value;
354 this[option.prefix] = value;
355 },
ffb5ea9a
CP
356 setValueByPrefix: function(prefix, value) {
357 this.optionHash[prefix].value = value;
358 this[prefix] = value;
359 },
c38a0240
CP
360 getOptionList: function() {
361 return this.optionList;
362 },
363 _get: function(x) {
364 return x.default_;
365 },
366 _setup: function() {
367 },
368 flush: function() {
ebb21d2e
CP
369 }
370});
371
372qwebirc.ui.OptionsPane = new Class({
c38a0240
CP
373 Implements: [Events],
374 initialize: function(parentElement, optionObject) {
ebb21d2e 375 this.parentElement = parentElement;
c38a0240 376 this.optionObject = optionObject;
ebb21d2e
CP
377
378 this.createElements();
379 },
380 createElements: function() {
381 var FE = function(element, parent) {
382 var n = new Element(element);
383 parent.appendChild(n);
384 return n;
385 };
ebb21d2e
CP
386
387 var t = FE("table", this.parentElement);
388 var tb = FE("tbody", t);
389
c38a0240
CP
390 this.boxList = [];
391
392 var optList = this.optionObject.getOptionList();
393 for(var i=0;i<optList.length;i++) {
394 var x = optList[i];
395
ebb21d2e 396 var row = FE("tr", tb);
a1fe9cd7
CP
397 var cell = FE("td", row);
398
a9cbd00d 399 x.id = qwebirc.util.generateID();
a1fe9cd7
CP
400 var ele = new x.Element(cell, x, i, this);
401 this.boxList.push([x, ele]);
402 var eleElement = cell.lastChild;
403 if(ele.wide)
404 cell.removeChild(eleElement);
c38a0240 405
a1fe9cd7
CP
406 var label = new Element("label", {"for": x.id});
407 cell.appendChild(label);
c38a0240 408
a1fe9cd7
CP
409 var eleParent;
410 if(ele.wide) {
411 label.set("text", x.label + ":");
412 cell.appendChild(eleElement);
413 } else {
414 label.set("text", x.label);
415 }
c38a0240
CP
416 }
417
418 var r = FE("tr", tb);
419 var cella = FE("td", r);
a1fe9cd7 420 var save = qwebirc.util.createInput("submit", cella);
c38a0240
CP
421 save.value = "Save";
422
423 save.addEvent("click", function() {
424 this.save();
425 this.fireEvent("close");
426 }.bind(this));
427
a1fe9cd7 428 var cancel = qwebirc.util.createInput("submit", cella);
c38a0240
CP
429 cancel.value = "Cancel";
430 cancel.addEvent("click", function() {
889ecb33 431 this.cancel();
c38a0240
CP
432 this.fireEvent("close");
433 }.bind(this));
434 },
435 save: function() {
436 this.boxList.forEach(function(x) {
437 var option = x[0];
438 var box = x[1];
439 this.optionObject.setValue(option, box.get());
ebb21d2e 440 }.bind(this));
82cf3f26
CP
441 this.boxList.forEach(function(x) {
442 x[1].applyChanges();
ebb21d2e 443 }.bind(this));
c38a0240 444 this.optionObject.flush();
889ecb33
CP
445 },
446 cancel: function() {
447 this.boxList.forEach(function(x) {
448 x[1].cancel();
449 }.bind(this));
ebb21d2e
CP
450 }
451});
c38a0240
CP
452
453qwebirc.ui.CookieOptions = new Class({
454 Extends: qwebirc.ui.Options,
455 _setup: function() {
456 this.__cookie = new Hash.Cookie("opt1", {duration: 3650, autoSave: false});
457 },
458 _get: function(x) {
459 var v = this.__cookie.get(x.optionId);
460 if(!$defined(v))
461 return x.default_;
462
463 return v;
464 },
465 flush: function() {
466 this.__cookie.erase();
467 this._setup();
468
469 this.getOptionList().forEach(function(x) {
470 this.__cookie.set(x.optionId, x.value);
471 }.bind(this));
472 this.__cookie.save();
473 }
474});
475
c0f2f367
CP
476qwebirc.ui.SuppliedArgOptions = new Class({
477 Extends: qwebirc.ui.CookieOptions,
478 initialize: function(ui, arg) {
ea29e3d7 479 var p = new QHash();
c0f2f367 480
8d1217eb
CP
481 if($defined(arg) && arg != "" && arg.length > 2) {
482 var checksum = arg.substr(arg.length - 2, 2);
483 var decoded = qwebirc.util.b64Decode(arg.substr(0, arg.length - 2));
484
10d70bfb
CP
485 if(decoded && (new qwebirc.util.crypto.MD5().digest(decoded).slice(0, 2) == checksum)) {
486 var p2 = qwebirc.util.parseURI("?" + decoded);
ea29e3d7
CP
487 p2.each(function(k, v) {
488 p.put(k, JSON.decode(v, true));
489 });
10d70bfb 490 }
c0f2f367
CP
491 }
492
493 this.parsedOptions = p;
494 this.parent(ui);
495 },
496 _get: function(x) {
497 if(x.settableByURL !== true)
498 return this.parent(x);
499
ea29e3d7 500 var opt = this.parsedOptions.get(String(x.optionId));
c0f2f367
CP
501 if(!$defined(opt))
502 return this.parent(x);
503
504 return opt;
505 },
506 serialise: function() {
507 var result = [];
508 this.getOptionList().forEach(function(x) {
509 if(x.settableByURL && x.default_ != x.value)
10d70bfb 510 result.push(x.optionId + "=" + JSON.encode(x.value));
c0f2f367
CP
511 }.bind(this));
512
8d1217eb
CP
513 var raw = result.join("&");
514 var checksum = new qwebirc.util.crypto.MD5().digest(raw).slice(0, 2);
515 return (qwebirc.util.b64Encode(raw)).replaceAll("=", "") + checksum;
c0f2f367
CP
516 }
517});
518
c38a0240 519qwebirc.ui.DefaultOptionsClass = new Class({
c0f2f367 520 Extends: qwebirc.ui.SuppliedArgOptions
c38a0240 521});