My First LitElement Web Component#
Description and reflections of create a very simple web component using LitElement's live tutorial
- My First LitElement Web Component
- Basic steps
- Writing your own - choices
- Creating reusable components - LitElement process
- Create Open Web components
- Creating a simple TrimesterDate component
- Practical example
- Finishing the trimester-date component
Basic steps#
Lot's not covered in this, including
- how to handle attributes passed to the component
- more flexible approaches to css
Impressive first impression#
First reactions, impressive. Produces a VSCode looking code editor and matching "browser" output all in the browser window. Though it's a bit of pain that it doesn't have the vim bindings.
Creating a component#
In summary,
A component lives in its own JS file, with three main sections
- Import various "includes"
- Define a class for the component (the code)
- Register the element with the browser
Properties#
Looks like basics of contemporary Javascript OO. But also with "React-like" nicety of components updating automatically when their properties change.
- Properties implemented as a hash/object
- Declare a getter for properties
static get properties() {
return { message: { type: String } };
}
- Initialise the property with a constructor
constructor() {
super();
this.message = 'Hello world! From my-element';
}
```
- Use it as an attribute
```javascript
return html`
<p>${this.message}</p>
`;
Logic in templates#
How to integrate conditionals and loops into components
return html`
<p>${this.message}</p>
<ul>${this.myArray.map(item => html`<li>${item}</li>`)}</ul>
${this.myBool ?
html`<p>Render some HTML if myBool is true</p>` :
html`<p>Render some other HTML if myBool is false</p>`}
`;
Events source#
- Add an event listener using
@event
- in this@click
<button @click=${this.clickHandler}>Click</button>
- Add an event handler as method to the class
clickHandler(event) {
console.log(event.target);
this.myBool = !this.myBool;
}
```
In operation, this illustrates how components are updated dynamically in response to changes to properties.
![Dynamically changing component display](webComponentButton.gif)
### Styling
[source](https://lit-element.polymer-project.org/try/style)
- include a css helper
- define styles by adding a styles getter (must be more complex ways to do this)
```javascript
static get styles() {
return css`
p {
font-family: Roboto;
font-size: 16px;
font-weight: 500;
}
.red {
color: red;
}
.blue {
color: blue;
}
`;
}
- apply styles
<p class="${this.myBool ? 'red' : 'blue' }">styled paragraph</p>
Writing your own - choices#
Two options
- Create reusable components to share with others e.g. casa. Typically published to npm. e.g. unbundled-web-components
- Create app-specific components e.g. Progress Checker. Part of the application code.
Creating reusable components - LitElement process#
LitElement suggests the following process
- Use a starter project
- Node.js and npm for dependency management
- Local dev server
- ESLint
- Testing with Karma (new to me)
- Static doc site auto built
But also identifies the open-wc project generator
Create Open Web components#
npm init @open-wc
- running a generator- Choose from two options
Web component
- for a single componentApplication
- application project (no shit Sherlock)- Add features eventually, including
- Linting - ESLint, Prettier, Husky and commitlint
- Testing - Karma
- Demoing - Storybook
- Building
Creating a simple TrimesterDate component#
Specification#
A component that changes the following
<trimester-date>Monday, Week 7</trimester-date>
into
<span class="trimester-date">Tuesday, Week 7 (15 June, 2020)<span>
Where the data that has been added in brackets is calculated by 1. Identifying the current trimester/term for a given institution based either on 1. The URL of the web page, or 2. The current default value 1. Identifying the date for the Monday of Week 7 for that trimester 2. Add 1 day to that date to get the date for Tuesday
Getting started#
- Create a folder and run
npm init @open-wc
- Use cursor keys to make choices, including
- scaffold a new project or upgrade new
- web component or application
- typescript or not
- Provide the tag name for my component
- Shows a default structure for appoval
- Choose dependencies via npm.
Which provides a full templated project, including source with a lot of the defaults in place. Looks like it's a functioning element by itself. Meaning it should run
Local dev server#
Yep, npm run start
works out of the box. All working.
Making changes#
The demo HTML is not yet standard - more learning to do - open-wc codelabs to the rescue?
All that was really needed was a bit of thinking and testing. I've rewritten the default index.html
from open-wc to use something a little bit more related to my thinking. Which works in the local test environment.
<script type="module" src="../trimester-date.js"></script>
<trimester-date title="fred">Hello</trimester-date>
The main problem is how to import the lit-element
resource (which npm has handled locally). Google search and Stack Overflow point to the CDN solution and "my" first web component is publicly available.
Only now need now to
- Turn this into a more scalable, sustainable process; and,
- Complete the code for the component
Building#
Optimising files to minimise impact on live. Only do this with final production.
Type of project matters 1. Building single page apps (SPA) - recommend roll-up 2. Building reusable components and libraries - recommend buildless development and how to publish web components to npm 3. Building websites or multi page apps (MPA) - no final recommendation yet, tips on building roll-up
I'm doing #2 at the moment.
Buildless recommends avoiding build tools (adding complexity) and instead relying on the browser.
More advice on developing without a build. It touched on evolution of web development (from just a text editor and web server) to complex toolsets that make it hard to learn (preach brother!). Instead, how much (and how) can be done with just the browser.
TO DO How to publish web components to npm
Rollup is recommended to "optimize your code for production and ensure it runs on all supported browsers". Built into the project generator.
Practical example#
The Open-WC site wasn't giving the end-to-end example I wanted. A google search revealed Building Components with Open Web Components
Use the open-wc generator#
As with the above
Install other dependencies#
Their example web component requires a bit more than mine.
Write the code - #
It's also a bit more complex. Good example of a more advanced component.
Using the web component#
Brings the component back in from npm. Proposes use of a CDN (unpkg.com) if testing where no install available.
- Load the calendar information from JSON/web service
- Figure out the current trimester from
- page URL,
- attribute, or
- default
- Get the current date string
- Identify week and day (if possible)
- Calculate the actual date
- Update the HTML to add the date
- Publish to npm??
- Get into a CDN??
- Test in Blackboard
Load calendar information#
Need to load the calendar data from somewhere. It's currently hard coded into other applications as JSON data. In theory, ES6 supports the importation of JSON data
e.g. import calendar from '../calendar.js'
would result in a variable named calendar with the information. That appears like it might work. At least initially.
Publish the component#
Building Components with Open Web Components suggests that standard approach with npm - so time to create an npm account../calendar.js
A bit more of a struggle with npm account login (password issues?) than expected, but is published
Make it available via a CDN#
Building Components with Open Web Components uses unpkg.com