import { createContext, useContextSelector } from '@fluentui/react-context-selector';
import { mount } from 'enzyme';
import * as React from 'react';
const TestContext = createContext<{ index: number }>({ index: -1 });
const TestComponent: React.FC<{ index: number; onUpdate?: () => void }> = props => {
const active = useContextSelector(TestContext, v => v.index === props.index);
React.useEffect(() => {
props.onUpdate && props.onUpdate();
});
return
;
};
describe('useContextSelector', () => {
it('propogates values via Context', () => {
const wrapper = mount(
,
);
expect(wrapper.find('div').prop('data-active')).toBe(true);
});
it('updates only on selector match', () => {
const onUpdate = jest.fn();
const wrapper = mount(
,
);
expect(wrapper.find('div').prop('data-active')).toBe(false);
expect(onUpdate).toBeCalledTimes(1);
// No match, (v.index: 2, p.index: 1)
wrapper.setProps({ value: { index: 2 } });
expect(wrapper.find('div').prop('data-active')).toBe(false);
expect(onUpdate).toBeCalledTimes(1);
// Match => update, (v.index: 1, p.index: 1)
wrapper.setProps({ value: { index: 1 } });
expect(wrapper.find('div').prop('data-active')).toBe(true);
expect(onUpdate).toBeCalledTimes(2);
// Match previous => no update, (v.index: 1, p.index: 1)
wrapper.setProps({ value: { index: 1 } });
expect(wrapper.find('div').prop('data-active')).toBe(true);
expect(onUpdate).toBeCalledTimes(2);
});
it('updates are propogated inside React.memo()', () => {
// https://reactjs.org/docs/react-api.html#reactmemo
// Will never pass updates
const MemoComponent = React.memo(TestComponent, () => true);
const onUpdate = jest.fn();
const wrapper = mount(
,
);
wrapper.setProps({ value: { index: 1 } });
expect(wrapper.find('div').prop('data-active')).toBe(true);
expect(onUpdate).toBeCalledTimes(2);
});
it('handles unsubscribe', () => {
const MemoComponent = React.memo(TestComponent);
const onUpdate = jest.fn();
const wrapper = mount(
,
);
wrapper.setProps({
children: [null, ],
});
expect(onUpdate).toBeCalledTimes(1);
});
});