forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			234 lines
		
	
	
	
		
			7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			234 lines
		
	
	
	
		
			7 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/. */
 | |
| 
 | |
| /**
 | |
|  * This module exports a provider that offers token alias engines.
 | |
|  */
 | |
| 
 | |
| import {
 | |
|   UrlbarProvider,
 | |
|   UrlbarUtils,
 | |
| } from "resource:///modules/UrlbarUtils.sys.mjs";
 | |
| 
 | |
| const lazy = {};
 | |
| 
 | |
| ChromeUtils.defineESModuleGetters(lazy, {
 | |
|   UrlbarPrefs: "resource:///modules/UrlbarPrefs.sys.mjs",
 | |
|   UrlbarResult: "resource:///modules/UrlbarResult.sys.mjs",
 | |
|   UrlbarSearchUtils: "resource:///modules/UrlbarSearchUtils.sys.mjs",
 | |
|   UrlbarTokenizer: "resource:///modules/UrlbarTokenizer.sys.mjs",
 | |
| });
 | |
| 
 | |
| /**
 | |
|  * Class used to create the provider.
 | |
|  */
 | |
| class ProviderTokenAliasEngines extends UrlbarProvider {
 | |
|   constructor() {
 | |
|     super();
 | |
|     this._engines = [];
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Returns the name of this provider.
 | |
|    *
 | |
|    * @returns {string} the name of this provider.
 | |
|    */
 | |
|   get name() {
 | |
|     return "TokenAliasEngines";
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Returns the type of this provider.
 | |
|    *
 | |
|    * @returns {integer} one of the types from UrlbarUtils.PROVIDER_TYPE.*
 | |
|    */
 | |
|   get type() {
 | |
|     return UrlbarUtils.PROVIDER_TYPE.HEURISTIC;
 | |
|   }
 | |
| 
 | |
|   get PRIORITY() {
 | |
|     // Beats UrlbarProviderSearchSuggestions and UrlbarProviderPlaces.
 | |
|     return 1;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Whether this provider should be invoked for the given context.
 | |
|    * If this method returns false, the providers manager won't start a query
 | |
|    * with this provider, to save on resources.
 | |
|    *
 | |
|    * @param {UrlbarQueryContext} queryContext The query context object
 | |
|    * @returns {boolean} Whether this provider should be invoked for the search.
 | |
|    */
 | |
|   async isActive(queryContext) {
 | |
|     let instance = this.queryInstance;
 | |
| 
 | |
|     // This is usually reset on canceling or completing the query, but since we
 | |
|     // query in isActive, it may not have been canceled by the previous call.
 | |
|     // It is an object with values { result: UrlbarResult, instance: Query }.
 | |
|     this._autofillData = null;
 | |
| 
 | |
|     // Once the user starts typing a search string after the token, we hand off
 | |
|     // suggestions to UrlbarProviderSearchSuggestions.
 | |
|     if (
 | |
|       !queryContext.searchString.startsWith("@") ||
 | |
|       queryContext.tokens.length != 1
 | |
|     ) {
 | |
|       return false;
 | |
|     }
 | |
| 
 | |
|     // Do not show token alias results in search mode.
 | |
|     if (queryContext.searchMode) {
 | |
|       return false;
 | |
|     }
 | |
| 
 | |
|     this._engines = await lazy.UrlbarSearchUtils.tokenAliasEngines();
 | |
|     if (!this._engines.length) {
 | |
|       return false;
 | |
|     }
 | |
| 
 | |
|     // Check the query was not canceled while this executed.
 | |
|     if (instance != this.queryInstance) {
 | |
|       return false;
 | |
|     }
 | |
| 
 | |
|     if (queryContext.trimmedSearchString == "@") {
 | |
|       return true;
 | |
|     }
 | |
| 
 | |
|     // If the user is typing a potential engine name, autofill it.
 | |
|     if (lazy.UrlbarPrefs.get("autoFill") && queryContext.allowAutofill) {
 | |
|       let result = this._getAutofillResult(queryContext);
 | |
|       if (result) {
 | |
|         this._autofillData = { result, instance };
 | |
|         return true;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Starts querying.
 | |
|    *
 | |
|    * @param {object} queryContext The query context object
 | |
|    * @param {Function} addCallback Callback invoked by the provider to add a new
 | |
|    *        result.
 | |
|    */
 | |
|   async startQuery(queryContext, addCallback) {
 | |
|     if (!this._engines || !this._engines.length) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     if (
 | |
|       this._autofillData &&
 | |
|       this._autofillData.instance == this.queryInstance
 | |
|     ) {
 | |
|       addCallback(this, this._autofillData.result);
 | |
|     }
 | |
| 
 | |
|     for (let { engine, tokenAliases } of this._engines) {
 | |
|       if (
 | |
|         tokenAliases[0].startsWith(queryContext.trimmedSearchString) &&
 | |
|         engine.name != this._autofillData?.result.payload.engine
 | |
|       ) {
 | |
|         let result = new lazy.UrlbarResult(
 | |
|           UrlbarUtils.RESULT_TYPE.SEARCH,
 | |
|           UrlbarUtils.RESULT_SOURCE.SEARCH,
 | |
|           ...lazy.UrlbarResult.payloadAndSimpleHighlights(queryContext.tokens, {
 | |
|             engine: [engine.name, UrlbarUtils.HIGHLIGHT.TYPED],
 | |
|             keyword: [tokenAliases[0], UrlbarUtils.HIGHLIGHT.TYPED],
 | |
|             query: ["", UrlbarUtils.HIGHLIGHT.TYPED],
 | |
|             icon: engine.getIconURL(),
 | |
|             providesSearchMode: true,
 | |
|           })
 | |
|         );
 | |
|         addCallback(this, result);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     this._autofillData = null;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Gets the provider's priority.
 | |
|    *
 | |
|    * @param {UrlbarQueryContext} queryContext The query context object
 | |
|    * @returns {number} The provider's priority for the given query.
 | |
|    */
 | |
|   getPriority(queryContext) {
 | |
|     return this.PRIORITY;
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * Cancels a running query.
 | |
|    *
 | |
|    * @param {object} queryContext The query context object
 | |
|    */
 | |
|   cancelQuery(queryContext) {
 | |
|     if (this._autofillData?.instance == this.queryInstance) {
 | |
|       this._autofillData = null;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   _getAutofillResult(queryContext) {
 | |
|     let lowerCaseSearchString = queryContext.searchString.toLowerCase();
 | |
| 
 | |
|     // The user is typing a specific engine. We should show a heuristic result.
 | |
|     for (let { engine, tokenAliases } of this._engines) {
 | |
|       for (let alias of tokenAliases) {
 | |
|         if (alias.startsWith(lowerCaseSearchString)) {
 | |
|           // We found the engine.
 | |
| 
 | |
|           // Stop adding an autofill result once the user has typed the full
 | |
|           // alias followed by a space. We enter search mode at that point.
 | |
|           if (
 | |
|             lowerCaseSearchString.startsWith(alias) &&
 | |
|             lazy.UrlbarTokenizer.REGEXP_SPACES_START.test(
 | |
|               lowerCaseSearchString.substring(alias.length)
 | |
|             )
 | |
|           ) {
 | |
|             return null;
 | |
|           }
 | |
| 
 | |
|           // Add an autofill result.  Append a space so the user can hit enter
 | |
|           // or the right arrow key and immediately start typing their query.
 | |
|           let aliasPreservingUserCase =
 | |
|             queryContext.searchString +
 | |
|             alias.substr(queryContext.searchString.length);
 | |
|           let value = aliasPreservingUserCase + " ";
 | |
|           let result = new lazy.UrlbarResult(
 | |
|             UrlbarUtils.RESULT_TYPE.SEARCH,
 | |
|             UrlbarUtils.RESULT_SOURCE.SEARCH,
 | |
|             ...lazy.UrlbarResult.payloadAndSimpleHighlights(
 | |
|               queryContext.tokens,
 | |
|               {
 | |
|                 engine: [engine.name, UrlbarUtils.HIGHLIGHT.TYPED],
 | |
|                 keyword: [aliasPreservingUserCase, UrlbarUtils.HIGHLIGHT.TYPED],
 | |
|                 query: ["", UrlbarUtils.HIGHLIGHT.TYPED],
 | |
|                 icon: engine.getIconURL(),
 | |
|                 providesSearchMode: true,
 | |
|               }
 | |
|             )
 | |
|           );
 | |
| 
 | |
|           // We set suggestedIndex = 0 instead of the heuristic because we
 | |
|           // don't want this result to be automatically selected. That way,
 | |
|           // users can press Tab to select the result, building on their
 | |
|           // muscle memory from tab-to-search.
 | |
|           result.suggestedIndex = 0;
 | |
| 
 | |
|           result.autofill = {
 | |
|             value,
 | |
|             selectionStart: queryContext.searchString.length,
 | |
|             selectionEnd: value.length,
 | |
|           };
 | |
|           return result;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|     return null;
 | |
|   }
 | |
| }
 | |
| 
 | |
| export var UrlbarProviderTokenAliasEngines = new ProviderTokenAliasEngines();
 | 
