hypermedia-app / lit-any-views

@lit-any view elements for functional building of complex HTML
https://lit-any.hypermedia.app
MIT License
1 stars 0 forks source link

DEPRECATED in favour of @hydrofoil/roadshow


@lit-any/views codecov

Building late-bound User Interface with lit-html without actually creating or using (too many) custom elements

Quick guide

1. Install

yarn add @lit-any/views lit-html lit-element

2. Set up how to render your content

You want to display an object which represents a person but the avatar comes in two flavors. Of course, it's possible to keep abusing if statements or drop in Polymer's <dom-if> templates.

With lit-any you can deconstruct your HTML by defining partial templates which will be rendered when they are really needed.

import { ViewTemplates } from '@lit-any/views'
import { html } from 'lit-html'

ViewTemplates.default.when
    .value(isPerson)
    .renders((renderFunc, person) => html`
        <person-element .name="${person.name}">
            <span slot="avatar">
                ${renderFunc(person.avatar, 'person-element-avatar')}
            </span>
        </person-element>
    `)

ViewTemplates.default.when
    .scope('person-element-avatar')
    .value(v => v.url)
    .renders((_, image) => html`
        <img src="https://github.com/hypermedia-app/lit-any-views/raw/master/${image.url}" alt="avatar" />
    `)

ViewTemplates.default.when
    .scope('person-element-avatar')
    .value(v => v.large)
    .renders((_, image) => html`
        <a href="https://github.com/hypermedia-app/lit-any-views/blob/master/${image.url}">
            <img src="https://github.com/hypermedia-app/lit-any-views/raw/master/${image.large}" alt="avatar" />
        </a>
    `)

function isPerson(value) {
    return value.type === 'Person'
}

This will set up the rendering so that <person-element> is displayed for a person but the avatar will be rendered based on the presence of the large property.

3a. Render anywhere

To do actual rendering you don't really need a dedicated custom element. Any container will be fine

<div id="personContainer"></div>
import { ViewTemplates } from '@lit-any/views'
import render from '@lit-any/views/lib/render'

const person = {
    type: 'Person',
    image: {
        url: 'https://s.gravatar.com/avatar/1497654c2d1af3cef4987234d1aced57?s=80',
        large: 'https://s.gravatar.com/avatar/1497654c2d1af3cef4987234d1aced57?s=800'
    }
};

const personContainer = document.querySelector('#personContainer');

render(ViewTemplates, { value: person }, personContainer);

3b. Render element

Alternatively you can use the lit-view instead of rendering directly to any node.

<lit-view id="personContainer"></lit-view>
const person = {
    type: 'Person',
    image: {
        url: 'https://s.gravatar.com/avatar/1497654c2d1af3cef4987234d1aced57?s=80',
        large: 'https://s.gravatar.com/avatar/1497654c2d1af3cef4987234d1aced57?s=800'
    }
};

const personContainer = document.querySelector('#personContainer');

personContainer.value = person;

More features

Scoping

When you want to display same data differently in different context.

import ViewTemplates from '@lit-any/lit-any/views';
import { html } from 'lit-html';
import * as moment from 'moment';

// show nicely formatted date by default
ViewTemplates.default.when
    .value(v => (v instanceOf Date))
    .renders((_, date) => html`<span>${moment(date).format('LL')}</span>`);

// but display calendar in 'event-large' scope
ViewTemplates.default.when
    .value(v => (v instanceOf Date))
    .scope('event-large')
    .renders((_, date) => html`<datetime-picker disabled .datetime="${date}"></datetime-picker>`);

Then set the scope on lit-view element:

<lit-view .value="{{someDate}}" template-scope="event-large"></lit-view>

Or pass to the render function:

import render from '@lit-any/lit-any/render';
import ViewTemplates from '@lit-any/lit-any/views';

const eventDateElement = document.querySelector('#eventDate').

render(ViewTemplates.default, {
    value: new Date(),
    scope: 'event-large',
}, eventDateElement);

Or override from parent template:

ViewTemplates.default.when
    .value(v => v.type === 'Event')
    .renders((renderChild, event) => html`
        <my-event-element>
            <div slot="calendar">
                ${renderChild(event.date, 'event-large')}
            </div>
        </my-event-element>
    `);