| Left: | ||
| Right: |
| OLD | NEW |
|---|---|
| (Empty) | |
| 1 (function() { | |
|
Felix Dahlke
2013/10/24 16:30:34
Opening braces should go on their own line, likewi
| |
| 2 /* Events */ | |
| 3 | |
| 4 var TabEventTarget = function() { | |
| 5 WrappedEventTarget.apply(this, arguments); | |
| 6 | |
| 7 this._tabs = {}; | |
| 8 | |
| 9 this._sharedListener = this._sharedListener.bind(this); | |
| 10 this._removeTab = this._removeTab.bind(this); | |
| 11 }; | |
| 12 TabEventTarget.prototype = { | |
| 13 __proto__: WrappedEventTarget.prototype, | |
| 14 _bindToTab: function(tab) { | |
| 15 return { | |
| 16 addListener: function(listener) { | |
| 17 var listeners = this._tabs[tab._id]; | |
| 18 | |
| 19 if (!listeners) { | |
| 20 this._tabs[tab._id] = listeners = []; | |
| 21 | |
| 22 if (Object.keys(this._tabs).length == 1) | |
| 23 this.addListener(this._sharedListener); | |
| 24 | |
| 25 tab.onRemoved.addListener(this._removeTab); | |
| 26 } | |
| 27 | |
| 28 listeners.push(listener); | |
| 29 }.bind(this), | |
| 30 removeListener: function(listener) { | |
| 31 var listeners = this._tabs[tab._id]; | |
| 32 if (!listeners) | |
| 33 return; | |
| 34 | |
| 35 var idx = listeners.indexOf(listener); | |
| 36 if (idx == -1) | |
| 37 return; | |
| 38 | |
| 39 listeners.splice(idx, 1); | |
| 40 | |
| 41 if (listeners.length == 0) | |
| 42 tab.onRemoved.removeListener(this._removeTab); | |
| 43 else { | |
| 44 if (listeners.length > 1) | |
| 45 return; | |
| 46 if (listeners[0] != this._removeTab) | |
| 47 return; | |
| 48 } | |
| 49 | |
| 50 this._removeTab(tab); | |
| 51 }.bind(this) | |
| 52 }; | |
| 53 }, | |
| 54 _sharedListener: function(tab) { | |
| 55 var listeners = this._tabs[tab._id]; | |
| 56 | |
| 57 if (!listeners) | |
| 58 return; | |
| 59 | |
| 60 // copy listeners before calling them, because of they | |
|
Felix Dahlke
2013/10/24 16:30:34
s/because of/because/
| |
| 61 // might add or remove other listeners, which must not be | |
| 62 // taken into account before the next occurrance of the event | |
|
Felix Dahlke
2013/10/24 16:30:34
s/occureace/occurence/
| |
| 63 listeners = listeners.slice(0); | |
| 64 | |
| 65 for (var i = 0; i < listeners.length; i++) | |
| 66 listeners[i](tab); | |
| 67 }, | |
| 68 _removeTab: function(tab) { | |
| 69 delete this._tabs[tab._id]; | |
| 70 | |
| 71 if (Object.keys(this._tabs).length == 0) | |
| 72 this.removeListener(this._sharedListener); | |
| 73 } | |
| 74 }; | |
| 75 | |
| 76 var BeforeNavigateTabEventTarget = function() { | |
| 77 TabEventTarget.call(this, chrome.tabs.onUpdated); | |
| 78 }; | |
| 79 BeforeNavigateTabEventTarget.prototype = { | |
| 80 __proto__: TabEventTarget.prototype, | |
| 81 _wrapListener: function(listener) { | |
| 82 return function(id, info, tab) { | |
| 83 if ("url" in info) | |
| 84 listener(new Tab(tab)); | |
| 85 }; | |
| 86 } | |
| 87 }; | |
| 88 | |
| 89 var CompletedTabEventTarget = function() { | |
| 90 TabEventTarget.call(this, chrome.tabs.onUpdated); | |
| 91 }; | |
| 92 CompletedTabEventTarget.prototype = { | |
| 93 __proto__: TabEventTarget.prototype, | |
| 94 _wrapListener: function(listener) { | |
| 95 return function(id, info, tab) { | |
| 96 if (info.status == "complete") | |
| 97 listener(new Tab(tab)); | |
| 98 }; | |
| 99 } | |
| 100 }; | |
| 101 | |
| 102 var ActivatedTabEventTarget = function() { | |
| 103 TabEventTarget.call(this, chrome.tabs.onActivated); | |
| 104 }; | |
| 105 ActivatedTabEventTarget.prototype = { | |
| 106 __proto__: TabEventTarget.prototype, | |
| 107 _wrapListener: function(listener) { | |
| 108 return function(info) { | |
| 109 chrome.tabs.get(info.tabId, function(tab) { | |
| 110 listener(new Tab(tab)); | |
| 111 }); | |
| 112 }; | |
| 113 } | |
| 114 } | |
| 115 | |
| 116 var RemovedTabEventTarget = function() { | |
| 117 TabEventTarget.call(this, chrome.tabs.onRemoved); | |
| 118 }; | |
| 119 RemovedTabEventTarget.prototype = { | |
| 120 __proto__: TabEventTarget.prototype, | |
| 121 _wrapListener: function(listener) { | |
| 122 return function(id) { | |
| 123 listener(new Tab({id: id})); | |
| 124 }; | |
| 125 } | |
| 126 }; | |
| 127 | |
| 128 var BeforeRequestEventTarget = function() { | |
| 129 WrappedEventTarget.call(this, chrome.webRequest.onBeforeRequest); | |
| 130 }; | |
| 131 BeforeRequestEventTarget.prototype = { | |
| 132 __proto__: WrappedEventTarget.prototype, | |
| 133 _wrapListener: function(listener) { | |
| 134 return function(details) { | |
| 135 var tab = null; | |
| 136 | |
| 137 if (details.tabId != -1) | |
| 138 tab = new Tab({id: details.tabId}); | |
| 139 | |
| 140 return {cancel: listener( | |
| 141 details.url, | |
| 142 details.type, | |
| 143 tab, | |
| 144 details.frameId, | |
| 145 details.parentFrameId | |
| 146 ) === false}; | |
| 147 }; | |
| 148 }, | |
| 149 _prepareExtraArguments: function(urls) { | |
| 150 return [urls ? {urls: urls} : {}, ["blocking"]]; | |
| 151 } | |
| 152 }; | |
| 153 | |
| 154 | |
| 155 /* Tabs */ | |
| 156 | |
| 157 var PageAction = function(tabId) { | |
| 158 this._tabId = tabId; | |
| 159 }; | |
| 160 PageAction.prototype = { | |
| 161 setIcon: function(path) { | |
| 162 chrome.pageAction.setIcon({tabId: this._tabId, path: path}); | |
| 163 }, | |
| 164 setTitle: function(title) { | |
| 165 chrome.pageAction.setTitle({tabId: this._tabId, title: title}); | |
| 166 }, | |
| 167 hide: function() { | |
| 168 chrome.pageAction.hide(this._tabId); | |
| 169 }, | |
| 170 show: function() { | |
| 171 chrome.pageAction.show(this._tabId); | |
| 172 } | |
| 173 }; | |
| 174 | |
| 175 Tab = function(tab) { | |
| 176 this._id = tab.id; | |
| 177 | |
| 178 this.url = tab.url; | |
| 179 this.pageAction = new PageAction(tab.id); | |
| 180 | |
| 181 this.onBeforeNavigate = ext.tabs.onBeforeNavigate._bindToTab(this); | |
| 182 this.onCompleted = ext.tabs.onCompleted._bindToTab(this); | |
| 183 this.onActivated = ext.tabs.onActivated._bindToTab(this); | |
| 184 this.onRemoved = ext.tabs.onRemoved._bindToTab(this); | |
| 185 }; | |
| 186 Tab.prototype = { | |
| 187 close: function() { | |
| 188 chrome.tabs.remove(this._id); | |
| 189 }, | |
| 190 activate: function() { | |
| 191 chrome.tabs.update(this._id, {selected: true}); | |
| 192 }, | |
| 193 sendMessage: function(message, responseCallback) { | |
| 194 chrome.tabs.sendMessage(this._id, message, responseCallback); | |
| 195 } | |
| 196 }; | |
| 197 | |
| 198 TabMap = function() { | |
| 199 this._map = {}; | |
| 200 this.delete = this.delete.bind(this); | |
| 201 }; | |
| 202 TabMap.prototype = { | |
| 203 get: function(tab) { | |
| 204 return (this._map[tab._id] || {}).value; | |
| 205 }, | |
| 206 set: function(tab, value) { | |
| 207 if (!(tab._id in this._map)) | |
| 208 tab.onRemoved.addListener(this.delete); | |
| 209 | |
| 210 this._map[tab._id] = {tab: tab, value: value}; | |
| 211 }, | |
| 212 has: function(tab) { | |
| 213 return tab._id in this._map; | |
| 214 }, | |
| 215 clear: function() { | |
| 216 for (var id in this._map) | |
| 217 this.delete(this._map[id].tab); | |
| 218 } | |
| 219 }; | |
| 220 TabMap.prototype["delete"] = function(tab) { | |
| 221 delete this._map[tab._id]; | |
| 222 tab.onRemoved.removeListener(this.delete); | |
| 223 }; | |
| 224 | |
| 225 | |
| 226 /* Windows */ | |
| 227 | |
| 228 Window = function(win) { | |
| 229 this._id = win.id; | |
| 230 this.visible = win.status != "minimized"; | |
| 231 }; | |
| 232 Window.prototype = { | |
| 233 getAllTabs: function(callback) { | |
| 234 chrome.tabs.query({windowId: this._id}, function(tabs) { | |
| 235 callback(tabs.map(function(tab) { | |
| 236 return new Tab(tab); | |
| 237 })); | |
| 238 }); | |
| 239 }, | |
| 240 getActiveTab: function(callback) { | |
| 241 chrome.tabs.query({windowId: this._id, active: true}, function(tabs) { | |
| 242 callback(new Tab(tabs[0])); | |
| 243 }); | |
| 244 }, | |
| 245 openTab: function(url, callback) { | |
| 246 var props = {windowId: this._id, url: url}; | |
| 247 | |
| 248 if (!callback) | |
| 249 chrome.tabs.create(props); | |
| 250 else | |
| 251 chrome.tabs.create(props, function(tab) { | |
| 252 callback(new Tab(tab)); | |
| 253 }); | |
| 254 } | |
| 255 }; | |
| 256 | |
| 257 | |
| 258 /* API */ | |
| 259 | |
| 260 ext.windows = { | |
| 261 getAll: function(callback) { | |
| 262 chrome.windows.getAll(function(windows) { | |
| 263 callback(windows.map(function(win) { | |
| 264 return new Window(win); | |
| 265 })); | |
| 266 }); | |
| 267 }, | |
| 268 getLastFocused: function(callback) { | |
| 269 chrome.windows.getLastFocused(function(win) { | |
| 270 callback(new Window(win)); | |
| 271 }); | |
| 272 } | |
| 273 }; | |
| 274 | |
| 275 ext.tabs = { | |
| 276 onBeforeNavigate: new BeforeNavigateTabEventTarget(), | |
| 277 onCompleted: new CompletedTabEventTarget(), | |
| 278 onActivated: new ActivatedTabEventTarget(), | |
| 279 onRemoved: new RemovedTabEventTarget() | |
| 280 }; | |
| 281 | |
| 282 ext.webRequest = { | |
| 283 onBeforeRequest: new BeforeRequestEventTarget(), | |
| 284 handlerBehaviorChanged: chrome.webRequest.handlerBehaviorChanged | |
| 285 }; | |
| 286 })(); | |
| OLD | NEW |