Skip to content

WebReflection/p-cool

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

18 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Pretty Cool Elements

Social Media Photo by Jamison McAndie on Unsplash

This module is a follow up of this Medium post, and it provides element mixins/behaviors, through class names, without names clashing.

Features

  • it addresses every single point touched in the Medium's post:
    • no name clashing
    • multiple mixins/behaviors attached/detached at any time
    • native Custom Elements builtin callbacks, associated to mixins/behaviors
  • it's Server Side Rendering compatible out of the box
  • it uses all the DOM primitives without needing an extra attribute (bloat-free layouts)
  • it's semantically bound with element's view (their classes and their dedicated style)
  • it's graceful enchancement out of the box, based on builtin extends
  • it provides a robust polyfilled version through vanilla-elements

Example

import {define} from 'p-cool';

define('my-div', {

  // to know when a behavior is attached or detached via class
  attachedCallback(element) {},
  detachedCallback(element) {}, // see ## About Callbacks

  // to observe connected/disconnected lifecycle
  connectedCallback(element) {},
  disconnectedCallback(element) {},

  // to observe specific attributes (omit to observe them all)
  observedAttributes: ['some-attribute'],

  // to know when observed attributes changed
  attributeChangedCallback(element, name, oldValue, newValue) {},
});
<div is="p-cool-div" class="my-div" some-attribute="ok">
  Hello Behaviors 👋
</div>

About PCool

With native Custom Elements, we need to reserve a single name in a shared global registry to pass through the upgrade, and callbacks, mechanism.

With p-cool elements, the registry is pre-populated with vanilla-elements extends through a p-cool-* prefix, so that <div is="p-cool-div">, and <p is="p-cool-p">, or <main is="p-cool-main"> are all valid, already registered, builtin extends, that brings mixins/behaviors to any element, and through their class name, as long as one, or mixin, is defined/attached, through the define(name, mixin) module's export.

<!doctype html>
<script type="module">
import {define} from '//unpkg.com/p-cool';

import Hero from './mixins/hero.js';
define('hero', Hero);

import Bottom from './mixins/bottom.js';
import AutoHide from './mixins/auto-hide.js';
define('bottom', Bottom);
define('auto-hide', AutoHide);
</script>
<style>
main { /* ... */ }
.hero { /* ... */ }
.bottom { /* ... */ }
</style>
<body is="p-cool-body" class="hero">
  <main>Hero</main>
  <footer is="p-cool-footer" class="bottom auto-hide">
    ...
  </footer>
</body>

To implement an element extend, the <p-cool> Custom Element is registered too, so that a page could be defined by non-builtin extends, with mixins/behaviors attached when, and if, needed.

<!doctype html>
<script type="module">
import {define} from '//unpkg.com/p-cool';

import Hero from './mixins/hero.js';
define('hero', Hero);

import Bottom from './mixins/bottom.js';
import AutoHide from './mixins/auto-hide.js';
define('bottom', Bottom);
define('auto-hide', AutoHide);
</script>
<body>
  <p-cool class="hero"></p-cool>
  <p-cool class="bottom auto-hide">
    ...
  </p-cool>
</body>

About Callbacks

attachedCallback

This callback is granted to be invoked only once, and before any other callback, whenever a mixin/behavior is attached through the element's class, somehow simulating what a constructor would do with Custom Elements.

This callback is ideal to add related event listeners, setup an element for the specific mixin/behavior, and so on.

Please note that if a mixin/behavior is detached, and then re-attached, this callback will be invoked again.

attributeChangedCallback

If any observedAttributes is specified, or if there is an attributeChangedCallback, this is invoked every time observed attributes change.

Like it is for Custom Elements, this callback is invoked, after a mixin/behavior is attached, hence after attachedCallback, but before connectedCallback.

This callback is also invoked during the element lifecycle, whenever observed attributes change, providing the oldValue and the newValue.

Both values are null if there was not attribute, or if the attribute got removed, replicating the native Custom Element behavior.

connectedCallback

This callback is granted to be invoked after an element gets a new mixin/behavior, if the element is already live, and every other time the element gets moved or re-appended on the DOM, exactly like it is for native Custom Elements.

Please note that when a mixin/behavior is attached, and there are observed attributes, this callback will be invoked after attributeChangedCallback.

disconnectedCallback

This callback is granted to be invoked when an element gets removed from the DOM, and it would never trigger if the connectedCallback didn't happen already.

Both callbacks are the ideal place to attach, on connected, and remove, on disconnected, timers, animations, or idle related callbacks, as even when elements get trashed, both callbacks are granted to be executed, and in the right order of events.

detachedCallback

This callback is not granted to be invoked if an element get trashed, but it's granted to be invoked after disconnectedCallback, if a mixin/behavior is removed from an element.

Please note that this callback is not really useful for elements that might be, or may not be, trashed, because there is no way to use a FinalizationRegistry and pass along the element, but it's very hando for those elements that never leave the DOM, but might change, over time, their classes, hence their mixins/behaviors.

import {define} from 'p-cool';

define('mixin', {
  attachedCallback(element) {
    console.log('mixin attached');
  },
  detachedCallback(element) {
    console.log('mixin detached');
  }
});

// example
document.body.innerHTML = `
  <div id="first" class="mixin">First</div>
  <div id="second" class="mixin">Second</div>
`;
// logs "mixin attached" twice

// will **not** "mixin detached"
first.remove();

// it **will** log "mixin detached"
second.classList.remove('mixin');

About Exports

This module offers the following exports:

  • p-cool with a define(name, mixin) export that does not polyfill Safari
  • p-cool/min with a minified define(name, mixin) export that does not polyfill Safari
  • p-cool/poly with a minified define(name, mixin) export that also does polyfill Safari
  • p-cool/behaviors with the internally used define and behaviors exports, plus constants, useful to potentially create other libraries or utilities on top of the same logic

The https://unpkg.com/p-cool points at the minified /poly variant, useful to quickly test, or develop, with this module.

Compatibility

Every ES2015+ compatible browser out of the box, including Safari/WebKit based browsers in the poly version.

About

Pretty Cool Elements

Resources

License

Stars

Watchers

Forks

Packages

No packages published