Skip to content

Commit

Permalink
Improve diffing of styles (#566)
Browse files Browse the repository at this point in the history
  • Loading branch information
vadimdemedes authored Mar 26, 2023
1 parent 2a67354 commit 690d48c
Show file tree
Hide file tree
Showing 3 changed files with 89 additions and 84 deletions.
6 changes: 1 addition & 5 deletions src/dom.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
// eslint-disable-next-line n/file-extension-in-import
import Yoga, {type Node as YogaNode} from 'yoga-wasm-web/auto';
import measureText from './measure-text.js';
import applyStyles, {type Styles} from './styles.js';
import {type Styles} from './styles.js';
import wrapText from './wrap-text.js';
import squashTextNodes from './squash-text-nodes.js';
import {type OutputTransformer} from './render-node-to-output.js';
Expand Down Expand Up @@ -159,10 +159,6 @@ export const setAttribute = (

export const setStyle = (node: DOMNode, style: Styles): void => {
node.style = style;

if (node.yogaNode) {
applyStyles(node.yogaNode, style);
}
};

export const createTextNode = (text: string): TextNode => {
Expand Down
143 changes: 74 additions & 69 deletions src/reconciler.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ import {
type ElementNames,
type DOMElement
} from './dom.js';
import {type Styles} from './styles.js';
import applyStyles, {type Styles} from './styles.js';
import {type OutputTransformer} from './render-node-to-output.js';

// We need to conditionally perform devtools connection to avoid
Expand All @@ -43,6 +43,41 @@ $ npm install --save-dev react-devtools-core
}
}

type AnyObject = Record<string, unknown>;

const diff = (before: AnyObject, after: AnyObject): AnyObject | undefined => {
if (before === after) {
return;
}

if (!before) {
return after;
}

const changed: AnyObject = {};
let isChanged = false;

for (const key of Object.keys(before)) {
const isDeleted = after ? !Object.hasOwnProperty.call(after, key) : true;

if (isDeleted) {
changed[key] = undefined;
isChanged = true;
}
}

if (after) {
for (const key of Object.keys(after)) {
if (after[key] !== before[key]) {
changed[key] = after[key];
isChanged = true;
}
}
}

return isChanged ? changed : undefined;
};

const cleanupYogaNode = (node?: YogaNode): void => {
node?.unsetMeasureFunc();
node?.freeRecursive();
Expand All @@ -54,6 +89,11 @@ type HostContext = {
isInsideText: boolean;
};

type UpdatePayload = {
props: Props | undefined;
style: Styles | undefined;
};

export default createReconciler<
ElementNames,
Props,
Expand All @@ -64,7 +104,7 @@ export default createReconciler<
unknown,
unknown,
HostContext,
Props,
UpdatePayload,
unknown,
unknown,
unknown
Expand Down Expand Up @@ -126,6 +166,11 @@ export default createReconciler<

if (key === 'style') {
setStyle(node, value as Styles);

if (node.yogaNode) {
applyStyles(node.yogaNode, value as Styles);
}

continue;
}

Expand Down Expand Up @@ -206,83 +251,43 @@ export default createReconciler<
rootNode.isStaticDirty = true;
}

const updatePayload: Props = {};
const keys = Object.keys(newProps);

for (const key of keys) {
if (newProps[key] !== oldProps[key]) {
const isStyle =
key === 'style' &&
typeof newProps['style'] === 'object' &&
typeof oldProps['style'] === 'object';

if (isStyle) {
const newStyle = newProps['style'] as Styles;
const oldStyle = oldProps['style'] as Styles;
const styleKeys = Object.keys(newStyle) as Array<keyof Styles>;

for (const styleKey of styleKeys) {
// Always include `borderColor` and `borderStyle` to ensure border is rendered,
// and `overflowX` and `overflowY` to ensure content is clipped,
// otherwise resulting `updatePayload` may not contain them
// if they weren't changed during this update
if (styleKey === 'borderStyle' || styleKey === 'borderColor') {
if (typeof updatePayload['style'] !== 'object') {
// Linter didn't like `= {} as Style`
const style: Styles = {};
updatePayload['style'] = style;
}

(updatePayload['style'] as any).borderStyle =
newStyle.borderStyle;
(updatePayload['style'] as any).borderColor =
newStyle.borderColor;
(updatePayload['style'] as any).overflowX = newStyle.overflowX;
(updatePayload['style'] as any).overflowY = newStyle.overflowY;
}

if (newStyle[styleKey] !== oldStyle[styleKey]) {
if (typeof updatePayload['style'] !== 'object') {
// Linter didn't like `= {} as Style`
const style: Styles = {};
updatePayload['style'] = style;
}

(updatePayload['style'] as any)[styleKey] = newStyle[styleKey];
}
}
const props = diff(oldProps, newProps);

continue;
}
const style = diff(
oldProps['style'] as Styles,
newProps['style'] as Styles
);

(updatePayload as any)[key] = newProps[key];
}
if (!props && !style) {
return null;
}

return updatePayload;
return {props, style};
},
commitUpdate(node, updatePayload) {
for (const [key, value] of Object.entries(updatePayload)) {
if (key === 'children') {
continue;
}
commitUpdate(node, {props, style}) {
if (props) {
for (const [key, value] of Object.entries(props)) {
if (key === 'style') {
setStyle(node, value as Styles);
continue;
}

if (key === 'style') {
setStyle(node, value as Styles);
continue;
}
if (key === 'internal_transform') {
node.internal_transform = value as OutputTransformer;
continue;
}

if (key === 'internal_transform') {
node.internal_transform = value as OutputTransformer;
continue;
}
if (key === 'internal_static') {
node.internal_static = true;
continue;
}

if (key === 'internal_static') {
node.internal_static = true;
continue;
setAttribute(node, key, value as DOMNodeAttribute);
}
}

setAttribute(node, key, value as DOMNodeAttribute);
if (style && node.yogaNode) {
applyStyles(node.yogaNode, style);
}
},
commitTextUpdate(node, _oldText, newText) {
Expand Down
24 changes: 14 additions & 10 deletions test/borders.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -410,24 +410,18 @@ test('nested boxes - fit-content box with emojis on flex-direction column', t =>
t.is(output, expected);
});

test('render border after update', async t => {
test('render border after update', t => {
const stdout = createStdout();

function Test() {
const [borderColor, setBorderColor] = useState<string | undefined>();

useEffect(() => {
setBorderColor('green');
}, []);

function Test({borderColor}: {borderColor?: string}) {
return (
<Box borderStyle="round" borderColor={borderColor}>
<Text>Hello World</Text>
</Box>
);
}

render(<Test />, {
const {rerender} = render(<Test />, {
stdout,
debug: true
});
Expand All @@ -437,7 +431,7 @@ test('render border after update', async t => {
boxen('Hello World', {width: 100, borderStyle: 'round'})
);

await delay(100);
rerender(<Test borderColor="green" />);

t.is(
(stdout.write as any).lastCall.args[0],
Expand All @@ -447,4 +441,14 @@ test('render border after update', async t => {
borderColor: 'green'
})
);

rerender(<Test />);

t.is(
(stdout.write as any).lastCall.args[0],
boxen('Hello World', {
width: 100,
borderStyle: 'round'
})
);
});

0 comments on commit 690d48c

Please sign in to comment.