Skip to content

Latest commit

 

History

History
249 lines (206 loc) · 7.81 KB

README.md

File metadata and controls

249 lines (206 loc) · 7.81 KB

lit-html workshop

In this workshop you will work on assigments. The lit-html assignments are done with the whole group, the lit-element assignments are done individually.

You can work on the assignment by copy pasting them into http://plnkr.co/edit/DI9312Rn54NuIH7FRzzD?p=preview, or by cloning this repository and running it locally (npm start).

The assignments should be executed in order, one step sets the stage for the next step. When in doubt, ask us or check the solution.

When you are done (or bored) there are advanced assignments which are free to pick in any order.

News app demo

Assignments

  1. Hello world
  2. Fetch first article
  3. Render articles
  4. Child component
  5. Read/unread toggle
  6. Filter articles
  7. Styling
  8. Polymer components
  9. Read/unread counter
  10. Template function

Solutions

  1. Hello world
  2. Fetch first article
  3. Render articles
  4. Child component
  5. Read/unread toggle
  6. Filter articles
  7. Styling
  8. Polymer components
  9. Read/unread counter
  10. Template function

Advanced assignments

These assignments have no solution. Pick a topic you are interested in.

Cheatsheet

Below are some code samples for quick reference, and some comparisons to how things are in in Polymer. Checkout the links below for more in-depth explanations.

Good reference material:

Other useful links:

Define a template

const template = html`
  <p>Hello world</p>
`;

Set variables inside a template

const template = html`
  <p>hello ${message}</p>
`;

Property binding (Polymer VS lit-html)

MyElement has two properties: users and menuItems

  • Polymer: change name to kebab-case, wrap in [[]]
  • lit-html: use camelCase name, add a . in front, wrap in ${}
html`
  <!-- polymer -->
  <my-element
    user="[[user]]"
    menu-items="[[menuItems]]">
  </my-element>

  <!-- lit-html -->
  <my-element
    .user=${this.user}
    .menuItems=${this.menuItems}>
  </my-element>
`;

Attribute binding (Polymer VS lit-html)

Attributes are set on the HTML element. in general you should set properties and only set attributes when you really need to for example to use it in CSS or for some native HTML elements

  • Polymer: like property binding, but add a $ behind it

  • lit-html: like property binding, but without a . in front

  • for boolean attributes Polymer did some hidden magic for you. in lit-html you need to add a ? in front to set this behavior. This will add/remove the attribute based on a boolean value.

html`
  <!-- polymer -->
  <img src$="[[imgSrc]]">
  <select>
    <option selected$="[[selected]]"></option>
  </select>

  <!-- lit-html -->
  <img src=${imgSrc}>
  <select>
    <option ?selected=${this.selected}></option>
  </select>
`;

Event listeners (Polymer VS lit-html)

  • Polymer: pass a string, polymer will try to find a function by that name your component
  • lit-html: pass (a reference to) the actual function which will be executed. This function can come from anywhere.
html`
  <!-- polymer -->
  <button on-click="_onButtonClicked">
    Click me
  </button>

  <!-- lit-html -->
  <!-- a function on your class instance -->
  <button @click=${this._onButtonClicked}>
    Click me
  </button>

  <!-- a function outside your class, for example a utility function -->
  <button @click=${onButtonClicked}>
    Click me
  </button>

  <!-- an inline function, useful for small changes -->
  <button @click=${()=> this.expanded = !this.expanded}>
    Click me
  </button>
`;

LitElement class basics

import { LitElement, html, css } from 'https://unpkg.com/[email protected]?module';

class MyElement extends LitElement {
  // your styles are defined in a static styles property
  static get styles() {
    return css`
      :host {
        display: block;
      }
    `;
  }

  static get properties() {
    return {
      message: { type: String }
    };
  }

  constructor() {
    super();

    // default value is no longer set in the static properties definition, but in the constructor
    this.message = 'World';
  }

  render() {
    return html`
      <p>Hello ${this.message}</p>

      <button @click=${this._onButtonClicked}>
        Click me!
      </button>
    `;
  }

  _onButtonClicked() {
    console.log('button clicked!');
  }
}

customElements.define('my-element', MyElement);

LitElement lifecycle

import { LitElement, html, css } from 'https://unpkg.com/[email protected]?module';

class MyElement extends LitElement {
  // styles: called once on element defintion, styles are the same for all instances
  static get styles() {
    return css``;
  }

  // properties: called once on element definition
  static get properties() {
    return {};
  }

  // called once for each instance of your component
  constructor() {
    super();
  }

  // called each time LitElement wants to render because a property changed
  // you can return a boolean to control whether or not the render should occur
  // changedProperties is a Map
  shouldUpdate(changedProperties) {
    if (changedProperties.has('somePropertyThatShouldNotCauseAnUpdate')) {
      return false;
    }
    return true;
  }

  // called each time LitElement wants to re-render your changes, if you did not
  // implement a custom shouldUpdate this will trigger each property c hange
  render() {
    return html``;
  }

  // called the first time your element has been rendered, can be useful when you need
  // to access your DOM elements
  // changedProperties is a Map
  firstUpdated(changedProperties) {
    this.shadowRoot.querySelector('...');
  }

  // called each time a render has occurred, this is useful to set up observers
  // changedProperties is a Map
  updated(changedProperties) {
    if (changedProperties.has('opened')) {
      this._onOpenedChanged();
    }

    if (changedProperties.has('message')) {
      this._doSomethingWithMessage(this.message);
    }
  }

}

customElements.define('my-element', MyElement);