Correctly proxy events for plugin components other than Network()
[KiwiIRC.git] / client / src / app.js
CommitLineData
eaaf73b0 1// Holds anything kiwi client specific (ie. front, gateway, _kiwi.plugs..)\r
9df70c63
D
2/**\r
3* @namespace\r
4*/\r
eaaf73b0 5var _kiwi = {};\r
9df70c63 6\r
41cba2fd 7_kiwi.misc = {};\r
eaaf73b0
D
8_kiwi.model = {};\r
9_kiwi.view = {};\r
10_kiwi.applets = {};\r
b62c8381
D
11\r
12\r
13/**\r
14 * A global container for third party access\r
15 * Will be used to access a limited subset of kiwi functionality\r
16 * and data (think: plugins)\r
17 */\r
eaaf73b0 18_kiwi.global = {\r
4c424593 19 build_version: '', // Kiwi IRC version this is built from (Set from index.html)\r
161b5673 20 settings: undefined, // Instance of _kiwi.model.DataStore\r
2a89ba11
D
21 plugins: undefined, // Instance of _kiwi.model.PluginManager\r
22 events: undefined, // Instance of PluginInterface\r
1e9f97d6
D
23 utils: {}, // References to misc. re-usable helpers / functions\r
24\r
25 initUtils: function() {\r
26 this.utils.randomString = randomString;\r
27 this.utils.secondsToTime = secondsToTime;\r
28 this.utils.parseISO8601 = parseISO8601;\r
29 this.utils.escapeRegex = escapeRegex;\r
30 this.utils.formatIRCMsg = formatIRCMsg;\r
31 this.utils.styleText = styleText;\r
32 this.utils.hsl2rgb = hsl2rgb;\r
33 },\r
34\r
ed8b91d6
D
35 rpc: function() {\r
36 _kiwi.gateway.rpc.call.call(_kiwi.gateway.rpc, arguments);\r
37 },\r
161b5673 38\r
f1452889
D
39 addMediaMessageType: function(match, buildHtml) {\r
40 _kiwi.view.MediaMessage.addType(match, buildHtml);\r
41 },\r
42\r
161b5673
D
43 // Event managers for plugins\r
44 components: {\r
45 EventComponent: function(event_source, proxy_event_name) {\r
9b609554
D
46 /*\r
47 * proxyEvent() listens for events then re-triggers them on its own\r
48 * event emitter. Why? So we can .off() on this emitter without\r
49 * effecting the source of events. Handy for plugins that we don't\r
50 * trust meddling with the core events.\r
51 *\r
52 * If listening for 'all' events the arguments are as follows:\r
53 * 1. Name of the triggered event\r
54 * 2. The event data\r
55 * For all other events, we only have one argument:\r
56 * 1. The event data\r
57 *\r
58 * When this is used via `new kiwi.components.Network()`, this listens\r
59 * for 'all' events so the first argument is the event name which is\r
60 * the connection ID. We don't want to re-trigger this event name so\r
61 * we need to juggle the arguments to find the real event name we want\r
62 * to emit.\r
63 */\r
161b5673 64 function proxyEvent(event_name, event_data) {\r
9b609554 65 if (proxy_event_name == 'all') {\r
9b609554 66 } else {\r
161b5673 67 event_data = event_name.event_data;\r
f1452889 68 event_name = event_name.event_name;\r
161b5673 69 }\r
f1452889 70\r
161b5673
D
71 this.trigger(event_name, event_data);\r
72 }\r
73\r
74 // The event we are to proxy\r
75 proxy_event_name = proxy_event_name || 'all';\r
76\r
161b5673
D
77 _.extend(this, Backbone.Events);\r
78 this._source = event_source;\r
79\r
80 // Proxy the events to this dispatcher\r
81 event_source.on(proxy_event_name, proxyEvent, this);\r
82\r
83 // Clean up this object\r
84 this.dispose = function () {\r
85 event_source.off(proxy_event_name, proxyEvent);\r
86 this.off();\r
87 delete this.event_source;\r
88 };\r
89 },\r
90\r
91 Network: function(connection_id) {\r
92 var connection_event;\r
93\r
a9b1c5d6 94 // If no connection id given, use all connections\r
161b5673
D
95 if (typeof connection_id !== 'undefined') {\r
96 connection_event = 'connection:' + connection_id.toString();\r
18ae1a72
D
97 } else {\r
98 connection_event = 'connection';\r
161b5673
D
99 }\r
100\r
a9b1c5d6
D
101 // Helper to get the network object\r
102 var getNetwork = function() {\r
103 var network = typeof connection_id === 'undefined' ?\r
104 _kiwi.app.connections.active_connection :\r
105 _kiwi.app.connections.getByConnectionId(connection_id);\r
106\r
107 return network ?\r
108 network :\r
109 undefined;\r
110 };\r
111\r
112 // Create the return object (events proxy from the gateway)\r
161b5673 113 var obj = new this.EventComponent(_kiwi.gateway, connection_event);\r
a9b1c5d6
D
114\r
115 // Proxy several gateway functions onto the return object\r
161b5673
D
116 var funcs = {\r
117 kiwi: 'kiwi', raw: 'raw', kick: 'kick', topic: 'topic',\r
118 part: 'part', join: 'join', action: 'action', ctcp: 'ctcp',\r
9de52d29 119 ctcpRequest: 'ctcpRequest', ctcpResponse: 'ctcpResponse',\r
72db27e4 120 notice: 'notice', msg: 'privmsg', changeNick: 'changeNick',\r
a9b1c5d6 121 channelInfo: 'channelInfo', mode: 'mode', quit: 'quit',\r
161b5673
D
122 };\r
123\r
161b5673
D
124 _.each(funcs, function(gateway_fn, func_name) {\r
125 obj[func_name] = function() {\r
126 var fn_name = gateway_fn;\r
127\r
128 // Add connection_id to the argument list\r
129 var args = Array.prototype.slice.call(arguments, 0);\r
130 args.unshift(connection_id);\r
131\r
132 // Call the gateway function on behalf of this connection\r
133 return _kiwi.gateway[fn_name].apply(_kiwi.gateway, args);\r
134 };\r
135 });\r
136\r
a9b1c5d6
D
137 // Add the networks getters/setters\r
138 obj.get = function() {\r
139 var network = getNetwork();\r
140 if (!network) {\r
141 return;\r
142 }\r
143\r
144 return network.get.apply(network, arguments);\r
145 };\r
146\r
147 obj.set = function() {\r
148 var network = getNetwork();\r
149 if (!network) {\r
150 return;\r
151 }\r
152\r
153 return network.set.apply(network, arguments);\r
154 };\r
155\r
161b5673
D
156 return obj;\r
157 },\r
158\r
159 ControlInput: function() {\r
160 var obj = new this.EventComponent(_kiwi.app.controlbox);\r
161 var funcs = {\r
d63a5832 162 run: 'processInput', addPluginIcon: 'addPluginIcon'\r
161b5673
D
163 };\r
164\r
165 _.each(funcs, function(controlbox_fn, func_name) {\r
166 obj[func_name] = function() {\r
167 var fn_name = controlbox_fn;\r
168 return _kiwi.app.controlbox[fn_name].apply(_kiwi.app.controlbox, arguments);\r
169 };\r
170 });\r
171\r
172 return obj;\r
173 }\r
174 },\r
175\r
176 // Entry point to start the kiwi application\r
6b8fbed0 177 init: function (opts, callback) {\r
c3d988bc 178 var jobs, locale, localeLoaded, textThemeLoaded, text_theme;\r
161b5673 179 opts = opts || {};\r
b62c8381 180\r
1e9f97d6
D
181 this.initUtils();\r
182\r
28990b2a 183 jobs = new JobManager();\r
c3d988bc 184 jobs.onFinish(function(locale, s, xhr) {\r
161b5673 185 _kiwi.app = new _kiwi.model.Application(opts);\r
0fa2ca42 186\r
161b5673 187 // Start the client up\r
6b8fbed0 188 _kiwi.app.initializeInterfaces();\r
62c18d2d 189\r
2a89ba11
D
190 // Event emitter to let plugins interface with parts of kiwi\r
191 _kiwi.global.events = new PluginInterface();\r
192\r
f66c4823
D
193 // Now everything has started up, load the plugin manager for third party plugins\r
194 _kiwi.global.plugins = new _kiwi.model.PluginManager();\r
195\r
c3d988bc
D
196 callback();\r
197 });\r
28990b2a 198\r
c3d988bc
D
199 textThemeLoaded = function(text_theme, s, xhr) {\r
200 opts.text_theme = text_theme;\r
28990b2a
D
201\r
202 jobs.finishJob('load_text_theme');\r
203 };\r
0fa2ca42 204\r
c3d988bc
D
205 localeLoaded = function(locale, s, xhr) {\r
206 if (locale) {\r
207 _kiwi.global.i18n = new Jed(locale);\r
208 } else {\r
209 _kiwi.global.i18n = new Jed();\r
210 }\r
211\r
212 jobs.finishJob('load_locale');\r
213 };\r
214\r
f66c4823
D
215 // Set up the settings datastore\r
216 _kiwi.global.settings = _kiwi.model.DataStore.instance('kiwi.settings');\r
217 _kiwi.global.settings.load();\r
218\r
d26a40c6 219 // Set the window title\r
2df39a5c 220 window.document.title = opts.server_settings.client.window_title || 'Kiwi IRC';\r
d26a40c6 221\r
28990b2a 222 jobs.registerJob('load_locale');\r
19c790f1 223 locale = _kiwi.global.settings.get('locale');\r
0fa2ca42 224 if (!locale) {\r
c3d988bc 225 $.getJSON(opts.base_path + '/assets/locales/magic.json', localeLoaded);\r
0fa2ca42 226 } else {\r
c3d988bc 227 $.getJSON(opts.base_path + '/assets/locales/' + locale + '.json', localeLoaded);\r
0fa2ca42 228 }\r
d2cf53f5 229\r
28990b2a 230 jobs.registerJob('load_text_theme');\r
3a146f27
D
231 text_theme = opts.server_settings.client.settings.text_theme || 'default';\r
232 $.getJSON(opts.base_path + '/assets/text_themes/' + text_theme + '.json', textThemeLoaded);\r
6b8fbed0
D
233 },\r
234\r
235 start: function() {\r
236 _kiwi.app.showStartup();\r
237 },\r
238\r
239 // Allow plugins to change the startup applet\r
240 registerStartupApplet: function(startup_applet_name) {\r
241 _kiwi.app.startup_applet_name = startup_applet_name;\r
e537d683
D
242 },\r
243\r
244 /**\r
245 * Open a new IRC connection\r
246 * @param {Object} connection_details {nick, host, port, ssl, password, options}\r
247 * @param {Function} callback function(err, network){}\r
248 */\r
249 newIrcConnection: function(connection_details, callback) {\r
250 _kiwi.gateway.newConnection(connection_details, callback);\r
d0734219
D
251 },\r
252\r
253\r
254 /**\r
255 * Taking settings from the server and URL, extract the default server/channel/nick settings\r
256 */\r
257 defaultServerSettings: function () {\r
258 var parts;\r
259 var defaults = {\r
260 nick: '',\r
261 server: '',\r
262 port: 6667,\r
263 ssl: false,\r
264 channel: '',\r
265 channel_key: ''\r
266 };\r
267 var uricheck;\r
268\r
269\r
270 /**\r
271 * Get any settings set by the server\r
272 * These settings may be changed in the server selection dialog or via URL parameters\r
273 */\r
274 if (_kiwi.app.server_settings.client) {\r
275 if (_kiwi.app.server_settings.client.nick)\r
276 defaults.nick = _kiwi.app.server_settings.client.nick;\r
277\r
278 if (_kiwi.app.server_settings.client.server)\r
279 defaults.server = _kiwi.app.server_settings.client.server;\r
280\r
281 if (_kiwi.app.server_settings.client.port)\r
282 defaults.port = _kiwi.app.server_settings.client.port;\r
283\r
284 if (_kiwi.app.server_settings.client.ssl)\r
285 defaults.ssl = _kiwi.app.server_settings.client.ssl;\r
286\r
287 if (_kiwi.app.server_settings.client.channel)\r
288 defaults.channel = _kiwi.app.server_settings.client.channel;\r
289\r
290 if (_kiwi.app.server_settings.client.channel_key)\r
291 defaults.channel_key = _kiwi.app.server_settings.client.channel_key;\r
292 }\r
293\r
294\r
295\r
296 /**\r
297 * Get any settings passed in the URL\r
298 * These settings may be changed in the server selection dialog\r
299 */\r
300\r
301 // Any query parameters first\r
302 if (getQueryVariable('nick'))\r
303 defaults.nick = getQueryVariable('nick');\r
304\r
305 if (window.location.hash)\r
306 defaults.channel = window.location.hash;\r
307\r
308\r
309 // Process the URL part by part, extracting as we go\r
310 parts = window.location.pathname.toString().replace(_kiwi.app.get('base_path'), '').split('/');\r
311\r
312 if (parts.length > 0) {\r
313 parts.shift();\r
314\r
315 if (parts.length > 0 && parts[0]) {\r
316 // Check to see if we're dealing with an irc: uri, or whether we need to extract the server/channel info from the HTTP URL path.\r
317 uricheck = parts[0].substr(0, 7).toLowerCase();\r
318 if ((uricheck === 'ircs%3a') || (uricheck.substr(0,6) === 'irc%3a')) {\r
319 parts[0] = decodeURIComponent(parts[0]);\r
320 // irc[s]://<host>[:<port>]/[<channel>[?<password>]]\r
321 uricheck = /^irc(s)?:(?:\/\/?)?([^:\/]+)(?::([0-9]+))?(?:(?:\/)([^\?]*)(?:(?:\?)(.*))?)?$/.exec(parts[0]);\r
322 /*\r
323 uricheck[1] = ssl (optional)\r
324 uricheck[2] = host\r
325 uricheck[3] = port (optional)\r
326 uricheck[4] = channel (optional)\r
327 uricheck[5] = channel key (optional, channel must also be set)\r
328 */\r
329 if (uricheck) {\r
330 if (typeof uricheck[1] !== 'undefined') {\r
331 defaults.ssl = true;\r
332 if (defaults.port === 6667) {\r
333 defaults.port = 6697;\r
334 }\r
335 }\r
336 defaults.server = uricheck[2];\r
337 if (typeof uricheck[3] !== 'undefined') {\r
338 defaults.port = uricheck[3];\r
339 }\r
340 if (typeof uricheck[4] !== 'undefined') {\r
341 defaults.channel = '#' + uricheck[4];\r
342 if (typeof uricheck[5] !== 'undefined') {\r
343 defaults.channel_key = uricheck[5];\r
344 }\r
345 }\r
346 }\r
347 parts = [];\r
348 } else {\r
349 // Extract the port+ssl if we find one\r
350 if (parts[0].search(/:/) > 0) {\r
351 defaults.port = parts[0].substring(parts[0].search(/:/) + 1);\r
352 defaults.server = parts[0].substring(0, parts[0].search(/:/));\r
353 if (defaults.port[0] === '+') {\r
354 defaults.port = parseInt(defaults.port.substring(1), 10);\r
355 defaults.ssl = true;\r
356 } else {\r
357 defaults.ssl = false;\r
358 }\r
359\r
360 } else {\r
361 defaults.server = parts[0];\r
362 }\r
363\r
364 parts.shift();\r
365 }\r
366 }\r
367\r
368 if (parts.length > 0 && parts[0]) {\r
369 defaults.channel = '#' + parts[0];\r
370 parts.shift();\r
371 }\r
372 }\r
373\r
374 // If any settings have been given by the server.. override any auto detected settings\r
375 /**\r
376 * Get any server restrictions as set in the server config\r
377 * These settings can not be changed in the server selection dialog\r
378 */\r
379 if (_kiwi.app.server_settings && _kiwi.app.server_settings.connection) {\r
380 if (_kiwi.app.server_settings.connection.server) {\r
381 defaults.server = _kiwi.app.server_settings.connection.server;\r
382 }\r
383\r
384 if (_kiwi.app.server_settings.connection.port) {\r
385 defaults.port = _kiwi.app.server_settings.connection.port;\r
386 }\r
387\r
388 if (_kiwi.app.server_settings.connection.ssl) {\r
389 defaults.ssl = _kiwi.app.server_settings.connection.ssl;\r
390 }\r
391\r
392 if (_kiwi.app.server_settings.connection.channel) {\r
393 defaults.channel = _kiwi.app.server_settings.connection.channel;\r
394 }\r
395\r
396 if (_kiwi.app.server_settings.connection.channel_key) {\r
397 defaults.channel_key = _kiwi.app.server_settings.connection.channel_key;\r
398 }\r
399\r
400 if (_kiwi.app.server_settings.connection.nick) {\r
401 defaults.nick = _kiwi.app.server_settings.connection.nick;\r
402 }\r
403 }\r
404\r
405 // Set any random numbers if needed\r
406 defaults.nick = defaults.nick.replace('?', Math.floor(Math.random() * 100000).toString());\r
407\r
408 if (getQueryVariable('encoding'))\r
409 defaults.encoding = getQueryVariable('encoding');\r
410\r
411 return defaults;\r
412 },\r
b62c8381
D
413};\r
414\r
415\r
416\r
417// If within a closure, expose the kiwi globals\r
418if (typeof global !== 'undefined') {\r
161b5673 419 global.kiwi = _kiwi.global;\r
eaaf73b0 420} else {\r
161b5673
D
421 // Not within a closure so set a var in the current scope\r
422 var kiwi = _kiwi.global;\r
6636cc9e 423}\r