Skip to content

Latest commit

 

History

History
115 lines (85 loc) · 3.48 KB

README.md

File metadata and controls

115 lines (85 loc) · 3.48 KB

babel-plugin-hoist-facc

Treat FACC as value types and hoist them to the highest scope

Install

Using npm:

npm install --save-dev @babel/core babel-plugin-hoist-facc

or using yarn:

yarn add @babel/core babel-plugin-hoist-facc --dev

What problem it solves?

It solves the problem of rendering optimization, which comes with FaCC pattern. In a few words: Function as Child Component can't be optimized by shouldComponentUpdate because the child function is always changes on render.

This plugin hoist the child function to highest possible scope to prevent its changes.

Example

class extends React.Component {
  render() {
    return <FaCC>{() => <div />}</FaCC>; // arrow function changes on each render
  }
}

⇩⇩⇩

var _ref = () => <div />;

class extends React.Component {
  render() {
    return <FaCC>{_ref}</FaCC>; // now it's constant
  }
}

Options

Option Defaults Description
unsafeHoistInClass false When true, enables the hoisting of FaCC children function, which refers to this, to the class constructor. Potentially it is unsafe due to context losing issue, but in regular React life this should`t happens. See example below.
warnIfCantHoist false When true, display a warnings about child functions that couldn't be hoisted.
loose false When true, class properties are compiled to use an assignment expression instead of Object.defineProperty.

More examples

["hoist-facc", { unsafeHoistInClass: true, loose: true }]

class extends React.Component {
  handleClick = () => {};

  render() {
    return <FaCC>{() => <button onClick={this.handleClick} />}</FaCC>;
  }
}

⇩⇩⇩

class extends React.Component {
  constructor(...args) {
    super(...args);

    this.handleClick = () => {};
    this._ref = () => <button onClick={this.handleClick} />;
  }

  render() {
    return <FaCC>{this._ref}</FaCC>;
  }
}

Pitfalls

There some cases it can't be hoisted automagically.

class extends React.Component {
  render() {
    const { id, label } = this.props;

    return <FaCC>{() => <div id={id}>{label}</div>}</FaCC>;
  }
}

The above code will not be transformed, because the id and label from the higher scope is used inside FaCC. You should proxy the props to the child function or access props directly trought this keyword to avoid this.

class extends React.Component {
  render() {
    return (
      <FaCC {...this.props}>
        {({ id }) => (
          <div id={id}>{this.props.label}</div>
        )}
      </FaCC>
    );
  }
}