Limit autocomplete for tags and fields to current library

Not done for advanced search

Should be easy for saved search, but not yet done
This commit is contained in:
Dan Stillman 2013-04-19 15:10:05 -04:00
parent 8469ce821b
commit 65c7a5988b
4 changed files with 81 additions and 56 deletions

View File

@ -1416,11 +1416,17 @@
|| fieldName == 'creator') { || fieldName == 'creator') {
t.setAttribute('type', 'autocomplete'); t.setAttribute('type', 'autocomplete');
t.setAttribute('autocompletesearch', 'zotero'); t.setAttribute('autocompletesearch', 'zotero');
var suffix = itemID ? itemID : ''; let params = {
if (field=='creator') { fieldName: fieldName,
suffix = elem.getAttribute('fieldMode') + '-' + suffix; libraryID: this.item.libraryID
} };
t.setAttribute('autocompletesearchparam', fieldName + '/' + suffix); if (field == 'creator') {
params.fieldMode = parseInt(elem.getAttribute('fieldMode'));
params.itemID = itemID ? itemID : '';
};
t.setAttribute(
'autocompletesearchparam', JSON.stringify(params)
);
t.setAttribute('ontextentered', t.setAttribute('ontextentered',
'document.getBindingParent(this).handleCreatorAutoCompleteSelect(this)'); 'document.getBindingParent(this).handleCreatorAutoCompleteSelect(this)');
} }

View File

@ -437,8 +437,14 @@
else { else {
t.setAttribute('type', 'autocomplete'); t.setAttribute('type', 'autocomplete');
t.setAttribute('autocompletesearch', 'zotero'); t.setAttribute('autocompletesearch', 'zotero');
var suffix = itemID ? itemID : ''; let params = {
t.setAttribute('autocompletesearchparam', fieldName + '/' + suffix); fieldName: fieldName,
libraryID: this.item.libraryID
};
params.itemID = itemID ? itemID : '';
t.setAttribute(
'autocompletesearchparam', JSON.stringify(params)
);
} }
var box = elem.parentNode; var box = elem.parentNode;

View File

@ -813,22 +813,20 @@
textbox.setAttribute('autocompletesearch', 'zotero'); textbox.setAttribute('autocompletesearch', 'zotero');
textbox.setAttribute('timeout', '250'); textbox.setAttribute('timeout', '250');
if (condition=='creator') var autocompleteParams = {
{ fieldName: condition
// 2 searches both single- and double-field creators };
var autocompleteCondition = condition + '/2' if (condition == 'creator') {
autocompleteParams.fieldMode = 2;
} }
else textbox.setAttribute(
{ 'autocompletesearchparam',
var autocompleteCondition = condition; JSON.stringify(autocompleteParams)
} );
textbox.setAttribute('autocompletesearchparam', autocompleteCondition);
} }
} }
if (!autocompleteCondition) if (!autocompleteParams) {
{
var textbox = document.getAnonymousNodes(this)[0]; var textbox = document.getAnonymousNodes(this)[0];
textbox.removeAttribute('type'); textbox.removeAttribute('type');
} }

View File

@ -44,7 +44,7 @@ function ZoteroAutoComplete() {
} }
ZoteroAutoComplete.prototype.startSearch = function(searchString, searchParam, previousResult, listener) { ZoteroAutoComplete.prototype.startSearch = function(searchString, searchParams, previousResult, listener) {
var result = Cc["@mozilla.org/autocomplete/simple-result;1"] var result = Cc["@mozilla.org/autocomplete/simple-result;1"]
.createInstance(Ci.nsIAutoCompleteSimpleResult); .createInstance(Ci.nsIAutoCompleteSimpleResult);
result.setSearchString(searchString); result.setSearchString(searchString);
@ -54,35 +54,35 @@ ZoteroAutoComplete.prototype.startSearch = function(searchString, searchParam, p
this._listener = listener; this._listener = listener;
this._cancelled = false; this._cancelled = false;
this._zotero.debug("Starting autocomplete search of type '" this._zotero.debug("Starting autocomplete search with data '"
+ searchParam + "'" + " with string '" + searchString + "'"); + searchParams + "'" + " and string '" + searchString + "'");
searchParams = JSON.parse(searchParams);
if (!searchParams) {
throw new Error("Invalid JSON passed to autocomplete");
}
var [fieldName, , subField] = searchParams.fieldName.split("-");
this.stopSearch(); this.stopSearch();
var self = this; var self = this;
var statement; var statement;
// Allow extra parameters to be passed in switch (fieldName) {
var pos = searchParam.indexOf('/');
if (pos!=-1){
var extra = searchParam.substr(pos + 1);
var searchParam = searchParam.substr(0, pos);
}
var searchParts = searchParam.split('-');
searchParam = searchParts[0];
switch (searchParam) {
case '': case '':
break; break;
case 'tag': case 'tag':
var sql = "SELECT DISTINCT name AS val, NULL AS comment FROM tags WHERE name LIKE ?"; var sql = "SELECT DISTINCT name AS val, NULL AS comment FROM tags WHERE name LIKE ?";
var sqlParams = [searchString + '%']; var sqlParams = [searchString + '%'];
if (extra){ if (typeof searchParams.libraryID != 'undefined') {
sql += " AND libraryID=?";
sqlParams.push(searchParams.libraryID);
}
if (searchParams.itemID) {
sql += " AND name NOT IN (SELECT name FROM tags WHERE tagID IN (" sql += " AND name NOT IN (SELECT name FROM tags WHERE tagID IN ("
+ "SELECT tagID FROM itemTags WHERE itemID = ?))"; + "SELECT tagID FROM itemTags WHERE itemID = ?))";
sqlParams.push(extra); sqlParams.push(searchParams.itemID);
} }
statement = this._zotero.DB.getStatement(sql, sqlParams); statement = this._zotero.DB.getStatement(sql, sqlParams);
@ -103,22 +103,24 @@ ZoteroAutoComplete.prototype.startSearch = function(searchString, searchParam, p
// 0 == search two-field creators // 0 == search two-field creators
// 1 == search single-field creators // 1 == search single-field creators
// 2 == search both // 2 == search both
var [fieldMode, itemID] = extra.split('-'); if (searchParams.fieldMode == 2) {
if (fieldMode==2)
{
var sql = "SELECT DISTINCT CASE fieldMode WHEN 1 THEN lastName " var sql = "SELECT DISTINCT CASE fieldMode WHEN 1 THEN lastName "
+ "WHEN 0 THEN firstName || ' ' || lastName END AS val, NULL AS comment " + "WHEN 0 THEN firstName || ' ' || lastName END AS val, NULL AS comment "
+ "FROM creators NATURAL JOIN creatorData WHERE CASE fieldMode " + "FROM creators NATURAL JOIN creatorData WHERE CASE fieldMode "
+ "WHEN 1 THEN lastName " + "WHEN 1 THEN lastName "
+ "WHEN 0 THEN firstName || ' ' || lastName END " + "WHEN 0 THEN firstName || ' ' || lastName END "
+ "LIKE ? ORDER BY val"; + "LIKE ? ";
var sqlParams = searchString + '%'; var sqlParams = [searchString + '%'];
if (typeof searchParams.libraryID != 'undefined') {
sql += " AND libraryID=?";
sqlParams.push(searchParams.libraryID);
}
sql += "ORDER BY val";
} }
else else
{ {
var sql = "SELECT DISTINCT "; var sql = "SELECT DISTINCT ";
if (fieldMode==1){ if (searchParams.fieldMode == 1) {
sql += "lastName AS val, creatorID || '-1' AS comment"; sql += "lastName AS val, creatorID || '-1' AS comment";
} }
// Retrieve the matches in the specified field // Retrieve the matches in the specified field
@ -138,22 +140,35 @@ ZoteroAutoComplete.prototype.startSearch = function(searchString, searchParam, p
} }
var fromSQL = " FROM creators NATURAL JOIN creatorData " var fromSQL = " FROM creators NATURAL JOIN creatorData "
+ "WHERE " + searchParts[2] + " LIKE ?1 " + "AND fieldMode=?2"; + "WHERE " + subField + " LIKE ?1 " + "AND fieldMode=?2";
var sqlParams = [searchString + '%', var sqlParams = [
fieldMode ? parseInt(fieldMode) : 0]; searchString + '%',
if (itemID){ searchParams.fieldMode ? searchParams.fieldMode : 0
];
if (searchParams.itemID) {
fromSQL += " AND creatorID NOT IN (SELECT creatorID FROM " fromSQL += " AND creatorID NOT IN (SELECT creatorID FROM "
+ "itemCreators WHERE itemID=?3)"; + "itemCreators WHERE itemID=?3)";
sqlParams.push(itemID); sqlParams.push(searchParams.itemID);
}
if (typeof searchParams.libraryID != 'undefined') {
if (searchParams.libraryID) {
fromSQL += " AND libraryID=?4";
sqlParams.push(searchParams.libraryID);
}
// The db query code doesn't properly replace numbered
// parameters with "IS NULL"
else {
fromSQL += " AND libraryID IS NULL";
}
} }
sql += fromSQL; sql += fromSQL;
// If double-field mode, include matches for just this field // If double-field mode, include matches for just this field
// as well (i.e. "Shakespeare"), and group to collapse repeats // as well (i.e. "Shakespeare"), and group to collapse repeats
if (fieldMode!=1){ if (searchParams.fieldMode != 1) {
sql = "SELECT * FROM (" + sql + " UNION SELECT DISTINCT " sql = "SELECT * FROM (" + sql + " UNION SELECT DISTINCT "
+ searchParts[2] + " AS val, creatorID || '-1' AS comment" + subField + " AS val, creatorID || '-1' AS comment"
+ fromSQL + ") GROUP BY val"; + fromSQL + ") GROUP BY val";
} }
@ -165,8 +180,8 @@ ZoteroAutoComplete.prototype.startSearch = function(searchString, searchParam, p
case 'dateModified': case 'dateModified':
case 'dateAdded': case 'dateAdded':
var sql = "SELECT DISTINCT DATE(" + searchParam + ", 'localtime') AS val, NULL AS comment FROM items " var sql = "SELECT DISTINCT DATE(" + fieldName + ", 'localtime') AS val, NULL AS comment FROM items "
+ "WHERE " + searchParam + " LIKE ? ORDER BY " + searchParam; + "WHERE " + fieldName + " LIKE ? ORDER BY " + fieldName;
var sqlParams = [searchString + '%']; var sqlParams = [searchString + '%'];
statement = this._zotero.DB.getStatement(sql, sqlParams); statement = this._zotero.DB.getStatement(sql, sqlParams);
break; break;
@ -181,9 +196,9 @@ ZoteroAutoComplete.prototype.startSearch = function(searchString, searchParam, p
break; break;
default: default:
var fieldID = this._zotero.ItemFields.getID(searchParam); var fieldID = this._zotero.ItemFields.getID(fieldName);
if (!fieldID) { if (!fieldID) {
this._zotero.debug("'" + searchParam + "' is not a valid autocomplete scope", 1); this._zotero.debug("'" + fieldName + "' is not a valid autocomplete scope", 1);
this.updateResults([], false, Ci.nsIAutoCompleteResult.RESULT_IGNORED); this.updateResults([], false, Ci.nsIAutoCompleteResult.RESULT_IGNORED);
return; return;
} }
@ -191,7 +206,7 @@ ZoteroAutoComplete.prototype.startSearch = function(searchString, searchParam, p
// We don't use date autocomplete anywhere, but if we're not // We don't use date autocomplete anywhere, but if we're not
// disallowing it altogether, we should at least do it right and // disallowing it altogether, we should at least do it right and
// use the user part of the multipart field // use the user part of the multipart field
var valueField = searchParam=='date' ? 'SUBSTR(value, 12, 100)' : 'value'; var valueField = fieldName == 'date' ? 'SUBSTR(value, 12, 100)' : 'value';
var sql = "SELECT DISTINCT " + valueField + " AS val, NULL AS comment " var sql = "SELECT DISTINCT " + valueField + " AS val, NULL AS comment "
+ "FROM itemData NATURAL JOIN itemDataValues " + "FROM itemData NATURAL JOIN itemDataValues "
@ -199,10 +214,10 @@ ZoteroAutoComplete.prototype.startSearch = function(searchString, searchParam, p
+ " LIKE ?2 " + " LIKE ?2 "
var sqlParams = [fieldID, searchString + '%']; var sqlParams = [fieldID, searchString + '%'];
if (extra){ if (searchParams.itemID) {
sql += "AND value NOT IN (SELECT value FROM itemData " sql += "AND value NOT IN (SELECT value FROM itemData "
+ "NATURAL JOIN itemDataValues WHERE fieldID=?1 AND itemID=?3) "; + "NATURAL JOIN itemDataValues WHERE fieldID=?1 AND itemID=?3) ";
sqlParams.push(extra); sqlParams.push(searchParams.itemID);
} }
sql += "ORDER BY value"; sql += "ORDER BY value";
statement = this._zotero.DB.getStatement(sql, sqlParams); statement = this._zotero.DB.getStatement(sql, sqlParams);