add: export note to markdown

This commit is contained in:
xiangyu 2022-04-30 22:23:24 +08:00
parent e7be379560
commit 738ebf2057
5 changed files with 235 additions and 77 deletions

View File

@ -59,7 +59,17 @@ class AddonEvents extends AddonBase {
public async onEditorEvent(message: EditorMessage) {
Zotero.debug(`Knowledge4Zotero: onEditorEvent\n${message.type}`);
if (message.type === "addNoteInstance") {
if (message.type === "openWorkspace") {
/*
message.content = {}
*/
await this._Addon.knowledge.openWorkspaceWindow();
} else if (message.type === "addNoteInstance") {
/*
message.content = {
editorInstance,
}
*/
let mainKnowledgeID = parseInt(
Zotero.Prefs.get("Knowledge4Zotero.mainKnowledgeID")
);
@ -71,22 +81,34 @@ class AddonEvents extends AddonBase {
Zotero.debug(`Knowledge4Zotero: main Knowledge`);
this._Addon.views.addEditorButton(
message.content.editorInstance,
"mainKnowledge",
isMainKnowledge ? "isMainKnowledge" : "setMainKnowledge",
"workspace",
isMainKnowledge ? "isMainKnowledge" : "notMainKnowledge",
isMainKnowledge
? "This Note is Knowledge Workspace"
: "Use Current Note as Knowledge Workspace",
new EditorMessage("setMainKnowledge", {})
? "Edit the main knowledge in Workspace"
: "Open Workspace",
"start",
new EditorMessage("openWorkspace", {})
);
this._Addon.views.addEditorButton(
message.content.editorInstance,
"addToKnowledge",
"addToKnowledge",
"Add Note Link to Knowledge Workspace",
"middle",
new EditorMessage("addToKnowledge", {
itemID: message.content.editorInstance._item.id,
})
);
this._Addon.views.addEditorButton(
message.content.editorInstance,
"export",
"export",
"Export Markdown with linked Notes",
"end",
new EditorMessage("export", {
itemID: message.content.editorInstance._item.id,
})
);
if (!message.content.editorInstance._knowledgeSelectionInitialized) {
this.addEditorDocumentEventListener(
message.content.editorInstance,
@ -95,12 +117,36 @@ class AddonEvents extends AddonBase {
);
message.content.editorInstance._knowledgeSelectionInitialized = true;
}
} else if (message.type === "enterWorkspace") {
/*
message.content = {
editorInstance,
params: "main" | "preview"
}
*/
if (message.content.params === "main") {
// This is a main knowledge, hide all buttons except the export button and add title
} else {
// This is a preview knowledge, hide openWorkspace button add show close botton
}
} else if (message.type === "leaveWorkspace") {
/*
message.content = {
editorInstance,
params: "main" | "preview"
}
*/
if (message.content.params === "main") {
// This is a main knowledge, show all buttons and remove title
} else {
// This is a preview knowledge, show openWorkspace button add hide close botton
}
} else if (message.type === "addToKnowledge") {
/*
message.content = {
editorInstance
}
*/
*/
Zotero.debug("Knowledge4Zotero: addToKnowledge");
this._Addon.knowledge.addLinkToNote(
undefined,
@ -112,8 +158,7 @@ class AddonEvents extends AddonBase {
message.content = {
editorInstance
}
*/
// TODO: Complete this part
*/
Zotero.debug("Knowledge4Zotero: setMainKnowledge");
let mainKnowledgeID = parseInt(
Zotero.Prefs.get("Knowledge4Zotero.mainKnowledgeID")
@ -132,29 +177,36 @@ class AddonEvents extends AddonBase {
message.content.editorInstance._item.id
);
// Set the button to selected state
this._Addon.views.changeEditorButton(
this._Addon.views.changeEditorButtonView(
message.content.event.target,
"isMainKnowledge",
"This Note is Knowledge Workspace"
"Edit the main knowledge in Workspace"
);
// TODO: update workspace window here
await this._Addon.knowledge.setWorkspaceMainNote();
await this._Addon.knowledge.setWorkspaceNote("main");
for (let editor of Zotero.Notes._editorInstances) {
await editor._initPromise;
if (editor._item.id === mainKnowledgeID) {
let button =
editor._iframeWindow.document.getElementById("mainKnowledge");
editor._iframeWindow.document.getElementById("workspace");
if (button) {
this._Addon.views.changeEditorButton(
this._Addon.views.changeEditorButtonView(
button,
"setMainKnowledge",
"Use Current Note as Knowledge Workspace"
"notMainKnowledge",
"Open Workspace"
);
}
}
}
}
} else if (message.type === "onNoteLink") {
/*
message.content = {
params: {
item: ZoteroItem | boolean,
infoText: string
}
}
*/
if (!message.content.params.item) {
Zotero.debug(`Knowledge4Zotero: ${message.content.params.infoText}`);
}
@ -163,7 +215,8 @@ class AddonEvents extends AddonBase {
);
let _window = this._Addon.knowledge.getWorkspaceWindow();
if (_window) {
this._Addon.knowledge.setWorkspacePreviewNote(
this._Addon.knowledge.setWorkspaceNote(
"preview",
message.content.params.item
);
(_window as Window).focus();
@ -203,6 +256,15 @@ class AddonEvents extends AddonBase {
this._Addon.knowledge.currentLine = currentLineIndex;
Zotero.debug(`Knowledge4Zotero: line ${currentLineIndex} selected.`);
}
} else if (message.type === "export") {
/*
message.content = {
editorInstance
}
*/
await this._Addon.knowledge.exportNoteToFile(
message.content.editorInstance._item
);
} else {
Zotero.debug(`Knowledge4Zotero: message not handled.`);
}

View File

@ -1,4 +1,4 @@
import { AddonBase } from "./base";
import { AddonBase, EditorMessage } from "./base";
const TreeModel = require("./treemodel");
@ -34,7 +34,7 @@ class Knowledge extends AddonBase {
);
this.workspaceWindow = win;
await this.waitWorkspaceRedy();
this.setWorkspaceMainNote();
this.setWorkspaceNote("main");
}
}
@ -51,35 +51,46 @@ class Knowledge extends AddonBase {
return t < 500;
}
async setWorkspaceMainNote(note: ZoteroItem = undefined) {
async getWorkspaceEditor(type: "main" | "preview" = "main") {
let _window = this.getWorkspaceWindow() as Window;
if (!_window) {
return;
}
await this.waitWorkspaceRedy();
return _window.document.getElementById(`zotero-note-editor-${type}`);
}
async setWorkspaceNote(
type: "main" | "preview" = "main",
note: ZoteroItem = undefined
) {
let _window = this.getWorkspaceWindow() as Window;
note = note || this.getWorkspaceNote();
if (!_window) {
return;
}
await this.waitWorkspaceRedy();
let noteEditor: any = _window.document.getElementById(
"zotero-note-editor-main"
);
noteEditor.mode = "edit";
noteEditor.viewMode = "library";
noteEditor.parent = null;
noteEditor.item = note;
}
async setWorkspacePreviewNote(note: ZoteroItem) {
let _window = this.getWorkspaceWindow() as Window;
if (!_window) {
return;
let noteEditor: any = await this.getWorkspaceEditor(type);
let lastEditorInstance = noteEditor._editorInstance as EditorInstance;
if (lastEditorInstance) {
await this._Addon.events.onEditorEvent(
new EditorMessage("leaveWorkspace", {
editorInstance: lastEditorInstance,
params: type,
})
);
}
await this.waitWorkspaceRedy();
let noteEditor: any = _window.document.getElementById(
"zotero-note-editor-preview"
);
noteEditor.mode = "edit";
noteEditor.viewMode = "library";
noteEditor.parent = null;
noteEditor.item = note;
await this._Addon.events.onEditorEvent(
new EditorMessage("enterWorkspace", {
editorInstance: noteEditor,
params: type,
})
);
}
getLinesInNote(note: ZoteroItem): string[] {
@ -396,6 +407,81 @@ class Knowledge extends AddonBase {
return text;
}
async exportNoteToFile(note: ZoteroItem, convertNoteLinks: boolean = true) {
let exporter = new Zotero_File_Exporter();
if (convertNoteLinks) {
let item: ZoteroItem = new Zotero.Item("note");
let noteLines = this.getLinesInNote(note);
let newLines = [];
for (let i in noteLines) {
newLines.push(noteLines[i]);
let linkIndex = noteLines[i].search(/zotero:\/\/note\//g);
while (linkIndex >= 0) {
let link = noteLines[i].substring(linkIndex);
link = link.substring(0, link.search('"'));
let res = await this.getNoteFromLink(link);
if (res.item && res.item.id !== note.id) {
Zotero.debug(`Knowledge4Zotero: Exporting sub-note ${link}`);
newLines.push("<blockquote>");
newLines.push(
`<p><strong>Linked Note: <a href="${link}" rel="noopener noreferrer nofollow">${res.item.getNoteTitle()}</a></strong></p>`
);
let linkedLines = this.getLinesInNote(res.item);
newLines = newLines.concat(linkedLines);
newLines.push("</blockquote>");
}
noteLines[i] = noteLines[i].substring(
noteLines[i].search(/zotero:\/\/note\//g)
);
noteLines[i] = noteLines[i].substring(
noteLines[i].search(/<\/a>/g) + "</a>".length
);
linkIndex = noteLines[i].search(/zotero:\/\/note\//g);
}
}
item.setNote(`<div data-schema-version="8">${newLines.join("\n")}</div>`);
item.saveTx();
exporter.items = [item];
exporter.save();
Zotero.Items.erase(item.id);
} else {
exporter.items = [note];
exporter.save();
}
}
async getNoteFromLink(uri: string) {
let [groupID, noteKey] = uri.substring("zotero://note/".length).split("/");
// User libraryID by defaultx
let libraryID = 1;
if (groupID !== "u") {
// Not a user item
let _groupID = parseInt(groupID);
libraryID = Zotero.Groups.getLibraryIDFromGroupID(_groupID);
}
if (!libraryID) {
return {
item: false,
infoText: "Library does not exist or access denied.",
};
}
let item = await Zotero.Items.getByLibraryAndKeyAsync(libraryID, noteKey);
if (!item || !item.isNote()) {
return {
item: false,
infoText: "Note does not exist or is not a note.",
};
}
return {
item: item,
infoText: "OK",
};
}
parseNoteHTML(note: ZoteroItem): Element {
note = note || this.getWorkspaceNote();
if (!note) {

View File

@ -13,8 +13,9 @@ class AddonViews extends AddonBase {
};
this.editorIcon = {
addToKnowledge: `<svg t="1651124422933" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="3269" width="24" height="24"><path d="M896.00324 352c70.7 0 128-57.3 128-128 0-70.6-57.4-128-128-128-70.7 0-128 57.3-128 128 0 18.8 4.1 36.7 11.3 52.8 2.7 6 1.4 13.1-3.3 17.8l-24.2 24.2c-5.7 5.7-14.9 6.3-21.2 1.2-38.1-30.1-86.3-48-138.6-48-18.8 0-37.1 2.3-54.6 6.7-6.9 1.7-14.1-1.4-17.7-7.5l-6.6-11.4c-3.4-5.8-2.7-13.1 1.6-18.3 18.6-22.6 29.7-51.6 29.3-83.2C543.10324 89 486.30324 32.6 417.00324 32c-70.6-0.6-128.1 56.1-129 126.3-0.9 69.5 56.5 128.6 126 129.6 9.4 0.1 18.5-0.7 27.4-2.5 6.8-1.4 13.6 1.7 17.1 7.7l2.2 3.8c4 7 2.2 15.9-4.2 20.7-42.4 32.3-73 79.4-84 133.6-1.5 7.4-8.1 12.7-15.7 12.7h-94.1c-6.6 0-12.6-4-14.9-10.2-18.1-48-64.3-82.2-118.5-82.8C58.70324 370.3 0.50324 427.6 0.00324 498.1-0.49676 569.2 57.00324 627 128.00324 627c56.7 0 104.8-36.9 121.6-87.9 2.2-6.6 8.3-11.1 15.2-11.1h92c7.6 0 14.2 5.4 15.7 12.9 9.5 46.7 33.5 88 67 119.2 5.4 5 6.6 13.2 2.9 19.6l-21.7 37.6c-3.7 6.3-11.1 9.4-18.2 7.4-11.1-3.1-22.7-4.7-34.8-4.7-69.7 0.1-127 56.8-127.8 126.6-0.8 71.7 57.4 130 129.1 129.4 69.5-0.6 126.3-57.3 126.9-126.8 0.3-28-8.5-53.9-23.5-75.1-3.6-5.1-3.9-11.8-0.8-17.2l24.9-43.1c3.9-6.7 12-9.7 19.3-7 23.7 8.6 49.3 13.2 76 13.2 34.9 0 67.9-8 97.3-22.2 7.6-3.7 16.7-0.9 20.9 6.4l37 64c-26.3 23.5-43 57.7-43 95.8 0 70.9 58 128.5 128.9 128 69.7-0.5 126.2-56.7 127.1-126.3 0.9-70.1-57-129.3-127.1-129.7-6.2 0-12.3 0.4-18.3 1.2-6.5 0.9-12.8-2.2-16.1-7.8l-39.2-67.9c-3.4-5.9-2.7-13.3 1.7-18.4 34.2-39.3 54.9-90.7 54.9-147 0-38.9-9.9-75.5-27.4-107.4-3.4-6.2-2.2-13.9 2.8-18.9l28.4-28.4c4.9-4.9 12.4-6 18.7-2.9 17.4 8.6 36.9 13.5 57.6 13.5z m0-192c35.3 0 64 28.7 64 64s-28.7 64-64 64-64-28.7-64-64 28.7-64 64-64zM128.00324 563c-35.3 0-64-28.7-64-64s28.7-64 64-64 64 28.7 64 64-28.7 64-64 64z m240 349c-35.3 0-64-28.7-64-64s28.7-64 64-64 64 28.7 64 64-28.7 64-64 64z m464-112c35.3 0 64 28.7 64 64s-28.7 64-64 64-64-28.7-64-64 28.7-64 64-64zM416.00324 224c-35.3 0-64-28.7-64-64s28.7-64 64-64 64 28.7 64 64-28.7 64-64 64z m289.1 385.1C674.90324 639.4 634.70324 656 592.00324 656s-82.9-16.6-113.1-46.9C448.60324 578.9 432.00324 538.7 432.00324 496s16.6-82.9 46.9-113.1C509.10324 352.6 549.30324 336 592.00324 336s82.9 16.6 113.1 46.9C735.40324 413.1 752.00324 453.3 752.00324 496s-16.6 82.9-46.9 113.1z" p-id="3270"></path></svg>`,
setMainKnowledge: `<svg t="1651124314636" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1689" width="24" height="24"><path d="M877.44 383.786667L624.426667 117.333333C594.986667 86.186667 554.88 69.12 512 69.12s-82.986667 17.066667-112.426667 48.213333L146.56 383.786667a148.266667 148.266667 0 0 0-40.746667 102.4v302.08c0 85.76 69.76 155.52 155.52 155.52h501.546667c85.76 0 155.52-69.76 155.52-155.52V485.973333c0-38.186667-14.506667-74.453333-40.96-102.186666z m-44.373333 404.266666c0 38.826667-31.573333 70.186667-70.186667 70.186667H261.333333c-38.826667 0-70.186667-31.573333-70.186666-70.186667V485.973333c0-16.213333 6.186667-31.786667 17.28-43.52L461.44 176c13.226667-13.866667 31.146667-21.546667 50.56-21.546667s37.333333 7.68 50.56 21.76l253.013333 266.453334c11.306667 11.733333 17.28 27.306667 17.28 43.52v301.866666z" p-id="1690"></path><path d="M608 687.786667h-192c-23.466667 0-42.666667 19.2-42.666667 42.666666s19.2 42.666667 42.666667 42.666667h192c23.466667 0 42.666667-19.2 42.666667-42.666667s-19.2-42.666667-42.666667-42.666666z" p-id="1691"></path></svg>`,
notMainKnowledge: `<svg t="1651124314636" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1689" width="24" height="24"><path d="M877.44 383.786667L624.426667 117.333333C594.986667 86.186667 554.88 69.12 512 69.12s-82.986667 17.066667-112.426667 48.213333L146.56 383.786667a148.266667 148.266667 0 0 0-40.746667 102.4v302.08c0 85.76 69.76 155.52 155.52 155.52h501.546667c85.76 0 155.52-69.76 155.52-155.52V485.973333c0-38.186667-14.506667-74.453333-40.96-102.186666z m-44.373333 404.266666c0 38.826667-31.573333 70.186667-70.186667 70.186667H261.333333c-38.826667 0-70.186667-31.573333-70.186666-70.186667V485.973333c0-16.213333 6.186667-31.786667 17.28-43.52L461.44 176c13.226667-13.866667 31.146667-21.546667 50.56-21.546667s37.333333 7.68 50.56 21.76l253.013333 266.453334c11.306667 11.733333 17.28 27.306667 17.28 43.52v301.866666z" p-id="1690"></path><path d="M608 687.786667h-192c-23.466667 0-42.666667 19.2-42.666667 42.666666s19.2 42.666667 42.666667 42.666667h192c23.466667 0 42.666667-19.2 42.666667-42.666667s-19.2-42.666667-42.666667-42.666666z" p-id="1691"></path></svg>`,
isMainKnowledge: `<svg t="1651124352868" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1850" width="24" height="24"><path d="M877.44 388.053333L624.426667 121.813333C594.986667 90.666667 554.88 73.386667 512 73.386667s-82.986667 17.066667-112.426667 48.213333L146.56 388.053333a148.266667 148.266667 0 0 0-40.746667 102.4v302.08c0 85.76 69.76 155.52 155.52 155.52h501.546667c85.76 0 155.52-69.76 155.52-155.52V490.453333c0-38.4-14.506667-74.666667-40.96-102.4zM608 777.386667h-192c-23.466667 0-42.666667-19.2-42.666667-42.666667s19.2-42.666667 42.666667-42.666667h192c23.466667 0 42.666667 19.2 42.666667 42.666667s-19.2 42.666667-42.666667 42.666667z" p-id="1851"></path></svg>`,
export: `<svg t="1651322116327" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="11894" width="24" height="24"><path d="M849.2 599v217H178.5V599c-0.7-23.7-20.1-42.7-44-42.7s-43.3 19-44 42.7v252.5c0 28.9 23.6 52.5 52.5 52.5h741.7c28.9 0 52.5-23.6 52.5-52.5V599c-0.7-23.7-20.1-42.7-44-42.7s-43.3 19-44 42.7z" p-id="11895"></path><path d="M482.7 135.4l-164 164c-17.1 17.1-17.1 45.1 0 62.2s45.1 17.1 62.2 0l85.7-85.7v314.8c0 26 21.3 47.2 47.2 47.2 26 0 47.2-21.3 47.2-47.2V276l85.7 85.7c17.1 17.1 45.1 17.1 62.2 0s17.1-45.1 0-62.2l-164-164c-17.1-17.2-45.1-17.2-62.2-0.1z" p-id="11896"></path></svg>`,
openWorkspaceCollectionView: `<svg t="1651317033804" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="2432" width="100%" height="100%"><path d="M874.9 459.4c-18.8 0-34 15.2-34 34v355.7c0 18.6-15.5 33.7-34.5 33.7H181.5c-19 0-34.5-15.1-34.5-33.7V232.3c0-18.6 15.5-33.7 34.5-33.7H541c18.8 0 34-15.2 34-34s-15.2-34-34-34H181.5C125 130.6 79 176.2 79 232.3v616.8c0 56 46 101.7 102.5 101.7h624.9c56.5 0 102.5-45.6 102.5-101.7V493.4c0-18.8-15.2-34-34-34z" fill="#b6b6b6" p-id="2433"></path><path d="M885.5 82.7H657.1c-18.8 0-34 15.2-34 34s15.2 34 34 34h169.7L358.5 619.1c-13.3 13.3-13.3 34.8 0 48.1 6.6 6.6 15.3 10 24 10s17.4-3.3 24-10l470-470v169.7c0 18.8 15.2 34 34 34s34-15.2 34-34V141.5c0.1-32.4-26.4-58.8-59-58.8z" fill="#b6b6b6" p-id="2434"></path></svg>`,
};
}
@ -24,19 +25,46 @@ class AddonViews extends AddonBase {
return editor;
}
async addEditorKnowledgeToolBar(editorInstances: EditorInstance) {
await editorInstances._initPromise;
const _document = editorInstances._iframeWindow.document;
const knowledgeToolBar = _document.createElement("div");
knowledgeToolBar.setAttribute("id", "knowledge-tools");
knowledgeToolBar.setAttribute("class", "toolbar");
const start = _document.createElement("div");
start.setAttribute("id", "knowledge-tools-start");
start.setAttribute("class", "start");
const middle = _document.createElement("div");
middle.setAttribute("id", "knowledge-tools-middle");
middle.setAttribute("class", "middle");
const end = _document.createElement("div");
end.setAttribute("id", "knowledge-tools-end");
end.setAttribute("class", "end");
knowledgeToolBar.append(start, middle, end);
_document
.getElementsByClassName("editor")[0]
.childNodes[0].before(knowledgeToolBar);
}
async addEditorButton(
editorInstances: EditorInstance,
id: string,
icon: string,
title: string,
position: "start" | "middle" | "end",
message: EditorMessage
) {
// Use Zotero.Notes._editorInstances to find current opened note editor
await editorInstances._initPromise;
let _document = editorInstances._iframeWindow.document;
let toolbar = _document.getElementsByClassName("middle")[0];
let button = _document.createElement("button");
const _document = editorInstances._iframeWindow.document;
let knowledgeToolBar = _document.getElementById("knowledge-tools");
if (!knowledgeToolBar) {
await this.addEditorKnowledgeToolBar(editorInstances);
}
const toolbar = _document.getElementById(`knowledge-tools-${position}`);
const button = _document.createElement("button");
button.setAttribute("class", "toolbar-button");
button.setAttribute("id", id);
button.setAttribute("title", title);
@ -49,9 +77,15 @@ class AddonViews extends AddonBase {
};
}
changeEditorButton(button: Element, icon: string, title: string) {
changeEditorButtonView(button: Element, icon: string, title: string = "") {
button.innerHTML = this.editorIcon[icon];
button.setAttribute("title", title);
if (title) {
button.setAttribute("title", title);
}
}
changeEditorButtonHidden(button: XUL.Element, hidden: boolean) {
button.hidden = hidden;
}
async scrollToLine(instance: EditorInstance, lineIndex: number) {
@ -90,7 +124,7 @@ class AddonViews extends AddonBase {
span1.append(span2, span3, span4);
treeRow.append(span1);
treeRow.addEventListener("click", (e) => {
this._Addon.knowledge.openWorkspaceWindow();
this._Addon.events.onEditorEvent(new EditorMessage("openWorkspace", {}));
});
treeRow.addEventListener("mouseover", (e: XULEvent) => {
treeRow.setAttribute(

View File

@ -1283,41 +1283,12 @@ function ZoteroProtocolHandler() {
let message = {
type: "onNoteLink",
content: {
params: {
item: false,
infoText: "",
},
params: await Zotero.Knowledge4Zotero.knowledge.getNoteFromLink(
uri.spec
),
},
};
let [groupID, noteKey] = uri.spec
.substring("zotero://note/".length)
.split("/");
// User libraryID by defaultx
let libraryID = 1;
if (groupID !== "u") {
// Not a user item
groupID = parseInt(groupID);
libraryID = Zotero.Groups.getLibraryIDFromGroupID(groupID);
}
if (!libraryID) {
message.content.params.infoText =
"Library does not exist or access denied.";
} else {
let item = await Zotero.Items.getByLibraryAndKeyAsync(
libraryID,
noteKey
);
if (!item || !item.isNote()) {
message.content.params.infoText =
"Note does not exist or is not a note.";
} else {
message.content.params.item = item;
}
}
Zotero.debug(`Note link ${libraryID} : ${noteKey} called.`);
Zotero.Knowledge4Zotero.events.onEditorEvent.bind(
Zotero.Knowledge4Zotero.events
)(message);

5
typing/global.d.ts vendored
View File

@ -149,6 +149,11 @@ declare class ZoteroCollection {
getChildItems: (arg1: boolean, arg2: boolean) => Array<ZoteroItem>;
}
declare class Zotero_File_Exporter {
items: ZoteroItem[];
save: () => void;
}
declare const Components: any;
declare const Services: any;