Skip to content

Latest commit

 

History

History
760 lines (712 loc) · 49.2 KB

README.md

File metadata and controls

760 lines (712 loc) · 49.2 KB

cristina10

A multi-project repo for TypeScript web server, front-end and database applications (Chirichella)

litclock/litimer

is an application to show some of the interesting and important aspects of web component development with typescript including:

  • Creating a new web project in seconds with the fantastic Vite using its Lit template in seconds.
  • Making a not so trivial Timer web component with Lit and JavaScript iterator/generator function.
  • Embedding a web component into the slots of the shadow DOM of another web component.

This project comes with a series of explanation and demonstration videos.

Part 1: Creating a web component project with Vite in a minute.

Part 2: The Anatomy of a Web Component: Counter

Watch Web Components Getting Started - 02 The Anatomy of a Web Component: Counter

Part 3: Crafting a not so Simple Web Component from Scratch in 5 minutes

Part 4: The Anatomy of a Web Component: Timer

  • Watch Web Components Getting Started - 04 Anatomy of Another Web Component - Timer the explanation of the error I had made in the previous video.
  • As explained in Iterators and Generators the function* syntax is a syntax sugar to create iterator objects where the yield expression emits a value for the next call. In my example the Lit directive asyncReplace works with the iterator/generator: html<slot>My Timer:</slot><span>${asyncReplace(this.timer)}</span> <slot name=after></slot>
  • The anatomy of the Timer web component:
@customElement('my-timer')
class MyTimer extends LitElement {
  @property({type:Boolean}) run = false
  async *countUp() {
    const start = Date.now()
    while (this.run) {
      const now = Date.now()
      yield now - start
      await new Promise((r) => setTimeout(r, 100))
    }
  }
  override attributeChangedCallback(name:string,old:string,value:string){
    super.attributeChangedCallback(name,old,value)
    if(this.run) this.timer = this.countUp()
  }
  @state() private timer = this.countUp()
  render() {
    return html`<slot>My Timer:</slot><span>${asyncReplace(this.timer)}</span><slot name=after></slot>`
  }
}
declare global { interface HTMLElementTagNameMap {'my-timer': MyTimer}}
  • @customElement decorator will automatically call window.cusomElements.define needed to register a web component.
  • run attribute is a tool for the outside world to start and stop the timer.
  • async *countUp() is the infinite looping ticking engine (an iterator generator method) to yield a value ten times a second. The value emitted is in milliseconds. The engine stops, when the run attribute is changed to false. Techically *countUp returns an iterator object from which to get the values yielded. This iterator object is kept in the this.timer member.
  • attributeChangedCallback is a standard browser-provided callback function for custom elements when an attribute is changed. LitElement is just a thin but brutally convenient layer to make custom element development a fun and joy. When the run property is changed, another generator instance (this.timer) is generated and the countUp pumping engine is restarted.
  • @state() private timer = this.countUp() is the actual generator object to get the next values from.
  • html ... tagged template literal function is the heart of Lit technology providing a diffing engine to make reactive rendering possible
  • <slot>My Timer:</slot> is a customizable label for the component. Slots can be replaced when the component is used on the web page like so <my-timer run><b>The Job is Running for:</b></my-timer>
  • The directice asyncReplace in ${asyncReplace(this.timer)} is a brutally powerful function from the Lit toolset import {asyncReplace} from "lit/directives/async-replace.js" to work with async iterator generators by reading the next values yielded by the iterator object this.timer in our case here created with a generator function.
  • <slot name=after></slot> is a named slot to make customizing the display of the web component more flexible. slots and shadow DOM are provided by the browser.

Part 5 and 6: Almost Everything You should Know About Slots and Shadow DOM

  querySlotElementAll(slot:string,name:string):[HTMLElement | null]  {
    const slotElement = this.shadowRoot?.querySelector(slot ? `slot[name=${slot}]` : `slot`) as HTMLSlotElement
    const slotChildren = slotElement.assignedElements()
    const selectedElements = slotChildren?.map((e) => e.nodeName == name.toUpperCase() ? [e] : Array.from(e.querySelectorAll(name)))       
    return selectedElements.reduce((acc, val) => acc.concat(val), []) as [HTMLElement | null]
  }
  ...
  //Or, here is an alternative using @queryAssignedNodes decorated members
  @queryAssignedNodes("") defaultSlotNodes!:NodeListOf<HTMLElement>
  querySlotElementAll(slotNodes:NodeListOf<HTMLElement>,name:string):[HTMLElement | null]  {
    const slotChildren = Array.from(slotNodes).filter(e=>e.querySelectorAll)
    const selectedElements = slotChildren?.map((e) => e.nodeName == name.toUpperCase() ? [e] : Array.from(e.querySelectorAll(name)))       
    return selectedElements.reduce((acc, val) => acc.concat(val), []) as [HTMLElement | null]
  }
  ...
  <button @click=${async () => {
    const clocks = this.querySlotElementAll("","my-timer")
    clocks?.forEach((c) => this.run ? c?.removeAttribute("run") : c?.setAttribute("run","true"))
    this.run = !this.run
    await new Promise((r) => setTimeout(r, 1)) // Add a chance to browser to update the screen before popping up alert
    // Alert will put all timers on halt while the alert is on
    window.alert(`${clocks.length} timers have been ${this.run ? `started` : `stopped` }`)
  }}>${this.run ? `Stop` : `Start`}</button>

Slots can be named and even a web component can be sandwitched into another instance of the same class:

  <my-timer run>
    <span>Embedding timer:</span> 
    <my-timer slot=after run>Embedded timer:</my-timer>
  </my-timer>

Part 7: Using MobX for Controlling Web Component Collaborations

Full explanation in this video Web Components Getting Started - 7 MobX for Controlling Web Component Collaborations

Dispatching Registration Costom Events from the connectedCallback of Timers

Dispatching the event from connectedCallback have to be done via the updateCallback Promise this.updateComplete.then(()=>dispatchTimerEvent(this,{type:"Connected"})) otherwise it will not work.

export const TTimerEvent = "timer"
export type TTimerActions = 
    {type: "Connected"} 
  | {type: "Disconnected"} 
function dispatchTimerEvent(el:HTMLElement,detail:TTimerActions):void {
    console.log(el,getParentElements(el),el.assignedSlot)
    el.dispatchEvent(new CustomEvent(TTimerEvent,{detail,composed:true,bubbles:true,cancelable:false}))
}
...
override connectedCallback(): void {
  super.connectedCallback()
  this.updateComplete.then(()=>dispatchTimerEvent(this,{type:"Connected"}))         
}  

//my-element.ts
import "./my-timer" 
import {TTimerEvent,TTimerActions} from "./my-timer" 
...
    <button @click=${async ()=> {
        this.timers.forEach(t => this.runTimer ? t.removeAttribute("run") : t.setAttribute("run","true"))
        this.runTimer = !this.runTimer
      }}>
      ${this.runTimer ? `Stop` : `Run`}
    </button>
...
timers:HTMLElement[] = []
registerTimer(timer: HTMLElement): void {this.timers.push(timer)}
unregisterTimer(timer: HTMLElement): void {this.timers.splice(this.timers.indexOf(timer), 1)} 
override connectedCallback():void {
  this.addEventListener(TTimerEvent,((e:CustomEvent)=>{
    const timer = e.composedPath()[0] as HTMLElement
    const detail = e.detail as TTimerActions
    switch(detail.type) {
      case "Connected": this.registerTimer(timer); break
      case "Disconnected": this.unregisterTimer(timer); break
    }
  }) as EventListener)
  super.connectedCallback()
}

Collecting Embedded Timer Components upon connectedCallback

To improve the performance for registering the timers when the my-element component is consstructed. Here again is terribly important to call querySlotElementAll within the updateComplete Promise

//my-element.ts
@queryAssignedNodes("") defSlotNodes!:NodeListOf<HTMLElement>
querySlotElementAll(slotNodes:NodeListOf<HTMLElement>,name:string):HTMLElement[] {
  const slotChildren = Array.from(slotNodes).filter(e=>e.querySelectorAll)
  const selectedElements = slotChildren.map(e => Array.from(e.querySelectorAll(name)))
  const arr = selectedElements.reduce((acc,val) => acc.concat(val),[]) as HTMLElement[]
  //console.log("querySlotElementAll",arr)
  return arr
}
override connectedCallback():void {
  this.updateComplete.then(()=>{
    const timers = this.querySlotElementAll(this.defSlotNodes,"my-timer")
    timers.forEach(t => this.registerTimer(t))  
  })
  super.connectedCallback()
}

The idea came from Shoelace tab-group where the updateComplete Promise is used to collect the tabs and tab panels:

  connectedCallback() {
    super.connectedCallback()
    ...
    this.updateComplete.then(() => {
      this.syncTabsAndPanels()
    })
  }
  syncTabsAndPanels() {
    this.tabs = this.getAllTabs()
    this.panels = this.getAllPanels()
    ...
  }
  getAllTabs(includeDisabled = false) {
    const slot = this.shadowRoot!.querySelector('slot[name="nav"]') as HTMLSlotElement
    return [...slot.assignedElements()].filter((el: any) => {
      return includeDisabled
        ? el.tagName.toLowerCase() === 'sl-tab'
        : el.tagName.toLowerCase() === 'sl-tab' && !el.disabled
    }) as SlTab[]
  }
  getAllPanels() {
    const slot = this.body.querySelector('slot')!
    return [...slot.assignedElements()].filter((el: any) => el.tagName.toLowerCase() === 'sl-tab-panel') as [SlTabPanel]
  }
  ```
- *updateComplete* is a getter in Lit ReactiveElement class, and what is interesting it is available in *connectedCallback*
```ts
  private __updatePromise!: Promise<boolean>
  get updateComplete(): Promise<boolean> {
    return this.getUpdateComplete()
  }
  protected getUpdateComplete(): Promise<boolean> {
    return this.__updatePromise
  }

tslitstrap

On-line sample on Azure white-pond

Bootstrap could be used easily via CDN, but TypeScript and Lit are not so it's better to create a Vite or WebPack project. Then the next major question is Shadow DOM or Light DOM web components you want to use?

The extreme complexity of styling Shadow DOM s because of its strict encapsulation is a major hindrance of web components becoming more popular.

  • Fortunately Custom Elements can be defined with Light DOM, too, Bootstrap Web Components #28131 is nice discussion with examples how Bootstrap could be used with light DOM custom elements.
  • In this discussion blikblum's example shows what can be achieved.
  • In using-bootstrap-in-web-components-shadowdom If you don't use shadow DOM, you don't need these at all.
    import bootstrap from './path/to/bootstrap.css';
    class MyElement extends LitElement {
      static styles = bootstrap; // If your build system already converted
                                // the stylesheet to a CSSResult
      static styles = unsafeCss(bootstrap); // If bootstrap is plain text
    }
    • This guy explains that rollutup an even webpack has tools to convert CSS to CSSResult. rollup-plugin-postcss-lit This plugin is pre-configured to work with Vite, just add it to plugins and your styles will be Lit-ified npm i -D rollup-plugin-postcss-lit
    import {LitElement, css} from 'lit'
    import {customElement} from 'lit/decorators.js'
    import myStyles from './styles.css'
    import otherStyles from './other-styles.scss'
    @customElement('my-component')
    export class MyComponent extends LitElement {
      // Add a single style
      static styles = myStyles;
      // Or more!
      static styles = [myStyles, otherStyles, css`
        .foo {
          color: ${...};
        }
      `];
      render() {
        // ...
      }
    }
    
    // vite.config.js/ts
    import postcssLit from 'rollup-plugin-postcss-lit';
    export default {
      plugins: [
        postcssLit(),
      ],
    };
    //And here are the options:
    postcssLit({
      // A glob (or array of globs) of files to include.
      // Default: **/*.{css,sss,pcss,styl,stylus,sass,scss,less}
      include: ...,
      // A glob (or array of globs) of files to exclude.
      // Default: null
      exclude: ...,
      // A string denoting the name of the package from which to import the `css`
      // template tag function. For lit-element this can be changed to 'lit-element'
      // Default: 'lit'
      importPackage: ...,
    }),
    At the end of the documentation of this plugin it says that "This plugin is meant to be used with rollup-plugin-postcss. If you only need to load plain css files in your LitElement components, consider using rollup-plugin-lit-css"
    /* css-in-css.css */
    :host {
      display: block;
    }
    h1 {
      color: hotpink;
    }
    
    import { LitElement, customElement, html } from 'lit-element';
    import style from './css-in-css.css';
    @customElement('css-in-css')
    class CSSInCSS extends LitElement {
      static get styles() {
        return [style];
      }
      render() {
        return html`<h1>It's Lit!</h1>`;
      }
    }
    
    import config from './rollup.config.rest.js'
    import litcss from 'rollup-plugin-lit-css';
    export default {
      ...config,
      plugins: [
        litcss({ include, exclude, uglify })
      ]
    }
  • The Bootstrap starter template on GitHub uses Bootstrap V4 and is configured with its own build toolset.
  • Installing Bootstrap and Lit:
    • npm i bootstrap bootstrap-icons @popperjs/core lit mobx @adobe/lit-mobx @lit/localize
    • npm i -D @types/bootstrap rollup-plugin-lit-css eslint copyfiles @lit/localize-tools
    • import bootstrap from "bootstrap" is when you want to use toast, see the ksenia25 videos
    • I have already experimented with Bootstrap in ksenia25 where I was experimenting with WebPack5 and I recorded a number of accompanying videos.
  • The Bootrap documentation describing webpack setup explains that you may use Bootstrap’s ready-to-use CSS by simply adding this line to your project’s entry point import "bootstrap/dist/css/bootstrap.min.css"
  • unsafeCSS seems to be a perfect tool to import Bootstrap css into LitElement as CSSResult. No need for fussing with the native CSSStyleSheet and replaceSync
    import bootstrapcss from "bootstrap/dist/css/bootstrap.min.css"
    //...
    class BootBase extends LitElement {
      static styles = [unsafeCSS(bootstrapcss)]
    }
    //MobiX is needed only for elements contacted to the app state store
    @customElement("boot-app")
    class BootApp extends MobxReactionUpdate(BootBase) {
      static styles = [...BootBase.styles, css`:host{text-align: center;}`]
      //...
    }
    declare global { interface HTMLElementTagNameMap { "boot-app": BootApp}}
    • Regular CSSStyleSheet and replaceSync could be use too as an alternative to unsafeCSS:
      unsafeCSS does something similar as this sequence
      const bootstrapstyles = new CSSStyleSheet()
      //@ts-ignore
      bootstrapstyles.replaceSync(bootstrapcss) 
      ...
      // Can be added this to LitElement classes like 
      static styles = [bootstrapstyles as CSSResultOrNative]
  • npm i @lit-labs/task is really light-weight task controller for executing long running async calls. It shows similarities React's useEffect. It can be installed as an NPM package.
  • Lit Locatlization
    • Forget about the sample is available in Lit Playground, check out the localize/examples provided in the GitHub repo of Lit.
    • Here are the steps to do:
      • In the source code use msg() function to sandwitch all string literals to be localized as explained in the sections Making Templates Localizable, Message Descriptions for Translators and Message Placeholders
        • The section on place holders is totally wrong, the correct format is: msg(str${detail.person.firstName} added) where str is tagged literal function from the localize package. When you receive this error message error TS2324: String literal with expressions must use the str tag when running the extract command, then use this str.
      • Make a lit-localize.json according to the example. Here is my version using the runtime mode adopted from the example:
      {
          "$schema": "https://raw.githubusercontent.com/lit/lit/main/packages/localize-tools/config.schema.json",
          "sourceLocale": "en",
          "targetLocales": ["hu-HU", "fr_FR"],
          "tsConfig": "./tsconfig.json",
          "output": {
              "mode": "runtime",
              "outputDir": "src/loc",
              "localeCodesModule": "src/loc/locale-codes.ts"
          },
          "interchange": {
            "format": "xliff",
            "xliffDir": "xliff"
          }
      }    
      • outputDir is the directory where lit-localize build is going to generate the TypeScript files from the xlf files previously generated with lit-localize extract
      • localeCodesModule is the name of the optional convenience file lit-localize build generates based on the settings in lit-localize.json
      • xliffDir is where lit-localize extract is going to generate the xlf files containing the string literals to be translated for each locaes defined in the targetLocales list.
      • The transform mode would be an overkill for mobile applications with limited amount of translatable scripts, usually fewer than a couple of thousands.
    • After having lit-localize.json in place, run npx lit-localize extract, which will generate xlf files for each target locales containing the default English texts.
    • Provide translations for the texts in the xlf files by adding <target> elements keeping the generated <source> elemnts untouched.
      • If you receice the error hu-HU message s2ec6be1b6544c74f is missing, using canonical text as fallback, then you probably replaced the texts in the <source> elemnts instead of adding <target>. This is an example how an xlf translation should look like:
        <trans-unit id="s2ec6be1b6544c74f">
          <source>People Re-loaded</source>
          <target>Személyek adatai betöltve</target>
        </trans-unit>
        <trans-unit id="sd976e241583c9b5f">
          <source><ph id="0">${detail.numberOfPeopleLoaded}</ph> persons found</source>
          <target>Felleltünk <ph id="0">${detail.numberOfPeopleLoaded}</ph> személyt</target>
        </trans-unit>
        In the literals with placeholders don't touch <ph id="0">${detail.numberOfPeopleLoaded}</ph> otherwise you will have Placeholder error in hu-HU localization of ...
    • Generate translator TypeScript modules with npx lit-localize build, which will generate ts files from the xlf files. These files will be dynamically imported when needed:
      import {sourceLocale, targetLocales} from "./loc/locale-codes" 
      type TLocales = typeof targetLocales[number]
      
      @localized()
      @customElement("boot-app")
      class BootApp extends MobxReactionUpdate(BootBase) {
        private _localeConfig = configureLocalization({
          sourceLocale,
          targetLocales,
          //loadLocale: (locale: string) => import(`./loc/${locale}`), // For development with npm run dev
          loadLocale: (locale: string) => import(`./loc/${locale}.js`), // For npm run build
        })  
        async setLocale(loc:TLocales):Promise<void> {await this._localeConfig.setLocale(loc)}
        override connectedCallback():void {
          ...
          this.setLocale("hu-HU")
          super.connectedCallback()
        }
        ...
      }
      Here I created a convenience TLocales type definition from the target locales string array. This type then is used in a convenience async setLocale member function, which is calling the setLocale function available in the localization configuration obbject from configureLocalization
    • Whenever you add new msg localizable messages/texts you should npm run loc1extract, translate the texts in the xlf files and then and run npm loc2generate
      "loc1extract":"lit-localize extract",
      "loc2generate":"lit-localize build"

jobwatch

Online version is available on Azure

Based on the JQuery Mobile implementation's design here is a more modern Lit/Vaadin/Vite/TS version. Project setup:

  • In the repo root folder npm init vite then select vanilla-ts, then cd into jobwatch
  • npm install in the jobwatch project folder to install dev dependencies
    • Remember to change target to ES2021 and comment out useDefineForClassFields in tsconfig, otherwise Lit will not work.
  • npm install lit
  • Check out package.json to see which Vaadin packages to install.
  • Using Vaadin's app Layout was a dead-end
    • I spent a hell lot of time on the Vaadin layouts including Lumo CSS Frameweork - see my frustration video here - as demonstrated in the video Vaadin Fusion CSS Layouts (DON'T WATCH WASTE OF TIME) since "Vaadin 21 and later include the CSS class names out of the box, so no need to install them separately anymore". So, it worked in the vaadin generated applications as expected, but I wasn't able to make it working in a "standalone" application.
    • I did't give up, I learned raw flexbox, and flexbox CSS directives worked fine with the Vaadin web components. So, I dropped all Vaadin layouting components, and implemented my own plain flexbox layout. I wanted a layout with an app bar, an optionally scrollable middle contents section and a sticky bottom button bar.
      • Actually, I could live without the sticky bottom button bar, but in a handheld mobile device in the right hand, the most convenient place for buttons are on the bottom.
    • So, I found Create a mobile app layout with Flexbox and https://github.com/wesbos/What-The-Flexbox/tree/master/flexbox-app-layout and based on that, I have elaborated a more-or-less fine solution with significant changes and code cleaning.
      • I used semantic HTML5 element names (section (for the concept of page), header, footer, main (for middle contents area)) consistently.
      • The footer had to be removed from the section flex container to make it a zero-bottom-positioned element: footer{position:fix; bottom:0;}
      • I created a PageBase class derived from Lit element and that class implements the layouting CSS, and all my page components are derived from this class.
  • The next challenge is to enable dark mode. I followed the instructions Dark Mode but it doesn't swith to dark theme.
  • The menu bar component is a quite simplistic, but it might be OK for the job.

Here is an accompanying (unlisted) video explaining this iteration of the work on Jobwatch Lit 2.0 Announced, Flexbox Layout for Jobwatch (Review)

Defining Custom Evemt Names and Payload Structures

I first tried this technique in paola18 based on an article Sunny Sun: How to Share Constants in Typescript Project - Avoid the magic strings, make your app maintainable and strongly typed

export const TCustomEvents = {ConfigDone: "configdone", LoginBack: "loginback"} as const
export type TConfigDonePayload = {detail:{saved:boolean},composed: true}

Thi is how to dispatch properly a custom event:

    const payload:TConfigDonePayload = {detail:{saved:false},composed:true}
    this.dispatchEvent(new CustomEvent(TCustomEvents.ConfigDone,payload))

It is terribly important to set the composed option to true in custom events, otherwise it doesn't bubbles up to the application component. In the type definition I simply defined the type for composed as boolean false; this is a brutally fancy feature of TypeScript; that's why we love it.

Event Listeners on the Application Component

If the events are dispatched with composed set to true, then the application web component can listen to all events:

  override connectedCallback():void {
    super.connectedCallback()
    // WARNING!!! DON'T DEFINE an EVENT LISTENER on THE SHADOW ROOT if you want to reference "THIS"
    // this.shadowRoot!.addEventListener( DON'T DO THIS )
    this.addEventListener(TCustomEvents.ConfigDone,this.onConfigDone as EventListener)
  }

Don't define the event listener on the shadow root, since then the this value is set to the shadow root object and not the actual component's. Note the tricky casting this.onConfigDone as EventListener to prevent TypeScript complaining: How to add a custom event listener?

buttonEl.addEventListener('myCustomEvent', ((event: CustomEvent) => {
 //do something
}) as EventListener);

vaadings (Getting Started with Vaadin)

Vaadin Fusion was implemented with JavaScript on top of LitElement, but it comes with full TzpeScript tzpe definitions, too. How to use Web Components, 2018 Dec 12 and accompanying videos are to be followed first:

Since the project is using package libraries, a bundler is required; so, I decided to use ViteJS, since it has a template for Lit projects, exactly as I did in my elena14, but this Vite lit-element template is to make library projects for web components, and not applications. For deployable regular web applications use the vanilla-ts template. Project setup is then

  • Run npm init vite on the repo root folder to craete vaadings subfolder
    • cd vaadings and npm install
  • Copy .eslintrc.json from the other project
    • npm i -D @typescript-eslint/eslint-plugin @typescript-eslint/parser eslint
  • npm install @vaadin/vaadin-button @vaadin/vaadin-grid @vaadin/vaadin-text-field for the example app as suggested in How to use Web Components in a JavaScript project

So this project is just a regular TypeScript project not using Lit Element.

tswebcomp

After the tsorchids remake, I started experimenting with web components foloowing JH Web Components Custom Elements, Templates and the Shadow DOM, 2019 and the subsequent two parts.

Here is my eloquent explanation video Web Component (Custom Element) Fundamentals with TypeScript.

I kept the tsorchids project, and I created a new folder tswebcomp. This project had only src/index.ts and dist/index.html I copied the files package.json (changed the project name and description), .eslintrc, tsconfig.json and .gitignore. I changed .gitignore, not to exclude the entire dist folder, just ignore dist/*.js

After cloning, the steps are exactly the same as in tsorchids project, but the project root folder is tswebcomp, of course.

For the full explanation of the videos, first watch JHerr's then mine. Here are the main points when working with web components:

  • What is the difference between an open and closed shadow root? Here is a nice explanation open-vs-closed-shadow-dom
    • The short answer is that with close shadow root, you should use the object returned by the attachShadow function and use that for building the shadow DOM.
  • Slots work only in the shadow DOM.
  • You can build custom elements (aka web components) with Bootstrap, too, without using the shadow DOM, but then you will not have slots and all the styling is cascading through the components in the elements defined in the innerHTML of the custom element.
    • To demonstrate that slots are available only in the shadow DOM I have added an attribute to my-product custom element. When shadow is no defined or its value is false, then the slot-replacement machinery is lost when instantiating a custom element:
      <my-product name="Note8" style="font-size: xx-large;" shadow>
          <span slot="price">$560</span>
          <h2>One of the best smartphones ever.</h2>
      </my-product> 
    This is a major limitation, and (almost) a (total) show stopper for using custom elements with lith DOM building librraies based on Bootstrap.
  • You cannot use input elements from the shadow DOM in combination with an outside form.
    • You have to add fordata event listener and then : (for more details see formdata-event example) Apple browsers don't support fordata events. Fortnately Apple is not relevant for business applications, and Chrome is available on all Apple platforms.
      handleFormData({formData}) {
        // add our name and value to the form's submission data if we're not disabled
        if (!this.input.disabled) {
          // https://developer.mozilla.org/en-US/docs/Web/API/FormData
          formData.append(this.input.name, this.input.value);
        }
      }    
      // find the <form>, and attach the `formdata` listener
      connectedCallback() {
        this._form = this.findContainingForm();
        if (this._form) {
          this._form.addEventListener('formdata', this._handleFormData);
        }
      }
      // remove the `formdata` listener if we're removed
      disconnectedCallback() {
        if (this._form) {
          this._form.removeEventListener('formdata', this._handleFormData);
          this._form = null;
        }
      }
  • All methods defined in the class (render, show, hide in our example) is fully accessible to the users from the web page.
  • If your attributeChangedCallback is not fired, possibly you have forgotten to define the static observedAttributes
  • Attributes are standard feature of every HTML element, and they can be accessed with getAttribute and setAttribute, but the browser will not generate any getter/setter for them automatically; this is your job to define getter/setter properties. You can have properties without attributes, and you can have attributes without properties, and you can link them as demonstrated in this example.
  • Using the standard template element gives the possibility that the user can define and customize the template on the HTML page without interferring the layout, since the browser doesn't render at all a template; a template is meant tor the web component developers to be rendered with JavaScript. See Using_templates_and_slots A major limitation is that a template has no machinery to generate parts dynamically. You can use regular JavaScript template literals with the placeholder expressions to create the inner text of templates, but gives no HTML diffing virtual dom feature, unlike the brilliant lit-html.
    • Slots work without templates, too. But, cloning a template is a lot faster than rendering a HTML string for hungreds of elements in a scrolling list for example.
  • Event handler functions that are defined with the classic method way have to be bounded when added to an event listener. Event handlers defined with lamda syntax are automatically bounded to the this keyword of the class.
    • If your event handler doesn't fire, check the handler function definition syntax and binding.
  • There is no such a thing as render callback function.
  • Here are the (lifecycle) callbacks:
    • constructor Lifecycle Callbacks suggest not to do any rendering and data fetching in the constructor it should be done in the connectedCallback
    • connectedCallback
    • disconnectedCallback
    • attributeChangedCallback
  • As for HTMLElementTagNameMap see typescriptlang/dom-manipulation When a lit-element project is created with Vite (see my elena14/elana14lit branch), a declaration section is generated at the bottom of the web component class TS file.
    declare global {
        interface HTMLElementTagNameMap {
          "my-product": MyProduct
        }
    }
  • Instead of regular events, dispatch CustomEvent and you can add data in the constructor
          const productDetails:TProductDetails = {name:this.name, price:854.5}
          this.dispatchEvent(new CustomEvent("buy",{detail:productDetails}))  
          //And the user can listen to that event like so:
          myProduct().addEventListener("buy",(e)=>alert('Buy Me, Please!' + JSON.stringify(e.detail)))
  • Actually, in practice when you create a custom element you don't have to render your template in the shadow DOM, you can implement it in the light DOM in innerText and you will have the regular global CSS space, which is excellent when you want to use Bootstrap with custom elements. BUT, there is an (almost (total)) show-stopper: slots don't work in light DOM, only in shadow DOM; that is, you cannot do something like <my-product name="Note 8"><h1 slot=price>999<h1><p>Great product</p></my-product> Instead, you are forced to apply a workaround (render-props) <my-product name="Note 8" price="<h1 slot=price>999<h1>" default="<p>Great product</p>"></my-product> This would work excellently with a tool like Lit but far from elegant. It was a crazy opinionated decision from the specification designers not to enable slots for the light DOM either.
  • I have implemented a way to clone automatically a style element in the shadow DOM This style (global) element is applied to the entire HTML page (the light DOM) but it has an ID, too:
    <style id="my-product-style">
        h2{color:blue;}
        button{color:red;}
    </style>

Therefore, the MyProduct custom element class can append it to its own shadow DOM:

    get name():string {return this.getAttribute("name")!}
    connectedCallback():void {
        let template = document.getElementById("my-product-template") as HTMLTemplateElement
        if(!template) {
            template = document.createElement("template")
            template.innerHTML = /*html*/`
                <!-- Add default template -->
            ` 
        }        
        const myProductStyles = document.getElementById("my-product-style")
        if(myProductStyles) this.root.appendChild(myProductStyles.cloneNode(true))
        this.root.appendChild(template.content.cloneNode(true))
        this.root.querySelector("button")?.addEventListener("click",this.buyMeButtonClick)
        this.render()
    }
    render():void { // Not a callback, this is just my function
        const h2 = this.root.querySelector("h2")
        //The first h2 element's text will be replaced with the product name
        if(h2) h2.innerText = this.name
    }

tsorchids

This is a remake of Jack Herrington Javascript Modules in detail - Part 1

After cloning,

  • cd tsorchids this is the project home/root folder
  • npm install to install the npm packages based on package.json
  • npm start to start tsc --watch to compile-watch the TypeScript sources into the dist folder.
  • in another terminal window npm run s4000 to start servor upon dist/index.html, or you can use Live Server, too.

Here is my quite lengthy explanation of almost every line No-framework, no-bundler TypeScript WebApplication Development 01 JH's Orchids Sample Remake.

The basic idea is to use only tsc for building the application. Here are the steps how to initialize the project. See my other projcet alessia8 how to configure ESLint for TypeScript

  • npm init
  • npm i -D typescript eslint
  • npx tsc --init
    • "target": "ESNext",
    • "module": "ESNext",
    • "rootDir": "./src",
    • "outDir": "./dist",
    • "inlineSourceMap": true,
    • "strictFunctionTypes": true, is terribly important, never leave it disabled.
  • npx eslint --init
    • How would you like to use ESLint? · problems
    • What type of modules does your project use? · esm
    • Which framework does your project use? · none
    • Does your project use TypeScript? · No / Yes
    • Where does your code run? · browser
    • What format do you want your config file to be in? · JSON
    • The config that you've selected requires the following dependencies:
      • @typescript-eslint/eslint-plugin@latest @typescript-eslint/parser@latest
      • Would you like to install them now with npm? · No / Yes
  • Add these TypeScript rules to .eslintrc:
    "@typescript-eslint/no-non-null-assertion":"off",
    "@typescript-eslint/explicit-function-return-type": "error", 
    "@typescript-eslint/explicit-module-boundary-types": ["warn", {"allowArgumentsExplicitlyTypedAsAny":true}], 
    "@typescript-eslint/no-explicit-any":"warn", 
    "quotes": ["error","double",{ "allowTemplateLiterals": true}], 
    "@typescript-eslint/semi": ["error", "never"], 
    "@typescript-eslint/ban-ts-comment": ["warn"]
    
  • I applied npx eslint src --fix to fix the fixable issues on the source files I copied from JHerr's repo.

Closures in JavaScript/TypeScript

The original implementation by JH was based on closures. Here all the local variables count, subscribers, orchids and image were referenced in the functions defined for the IStore object, all of these variables are actually created on the heap and they are kept alive with reference counting.
The store (global) object is created at the end of the module export const store = createStore(), so the local variables referenced by these functions are kept forever. The function createStore with its local variables behaves like an object constructor. See emulating_private_methods_with_closures

const createStore = ():IStore => {
  let count = 0
  const subscribers: Array<TStoreEventHandler> = []
  const orchids = [1,2,3,4,5,6,7].map(i => `orchids/orchid${i}.jpg`)
  let image = orchids[0]
  return {
    getCount: ():number => count,
    getImage: ():string => image,
    incrementCount: ():void => {
      count += 1
      subscribers.forEach(s => s("CountIncremented"))
    },
    setImage: (img:string):void => {
      image = img
      subscribers.forEach(s => s("ImageSet"))
    },
    subscribe: (fn:TStoreEventHandler):void => {
      subscribers.push(fn)
    },
    orchids,
  }
}
export const store = createStore()

When event listener functions are created they are kept alive until the object to which they are registered is live. So, the createBuytools for example has two event listener functions and both of them references the store object that the function is received as incoming parameter. These anonymous functions are linked to the button (addEventListener) object and the store (subscribe) object. Since these functions reference the store object it is a closure again. The store object could have been a local variable, too, in this regard it doesn't matter.

export const createBuytools = (el:HTMLElement,store:IStore):void => {
    ...
    document.getElementById(ADDTOCART)!.addEventListener("click", () => {
        store.incrementCount()
    }); 
    store.subscribe((ev:TStoreEventType) => {
        if(ev == "ImageSet") (document.getElementById(ORCHIDIMAGE)! as HTMLImageElement).src = store.getImage()
    })
}

Loading Modules Dinamically Implementing a Micro FrontEnd (Micro-FE) Architecture

This second part is based on JH: Javascript Modules in detail - Part 2. Servor is used to mimic the multiple miro-FE provider concept running on different ports. Servor has hot reload as well as built in CORS enabled without any configuration, which is very important for testing multi-site/multi-server applications.

  • Make sure to navigate to the home folder (tsorchid) of the project (not the root folder of the mult-project repo)
  • npx servor dist index.html 4000 --browse --reload starts servor on the dist folder opening index.html on port 4000 opening a browser and reloading when any file is changed.
  • I added a script command to package.json, too "s4000":"npx servor dist index.html 4000 --browse --reload" using npm run s4000

Eventually, I lost interest since this kind of JavaScript source module distribution (federation) might primarily be interesting for wbesite, portal builders. Business applications are communicationg via API's and for single signon with HTTP redirections. On the other hand, having a business app, production data collection, warehousing mobile scanner, and the application manufacturer could implement data validations, user interface composition with this dynamic JavaScript module loading. Yep, definitely thhis would be very important for a major out-of-the-box standard project. The configuration details should have been stored in a separate JavaScript module, which is not part of the standard application deplyment, so every time a new version is delivered, the configuration JavaScript module would be kept as is. In a cloud service scenario each tenant could have its configuration data table with the urls for the custom modules; but, in this scenario allowing a customer to inject its own custom JavaScript code would be a major risk; so this would work best in a customer-dedicated deployment, where the standard application would be deployed on the customers server or virtual machine in a cloud. So, this github.com/jherr/orchids-esm-part-2 could be vey useful in case I decided to start a major project with Miro-FE.

In the index.html he used <component src="http://localhost:4002/header.js"></component> and <component src="http://localhost:4002/banner.js" text="Buy these orchids!"></component> these component tags were just arbitrary names, these are not custom elements, they have no semantics, just like spans, sections or divs. They can even have attributes, but the browser doesn't do anything with these attributes the src on a component element means nothing to the browser: it's the application that has to handle these attributes. Here is a function that actually loads the JavaScript modules defined in the src attributes in these arbitraryly invented component elements:

export const loadComponents = () => {
  document.querySelectorAll('component').forEach(el => {
    const props = {};
    Array.from(el.attributes).forEach(({ name, value }) => {
      props[name] = value;
    });
    import(el.attributes.src.value)
      .then((mod) => {
        mod.default(el, props);
      });
  });
};

It collects all the attributes into an object const props = {}; note the syntax he used to create member in the object with the array syntax props[name] = value. Thereafter, the import async function was used to load the JavaScript module. He could have used the async/await syntax:

    const mod = await import(el.attributes.src.value)
    mod.default(el, props)

He used the default function passing all the attribute object. This is a great example why it is meaningful to define a default function. This is a pretty neat dynamic custom module loader implementation. What is not really elegant in the solution is the hard-coded URL's in the component JavaScript codes import { getStore } from 'http://localhost:4001/framework.js'

Ranting on the Event of Lit 2.0 Officially Announced

A Quick subjective review of the Web Component landscape in 2021 Sept

Lit-HTML is eye-opening brutally excellent tool for both regular reactive web application development as well as for web components. Lit Element is fantastic, too, for web components. These tools themselves are awesome, mature error-free, brilliant. The problem is with the entire web component ecosystem: there are no really great, reliable professional quality components libraries for business applications. Vaadin Fusion would be really decent, but it is tied heavily to their Java server stack. Shoelace looks awesome, but it has no data-grid/table/virtual list or any list component at all. SAP UI5 Web Components (not to confuse with the plain UI5), has a quite decent UI5Table component, but it is not up to what is expected for a demanding business app (Vaadin Grid, even the free version, is almost perfect). UI5 web components uses lit-html but not Lit Element. Material Components are version 0.25 and TB developed. As for Microsoft FAST/FluidUI web components, the technology is brutally similar to lit-html/Lit Element, but their data-grid component is pathetic. Telerik a major component manufacturer has no web component version of their libraries. Haunted has a great concept for hook-paradigm, and is based on lit-html, but it uses a an old version of lit-html which doesn't work with any component libraries.

Lit-Plugin Visual Studio Code extension is fine, but it is far from perfect, when the application is getting bigger, its code completion, context sensitive support for listing properties, @-syntax event handlers is getting hit or miss. When using React/TypeScript or Angular, Visual Studio Code programmer experience is mature and reliable in this regard. In Angular you even have the @output concept, which is totally missing from the Custom Element specification. When you have a component toolset like React or Angular, you don't need web components, totally irrelevant, a major hassle and additional complexity, web components are alien to Angular/React.

On the other hand, Lit is so brilliant that, if you are open to build your own web component library, it would be the best choice. Vaadin Fusion library is really decent but their tight coupling with Java server is sever limitation, unfortunately.

Other Videos Worth Watching