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:
parent
8469ce821b
commit
65c7a5988b
|
@ -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 = {
|
||||||
|
fieldName: fieldName,
|
||||||
|
libraryID: this.item.libraryID
|
||||||
|
};
|
||||||
if (field == 'creator') {
|
if (field == 'creator') {
|
||||||
suffix = elem.getAttribute('fieldMode') + '-' + suffix;
|
params.fieldMode = parseInt(elem.getAttribute('fieldMode'));
|
||||||
}
|
params.itemID = itemID ? itemID : '';
|
||||||
t.setAttribute('autocompletesearchparam', fieldName + '/' + suffix);
|
};
|
||||||
|
t.setAttribute(
|
||||||
|
'autocompletesearchparam', JSON.stringify(params)
|
||||||
|
);
|
||||||
t.setAttribute('ontextentered',
|
t.setAttribute('ontextentered',
|
||||||
'document.getBindingParent(this).handleCreatorAutoCompleteSelect(this)');
|
'document.getBindingParent(this).handleCreatorAutoCompleteSelect(this)');
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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');
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
|
|
Loading…
Reference in New Issue
Block a user