add: preference compatibility for Zotero 6 & 7

This commit is contained in:
xiangyu 2022-12-16 18:11:47 +08:00
parent 583fbf166e
commit 1ab8af4ad2
15 changed files with 220 additions and 37 deletions

21
addon/bootstrap.js vendored
View File

@ -69,11 +69,12 @@ async function startup({ id, version, resourceURI, rootURI }, reason) {
rootURI = resourceURI.spec;
}
Services.scriptloader.loadSubScript(
`${rootURI}/chrome/content/scripts/index.js`
);
const ctx = { Zotero, rootURI };
Zotero.AddonTemplate.rootURI = rootURI;
Services.scriptloader.loadSubScript(
`${rootURI}/chrome/content/scripts/index.js`,
ctx
);
if (Zotero.platformMajorVersion >= 102) {
var aomStartup = Components.classes[
@ -86,12 +87,12 @@ async function startup({ id, version, resourceURI, rootURI }, reason) {
["locale", "__addonRef__", "zh-CN", rootURI + "chrome/locale/zh-CN/"],
]);
Zotero.PreferencePanes.register({
pluginID: "__addonID__",
src: rootURI + "chrome/content/preferences.xhtml",
extraDTD: [`chrome://__addonRef__/locale/overlay.dtd`],
defaultXUL: true,
});
// Zotero.PreferencePanes.register({
// pluginID: "__addonID__",
// src: rootURI + "chrome/content/preferences.xhtml",
// extraDTD: ["chrome://__addonRef__/locale/overlay.dtd"],
// defaultXUL: true,
// });
}
}

View File

@ -1,4 +1,3 @@
content __addonRef__ chrome/content/
skin __addonRef__ default chrome/skin/default/__addonRef__/
locale __addonRef__ en-US chrome/locale/en-US/
locale __addonRef__ zh-CN chrome/locale/zh-CN/

View File

Before

Width:  |  Height:  |  Size: 677 B

After

Width:  |  Height:  |  Size: 677 B

View File

Before

Width:  |  Height:  |  Size: 836 B

After

Width:  |  Height:  |  Size: 836 B

View File

@ -6,8 +6,16 @@
<label><html:h2>Addon Template Example</html:h2></label>
<checkbox
id="zotero-prefpane-__addonRef__-enable"
preference="pref-__addonRef__-enable"
label="&zotero.__addonRef__.pref.enable.label;"
/>
<hbox>
<html:input
type="text"
id="zotero-prefpane-__addonRef__-input"
></html:input>
<html:label for="zotero-prefpane-__addonRef__-input"
>&zotero.__addonRef__.pref.input.label;</html:label
>
</hbox>
</groupbox>
</vbox>

View File

@ -1,5 +1,6 @@
<!ENTITY zotero.__addonRef__.itemmenu.test.label "addon template">
<!ENTITY zotero.__addonRef__.pref.enable.label "Enable">
<!ENTITY zotero.__addonRef__.pref.input.label "Input">
<!ENTITY zotero.__addonRef__.help.version.label "__addonName__ VERSION __buildVersion__">
<!ENTITY zotero.__addonRef__.help.releasetime.label "Build __buildTime__">

View File

@ -1,5 +1,6 @@
<!ENTITY zotero.__addonRef__.itemmenu.test.label "插件模板">
<!ENTITY zotero.__addonRef__.pref.enable.label "开启">
<!ENTITY zotero.__addonRef__.pref.input.label "输入">
<!ENTITY zotero.__addonRef__.help.version.label "__addonName__ 版本 __buildVersion__">
<!ENTITY zotero.__addonRef__.help.releasetime.label "Build __buildTime__">

View File

@ -12,7 +12,7 @@
em:creator="__author__"
em:description="__description__"
em:homepageURL="__homepage__"
em:iconURL="chrome://__addonRef__/skin/favicon.png"
em:iconURL="chrome://__addonRef__/content/icons/favicon.png"
em:optionsURL="chrome://__addonRef__/content/preferences.xul"
em:updateURL="__updaterdf__"
em:multiprocessCompatible="true"

View File

@ -5,8 +5,8 @@
"description": "__description__",
"author": "__author__",
"icons": {
"48": "chrome/skin/default/__addonRef__/favicon@0.5x.png",
"96": "chrome/skin/default/__addonRef__/favicon.png"
"48": "chrome/content/icons/favicon@0.5x.png",
"96": "chrome/content/icons/favicon.png"
},
"applications": {
"zotero": {

View File

@ -28,10 +28,11 @@ class AddonEvents extends AddonModule {
};
}
public async onInit(_Zotero: _ZoteroConstructable) {
public async onInit(_Zotero: _ZoteroConstructable, rootURI) {
this._Addon.Zotero = _Zotero;
this._Addon.rootURI = rootURI;
// This function is the setup code of the addon
Zotero.debug(`${addonName}: init called`);
this._Addon.Utils.Tool.log(`${addonName}: init called`);
// alert(112233);
// Reset prefs
@ -72,7 +73,7 @@ class AddonEvents extends AddonModule {
public onUnInit(): void {
const Zotero = this._Addon.Zotero;
Zotero.debug(`${addonName}: uninit called`);
this._Addon.Utils.Tool.log(`${addonName}: uninit called`);
// Remove elements and do clean up
this._Addon.views.unInitViews();
// Remove addon object

View File

@ -1,10 +1,7 @@
import Addon from "./addon";
const Zotero = Components.classes["@zotero.org/Zotero;1"].getService(
Components.interfaces.nsISupports
).wrappedJSObject;
if (!Zotero.AddonTemplate) {
Zotero.AddonTemplate = new Addon();
Zotero.AddonTemplate.events.onInit(Zotero);
// @ts-ignore
Zotero.AddonTemplate.events.onInit(Zotero, rootURI);
}

View File

@ -1,6 +1,6 @@
import Addon from "./addon";
import AddonModule from "./module";
import { addonName } from "../package.json";
import { addonName, addonRef } from "../package.json";
class AddonPrefs extends AddonModule {
private _window: Window;
@ -11,15 +11,36 @@ class AddonPrefs extends AddonModule {
// This function is called when the prefs window is opened
// See addon/chrome/content/preferences.xul onpaneload
this._window = _window;
Zotero.debug(`${addonName}: init preferences`);
this._Addon.Utils.Tool.log(`${addonName}: init preferences`);
this.updatePrefsUI();
this.bindPrefEvents();
}
private updatePrefsUI() {
// You can initialize some UI elements on prefs window
// with this._window.document
// Or bind some events to the elements
Zotero.debug(`${addonName}: init preferences UI`);
this._Addon.Utils.Tool.log(`${addonName}: init preferences UI`);
}
private bindPrefEvents() {
this._window.document
.querySelector(`#zotero-prefpane-${addonRef}-enable`)
?.addEventListener("command", (e) => {
this._Addon.Utils.Tool.log(e);
this._window.alert(
`Successfully changed to ${(e.target as XUL.Checkbox).checked}!`
);
});
this._window.document
.querySelector(`#zotero-prefpane-${addonRef}-input`)
?.addEventListener("change", (e) => {
this._Addon.Utils.Tool.log(e);
this._window.alert(
`Successfully changed to ${(e.target as HTMLInputElement).value}!`
);
});
}
}

View File

@ -21,12 +21,19 @@ class AddonUtils extends AddonModule {
// Check if it's running on Zotero 7 (Firefox 102)
isZotero7: () => Zotero.platformMajorVersion >= 102,
// Firefox 102 support DOMParser natively
getDOMParser: () =>
this.Compat.isZotero7()
? new DOMParser()
: Components.classes[
"@mozilla.org/xmlextras/domparser;1"
].createInstance(Components.interfaces.nsIDOMParser),
getDOMParser: () => {
if (this.Compat.isZotero7()) {
return new DOMParser();
}
try {
return new (this.Compat.getZotero().getMainWindow().DOMParser)();
} catch (e) {
return Components.classes[
"@mozilla.org/xmlextras/domparser;1"
].createInstance(Components.interfaces.nsIDOMParser);
}
},
// create XUL element
createXULElement: (doc: Document, type: string) => {
if (this.Compat.isZotero7()) {
@ -39,6 +46,111 @@ class AddonUtils extends AddonModule {
) as XUL.Element;
}
},
parseXHTMLToFragment: (
str: string,
entities: string[] = [],
defaultXUL = true
) => {
// Adapted from MozXULElement.parseXULToFragment
/* eslint-disable indent */
let parser = this.Compat.getDOMParser();
// parser.forceEnableXULXBL();
const xulns =
"http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul";
const htmlns = "http://www.w3.org/1999/xhtml";
const wrappedStr = `${
entities.length
? `<!DOCTYPE bindings [ ${entities.reduce(
(preamble, url, index) => {
return (
preamble +
`<!ENTITY % _dtd-${index} SYSTEM "${url}"> %_dtd-${index}; `
);
},
""
)}]>`
: ""
}
<html:div xmlns="${defaultXUL ? xulns : htmlns}"
xmlns:xul="${xulns}" xmlns:html="${htmlns}">
${str}
</html:div>`;
this.Tool.log(wrappedStr, parser);
let doc = parser.parseFromString(wrappedStr, "text/xml");
/* eslint-enable indent */
console.log(doc);
if (doc.documentElement.localName === "parsererror") {
throw new Error("not well-formed XHTML");
}
// We use a range here so that we don't access the inner DOM elements from
// JavaScript before they are imported and inserted into a document.
let range = doc.createRange();
range.selectNodeContents(doc.querySelector("div"));
return range.extractContents();
},
prefPaneCache: { win: undefined, listeners: [], ids: [] },
registerPrefPane: (options: PrefPaneOptions) => {
const windowListener = {
onOpenWindow: (xulWindow) => {
const win: Window = xulWindow
.QueryInterface(Components.interfaces.nsIInterfaceRequestor)
.getInterface(Components.interfaces.nsIDOMWindow);
win.addEventListener(
"load",
async () => {
if (
win.location.href ===
"chrome://zotero/content/preferences/preferences.xul"
) {
this.Tool.log("registerPrefPane:detected", options);
const Zotero = this.Compat.getZotero();
options.id || (options.id = `plugin-${new Date().getTime()}`);
const src = `<prefpane id="${
options.id
}" insertafter="zotero-prefpane-advanced" label="${
options.label || options.pluginID
}" image="${options.image || ""}">
${(await Zotero.File.getContentsAsync(options.src)) as string}
</prefpane>`;
const frag = this.Compat.parseXHTMLToFragment(
src,
options.extraDTD,
options.defaultXUL
);
this.Tool.log(frag);
const prefWindow = win.document.querySelector("prefwindow");
prefWindow.appendChild(frag);
const prefPane = win.document.querySelector(`#${options.id}`);
// @ts-ignore
prefWindow.addPane(prefPane);
this.Compat.prefPaneCache.win = win;
this.Compat.prefPaneCache.listeners.push(windowListener);
this.Compat.prefPaneCache.ids.push(options.id);
if (options.onload) {
options.onload(win);
}
}
},
false
);
},
};
Services.wm.addListener(windowListener);
},
unregisterPrefPane: () => {
this.Compat.prefPaneCache.listeners.forEach((l) =>
Services.wm.removeListener(l)
);
const win = this.Compat.prefPaneCache.win;
if (win && !win.closed) {
this.Compat.prefPaneCache.ids.forEach((id) =>
win.document.querySelector(id)?.remove()
);
}
},
};
this.Tool = {
getCopyHelper: () => new CopyHelper(),
@ -84,7 +196,7 @@ class AddonUtils extends AddonModule {
},
log: (...data: any[]) => {
try {
this._Addon.Zotero.getMainWindow().console.log(data);
this._Addon.Zotero.getMainWindow().console.log(...data);
for (const d of data) {
this._Addon.Zotero.debug(d);
}

View File

@ -11,7 +11,7 @@ class AddonViews extends AddonModule {
this.progressWindowIcon = {
success: "chrome://zotero/skin/tick.png",
fail: "chrome://zotero/skin/cross.png",
default: `chrome://${addonRef}/skin/favicon.png`,
default: `chrome://${addonRef}/content/icons/favicon.png`,
};
}
@ -19,9 +19,9 @@ class AddonViews extends AddonModule {
const Zotero = this._Addon.Zotero;
// You can init the UI elements that
// cannot be initialized with overlay.xul
Zotero.debug("Initializing UI");
this._Addon.Utils.Tool.log("Initializing UI");
const menuIcon =
"";
'url("chrome://addontemplate/content/icons/favicon@0.5x.png")';
// item menuitem with icon
this._Addon.Utils.UI.insertMenuItem("item", {
tag: "menuitem",
@ -62,12 +62,32 @@ class AddonViews extends AddonModule {
public initPrefs() {
const Zotero = this._Addon.Zotero;
this._Addon.Utils.Tool.log(this._Addon.rootURI);
const prefOptions = {
pluginID: addonID,
src: this._Addon.rootURI + "chrome/content/preferences.xhtml",
label: "Template",
image: `chrome://${addonRef}/content/icons/favicon.png`,
extraDTD: [`chrome://${addonRef}/locale/overlay.dtd`],
defaultXUL: true,
onload: (win: Window) => {
this._Addon.prefs.initPreferences(win);
},
};
if (this._Addon.Utils.Compat.isZotero7()) {
Zotero.PreferencePanes.register(prefOptions);
} else {
this._Addon.Utils.Compat.registerPrefPane(prefOptions);
}
}
public unInitViews() {
const Zotero = this._Addon.Zotero;
Zotero.debug("Uninitializing UI");
this._Addon.Utils.Tool.log("Uninitializing UI");
this._Addon.Utils.UI.removeAddonElements();
if (!this._Addon.Utils.Compat.isZotero7()) {
this._Addon.Utils.Compat.unregisterPrefPane();
}
}
public showProgressWindow(

22
typing/global.d.ts vendored
View File

@ -3,6 +3,14 @@ declare interface ZoteroCompat {
isZotero7: () => boolean;
getDOMParser: () => DOMParser;
createXULElement: (doc: Document, type: string) => XUL.Element;
parseXHTMLToFragment: (
str: string,
entities: string[],
defaultXUL?: boolean
) => DocumentFragment;
prefPaneCache: { win: Window; listeners: any[]; ids: string[] };
registerPrefPane: (options: PrefPaneOptions) => void;
unregisterPrefPane: () => void;
}
declare interface ZoteroTool {
@ -79,6 +87,20 @@ declare interface MenuitemOptions {
subElementOptions?: Array<MenuitemOptions>;
}
declare interface PrefPaneOptions {
pluginID: string;
src: string;
id?: string;
parent?: string;
label?: string;
image?: string;
extraDTD?: string[];
scripts?: string[];
defaultXUL?: boolean;
// Only for Zotero 6
onload?: (win: Window) => any;
}
declare class CopyHelper {
addText: (source: string, type: "text/html" | "text/unicode") => CopyHelper;
addImage: (source: string) => CopyHelper;