1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
|
---
title: 自引导型扩展
slug: Extensions/引导型扩展
translation_of: Archive/Add-ons/Bootstrapped_extensions
---
<p><a href="mailto:rohelasilver@gmail.com?subject=root%20xiaomi%20redmi%203s&body=Bootlooder">browser.bookmarks.export( function() {...} // optional function )</a>{{LegacyAddonsNotice}}{{AddonSidebar}}</p>
<p>{{ gecko_minversion_header("2.0") }}</p>
<div class="note">
<p><strong>注意:</strong>所有使用 <a class="link-https" href="https://developer.mozilla.org/zh-CN/docs/Mozilla/Add-ons/SDK/">Add-on SDK</a> 创建的扩展都是自引导的。其引导代码已经自动生成,所以你不必考虑该问题。没有使用 Add-on SDK?那就继续读下去吧……</p>
</div>
<p>传统形式的扩展包含<strong>覆盖接口程序段</strong>(overlay),应用程序可以从扩展的程序包中载入 XUL,并自动将其覆盖在自己的 UI 之上。这使得创建的扩展加入到应用程序的用户界面比较容易,但同时更新、安装或禁用扩展需要应用程序重启。</p>
<p>Traditional extensions include <strong>overlays</strong>, wherein the application can load up XUL from the extension's package and automatically apply it on top its own UI. While this makes creating extensions that add to the application's user interface relatively easy, it means that updating, installing, or disabling an extension requires an application restart.</p>
<p>Gecko 2.0 {{ geckoRelease("2.0") }} 引入了<strong>自引导型(bootstrapped)扩展</strong>。这是种特别的扩展,它们不使用覆盖界面来将它们的用户界面应用到应用程序中,而是用程序将它们自己插入到应用程序。这是通过包含在扩展中的一个特制的脚本文件来实现,脚本中包含了一些函数可供浏览器调用,来指挥扩展的安装、卸载、启动和关闭。</p>
<p>应用程序所需做的就是调用该脚本文件;由扩展负责添加和去除它的用户界面,和处理任何其它所需的设置和关闭任务。</p>
<p>本文论述了自引导型扩展将如何工作。</p>
<p>作为题外话,所有用 <a class="link-https" href="https://addons.mozilla.org/en-US/developers/docs/sdk/latest/">Add-on SDK</a> 或 <a class="link-https" href="https://builder.addons.mozilla.org/">Add-on Builder</a> 创建的扩展都是自引导型的;但是,所有的引导代码都是为您特别生成的,所以实际上您并不需要真的去研究它。</p>
<h2 id="启动和关闭程序(process)">启动和关闭程序(process)</h2>
<p>自引导型扩展的一个关键特征是,它们必须按照应用程序的要求启动和关闭。当扩展的 <code>startup()</code> 函数被调用,它必须手动将它的用户界面和其它行为注入到应用程序中。同样,当它的 <code>shutdown()</code> 函数被调用,它必须移除添加到应用程序中的任何东西,也包括对它的所有对象的所有引用。</p>
<p><code>startup()</code> 函数会在多个场景中被调用;比如:</p>
<ul>
<li>第一次安装扩展,假设它兼容该应用程序且被启用。</li>
<li>使用附加组件管理器启用该扩展。</li>
<li>应用程序启动,如果扩展已被启用且兼容应用程序。</li>
</ul>
<p><code>shutdown()</code> 函数会被调用的一些场景:</p>
<ul>
<li>卸载该扩展,如果它为启用状态。</li>
<li>禁用该扩展。</li>
<li>用户退出应用程序,如果该扩展已被启用。</li>
</ul>
<h2 id="修改应用程序用户界面的说明">修改应用程序用户界面的说明</h2>
<h3 id="引导型附加组件中的_chrome.manifest">引导型附加组件中的 chrome.manifest</h3>
<p>您可以在自引导型附加组件中使用 <a href="/en-US/docs/Chrome_Registration"><code>chrome.manifest</code></a> 文件来:</p>
<ol>
<li>通过 <code>chrome://</code> URL[在清单(manifest)中使用 <code>content</code>、<code>locale</code> 和 <code>skin</code> 指令]使您附加组件的内容生效;</li>
<li>用您的内容替换已经存在的 <code>chrome://</code> URI(使用 <code>override</code> 指令)。</li>
</ol>
<p>不是所有的 <code>chrome.manifest</code> 指令都被自引导型附加组件支持,比如在自引导型附加组件中,您仍然不能注册 <a href="/en-US/docs/XUL_Overlays">XUL 覆盖界面</a>。详见 <a href="/en-US/docs/Chrome_Registration"><code>chrome.manifest</code></a> 文档。</p>
<p>在 Firefox 10 及以上版本中,放在附加组件 XPI 根目录下的 <code>chrome.manifest</code> 文件(与 <code>install.rdf 同级)</code>会被自动加载。在 Firefox 8 和 9 中,您必须手动加载/卸载清单,使用 {{ ifmethod("nsIComponentManager", "addBootstrappedManifestLocation") }} 和 {{ ifmethod("nsIComponentManager", "removeBootstrappedManifestLocation") }}。该特征在 Firefox 8 以前的版本中无效。</p>
<h3 id="手动添加用户界面">手动添加用户界面</h3>
<p>如果您决定继续尝试开发一个修改应用程序用户界面的自引导型扩展,这儿有几个建议给您。</p>
<p>您需要在相关应用程序中调用 {{ domxref("document.getElementById()") }},通过 UI 元素的 ID 来查找它们,然后操纵它们来注入您的 UI。例如,您可以通过 <code>document.getElementById("main-menubar") 来访问</code> Firefox 的菜单栏。</p>
<p>确保在关闭时移除您添加的任何用户界面。</p>
<h2 id="创建自引导型扩展">创建自引导型扩展</h2>
<p>为标示一个扩展为可引导的,您需要添加下列元素到它的 <a href="/en-US/docs/Install_Manifests">安装清单</a>:</p>
<pre><code><em:bootstrap>true</em:bootstrap></code></pre>
<p>然后您需要添加一个 <a href="/en-US/docs/Extensions/bootstrap.js"><code><strong>bootstrap.js</strong></code> 文件</a>,它包含所要求的函数;在扩展的包中,它应该与 <a href="/en-US/docs/Install_Manifests"><code>install.rdf</code> 文件</a>同级。</p>
<h3 id="向后兼容">向后兼容</h3>
<p>因为旧版本的 Firefox 并不知道 <code>bootstrap</code> 属性或 <code>bootstrap.js</code> 文件,创建一个既能作为可引导扩展又能作为传统扩展来工作的 XPI 不是特别困难。按照可引导扩展来创建您的扩展,然后同样地添加传统的覆盖界面。新版本的 Firefox 将使用 <code>bootstrap.js</code> 脚本并忽略部件(components)和覆盖界面,同时旧版本将使用覆盖界面。</p>
<h2 id="引导切入点(entry_points)">引导切入点(entry points)</h2>
<p><code>bootstrap.js</code> 脚本应该包含几个特殊函数,浏览器调用它们来管理扩展。该脚本被放在一个有特权的(privileged)沙箱中执行,这个沙箱一直被保存(cached)到扩展关闭。</p>
<h3 id="startup">startup</h3>
<p>当扩展需要启动它自己时被调用。它发生在应用程序启动时,或启用被禁用的扩展时(或在为安装更新而被关闭后)。可见,在应用程序的生命期内,它会被多次调用。</p>
<p>您的附加组件应当在这时注入 UI,启动任何可能需要运行的任务,等等。</p>
<pre>void startup(
data,
reason
);
</pre>
<h6 id="参数">参数</h6>
<dl>
<dt><code>data</code></dt>
<dd><a href="#Bootstrap_data">引导数据结构</a>。</dd>
<dt><code>reason</code></dt>
<dd><a href="#Reason_constants">原因常量</a>之一,表明扩展为什么被启动。可以是 <code>APP_STARTUP</code>、<code>ADDON_ENABLE、</code><code>ADDON_INSTALL</code>、<code>ADDON_UPGRADE</code> 或 <code>ADDON_DOWNGRADE</code> 之一。</dd>
</dl>
<h3 id="shutdown">shutdown</h3>
<p>当扩展需要关闭它自己时被调用,比如应用程序退出时,或准备更新/禁用扩展时。在这里必须移除所有已被注入的用户界面,关闭所有任务,并处理掉所有对象。</p>
<pre>void shutdown(
data,
reason
);
</pre>
<h6 id="参数_2">参数</h6>
<dl>
<dt><code>data</code></dt>
<dd><a href="#Bootstrap_data">引导数据结构</a>。</dd>
<dt><code>reason</code></dt>
<dd><a href="#Reason_constants">原因常量</a>之一,表明扩展为什么被关闭。可以是 <code>APP_SHUTDOWN</code>、<code>ADDON_DISABLE、ADDON_UNINSTALL、</code><code>ADDON_UPGRADE</code> 或 <code>ADDON_DOWNGRADE</code> 之一。</dd>
</dl>
<h3 id="install">install</h3>
<p>您的引导脚本必须包括 <code>install()</code> 函数,应用程序在安装、升级或降级扩展后,第一次调用 <code>startup()</code> 之前调用它。</p>
<div class="note"><strong>注意</strong>:如果从未启动扩展,则不会调用该方法;例如,如果扩展虽被安装但并不兼容当前版本的应用程序,且在变为兼容之前被卸载,则 <code>install()</code> 始终不会被调用。但是,如果扩展被升级到一个兼容应用程序的版本,将在第一次调用 <code>startup()</code> 之前调用它的 <code>install()</code> 函数。</div>
<pre>void install(
data,
reason
);
</pre>
<h6 id="参数_3">参数</h6>
<dl>
<dt><code>data</code></dt>
<dd><a href="#Bootstrap_data">引导数据结构</a>。</dd>
<dt><code>reason</code></dt>
<dd><a href="#Reason_constants">原因常量</a>之一,表明扩展为什么被安装。可以是 <code>ADDON_INSTALL</code>、<code>ADDON_UPGRADE</code> 或 <code>ADDON_DOWNGRADE</code> 之一。</dd>
</dl>
<h3 id="uninstall">uninstall</h3>
<p>该函数在卸载扩展的某个版本之前,最后一次调用 <code>shutdown()</code> 之后被调用。如果始终未调用 <code>install()</code>,也不会调用此函数。</p>
<div class="note"><strong>注意:</strong>记住<code>这一点很</code>重要——即使对于当前被禁用或不兼容当前应用程序的扩展,都会被调用 <code>uninstall()</code>。鉴于此,在实现该函数时,优雅地处理那些也许并不存在于应用程序中的 API 变得非常重要。如果一个第三方应用程序在 Firefox 未运行时移除了该扩展,该函数也不会被调用。</div>
<div class="note"><strong>Note:</strong> The uninstall function fires on downgrade and upgrade as well so you should make sure it is an uninstall by doing this:<br>
<code>function uninstall(aData, aReason) {</code><br>
<code> if (aReason == ADDON_UNINSTALL) {</code><br>
<code> console.log('really uninstalling');</code><br>
<code> } else {</code><br>
<code> console.log('not a permanent uninstall, likely an upgrade or downgrade');</code><br>
<code> }</code><br>
<code>}</code></div>
<pre>void uninstall(
data,
reason
);
</pre>
<h6 id="参数_4">参数</h6>
<dl>
<dt><code>data</code></dt>
<dd><a href="#Bootstrap_data">引导数据结构</a>。</dd>
<dt><code>reason</code></dt>
<dd><a href="#Reason_constants">原因常量</a>之一,表明扩展为什么被卸载。可以是 <code>ADDON_UNINSTALL</code>、<code>ADDON_UPGRADE</code> 或 <code>ADDON_DOWNGRADE</code> 之一。</dd>
</dl>
<h2 id="原因常量">原因常量</h2>
<p>上述引导函数都接受一个 <code>reason</code> 参数,该参数向扩展解释为什么调用该函数。 这些原因常量如下:</p>
<table class="standard-table">
<tbody>
<tr>
<td class="header">常量</td>
<td class="header">值</td>
<td class="header">描述</td>
</tr>
<tr>
<td><code>APP_STARTUP</code></td>
<td>1</td>
<td>应用程序启动。</td>
</tr>
<tr>
<td><code>APP_SHUTDOWN</code></td>
<td>2</td>
<td>应用程序关闭。</td>
</tr>
<tr>
<td><code>ADDON_ENABLE</code></td>
<td>3</td>
<td>启用附加组件。</td>
</tr>
<tr>
<td><code>ADDON_DISABLE</code></td>
<td>4</td>
<td>禁用附加组件。(同时被<a class="link-https" href="https://bugzilla.mozilla.org/show_bug.cgi?id=620541">错用于卸载</a>)</td>
</tr>
<tr>
<td><code>ADDON_INSTALL</code></td>
<td>5</td>
<td>安装附加组件。</td>
</tr>
<tr>
<td><code>ADDON_UNINSTALL</code></td>
<td>6</td>
<td>卸载附加组件。</td>
</tr>
<tr>
<td><code>ADDON_UPGRADE</code></td>
<td>7</td>
<td>升级附加组件。</td>
</tr>
<tr>
<td><code>ADDON_DOWNGRADE</code></td>
<td>8</td>
<td>降级附加组件。</td>
</tr>
</tbody>
</table>
<h2 id="引导数据_2"><a name="引导数据"></a>引导数据</h2>
<p>每个切入点被传入一个简单的数据结构,它包含一些关于引导附加组件的有用信息。调用 <code><a href="/en-US/docs/Addons/Add-on_Manager/AddonManager#getAddonByID()">AddonManager.getAddonByID()</a></code> 可以获得更多关于附加组件的信息。该数据是一个简单的 JavaScript 对象,它包括以下属性:</p>
<table class="standard-table">
<tbody>
<tr>
<td class="header">属性</td>
<td class="header">类型</td>
<td class="header">描述</td>
</tr>
<tr>
<td><code>id</code></td>
<td><code>string</code></td>
<td>被引导附加组件的 ID。</td>
</tr>
<tr>
<td><code>version</code></td>
<td><code>string</code></td>
<td>被引导附加组件的版本。</td>
</tr>
<tr>
<td><code>installPath</code></td>
<td><code>nsIFile</code></td>
<td>被引导附加组件的安装位置。可能是一个目录或一个 XPI 文件,这取决于安装该附加组件时是否解包。</td>
</tr>
<tr>
<td><code>resourceURI</code></td>
<td><code>nsIURI</code></td>
<td>附加组件文件根目录的 URI,可能是 <code>jar:</code> 或 <code>file:</code> URI,这取决于安装该附加组件时是否解包。{{ gecko_minversion_inline("7.0") }}</td>
</tr>
<tr>
<td><code>oldVersion</code></td>
<td><code>string</code></td>
<td>以前的安装版本,如果原因是 <code>ADDON_UPGRADE</code> 或 <code>ADDON_DOWNGRADE</code>,并且方法是 <code>install</code> 或 <code>startup</code>。{{ gecko_minversion_inline("22.0") }}</td>
</tr>
<tr>
<td><code>newVersion</code></td>
<td><code>string</code></td>
<td>将要安装的版本,如果原因是 <code>ADDON_UPGRADE</code> 或 <code>ADDON_DOWNGRADE</code>,并且方法是 <code>shutdown</code> 或 <code>uninstall</code>。{{ gecko_minversion_inline("22.0") }}</td>
</tr>
</tbody>
</table>
<div class="note">
<p><strong>注意:</strong>附加组件可能会在程序启动时升降级,此时 <code>startup</code> 方法的原因是 <code>APP_STARTUP</code>,并且不会设置 <code>oldVersion</code> 属性。 另外需注意,在某些情形下有可能发生附加组件升降级但 <code>uninstall</code> 方法没有被调用的情况。</p>
</div>
<h2 id="拓展调试器">拓展调试器</h2>
<p>From Firefox 31 onwards, you can use the <a href="/en-US/Add-ons/Add-on_Debugger">Add-on Debugger</a> to debug bootstrapped add-ons.</p>
<h2 id="本地化(L10n)">本地化(L10n)</h2>
<p>Localizing bootstrapped add-ons is very much the same since Firefox 7, as that is when chrome.manifest compatibility landed.</p>
<h3 id="JS_and_JSM_Files_-_Using_Property_Files">JS and JSM Files - Using Property Files</h3>
<p>To localize your .js and .jsm files you have to use <a href="/en-US/docs/XUL/Tutorial/Property_Files">property files</a>.</p>
<p>The absolute minimum needed here is:</p>
<ol>
<li>File: install.rdf</li>
<li>File: chrome.manifest</li>
<li>File: bootstrap.js</li>
<li>Folder: locale
<ol>
<li>Folder: VALID_LOCALE_HERE
<ol>
<li>File: ANYTHING.properties</li>
</ol>
</li>
</ol>
</li>
</ol>
<p>In the locale folder you must have folders for each of the languages you want to provide; each folder must be named a valid locale (ex: en-US). Inside this folder must be a property file. Inside the chrome.manifest file these locale must be defined. For example if you had a subfolder of en-US in locale folder your chrome.manifest file will have to contain: <code>locale NAME_OF_YOUR_ADDON en-US locale/en-US/</code></p>
<p>Here is an example: <a href="https://github.com/Noitidart/l10n/tree/properties">GitHub :: l10n-properties</a> - on startup of this add-on it will show a prompt saying USA or Great Britain, which ever it deems closest to your locale. You can test different locale by going to about:config and changing preference of general.useragent.locale to en-US and then to en-GB and disabling then re-enabling the add-on.</p>
<h3 id="XUL_and_HTML_Files_-_Using_Entities_from_DTD_Files">XUL and HTML Files - Using Entities from DTD Files</h3>
<p>Many times HTML pages are used, however they cannot be localized with DTD files. There are three changes you must make:</p>
<ol>
<li>You have to change the HTML file's extension to be <code>.xhtml</code></li>
<li>The doctype must be defined point to a DTD file in your locale folder such as: <code><!DOCTYPE html SYSTEM <span class="pl-s1">"chrome://l10n/locale/mozilla.dtd"</span>></code></li>
<li>Must add xmlns attribute to html tag for example: <code><<span class="pl-ent">html</span> <span class="pl-e">xmlns</span>=<span class="pl-s1"><span class="pl-pds">"</span>http://www.w3.org/1999/xhtml<span class="pl-pds">"</span></span>></code></li>
<li>If you have multiple DTD files read on here: <a href="/en-US/docs/Using_multiple_DTDs">Using multiple DTDs</a></li>
</ol>
<p>The bare minimum needed is:</p>
<ol>
<li>File: install.rdf</li>
<li>File: chrome.manifest</li>
<li>File: bootstrap.js</li>
<li>Folder: locale
<ol>
<li>Folder: VALID_LOCALE_HERE
<ol>
<li>File: ANYTHING.dtd</li>
</ol>
</li>
</ol>
</li>
</ol>
<p>The chrome.manifest file must include a definition for content for example: <code>content NAME_OF_YOUR_ADDON ./</code></p>
<p>The chrome.manifest file must also include a line pointing to the locale, just like in the above property section, if you had a folder named en-US in locale, the chrome.manifest file should contain: <code>locale NAME_OF_YOUR_ADDON en-US locale/en-US/</code></p>
<p>Here is an example add-on that opens an HTML page and a XUL page on install: <a href="https://github.com/Noitidart/l10n/tree/c456cc82a8a66b6d552cd8c2299cd2babc383af0">GitHub :: l10n-xhtml-xul</a>. Here is an example showing how to use a localized HTML page as an options page: <a href="https://github.com/Noitidart/l10n/tree/html-options">GitHub :: l10n-html-options</a>. You can go to about:config and change the value of the preference <code>general.useragent.locale </code>to <code>en-US</code> and then to <code>en-GB</code> and then reload the open pages to see the localization change.</p>
<h2 id="延伸阅读">延伸阅读</h2>
<ul>
<li>Wladimir Palant 解释了<a class="external" href="http://adblockplus.org/blog/how-many-hacks-does-it-take-to-make-your-extension-install-without-a-restart">转化已存在的扩展时发现的问题和漏洞</a>,包括一些解决方案和变通方法。</li>
<li>Mark Finkle 为<a class="external" href="http://starkravingfinkle.org/blog/2011/01/bootstrap-jones-adventures-in-restartless-add-ons/">移动 Firefox 中无需重启的附加组件</a>提供了一些简单示例代码,向自引导型扩展<a class="external" href="http://starkravingfinkle.org/blog/2011/01/restartless-add-ons-more-resources/">添加资源(如选项窗口)</a>,<a class="external" href="http://starkravingfinkle.org/blog/2011/01/restartless-add-ons-%e2%80%93-default-preferences/">使用默认参数</a>但不使用 <code>default/preferences/prefs.js</code> 文件。</li>
<li>Kris Maglione 写的关于在自引导型扩展中<a class="external" href="http://maglione-k.users.sourceforge.net/bootstrapped.xhtml">清理程序的要求</a>。</li>
<li>Edward Lee 炫耀了一些<a class="external" href="http://ed.agadak.net/2011/01/restartless-add-on-example-code">有用的编码模式和示例</a>,您可以用于您的自引导型附加组件。</li>
<li>Erik Vold 的关于无需重启附加组件的快速指南:
<ul>
<li><a class="external" href="http://erikvold.com/blog/index.cfm/2010/11/3/restartless-restart-addon-for-firefox">第一部分:简介</a></li>
<li><a class="external" href="http://erikvold.com/blog/index.cfm/2011/1/2/restartless-firefox-addons-part-2-includes">第二部分:包含(includes)</a></li>
<li><a class="external" href="http://erikvold.com/blog/index.cfm/2011/2/7/restartless-firefox-addons-part-3-icons">第三部分:图标</a></li>
<li><a class="external" href="http://erikvold.com/blog/index.cfm/2011/2/18/restartless-firefox-addons-part-4-localization-l10n">第四部分:本地化(l10n)</a></li>
<li><a class="external" href="http://erikvold.com/blog/index.cfm/2011/4/1/restartless-firefox-addons-part-5-logging">第五部分:日志</a></li>
<li><a class="external" href="http://erikvold.com/blog/index.cfm/2011/6/14/restartless-firefox-addons-part-6-better-includes">第六部分:更好地包含</a></li>
<li><a class="external" href="http://erikvold.com/blog/index.cfm/2011/6/19/restartless-firefox-addons-part-7-css">第七部分:CSS</a></li>
<li><a class="external" href="http://erikvold.com/blog/index.cfm/2011/6/19/restartless-firefox-addons-part-8-require-commonjs-and-jetpack">第八部分:要求(Require)、CommonJS 和 Jetpack</a></li>
</ul>
</li>
<li>Firefox 7 及以上版本中<a href="/en-US/docs/Extensions/Inline_Options">内联选项(Inline Options)</a>的文档。</li>
</ul>
|