Данная статья является вольным переводом урока от 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.
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.