2010-07-06 09:02:35 +00:00
|
|
|
/*
|
|
|
|
***** BEGIN LICENSE BLOCK *****
|
|
|
|
|
2010-12-26 19:05:52 +00:00
|
|
|
Copyright © 2009 Center for History and New Media
|
2010-07-06 09:02:35 +00:00
|
|
|
George Mason University, Fairfax, Virginia, USA
|
|
|
|
http://zotero.org
|
|
|
|
|
|
|
|
This file is part of Zotero.
|
|
|
|
|
|
|
|
Zotero is free software: you can redistribute it and/or modify
|
2011-05-18 18:34:22 +00:00
|
|
|
it under the terms of the GNU Affero General Public License as published by
|
2010-07-06 09:02:35 +00:00
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
Zotero is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2011-05-18 18:34:22 +00:00
|
|
|
GNU Affero General Public License for more details.
|
2010-07-06 09:02:35 +00:00
|
|
|
|
2011-05-18 18:34:22 +00:00
|
|
|
You should have received a copy of the GNU Affero General Public License
|
2010-07-06 09:02:35 +00:00
|
|
|
along with Zotero. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
|
|
|
|
Based on nsChromeExtensionHandler example code by Ed Anuff at
|
|
|
|
http://kb.mozillazine.org/Dev_:_Extending_the_Chrome_Protocol
|
|
|
|
|
|
|
|
***** END LICENSE BLOCK *****
|
|
|
|
*/
|
|
|
|
|
2006-05-27 00:20:27 +00:00
|
|
|
const Cc = Components.classes;
|
|
|
|
const Ci = Components.interfaces;
|
|
|
|
|
2011-06-14 00:36:21 +00:00
|
|
|
/** XPCOM files to be loaded for all modes **/
|
|
|
|
const xpcomFilesAll = [
|
2008-08-13 06:38:47 +00:00
|
|
|
'zotero',
|
2016-11-27 03:41:26 +00:00
|
|
|
'dataDirectory',
|
2011-06-14 00:36:21 +00:00
|
|
|
'date',
|
|
|
|
'debug',
|
|
|
|
'error',
|
|
|
|
'file',
|
|
|
|
'http',
|
|
|
|
'mimeTypeHandler',
|
|
|
|
'openurl',
|
|
|
|
'ipc',
|
2016-11-27 03:41:26 +00:00
|
|
|
'profile',
|
2011-06-14 00:36:21 +00:00
|
|
|
'progressWindow',
|
2016-12-12 12:29:59 +00:00
|
|
|
'proxy',
|
2011-06-14 00:36:21 +00:00
|
|
|
'translation/translate',
|
|
|
|
'translation/translate_firefox',
|
2013-08-17 07:08:59 +00:00
|
|
|
'translation/translator',
|
2011-06-14 00:36:21 +00:00
|
|
|
'translation/tlds',
|
2011-08-31 23:25:48 +00:00
|
|
|
'utilities',
|
2015-02-19 08:25:15 +00:00
|
|
|
'isbn',
|
2011-08-31 23:25:48 +00:00
|
|
|
'utilities_internal',
|
|
|
|
'utilities_translate'
|
2011-06-14 00:36:21 +00:00
|
|
|
];
|
|
|
|
|
|
|
|
/** XPCOM files to be loaded only for local translation and DB access **/
|
|
|
|
const xpcomFilesLocal = [
|
2013-09-05 17:37:09 +00:00
|
|
|
'libraryTreeView',
|
2011-06-14 00:36:21 +00:00
|
|
|
'collectionTreeView',
|
2015-11-03 22:06:18 +00:00
|
|
|
'collectionTreeRow',
|
2008-08-13 06:38:47 +00:00
|
|
|
'annotate',
|
Update zotero:// extensions (report, timeline, etc.) for async DB, and more
- Protocol handler extensions can now handle promises and can also make
data available as it's ready instead of all at once (e.g., reports now
output one entry at a time)
- zotero:// URL syntaxes are now more consistent and closer to the web
API (old URLs should work, but some may currently be broken)
Also:
- Code to generate server API, currently available for testing via
zotero://data URLs but eventually moving to HTTP -- zotero://data URLs match
web API URLs, with a different prefix for the personal library (/library vs.
/users/12345)
- Miscellaneous fixes to data objects
Under the hood:
- Extensions now return an AsyncChannel, which is an nsIChannel implementation
that takes a promise-yielding generator that returns a string,
nsIAsyncInputStream, or file that will be used for the channel's data
- New function Zotero.Utilities.Internal.getAsyncInputStream() takes a
generator that yields either promises or strings and returns an async input
stream filled with the yielded strings
- Zotero.Router parsers URLs and extract parameters
- Zotero.Item.toResponseJSON()
2014-09-08 20:51:05 +00:00
|
|
|
'api',
|
2008-08-13 06:38:47 +00:00
|
|
|
'attachments',
|
2010-05-30 11:25:25 +00:00
|
|
|
'cite',
|
2011-07-19 03:52:02 +00:00
|
|
|
'cookieSandbox',
|
2015-06-02 04:29:40 +00:00
|
|
|
'data/library',
|
|
|
|
'data/libraries',
|
Async DB megacommit
Promise-based rewrite of most of the codebase, with asynchronous database and file access -- see https://github.com/zotero/zotero/issues/518 for details.
WARNING: This includes backwards-incompatible schema changes.
An incomplete list of other changes:
- Schema overhaul
- Replace main tables with new versions with updated schema
- Enable real foreign key support and remove previous triggers
- Don't use NULLs for local libraryID, which broke the UNIQUE index
preventing object key duplication. All code (Zotero and third-party)
using NULL for the local library will need to be updated to use 0
instead (already done for Zotero code)
- Add 'compatibility' DB version that can be incremented manually to break DB
compatibility with previous versions. 'userdata' upgrades will no longer
automatically break compatibility.
- Demote creators and tags from first-class objects to item properties
- New API syncing properties
- 'synced'/'version' properties to data objects
- 'etag' to groups
- 'version' to libraries
- Create Zotero.DataObject that other objects inherit from
- Consolidate data object loading into Zotero.DataObjects
- Change object reloading so that only the loaded and changed parts of objects are reloaded, instead of reloading all data from the database (with some exceptions, including item primary data)
- Items and collections now have .parentItem and .parentKey properties, replacing item.getSource() and item.getSourceKey()
- New function Zotero.serial(fn), to wrap an async function such that all calls are run serially
- New function Zotero.Utilities.Internal.forEachChunkAsync(arr, chunkSize, func)
- Add tag selector loading message
- Various API and name changes, since everything was breaking anyway
Known broken things:
- Syncing (will be completely rewritten for API syncing)
- Translation architecture (needs promise-based rewrite)
- Duplicates view
- DB integrity check (from schema changes)
- Dragging (may be difficult to fix)
Lots of other big and little things are certainly broken, particularly with the UI, which can be affected by async code in all sorts of subtle ways.
2014-08-06 21:38:05 +00:00
|
|
|
'data/dataObject',
|
2008-08-13 06:38:47 +00:00
|
|
|
'data/dataObjects',
|
2014-08-06 21:38:04 +00:00
|
|
|
'data/dataObjectUtilities',
|
2008-08-13 06:38:47 +00:00
|
|
|
'data/cachedTypes',
|
Async DB megacommit
Promise-based rewrite of most of the codebase, with asynchronous database and file access -- see https://github.com/zotero/zotero/issues/518 for details.
WARNING: This includes backwards-incompatible schema changes.
An incomplete list of other changes:
- Schema overhaul
- Replace main tables with new versions with updated schema
- Enable real foreign key support and remove previous triggers
- Don't use NULLs for local libraryID, which broke the UNIQUE index
preventing object key duplication. All code (Zotero and third-party)
using NULL for the local library will need to be updated to use 0
instead (already done for Zotero code)
- Add 'compatibility' DB version that can be incremented manually to break DB
compatibility with previous versions. 'userdata' upgrades will no longer
automatically break compatibility.
- Demote creators and tags from first-class objects to item properties
- New API syncing properties
- 'synced'/'version' properties to data objects
- 'etag' to groups
- 'version' to libraries
- Create Zotero.DataObject that other objects inherit from
- Consolidate data object loading into Zotero.DataObjects
- Change object reloading so that only the loaded and changed parts of objects are reloaded, instead of reloading all data from the database (with some exceptions, including item primary data)
- Items and collections now have .parentItem and .parentKey properties, replacing item.getSource() and item.getSourceKey()
- New function Zotero.serial(fn), to wrap an async function such that all calls are run serially
- New function Zotero.Utilities.Internal.forEachChunkAsync(arr, chunkSize, func)
- Add tag selector loading message
- Various API and name changes, since everything was breaking anyway
Known broken things:
- Syncing (will be completely rewritten for API syncing)
- Translation architecture (needs promise-based rewrite)
- Duplicates view
- DB integrity check (from schema changes)
- Dragging (may be difficult to fix)
Lots of other big and little things are certainly broken, particularly with the UI, which can be affected by async code in all sorts of subtle ways.
2014-08-06 21:38:05 +00:00
|
|
|
'data/notes',
|
2008-08-13 06:38:47 +00:00
|
|
|
'data/item',
|
|
|
|
'data/items',
|
|
|
|
'data/collection',
|
|
|
|
'data/collections',
|
2015-06-02 04:29:40 +00:00
|
|
|
'data/feedItem',
|
|
|
|
'data/feedItems',
|
|
|
|
'data/feed',
|
|
|
|
'data/feeds',
|
2008-08-13 06:38:47 +00:00
|
|
|
'data/creators',
|
2.0b3 megacommit
- Support for group libraries
- General support for multiple libraries of different types
- Streamlined sync support
- Using solely libraryID and key rather than itemID, and removed all itemID-changing code
- Combined two requests for increased performance and decreased server load
- Added warning on user account change
- Provide explicit error message on SSL failure
- Removed snapshot and link toolbar buttons and changed browser context menu options and drags to create parent items + snapshots
- Closes #786, Add numPages field
- Fixes #1063, Duplicate item with tags broken in Sync Preview
- Added better purging of deleted tags
- Added local user key before first sync
- Add clientDateModified to all objects for more flexibility in syncing
- Added new triples-based Relation object type, currently used to store links between items copied between local and group libraries
- Updated zotero.org translator for groups
- Additional trigger-based consistency checks
- Fixed broken URL drag in Firefox 3.5
- Disabled zeroconf menu option (no longer functional)
Developer-specific changes:
- Overhauled data layer
- Data object constructors no longer take arguments (return to 1.0-like API)
- Existing objects can be retrieved by setting id or library/key properties
- id/library/key must be set for new objects before other fields
- New methods:
- ZoteroPane.getSelectedLibraryID()
- ZoteroPane.getSelectedGroup(asID)
- ZoteroPane.addItemFromDocument(doc, itemType, saveSnapshot)
- ZoteroPane.addItemFromURL(url, itemType)
- ZoteroPane.canEdit()
- Zotero.CollectionTreeView.selectLibrary(libraryID)
- New Zotero.URI methods
- Changed methods
- Many data object methods now take a libraryID
- ZoteroPane.addAttachmentFromPage(link, itemID)
- Removed saveItem and saveAttachments parameters from Zotero.Translate constructor
- translate() now takes a libraryID, null for local library, or false to not save items (previously on constructor)
- saveAttachments is now a translate() parameter
- Zotero.flattenArguments() better handles passed objects
- Zotero.File.getFileHash() (not currently used)
2009-05-14 18:23:40 +00:00
|
|
|
'data/group',
|
|
|
|
'data/groups',
|
2008-08-13 06:38:47 +00:00
|
|
|
'data/itemFields',
|
2.0b3 megacommit
- Support for group libraries
- General support for multiple libraries of different types
- Streamlined sync support
- Using solely libraryID and key rather than itemID, and removed all itemID-changing code
- Combined two requests for increased performance and decreased server load
- Added warning on user account change
- Provide explicit error message on SSL failure
- Removed snapshot and link toolbar buttons and changed browser context menu options and drags to create parent items + snapshots
- Closes #786, Add numPages field
- Fixes #1063, Duplicate item with tags broken in Sync Preview
- Added better purging of deleted tags
- Added local user key before first sync
- Add clientDateModified to all objects for more flexibility in syncing
- Added new triples-based Relation object type, currently used to store links between items copied between local and group libraries
- Updated zotero.org translator for groups
- Additional trigger-based consistency checks
- Fixed broken URL drag in Firefox 3.5
- Disabled zeroconf menu option (no longer functional)
Developer-specific changes:
- Overhauled data layer
- Data object constructors no longer take arguments (return to 1.0-like API)
- Existing objects can be retrieved by setting id or library/key properties
- id/library/key must be set for new objects before other fields
- New methods:
- ZoteroPane.getSelectedLibraryID()
- ZoteroPane.getSelectedGroup(asID)
- ZoteroPane.addItemFromDocument(doc, itemType, saveSnapshot)
- ZoteroPane.addItemFromURL(url, itemType)
- ZoteroPane.canEdit()
- Zotero.CollectionTreeView.selectLibrary(libraryID)
- New Zotero.URI methods
- Changed methods
- Many data object methods now take a libraryID
- ZoteroPane.addAttachmentFromPage(link, itemID)
- Removed saveItem and saveAttachments parameters from Zotero.Translate constructor
- translate() now takes a libraryID, null for local library, or false to not save items (previously on constructor)
- saveAttachments is now a translate() parameter
- Zotero.flattenArguments() better handles passed objects
- Zotero.File.getFileHash() (not currently used)
2009-05-14 18:23:40 +00:00
|
|
|
'data/relations',
|
2016-10-05 03:44:07 +00:00
|
|
|
'data/search',
|
|
|
|
'data/searchConditions',
|
|
|
|
'data/searches',
|
2008-08-13 06:38:47 +00:00
|
|
|
'data/tags',
|
|
|
|
'db',
|
Duplicate detection:
- Adds a per-library "Duplicate Items" virtual search to the source list -- shows up by default for "My Library" but can be added to and removed from all libraries
- Current matching algorithm is very basic: finds exact title matches (after normalizing case/diacritics/punctuation/spacing) and DOI/ISBN matches (untested)
- In duplicates view, sets are selected automatically; in other views, duplicate items can be selected manually and the merge interface can be brought up with "Merge Items" in the context menu
- Can select a master item and individual fields to merge from other versions
- Word processor integration code will automatically find mapped replacements and update documents with new item keys
Possible future improvements:
- Improved detection algorithms
- UI tweaks
- Currently if any items differ, all available versions will be shown as master item options, even if only one item is different; probably the earliest equivalent item should be shown for each distinct version
- Caching of results for performance
- Confidence scale
- Creator version selection (currently the creators from the chosen master item are kept)
- Merging of matching child items
- Better sorting of duplicates if not clustered together by the selected sort column
- Relation path compression when merging items that are already mapped to previously removed duplicates
Other changes in this commit:
- Don't show Trash in word processor integration windows
- Consider items in trash to be missing in word processor documents
- Selection of special views (Trash, Unfiled, Duplicates) is now restored properly in new windows
- Disabled field transform context menu when item isn't editable
- Left/right arrow now expands/collapses all selected items instead of just the last-selected row
- Relation deletions are now synced
- The same items row is now reselected after item deletion
- (dev) Zotero.Item.getNotes(), Zotero.Item.getAttachments(), and Zotero.Item.getTags() now return empty arrays rather than FALSE if no matches -- tests on those return values in third-party code will need to be changed
- (dev) New function Zotero.Utilities.removeDiacritics(str, lowercaseOnly) -- could be used to generate ASCII BibTeX keys
- (dev) New 'tempTable' search condition can take a table to join against -- useful for implementing virtual source lists
- (dev) Significant UI code cleanup
- (dev) Moved all item pane content into itemPane.xul
- Probably various other things
Needless to say, this needs testing.
2011-07-22 21:24:38 +00:00
|
|
|
'duplicates',
|
2014-11-07 04:08:56 +00:00
|
|
|
'feedReader',
|
2008-08-13 06:38:47 +00:00
|
|
|
'fulltext',
|
|
|
|
'id',
|
|
|
|
'integration',
|
|
|
|
'itemTreeView',
|
2017-07-21 09:30:49 +00:00
|
|
|
'locale',
|
2011-02-09 03:22:06 +00:00
|
|
|
'locateManager',
|
2008-08-13 06:38:47 +00:00
|
|
|
'mime',
|
|
|
|
'notifier',
|
|
|
|
'quickCopy',
|
2018-01-20 08:45:00 +00:00
|
|
|
'recognizePDF',
|
2008-08-13 06:38:47 +00:00
|
|
|
'report',
|
Update zotero:// extensions (report, timeline, etc.) for async DB, and more
- Protocol handler extensions can now handle promises and can also make
data available as it's ready instead of all at once (e.g., reports now
output one entry at a time)
- zotero:// URL syntaxes are now more consistent and closer to the web
API (old URLs should work, but some may currently be broken)
Also:
- Code to generate server API, currently available for testing via
zotero://data URLs but eventually moving to HTTP -- zotero://data URLs match
web API URLs, with a different prefix for the personal library (/library vs.
/users/12345)
- Miscellaneous fixes to data objects
Under the hood:
- Extensions now return an AsyncChannel, which is an nsIChannel implementation
that takes a promise-yielding generator that returns a string,
nsIAsyncInputStream, or file that will be used for the channel's data
- New function Zotero.Utilities.Internal.getAsyncInputStream() takes a
generator that yields either promises or strings and returns an async input
stream filled with the yielded strings
- Zotero.Router parsers URLs and extract parameters
- Zotero.Item.toResponseJSON()
2014-09-08 20:51:05 +00:00
|
|
|
'router',
|
2008-08-13 06:38:47 +00:00
|
|
|
'schema',
|
2011-06-14 00:36:21 +00:00
|
|
|
'server',
|
2017-07-04 22:03:13 +00:00
|
|
|
'streamer',
|
2008-09-16 19:14:52 +00:00
|
|
|
'style',
|
2008-08-13 06:38:47 +00:00
|
|
|
'sync',
|
2015-07-20 21:27:55 +00:00
|
|
|
'sync/syncAPIClient',
|
|
|
|
'sync/syncEngine',
|
2016-05-06 07:08:22 +00:00
|
|
|
'sync/syncExceptions',
|
2015-07-20 21:27:55 +00:00
|
|
|
'sync/syncEventListeners',
|
2015-11-12 07:54:51 +00:00
|
|
|
'sync/syncFullTextEngine',
|
2015-07-20 21:27:55 +00:00
|
|
|
'sync/syncLocal',
|
|
|
|
'sync/syncRunner',
|
|
|
|
'sync/syncUtilities',
|
2008-08-31 23:36:01 +00:00
|
|
|
'storage',
|
2015-10-29 07:41:54 +00:00
|
|
|
'storage/storageEngine',
|
|
|
|
'storage/storageLocal',
|
|
|
|
'storage/storageRequest',
|
|
|
|
'storage/storageResult',
|
|
|
|
'storage/storageUtilities',
|
2011-11-14 08:42:06 +00:00
|
|
|
'storage/streamListener',
|
Zotero File Storage megacommit
- Group file sync via Zotero File Storage
- Split file syncing into separate modules for ZFS and WebDAV
- Dragging items between libraries copies child notes, snapshots/files, and links based on checkboxes for each (enabled by default) in the Zotero preferences
- Sync errors now trigger an exclamation/error icon separate from the sync icon, with a popup window displaying the error and an option to report it
- Various errors that could cause perpetual sync icon spinning now stop the sync properly
- Zotero.Utilities.md5(str) is now md5(strOrFile, base64)
- doPost(), doHead(), and retrieveSource() now takes a headers parameter instead of requestContentType
- doHead() can now accept an nsIURI (with login credentials), is a background request, and isn't cached
- When library access or file writing access is denied during sync, display a warning and then reset local group to server version
- Perform additional steps (e.g., removing local groups) when switching sync users to prevent errors
- Compare hash as well as mod time when checking for modified local files
- Don't trigger notifications when removing groups from the client
- Clear relation links to items in removed groups
- Zotero.Item.attachmentHash property to get file MD5
- importFromFile() now takes libraryID as a third parameter
- Zotero.Attachments.getNumFiles() returns the number of files in the attachment directory
- Zotero.Attachments.copyAttachmentToLibrary() copies an attachment item, including files, to another library
- Removed Zotero.File.getFileHash() in favor of updated Zotero.Utilities.md5()
- Zotero.File.copyDirectory(dir, newDir) copies all files from dir into newDir
- Preferences shuffling: OpenURL to Advanced, import/export character set options to Export, "Include URLs of paper articles in references" to Styles
- Other stuff I don't remember
Suffice it to say, this could use testing.
2009-09-13 07:23:29 +00:00
|
|
|
'storage/zfs',
|
|
|
|
'storage/webdav',
|
2013-03-03 08:38:02 +00:00
|
|
|
'syncedSettings',
|
2008-08-13 06:38:47 +00:00
|
|
|
'timeline',
|
2.0b3 megacommit
- Support for group libraries
- General support for multiple libraries of different types
- Streamlined sync support
- Using solely libraryID and key rather than itemID, and removed all itemID-changing code
- Combined two requests for increased performance and decreased server load
- Added warning on user account change
- Provide explicit error message on SSL failure
- Removed snapshot and link toolbar buttons and changed browser context menu options and drags to create parent items + snapshots
- Closes #786, Add numPages field
- Fixes #1063, Duplicate item with tags broken in Sync Preview
- Added better purging of deleted tags
- Added local user key before first sync
- Add clientDateModified to all objects for more flexibility in syncing
- Added new triples-based Relation object type, currently used to store links between items copied between local and group libraries
- Updated zotero.org translator for groups
- Additional trigger-based consistency checks
- Fixed broken URL drag in Firefox 3.5
- Disabled zeroconf menu option (no longer functional)
Developer-specific changes:
- Overhauled data layer
- Data object constructors no longer take arguments (return to 1.0-like API)
- Existing objects can be retrieved by setting id or library/key properties
- id/library/key must be set for new objects before other fields
- New methods:
- ZoteroPane.getSelectedLibraryID()
- ZoteroPane.getSelectedGroup(asID)
- ZoteroPane.addItemFromDocument(doc, itemType, saveSnapshot)
- ZoteroPane.addItemFromURL(url, itemType)
- ZoteroPane.canEdit()
- Zotero.CollectionTreeView.selectLibrary(libraryID)
- New Zotero.URI methods
- Changed methods
- Many data object methods now take a libraryID
- ZoteroPane.addAttachmentFromPage(link, itemID)
- Removed saveItem and saveAttachments parameters from Zotero.Translate constructor
- translate() now takes a libraryID, null for local library, or false to not save items (previously on constructor)
- saveAttachments is now a translate() parameter
- Zotero.flattenArguments() better handles passed objects
- Zotero.File.getFileHash() (not currently used)
2009-05-14 18:23:40 +00:00
|
|
|
'uri',
|
2014-08-09 22:01:28 +00:00
|
|
|
'users',
|
2011-06-14 00:36:21 +00:00
|
|
|
'translation/translate_item',
|
2013-08-17 07:08:59 +00:00
|
|
|
'translation/translators',
|
2011-06-14 00:36:21 +00:00
|
|
|
'server_connector'
|
2008-08-13 06:38:47 +00:00
|
|
|
];
|
Initial Zotero 1.5 Megacommit
Apologies for the massive (and, due to data_access.js splitting, difficult-to-follow) commit. Please note that external code that accesses the data layer may need to be tweaked for compatibility. Here's a comprehensive-as-possible changelog:
- Added server sync functionality (incomplete)
- Overhaul of data layer
- Split data_access.js into separate files (item.js, items.js, creator.js, etc.)
- Made creators and collections first-class objects, similar to items
- Constructors now take id as first parameter, e.g. new Zotero.Item(1234, 'book'), to allow explicit id setting and id changing
- Made various data layer operations (including attachment fields) require a save() rather than making direct DB changes
- Better handling of unsaved objects
- Item.setCreator() now takes creator objects instead of creator ids, and Item.save() will auto-save unsaved creators
- clone() now works on unsaved objects
- Newly created object instances are now disabled after save() to force refetch of globally accessible instance using Zotero.(Items|Creators|etc.).get()
- Added secondary lookup key to data objects
- Deprecated getID() and getItemType() methods in favor of .id and .itemTypeID properties
- toArray() deprecated in favor of serialize(), which has a somewhat modified format
- Added support for multiple creators with identical data -- currently unimplemented in interface and most of data layer
- Added Item.diff() for comparing item metadata
- Database changes
- Added SQLite triggers to enforce foreign key constraints
- Added Zotero.DB.transactionVacuum flag to run a VACUUM after a transaction
- Added Zotero.DB.transactionDate, .transactionDateTime, and transactionTimestamp to retrieve consistent timestamps for entire transaction
- Properly store 64-bit integers
- Set PRAGMA locking_mode=EXCLUSIVE on database
- Set SQLite page size to 4096 on new databases
- Set SQLite page cache to 8MB
- Do some database cleanup and integrity checking on migration from 1.0 branch
- Removed IF NOT EXISTS from userdata.sql CREATE statements -- userdata.sql is now processed only on DB initialization
- Removed itemNoteTitles table and moved titles into itemNotes
- Abstracted metadata edit box and note box into flexible XBL bindings with various modes, including read-only states
- Massive speed-up of item tree view
- Several fixes from 1.0 branch for Fx3 compatibility
- Added Notifier observer to log delete events for syncing
- Zotero.Utilities changes
- New methods getSQLDataType() and md5()
- Removed onError from Zotero.Utilities.HTTP.doGet()
- Don't display more than 1024 characters in doPost() debug output
- Don't display passwords in doPost() debug output
- Added Zotero.Notifier.untrigger() -- currently unused
- Added Zotero.reloadDataObjects() to reset all in-memory objects
- Added |chars| parameter to Zotero.randomString(len, chars)
- Added Zotero.Date.getUnixTimestamp() and Date.toUnixTimestamp(JSDate)
- Adjusted zotero-service.js to simplify file inclusion
Various things (such as tags) are temporarily broken.
2008-05-04 08:32:48 +00:00
|
|
|
|
2011-06-14 00:36:21 +00:00
|
|
|
/** XPCOM files to be loaded only for connector translation and DB access **/
|
|
|
|
const xpcomFilesConnector = [
|
|
|
|
'connector/translate_item',
|
|
|
|
'connector/translator',
|
|
|
|
'connector/connector',
|
2011-07-20 03:35:21 +00:00
|
|
|
'connector/connector_firefox',
|
2011-06-30 01:08:30 +00:00
|
|
|
'connector/cachedTypes',
|
|
|
|
'connector/repo',
|
|
|
|
'connector/typeSchemaData'
|
2011-06-14 00:36:21 +00:00
|
|
|
];
|
2010-05-30 11:25:25 +00:00
|
|
|
|
2011-06-14 00:36:21 +00:00
|
|
|
Components.utils.import("resource://gre/modules/XPCOMUtils.jsm");
|
2013-11-05 20:52:40 +00:00
|
|
|
Components.utils.import("resource://gre/modules/Services.jsm");
|
2010-05-30 11:25:25 +00:00
|
|
|
|
2011-06-14 00:36:21 +00:00
|
|
|
var instanceID = (new Date()).getTime();
|
|
|
|
var isFirstLoadThisSession = true;
|
|
|
|
var zContext = null;
|
2016-12-16 08:18:29 +00:00
|
|
|
var initCallbacks = [];
|
2013-11-30 06:55:48 +00:00
|
|
|
var zInitOptions = {};
|
2017-05-31 15:28:47 +00:00
|
|
|
Components.utils.import('resource://zotero/require.js');
|
2011-06-14 00:36:21 +00:00
|
|
|
|
|
|
|
ZoteroContext = function() {}
|
|
|
|
ZoteroContext.prototype = {
|
2017-05-31 15:28:47 +00:00
|
|
|
require,
|
2017-05-24 04:58:41 +00:00
|
|
|
|
2011-06-14 00:36:21 +00:00
|
|
|
/**
|
|
|
|
* Convenience method to replicate window.alert()
|
|
|
|
**/
|
|
|
|
// TODO: is this still used? if so, move to zotero.js
|
|
|
|
"alert":function alert(msg){
|
|
|
|
this.Zotero.debug("alert() is deprecated from Zotero XPCOM");
|
|
|
|
Cc["@mozilla.org/embedcomp/prompt-service;1"]
|
|
|
|
.getService(Ci.nsIPromptService)
|
|
|
|
.alert(null, "", msg);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Convenience method to replicate window.confirm()
|
|
|
|
**/
|
|
|
|
// TODO: is this still used? if so, move to zotero.js
|
|
|
|
"confirm":function confirm(msg){
|
|
|
|
this.Zotero.debug("confirm() is deprecated from Zotero XPCOM");
|
|
|
|
return Cc["@mozilla.org/embedcomp/prompt-service;1"]
|
|
|
|
.getService(Ci.nsIPromptService)
|
|
|
|
.confirm(null, "", msg);
|
|
|
|
},
|
|
|
|
|
|
|
|
"Cc":Cc,
|
|
|
|
"Ci":Ci,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Convenience method to replicate window.setTimeout()
|
|
|
|
**/
|
|
|
|
"setTimeout":function setTimeout(func, ms){
|
2016-05-17 15:49:13 +00:00
|
|
|
return this.Zotero.setTimeout(func, ms);
|
|
|
|
},
|
|
|
|
|
|
|
|
"clearTimeout":function setTimeout(id) {
|
|
|
|
this.Zotero.clearTimeout(id);
|
2011-06-14 00:36:21 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Switches in or out of connector mode
|
|
|
|
*/
|
|
|
|
"switchConnectorMode":function(isConnector) {
|
|
|
|
if(isConnector !== this.isConnector) {
|
2013-11-05 20:52:40 +00:00
|
|
|
Services.obs.notifyObservers(zContext.Zotero, "zotero-before-reload", isConnector ? "connector" : "full");
|
2013-08-09 15:10:38 +00:00
|
|
|
zContext.Zotero.shutdown().then(function() {
|
|
|
|
// create a new zContext
|
|
|
|
makeZoteroContext(isConnector);
|
2014-04-08 22:47:32 +00:00
|
|
|
return zContext.Zotero.init(zInitOptions);
|
2013-08-09 15:10:38 +00:00
|
|
|
}).done();
|
2011-06-14 00:36:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return zContext;
|
2015-03-09 18:25:49 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Shuts down Zotero, calls a callback (that may return a promise),
|
|
|
|
* then reinitializes Zotero. Returns a promise that is resolved
|
|
|
|
* when this process completes.
|
|
|
|
*/
|
2015-07-19 21:58:58 +00:00
|
|
|
"reinit":function(cb, isConnector, options = {}) {
|
2015-03-09 18:25:49 +00:00
|
|
|
Services.obs.notifyObservers(zContext.Zotero, "zotero-before-reload", isConnector ? "connector" : "full");
|
|
|
|
return zContext.Zotero.shutdown().then(function() {
|
|
|
|
return cb ? cb() : false;
|
|
|
|
}).finally(function() {
|
|
|
|
makeZoteroContext(isConnector);
|
2015-07-19 21:58:58 +00:00
|
|
|
var o = {};
|
|
|
|
Object.assign(o, zInitOptions);
|
|
|
|
Object.assign(o, options);
|
|
|
|
zContext.Zotero.init(o);
|
2015-03-09 18:25:49 +00:00
|
|
|
});
|
2011-06-14 00:36:21 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The class from which the Zotero global XPCOM context is constructed
|
|
|
|
*
|
|
|
|
* @constructor
|
|
|
|
* This runs when ZoteroService is first requested to load all applicable scripts and initialize
|
|
|
|
* Zotero. Calls to other XPCOM components must be in here rather than in top-level code, as other
|
|
|
|
* components may not have yet been initialized.
|
|
|
|
*/
|
|
|
|
function makeZoteroContext(isConnector) {
|
|
|
|
if(zContext) {
|
|
|
|
// Swap out old zContext
|
|
|
|
var oldzContext = zContext;
|
|
|
|
// Create new zContext
|
|
|
|
zContext = new ZoteroContext();
|
|
|
|
// Swap in old Zotero object, so that references don't break, but empty it
|
|
|
|
zContext.Zotero = oldzContext.Zotero;
|
|
|
|
for(var key in zContext.Zotero) delete zContext.Zotero[key];
|
|
|
|
} else {
|
|
|
|
zContext = new ZoteroContext();
|
|
|
|
zContext.Zotero = function() {};
|
|
|
|
}
|
|
|
|
|
2013-01-21 06:58:01 +00:00
|
|
|
var subscriptLoader = Cc["@mozilla.org/moz/jssubscript-loader;1"].getService(Ci.mozIJSSubScriptLoader);
|
|
|
|
|
2011-06-14 00:36:21 +00:00
|
|
|
// Load zotero.js first
|
2013-01-21 06:58:01 +00:00
|
|
|
subscriptLoader.loadSubScript("chrome://zotero/content/xpcom/" + xpcomFilesAll[0] + ".js", zContext);
|
2011-06-14 00:36:21 +00:00
|
|
|
|
|
|
|
// Load CiteProc into Zotero.CiteProc namespace
|
|
|
|
zContext.Zotero.CiteProc = {"Zotero":zContext.Zotero};
|
2013-01-21 06:58:01 +00:00
|
|
|
subscriptLoader.loadSubScript("chrome://zotero/content/xpcom/citeproc.js", zContext.Zotero.CiteProc);
|
|
|
|
|
|
|
|
// Load XRegExp object into Zotero.XRegExp
|
|
|
|
const xregexpFiles = [
|
|
|
|
/**Core functions**/
|
|
|
|
'xregexp',
|
|
|
|
|
|
|
|
/**Addons**/
|
|
|
|
'addons/build', //adds ability to "build regular expressions using named subpatterns, for readability and pattern reuse"
|
|
|
|
'addons/matchrecursive', //adds ability to "match recursive constructs using XRegExp pattern strings as left and right delimiters"
|
2011-06-14 00:36:21 +00:00
|
|
|
|
2013-01-21 06:58:01 +00:00
|
|
|
/**Unicode support**/
|
|
|
|
'addons/unicode/unicode-base', //required for all other unicode packages. Adds \p{Letter} category
|
|
|
|
|
|
|
|
//'addons/unicode/unicode-blocks', //adds support for all Unicode blocks (e.g. InArabic, InCyrillic_Extended_A, etc.)
|
|
|
|
'addons/unicode/unicode-categories', //adds support for all Unicode categories (e.g. Punctuation, Lowercase_Letter, etc.)
|
|
|
|
//'addons/unicode/unicode-properties', //adds Level 1 Unicode properties (e.g. Uppercase, White_Space, etc.)
|
|
|
|
//'addons/unicode/unicode-scripts' //adds support for all Unicode scripts (e.g. Gujarati, Cyrillic, etc.)
|
|
|
|
'addons/unicode/unicode-zotero' //adds support for some Unicode categories used in Zotero
|
|
|
|
];
|
|
|
|
for (var i=0; i<xregexpFiles.length; i++) {
|
|
|
|
subscriptLoader.loadSubScript("chrome://zotero/content/xpcom/xregexp/" + xregexpFiles[i] + ".js", zContext);
|
2012-12-19 12:36:08 +00:00
|
|
|
}
|
2011-06-14 00:36:21 +00:00
|
|
|
|
|
|
|
// Load remaining xpcomFiles
|
|
|
|
for (var i=1; i<xpcomFilesAll.length; i++) {
|
|
|
|
try {
|
2013-01-21 06:58:01 +00:00
|
|
|
subscriptLoader.loadSubScript("chrome://zotero/content/xpcom/" + xpcomFilesAll[i] + ".js", zContext);
|
2011-06-14 00:36:21 +00:00
|
|
|
}
|
|
|
|
catch (e) {
|
|
|
|
Components.utils.reportError("Error loading " + xpcomFilesAll[i] + ".js", zContext);
|
|
|
|
throw (e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load xpcomFiles for specific mode
|
2016-05-03 21:51:12 +00:00
|
|
|
for (let xpcomFile of (isConnector ? xpcomFilesConnector : xpcomFilesLocal)) {
|
2011-06-14 00:36:21 +00:00
|
|
|
try {
|
2017-07-21 09:30:49 +00:00
|
|
|
subscriptLoader.loadSubScript("chrome://zotero/content/xpcom/" + xpcomFile + ".js", zContext, "UTF-8");
|
2011-06-14 00:36:21 +00:00
|
|
|
}
|
|
|
|
catch (e) {
|
2016-04-04 10:33:15 +00:00
|
|
|
dump("Error loading " + xpcomFile + ".js\n\n");
|
|
|
|
dump(e + "\n\n");
|
2011-06-14 00:36:21 +00:00
|
|
|
Components.utils.reportError("Error loading " + xpcomFile + ".js", zContext);
|
|
|
|
throw (e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load RDF files into Zotero.RDF.AJAW namespace (easier than modifying all of the references)
|
|
|
|
const rdfXpcomFiles = [
|
2012-04-26 22:36:17 +00:00
|
|
|
'rdf/init',
|
2011-06-14 00:36:21 +00:00
|
|
|
'rdf/uri',
|
|
|
|
'rdf/term',
|
|
|
|
'rdf/identity',
|
|
|
|
'rdf/match',
|
|
|
|
'rdf/n3parser',
|
|
|
|
'rdf/rdfparser',
|
2012-04-30 01:30:37 +00:00
|
|
|
'rdf/serialize'
|
2011-06-14 00:36:21 +00:00
|
|
|
];
|
2012-05-05 20:53:00 +00:00
|
|
|
zContext.Zotero.RDF = {Zotero:zContext.Zotero};
|
2011-06-14 00:36:21 +00:00
|
|
|
for (var i=0; i<rdfXpcomFiles.length; i++) {
|
2013-01-21 06:58:01 +00:00
|
|
|
subscriptLoader.loadSubScript("chrome://zotero/content/xpcom/" + rdfXpcomFiles[i] + ".js", zContext.Zotero.RDF);
|
2009-07-06 10:13:02 +00:00
|
|
|
}
|
2011-06-14 00:36:21 +00:00
|
|
|
|
2012-02-09 04:05:34 +00:00
|
|
|
if(isStandalone()) {
|
|
|
|
// If isStandalone, load standalone.js
|
2013-01-21 06:58:01 +00:00
|
|
|
subscriptLoader.loadSubScript("chrome://zotero/content/xpcom/standalone.js", zContext);
|
2012-02-09 04:05:34 +00:00
|
|
|
}
|
|
|
|
|
2011-06-14 00:36:21 +00:00
|
|
|
// add connector-related properties
|
|
|
|
zContext.Zotero.isConnector = isConnector;
|
|
|
|
zContext.Zotero.instanceID = instanceID;
|
2016-10-15 19:20:09 +00:00
|
|
|
zContext.Zotero.__defineGetter__("isFirstLoadThisSession", function() { return isFirstLoadThisSession; });
|
2011-06-14 00:36:21 +00:00
|
|
|
};
|
2006-05-27 00:20:27 +00:00
|
|
|
|
2011-06-14 00:36:21 +00:00
|
|
|
/**
|
|
|
|
* The class representing the Zotero service, and affiliated XPCOM goop
|
|
|
|
*/
|
2011-07-02 22:04:04 +00:00
|
|
|
function ZoteroService() {
|
2008-07-03 16:44:52 +00:00
|
|
|
try {
|
2011-08-24 01:59:56 +00:00
|
|
|
var start = Date.now();
|
|
|
|
|
2011-06-14 00:36:21 +00:00
|
|
|
if(isFirstLoadThisSession) {
|
|
|
|
makeZoteroContext(false);
|
Async DB megacommit
Promise-based rewrite of most of the codebase, with asynchronous database and file access -- see https://github.com/zotero/zotero/issues/518 for details.
WARNING: This includes backwards-incompatible schema changes.
An incomplete list of other changes:
- Schema overhaul
- Replace main tables with new versions with updated schema
- Enable real foreign key support and remove previous triggers
- Don't use NULLs for local libraryID, which broke the UNIQUE index
preventing object key duplication. All code (Zotero and third-party)
using NULL for the local library will need to be updated to use 0
instead (already done for Zotero code)
- Add 'compatibility' DB version that can be incremented manually to break DB
compatibility with previous versions. 'userdata' upgrades will no longer
automatically break compatibility.
- Demote creators and tags from first-class objects to item properties
- New API syncing properties
- 'synced'/'version' properties to data objects
- 'etag' to groups
- 'version' to libraries
- Create Zotero.DataObject that other objects inherit from
- Consolidate data object loading into Zotero.DataObjects
- Change object reloading so that only the loaded and changed parts of objects are reloaded, instead of reloading all data from the database (with some exceptions, including item primary data)
- Items and collections now have .parentItem and .parentKey properties, replacing item.getSource() and item.getSourceKey()
- New function Zotero.serial(fn), to wrap an async function such that all calls are run serially
- New function Zotero.Utilities.Internal.forEachChunkAsync(arr, chunkSize, func)
- Add tag selector loading message
- Various API and name changes, since everything was breaking anyway
Known broken things:
- Syncing (will be completely rewritten for API syncing)
- Translation architecture (needs promise-based rewrite)
- Duplicates view
- DB integrity check (from schema changes)
- Dragging (may be difficult to fix)
Lots of other big and little things are certainly broken, particularly with the UI, which can be affected by async code in all sorts of subtle ways.
2014-08-06 21:38:05 +00:00
|
|
|
zContext.Zotero.init(zInitOptions)
|
2013-08-12 00:51:16 +00:00
|
|
|
.catch(function (e) {
|
2014-04-08 22:47:32 +00:00
|
|
|
if (e === "ZOTERO_SHOULD_START_AS_CONNECTOR") {
|
2013-12-11 19:50:48 +00:00
|
|
|
// if Zotero should start as a connector, reload it
|
2014-04-08 22:47:32 +00:00
|
|
|
return zContext.Zotero.shutdown()
|
|
|
|
.then(function() {
|
2013-12-11 19:50:48 +00:00
|
|
|
makeZoteroContext(true);
|
2014-04-08 22:47:32 +00:00
|
|
|
return zContext.Zotero.init(zInitOptions);
|
|
|
|
})
|
|
|
|
}
|
2016-09-06 00:41:35 +00:00
|
|
|
dump(e + "\n\n");
|
|
|
|
Components.utils.reportError(e);
|
2016-12-05 10:17:21 +00:00
|
|
|
if (!zContext.Zotero.startupError) {
|
2017-09-11 07:52:10 +00:00
|
|
|
zContext.Zotero.startupError = e.stack || e;
|
2016-12-06 11:12:54 +00:00
|
|
|
}
|
|
|
|
if (!isStandalone()) {
|
|
|
|
throw e;
|
2016-12-05 10:17:21 +00:00
|
|
|
}
|
2013-08-12 00:51:16 +00:00
|
|
|
})
|
|
|
|
.then(function () {
|
2016-11-25 19:08:50 +00:00
|
|
|
if (isStandalone()) {
|
2016-11-22 09:35:57 +00:00
|
|
|
if (zContext.Zotero.startupErrorHandler || zContext.Zotero.startupError) {
|
|
|
|
if (zContext.Zotero.startupErrorHandler) {
|
|
|
|
zContext.Zotero.startupErrorHandler();
|
|
|
|
}
|
|
|
|
else if (zContext.Zotero.startupError) {
|
2017-09-11 07:52:30 +00:00
|
|
|
try {
|
|
|
|
zContext.Zotero.startupError =
|
|
|
|
zContext.Zotero.Utilities.Internal.filterStack(
|
|
|
|
zContext.Zotero.startupError
|
|
|
|
);
|
|
|
|
}
|
|
|
|
catch (e) {}
|
|
|
|
|
2016-12-16 10:59:47 +00:00
|
|
|
let ps = Cc["@mozilla.org/embedcomp/prompt-service;1"]
|
|
|
|
.getService(Ci.nsIPromptService);
|
|
|
|
let buttonFlags = (ps.BUTTON_POS_0) * (ps.BUTTON_TITLE_IS_STRING)
|
|
|
|
+ (ps.BUTTON_POS_1) * (ps.BUTTON_TITLE_IS_STRING);
|
|
|
|
// Get the stringbundle manually
|
|
|
|
let errorStr = "Error";
|
|
|
|
let quitStr = "Quit";
|
|
|
|
let checkForUpdateStr = "Check for Update";
|
|
|
|
try {
|
|
|
|
let src = 'chrome://zotero/locale/zotero.properties';
|
|
|
|
let stringBundleService = Components.classes["@mozilla.org/intl/stringbundle;1"]
|
|
|
|
.getService(Components.interfaces.nsIStringBundleService);
|
2017-07-09 10:56:56 +00:00
|
|
|
let stringBundle = stringBundleService.createBundle(src);
|
2016-12-16 10:59:47 +00:00
|
|
|
errorStr = stringBundle.GetStringFromName('general.error');
|
|
|
|
checkForUpdateStr = stringBundle.GetStringFromName('general.checkForUpdate');
|
|
|
|
quitStr = stringBundle.GetStringFromName('general.quit');
|
|
|
|
}
|
|
|
|
catch (e) {}
|
|
|
|
let index = ps.confirmEx(
|
|
|
|
null,
|
|
|
|
errorStr,
|
|
|
|
zContext.Zotero.startupError,
|
|
|
|
buttonFlags,
|
|
|
|
checkForUpdateStr,
|
|
|
|
quitStr,
|
|
|
|
null,
|
|
|
|
null,
|
|
|
|
{}
|
|
|
|
);
|
|
|
|
if (index == 0) {
|
|
|
|
Components.classes["@mozilla.org/embedcomp/window-watcher;1"]
|
|
|
|
.getService(Components.interfaces.nsIWindowWatcher)
|
|
|
|
.openWindow(null, 'chrome://mozapps/content/update/updates.xul',
|
|
|
|
'updateChecker', 'chrome,centerscreen,modal', null);
|
|
|
|
}
|
2016-11-22 09:35:57 +00:00
|
|
|
}
|
|
|
|
zContext.Zotero.Utilities.Internal.quitZotero();
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2013-08-12 00:51:16 +00:00
|
|
|
zContext.Zotero.debug("Initialized in "+(Date.now() - start)+" ms");
|
2015-02-25 03:06:18 +00:00
|
|
|
isFirstLoadThisSession = false;
|
|
|
|
});
|
2016-12-16 08:18:29 +00:00
|
|
|
|
|
|
|
let cb;
|
|
|
|
while (cb = initCallbacks.shift()) {
|
|
|
|
cb(zContext.Zotero);
|
|
|
|
}
|
2011-06-14 00:36:21 +00:00
|
|
|
}
|
2013-08-12 00:51:16 +00:00
|
|
|
else {
|
|
|
|
zContext.Zotero.debug("Already initialized");
|
|
|
|
}
|
2011-06-14 00:36:21 +00:00
|
|
|
this.wrappedJSObject = zContext.Zotero;
|
|
|
|
} catch(e) {
|
2015-04-11 00:46:57 +00:00
|
|
|
var msg = e instanceof Error
|
|
|
|
? e.name + ': ' + e.message + '\n' + e.fileName + ':' + e.lineNumber + '\n' + e.stack
|
|
|
|
: '' + e;
|
|
|
|
dump(msg + '\n');
|
2008-07-03 16:44:52 +00:00
|
|
|
Components.utils.reportError(e);
|
2011-06-14 00:36:21 +00:00
|
|
|
throw e;
|
2008-07-03 16:44:52 +00:00
|
|
|
}
|
2006-05-27 00:20:27 +00:00
|
|
|
}
|
|
|
|
|
2010-07-06 09:02:35 +00:00
|
|
|
ZoteroService.prototype = {
|
2011-07-02 22:04:04 +00:00
|
|
|
contractID: '@zotero.org/Zotero;1',
|
|
|
|
classDescription: 'Zotero',
|
|
|
|
classID: Components.ID('{e4c61080-ec2d-11da-8ad9-0800200c9a66}'),
|
2011-06-14 00:36:21 +00:00
|
|
|
QueryInterface: XPCOMUtils.generateQI([Components.interfaces.nsISupports,
|
|
|
|
Components.interfaces.nsIProtocolHandler])
|
2010-07-06 09:02:35 +00:00
|
|
|
}
|
2006-05-27 00:20:27 +00:00
|
|
|
|
2016-12-16 08:18:29 +00:00
|
|
|
function addInitCallback(callback) {
|
|
|
|
if (zContext && zContext.Zotero) {
|
|
|
|
callback(zContext.Zotero);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
initCallbacks.push(callback);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-07-03 00:07:56 +00:00
|
|
|
var _isStandalone = null;
|
|
|
|
/**
|
|
|
|
* Determine whether Zotero Standalone is running
|
|
|
|
*/
|
|
|
|
function isStandalone() {
|
|
|
|
if(_isStandalone === null) {
|
|
|
|
var appInfo = Components.classes["@mozilla.org/xre/app-info;1"].
|
|
|
|
getService(Components.interfaces.nsIXULAppInfo);
|
|
|
|
_isStandalone = appInfo.ID === 'zotero@chnm.gmu.edu';
|
|
|
|
}
|
|
|
|
return _isStandalone;
|
|
|
|
}
|
|
|
|
|
2016-12-26 22:09:29 +00:00
|
|
|
function getOS() {
|
|
|
|
return Services.appinfo.OS;
|
|
|
|
}
|
|
|
|
|
|
|
|
function isMac() {
|
|
|
|
return getOS() == "Darwin";
|
|
|
|
}
|
|
|
|
|
|
|
|
function isWin() {
|
|
|
|
return getOS() == "WINNT";
|
|
|
|
}
|
|
|
|
|
|
|
|
function isLinux() {
|
|
|
|
return getOS() == "Linux";
|
|
|
|
}
|
|
|
|
|
2011-07-02 22:04:04 +00:00
|
|
|
/**
|
|
|
|
* The class representing the Zotero command line handler
|
|
|
|
*/
|
|
|
|
function ZoteroCommandLineHandler() {}
|
|
|
|
ZoteroCommandLineHandler.prototype = {
|
|
|
|
/* nsICommandLineHandler */
|
|
|
|
handle : function(cmdLine) {
|
2017-01-14 22:03:22 +00:00
|
|
|
// Force debug output to window
|
2015-03-24 07:38:11 +00:00
|
|
|
if (cmdLine.handleFlag("ZoteroDebug", false)) {
|
2017-01-14 22:03:22 +00:00
|
|
|
zInitOptions.forceDebugLog = 2;
|
|
|
|
}
|
|
|
|
// Force debug output to text console
|
|
|
|
else if (cmdLine.handleFlag("ZoteroDebugText", false)) {
|
|
|
|
zInitOptions.forceDebugLog = 1;
|
2013-11-30 06:55:48 +00:00
|
|
|
}
|
|
|
|
|
2017-09-11 07:49:06 +00:00
|
|
|
zInitOptions.forceDataDir = cmdLine.handleFlagWithParam("datadir", false);
|
|
|
|
|
2013-11-30 06:55:48 +00:00
|
|
|
// handler to open Zotero pane at startup in Zotero for Firefox
|
|
|
|
if (!isStandalone() && cmdLine.handleFlag("ZoteroPaneOpen", false)) {
|
|
|
|
zInitOptions.openPane = true;
|
|
|
|
}
|
|
|
|
|
2016-12-16 08:18:29 +00:00
|
|
|
if (cmdLine.handleFlag("ZoteroTest", false)) {
|
|
|
|
zInitOptions.test = true;
|
|
|
|
}
|
|
|
|
if (cmdLine.handleFlag("ZoteroAutomatedTest", false)) {
|
|
|
|
zInitOptions.automatedTest = true;
|
|
|
|
}
|
|
|
|
if (cmdLine.handleFlag("ZoteroSkipBundledFiles", false)) {
|
|
|
|
zInitOptions.skipBundledFiles = true;
|
|
|
|
}
|
|
|
|
|
2011-07-02 22:04:04 +00:00
|
|
|
// handler for Zotero integration commands
|
|
|
|
// this is typically used on Windows only, via WM_COPYDATA rather than the command line
|
|
|
|
var agent = cmdLine.handleFlagWithParam("ZoteroIntegrationAgent", false);
|
|
|
|
if(agent) {
|
|
|
|
// Don't open a new window
|
|
|
|
cmdLine.preventDefault = true;
|
|
|
|
|
|
|
|
var command = cmdLine.handleFlagWithParam("ZoteroIntegrationCommand", false);
|
|
|
|
var docId = cmdLine.handleFlagWithParam("ZoteroIntegrationDocument", false);
|
|
|
|
|
2016-12-16 08:18:29 +00:00
|
|
|
zContext.Zotero.Integration.execCommand(agent, command, docId);
|
2011-07-02 22:04:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// handler for Windows IPC commands
|
2011-07-03 03:07:01 +00:00
|
|
|
var ipcParam = cmdLine.handleFlagWithParam("ZoteroIPC", false);
|
|
|
|
if(ipcParam) {
|
2011-07-02 22:04:04 +00:00
|
|
|
// Don't open a new window
|
|
|
|
cmdLine.preventDefault = true;
|
2016-12-16 08:18:29 +00:00
|
|
|
if (!zContext) new ZoteroService();
|
|
|
|
let Zotero = zContext.Zotero;
|
|
|
|
Zotero.setTimeout(() => Zotero.IPC.parsePipeInput(ipcParam), 0);
|
2011-07-02 22:04:04 +00:00
|
|
|
}
|
|
|
|
|
2011-07-03 00:07:56 +00:00
|
|
|
if(isStandalone()) {
|
2016-12-16 08:18:29 +00:00
|
|
|
var fileToOpen;
|
|
|
|
// Special handler for "zotero" URIs at the command line to prevent them from opening a new window
|
2011-07-02 22:04:04 +00:00
|
|
|
var param = cmdLine.handleFlagWithParam("url", false);
|
2016-12-16 08:18:29 +00:00
|
|
|
if (param) {
|
2011-07-02 22:04:04 +00:00
|
|
|
var uri = cmdLine.resolveURI(param);
|
|
|
|
if(uri.schemeIs("zotero")) {
|
2017-07-29 21:13:46 +00:00
|
|
|
addInitCallback(function (Zotero) {
|
|
|
|
Zotero.uiReadyPromise
|
|
|
|
.then(function () {
|
|
|
|
// Check for existing window and focus it
|
|
|
|
var wm = Components.classes["@mozilla.org/appshell/window-mediator;1"]
|
|
|
|
.getService(Components.interfaces.nsIWindowMediator);
|
|
|
|
var win = wm.getMostRecentWindow("navigator:browser");
|
|
|
|
if (win) {
|
|
|
|
win.focus();
|
|
|
|
win.ZoteroPane.loadURI(uri.spec)
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
2016-12-16 08:18:29 +00:00
|
|
|
}
|
|
|
|
// See below
|
|
|
|
else if (uri.schemeIs("file")) {
|
|
|
|
Components.utils.import("resource://gre/modules/osfile.jsm")
|
|
|
|
fileToOpen = OS.Path.fromFileURI(uri.spec)
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
dump(`Not handling URL: ${uri.spec}\n\n`);
|
2011-08-17 04:51:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-17 21:08:03 +00:00
|
|
|
param = cmdLine.handleFlag("debugger", false);
|
|
|
|
if (param) {
|
|
|
|
try {
|
|
|
|
let portOrPath = Services.prefs.getBranch('').getIntPref('devtools.debugger.remote-port');
|
|
|
|
|
|
|
|
const { devtools } = Components.utils.import("resource://devtools/shared/Loader.jsm", {});
|
|
|
|
const { DebuggerServer } = devtools.require("devtools/server/main");
|
|
|
|
|
|
|
|
if (!DebuggerServer.initialized) {
|
|
|
|
dump("Initializing devtools server\n");
|
|
|
|
DebuggerServer.init();
|
|
|
|
DebuggerServer.allowChromeProcess = true;
|
|
|
|
DebuggerServer.addBrowserActors();
|
|
|
|
}
|
|
|
|
|
|
|
|
let listener = DebuggerServer.createListener();
|
|
|
|
listener.portOrPath = portOrPath;
|
|
|
|
listener.open();
|
|
|
|
|
|
|
|
dump("Debugger server started on " + portOrPath + "\n\n");
|
|
|
|
}
|
|
|
|
catch (e) {
|
|
|
|
dump(e + "\n\n");
|
|
|
|
Components.utils.reportError(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-16 08:18:29 +00:00
|
|
|
// In Fx49-based Mac Standalone, if Zotero is closed, an associated file is launched, and
|
|
|
|
// Zotero hasn't been opened before, a -file parameter is passed and two main windows open.
|
|
|
|
// Subsequent file openings when closed result in -url with file:// URLs (converted above)
|
2016-12-26 22:09:29 +00:00
|
|
|
// and don't result in two windows. Here we prevent the double window.
|
2016-12-16 08:18:29 +00:00
|
|
|
param = fileToOpen;
|
|
|
|
if (!param) {
|
|
|
|
param = cmdLine.handleFlagWithParam("file", false);
|
2016-12-26 22:09:29 +00:00
|
|
|
if (param && isMac()) {
|
2016-12-16 08:18:29 +00:00
|
|
|
cmdLine.preventDefault = true;
|
2011-07-02 22:04:04 +00:00
|
|
|
}
|
|
|
|
}
|
2016-12-16 08:18:29 +00:00
|
|
|
if (param) {
|
|
|
|
addInitCallback(function (Zotero) {
|
|
|
|
// Wait to handle things that require the UI until after it's loaded
|
|
|
|
Zotero.uiReadyPromise
|
|
|
|
.then(function () {
|
|
|
|
var file = Components.classes["@mozilla.org/file/local;1"].
|
|
|
|
createInstance(Components.interfaces.nsILocalFile);
|
|
|
|
file.initWithPath(param);
|
|
|
|
|
|
|
|
if(file.leafName.substr(-4).toLowerCase() === ".csl"
|
|
|
|
|| file.leafName.substr(-8).toLowerCase() === ".csl.txt") {
|
|
|
|
// Install CSL file
|
2017-12-08 05:04:00 +00:00
|
|
|
Zotero.Styles.install({ file: file.path }, file.path);
|
2016-12-16 08:18:29 +00:00
|
|
|
} else {
|
|
|
|
// Ask before importing
|
|
|
|
var checkState = {
|
|
|
|
value: Zotero.Prefs.get('import.createNewCollection.fromFileOpenHandler')
|
|
|
|
};
|
|
|
|
if (Components.classes["@mozilla.org/embedcomp/prompt-service;1"]
|
|
|
|
.getService(Components.interfaces.nsIPromptService)
|
|
|
|
.confirmCheck(null, Zotero.getString('ingester.importFile.title'),
|
|
|
|
Zotero.getString('ingester.importFile.text', [file.leafName]),
|
|
|
|
Zotero.getString('ingester.importFile.intoNewCollection'),
|
|
|
|
checkState)) {
|
|
|
|
Zotero.Prefs.set(
|
|
|
|
'import.createNewCollection.fromFileOpenHandler', checkState.value
|
|
|
|
);
|
|
|
|
|
|
|
|
// Perform file import in front window
|
|
|
|
var wm = Components.classes["@mozilla.org/appshell/window-mediator;1"]
|
|
|
|
.getService(Components.interfaces.nsIWindowMediator);
|
|
|
|
var browserWindow = wm.getMostRecentWindow("navigator:browser");
|
|
|
|
browserWindow.Zotero_File_Interface.importFile(file, checkState.value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
2015-04-16 21:35:38 +00:00
|
|
|
}
|
2011-07-02 22:04:04 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
contractID: "@mozilla.org/commandlinehandler/general-startup;1?type=zotero",
|
|
|
|
classDescription: "Zotero Command Line Handler",
|
|
|
|
classID: Components.ID("{531828f8-a16c-46be-b9aa-14845c3b010f}"),
|
|
|
|
service: true,
|
|
|
|
_xpcom_categories: [{category:"command-line-handler", entry:"m-zotero"}],
|
|
|
|
QueryInterface: XPCOMUtils.generateQI([Components.interfaces.nsICommandLineHandler,
|
|
|
|
Components.interfaces.nsISupports])
|
|
|
|
};
|
|
|
|
|
2016-10-08 23:15:49 +00:00
|
|
|
var NSGetFactory = XPCOMUtils.generateNSGetFactory([ZoteroService, ZoteroCommandLineHandler]);
|