Skip to content

Как создавать кастомный рендерер для React 16

Notifications You must be signed in to change notification settings

Asmadek/Making-a-custom-React-renderer

 
 

Repository files navigation

Создание кастомного рендерера для React 16

Введение

Данная статья является вольным переводом урока от nitin42, ссылка на оригинал https://github.com/nitin42/Making-a-custom-React-renderer

Данная статья состоит из пяти частей. Первая часть даёт представление о том, что такое рендереры и с чем их едят. А оставшиеся четыре - о этапах создания самописного рендерера.

Что такое рендерер

React, которым мы все пользуемся для разработки сайтов состаит из двух основных библиотек - React и ReactDOM. Создавать различные компоненты нам помогает React, а вот ReactDOM занимается превращением их в HTML для отображения в браузере.

Вот этот самый ReactDOM и является тем самым рендерером. Другими словами, рендерер это библиотека которая позволяет визуализировать наш набор компонентов с данными внутри них.

Кроме ReactDOM, отвественного за создание веб-страниц, существует большое количество рендереров для совершенно разных платформ. Самым известным примером является React-Native, для создание приложений на Android и iOS с помощью React.

Основная цель создания кастомных рендереров - распространение идеалогии React - "Выучил один раз - пиши везде". Так уже сейчас на React можно создавать приложения для мобильных платформ, VR, AR, IoT и другие направления.

Для рендереров даже есть свой awesome-репозиторий, где вы можете найти несколько примером рендереров.

До недавнего времени создание своих рендереров было затруднено текущим состоянием библиотеки React, которая была не так радушна для создателей своих рендереров. Но с приходом Fiber ситуация сильно изменилась. API значительно упростилось и теперь все карты в руках разработчиков.

И в этой статье будет изложен пример создания своего рендерера, который будет из имеющихся компонентов React генерировать Word-документ. За основу будет взята библиотека officegen, которая позволяет генерировать Open Office XML файлы, другими словами файлы формата docx.

Part-I

This is part one of the tutorial. In this section, we will create a React reconciler for the current targeted version of React 16.0.0-alpha.4. We are going to implement the renderer using Fiber. Earlier, React was using a stack renderer as it was implemented on the traditional JavaScript stack. On the other hand, Fiber is influenced by algebraic effects and functional ideas. It can be thought of as a JavaScript object that contains information about a component, its input, and its output.

Before we proceed further, I'll recommend you to read this documentation on Fiber architecture by Andrew Clark. This will make things easier for you.

Let's get started!

We will first install the dependencies.

npm install react-dom@16.0.0-alpha.4 fbjs@0.8.4

Let's import ReactFiberReconciler from react-dom and other modules also.

import ReactFiberReconciler from 'react-dom/lib/ReactFiberReconciler';
import emptyObject from 'fbjs/lib/emptyObject';
import createElement from './utils/createElement';

Notice we have also imported createElement function. Don't worry, we will implement it afterwards.

We will create a reconciler instance using ReactFiberReconciler which accepts a host config object. In this object we will define some methods which can be thought of as lifecycle of a renderer (update, append children, remove children, commit). React will manage all the non-host components, stateless and composites.

const WordRenderer = ReactFiberReconciler({

  // Creates component instance
  createInstance(
    type,
    props,
    rootContainerInstance,
    hostContext,
    internalInstanceHandle,
  ) {
    return createElement(type, props, rootContainerInstance);
  },
  
  appendInitialChild(parentInstance, child) {
    if (parentInstance.appendChild) {
      parentInstance.appendChild(child);
    } else {
      parentInstance.document = child;
    }
  },

  appendChild(parentInstance, child) {
    if (parentInstance.appendChild) {
      parentInstance.appendChild(child);
    } else {
      parentInstance.document = child;
    }
  },

  removeChild(parentInstance, child) {
    parentInstance.removeChild(child);
  },

  insertBefore(parentInstance, child, beforeChild) {
    // noop
  },

  // This is the final method which is called before flushing the root component to the host environment.
  finalizeInitialChildren(testElement, type, props, rootContainerInstance) {
    return false;
  },

  prepareUpdate(testElement, type, oldProps, newProps, hostContext) {
    return true;
  },

  commitUpdate(
    instance,
    type,
    oldProps,
    newProps,
    rootContainerInstance,
    internalInstanceHandle,
  ) {
    // noop
  },
  
  // This is called after initializeFinalChildren
  commitMount(
    instance,
    type,
    newProps,
    rootContainerInstance,
    internalInstanceHandle,
  ) {
    // noop
  },

  getRootHostContext() {
    return emptyObject;
  },

  getChildHostContext() {
    return emptyObject;
  },
  
  getPublicInstance(inst) {
    return inst;
  },

  // These are necessary for any global side-effects that you need to produce in the host environment
  
  prepareForCommit() {
    // noop
  },

  resetAfterCommit() {
    // noop
  },
  
  // The following methods are for the specific text nodes. In our example, we don't have any specific text nodes so we return false or noop them
  
  shouldSetTextContent(props) {
    return false;
  },

  resetTextContent(testElement) {
    // noop
  },

  createTextInstance(
    text,
    rootContainerInstance,
    hostContext,
    internalInstanceHandle,
  ) {
    return text;
  },

  commitTextUpdate(textInstance, oldText, newText) {
    textInstance.chidren = newText;
  },
  
  useSyncScheduling: true,
});

Let's break down our host config -

createInstance

This method creates a component instance with type, props, rootContainerInstance, hostContext and internalInstanceHandle.

Example - Let's say we render,

<Text>Hello World</Text>

createInstance will then return the information about the type of an element (' TEXT '), props ( { children: 'Hello World' } ), rootContainerInstance(WordDocument), hostContext ({}) and internalInstanceHandle.

internalInstanceHandle contains information about the tag, type, key, stateNode, and the return fiber. This object (fiber) further contains information about -

  • tag
  • key
  • type
  • stateNode
  • return
  • child
  • sibling
  • index
  • ref
  • pendingProps
  • memoizedProps
  • updateQueue
  • memoizedState
  • effectTag
  • nextEffect
  • firstEffect
  • pendingWorkPriority
  • progressedPriority
  • progressedChild
  • progressedFirstDeletion
  • progressedLastDeletion
  • alternate

appendInitialChild

It appends the children. If children are wrapped inside a parent component (eg: Document), then we will add all the children to it else we will create a property called document on a parent node and append all the children to it. This will be helpful when we will parse the input component and make a call to the render method of our component.

Example -

const data = document.render(); // returns the output

prepareUpdate

It computes the diff for an instance. Fiber can reuse this work even if it pauses or abort rendering a part of the tree.

commitUpdate

Commit the update or apply the diff calculated to the host environment's node (WordDocument).

hostContext

Host context is an internal object which our renderer may use based on the current location in the tree. In DOM, this object is required to make correct calls for example to create an element in html or in MathMl based on current context.

getPublicInstance

This is an identity relation which means that it always returns the same value that was used as its argument. It was added for the TestRenderers.

We're done with the Part One of our tutorial. I know some concepts are difficult to grok solely by looking at code. Initially it feels agitating but keep trying it and it will eventually make sense. When I first started learning about the Fiber architecture, I couldn't understand anything at all. I was frustated and dismayed but I used console.log() in every section of the above code and tried to understand its implementation and then there was this "Aha Aha" moment and it finally helped me to build redocx. Its a little perplexing to understand but you will get it eventually.

If you still have any doubts, DMs are open. I'm at @NTulswani on Twitter.

Continue to Part-II

About

Как создавать кастомный рендерер для React 16

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • JavaScript 100.0%