Home

Awesome

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

OptionDefaultsDescription
unsafeHoistInClassfalseWhen 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.
warnIfCantHoistfalseWhen true, display a warnings about child functions that couldn't be hoisted.
loosefalseWhen 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>
    );
  }
}