From 1ab8af4ad24257903b53019e24a2f0439d8facdf Mon Sep 17 00:00:00 2001 From: xiangyu <3170102889@zju.edu.cn> Date: Fri, 16 Dec 2022 18:11:47 +0800 Subject: [PATCH] add: preference compatibility for Zotero 6 & 7 --- addon/bootstrap.js | 21 +-- addon/chrome.manifest | 1 - .../icons}/favicon.png | Bin .../icons}/favicon@0.5x.png | Bin addon/chrome/content/preferences.xhtml | 10 +- addon/chrome/locale/en-US/overlay.dtd | 1 + addon/chrome/locale/zh-CN/overlay.dtd | 1 + addon/install.rdf | 2 +- addon/manifest.json | 4 +- src/events.ts | 7 +- src/index.ts | 7 +- src/prefs.ts | 27 +++- src/utils.ts | 126 +++++++++++++++++- src/views.ts | 28 +++- typing/global.d.ts | 22 +++ 15 files changed, 220 insertions(+), 37 deletions(-) rename addon/chrome/{skin/default/addontemplate => content/icons}/favicon.png (100%) rename addon/chrome/{skin/default/addontemplate => content/icons}/favicon@0.5x.png (100%) diff --git a/addon/bootstrap.js b/addon/bootstrap.js index 39826cf..d3c07b0 100644 --- a/addon/bootstrap.js +++ b/addon/bootstrap.js @@ -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, + // }); } } diff --git a/addon/chrome.manifest b/addon/chrome.manifest index 6c33bcd..713e1b0 100644 --- a/addon/chrome.manifest +++ b/addon/chrome.manifest @@ -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/ diff --git a/addon/chrome/skin/default/addontemplate/favicon.png b/addon/chrome/content/icons/favicon.png similarity index 100% rename from addon/chrome/skin/default/addontemplate/favicon.png rename to addon/chrome/content/icons/favicon.png diff --git a/addon/chrome/skin/default/addontemplate/favicon@0.5x.png b/addon/chrome/content/icons/favicon@0.5x.png similarity index 100% rename from addon/chrome/skin/default/addontemplate/favicon@0.5x.png rename to addon/chrome/content/icons/favicon@0.5x.png diff --git a/addon/chrome/content/preferences.xhtml b/addon/chrome/content/preferences.xhtml index 60607f3..ad73f90 100644 --- a/addon/chrome/content/preferences.xhtml +++ b/addon/chrome/content/preferences.xhtml @@ -6,8 +6,16 @@ + + + &zotero.__addonRef__.pref.input.label; + diff --git a/addon/chrome/locale/en-US/overlay.dtd b/addon/chrome/locale/en-US/overlay.dtd index 319c425..beffe6c 100644 --- a/addon/chrome/locale/en-US/overlay.dtd +++ b/addon/chrome/locale/en-US/overlay.dtd @@ -1,5 +1,6 @@ + \ No newline at end of file diff --git a/addon/chrome/locale/zh-CN/overlay.dtd b/addon/chrome/locale/zh-CN/overlay.dtd index 8417667..8d0a76f 100644 --- a/addon/chrome/locale/zh-CN/overlay.dtd +++ b/addon/chrome/locale/zh-CN/overlay.dtd @@ -1,5 +1,6 @@ + \ No newline at end of file diff --git a/addon/install.rdf b/addon/install.rdf index 15fd8bf..77666f5 100644 --- a/addon/install.rdf +++ b/addon/install.rdf @@ -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" diff --git a/addon/manifest.json b/addon/manifest.json index c7bc69b..029bf11 100644 --- a/addon/manifest.json +++ b/addon/manifest.json @@ -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": { diff --git a/src/events.ts b/src/events.ts index 441ad37..52fd8eb 100644 --- a/src/events.ts +++ b/src/events.ts @@ -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 diff --git a/src/index.ts b/src/index.ts index 845bb14..a58cf92 100644 --- a/src/index.ts +++ b/src/index.ts @@ -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); } diff --git a/src/prefs.ts b/src/prefs.ts index 4f10cfc..82ad941 100644 --- a/src/prefs.ts +++ b/src/prefs.ts @@ -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}!` + ); + }); } } diff --git a/src/utils.ts b/src/utils.ts index 9de9d8a..e7304c0 100644 --- a/src/utils.ts +++ b/src/utils.ts @@ -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 + ? ` { + return ( + preamble + + ` %_dtd-${index}; ` + ); + }, + "" + )}]>` + : "" + } + + ${str} + `; + 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 = ` + ${(await Zotero.File.getContentsAsync(options.src)) as string} + `; + 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); } diff --git a/src/views.ts b/src/views.ts index a1bc6bf..d8299d8 100644 --- a/src/views.ts +++ b/src/views.ts @@ -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( diff --git a/typing/global.d.ts b/typing/global.d.ts index 415b7fe..75c2131 100644 --- a/typing/global.d.ts +++ b/typing/global.d.ts @@ -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; } +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;