0.9.0 โ€ข Published 2 months ago

@lirx/dom v0.9.0

Weekly downloads
-
License
MIT
Repository
github
Last release
2 months ago

comment: <> (https://github.com/tusharmath/reactive-dom#virtualdomvsreactivedom)

@lirx/dom gives you many tools to write and build dynamic html templates and components with a syntax similar to Angular. However, by design, it strongly outperforms React and Angular due to its unique architecture: when an Observable changes, it updates only the relevant DOM nodes, ensuring maximal efficiency.

To simplify: you'll create some dynamic variables, and @lirx/dom will take care to refresh the DOM, with the precision of a surgeon and the speed of a rocket.

Currently @lirx/dom is a niche framework, but it has been built to have the absolute best performances.

The current frameworks mostly use differ algorithms on complex data structures to refresh the DOM. The developers have a very simple and intuitive way to define their data, and see them magically reflected on the DOM. However, it costs a lot of javascript execution time, especially when changes append on the top-most parents. This cost is usually ignored because our machines are fast enough in most cases. And let's be honest: developers are lazy or constrained by time by the direction. Non optimized code is common and frequent.

@lirx/dom is done for perfectionists: using Observables constraint you to work in Reactive Programming, using dynamic variables and Stores. It's complex, and discouraging for beginners, but it's so much more powerful and less error-prone. Observables are extremely strong, and gives you full control over async streams of data. @lirx/dom does a great usage of them, which optimizes the rendering of the DOM to the maximum. It results in performances as fast as native DOM manipulation, and code as small as possible, at the cost of a more complicated initial learning curve.

In these cases, @lirx/dom is perfect:

  • you're comfortable with observables
  • you want to create some components embedded and working everywhere, with very small footprints
  • you want to create an application requiring critical performances (ex: displaying millions of DOM nodes, or working on very low-end devices)
  • you have the time a play with an exotic framework

    However, it's overcomplicated in these cases:

  • you're a beginner in javascript and never head of Observables

  • you don't give a f*ck about performances (like probably most of the developers ๐Ÿ˜’)

Moreover, it comes with an AOT plugin for rollup, which strongly optimizes your components and generates very small bundles.

It's light, fast, and simple ! Give it a try !

๐Ÿ“‘ Example

/** COMPONENT **/

// the interface of the data available in the template
interface IData {
  readonly $input: IObserver<string>;
  readonly input$: IObservable<string>;
  readonly remaining$: IObservable<number>;
  readonly valid$: IObservable<boolean>;
}

interface IAppHelloWorldComponentConfig {
  data: IData;
}

export const AppHelloWorldComponent = createComponent<IAppHelloWorldComponentConfig>({
  name: 'app-hello-world',
  template: compileReactiveHTMLAsComponentTemplate({
    html: `
      <div class="input-container">
        <input
          [value]="$.input$"
          (input)="() => $.$input(node.value)"
        >
      </div>
      <div
        class="max-length-container"
        [class.valid]="$.valid$"
      >
        Length: {{ $.remaining$ }} / 10
      </div>
   `,
  }),
  styles: [compileStyleAsComponentStyle(`
    :host {
      display: block;
    }

    :host > .max-length-container:not(.valid) {
      color: red;
    }
  `)],
  init: (node: VirtualCustomElementNode<IAppHelloWorldComponentConfig>): IData => {
    // 'input' is a source which contains and emits the value of our input
    const { emit: $input, subscribe: input$ } = let$$('');

    // 'remaining' is an observable whose value is computed from the length of 'input'
    const remaining$ = map$$(input$, (value: string) => value.length);

    // 'valid' is an observable whose value is true if 'remaining' is less than 10
    const valid$ = map$$(remaining$, (value: number) => (value <= 10));

    return {
      $input,
      input$,
      remaining$,
      valid$,
    };
  },
});

Click here to see the live demo

๐Ÿ“ฆ Installation

yarn add @lirx/dom
# or
npm install @lirx/dom--save

Click here to read the installation manual


Differences with other popular frameworks:

FeatureAngularVirtual DOM (React, Vue)@lirx/dom
Semanticshtml with special flavourjsx or hyperscripthtml with special flavour
Memorymedium: data are directly reflected on the nodes, but the framework itself is heavyhigh a lot of virtual DOM elements are created every time the DOM updates, and the number of virtual nodes is also linearly proportional to the size of the DOM tree.very low: once the data pipeline is set, on every update the data is directly reflected on the node.
CPUmedium: when zoneJs triggers, all expressions in the html are evaluated and reflected on the nodeshigh because a lot of time is spent regenerating the Virtual DOM, calculating the diff and figuring out what changed.low: the nodes subscribe only to the part of the data that is needed for rendering / updating them. It's almost unbeatable, because when the data changes, it directly updates the nodes.
Size~50KB~10KB (preact)~8KB (with jit compiler), ~4KB (aot)

*size is calculated for similar 'hello world' projects, compiled, minified and gzipped.

The current frameworks are pretty efficient, however, they are far from the perfect optimization. @lirx/dom tries to do better by conciliating an elegant syntax with maximal performances.

For new incomers, learning observables may be discouraging, as it is a totally different manner to think your code, but once you're comfortable with this principle, you'll fully enjoy the potential, and the performances they provide:

  • fewer errors, especially on computed properties
  • better resource managements: cancellation is part of observables
  • faster rendering and updating

Obviously, current popular frameworks are more mature and offers more tools, having a very important community. However, this project may close the gap in the future.