/g, '')
+ .replace(/<\/m:center>/g, '
')
+ //
+ .replace(//g, '')
+ .replace(/<\/m:underline>/g, '');
+};
+
+Zotero_Import_Mendeley.prototype._saveItems = async function (libraryID, json) {
+ var idMap = new Map();
+ await Zotero.DB.executeTransaction(async function () {
+ for (let itemJSON of json) {
+ let item = new Zotero.Item;
+ item.libraryID = libraryID;
+ if (itemJSON.key) {
+ item.key = itemJSON.key;
+ await item.loadPrimaryData();
+ }
+
+ // Remove external id before save
+ let toSave = Object.assign({}, itemJSON);
+ delete toSave.documentID;
+
+ item.fromJSON(toSave);
+ await item.save({
+ skipSelect: true,
+ skipDateModifiedUpdate: true
+ });
+ if (itemJSON.documentID) {
+ idMap.set(itemJSON.documentID, item.id);
+ }
+ }
+ }.bind(this));
+ return idMap;
+};
+
+/**
+ * Saves attachments and extracted annotations for a given document
+ */
+Zotero_Import_Mendeley.prototype._saveFilesAndAnnotations = async function (files, libraryID, parentItemID, annotations) {
+ for (let file of files) {
+ try {
+ if (!file.fileURL) continue;
+
+ let path = OS.Path.fromFileURI(file.fileURL);
+
+ let attachment;
+ if (await OS.File.exists(path)) {
+ let options = {
+ libraryID,
+ parentItemID,
+ file: path
+ };
+ // If file is in Mendeley downloads folder, import it
+ if (OS.Path.dirname(path).endsWith(OS.Path.join('Mendeley Desktop', 'Downloaded'))) {
+ attachment = await Zotero.Attachments.importFromFile(options);
+ }
+ // Otherwise link it
+ else {
+ attachment = await Zotero.Attachments.linkFromFile(options);
+ }
+ attachment.relations = {
+ 'mendeleyDB:fileHash': file.hash,
+ 'mendeleyDB:fileUUID': file.uuid
+ };
+ await attachment.saveTx({
+ skipSelect: true
+ });
+ }
+ else {
+ Zotero.warn(path + " not found -- not importing");
+ }
+
+ if (annotations) {
+ await this._saveAnnotations(
+ // We have annotations from all files for this document, so limit to just those on
+ // this file
+ annotations.filter(a => a.hash == file.hash),
+ parentItemID,
+ attachment ? attachment.id : null
+ );
+ }
+ }
+ catch (e) {
+ Zotero.logError(e);
+ }
+ }
+}
+
+Zotero_Import_Mendeley.prototype._saveAnnotations = async function (annotations, parentItemID, attachmentItemID) {
+ if (!annotations.length) return;
+ var noteStrings = [];
+ var parentItem = Zotero.Items.get(parentItemID);
+ var libraryID = parentItem.libraryID;
+ if (attachmentItemID) {
+ var attachmentItem = Zotero.Items.get(attachmentItemID);
+ var attachmentURIPath = Zotero.API.getLibraryPrefix(libraryID) + '/items/' + attachmentItem.key;
+ }
+
+ for (let annotation of annotations) {
+ if (!annotation.note || !annotation.note.trim()) continue;
+
+ let linkStr;
+ let linkText = `note on p. ${annotation.page}`;
+ if (attachmentItem) {
+ let url = `zotero://open-pdf/${attachmentURIPath}?page=${annotation.page}`;
+ linkStr = `${linkText}`;
+ }
+ else {
+ linkStr = linkText;
+ }
+
+ noteStrings.push(
+ Zotero.Utilities.text2html(annotation.note.trim())
+ + `(${linkStr})
`
+ );
+ }
+
+ if (!noteStrings.length) return;
+
+ let note = new Zotero.Item('note');
+ note.libraryID = libraryID;
+ note.parentItemID = parentItemID;
+ note.setNote('' + Zotero.getString('extractedAnnotations') + '
\n' + noteStrings.join('\n'));
+ return note.saveTx({
+ skipSelect: true
+ });
+};
diff --git a/chrome/content/zotero/import/mendeley/mendeleySchemaMap.js b/chrome/content/zotero/import/mendeley/mendeleySchemaMap.js
new file mode 100644
index 000000000..af8e929e3
--- /dev/null
+++ b/chrome/content/zotero/import/mendeley/mendeleySchemaMap.js
@@ -0,0 +1,102 @@
+var map = {
+ 83: {
+ itemTypes: {
+ Bill: "bill",
+ Book: "book",
+ BookSection: "bookSection",
+ Case: "case",
+ ComputerProgram: "computerProgram",
+ ConferenceProceedings: "conferencePaper",
+ EncyclopediaArticle: "encyclopediaArticle",
+ Film: "film",
+ Generic: "document",
+ JournalArticle: "journalArticle",
+ MagazineArticle: "magazineArticle",
+ NewspaperArticle: "newspaperArticle",
+ Patent: "patent",
+ Report: "report",
+ Statute: "statute",
+ TelevisionBroadcast: "tvBroadcast",
+ Thesis: "thesis",
+ WebPage: "webpage",
+ WorkingPaper: "report"
+ },
+ fields: {
+ id: "",
+ uuid: "",
+ reviewedArticle: "",
+ revisionNumber: "",
+ publisher: "publisher",
+ reprintEdition: "",
+ series: "seriesTitle",
+ seriesNumber: "seriesNumber",
+ sections: "section",
+ seriesEditor: "creator[seriesEditor]", // falls back to editor if necessary
+ owner: "",
+ pages: "func[pages]",
+ month: "", // handled explicitly
+ originalPublication: "",
+ publication: "publicationTitle",
+ publicLawNumber: "publicLawNumber",
+ pmid: "extra[PMID]",
+ sourceType: "",
+ session: "session",
+ shortTitle: "shortTitle",
+ volume: "volume",
+ year: "", // handled explicitly
+ userType: "type",
+ country: "place[country]",
+ dateAccessed: "accessDate",
+ committee: "committee",
+ counsel: "creator[counsel]",
+ doi: "DOI",
+ edition: "edition",
+ day: "", // handled explicitly
+ department: "",
+ citationKey: "citationKey", // put in Extra
+ city: "place[city]",
+ chapter: "",
+ codeSection: "section",
+ codeVolume: "codeVolume",
+ code: "code",
+ codeNumber: "codeNumber",
+ issue: "issue",
+ language: "language",
+ isbn: "ISBN",
+ issn: "ISSN",
+ length: "",
+ medium: "medium",
+ lastUpdate: "",
+ legalStatus: "legalStatus",
+ hideFromMendeleyWebIndex: "",
+ institution: "publisher",
+ genre: "genre",
+ internationalTitle: "",
+ internationalUserType: "",
+ internationalAuthor: "",
+ internationalNumber: "",
+ deletionPending: "",
+ favourite: "", // tag?
+ confirmed: "", // tag?
+ deduplicated: "",
+ read: "", // tag?
+ type: "", // item type handled separately
+ title: "title",
+ privacy: "",
+ applicationNumber: "applicationNumber",
+ arxivId: "extra[arXiv]",
+ advisor: "",
+ articleColumn: "",
+ modified: "func[fromUnixtime:dateModified]",
+ abstract: "abstractNote",
+ added: "func[fromUnixtime:dateAdded]",
+ note: "func[note]",
+ importer: ""
+ },
+ creatorTypes: {
+ DocumentAuthor: "author",
+ DocumentEditor: "editor",
+ DocumentTranslator: "translator"
+ }
+ }
+};
diff --git a/chrome/content/zotero/xpcom/data/collection.js b/chrome/content/zotero/xpcom/data/collection.js
index ca81a6f78..8d263aae3 100644
--- a/chrome/content/zotero/xpcom/data/collection.js
+++ b/chrome/content/zotero/xpcom/data/collection.js
@@ -712,7 +712,7 @@ Zotero.Collection.prototype.toJSON = function (options = {}) {
obj.name = this.name;
obj.parentCollection = this.parentKey ? this.parentKey : false;
- obj.relations = {}; // TEMP
+ obj.relations = this.getRelations();
return this._postToJSON(env);
}
diff --git a/chrome/content/zotero/xpcom/data/dataObject.js b/chrome/content/zotero/xpcom/data/dataObject.js
index 342aeede2..4d2d29f34 100644
--- a/chrome/content/zotero/xpcom/data/dataObject.js
+++ b/chrome/content/zotero/xpcom/data/dataObject.js
@@ -285,7 +285,7 @@ Zotero.DataObject.prototype._setParentKey = function(key) {
/**
* Returns all relations of the object
*
- * @return {Object} - Object with predicates as keys and arrays of URIs as values
+ * @return {Object} - Object with predicates as keys and arrays of values
*/
Zotero.DataObject.prototype.getRelations = function () {
this._requireData('relations');
@@ -410,7 +410,7 @@ Zotero.DataObject.prototype.setRelations = function (newRelations) {
// Limit predicates to letters and colons for now
for (let p in newRelations) {
- if (!/[a-z]+:[a-z]+/.test(p)) {
+ if (!/^[a-z]+:[a-z]+$/i.test(p)) {
throw new Error(`Invalid relation predicate '${p}'`);
}
}
diff --git a/chrome/content/zotero/xpcom/data/relations.js b/chrome/content/zotero/xpcom/data/relations.js
index 3b259fda8..66c6dcbd0 100644
--- a/chrome/content/zotero/xpcom/data/relations.js
+++ b/chrome/content/zotero/xpcom/data/relations.js
@@ -32,7 +32,8 @@ Zotero.Relations = new function () {
this._namespaces = {
dc: 'http://purl.org/dc/elements/1.1/',
- owl: 'http://www.w3.org/2002/07/owl#'
+ owl: 'http://www.w3.org/2002/07/owl#',
+ mendeleyDB: 'http://zotero.org/namespaces/mendeleyDB#'
};
var _types = ['collection', 'item'];
diff --git a/chrome/content/zotero/xpcom/mime.js b/chrome/content/zotero/xpcom/mime.js
index 15859e940..66b87cce2 100644
--- a/chrome/content/zotero/xpcom/mime.js
+++ b/chrome/content/zotero/xpcom/mime.js
@@ -47,8 +47,9 @@ Zotero.MIME = new function(){
["\uFFFDPNG", 'image/png', 0],
["JFIF", 'image/jpeg'],
["FLV", "video/x-flv", 0],
- ["\u0000\u0000\u0001\u0000", "image/vnd.microsoft.icon", 0]
-
+ ["\u0000\u0000\u0001\u0000", "image/vnd.microsoft.icon", 0],
+ ["\u0053\u0051\u004C\u0069\u0074\u0065\u0020\u0066"
+ + "\u006F\u0072\u006D\u0061\u0074\u0020\u0033\u0000", "application/x-sqlite3", 0]
];
var _extensions = {
diff --git a/chrome/content/zotero/zoteroPane.xul b/chrome/content/zotero/zoteroPane.xul
index 1d5817c32..af44a9dae 100644
--- a/chrome/content/zotero/zoteroPane.xul
+++ b/chrome/content/zotero/zoteroPane.xul
@@ -49,7 +49,7 @@
-
+
diff --git a/chrome/locale/en-US/zotero/zotero.dtd b/chrome/locale/en-US/zotero/zotero.dtd
index dd99f00cf..e16c2cb74 100644
--- a/chrome/locale/en-US/zotero/zotero.dtd
+++ b/chrome/locale/en-US/zotero/zotero.dtd
@@ -202,6 +202,14 @@
+
+
+
+
+
+
+
+
diff --git a/chrome/locale/en-US/zotero/zotero.properties b/chrome/locale/en-US/zotero/zotero.properties
index bfbe3ec44..57656b555 100644
--- a/chrome/locale/en-US/zotero/zotero.properties
+++ b/chrome/locale/en-US/zotero/zotero.properties
@@ -69,6 +69,7 @@ general.processing = Processing
general.submitted = Submitted
general.thanksForHelpingImprove = Thanks for helping to improve %S!
general.describeProblem = Briefly describe the problem:
+general.nMegabytes = %S MB
general.operationInProgress = A Zotero operation is currently in progress.
general.operationInProgress.waitUntilFinished = Please wait until it has finished.
@@ -688,10 +689,12 @@ fileInterface.importComplete = Import Complete
fileInterface.itemsWereImported = %1$S item was imported;%1$S items were imported
fileInterface.itemsExported = Exporting items…
fileInterface.import = Import
+fileInterface.chooseAppDatabaseToImport = Choose the %S database to import
fileInterface.export = Export
fileInterface.exportedItems = Exported Items
fileInterface.imported = Imported
fileInterface.unsupportedFormat = The selected file is not in a supported format.
+fileInterface.appDatabase = %S Database
fileInterface.viewSupportedFormats = View Supported Formats…
fileInterface.untitledBibliography = Untitled Bibliography
fileInterface.bibliographyHTMLTitle = Bibliography
@@ -1075,6 +1078,7 @@ rtfScan.rtf = Rich Text Format (.rtf)
rtfScan.saveTitle = Select a location in which to save the formatted file
rtfScan.scannedFileSuffix = (Scanned)
+extractedAnnotations = Extracted Annotations
file.accessError.theFileCannotBeCreated = The file '%S' cannot be created.
file.accessError.theFileCannotBeUpdated = The file '%S' cannot be updated.
diff --git a/chrome/skin/default/zotero/importWizard.css b/chrome/skin/default/zotero/importWizard.css
new file mode 100644
index 000000000..308fe0d42
--- /dev/null
+++ b/chrome/skin/default/zotero/importWizard.css
@@ -0,0 +1,44 @@
+.wizard-header-label {
+ font-size: 16px;
+ font-weight: bold;
+}
+
+/* Start */
+wizard[currentpageid="page-start"] .wizard-header-label {
+ padding-top: 24px;
+}
+
+wizard[currentpageid="page-start"] .wizard-page-box {
+ margin-top: -2px;
+ padding-top: 0;
+}
+
+radiogroup {
+ font-size: 14px;
+ margin-top: 4px;
+}
+
+radio {
+ padding-top: 5px;
+}
+
+/* File options */
+wizard[currentpageid="page-file-options"] .wizard-header {
+ display: none;
+}
+
+#file-options-header {
+ font-size: 15px;
+ font-weight: bold;
+ margin-bottom: 6px;
+}
+
+listbox, #result-description {
+ font-size: 13px;
+}
+
+#result-report-error {
+ margin-top: 13px;
+ margin-left: 0;
+ font-size: 13px;
+}