mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
550 lines
19 KiB
JavaScript
550 lines
19 KiB
JavaScript
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
||
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
||
|
|
||
|
"use strict";
|
||
|
|
||
|
const Cu = Components.utils;
|
||
|
|
||
|
Cu.import("resource://gre/modules/XPCOMUtils.jsm");
|
||
|
|
||
|
XPCOMUtils.defineLazyModuleGetter(this, "AutocompletePopup",
|
||
|
"resource:///modules/devtools/AutocompletePopup.jsm");
|
||
|
this.EXPORTED_SYMBOLS = ["SelectorSearch"];
|
||
|
|
||
|
// Maximum number of selector suggestions shown in the panel.
|
||
|
const MAX_SUGGESTIONS = 15;
|
||
|
|
||
|
/**
|
||
|
* Converts any input box on a page to a CSS selector search and suggestion box.
|
||
|
*
|
||
|
* @constructor
|
||
|
* @param nsIDOMDocument aContentDocument
|
||
|
* The content document which inspector is attached to.
|
||
|
* @param nsiInputElement aInputNode
|
||
|
* The input element to which the panel will be attached and from where
|
||
|
* search input will be taken.
|
||
|
* @param Function aCallback
|
||
|
* The method to callback when a search is available.
|
||
|
* This method is called with the matched node as the first argument.
|
||
|
*/
|
||
|
this.SelectorSearch = function(aContentDocument, aInputNode, aCallback) {
|
||
|
this.doc = aContentDocument;
|
||
|
this.callback = aCallback;
|
||
|
this.searchBox = aInputNode;
|
||
|
this.panelDoc = this.searchBox.ownerDocument;
|
||
|
|
||
|
// initialize variables.
|
||
|
this._lastSearched = null;
|
||
|
this._lastValidSearch = "";
|
||
|
this._lastToLastValidSearch = null;
|
||
|
this._searchResults = null;
|
||
|
this._searchSuggestions = {};
|
||
|
this._searchIndex = 0;
|
||
|
|
||
|
// bind!
|
||
|
this._showPopup = this._showPopup.bind(this);
|
||
|
this._onHTMLSearch = this._onHTMLSearch.bind(this);
|
||
|
this._onSearchKeypress = this._onSearchKeypress.bind(this);
|
||
|
this._onListBoxKeypress = this._onListBoxKeypress.bind(this);
|
||
|
|
||
|
// Options for the AutocompletePopup.
|
||
|
let options = {
|
||
|
panelId: "inspector-searchbox-panel",
|
||
|
listBoxId: "searchbox-panel-listbox",
|
||
|
fixedWidth: true,
|
||
|
autoSelect: true,
|
||
|
position: "before_start",
|
||
|
direction: "ltr",
|
||
|
onClick: this._onListBoxKeypress,
|
||
|
onKeypress: this._onListBoxKeypress,
|
||
|
};
|
||
|
this.searchPopup = new AutocompletePopup(this.panelDoc, options);
|
||
|
|
||
|
// event listeners.
|
||
|
this.searchBox.addEventListener("command", this._onHTMLSearch, true);
|
||
|
this.searchBox.addEventListener("keypress", this._onSearchKeypress, true);
|
||
|
}
|
||
|
|
||
|
this.SelectorSearch.prototype = {
|
||
|
|
||
|
// The possible states of the query.
|
||
|
States: {
|
||
|
CLASS: "class",
|
||
|
ID: "id",
|
||
|
TAG: "tag",
|
||
|
},
|
||
|
|
||
|
// The current state of the query.
|
||
|
_state: null,
|
||
|
|
||
|
// The query corresponding to last state computation.
|
||
|
_lastStateCheckAt: null,
|
||
|
|
||
|
/**
|
||
|
* Computes the state of the query. State refers to whether the query
|
||
|
* currently requires a class suggestion, or a tag, or an Id suggestion.
|
||
|
* This getter will effectively compute the state by traversing the query
|
||
|
* character by character each time the query changes.
|
||
|
*
|
||
|
* @example
|
||
|
* '#f' requires an Id suggestion, so the state is States.ID
|
||
|
* 'div > .foo' requires class suggestion, so state is States.CLASS
|
||
|
*/
|
||
|
get state() {
|
||
|
if (!this.searchBox || !this.searchBox.value) {
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
let query = this.searchBox.value;
|
||
|
if (this._lastStateCheckAt == query) {
|
||
|
// If query is the same, return early.
|
||
|
return this._state;
|
||
|
}
|
||
|
this._lastStateCheckAt = query;
|
||
|
|
||
|
this._state = null;
|
||
|
let subQuery = "";
|
||
|
// Now we iterate over the query and decide the state character by character.
|
||
|
// The logic here is that while iterating, the state can go from one to
|
||
|
// another with some restrictions. Like, if the state is Class, then it can
|
||
|
// never go to Tag state without a space or '>' character; Or like, a Class
|
||
|
// state with only '.' cannot go to an Id state without any [a-zA-Z] after
|
||
|
// the '.' which means that '.#' is a selector matching a class name '#'.
|
||
|
// Similarily for '#.' which means a selctor matching an id '.'.
|
||
|
for (let i = 1; i <= query.length; i++) {
|
||
|
// Calculate the state.
|
||
|
subQuery = query.slice(0, i);
|
||
|
let [secondLastChar, lastChar] = subQuery.slice(-2);
|
||
|
switch (this._state) {
|
||
|
case null:
|
||
|
// This will happen only in the first iteration of the for loop.
|
||
|
lastChar = secondLastChar;
|
||
|
case this.States.TAG:
|
||
|
this._state = lastChar == "."
|
||
|
? this.States.CLASS
|
||
|
: lastChar == "#"
|
||
|
? this.States.ID
|
||
|
: this.States.TAG;
|
||
|
break;
|
||
|
|
||
|
case this.States.CLASS:
|
||
|
if (subQuery.match(/[\.]+[^\.]*$/)[0].length > 2) {
|
||
|
// Checks whether the subQuery has atleast one [a-zA-Z] after the '.'.
|
||
|
this._state = (lastChar == " " || lastChar == ">")
|
||
|
? this.States.TAG
|
||
|
: lastChar == "#"
|
||
|
? this.States.ID
|
||
|
: this.States.CLASS;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case this.States.ID:
|
||
|
if (subQuery.match(/[#]+[^#]*$/)[0].length > 2) {
|
||
|
// Checks whether the subQuery has atleast one [a-zA-Z] after the '#'.
|
||
|
this._state = (lastChar == " " || lastChar == ">")
|
||
|
? this.States.TAG
|
||
|
: lastChar == "."
|
||
|
? this.States.CLASS
|
||
|
: this.States.ID;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
return this._state;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Removes event listeners and cleans up references.
|
||
|
*/
|
||
|
destroy: function SelectorSearch_destroy() {
|
||
|
// event listeners.
|
||
|
this.searchBox.removeEventListener("command", this._onHTMLSearch, true);
|
||
|
this.searchBox.removeEventListener("keypress", this._onSearchKeypress, true);
|
||
|
this.searchPopup.destroy();
|
||
|
this.searchPopup = null;
|
||
|
this.searchBox = null;
|
||
|
this.doc = null;
|
||
|
this.panelDoc = null;
|
||
|
this._searchResults = null;
|
||
|
this._searchSuggestions = null;
|
||
|
this.callback = null;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* The command callback for the input box. This function is automatically
|
||
|
* invoked as the user is typing if the input box type is search.
|
||
|
*/
|
||
|
_onHTMLSearch: function SelectorSearch__onHTMLSearch() {
|
||
|
let query = this.searchBox.value;
|
||
|
if (query == this._lastSearched) {
|
||
|
return;
|
||
|
}
|
||
|
this._lastSearched = query;
|
||
|
this._searchIndex = 0;
|
||
|
|
||
|
if (query.length == 0) {
|
||
|
this._lastValidSearch = "";
|
||
|
this.searchBox.removeAttribute("filled");
|
||
|
this.searchBox.classList.remove("devtools-no-search-result");
|
||
|
if (this.searchPopup.isOpen) {
|
||
|
this.searchPopup.hidePopup();
|
||
|
}
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
this.searchBox.setAttribute("filled", true);
|
||
|
try {
|
||
|
this._searchResults = this.doc.querySelectorAll(query);
|
||
|
}
|
||
|
catch (ex) {
|
||
|
this._searchResults = [];
|
||
|
}
|
||
|
if (this._searchResults.length > 0) {
|
||
|
this._lastValidSearch = query;
|
||
|
// Even though the selector matched atleast one node, there is still
|
||
|
// possibility of suggestions.
|
||
|
if (query.match(/[\s>+]$/)) {
|
||
|
// If the query has a space or '>' at the end, create a selector to match
|
||
|
// the children of the selector inside the search box by adding a '*'.
|
||
|
this._lastValidSearch += "*";
|
||
|
}
|
||
|
else if (query.match(/[\s>+][\.#a-zA-Z][\.#>\s+]*$/)) {
|
||
|
// If the query is a partial descendant selector which does not matches
|
||
|
// any node, remove the last incomplete part and add a '*' to match
|
||
|
// everything. For ex, convert 'foo > b' to 'foo > *' .
|
||
|
let lastPart = query.match(/[\s>+][\.#a-zA-Z][^>\s+]*$/)[0];
|
||
|
this._lastValidSearch = query.slice(0, -1 * lastPart.length + 1) + "*";
|
||
|
}
|
||
|
|
||
|
if (!query.slice(-1).match(/[\.#\s>+]/)) {
|
||
|
// Hide the popup if we have some matching nodes and the query is not
|
||
|
// ending with [.# >] which means that the selector is not at the
|
||
|
// beginning of a new class, tag or id.
|
||
|
if (this.searchPopup.isOpen) {
|
||
|
this.searchPopup.hidePopup();
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
this.showSuggestions();
|
||
|
}
|
||
|
this.searchBox.classList.remove("devtools-no-search-result");
|
||
|
this.callback(this._searchResults[0]);
|
||
|
}
|
||
|
else {
|
||
|
if (query.match(/[\s>+]$/)) {
|
||
|
this._lastValidSearch = query + "*";
|
||
|
}
|
||
|
else if (query.match(/[\s>+][\.#a-zA-Z][\.#>\s+]*$/)) {
|
||
|
let lastPart = query.match(/[\s+>][\.#a-zA-Z][^>\s+]*$/)[0];
|
||
|
this._lastValidSearch = query.slice(0, -1 * lastPart.length + 1) + "*";
|
||
|
}
|
||
|
this.searchBox.classList.add("devtools-no-search-result");
|
||
|
this.showSuggestions();
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Handles keypresses inside the input box.
|
||
|
*/
|
||
|
_onSearchKeypress: function SelectorSearch__onSearchKeypress(aEvent) {
|
||
|
let query = this.searchBox.value;
|
||
|
switch(aEvent.keyCode) {
|
||
|
case aEvent.DOM_VK_ENTER:
|
||
|
case aEvent.DOM_VK_RETURN:
|
||
|
if (query == this._lastSearched) {
|
||
|
this._searchIndex = (this._searchIndex + 1) % this._searchResults.length;
|
||
|
}
|
||
|
else {
|
||
|
this._onHTMLSearch();
|
||
|
return;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case aEvent.DOM_VK_UP:
|
||
|
if (this.searchPopup.isOpen && this.searchPopup.itemCount > 0) {
|
||
|
this.searchPopup.focus();
|
||
|
if (this.searchPopup.selectedIndex == this.searchPopup.itemCount - 1) {
|
||
|
this.searchPopup.selectedIndex =
|
||
|
Math.max(0, this.searchPopup.itemCount - 2);
|
||
|
}
|
||
|
else {
|
||
|
this.searchPopup.selectedIndex = this.searchPopup.itemCount - 1;
|
||
|
}
|
||
|
this.searchBox.value = this.searchPopup.selectedItem.label;
|
||
|
}
|
||
|
else if (--this._searchIndex < 0) {
|
||
|
this._searchIndex = this._searchResults.length - 1;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case aEvent.DOM_VK_DOWN:
|
||
|
if (this.searchPopup.isOpen && this.searchPopup.itemCount > 0) {
|
||
|
this.searchPopup.focus();
|
||
|
this.searchPopup.selectedIndex = 0;
|
||
|
this.searchBox.value = this.searchPopup.selectedItem.label;
|
||
|
}
|
||
|
this._searchIndex = (this._searchIndex + 1) % this._searchResults.length;
|
||
|
break;
|
||
|
|
||
|
case aEvent.DOM_VK_TAB:
|
||
|
if (this.searchPopup.isOpen &&
|
||
|
this.searchPopup.getItemAtIndex(this.searchPopup.itemCount - 1)
|
||
|
.preLabel == query) {
|
||
|
this.searchPopup.selectedIndex = this.searchPopup.itemCount - 1;
|
||
|
this.searchBox.value = this.searchPopup.selectedItem.label;
|
||
|
this._onHTMLSearch();
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case aEvent.DOM_VK_BACK_SPACE:
|
||
|
case aEvent.DOM_VK_DELETE:
|
||
|
// need to throw away the lastValidSearch.
|
||
|
this._lastToLastValidSearch = null;
|
||
|
// This gets the most complete selector from the query. For ex.
|
||
|
// '.foo.ba' returns '.foo' , '#foo > .bar.baz' returns '#foo > .bar'
|
||
|
// '.foo +bar' returns '.foo +' and likewise.
|
||
|
this._lastValidSearch = (query.match(/(.*)[\.#][^\.# ]{0,}$/) ||
|
||
|
query.match(/(.*[\s>+])[a-zA-Z][^\.# ]{0,}$/) ||
|
||
|
["",""])[1];
|
||
|
return;
|
||
|
|
||
|
default:
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
aEvent.preventDefault();
|
||
|
aEvent.stopPropagation();
|
||
|
if (this._searchResults.length > 0) {
|
||
|
this.callback(this._searchResults[this._searchIndex]);
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Handles keypress and mouse click on the suggestions richlistbox.
|
||
|
*/
|
||
|
_onListBoxKeypress: function SelectorSearch__onListBoxKeypress(aEvent) {
|
||
|
switch(aEvent.keyCode || aEvent.button) {
|
||
|
case aEvent.DOM_VK_ENTER:
|
||
|
case aEvent.DOM_VK_RETURN:
|
||
|
case aEvent.DOM_VK_TAB:
|
||
|
case 0: // left mouse button
|
||
|
aEvent.stopPropagation();
|
||
|
aEvent.preventDefault();
|
||
|
this.searchBox.value = this.searchPopup.selectedItem.label;
|
||
|
this.searchBox.focus();
|
||
|
this._onHTMLSearch();
|
||
|
break;
|
||
|
|
||
|
case aEvent.DOM_VK_UP:
|
||
|
if (this.searchPopup.selectedIndex == 0) {
|
||
|
this.searchPopup.selectedIndex = -1;
|
||
|
aEvent.stopPropagation();
|
||
|
aEvent.preventDefault();
|
||
|
this.searchBox.focus();
|
||
|
}
|
||
|
else {
|
||
|
let index = this.searchPopup.selectedIndex;
|
||
|
this.searchBox.value = this.searchPopup.getItemAtIndex(index - 1).label;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case aEvent.DOM_VK_DOWN:
|
||
|
if (this.searchPopup.selectedIndex == this.searchPopup.itemCount - 1) {
|
||
|
this.searchPopup.selectedIndex = -1;
|
||
|
aEvent.stopPropagation();
|
||
|
aEvent.preventDefault();
|
||
|
this.searchBox.focus();
|
||
|
}
|
||
|
else {
|
||
|
let index = this.searchPopup.selectedIndex;
|
||
|
this.searchBox.value = this.searchPopup.getItemAtIndex(index + 1).label;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case aEvent.DOM_VK_BACK_SPACE:
|
||
|
aEvent.stopPropagation();
|
||
|
aEvent.preventDefault();
|
||
|
this.searchBox.focus();
|
||
|
if (this.searchBox.selectionStart > 0) {
|
||
|
this.searchBox.value =
|
||
|
this.searchBox.value.substring(0, this.searchBox.selectionStart - 1);
|
||
|
}
|
||
|
this._lastToLastValidSearch = null;
|
||
|
let query = this.searchBox.value;
|
||
|
this._lastValidSearch = (query.match(/(.*)[\.#][^\.# ]{0,}$/) ||
|
||
|
query.match(/(.*[\s>+])[a-zA-Z][^\.# ]{0,}$/) ||
|
||
|
["",""])[1];
|
||
|
this._onHTMLSearch();
|
||
|
break;
|
||
|
}
|
||
|
},
|
||
|
|
||
|
|
||
|
/**
|
||
|
* Populates the suggestions list and show the suggestion popup.
|
||
|
*/
|
||
|
_showPopup: function SelectorSearch__showPopup(aList, aFirstPart) {
|
||
|
// Sort alphabetically in increaseing order.
|
||
|
aList = aList.sort();
|
||
|
// Sort based on count= in decreasing order.
|
||
|
aList = aList.sort(function([a1,a2], [b1,b2]) {
|
||
|
return a2 < b2;
|
||
|
});
|
||
|
|
||
|
let total = 0;
|
||
|
let query = this.searchBox.value;
|
||
|
let toLowerCase = false;
|
||
|
let items = [];
|
||
|
// In case of tagNames, change the case to small.
|
||
|
if (query.match(/.*[\.#][^\.#]{0,}$/) == null) {
|
||
|
toLowerCase = true;
|
||
|
}
|
||
|
for (let [value, count] of aList) {
|
||
|
// for cases like 'div ' or 'div >' or 'div+'
|
||
|
if (query.match(/[\s>+]$/)) {
|
||
|
value = query + value;
|
||
|
}
|
||
|
// for cases like 'div #a' or 'div .a' or 'div > d' and likewise
|
||
|
else if (query.match(/[\s>+][\.#a-zA-Z][^\s>+\.#]*$/)) {
|
||
|
let lastPart = query.match(/[\s>+][\.#a-zA-Z][^>\s+\.#]*$/)[0];
|
||
|
value = query.slice(0, -1 * lastPart.length + 1) + value;
|
||
|
}
|
||
|
// for cases like 'div.class' or '#foo.bar' and likewise
|
||
|
else if (query.match(/[a-zA-Z][#\.][^#\.\s+>]*$/)) {
|
||
|
let lastPart = query.match(/[a-zA-Z][#\.][^#\.\s>+]*$/)[0];
|
||
|
value = query.slice(0, -1 * lastPart.length + 1) + value;
|
||
|
}
|
||
|
let item = {
|
||
|
preLabel: query,
|
||
|
label: value,
|
||
|
count: count
|
||
|
};
|
||
|
if (toLowerCase) {
|
||
|
item.label = value.toLowerCase();
|
||
|
}
|
||
|
items.unshift(item);
|
||
|
if (++total > MAX_SUGGESTIONS - 1) {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (total > 0) {
|
||
|
this.searchPopup.setItems(items);
|
||
|
this.searchPopup.openPopup(this.searchBox);
|
||
|
}
|
||
|
else {
|
||
|
this.searchPopup.hidePopup();
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Suggests classes,ids and tags based on the user input as user types in the
|
||
|
* searchbox.
|
||
|
*/
|
||
|
showSuggestions: function SelectorSearch_showSuggestions() {
|
||
|
let query = this.searchBox.value;
|
||
|
if (this._lastValidSearch != "" &&
|
||
|
this._lastToLastValidSearch != this._lastValidSearch) {
|
||
|
this._searchSuggestions = {
|
||
|
ids: new Map(),
|
||
|
classes: new Map(),
|
||
|
tags: new Map(),
|
||
|
};
|
||
|
|
||
|
let nodes = [];
|
||
|
try {
|
||
|
nodes = this.doc.querySelectorAll(this._lastValidSearch);
|
||
|
} catch (ex) {}
|
||
|
for (let node of nodes) {
|
||
|
this._searchSuggestions.ids.set(node.id, 1);
|
||
|
this._searchSuggestions.tags
|
||
|
.set(node.tagName,
|
||
|
(this._searchSuggestions.tags.get(node.tagName) || 0) + 1);
|
||
|
for (let className of node.classList) {
|
||
|
this._searchSuggestions.classes
|
||
|
.set(className,
|
||
|
(this._searchSuggestions.classes.get(className) || 0) + 1);
|
||
|
}
|
||
|
}
|
||
|
this._lastToLastValidSearch = this._lastValidSearch;
|
||
|
}
|
||
|
else if (this._lastToLastValidSearch != this._lastValidSearch) {
|
||
|
this._searchSuggestions = {
|
||
|
ids: new Map(),
|
||
|
classes: new Map(),
|
||
|
tags: new Map(),
|
||
|
};
|
||
|
|
||
|
if (query.length == 0) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
let nodes = null;
|
||
|
if (this.state == this.States.CLASS) {
|
||
|
nodes = this.doc.querySelectorAll("[class]");
|
||
|
for (let node of nodes) {
|
||
|
for (let className of node.classList) {
|
||
|
this._searchSuggestions.classes
|
||
|
.set(className,
|
||
|
(this._searchSuggestions.classes.get(className) || 0) + 1);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else if (this.state == this.States.ID) {
|
||
|
nodes = this.doc.querySelectorAll("[id]");
|
||
|
for (let node of nodes) {
|
||
|
this._searchSuggestions.ids.set(node.id, 1);
|
||
|
}
|
||
|
}
|
||
|
else if (this.state == this.States.TAG) {
|
||
|
nodes = this.doc.getElementsByTagName("*");
|
||
|
for (let node of nodes) {
|
||
|
this._searchSuggestions.tags
|
||
|
.set(node.tagName,
|
||
|
(this._searchSuggestions.tags.get(node.tagName) || 0) + 1);
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
return;
|
||
|
}
|
||
|
this._lastToLastValidSearch = this._lastValidSearch;
|
||
|
}
|
||
|
|
||
|
// Filter the suggestions based on search box value.
|
||
|
let result = [];
|
||
|
let firstPart = "";
|
||
|
if (this.state == this.States.TAG) {
|
||
|
// gets the tag that is being completed. For ex. 'div.foo > s' returns 's',
|
||
|
// 'di' returns 'di' and likewise.
|
||
|
firstPart = (query.match(/[\s>+]?([a-zA-Z]*)$/) || ["",query])[1];
|
||
|
for (let [tag, count] of this._searchSuggestions.tags) {
|
||
|
if (tag.toLowerCase().startsWith(firstPart.toLowerCase())) {
|
||
|
result.push([tag, count]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else if (this.state == this.States.CLASS) {
|
||
|
// gets the class that is being completed. For ex. '.foo.b' returns 'b'
|
||
|
firstPart = query.match(/\.([^\.]*)$/)[1];
|
||
|
for (let [className, count] of this._searchSuggestions.classes) {
|
||
|
if (className.startsWith(firstPart)) {
|
||
|
result.push(["." + className, count]);
|
||
|
}
|
||
|
}
|
||
|
firstPart = "." + firstPart;
|
||
|
}
|
||
|
else if (this.state == this.States.ID) {
|
||
|
// gets the id that is being completed. For ex. '.foo#b' returns 'b'
|
||
|
firstPart = query.match(/#([^#]*)$/)[1];
|
||
|
for (let [id, count] of this._searchSuggestions.ids) {
|
||
|
if (id.startsWith(firstPart)) {
|
||
|
result.push(["#" + id, 1]);
|
||
|
}
|
||
|
}
|
||
|
firstPart = "#" + firstPart;
|
||
|
}
|
||
|
|
||
|
this._showPopup(result, firstPart);
|
||
|
},
|
||
|
};
|