forked from mirrors/gecko-dev
		
	 f1b3d2fa50
			
		
	
	
		f1b3d2fa50
		
	
	
	
	
		
			
			lit is a library for building Web Components. It is released under the BSD 3-Clause license. Website: https://lit.dev/ GitHub: https://github.com/lit/lit/ Differential Revision: https://phabricator.services.mozilla.com/D159765
		
			
				
	
	
		
			144 lines
		
	
	
	
		
			4 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			144 lines
		
	
	
	
		
			4 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/. */
 | |
| 
 | |
| import { LitElement } from "./vendor/lit.all.mjs";
 | |
| 
 | |
| /**
 | |
|  * Helper for our replacement of @query. Used with `static queries` property.
 | |
|  *
 | |
|  * https://github.com/lit/lit/blob/main/packages/reactive-element/src/decorators/query.ts
 | |
|  */
 | |
| function query(el, selector) {
 | |
|   return () => el.renderRoot.querySelector(selector);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Helper for our replacement of @queryAll. Used with `static queries` property.
 | |
|  *
 | |
|  * https://github.com/lit/lit/blob/main/packages/reactive-element/src/decorators/query-all.ts
 | |
|  */
 | |
| function queryAll(el, selector) {
 | |
|   return () => el.renderRoot.querySelectorAll(selector);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * MozLitElement provides extensions to the lit-provided LitElement class.
 | |
|  *
 | |
|  *******
 | |
|  *
 | |
|  * `@query` support (define a getter for a querySelector):
 | |
|  *
 | |
|  * static get queries() {
 | |
|  *   return {
 | |
|  *     propertyName: ".aNormal .cssSelector",
 | |
|  *     anotherName: { all: ".selectorFor .querySelectorAll" },
 | |
|  *   };
 | |
|  * }
 | |
|  *
 | |
|  * This example would add properties that would be written like this without
 | |
|  * using `queries`:
 | |
|  *
 | |
|  * get propertyName() {
 | |
|  *   return this.renderRoot?.querySelector(".aNormal .cssSelector");
 | |
|  * }
 | |
|  *
 | |
|  * get anotherName() {
 | |
|  *   return this.renderRoot?.querySelectorAll(".selectorFor .querySelectorAll");
 | |
|  * }
 | |
|  *******
 | |
|  *
 | |
|  * Automatic Fluent support for shadow DOM.
 | |
|  *
 | |
|  * Fluent requires that a shadowRoot be connected before it can use Fluent.
 | |
|  * Shadow roots will get connected automatically.
 | |
|  *
 | |
|  *******
 | |
|  *
 | |
|  * Test helper for sending events after a change: `dispatchOnUpdateComplete`
 | |
|  *
 | |
|  * When some async stuff is going on and you want to wait for it in a test, you
 | |
|  * can use `this.dispatchOnUpdateComplete(myEvent)` and have the test wait on
 | |
|  * your event.
 | |
|  *
 | |
|  * The component will then wait for your reactive property change to take effect
 | |
|  * and dispatch the desired event.
 | |
|  *
 | |
|  * Example:
 | |
|  *
 | |
|  * async onClick() {
 | |
|  *   let response = await this.getServerResponse(this.data);
 | |
|  *   // Show the response status to the user.
 | |
|  *   this.responseStatus = respose.status;
 | |
|  *   this.dispatchOnUpdateComplete(
 | |
|  *     new CustomEvent("status-shown")
 | |
|  *   );
 | |
|  * }
 | |
|  *
 | |
|  * add_task(async testButton() {
 | |
|  *   let button = this.setupAndGetButton();
 | |
|  *   button.click();
 | |
|  *   await BrowserTestUtils.waitForEvent(button, "status-shown");
 | |
|  * });
 | |
|  */
 | |
| export class MozLitElement extends LitElement {
 | |
|   constructor() {
 | |
|     super();
 | |
|     let { queries } = this.constructor;
 | |
|     if (queries) {
 | |
|       for (let [name, selector] of Object.entries(queries)) {
 | |
|         if (selector.all) {
 | |
|           Object.defineProperty(this, name, {
 | |
|             get: queryAll(this, selector.all),
 | |
|           });
 | |
|         } else {
 | |
|           Object.defineProperty(this, name, {
 | |
|             get: query(this, selector),
 | |
|           });
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * The URL for this component's styles. To make development in Storybook
 | |
|    * easier this will use the chrome:// URL when in product (feature detected
 | |
|    * by AppConstants existing) and a relative path for Storybook.
 | |
|    *
 | |
|    * LOCAL_NAME should be the kebab-cased name of the element. It is added by
 | |
|    * the `./mach addwidget` command.
 | |
|    */
 | |
|   static get stylesheetUrl() {
 | |
|     if (this.useChromeStylesheet) {
 | |
|       return `chrome://global/content/elements/${this.LOCAL_NAME}.css`;
 | |
|     }
 | |
|     return `./${this.LOCAL_NAME}/${this.LOCAL_NAME}.css`;
 | |
|   }
 | |
| 
 | |
|   static get useChromeStylesheet() {
 | |
|     return (
 | |
|       typeof AppConstants != "undefined" ||
 | |
|       (typeof Cu != "undefined" && Cu.isInAutomation)
 | |
|     );
 | |
|   }
 | |
| 
 | |
|   connectedCallback() {
 | |
|     super.connectedCallback();
 | |
|     if (!this._l10nRootConnected && document.l10n) {
 | |
|       document.l10n.connectRoot(this.renderRoot);
 | |
|       this._l10nRootConnected = true;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   async dispatchOnUpdateComplete(event) {
 | |
|     await this.updateComplete;
 | |
|     this.dispatchEvent(event);
 | |
|   }
 | |
| 
 | |
|   update() {
 | |
|     super.update();
 | |
|     if (document.l10n) {
 | |
|       document.l10n.translateFragment(this.renderRoot);
 | |
|     }
 | |
|   }
 | |
| }
 |