OLD | NEW |
1 /* | 1 /* |
2 * This file is part of Adblock Plus <http://adblockplus.org/>, | 2 * This file is part of Adblock Plus <http://adblockplus.org/>, |
3 * Copyright (C) 2006-2013 Eyeo GmbH | 3 * Copyright (C) 2006-2013 Eyeo GmbH |
4 * | 4 * |
5 * Adblock Plus is free software: you can redistribute it and/or modify | 5 * Adblock Plus is free software: you can redistribute it and/or modify |
6 * it under the terms of the GNU General Public License version 3 as | 6 * it under the terms of the GNU General Public License version 3 as |
7 * published by the Free Software Foundation. | 7 * published by the Free Software Foundation. |
8 * | 8 * |
9 * Adblock Plus is distributed in the hope that it will be useful, | 9 * Adblock Plus is distributed in the hope that it will be useful, |
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of | 10 * but WITHOUT ANY WARRANTY; without even the implied warranty of |
(...skipping 170 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
181 openTab: function(url, callback) | 181 openTab: function(url, callback) |
182 { | 182 { |
183 var tab = this._win.openTab(); | 183 var tab = this._win.openTab(); |
184 tab.url = url; | 184 tab.url = url; |
185 | 185 |
186 if (callback) | 186 if (callback) |
187 callback(new Tab(tab)); | 187 callback(new Tab(tab)); |
188 } | 188 } |
189 }; | 189 }; |
190 | 190 |
191 if (safari.extension.globalPage.contentWindow == window) | 191 |
| 192 /* Background page proxy */ |
| 193 |
| 194 var proxy = { |
| 195 tabs: [], |
| 196 objects: [], |
| 197 |
| 198 registerObject: function(obj, objects) |
| 199 { |
| 200 var objectId = objects.indexOf(obj); |
| 201 |
| 202 if (objectId == -1) |
| 203 objectId = objects.push(obj) - 1; |
| 204 |
| 205 return objectId; |
| 206 }, |
| 207 serializeSequence: function(sequence, objects, memo) |
| 208 { |
| 209 if (!memo) |
| 210 memo = {specs: [], arrays: []}; |
| 211 |
| 212 var items = []; |
| 213 for (var i = 0; i < sequence.length; i++) |
| 214 items.push(this.serialize(sequence[i], objects, memo)); |
| 215 |
| 216 return items; |
| 217 }, |
| 218 serialize: function(obj, objects, memo) |
| 219 { |
| 220 if (typeof obj == "object" && obj != null || typeof obj == "function") |
| 221 { |
| 222 if (obj.constructor == Array) |
| 223 { |
| 224 if (!memo) |
| 225 memo = {specs: [], arrays: []}; |
| 226 |
| 227 var idx = memo.arrays.indexOf(obj); |
| 228 if (idx != -1) |
| 229 return memo.specs[idx]; |
| 230 |
| 231 var spec = {type: "array"}; |
| 232 memo.specs.push(spec); |
| 233 memo.arrays.push(obj); |
| 234 |
| 235 spec.items = this.serializeSequence(obj, objects, memo); |
| 236 return spec; |
| 237 } |
| 238 |
| 239 if (obj.constructor != Date && obj.constructor != RegExp) |
| 240 return {type: "object", objectId: this.registerObject(obj, objects)}; |
| 241 } |
| 242 |
| 243 return {type: "value", value: obj}; |
| 244 }, |
| 245 createCallback: function(callbackId, tab) |
| 246 { |
| 247 var proxy = this; |
| 248 |
| 249 return function() |
| 250 { |
| 251 var idx = proxy.tabs.indexOf(tab); |
| 252 |
| 253 if (idx != -1) { |
| 254 var objects = proxy.objects[idx]; |
| 255 |
| 256 tab.page.dispatchMessage("proxyCallback", |
| 257 { |
| 258 callbackId: callbackId, |
| 259 contextId: proxy.registerObject(this, objects), |
| 260 args: proxy.serializeSequence(arguments, objects) |
| 261 }); |
| 262 } |
| 263 }; |
| 264 }, |
| 265 deserialize: function(spec, objects, tab, memo) |
| 266 { |
| 267 switch (spec.type) |
| 268 { |
| 269 case "value": |
| 270 return spec.value; |
| 271 case "hosted": |
| 272 return objects[spec.objectId]; |
| 273 case "callback": |
| 274 return this.createCallback(spec.callbackId, tab); |
| 275 case "object": |
| 276 case "array": |
| 277 if (!memo) |
| 278 memo = {specs: [], objects: []}; |
| 279 |
| 280 var idx = memo.specs.indexOf(spec); |
| 281 if (idx != -1) |
| 282 return memo.objects[idx]; |
| 283 |
| 284 var obj; |
| 285 if (spec.type == "array") |
| 286 obj = []; |
| 287 else |
| 288 obj = {}; |
| 289 |
| 290 memo.specs.push(spec); |
| 291 memo.objects.push(obj); |
| 292 |
| 293 if (spec.type == "array") |
| 294 for (var i = 0; i < spec.items.length; i++) |
| 295 obj.push(this.deserialize(spec.items[i], objects, tab, memo)); |
| 296 else |
| 297 for (var k in spec.properties) |
| 298 obj[k] = this.deserialize(spec.properties[k], objects, tab, memo); |
| 299 |
| 300 return obj; |
| 301 } |
| 302 }, |
| 303 createObjectCache: function(tab) |
| 304 { |
| 305 var objects = [window]; |
| 306 |
| 307 this.tabs.push(tab); |
| 308 this.objects.push(objects); |
| 309 |
| 310 tab.addEventListener("close", function() |
| 311 { |
| 312 var idx = this.tabs.indexOf(tab); |
| 313 |
| 314 if (idx != -1) |
| 315 { |
| 316 this.tabs.splice(idx, 1); |
| 317 this.objects.splice(idx, 1); |
| 318 } |
| 319 }.bind(this)); |
| 320 |
| 321 return objects; |
| 322 }, |
| 323 getObjectCache: function(tab) |
| 324 { |
| 325 var idx = this.tabs.indexOf(tab); |
| 326 var objects; |
| 327 |
| 328 if (idx != -1) |
| 329 objects = this.objects[idx]; |
| 330 else |
| 331 objects = this.objects[idx] = this.createObjectCache(tab); |
| 332 |
| 333 return objects; |
| 334 }, |
| 335 fail: function(error) |
| 336 { |
| 337 if (error instanceof Error) |
| 338 error = error.message; |
| 339 return {succeed: false, error: error}; |
| 340 }, |
| 341 _handleMessage: function(message, tab) |
| 342 { |
| 343 var objects = this.getObjectCache(tab); |
| 344 |
| 345 switch (message.type) |
| 346 { |
| 347 case "getProperty": |
| 348 var obj = objects[message.objectId]; |
| 349 |
| 350 try |
| 351 { |
| 352 var value = obj[message.property]; |
| 353 } |
| 354 catch (e) |
| 355 { |
| 356 return this.fail(e); |
| 357 } |
| 358 |
| 359 return {succeed: true, result: this.serialize(value, objects)}; |
| 360 case "setProperty": |
| 361 var obj = objects[message.objectId]; |
| 362 var value = this.deserialize(message.value, objects, tab); |
| 363 |
| 364 try |
| 365 { |
| 366 obj[message.property] = value; |
| 367 } |
| 368 catch (e) |
| 369 { |
| 370 return this.fail(e); |
| 371 } |
| 372 |
| 373 return {succeed: true}; |
| 374 case "callFunction": |
| 375 var func = objects[message.functionId]; |
| 376 var context = objects[message.contextId]; |
| 377 |
| 378 var args = []; |
| 379 for (var i = 0; i < message.args.length; i++) |
| 380 args.push(this.deserialize(message.args[i], objects, tab)); |
| 381 |
| 382 try |
| 383 { |
| 384 var result = func.apply(context, args); |
| 385 } |
| 386 catch (e) |
| 387 { |
| 388 return this.fail(e); |
| 389 } |
| 390 |
| 391 return {succeed: true, result: this.serialize(result, objects)}; |
| 392 case "inspectObject": |
| 393 var obj = objects[message.objectId]; |
| 394 var objectInfo = {properties: {}, isFunction: typeof obj == "function"
}; |
| 395 |
| 396 Object.getOwnPropertyNames(obj).forEach(function(prop) |
| 397 { |
| 398 objectInfo.properties[prop] = { |
| 399 enumerable: Object.prototype.propertyIsEnumerable.call(obj, prop) |
| 400 }; |
| 401 }); |
| 402 |
| 403 if (obj.__proto__) |
| 404 objectInfo.prototypeId = this.registerObject(obj.__proto__, objects)
; |
| 405 |
| 406 if (obj == Object.prototype) |
| 407 objectInfo.prototypeOf = "Object"; |
| 408 if (obj == Function.prototype) |
| 409 objectInfo.prototypeOf = "Function"; |
| 410 |
| 411 return objectInfo; |
| 412 } |
| 413 } |
| 414 }; |
| 415 |
| 416 |
| 417 /* Web request blocking */ |
| 418 |
| 419 ext.webRequest = { |
| 420 onBeforeRequest: { |
| 421 _listeners: [], |
| 422 _urlPatterns: [], |
| 423 |
| 424 _handleMessage: function(message, tab) |
| 425 { |
| 426 tab = new Tab(tab); |
| 427 |
| 428 for (var i = 0; i < this._listeners.length; i++) |
| 429 { |
| 430 var regex = this._urlPatterns[i]; |
| 431 |
| 432 if ((!regex || regex.test(message.url)) && this._listeners[i](message.
url, message.type, tab, 0, -1) === false) |
| 433 return false; |
| 434 } |
| 435 |
| 436 return true; |
| 437 }, |
| 438 addListener: function(listener, urls) |
| 439 { |
| 440 var regex; |
| 441 |
| 442 if (urls) |
| 443 regex = new RegExp("^(?:" + urls.map(function(url) |
| 444 { |
| 445 return url.split("*").map(function(s) |
| 446 { |
| 447 return s.replace(/([.?+^$[\]\\(){}|-])/g, "\\$1"); |
| 448 }).join(".*"); |
| 449 }).join("|") + ")($|[?#])"); |
| 450 |
| 451 this._listeners.push(listener); |
| 452 this._urlPatterns.push(regex); |
| 453 }, |
| 454 removeListener: function(listener) |
| 455 { |
| 456 var idx = this._listeners.indexOf(listener); |
| 457 |
| 458 if (idx != -1) |
| 459 { |
| 460 this._listeners.splice(idx, 1); |
| 461 this._urlPatterns.splice(idx, 1); |
| 462 } |
| 463 } |
| 464 }, |
| 465 handlerBehaviorChanged: function() {} |
| 466 }; |
| 467 |
| 468 |
| 469 /* Synchronous messaging */ |
| 470 |
| 471 safari.application.addEventListener("message", function(event) |
192 { | 472 { |
193 /* Background page proxy */ | 473 if (event.name == "canLoad") |
194 | 474 { |
195 var proxy = { | 475 var handler; |
196 tabs: [], | 476 |
197 objects: [], | 477 switch (event.message.type) |
198 | 478 { |
199 registerObject: function(obj, objects) | 479 case "proxy": |
200 { | 480 handler = proxy; |
201 var objectId = objects.indexOf(obj); | 481 break; |
202 | 482 case "webRequest": |
203 if (objectId == -1) | 483 handler = ext.webRequest.onBeforeRequest; |
204 objectId = objects.push(obj) - 1; | 484 break; |
205 | 485 } |
206 return objectId; | 486 |
207 }, | 487 event.message = handler._handleMessage(event.message.payload, event.target
); |
208 serializeSequence: function(sequence, objects, memo) | 488 } |
209 { | 489 }, true); |
210 if (!memo) | |
211 memo = {specs: [], arrays: []}; | |
212 | |
213 var items = []; | |
214 for (var i = 0; i < sequence.length; i++) | |
215 items.push(this.serialize(sequence[i], objects, memo)); | |
216 | |
217 return items; | |
218 }, | |
219 serialize: function(obj, objects, memo) | |
220 { | |
221 if (typeof obj == "object" && obj != null || typeof obj == "function") | |
222 { | |
223 if (obj.constructor == Array) | |
224 { | |
225 if (!memo) | |
226 memo = {specs: [], arrays: []}; | |
227 | |
228 var idx = memo.arrays.indexOf(obj); | |
229 if (idx != -1) | |
230 return memo.specs[idx]; | |
231 | |
232 var spec = {type: "array"}; | |
233 memo.specs.push(spec); | |
234 memo.arrays.push(obj); | |
235 | |
236 spec.items = this.serializeSequence(obj, objects, memo); | |
237 return spec; | |
238 } | |
239 | |
240 if (obj.constructor != Date && obj.constructor != RegExp) | |
241 return {type: "object", objectId: this.registerObject(obj, objects)}
; | |
242 } | |
243 | |
244 return {type: "value", value: obj}; | |
245 }, | |
246 createCallback: function(callbackId, tab) | |
247 { | |
248 var proxy = this; | |
249 | |
250 return function() | |
251 { | |
252 var idx = proxy.tabs.indexOf(tab); | |
253 | |
254 if (idx != -1) { | |
255 var objects = proxy.objects[idx]; | |
256 | |
257 tab.page.dispatchMessage("proxyCallback", | |
258 { | |
259 callbackId: callbackId, | |
260 contextId: proxy.registerObject(this, objects), | |
261 args: proxy.serializeSequence(arguments, objects) | |
262 }); | |
263 } | |
264 }; | |
265 }, | |
266 deserialize: function(spec, objects, tab, memo) | |
267 { | |
268 switch (spec.type) | |
269 { | |
270 case "value": | |
271 return spec.value; | |
272 case "hosted": | |
273 return objects[spec.objectId]; | |
274 case "callback": | |
275 return this.createCallback(spec.callbackId, tab); | |
276 case "object": | |
277 case "array": | |
278 if (!memo) | |
279 memo = {specs: [], objects: []}; | |
280 | |
281 var idx = memo.specs.indexOf(spec); | |
282 if (idx != -1) | |
283 return memo.objects[idx]; | |
284 | |
285 var obj; | |
286 if (spec.type == "array") | |
287 obj = []; | |
288 else | |
289 obj = {}; | |
290 | |
291 memo.specs.push(spec); | |
292 memo.objects.push(obj); | |
293 | |
294 if (spec.type == "array") | |
295 for (var i = 0; i < spec.items.length; i++) | |
296 obj.push(this.deserialize(spec.items[i], objects, tab, memo)); | |
297 else | |
298 for (var k in spec.properties) | |
299 obj[k] = this.deserialize(spec.properties[k], objects, tab, memo
); | |
300 | |
301 return obj; | |
302 } | |
303 }, | |
304 createObjectCache: function(tab) | |
305 { | |
306 var objects = [window]; | |
307 | |
308 this.tabs.push(tab); | |
309 this.objects.push(objects); | |
310 | |
311 tab.addEventListener("close", function() | |
312 { | |
313 var idx = this.tabs.indexOf(tab); | |
314 | |
315 if (idx != -1) | |
316 { | |
317 this.tabs.splice(idx, 1); | |
318 this.objects.splice(idx, 1); | |
319 } | |
320 }.bind(this)); | |
321 | |
322 return objects; | |
323 }, | |
324 getObjectCache: function(tab) | |
325 { | |
326 var idx = this.tabs.indexOf(tab); | |
327 var objects; | |
328 | |
329 if (idx != -1) | |
330 objects = this.objects[idx]; | |
331 else | |
332 objects = this.objects[idx] = this.createObjectCache(tab); | |
333 | |
334 return objects; | |
335 }, | |
336 fail: function(error) | |
337 { | |
338 if (error instanceof Error) | |
339 error = error.message; | |
340 return {succeed: false, error: error}; | |
341 }, | |
342 _handleMessage: function(message, tab) | |
343 { | |
344 var objects = this.getObjectCache(tab); | |
345 | |
346 switch (message.type) | |
347 { | |
348 case "getProperty": | |
349 var obj = objects[message.objectId]; | |
350 | |
351 try | |
352 { | |
353 var value = obj[message.property]; | |
354 } | |
355 catch (e) | |
356 { | |
357 return this.fail(e); | |
358 } | |
359 | |
360 return {succeed: true, result: this.serialize(value, objects)}; | |
361 case "setProperty": | |
362 var obj = objects[message.objectId]; | |
363 var value = this.deserialize(message.value, objects, tab); | |
364 | |
365 try | |
366 { | |
367 obj[message.property] = value; | |
368 } | |
369 catch (e) | |
370 { | |
371 return this.fail(e); | |
372 } | |
373 | |
374 return {succeed: true}; | |
375 case "callFunction": | |
376 var func = objects[message.functionId]; | |
377 var context = objects[message.contextId]; | |
378 | |
379 var args = []; | |
380 for (var i = 0; i < message.args.length; i++) | |
381 args.push(this.deserialize(message.args[i], objects, tab)); | |
382 | |
383 try | |
384 { | |
385 var result = func.apply(context, args); | |
386 } | |
387 catch (e) | |
388 { | |
389 return this.fail(e); | |
390 } | |
391 | |
392 return {succeed: true, result: this.serialize(result, objects)}; | |
393 case "inspectObject": | |
394 var obj = objects[message.objectId]; | |
395 var objectInfo = {properties: {}, isFunction: typeof obj == "functio
n"}; | |
396 | |
397 Object.getOwnPropertyNames(obj).forEach(function(prop) | |
398 { | |
399 objectInfo.properties[prop] = { | |
400 enumerable: Object.prototype.propertyIsEnumerable.call(obj, prop
) | |
401 }; | |
402 }); | |
403 | |
404 if (obj.__proto__) | |
405 objectInfo.prototypeId = this.registerObject(obj.__proto__, object
s); | |
406 | |
407 if (obj == Object.prototype) | |
408 objectInfo.prototypeOf = "Object"; | |
409 if (obj == Function.prototype) | |
410 objectInfo.prototypeOf = "Function"; | |
411 | |
412 return objectInfo; | |
413 } | |
414 } | |
415 }; | |
416 | |
417 | |
418 /* Web request blocking */ | |
419 | |
420 ext.webRequest = { | |
421 onBeforeRequest: { | |
422 _listeners: [], | |
423 _urlPatterns: [], | |
424 | |
425 _handleMessage: function(message, tab) | |
426 { | |
427 tab = new Tab(tab); | |
428 | |
429 for (var i = 0; i < this._listeners.length; i++) | |
430 { | |
431 var regex = this._urlPatterns[i]; | |
432 | |
433 if ((!regex || regex.test(message.url)) && this._listeners[i](messag
e.url, message.type, tab, 0, -1) === false) | |
434 return false; | |
435 } | |
436 | |
437 return true; | |
438 }, | |
439 addListener: function(listener, urls) | |
440 { | |
441 var regex; | |
442 | |
443 if (urls) | |
444 regex = new RegExp("^(?:" + urls.map(function(url) | |
445 { | |
446 return url.split("*").map(function(s) | |
447 { | |
448 return s.replace(/([.?+^$[\]\\(){}|-])/g, "\\$1"); | |
449 }).join(".*"); | |
450 }).join("|") + ")($|[?#])"); | |
451 | |
452 this._listeners.push(listener); | |
453 this._urlPatterns.push(regex); | |
454 }, | |
455 removeListener: function(listener) | |
456 { | |
457 var idx = this._listeners.indexOf(listener); | |
458 | |
459 if (idx != -1) | |
460 { | |
461 this._listeners.splice(idx, 1); | |
462 this._urlPatterns.splice(idx, 1); | |
463 } | |
464 } | |
465 }, | |
466 handlerBehaviorChanged: function() {} | |
467 }; | |
468 | |
469 | |
470 /* Synchronous messaging */ | |
471 | |
472 safari.application.addEventListener("message", function(event) | |
473 { | |
474 if (event.name == "canLoad") | |
475 { | |
476 var handler; | |
477 | |
478 switch (event.message.type) | |
479 { | |
480 case "proxy": | |
481 handler = proxy; | |
482 break; | |
483 case "webRequest": | |
484 handler = ext.webRequest.onBeforeRequest; | |
485 break; | |
486 } | |
487 | |
488 event.message = handler._handleMessage(event.message.payload, event.targ
et); | |
489 } | |
490 }, true); | |
491 } | |
492 | 490 |
493 | 491 |
494 /* API */ | 492 /* API */ |
495 | 493 |
496 ext.windows = { | 494 ext.windows = { |
497 getAll: function(callback) | 495 getAll: function(callback) |
498 { | 496 { |
499 callback(safari.application.browserWindows.map(function(win) | 497 callback(safari.application.browserWindows.map(function(win) |
500 { | 498 { |
501 return new Window(win); | 499 return new Window(win); |
(...skipping 19 matching lines...) Expand all Loading... |
521 } | 519 } |
522 }; | 520 }; |
523 | 521 |
524 ext.onMessage = new MessageEventTarget(safari.application); | 522 ext.onMessage = new MessageEventTarget(safari.application); |
525 | 523 |
526 // TODO: Implement context menu | 524 // TODO: Implement context menu |
527 ext.contextMenus = { | 525 ext.contextMenus = { |
528 create: function(title, contexts, onclick) {}, | 526 create: function(title, contexts, onclick) {}, |
529 removeAll: function(callback) {} | 527 removeAll: function(callback) {} |
530 }; | 528 }; |
531 | |
532 // Safari will load the bubble once, and then show it everytime the icon is | |
533 // clicked. While Chrome loads it everytime you click the icon. So in order to | |
534 // force the same behavior in Safari, we are going to reload the page of the | |
535 // bubble everytime it is shown. | |
536 if (safari.extension.globalPage.contentWindow != window) | |
537 safari.application.addEventListener("popover", function() | |
538 { | |
539 document.documentElement.style.display = "none"; | |
540 document.location.reload(); | |
541 }, true); | |
542 })(); | 529 })(); |
OLD | NEW |