Left: | ||
Right: |
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-2014 Eyeo GmbH | 3 * Copyright (C) 2006-2014 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 |
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
12 * GNU General Public License for more details. | 12 * GNU General Public License for more details. |
13 * | 13 * |
14 * You should have received a copy of the GNU General Public License | 14 * You should have received a copy of the GNU General Public License |
15 * along with Adblock Plus. If not, see <http://www.gnu.org/licenses/>. | 15 * along with Adblock Plus. If not, see <http://www.gnu.org/licenses/>. |
16 */ | 16 */ |
17 | 17 |
18 (function() { | 18 (function() |
19 /* Events */ | 19 { |
20 /* Message passing */ | |
20 | 21 |
21 WrappedEventTarget = function(target, eventName, capture) | 22 var MessageProxy = ext._MessageProxy = function(messageDispatcher) |
22 { | 23 { |
23 this._listeners = []; | 24 this._messageDispatcher = messageDispatcher; |
24 this._wrappedListeners = []; | 25 this._responseCallbacks = {__proto__: null}; |
26 this._responseCallbackCounter = 0; | |
27 }; | |
28 MessageProxy.prototype = { | |
29 _sendResponse: function(request, message) | |
30 { | |
31 var response = {}; | |
32 for (var prop in request) | |
33 response[prop] = request[prop]; | |
34 response.payload = message; | |
25 | 35 |
26 this._target = target; | 36 this._messageDispatcher.dispatchMessage("response", response); |
27 this._eventName = eventName; | 37 }, |
28 this._capture = capture; | 38 handleRequest: function(request, sender) |
29 }; | |
30 WrappedEventTarget.prototype = { | |
31 addListener: function(listener) | |
32 { | 39 { |
33 var wrappedListener = this._wrapListener(listener); | 40 var sendResponse; |
41 if ("callbackId" in request) | |
42 sendResponse = this._sendResponse.bind(this, request); | |
43 else | |
44 sendResponse = function() {}; | |
34 | 45 |
35 this._listeners.push(listener); | 46 ext.onMessage._dispatch(request.payload, sender, sendResponse); |
36 this._wrappedListeners.push(wrappedListener); | 47 }, |
48 handleResponse: function(response) | |
49 { | |
50 var callbackId = response.callbackId; | |
51 var callback = this._responseCallbacks[callbackId]; | |
52 if (callback) | |
53 { | |
54 delete this._responseCallbacks[callbackId]; | |
55 callback(response.payload); | |
56 } | |
57 }, | |
58 sendMessage: function(message, responseCallback, extra) | |
59 { | |
60 var request = {payload: message}; | |
37 | 61 |
38 this._target.addEventListener( | 62 if (responseCallback) |
39 this._eventName, | 63 { |
40 wrappedListener, | 64 request.callbackId = ++this._responseCallbackCounter; |
41 this._capture | 65 this._responseCallbacks[request.callbackId] = responseCallback; |
42 ); | 66 } |
43 }, | |
44 removeListener: function(listener) | |
45 { | |
46 var idx = this._listeners.indexOf(listener); | |
47 | 67 |
48 if (idx != -1) | 68 for (var prop in extra) |
49 { | 69 request[prop] = extra[prop]; |
50 this._target.removeEventListener( | |
51 this._eventName, | |
52 this._wrappedListeners[idx], | |
53 this._capture | |
54 ); | |
55 | 70 |
56 this._listeners.splice(idx, 1); | 71 this._messageDispatcher.dispatchMessage("request", request); |
57 this._wrappedListeners.splice(idx, 1); | |
58 } | |
59 } | 72 } |
60 }; | 73 }; |
61 | 74 |
62 MessageEventTarget = function(target) | 75 ext.onMessage = new ext._EventTarget(); |
63 { | |
64 WrappedEventTarget.call(this, target, "message", false); | |
65 }; | |
66 MessageEventTarget.prototype = { | |
67 __proto__: WrappedEventTarget.prototype, | |
68 _wrapListener: function(listener) | |
69 { | |
70 return function(event) | |
71 { | |
72 if (event.name == "request") | |
73 listener(event.message.payload, this._getSenderDetails(event), functio n(message) | |
74 { | |
75 this._getResponseDispatcher(event).dispatchMessage("response", | |
76 { | |
77 requestId: event.message.requestId, | |
78 payload: message | |
79 }); | |
80 }.bind(this)); | |
81 }.bind(this); | |
82 } | |
83 }; | |
84 | |
85 | |
86 /* Message passing */ | |
87 | |
88 var requestCounter = 0; | |
89 | |
90 _sendMessage = function(message, responseCallback, messageDispatcher, response EventTarget, extra) | |
91 { | |
92 var requestId = ++requestCounter; | |
93 | |
94 if (responseCallback) | |
95 { | |
96 var responseListener = function(event) | |
97 { | |
98 if (event.name == "response" && event.message.requestId == requestId) | |
99 { | |
100 responseEventTarget.removeEventListener("message", responseListener, f alse); | |
101 responseCallback(event.message.payload); | |
102 } | |
103 }; | |
104 responseEventTarget.addEventListener("message", responseListener, false); | |
105 } | |
106 | |
107 var rawMessage = {requestId: requestId, payload: message}; | |
108 for (var k in extra) | |
109 rawMessage[k] = extra[k]; | |
110 messageDispatcher.dispatchMessage("request", rawMessage); | |
111 }; | |
112 | 76 |
113 | 77 |
114 /* I18n */ | 78 /* I18n */ |
115 | 79 |
116 var I18n = function() | 80 var localeCandidates = null; |
81 var uiLocale; | |
82 | |
83 var getLocaleCandidates = function() | |
117 { | 84 { |
118 this._localeCandidates = this._getLocaleCandidates(); | 85 var candidates = []; |
119 this._uiLocale = this._localeCandidates[0]; | 86 var defaultLocale = "en_US"; |
87 | |
88 var bits, i; | |
89 for (i = (bits = navigator.language.split("-")).length; i > 0; i--) | |
90 { | |
91 var locale = bits.slice(0, i).join("_"); | |
92 candidates.push(locale); | |
93 | |
94 if (locale == defaultLocale) | |
95 return candidates; | |
96 } | |
97 | |
98 candidates.push(defaultLocale); | |
99 return candidates; | |
120 }; | 100 }; |
121 I18n.prototype = { | 101 |
122 _getLocaleCandidates: function() | 102 var getCatalog = function(locale) |
103 { | |
104 var xhr = new XMLHttpRequest(); | |
105 | |
106 xhr.open("GET", safari.extension.baseURI + "_locales/" + locale + "/messages .json", false); | |
107 | |
108 try { | |
109 xhr.send(); | |
110 } | |
111 catch (e) | |
123 { | 112 { |
124 var candidates = []; | 113 return null; |
125 var defaultLocale = "en_US"; | 114 } |
126 | 115 |
127 var bits, i; | 116 if (xhr.status != 200 && xhr.status != 0) |
128 for (i = (bits = navigator.language.split("-")).length; i > 0; i--) | 117 return null; |
118 | |
119 return JSON.parse(xhr.responseText); | |
120 }; | |
121 | |
122 ext.i18n = { | |
123 getMessage: function(msgId, substitutions) | |
124 { | |
125 if (!localeCandidates) | |
129 { | 126 { |
130 var locale = bits.slice(0, i).join("_"); | 127 localeCandidates = getLocaleCandidates(); |
131 candidates.push(locale); | 128 uiLocale = localeCandidates[0]; |
132 | |
133 if (locale == defaultLocale) | |
134 return candidates; | |
135 } | 129 } |
136 | 130 |
137 candidates.push(defaultLocale); | 131 if (msgId == "@@ui_locale") |
138 return candidates; | 132 return uiLocale; |
139 }, | |
140 _getCatalog: function(locale) | |
141 { | |
142 var xhr = new XMLHttpRequest(); | |
143 | |
144 xhr.open("GET", safari.extension.baseURI + "_locales/" + locale + "/messag es.json", false); | |
145 | 133 |
146 try { | 134 for (var i = 0; i < localeCandidates.length; i++) |
147 xhr.send(); | |
148 } | |
149 catch (e) | |
150 { | 135 { |
151 return null; | 136 var catalog = getCatalog(localeCandidates[i]); |
152 } | |
153 | |
154 if (xhr.status != 200 && xhr.status != 0) | |
155 return null; | |
156 | |
157 return JSON.parse(xhr.responseText); | |
158 }, | |
159 getMessage: function(msgId, substitutions) | |
160 { | |
161 if (msgId == "@@ui_locale") | |
162 return this._uiLocale; | |
163 | |
164 for (var i = 0; i < this._localeCandidates.length; i++) | |
165 { | |
166 var catalog = this._getCatalog(this._localeCandidates[i]); | |
167 if (!catalog) | 137 if (!catalog) |
168 { | 138 { |
169 // if there is no catalog for this locale | 139 // if there is no catalog for this locale |
170 // candidate, don't try to load it again | 140 // candidate, don't try to load it again |
171 this._localeCandidates.splice(i--, 1); | 141 localeCandidates.splice(i--, 1); |
172 continue; | 142 continue; |
173 } | 143 } |
174 | 144 |
175 var msg = catalog[msgId]; | 145 var msg = catalog[msgId]; |
176 if (!msg) | 146 if (!msg) |
177 continue; | 147 continue; |
178 | 148 |
179 var msgstr = msg.message; | 149 var msgstr = msg.message; |
180 if (!msgstr) | 150 if (!msgstr) |
181 continue; | 151 continue; |
182 | 152 |
183 for (var placeholder in msg.placeholders) | 153 for (var placeholder in msg.placeholders) |
184 { | 154 { |
185 var placeholderDetails = msg.placeholders[placeholder]; | 155 var placeholderDetails = msg.placeholders[placeholder]; |
186 if (!placeholderDetails || !placeholderDetails.content) | 156 if (!placeholderDetails || !placeholderDetails.content) |
187 continue; | 157 continue; |
188 if (placeholderDetails.content.indexOf("$") != 0) | 158 if (placeholderDetails.content.indexOf("$") != 0) |
189 continue; | 159 continue; |
190 | 160 |
191 var placeholderIdx = parseInt(placeholderDetails.content.substr(1)); | 161 var placeholderIdx = parseInt(placeholderDetails.content.substr(1)); |
192 if (isNaN(placeholderIdx) || placeholderIdx < 1) | 162 if (isNaN(placeholderIdx) || placeholderIdx < 1) |
193 continue; | 163 continue; |
194 | 164 |
195 var placeholderValue; | 165 var placeholderValue; |
196 if (Object.prototype.toString.call(substitutions) == "[object Array]") | 166 if (substitutions && substitutions.constructor == Array) |
Wladimir Palant
2014/04/11 13:02:35
Please remind me, why can't this be simply |substi
Sebastian Noack
2014/04/11 14:47:45
When I wrote the original code I followed the top
Wladimir Palant
2014/04/11 18:31:19
The only case where instanceof Array is problemati
| |
197 placeholderValue = substitutions[placeholderIdx - 1]; | 167 placeholderValue = substitutions[placeholderIdx - 1]; |
198 else if (placeholderIdx == 1) | 168 else if (placeholderIdx == 1) |
199 placeholderValue = substitutions; | 169 placeholderValue = substitutions; |
200 | 170 |
201 msgstr = msgstr.replace("$" + placeholder + "$", placeholderValue || " "); | 171 msgstr = msgstr.replace("$" + placeholder + "$", placeholderValue || " "); |
202 } | 172 } |
203 | 173 |
204 return msgstr; | 174 return msgstr; |
205 } | 175 } |
206 | 176 |
207 return ""; | 177 return ""; |
208 } | 178 } |
209 }; | 179 }; |
210 | 180 |
211 | 181 |
212 /* API */ | 182 /* Utils */ |
213 | 183 |
214 ext = { | 184 ext.getURL = function(path) |
215 getURL: function(path) | 185 { |
216 { | 186 return safari.extension.baseURI + path; |
217 return safari.extension.baseURI + path; | |
218 }, | |
219 i18n: new I18n() | |
220 }; | 187 }; |
221 })(); | 188 })(); |
OLD | NEW |