534 lines
14 KiB
JavaScript
534 lines
14 KiB
JavaScript
/*
|
|
***** BEGIN LICENSE BLOCK *****
|
|
|
|
Copyright © 2009 Center for History and New Media
|
|
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
|
|
it under the terms of the GNU Affero General Public License as published by
|
|
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
|
|
GNU Affero General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Affero General Public License
|
|
along with Zotero. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
***** END LICENSE BLOCK *****
|
|
*/
|
|
|
|
Zotero.History = new function(){
|
|
this.begin = begin;
|
|
this.setAssociatedID = setAssociatedID;
|
|
this.add = add;
|
|
this.modify = modify;
|
|
this.remove = remove;
|
|
this.commit = commit;
|
|
this.cancel = cancel;
|
|
this.getPreviousEvent = getPreviousEvent;
|
|
this.getNextEvent = getNextEvent;
|
|
this.undo = undo;
|
|
this.redo = redo;
|
|
this.clear = clear;
|
|
this.clearAfter = clearAfter;
|
|
|
|
var _firstTime = true;
|
|
var _currentID = 0;
|
|
var _activeID;
|
|
var _activeEvent;
|
|
var _maxID = 0;
|
|
|
|
|
|
/**
|
|
* Begin a transaction set
|
|
*
|
|
* event: 'item-add', 'item-delete', 'item-modify', 'collection-add',
|
|
* 'collection-modify', 'collection-delete'...
|
|
*
|
|
* id: An id or array of ids that will be passed to
|
|
* Zotero.Notifier.trigger() on an undo or redo
|
|
**/
|
|
function begin(event, id){
|
|
if (_activeID){
|
|
throw('History transaction set already in progress');
|
|
}
|
|
|
|
// If running for the first time this session or we're in the middle of
|
|
// the history, clear any transaction sets after the current position
|
|
if (_firstTime || _currentID<_maxID){
|
|
_firstTime = false;
|
|
this.clearAfter();
|
|
}
|
|
|
|
Zotero.debug('Beginning history transaction set ' + event);
|
|
var sql = "INSERT INTO transactionSets (event, id) VALUES "
|
|
+ "('" + event + "', ";
|
|
if (!id){
|
|
sql += '0';
|
|
}
|
|
// If array, insert hyphen-delimited string
|
|
else if (typeof id=='object'){
|
|
sql += "'" + id.join('-') + "'"
|
|
}
|
|
else {
|
|
sql += id;
|
|
}
|
|
sql += ")";
|
|
|
|
Zotero.DB.beginTransaction();
|
|
_activeID = Zotero.DB.query(sql);
|
|
_activeEvent = event;
|
|
}
|
|
|
|
|
|
/**
|
|
* Associate an id or array of ids with the transaction set --
|
|
* for use if the ids weren't available at when begin() was called
|
|
*
|
|
* id: An id or array of ids that will be passed to
|
|
* Zotero.Notifier.trigger() on an undo or redo
|
|
**/
|
|
function setAssociatedID(id){
|
|
if (!_activeID){
|
|
throw('Cannot call setAssociatedID() with no history transaction set in progress');
|
|
}
|
|
|
|
var sql = "UPDATE transactionSets SET id=";
|
|
if (!id){
|
|
sql += '0';
|
|
}
|
|
// If array, insert hyphen-delimited string
|
|
else if (typeof id=='object'){
|
|
sql += "'" + id.join('-') + "'"
|
|
}
|
|
else {
|
|
sql += id;
|
|
}
|
|
sql += " WHERE transactionSetID=" + _activeID;
|
|
Zotero.DB.query(sql);
|
|
}
|
|
|
|
|
|
/**
|
|
* Add an add transaction to the current set
|
|
*
|
|
* Can be called before or after an INSERT statement
|
|
*
|
|
* key is a hyphen-delimited list of columns identifying the row
|
|
* e.g. 'itemID-creatorID'
|
|
*
|
|
* keyValues is a hyphen-delimited list of values matching the key parts
|
|
* e.g. '1-1'
|
|
**/
|
|
function add(table, key, keyValues){
|
|
return _addTransaction('add', table, key, keyValues);
|
|
}
|
|
|
|
|
|
/**
|
|
* Add a modify transaction to the current set
|
|
*
|
|
* Must be called before an UPDATE statement
|
|
*
|
|
* key is a hyphen-delimited list of columns identifying the row
|
|
* e.g. 'itemID-creatorID'
|
|
*
|
|
* keyValues is an array or hyphen-delimited string of values matching
|
|
* the key parts (e.g. [1, 1] or '1-1')
|
|
*
|
|
* _field_ is optional -- otherwise all fields are saved
|
|
**/
|
|
function modify(table, key, keyValues, field){
|
|
return _addTransaction('modify', table, key, keyValues, field);
|
|
}
|
|
|
|
|
|
/**
|
|
* Add a remove transaction to the current set
|
|
*
|
|
* Must be called before a DELETE statement
|
|
*
|
|
* key is a hyphen-delimited list of columns identifying the row
|
|
* e.g. 'itemID-creatorID'
|
|
*
|
|
* keyValues is a hyphen-delimited list of values matching the key parts
|
|
* e.g. '1-1'
|
|
**/
|
|
function remove(table, key, keyValues){
|
|
return _addTransaction('remove', table, key, keyValues);
|
|
}
|
|
|
|
|
|
/**
|
|
* Commit the current transaction set
|
|
**/
|
|
function commit(){
|
|
Zotero.debug('Committing history transaction set ' + _activeEvent);
|
|
Zotero.DB.commitTransaction();
|
|
_currentID = _activeID;
|
|
_maxID = _activeID;
|
|
_activeID = null;
|
|
_activeEvent = null;
|
|
}
|
|
|
|
|
|
/**
|
|
* Cancel the current transaction set
|
|
**/
|
|
function cancel(){
|
|
Zotero.debug('Cancelling history transaction set ' + _activeEvent);
|
|
Zotero.DB.rollbackTransaction();
|
|
_activeID = null;
|
|
_activeEvent = null;
|
|
}
|
|
|
|
|
|
/**
|
|
* Get the next event to undo, or false if none
|
|
**/
|
|
function getPreviousEvent(){
|
|
if (!_currentID){
|
|
return false;
|
|
}
|
|
|
|
var sql = "SELECT event FROM transactionSets WHERE transactionSetID="
|
|
+ _currentID;
|
|
return Zotero.DB.valueQuery(sql);
|
|
}
|
|
|
|
|
|
/**
|
|
* Get the next event to redo, or false if none
|
|
**/
|
|
function getNextEvent(){
|
|
var sql = "SELECT event FROM transactionSets WHERE transactionSetID="
|
|
+ (_currentID + 1);
|
|
return Zotero.DB.valueQuery(sql);
|
|
}
|
|
|
|
|
|
/**
|
|
* Undo the last transaction set
|
|
**/
|
|
function undo(){
|
|
if (!_currentID){
|
|
throw('No transaction set to undo');
|
|
return false;
|
|
}
|
|
|
|
var id = _currentID;
|
|
Zotero.debug('Undoing transaction set ' + id);
|
|
Zotero.DB.beginTransaction();
|
|
var undone = _do('undo');
|
|
_currentID--;
|
|
Zotero.DB.commitTransaction();
|
|
_reloadAndNotify(id);
|
|
return true;
|
|
}
|
|
|
|
|
|
/**
|
|
* Redo the next transaction set
|
|
**/
|
|
function redo(){
|
|
var id = _currentID + 1;
|
|
Zotero.debug('Redoing transaction set ' + id);
|
|
Zotero.DB.beginTransaction();
|
|
var redone = _do('redo');
|
|
_currentID++;
|
|
Zotero.DB.commitTransaction();
|
|
_reloadAndNotify(id, true);
|
|
return redone;
|
|
}
|
|
|
|
|
|
/**
|
|
* Clear the entire history
|
|
**/
|
|
function clear(){
|
|
Zotero.DB.beginTransaction();
|
|
Zotero.DB.query("DELETE FROM transactionSets");
|
|
Zotero.DB.query("DELETE FROM transactions");
|
|
Zotero.DB.query("DELETE FROM transactionLog");
|
|
_currentID = null;
|
|
_activeID = null;
|
|
_activeEvent = null;
|
|
_maxID = null;
|
|
Zotero.DB.commitTransaction();
|
|
}
|
|
|
|
|
|
/**
|
|
* Clear all transactions in history after the current one
|
|
**/
|
|
function clearAfter(){
|
|
Zotero.DB.beginTransaction();
|
|
var min = Zotero.DB.valueQuery("SELECT MIN(transactionID) FROM "
|
|
+ "transactions WHERE transactionSetID=" + (_currentID + 1));
|
|
|
|
if (!min){
|
|
Zotero.DB.commitTransaction();
|
|
return;
|
|
}
|
|
|
|
Zotero.DB.query("DELETE FROM transactionLog "
|
|
+ "WHERE transactionID>=" + min);
|
|
Zotero.DB.query("DELETE FROM transactions "
|
|
+ "WHERE transactionID>=" + min);
|
|
Zotero.DB.query("DELETE FROM transactionSets "
|
|
+ "WHERE transactionSetID>" + _currentID);
|
|
|
|
_maxID = _currentID;
|
|
_activeID = null;
|
|
Zotero.DB.commitTransaction();
|
|
return;
|
|
}
|
|
|
|
|
|
//
|
|
// Private methods
|
|
//
|
|
|
|
function _addTransaction(action, table, key, keyValues, field){
|
|
if (!_activeID){
|
|
throw('Cannot add history transaction with no transaction set in progress');
|
|
}
|
|
|
|
if (typeof keyValues == 'object'){
|
|
keyValues = keyValues.join('-');
|
|
}
|
|
|
|
var contextString = table + '.' + key + '.' + keyValues;
|
|
var context = _parseContext(contextString);
|
|
var fromClause = _contextToSQLFrom(context);
|
|
|
|
var sql = "INSERT INTO transactions (transactionSetID, context, action) "
|
|
+ "VALUES (" + _activeID + ", '" + contextString
|
|
+ "', '" + action + "')";
|
|
|
|
var transactionID = Zotero.DB.query(sql);
|
|
|
|
switch (action){
|
|
case 'add':
|
|
// No need to store an add, since we'll just delete it to reverse
|
|
break;
|
|
case 'modify':
|
|
// Only save one field -- _do() won't know about this, but the
|
|
// UPDATE statements on the other fields just won't do anything
|
|
if (field){
|
|
var sql = "INSERT INTO transactionLog SELECT " + transactionID
|
|
+ ", '" + field + "', " + field + fromClause;
|
|
Zotero.DB.query(sql);
|
|
break;
|
|
}
|
|
// Fall through if no field specified and save all
|
|
case 'remove':
|
|
var cols = Zotero.DB.getColumns(table);
|
|
for (var i in cols){
|
|
// If column is not part of the key, log it
|
|
if (!context['keys'].indexOf(cols[i]) === -1){
|
|
var sql = "INSERT INTO transactionLog "
|
|
+ "SELECT " + transactionID + ", '" + cols[i]
|
|
+ "', " + cols[i] + fromClause;
|
|
Zotero.DB.query(sql);
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
Zotero.DB.rollbackTransaction();
|
|
throw("Invalid history action '" + action + "'");
|
|
}
|
|
}
|
|
|
|
|
|
function _do(mode){
|
|
switch (mode){
|
|
case 'undo':
|
|
var id = _currentID;
|
|
break;
|
|
case 'redo':
|
|
var id = _currentID + 1;
|
|
break;
|
|
}
|
|
|
|
var sql = "SELECT transactionID, context, action FROM transactions "
|
|
+ "WHERE transactionSetID=" + id;
|
|
var transactions = Zotero.DB.query(sql);
|
|
|
|
if (!transactions){
|
|
throw('Transaction set not found for '
|
|
+ (mode=='undo' ? 'current' : 'next') + id);
|
|
}
|
|
|
|
for (var i in transactions){
|
|
var transactionID = transactions[i]['transactionID'];
|
|
var context = _parseContext(transactions[i]['context']);
|
|
|
|
// If in redo mode, swap 'add' and 'remove'
|
|
if (mode=='redo'){
|
|
switch (transactions[i]['action']){
|
|
case 'add':
|
|
transactions[i]['action'] = 'remove';
|
|
break;
|
|
case 'remove':
|
|
transactions[i]['action'] = 'add';
|
|
break;
|
|
}
|
|
}
|
|
|
|
switch (transactions[i]['action']){
|
|
case 'add':
|
|
var fromClause = _contextToSQLFrom(context);
|
|
|
|
// First, store the row we're about to delete for later redo
|
|
var cols = Zotero.DB.getColumns(context['table']);
|
|
for (var i in cols){
|
|
// If column is not part of the key, log it
|
|
if (!context['keys'].indexOf(cols[i]) === -1){
|
|
var sql = "INSERT INTO transactionLog "
|
|
+ "SELECT " + transactionID + ", '" + cols[i]
|
|
+ "', " + cols[i] + fromClause;
|
|
Zotero.DB.query(sql);
|
|
}
|
|
}
|
|
|
|
// And delete the row
|
|
var sql = "DELETE" + fromClause;
|
|
Zotero.DB.query(sql);
|
|
break;
|
|
|
|
case 'modify':
|
|
// Retrieve old values
|
|
var sql = "SELECT field, value FROM transactionLog "
|
|
+ "WHERE transactionID=" + transactionID;
|
|
var oldFieldValues = Zotero.DB.query(sql);
|
|
|
|
// Retrieve new values
|
|
var sql = "SELECT *" + _contextToSQLFrom(context);
|
|
var newValues = Zotero.DB.rowQuery(sql);
|
|
|
|
// Update row with old values
|
|
var sql = "UPDATE " + context['table'] + " SET ";
|
|
var values = [];
|
|
for (var i in oldFieldValues){
|
|
sql += oldFieldValues[i]['field'] + '=?, ';
|
|
values.push(oldFieldValues[i]['value']);
|
|
}
|
|
sql = sql.substr(0, sql.length-2) + _contextToSQLWhere(context);
|
|
Zotero.DB.query(sql, values);
|
|
|
|
// Update log with new values for later redo
|
|
for (var i in newValues){
|
|
if (context['keys'].indexOf(i) === -1){
|
|
var sql = "UPDATE transactionLog SET "
|
|
+ "value=? WHERE transactionID=? AND field=?";
|
|
Zotero.DB.query(sql, [i, newValues[i], transactionID]);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 'remove':
|
|
// Retrieve old values
|
|
var sql = "SELECT field, value FROM transactionLog "
|
|
+ "WHERE transactionID=" + transactionID;
|
|
var oldFieldValues = Zotero.DB.query(sql);
|
|
|
|
// Add key to parameters
|
|
var fields = [], values = [], marks = [];
|
|
for (var i=0; i<context['keys'].length; i++){
|
|
fields.push(context['keys'][i]);
|
|
values.push(context['values'][i]);
|
|
marks.push('?');
|
|
}
|
|
|
|
// Add other fields to parameters
|
|
for (var i in oldFieldValues){
|
|
fields.push(oldFieldValues[i]['field']);
|
|
values.push(oldFieldValues[i]['value']);
|
|
marks.push('?');
|
|
}
|
|
|
|
// Insert old values into table
|
|
var sql = "INSERT INTO " + context['table'] + "("
|
|
+ fields.join() + ") VALUES (" + marks.join() + ")";
|
|
Zotero.DB.query(sql, values);
|
|
|
|
// Delete restored data from transactionLog
|
|
var sql = "DELETE FROM transactionLog WHERE transactionID="
|
|
+ transactionID;
|
|
Zotero.DB.query(sql);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
function _parseContext(context){
|
|
var parts = context.split('.');
|
|
var parsed = {
|
|
table:parts[0],
|
|
keys:parts[1].split('-'),
|
|
values:parts[2].split('-')
|
|
}
|
|
if (parsed['keys'].length!=parsed['values'].length){
|
|
throw("Different number of keys and values in _parseContext('"
|
|
+ context + "')");
|
|
}
|
|
|
|
return parsed;
|
|
}
|
|
|
|
|
|
function _contextToSQLFrom(parsed){
|
|
return " FROM " + parsed['table'] + _contextToSQLWhere(parsed);
|
|
}
|
|
|
|
|
|
function _contextToSQLWhere(parsed){
|
|
var sql = " WHERE ";
|
|
for (var i=0; i<parsed['keys'].length; i++){
|
|
// DEBUG: type?
|
|
sql += parsed['keys'][i] + "='" + parsed['values'][i] + "' AND ";
|
|
}
|
|
return sql.substr(0, sql.length-5);
|
|
}
|
|
|
|
|
|
/**
|
|
* Get the ids associated with a particular transaction set
|
|
**/
|
|
function _getSetData(transactionSetID){
|
|
var sql = "SELECT event, id FROM transactionSets WHERE transactionSetID="
|
|
+ transactionSetID;
|
|
return Zotero.DB.rowQuery(sql);
|
|
}
|
|
|
|
|
|
function _reloadAndNotify(transactionSetID, redo){
|
|
var data = _getSetData(transactionSetID);
|
|
var eventParts = data['event'].split('-'); // e.g. modify-item
|
|
if (redo){
|
|
switch (eventParts[0]){
|
|
case 'add':
|
|
eventParts[0] = 'remove';
|
|
break;
|
|
case 'remove':
|
|
eventParts[0] = 'add';
|
|
break;
|
|
}
|
|
}
|
|
switch (eventParts[1]){
|
|
case 'item':
|
|
Zotero.Items.reload(data['id']);
|
|
break;
|
|
}
|
|
|
|
Zotero.Notifier.trigger(eventParts[0], eventParts[1], data['id']);
|
|
}
|
|
}
|