uce
    TypeScript icon, indicating that this package has built-in type declarations

    1.16.5 • Public • Published

    µce

    Downloads Build Status Coverage Status

    windflower

    Social Media Photo by Dawid Zawiła on Unsplash

    µhtml based Custom Elements.

    📣 Community Announcement

    Please ask questions in the dedicated discussions repository, to help the community around this project grow


    API Overview

    µce exports render, html, and svg, from µhtml, plus its own way to define components.

    In version 1.2, it exports also a plain-tag named css, useful to trigger CSS minifiers.

    Check out the test page or this code pen playground.

    // list of all exports
    import {define, render, html, svg, css} from 'uce';
    
    define('my-component', {
    
      // if specified, it can extend built-ins too.
      // by default it's 'element', as HTMLElement
      extends: 'div',
    
      // if specified, it injects once per class definition
      // a global <style> element in the document <head>, but
      // *not* in the shadowRoot. You can render style just fine
      // within the render, if shadow DOM is desired.
      // For the one-off global case, this method will be invoked with
      // a selector like `div[is="my-component"]` or `some-component`
      style: selector => css`${selector} {
        font-weight: bold;
      }`,
    
      // if specified, it's like the constructor but
      // it's granted to be invoked *only once* on bootstrap
      // and *always* before connected/attributeChanged/props
      init() {
        // µhtml is provided automatically via this.html
        // it will populate the shadow root, even if closed
        // or simply the node, if no attachShadow is defined
        this.html`<h1>Hello 👋 µce</h1>`;
        // a default props access example
        // <my-ce name="ag" />
        console.log(this.props.name); // "ag"
      },
    
      // if there is a render method, and no `init`,
      // this method will be invoked automatically on bootstrap.
      // element.render(), if present, is also invoked automatically
      // when `props` are defined as accessors, and one of these is
      // set during some outer component render()
      render() {
        this.html`<h1>Hello again!</h1>`;
      },
    
      // by default, props resolves all attributes by name
      // const {prop} = this.props; will be an alias for
      // this.getAttribute('prop') operation,
      // but it can simulate what React props do,
      // meaning that if it's defined as object,
      // all properties will trigger automatically
      // a render() call, if there is a render,
      // and properties are set as accessor, so that
      // the syntax to trigger these is .prop=${value}
      // as opposite of the default prop=${value}
      // which is observable, but it can hold only strings.
      // props: {prop: value} will make this.prop work.
      // If you don't want any of this machinery around props
      // you can opt out by defining it as null.
      // Bear in mind, the way to pass props as accessors,
      // is by prefixing the attribute via `.`, that is:
      // this.html`<my-comp .prop=${value}/>`;
      props: null,
    
      // if present, all names will be automatically bound to the element
      // right before initialization (el.method = el.method.bind(el))
      // this allows usage of methods instead of `this` for inner components
      bound: ['method'],
    
      // if specified, it renders within its Shadow DOM
      // compatible with both open and closed modes
      attachShadow: {mode: 'closed'},
    
      // if specified, observe the list of attributes
      observedAttributes: ['test'],
    
      // if specified, will be notified per each
      // observed attribute change
      attributeChanged(name, oldValue, newValue){},
    
      // if specified, will be invoked when the node
      // is either appended live, or removed
      connected() {},
      disconnected() {},
    
      // events are automatically attached, as long
      // as they start with the `on` prefix
      // the context is *always* the component,
      // you'll never need to bind a method here
      onClick(event) {
        console.log(this); // always the current Custom Element
      },
    
      // if specified with `on` prefix and `Options` suffix,
      // allows adding the listener with a proper third argument
      onClickOptions: {once: true}, // or true, or default false
    
      // any other method, property, or getter/setter will be
      // properly configured in the defined class prototype
      get test() { return Math.random(); },
    
      set test(value) { console.log(value); },
    
      sharedData: [1, 2, 3],
    
      method() {
        return this.test;
      }
    
    });

    F.A.Q.

    Which polyfill should I use?

    The @ungap/custom-elements is the recommended polyfill to grant every Custom Elements V1 feature is available in every browser.

    However, if no builtin extend is used, but legacy needs to be supported, including @webreflection/custom-elements-no-builtin on top of the page should patch IE 11 and other legacy browsers.

    How to avoid bundling µce per each component?

    This module reserves, in the Custom Elements Registry a uce-lib class, which only purpose is to provide all exports as static getters.

    // whenever uce library is loaded
    customElements
      .whenDefined('uce-lib')
      .then(({define, render, html, svg} = customElements.get('uce-lib')) => {
        // that's it: ready to go 🎉
        define('my-component', {
          init() {
            console.log('this is awesome!');
          }
        });
      }
    );

    Using a helper

    "There's a module for that", it's called once-defined:

    import when from 'once-defined';
    
    when('uce-lib').then(({define, render, html, svg}) => {
      // define your Custom Element
    });
    Without classes, how does one define private properties?

    Private properties can be created via a WeakMap, which is indeed how Babel transforms these anyway.

    const privates = new WeakMap;
    define('ce-with-privates', {
      init() {
        // define these once
        privates.set(this, {test: 1, other: '2'});
      },
      method() {
        // and use it anywhere you need them
        const {test, other} = privates.get(this);
        console.log(test, other);
      }
    });
    Without classes, how does one extend other components?

    There are at least two ways to extend an uce component:

    • define via uce your base component, and use extends: "base-comp-name" to extend it (built-ins supported!)
    • use one or more mixin through object literals

    Object literals have indeed been used as mixin for a very long time, and the pattern with uce would be very similar.

    The only warning is that Object.assign, as well as object {...spread}, lose getters and setters in the process, so that if you want to extend more complex components, you should consider using assignProperties, or a similar helper.

    import $ from 'assign-properties';
    const mixin = (...components) => $({}, ...components);
    
    // a component literal definition
    const NamedElement = {
      get name () { return this.tagName; }
    };
    
    // a generic NamedElement mixin
    const FirstComponent = mixin(NamedElement, {
      method() {
        console.log(this.name);
      }
    });
    
    // define it via the FirstComponent mixin
    define('first-component', FirstComponent);
    
    // define it via mixin
    define('first-component', mixin(FirstComponent, {
      otherThing() {}
    }));
    How different is µce from others?

    I have written a gist that compares uce vs lit-element, so that most obvious differences are highlighted, but basically uce provides pretty much everything other libraries provide and vice-versa, and choosing one or another should be driven by personal taste and style, as long as most relevant differences are clear.

    That is: uce is neither superior nor inferior to others, it tries to be as simple and concise as possible, and it has great pontentials when used via uce-template too.

    What happened between 1.2 and 1.5?

    A wrong npm publish happened, as 1.5.0 has been pushed for no reason between 0.5 an 0.6, so that latest was picking up actually an older version of the library.

    My apologies.

    What's new in v1.2?

    So far, the only missing utility for non Shadow DOM cases, is a way to define once a generic style associated with a component, which is why the special style: (selector) => css property has been added, so that any component can automatically define any specific style, using the selector to confine inner nodes directives.

    The css export is a plain template literal tag, which is completely optional, but it might help minifiers, or rollup plugins, to minify that code too.

    // note: the css import is optional
    import {define, css} from 'uce';
    
    define('very-important', {
      style: sel => css`
        ${sel} {
          font-weight: bold;
          text-transform: uppercase;
        }
        ${sel}:hover {
          font-size: 2rem;
        }
      `
    });

    If the element doesn't extend a built-in, the received sel, as selector, will simply be its name, otherwise it'll be the built-in name with its [is="..."] attribute.

    Please note the style won't interfere, or be attached anyhow, with the regular element.style or this.style, within a method, which is actually why I've chosen that name, so it's clear it's about the generic class/component style, and not its property.

    Install

    npm i uce

    DownloadsWeekly Downloads

    169

    Version

    1.16.5

    License

    ISC

    Unpacked Size

    104 kB

    Total Files

    10

    Last publish

    Collaborators

    • webreflection