static-cms/packages/core/test/harnesses/widget.harness.tsx
2023-09-06 16:30:51 -04:00

71 lines
2.2 KiB
TypeScript

/* eslint-disable import/prefer-default-export */
import { act } from '@testing-library/react';
import React from 'react';
import MediaLibraryModal from '@staticcms/core/components/media-library/MediaLibraryModal';
import { store } from '@staticcms/core/store';
import { createMockWidgetControlProps } from '@staticcms/test/data/widgets.mock';
import { renderWithProviders } from '@staticcms/test/test-utils';
import type { BaseField, UnknownField, WidgetControlProps } from '@staticcms/core/interface';
import type { FC } from 'react';
export interface WidgetControlHarnessOptions {
useFakeTimers?: boolean;
withMediaLibrary?: boolean;
}
export const createWidgetControlHarness = <T, F extends BaseField = UnknownField>(
Component: FC<WidgetControlProps<T, F>>,
defaults: Omit<Partial<WidgetControlProps<T, F>>, 'field'> &
Pick<WidgetControlProps<T, F>, 'field'>,
options?: WidgetControlHarnessOptions,
) => {
type Params = Parameters<typeof createMockWidgetControlProps<T, F>>[0];
type Props = Omit<Params, 'field'> & Pick<Partial<Params>, 'field'>;
return (renderProps?: Props, renderOptions?: WidgetControlHarnessOptions) => {
const { useFakeTimers = false, withMediaLibrary = false } = renderOptions ?? options ?? {};
if (useFakeTimers) {
jest.useFakeTimers({ now: new Date(2023, 1, 12, 10, 15, 35, 0) });
} else {
jest.useRealTimers();
}
const field = renderProps?.field ?? defaults.field;
const props = createMockWidgetControlProps<T, F>({ ...defaults, ...renderProps, field });
const result = renderWithProviders(
<>
<Component {...props} />
{withMediaLibrary ? <MediaLibraryModal key="library" /> : null}
</>,
);
if (useFakeTimers) {
act(() => {
jest.advanceTimersByTime(1000);
});
}
const rerender = (rerenderProps?: Omit<Props, 'field'>) => {
const finalRerenderProps = {
...props,
...rerenderProps,
} as WidgetControlProps<T, F>;
result.rerender(
<>
<Component {...finalRerenderProps} />
{withMediaLibrary ? <MediaLibraryModal key="library" /> : null}
</>,
);
return { props: rerenderProps };
};
return { ...result, props, rerender, store };
};
};