function createDom(fiber) {
const dom =
fiber.type === "TEXT_ELEMENT"
? document.createTextNode(fiber.props.nodeValue)
: document.createElement(fiber.type);
Object.keys(fiber.props).forEach((key) => {
if (key !== "children") {
if (key === "style") {
Object.keys(fiber.props.style).forEach((styleKey) => {
dom.style[styleKey] = fiber.props.style[styleKey];
});
} else {
dom[key] = fiber.props[key];
}
}
});
return dom;
}
function createTextNode(text) {
return {
type: "TEXT_ELEMENT",
props: {
nodeValue: text,
children: [],
},
};
}
function createElement(type, props, ...children) {
return {
type,
props: {
...props,
children: children.map((child) => {
return typeof child === "object" ? child : createTextNode(child);
}),
},
};
}
let nextUnitOfWork = null;
let wipRoot = null;
let currentRoot = null;
let deletions = null;
function commitRoot() {
deletions.forEach(commitWork);
commitWork(wipRoot.child);
currentRoot = wipRoot;
wipRoot = null;
}
function commitWork(fiber) {
if (!fiber) return;
let domParentFiber = fiber.parent;
while (!domParentFiber.dom) {
domParentFiber = domParentFiber.parent;
}
const domParent = domParentFiber.dom;
if (fiber.effectTag === "PLACEMENT" && fiber.dom) {
domParent.appendChild(fiber.dom);
} else if (fiber.effectTag === "DELETION") {
commitDeletion(fiber, domParent);
} else if (fiber.effectTag === "UPDATE" && fiber.dom) {
updateDom(fiber.dom, fiber.alternate.props, fiber.props);
}
commitWork(fiber.child);
commitWork(fiber.sibling);
}
function commitDeletion(fiber, domParent) {
if (fiber.dom) {
domParent.removeChild(fiber.dom);
} else {
commitDeletion(fiber.child, domParent);
}
}
function updateDom(dom, prevProps, nextProps) {
const isEvent = (key) => key.startsWith("on");
Object.keys(prevProps)
.filter(isEvent)
.filter((key) => !(key in nextProps) || prevProps[key] != nextProps[key])
.forEach((key) => {
const eventType = key.toLowerCase().substring(2);
dom.removeEventListener(eventType, prevProps[key]);
});
Object.keys(nextProps)
.filter(isEvent)
.filter((key) => prevProps[key] != nextProps[key])
.forEach((key) => {
const eventType = key.toLowerCase().substring(2);
dom.addEventListener(eventType, nextProps[key]);
});
Object.keys(prevProps)
.filter((key) => key !== "children")
.filter((key) => !(key in nextProps))
.forEach((key) => {
dom[key] = "";
});
Object.keys(nextProps)
.filter((key) => key !== "children")
.filter((key) => !(key in nextProps) || prevProps[key] !== nextProps[key])
.forEach((key) => {
dom[key] = nextProps[key];
});
}
function workLoop(deadline) {
let shouldYield = false;
while (nextUnitOfWork && !shouldYield) {
console.log("workLoop");
nextUnitOfWork = performUnitOfWork(nextUnitOfWork);
shouldYield = deadline.timeRemaining() < 1;
}
requestIdleCallback(workLoop);
if (!nextUnitOfWork && wipRoot) {
commitRoot();
}
}
requestIdleCallback(workLoop);
function updateHostComponent(fiber) {
if (!fiber.dom) {
fiber.dom = createDom(fiber);
}
reconcileChildren(fiber, fiber.props.children);
}
let wipFiber = null;
let hookIndex = 0;
function useState(init) {
const oldHook = wipFiber.alternate && wipFiber.alternate.hooks[hookIndex];
const hook = {
state: oldHook ? oldHook.state : init,
queue: [],
};
const actions = oldHook ? oldHook.queue : [];
actions.forEach((action) => (hook.state = action(hook.state)));
setState = (action) => {
hook.queue.push(action);
wipRoot = {
dom: currentRoot.dom,
props: currentRoot.props,
alternate: currentRoot,
};
nextUnitOfWork = wipRoot;
deletions = [];
};
wipFiber.hooks.push(hook);
hookIndex++;
return [hook.state, setState];
}
function updateFunctionComponent(fiber) {
wipFiber = fiber;
hookIndex = 0;
wipFiber.hooks = [];
fiber.dom = null;
const children = [fiber.type(fiber.props)];
reconcileChildren(fiber, children);
}
function performUnitOfWork(fiber) {
if (!fiber.dom) {
fiber.dom = createDom(fiber);
}
const isFunctionComponent = typeof fiber.type === "function";
if (isFunctionComponent) {
updateFunctionComponent(fiber);
} else {
updateHostComponent(fiber);
}
if (fiber.child) {
return fiber.child;
}
let nextFiber = fiber;
while (nextFiber) {
if (nextFiber.sibling) {
return nextFiber.sibling;
}
nextFiber = nextFiber.parent;
}
}
function reconcileChildren(wipFiber, elements) {
let index = 0;
let oldFiber = wipFiber.alternate && wipFiber.alternate.child;
let prevSibling = null;
let newFiber = null;
while (index < elements.length || oldFiber) {
let element = elements[index];
const sameType = oldFiber && element && oldFiber.type === element.type;
if (sameType) {
newFiber = {
type: oldFiber.type,
props: element.props,
dom: oldFiber.dom,
parent: wipFiber,
alternate: oldFiber,
effectTag: "UPDATE",
};
}
if (element && !sameType) {
newFiber = {
type: element.type,
props: element.props,
dom: null,
parent: wipFiber,
alternate: null,
effectTag: "PLACEMENT",
};
}
if (oldFiber && !sameType) {
oldFiber.effectTag = "DELETION";
deletions.push(oldFiber);
}
if (oldFiber) {
oldFiber = oldFiber.sibling;
}
if (index === 0) {
wipFiber.child = newFiber;
} else {
prevSibling.sibling = newFiber;
}
prevSibling = newFiber;
index++;
}
}
function render(element, container) {
console.log(element, container);
wipRoot = {
dom: container,
props: {
children: [element],
},
alternate: currentRoot,
sibling: null,
child: null,
parent: null,
};
nextUnitOfWork = wipRoot;
deletions = [];
}
const React = {
render,
createElement,
};
export default React;
import React from "./core/react.js";
const App = React.createElement(
"div",
{ id: "app", style: { color: "red" } },
"hello world",
React.createElement(
"div",
{ style: { backgroundColor: "yellow" } },
React.createElement("span", {}, "span1"),
),
React.createElement("div", {}, "span2"),
);
React.render(App, document.getElementById("root"));