import React, { memo, useContext, useRef, useState } from 'react'; import Menu from '../index'; import MenuContext from '../MenuContext'; import { render, fireEvent } from '../../../tests/utils'; // we use'memo' here in order to only render inner component while context changed. const CacheInner = memo(() => { const countRef = useRef(0); countRef.current++; // subscribe anchor context useContext(MenuContext); return (
Child Rendering Count: {countRef.current}
); }); const CacheOuter = () => { // We use 'useState' here in order to trigger parent component rendering. const [count, setCount] = useState(1); const handleClick = () => { setCount(count + 1); }; // During each rendering phase, the cached context value returned from method 'Menu#getMemoizedContextValue' will take effect. // So 'CacheInner' component won't rerender. return (
Parent Rendering Count: {count}
); }; it("Rendering on Menu without changed MenuContext won't trigger rendering on child component.", () => { const { container, unmount } = render(); const childCount = container.querySelector('#child_count')?.textContent; fireEvent.click(container.querySelector('#parent_btn')!); expect(container.querySelector('#parent_count')?.textContent).toBe('2'); // child component won't rerender expect(container.querySelector('#child_count')?.textContent).toBe(childCount); fireEvent.click(container.querySelector('#parent_btn')!); expect(container.querySelector('#parent_count')?.textContent).toBe('3'); // child component won't rerender expect(container.querySelector('#child_count')?.textContent).toBe(childCount); // in order to depress warning "Warning: An update to Menu inside a test was not wrapped in act(...)." unmount(); });