mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-06 06:31:27 +08:00
Merge pull request #30962 from storybookjs/jeppe/remove-deprecated-test-provider-api
Core: Remove deprecated parts of test provider API
This commit is contained in:
commit
5fdb23a03f
@ -14,15 +14,6 @@ import { store as mockStore } from '../manager-store.mock';
|
|||||||
import { TestProviderRender } from './TestProviderRender';
|
import { TestProviderRender } from './TestProviderRender';
|
||||||
|
|
||||||
const managerContext: any = {
|
const managerContext: any = {
|
||||||
state: {
|
|
||||||
testProviders: {
|
|
||||||
'test-provider-id': {
|
|
||||||
id: 'test-provider-id',
|
|
||||||
name: 'Test Provider',
|
|
||||||
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
api: {
|
api: {
|
||||||
getDocsUrl: fn(({ subpath }) => `https://storybook.js.org/docs/${subpath}`).mockName(
|
getDocsUrl: fn(({ subpath }) => `https://storybook.js.org/docs/${subpath}`).mockName(
|
||||||
'api::getDocsUrl'
|
'api::getDocsUrl'
|
||||||
|
@ -46,8 +46,6 @@ addons.register(ADDON_ID, (api) => {
|
|||||||
|
|
||||||
addons.add(TEST_PROVIDER_ID, {
|
addons.add(TEST_PROVIDER_ID, {
|
||||||
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
||||||
runnable: true,
|
|
||||||
name: 'Component tests',
|
|
||||||
render: () => {
|
render: () => {
|
||||||
const [isModalOpen, setModalOpen] = useState(false);
|
const [isModalOpen, setModalOpen] = useState(false);
|
||||||
const {
|
const {
|
||||||
@ -94,6 +92,6 @@ addons.register(ADDON_ID, (api) => {
|
|||||||
}
|
}
|
||||||
return <SidebarContextMenu context={context} api={api} />;
|
return <SidebarContextMenu context={context} api={api} />;
|
||||||
},
|
},
|
||||||
} satisfies Omit<Addon_TestProviderType, 'id'>);
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -1,73 +0,0 @@
|
|||||||
import type { Addon_TestProviderState, Addon_TestProviderType } from 'storybook/internal/types';
|
|
||||||
|
|
||||||
type DateNow = number;
|
|
||||||
|
|
||||||
export type TestProviderId = Addon_TestProviderType['id'];
|
|
||||||
export type TestProviderConfig = Addon_TestProviderType;
|
|
||||||
export type TestProviderState<Details extends { [key: string]: any } = NonNullable<unknown>> =
|
|
||||||
Addon_TestProviderState<Details>;
|
|
||||||
|
|
||||||
export type TestProviders = Record<TestProviderId, TestProviderConfig & TestProviderState>;
|
|
||||||
|
|
||||||
export type TestingModuleRunRequestPayload = {
|
|
||||||
providerId: TestProviderId;
|
|
||||||
// TODO: Avoid needing to do a fetch request server-side to retrieve the index
|
|
||||||
indexUrl: string; // e.g. http://localhost:6006/index.json
|
|
||||||
storyIds?: string[]; // ['button--primary', 'button--secondary']
|
|
||||||
};
|
|
||||||
|
|
||||||
export type TestingModuleProgressReportPayload =
|
|
||||||
| {
|
|
||||||
providerId: TestProviderId;
|
|
||||||
status: 'success' | 'pending' | 'cancelled';
|
|
||||||
cancellable?: boolean;
|
|
||||||
progress?: TestingModuleProgressReportProgress;
|
|
||||||
details?: { [key: string]: any };
|
|
||||||
}
|
|
||||||
| {
|
|
||||||
providerId: TestProviderId;
|
|
||||||
status: 'failed';
|
|
||||||
progress?: TestingModuleProgressReportProgress;
|
|
||||||
details?: { [key: string]: any };
|
|
||||||
error: {
|
|
||||||
name: string;
|
|
||||||
message: string;
|
|
||||||
stack?: string;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
| {
|
|
||||||
providerId: TestProviderId;
|
|
||||||
details: { [key: string]: any };
|
|
||||||
};
|
|
||||||
|
|
||||||
export type TestingModuleCrashReportPayload = {
|
|
||||||
providerId: TestProviderId;
|
|
||||||
error: {
|
|
||||||
message: string;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export type TestingModuleProgressReportProgress = {
|
|
||||||
startedAt: DateNow;
|
|
||||||
finishedAt?: DateNow;
|
|
||||||
numTotalTests?: number;
|
|
||||||
numPassedTests?: number;
|
|
||||||
numFailedTests?: number;
|
|
||||||
numPendingTests?: number;
|
|
||||||
percentageCompleted?: number;
|
|
||||||
};
|
|
||||||
|
|
||||||
export type Status = 'success' | 'failed' | 'pending';
|
|
||||||
|
|
||||||
export type TestingModuleCancelTestRunRequestPayload = {
|
|
||||||
providerId: TestProviderId;
|
|
||||||
};
|
|
||||||
|
|
||||||
export type TestingModuleCancelTestRunResponsePayload =
|
|
||||||
| {
|
|
||||||
status: 'success';
|
|
||||||
}
|
|
||||||
| {
|
|
||||||
status: 'failed';
|
|
||||||
message: string;
|
|
||||||
};
|
|
@ -85,12 +85,6 @@ enum events {
|
|||||||
ARGTYPES_INFO_RESPONSE = 'argtypesInfoResponse',
|
ARGTYPES_INFO_RESPONSE = 'argtypesInfoResponse',
|
||||||
CREATE_NEW_STORYFILE_REQUEST = 'createNewStoryfileRequest',
|
CREATE_NEW_STORYFILE_REQUEST = 'createNewStoryfileRequest',
|
||||||
CREATE_NEW_STORYFILE_RESPONSE = 'createNewStoryfileResponse',
|
CREATE_NEW_STORYFILE_RESPONSE = 'createNewStoryfileResponse',
|
||||||
|
|
||||||
TESTING_MODULE_CRASH_REPORT = 'testingModuleCrashReport',
|
|
||||||
TESTING_MODULE_PROGRESS_REPORT = 'testingModuleProgressReport',
|
|
||||||
TESTING_MODULE_RUN_REQUEST = 'testingModuleRunRequest',
|
|
||||||
TESTING_MODULE_CANCEL_TEST_RUN_REQUEST = 'testingModuleCancelTestRunRequest',
|
|
||||||
TESTING_MODULE_CANCEL_TEST_RUN_RESPONSE = 'testingModuleCancelTestRunResponse',
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Enables: `import Events from ...`
|
// Enables: `import Events from ...`
|
||||||
@ -155,11 +149,6 @@ export const {
|
|||||||
SAVE_STORY_RESPONSE,
|
SAVE_STORY_RESPONSE,
|
||||||
ARGTYPES_INFO_REQUEST,
|
ARGTYPES_INFO_REQUEST,
|
||||||
ARGTYPES_INFO_RESPONSE,
|
ARGTYPES_INFO_RESPONSE,
|
||||||
TESTING_MODULE_CRASH_REPORT,
|
|
||||||
TESTING_MODULE_PROGRESS_REPORT,
|
|
||||||
TESTING_MODULE_RUN_REQUEST,
|
|
||||||
TESTING_MODULE_CANCEL_TEST_RUN_REQUEST,
|
|
||||||
TESTING_MODULE_CANCEL_TEST_RUN_RESPONSE,
|
|
||||||
} = events;
|
} = events;
|
||||||
|
|
||||||
export * from './data/create-new-story';
|
export * from './data/create-new-story';
|
||||||
@ -168,5 +157,4 @@ export * from './data/argtypes-info';
|
|||||||
export * from './data/request-response';
|
export * from './data/request-response';
|
||||||
export * from './data/save-story';
|
export * from './data/save-story';
|
||||||
export * from './data/whats-new';
|
export * from './data/whats-new';
|
||||||
export * from './data/testing-module';
|
|
||||||
export * from './data/phases';
|
export * from './data/phases';
|
||||||
|
@ -24,13 +24,6 @@ import type {
|
|||||||
|
|
||||||
import { dedent } from 'ts-dedent';
|
import { dedent } from 'ts-dedent';
|
||||||
|
|
||||||
import { TEST_PROVIDER_ID as ADDON_TEST_PROVIDER_ID } from '../../../../addons/test/src/constants';
|
|
||||||
import {
|
|
||||||
TESTING_MODULE_CRASH_REPORT,
|
|
||||||
TESTING_MODULE_PROGRESS_REPORT,
|
|
||||||
type TestingModuleCrashReportPayload,
|
|
||||||
type TestingModuleProgressReportPayload,
|
|
||||||
} from '../../core-events';
|
|
||||||
import { cleanPaths, sanitizeError } from '../../telemetry/sanitize';
|
import { cleanPaths, sanitizeError } from '../../telemetry/sanitize';
|
||||||
import { initCreateNewStoryChannel } from '../server-channel/create-new-story-channel';
|
import { initCreateNewStoryChannel } from '../server-channel/create-new-story-channel';
|
||||||
import { initFileSearchChannel } from '../server-channel/file-search-channel';
|
import { initFileSearchChannel } from '../server-channel/file-search-channel';
|
||||||
@ -285,55 +278,6 @@ export const experimental_serverChannel = async (
|
|||||||
initFileSearchChannel(channel, options, coreOptions);
|
initFileSearchChannel(channel, options, coreOptions);
|
||||||
initCreateNewStoryChannel(channel, options, coreOptions);
|
initCreateNewStoryChannel(channel, options, coreOptions);
|
||||||
|
|
||||||
if (!options.disableTelemetry) {
|
|
||||||
channel.on(
|
|
||||||
TESTING_MODULE_PROGRESS_REPORT,
|
|
||||||
async (payload: TestingModuleProgressReportPayload) => {
|
|
||||||
if (payload.providerId === ADDON_TEST_PROVIDER_ID) {
|
|
||||||
// addon-test does its own telemetry
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const status = 'status' in payload ? payload.status : undefined;
|
|
||||||
const progress = 'progress' in payload ? payload.progress : undefined;
|
|
||||||
const error = 'error' in payload ? payload.error : undefined;
|
|
||||||
|
|
||||||
if ((status === 'success' || status === 'cancelled') && progress?.finishedAt) {
|
|
||||||
await telemetry('testing-module-completed-report', {
|
|
||||||
provider: payload.providerId,
|
|
||||||
duration: progress?.finishedAt - progress?.startedAt,
|
|
||||||
numTotalTests: progress?.numTotalTests,
|
|
||||||
numFailedTests: progress?.numFailedTests,
|
|
||||||
numPassedTests: progress?.numPassedTests,
|
|
||||||
status,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
if (status === 'failed') {
|
|
||||||
await telemetry('testing-module-completed-report', {
|
|
||||||
provider: payload.providerId,
|
|
||||||
status: 'failed',
|
|
||||||
...(options.enableCrashReports && {
|
|
||||||
error: error && sanitizeError(error),
|
|
||||||
}),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
channel.on(TESTING_MODULE_CRASH_REPORT, async (payload: TestingModuleCrashReportPayload) => {
|
|
||||||
if (payload.providerId === ADDON_TEST_PROVIDER_ID) {
|
|
||||||
// addon-test does its own telemetry
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
await telemetry('testing-module-crash-report', {
|
|
||||||
provider: payload.providerId,
|
|
||||||
...(options.enableCrashReports && {
|
|
||||||
error: cleanPaths(payload.error.message),
|
|
||||||
}),
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return channel;
|
return channel;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1,148 +0,0 @@
|
|||||||
import {
|
|
||||||
TESTING_MODULE_CANCEL_TEST_RUN_REQUEST,
|
|
||||||
TESTING_MODULE_RUN_REQUEST,
|
|
||||||
type TestProviderId,
|
|
||||||
type TestProviderState,
|
|
||||||
type TestProviders,
|
|
||||||
type TestingModuleRunRequestPayload,
|
|
||||||
} from 'storybook/internal/core-events';
|
|
||||||
import { Addon_TypesEnum, type StoryId } from 'storybook/internal/types';
|
|
||||||
|
|
||||||
import invariant from 'tiny-invariant';
|
|
||||||
|
|
||||||
import type { ModuleFn } from '../lib/types';
|
|
||||||
|
|
||||||
export type SubState = {
|
|
||||||
testProviders: TestProviders;
|
|
||||||
};
|
|
||||||
|
|
||||||
const initialTestProviderState: TestProviderState = {
|
|
||||||
details: {} as { [key: string]: any },
|
|
||||||
cancellable: false,
|
|
||||||
cancelling: false,
|
|
||||||
running: false,
|
|
||||||
failed: false,
|
|
||||||
crashed: false,
|
|
||||||
};
|
|
||||||
|
|
||||||
interface RunOptions {
|
|
||||||
entryId?: StoryId;
|
|
||||||
}
|
|
||||||
|
|
||||||
export type SubAPI = {
|
|
||||||
getTestProviderState(id: string): TestProviderState | undefined;
|
|
||||||
updateTestProviderState(id: TestProviderId, update: Partial<TestProviderState>): void;
|
|
||||||
clearTestProviderState(id: TestProviderId): void;
|
|
||||||
runTestProvider(id: TestProviderId, options?: RunOptions): () => void;
|
|
||||||
cancelTestProvider(id: TestProviderId): void;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const init: ModuleFn<SubAPI, SubState> = ({ store, fullAPI }) => {
|
|
||||||
const state: SubState = {
|
|
||||||
testProviders: store.getState().testProviders || {},
|
|
||||||
};
|
|
||||||
|
|
||||||
const api: SubAPI = {
|
|
||||||
getTestProviderState(id) {
|
|
||||||
const { testProviders } = store.getState();
|
|
||||||
return testProviders?.[id];
|
|
||||||
},
|
|
||||||
updateTestProviderState(id, update) {
|
|
||||||
return store.setState(
|
|
||||||
({ testProviders }) => {
|
|
||||||
const currentState = testProviders[id];
|
|
||||||
const updatedState = currentState.stateUpdater?.(currentState, update) ?? {
|
|
||||||
...currentState,
|
|
||||||
...update,
|
|
||||||
details: { ...currentState.details, ...update.details },
|
|
||||||
};
|
|
||||||
return { testProviders: { ...testProviders, [id]: updatedState } };
|
|
||||||
},
|
|
||||||
{ persistence: 'session' }
|
|
||||||
);
|
|
||||||
},
|
|
||||||
clearTestProviderState(id) {
|
|
||||||
const update = {
|
|
||||||
cancelling: false,
|
|
||||||
running: false,
|
|
||||||
failed: false,
|
|
||||||
crashed: false,
|
|
||||||
progress: undefined,
|
|
||||||
};
|
|
||||||
return store.setState(
|
|
||||||
({ testProviders }) => {
|
|
||||||
return { testProviders: { ...testProviders, [id]: { ...testProviders[id], ...update } } };
|
|
||||||
},
|
|
||||||
{ persistence: 'session' }
|
|
||||||
);
|
|
||||||
},
|
|
||||||
runTestProvider(id, options) {
|
|
||||||
const index = store.getState().index;
|
|
||||||
invariant(index, 'The index is currently unavailable');
|
|
||||||
|
|
||||||
api.updateTestProviderState(id, {
|
|
||||||
running: true,
|
|
||||||
failed: false,
|
|
||||||
crashed: false,
|
|
||||||
progress: undefined,
|
|
||||||
});
|
|
||||||
|
|
||||||
const indexUrl = new URL('index.json', window.location.href).toString();
|
|
||||||
|
|
||||||
if (!options?.entryId) {
|
|
||||||
const payload: TestingModuleRunRequestPayload = {
|
|
||||||
providerId: id,
|
|
||||||
indexUrl,
|
|
||||||
};
|
|
||||||
|
|
||||||
fullAPI.emit(TESTING_MODULE_RUN_REQUEST, payload);
|
|
||||||
|
|
||||||
return () => api.cancelTestProvider(id);
|
|
||||||
}
|
|
||||||
|
|
||||||
const entry = index[options.entryId];
|
|
||||||
invariant(entry, `No entry found in the index for id '${options.entryId}'`);
|
|
||||||
|
|
||||||
const findStories = (entryId: StoryId, results: StoryId[] = []): StoryId[] => {
|
|
||||||
const node = index[entryId];
|
|
||||||
if (node.type === 'story') {
|
|
||||||
results.push(node.id);
|
|
||||||
} else if ('children' in node) {
|
|
||||||
node.children.forEach((childId) => findStories(childId, results));
|
|
||||||
}
|
|
||||||
return results;
|
|
||||||
};
|
|
||||||
|
|
||||||
const payload: TestingModuleRunRequestPayload = {
|
|
||||||
providerId: id,
|
|
||||||
indexUrl,
|
|
||||||
storyIds: findStories(options.entryId),
|
|
||||||
};
|
|
||||||
fullAPI.emit(TESTING_MODULE_RUN_REQUEST, payload);
|
|
||||||
return () => api.cancelTestProvider(id);
|
|
||||||
},
|
|
||||||
cancelTestProvider(id) {
|
|
||||||
api.updateTestProviderState(id, { cancelling: true });
|
|
||||||
fullAPI.emit(TESTING_MODULE_CANCEL_TEST_RUN_REQUEST, { providerId: id });
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
const initModule = async () => {
|
|
||||||
const initialState: TestProviders = Object.fromEntries(
|
|
||||||
Object.entries(fullAPI.getElements(Addon_TypesEnum.experimental_TEST_PROVIDER)).map(
|
|
||||||
([id, config]) => [
|
|
||||||
id,
|
|
||||||
{
|
|
||||||
...config,
|
|
||||||
...initialTestProviderState,
|
|
||||||
...(state?.testProviders?.[id] || {}),
|
|
||||||
running: false,
|
|
||||||
} as TestProviders[0],
|
|
||||||
]
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
store.setState({ testProviders: initialState }, { persistence: 'session' });
|
|
||||||
};
|
|
||||||
return { init: initModule, state, api };
|
|
||||||
};
|
|
@ -49,7 +49,6 @@ import { noArrayMerge } from './lib/merge';
|
|||||||
import type { ModuleFn } from './lib/types';
|
import type { ModuleFn } from './lib/types';
|
||||||
import * as addons from './modules/addons';
|
import * as addons from './modules/addons';
|
||||||
import * as channel from './modules/channel';
|
import * as channel from './modules/channel';
|
||||||
import * as testProviders from './modules/experimental_testmodule';
|
|
||||||
import * as globals from './modules/globals';
|
import * as globals from './modules/globals';
|
||||||
import * as layout from './modules/layout';
|
import * as layout from './modules/layout';
|
||||||
import * as notifications from './modules/notifications';
|
import * as notifications from './modules/notifications';
|
||||||
@ -79,7 +78,6 @@ export type State = layout.SubState &
|
|||||||
stories.SubState &
|
stories.SubState &
|
||||||
refs.SubState &
|
refs.SubState &
|
||||||
notifications.SubState &
|
notifications.SubState &
|
||||||
testProviders.SubState &
|
|
||||||
version.SubState &
|
version.SubState &
|
||||||
url.SubState &
|
url.SubState &
|
||||||
shortcuts.SubState &
|
shortcuts.SubState &
|
||||||
@ -98,7 +96,6 @@ export type API = addons.SubAPI &
|
|||||||
globals.SubAPI &
|
globals.SubAPI &
|
||||||
layout.SubAPI &
|
layout.SubAPI &
|
||||||
notifications.SubAPI &
|
notifications.SubAPI &
|
||||||
testProviders.SubAPI &
|
|
||||||
shortcuts.SubAPI &
|
shortcuts.SubAPI &
|
||||||
settings.SubAPI &
|
settings.SubAPI &
|
||||||
version.SubAPI &
|
version.SubAPI &
|
||||||
@ -170,7 +167,6 @@ class ManagerProvider extends Component<ManagerProviderProps, State> {
|
|||||||
addons,
|
addons,
|
||||||
layout,
|
layout,
|
||||||
notifications,
|
notifications,
|
||||||
testProviders,
|
|
||||||
settings,
|
settings,
|
||||||
shortcuts,
|
shortcuts,
|
||||||
stories,
|
stories,
|
||||||
|
@ -2,12 +2,16 @@ import type { ComponentProps, FC, SyntheticEvent } from 'react';
|
|||||||
import React, { useMemo, useState } from 'react';
|
import React, { useMemo, useState } from 'react';
|
||||||
|
|
||||||
import { TooltipLinkList, WithTooltip } from 'storybook/internal/components';
|
import { TooltipLinkList, WithTooltip } from 'storybook/internal/components';
|
||||||
import { type TestProviders } from 'storybook/internal/core-events';
|
import {
|
||||||
import { type API_HashEntry, Addon_TypesEnum } from 'storybook/internal/types';
|
type API_HashEntry,
|
||||||
|
type Addon_Collection,
|
||||||
|
type Addon_TestProviderType,
|
||||||
|
Addon_TypesEnum,
|
||||||
|
} from 'storybook/internal/types';
|
||||||
|
|
||||||
import { EllipsisIcon } from '@storybook/icons';
|
import { EllipsisIcon } from '@storybook/icons';
|
||||||
|
|
||||||
import { useStorybookState } from 'storybook/manager-api';
|
import { useStorybookApi } from 'storybook/manager-api';
|
||||||
import type { API } from 'storybook/manager-api';
|
import type { API } from 'storybook/manager-api';
|
||||||
import { styled } from 'storybook/theming';
|
import { styled } from 'storybook/theming';
|
||||||
|
|
||||||
@ -55,12 +59,10 @@ export const useContextMenu = (context: API_HashEntry, links: Link[], api: API)
|
|||||||
* instead of a simple boolean to ensure that the links are recalculated
|
* instead of a simple boolean to ensure that the links are recalculated
|
||||||
*/
|
*/
|
||||||
const providerLinks = useMemo(() => {
|
const providerLinks = useMemo(() => {
|
||||||
const testProviders = api.getElements(
|
const registeredTestProviders = api.getElements(Addon_TypesEnum.experimental_TEST_PROVIDER);
|
||||||
Addon_TypesEnum.experimental_TEST_PROVIDER
|
|
||||||
) as any as TestProviders;
|
|
||||||
|
|
||||||
if (hoverCount) {
|
if (hoverCount) {
|
||||||
return generateTestProviderLinks(testProviders, context);
|
return generateTestProviderLinks(registeredTestProviders, context);
|
||||||
}
|
}
|
||||||
return [];
|
return [];
|
||||||
}, [api, context, hoverCount]);
|
}, [api, context, hoverCount]);
|
||||||
@ -109,8 +111,10 @@ const LiveContextMenu: FC<{ context: API_HashEntry } & ComponentProps<typeof Too
|
|||||||
links,
|
links,
|
||||||
...rest
|
...rest
|
||||||
}) => {
|
}) => {
|
||||||
const { testProviders } = useStorybookState();
|
const registeredTestProviders = useStorybookApi().getElements(
|
||||||
const providerLinks: Link[] = generateTestProviderLinks(testProviders, context);
|
Addon_TypesEnum.experimental_TEST_PROVIDER
|
||||||
|
);
|
||||||
|
const providerLinks: Link[] = generateTestProviderLinks(registeredTestProviders, context);
|
||||||
const groups = Array.isArray(links[0]) ? (links as Link[][]) : [links as Link[]];
|
const groups = Array.isArray(links[0]) ? (links as Link[][]) : [links as Link[]];
|
||||||
const all = groups.concat([providerLinks]);
|
const all = groups.concat([providerLinks]);
|
||||||
|
|
||||||
@ -118,15 +122,15 @@ const LiveContextMenu: FC<{ context: API_HashEntry } & ComponentProps<typeof Too
|
|||||||
};
|
};
|
||||||
|
|
||||||
export function generateTestProviderLinks(
|
export function generateTestProviderLinks(
|
||||||
testProviders: TestProviders,
|
registeredTestProviders: Addon_Collection<Addon_TestProviderType>,
|
||||||
context: API_HashEntry
|
context: API_HashEntry
|
||||||
): Link[] {
|
): Link[] {
|
||||||
return Object.entries(testProviders)
|
return Object.entries(registeredTestProviders)
|
||||||
.map(([testProviderId, state]) => {
|
.map(([testProviderId, state]) => {
|
||||||
if (!state) {
|
if (!state) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
const content = state.sidebarContextMenu?.({ context, state });
|
const content = state.sidebarContextMenu?.({ context });
|
||||||
|
|
||||||
if (!content) {
|
if (!content) {
|
||||||
return null;
|
return null;
|
||||||
|
@ -1,113 +0,0 @@
|
|||||||
import React from 'react';
|
|
||||||
|
|
||||||
import { Button, ProgressSpinner, TooltipNote, WithTooltip } from 'storybook/internal/components';
|
|
||||||
import type { TestProviders } from 'storybook/internal/core-events';
|
|
||||||
|
|
||||||
import { PlayHollowIcon, StopAltIcon } from '@storybook/icons';
|
|
||||||
|
|
||||||
import { useStorybookApi } from 'storybook/manager-api';
|
|
||||||
import { styled } from 'storybook/theming';
|
|
||||||
|
|
||||||
const Container = styled.div({
|
|
||||||
display: 'flex',
|
|
||||||
justifyContent: 'space-between',
|
|
||||||
padding: '8px 0',
|
|
||||||
});
|
|
||||||
|
|
||||||
const Info = styled.div({
|
|
||||||
display: 'flex',
|
|
||||||
flexDirection: 'column',
|
|
||||||
marginLeft: 8,
|
|
||||||
});
|
|
||||||
|
|
||||||
const Actions = styled.div({
|
|
||||||
display: 'flex',
|
|
||||||
gap: 4,
|
|
||||||
});
|
|
||||||
|
|
||||||
const TitleWrapper = styled.div<{ crashed?: boolean }>(({ crashed, theme }) => ({
|
|
||||||
fontSize: theme.typography.size.s1,
|
|
||||||
fontWeight: crashed ? 'bold' : 'normal',
|
|
||||||
color: crashed ? theme.color.negativeText : theme.color.defaultText,
|
|
||||||
}));
|
|
||||||
|
|
||||||
const DescriptionWrapper = styled.div(({ theme }) => ({
|
|
||||||
fontSize: theme.typography.size.s1,
|
|
||||||
color: theme.textMutedColor,
|
|
||||||
}));
|
|
||||||
|
|
||||||
const Progress = styled(ProgressSpinner)({
|
|
||||||
margin: 4,
|
|
||||||
});
|
|
||||||
|
|
||||||
const StopIcon = styled(StopAltIcon)({
|
|
||||||
width: 10,
|
|
||||||
});
|
|
||||||
|
|
||||||
export const LegacyRender = ({ ...state }: TestProviders[keyof TestProviders]) => {
|
|
||||||
const Description = state.description!;
|
|
||||||
const Title = state.title!;
|
|
||||||
const api = useStorybookApi();
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Container>
|
|
||||||
<Info>
|
|
||||||
<TitleWrapper crashed={state.crashed} id="testing-module-title">
|
|
||||||
<Title {...state} />
|
|
||||||
</TitleWrapper>
|
|
||||||
<DescriptionWrapper id="testing-module-description">
|
|
||||||
<Description {...state} />
|
|
||||||
</DescriptionWrapper>
|
|
||||||
</Info>
|
|
||||||
|
|
||||||
<Actions>
|
|
||||||
{state.runnable && (
|
|
||||||
<>
|
|
||||||
{state.running && state.cancellable ? (
|
|
||||||
<WithTooltip
|
|
||||||
hasChrome={false}
|
|
||||||
trigger="hover"
|
|
||||||
tooltip={<TooltipNote note={`Stop ${state.name}`} />}
|
|
||||||
>
|
|
||||||
<Button
|
|
||||||
aria-label={`Stop ${state.name}`}
|
|
||||||
size="medium"
|
|
||||||
variant="ghost"
|
|
||||||
padding="none"
|
|
||||||
onClick={() => api.cancelTestProvider(state.id)}
|
|
||||||
disabled={state.cancelling}
|
|
||||||
>
|
|
||||||
<Progress
|
|
||||||
percentage={
|
|
||||||
state.progress?.percentageCompleted ??
|
|
||||||
(state.details as any)?.buildProgressPercentage
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<StopIcon />
|
|
||||||
</Progress>
|
|
||||||
</Button>
|
|
||||||
</WithTooltip>
|
|
||||||
) : (
|
|
||||||
<WithTooltip
|
|
||||||
hasChrome={false}
|
|
||||||
trigger="hover"
|
|
||||||
tooltip={<TooltipNote note={`Start ${state.name}`} />}
|
|
||||||
>
|
|
||||||
<Button
|
|
||||||
aria-label={`Start ${state.name}`}
|
|
||||||
size="medium"
|
|
||||||
variant="ghost"
|
|
||||||
padding="small"
|
|
||||||
onClick={() => api.runTestProvider(state.id)}
|
|
||||||
disabled={state.crashed || state.running}
|
|
||||||
>
|
|
||||||
<PlayHollowIcon />
|
|
||||||
</Button>
|
|
||||||
</WithTooltip>
|
|
||||||
)}
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</Actions>
|
|
||||||
</Container>
|
|
||||||
);
|
|
||||||
};
|
|
@ -10,7 +10,7 @@ import { mockDataset } from './mockdata';
|
|||||||
import type { RefType } from './types';
|
import type { RefType } from './types';
|
||||||
|
|
||||||
const managerContext = {
|
const managerContext = {
|
||||||
state: { docsOptions: {}, testProviders: {} },
|
state: { docsOptions: {} },
|
||||||
api: {
|
api: {
|
||||||
on: fn().mockName('api::on'),
|
on: fn().mockName('api::on'),
|
||||||
off: fn().mockName('api::off'),
|
off: fn().mockName('api::off'),
|
||||||
|
@ -35,7 +35,6 @@ const managerContext: any = {
|
|||||||
autodocs: 'tag',
|
autodocs: 'tag',
|
||||||
docsMode: false,
|
docsMode: false,
|
||||||
},
|
},
|
||||||
testProviders: {},
|
|
||||||
},
|
},
|
||||||
api: {
|
api: {
|
||||||
emit: fn().mockName('api::emit'),
|
emit: fn().mockName('api::emit'),
|
||||||
|
@ -1,13 +1,16 @@
|
|||||||
import React, { type FC, useEffect, useState } from 'react';
|
import React, { type FC, useEffect, useState } from 'react';
|
||||||
|
|
||||||
import { Addon_TypesEnum } from 'storybook/internal/types';
|
import {
|
||||||
|
type Addon_Collection,
|
||||||
|
type Addon_TestProviderType,
|
||||||
|
Addon_TypesEnum,
|
||||||
|
} from 'storybook/internal/types';
|
||||||
|
|
||||||
import type { Meta, StoryObj } from '@storybook/react-vite';
|
import type { Meta, StoryObj } from '@storybook/react-vite';
|
||||||
|
|
||||||
import { type API, ManagerContext } from 'storybook/manager-api';
|
import { type API, ManagerContext } from 'storybook/manager-api';
|
||||||
import { expect, fireEvent, fn, waitFor, within } from 'storybook/test';
|
import { expect, fireEvent, fn, waitFor, within } from 'storybook/test';
|
||||||
|
|
||||||
import type { TestProviders } from '../../../core-events';
|
|
||||||
import type { TestProviderStateByProviderId } from '../../../shared/test-provider-store';
|
import type { TestProviderStateByProviderId } from '../../../shared/test-provider-store';
|
||||||
import { SidebarBottomBase } from './SidebarBottom';
|
import { SidebarBottomBase } from './SidebarBottom';
|
||||||
|
|
||||||
@ -52,32 +55,16 @@ const managerContext: any = {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const registeredTestProviders: TestProviders = {
|
const registeredTestProviders: Addon_Collection<Addon_TestProviderType> = {
|
||||||
'component-tests': {
|
'component-tests': {
|
||||||
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
||||||
id: 'component-tests',
|
id: 'component-tests',
|
||||||
name: 'Component tests',
|
|
||||||
render: () => <div>Component tests</div>,
|
render: () => <div>Component tests</div>,
|
||||||
runnable: true,
|
|
||||||
details: {},
|
|
||||||
cancellable: true,
|
|
||||||
cancelling: false,
|
|
||||||
running: false,
|
|
||||||
failed: false,
|
|
||||||
crashed: false,
|
|
||||||
},
|
},
|
||||||
'visual-tests': {
|
'visual-tests': {
|
||||||
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
||||||
id: 'visual-tests',
|
id: 'visual-tests',
|
||||||
name: 'Visual tests',
|
|
||||||
render: () => <div>Visual tests</div>,
|
render: () => <div>Visual tests</div>,
|
||||||
runnable: true,
|
|
||||||
details: {},
|
|
||||||
cancellable: true,
|
|
||||||
cancelling: false,
|
|
||||||
running: false,
|
|
||||||
failed: false,
|
|
||||||
crashed: false,
|
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
const testProviderStates: TestProviderStateByProviderId = {
|
const testProviderStates: TestProviderStateByProviderId = {
|
||||||
@ -149,15 +136,7 @@ export const DynamicHeight: Story = {
|
|||||||
'dynamic-height': {
|
'dynamic-height': {
|
||||||
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
||||||
id: 'dynamic-height',
|
id: 'dynamic-height',
|
||||||
name: 'Dynamic height',
|
|
||||||
render: () => <DynamicHeightDemo />,
|
render: () => <DynamicHeightDemo />,
|
||||||
runnable: true,
|
|
||||||
details: {},
|
|
||||||
cancellable: true,
|
|
||||||
cancelling: false,
|
|
||||||
running: false,
|
|
||||||
failed: false,
|
|
||||||
crashed: false,
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -1,13 +1,11 @@
|
|||||||
import React, { Fragment, useEffect, useLayoutEffect, useRef, useState } from 'react';
|
import React, { Fragment, useEffect, useLayoutEffect, useRef, useState } from 'react';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
TESTING_MODULE_CRASH_REPORT,
|
type API_FilterFunction,
|
||||||
TESTING_MODULE_PROGRESS_REPORT,
|
type Addon_Collection,
|
||||||
type TestProviders,
|
type Addon_TestProviderType,
|
||||||
type TestingModuleCrashReportPayload,
|
Addon_TypesEnum,
|
||||||
type TestingModuleProgressReportPayload,
|
} from 'storybook/internal/types';
|
||||||
} from 'storybook/internal/core-events';
|
|
||||||
import { type API_FilterFunction } from 'storybook/internal/types';
|
|
||||||
|
|
||||||
import {
|
import {
|
||||||
experimental_useStatusStore,
|
experimental_useStatusStore,
|
||||||
@ -89,7 +87,7 @@ interface SidebarBottomProps {
|
|||||||
hasStatuses: boolean;
|
hasStatuses: boolean;
|
||||||
isDevelopment?: boolean;
|
isDevelopment?: boolean;
|
||||||
testProviderStates: TestProviderStateByProviderId;
|
testProviderStates: TestProviderStateByProviderId;
|
||||||
registeredTestProviders: TestProviders;
|
registeredTestProviders: Addon_Collection<Addon_TestProviderType>;
|
||||||
onRunAll: () => void;
|
onRunAll: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -126,38 +124,6 @@ export const SidebarBottomBase = ({
|
|||||||
api.experimental_setFilter('sidebar-bottom-filter', filter);
|
api.experimental_setFilter('sidebar-bottom-filter', filter);
|
||||||
}, [api, warningCount, errorCount, warningsActive, errorsActive]);
|
}, [api, warningCount, errorCount, warningsActive, errorsActive]);
|
||||||
|
|
||||||
// Register listeners before the first render
|
|
||||||
useLayoutEffect(() => {
|
|
||||||
const onCrashReport = ({ providerId, ...details }: TestingModuleCrashReportPayload) => {
|
|
||||||
api.updateTestProviderState(providerId, {
|
|
||||||
error: { name: 'Crashed!', message: details.error.message },
|
|
||||||
running: false,
|
|
||||||
crashed: true,
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const onProgressReport = async ({
|
|
||||||
providerId,
|
|
||||||
...result
|
|
||||||
}: TestingModuleProgressReportPayload) => {
|
|
||||||
const statusResult = 'status' in result ? result.status : undefined;
|
|
||||||
api.updateTestProviderState(
|
|
||||||
providerId,
|
|
||||||
statusResult === 'failed'
|
|
||||||
? { ...result, running: false, failed: true }
|
|
||||||
: { ...result, running: statusResult === 'pending' }
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
api.on(TESTING_MODULE_CRASH_REPORT, onCrashReport);
|
|
||||||
api.on(TESTING_MODULE_PROGRESS_REPORT, onProgressReport);
|
|
||||||
|
|
||||||
return () => {
|
|
||||||
api.off(TESTING_MODULE_CRASH_REPORT, onCrashReport);
|
|
||||||
api.off(TESTING_MODULE_PROGRESS_REPORT, onProgressReport);
|
|
||||||
};
|
|
||||||
}, [api, registeredTestProviders]);
|
|
||||||
|
|
||||||
if (
|
if (
|
||||||
!warningCount &&
|
!warningCount &&
|
||||||
!errorCount &&
|
!errorCount &&
|
||||||
@ -199,7 +165,8 @@ export const SidebarBottomBase = ({
|
|||||||
|
|
||||||
export const SidebarBottom = ({ isDevelopment }: { isDevelopment?: boolean }) => {
|
export const SidebarBottom = ({ isDevelopment }: { isDevelopment?: boolean }) => {
|
||||||
const api = useStorybookApi();
|
const api = useStorybookApi();
|
||||||
const { notifications, testProviders: registeredTestProviders } = useStorybookState();
|
const registeredTestProviders = api.getElements(Addon_TypesEnum.experimental_TEST_PROVIDER);
|
||||||
|
const { notifications } = useStorybookState();
|
||||||
const { hasStatuses, errorCount, warningCount } = experimental_useStatusStore((statuses) => {
|
const { hasStatuses, errorCount, warningCount } = experimental_useStatusStore((statuses) => {
|
||||||
return Object.values(statuses).reduce(
|
return Object.values(statuses).reduce(
|
||||||
(result, storyStatuses) => {
|
(result, storyStatuses) => {
|
||||||
|
@ -1,9 +1,12 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
|
||||||
import type { Listener } from 'storybook/internal/channels';
|
import type { Listener } from 'storybook/internal/channels';
|
||||||
import { type TestProviders } from 'storybook/internal/core-events';
|
|
||||||
import type { TestProviderStateByProviderId } from 'storybook/internal/types';
|
import type { TestProviderStateByProviderId } from 'storybook/internal/types';
|
||||||
import { Addon_TypesEnum } from 'storybook/internal/types';
|
import {
|
||||||
|
type Addon_Collection,
|
||||||
|
type Addon_TestProviderType,
|
||||||
|
Addon_TypesEnum,
|
||||||
|
} from 'storybook/internal/types';
|
||||||
|
|
||||||
import type { Meta, StoryObj } from '@storybook/react-vite';
|
import type { Meta, StoryObj } from '@storybook/react-vite';
|
||||||
|
|
||||||
@ -19,38 +22,21 @@ const TestProvider = styled.div({
|
|||||||
fontSize: 12,
|
fontSize: 12,
|
||||||
});
|
});
|
||||||
|
|
||||||
const baseState = {
|
const registeredTestProviders: Addon_Collection<Addon_TestProviderType> = {
|
||||||
details: {},
|
|
||||||
cancellable: false,
|
|
||||||
cancelling: false,
|
|
||||||
running: false,
|
|
||||||
failed: false,
|
|
||||||
crashed: false,
|
|
||||||
};
|
|
||||||
|
|
||||||
const registeredTestProviders: TestProviders = {
|
|
||||||
'component-tests': {
|
'component-tests': {
|
||||||
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
||||||
id: 'component-tests',
|
id: 'component-tests',
|
||||||
name: 'Component tests',
|
|
||||||
render: () => <TestProvider>Component tests</TestProvider>,
|
render: () => <TestProvider>Component tests</TestProvider>,
|
||||||
runnable: true,
|
|
||||||
...baseState,
|
|
||||||
},
|
},
|
||||||
'visual-tests': {
|
'visual-tests': {
|
||||||
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
||||||
id: 'visual-tests',
|
id: 'visual-tests',
|
||||||
name: 'Visual tests',
|
|
||||||
render: () => <TestProvider>Visual tests</TestProvider>,
|
render: () => <TestProvider>Visual tests</TestProvider>,
|
||||||
runnable: true,
|
|
||||||
...baseState,
|
|
||||||
},
|
},
|
||||||
linting: {
|
linting: {
|
||||||
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
||||||
id: 'linting',
|
id: 'linting',
|
||||||
name: 'Linting',
|
|
||||||
render: () => <TestProvider>Linting</TestProvider>,
|
render: () => <TestProvider>Linting</TestProvider>,
|
||||||
...baseState,
|
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1,17 +1,19 @@
|
|||||||
import React, { type SyntheticEvent, useCallback, useEffect, useRef, useState } from 'react';
|
import React, { type SyntheticEvent, useCallback, useEffect, useRef, useState } from 'react';
|
||||||
|
|
||||||
|
import { once } from 'storybook/internal/client-logger';
|
||||||
import { Button, IconButton, TooltipNote } from 'storybook/internal/components';
|
import { Button, IconButton, TooltipNote } from 'storybook/internal/components';
|
||||||
import { WithTooltip } from 'storybook/internal/components';
|
import { WithTooltip } from 'storybook/internal/components';
|
||||||
import { type TestProviders } from 'storybook/internal/core-events';
|
import type {
|
||||||
|
Addon_Collection,
|
||||||
|
Addon_TestProviderType,
|
||||||
|
TestProviderStateByProviderId,
|
||||||
|
} from 'storybook/internal/types';
|
||||||
|
|
||||||
import { ChevronSmallUpIcon, PlayAllHollowIcon, SweepIcon } from '@storybook/icons';
|
import { ChevronSmallUpIcon, PlayAllHollowIcon, SweepIcon } from '@storybook/icons';
|
||||||
|
|
||||||
import { internal_fullTestProviderStore } from '#manager-stores';
|
import { internal_fullTestProviderStore } from '#manager-stores';
|
||||||
import { keyframes, styled } from 'storybook/theming';
|
import { keyframes, styled } from 'storybook/theming';
|
||||||
|
|
||||||
import type { TestProviderStateByProviderId } from '../../../shared/test-provider-store';
|
|
||||||
import { LegacyRender } from './LegacyRender';
|
|
||||||
|
|
||||||
const DEFAULT_HEIGHT = 500;
|
const DEFAULT_HEIGHT = 500;
|
||||||
|
|
||||||
const spin = keyframes({
|
const spin = keyframes({
|
||||||
@ -162,7 +164,7 @@ const TestProvider = styled.div(({ theme }) => ({
|
|||||||
}));
|
}));
|
||||||
|
|
||||||
interface TestingModuleProps {
|
interface TestingModuleProps {
|
||||||
registeredTestProviders: TestProviders;
|
registeredTestProviders: Addon_Collection<Addon_TestProviderType>;
|
||||||
testProviderStates: TestProviderStateByProviderId;
|
testProviderStates: TestProviderStateByProviderId;
|
||||||
hasStatuses: boolean;
|
hasStatuses: boolean;
|
||||||
clearStatuses: () => void;
|
clearStatuses: () => void;
|
||||||
@ -271,11 +273,17 @@ export const TestingModule = ({
|
|||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Content ref={contentRef}>
|
<Content ref={contentRef}>
|
||||||
{Object.values(registeredTestProviders).map((state) => {
|
{Object.values(registeredTestProviders).map((registeredTestProvider) => {
|
||||||
const { render: Render } = state;
|
const { render: Render, id } = registeredTestProvider;
|
||||||
|
if (!Render) {
|
||||||
|
once.warn(
|
||||||
|
`No render function found for test provider with id '${id}', skipping...`
|
||||||
|
);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
return (
|
return (
|
||||||
<TestProvider key={state.id} data-module-id={state.id}>
|
<TestProvider key={id} data-module-id={id}>
|
||||||
{Render ? <Render {...state} /> : <LegacyRender {...state} />}
|
<Render />
|
||||||
</TestProvider>
|
</TestProvider>
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
|
@ -1,5 +1,11 @@
|
|||||||
import React, { useState } from 'react';
|
import React, { useState } from 'react';
|
||||||
|
|
||||||
|
import {
|
||||||
|
type Addon_Collection,
|
||||||
|
type Addon_TestProviderType,
|
||||||
|
Addon_TypesEnum,
|
||||||
|
} from 'storybook/internal/types';
|
||||||
|
|
||||||
import type { Meta, StoryObj } from '@storybook/react-vite';
|
import type { Meta, StoryObj } from '@storybook/react-vite';
|
||||||
|
|
||||||
import { action } from 'storybook/actions';
|
import { action } from 'storybook/actions';
|
||||||
@ -17,43 +23,28 @@ const managerContext: any = {
|
|||||||
autodocs: 'tag',
|
autodocs: 'tag',
|
||||||
docsMode: false,
|
docsMode: false,
|
||||||
},
|
},
|
||||||
testProviders: {
|
|
||||||
'component-tests': {
|
|
||||||
type: 'experimental_TEST_PROVIDER',
|
|
||||||
id: 'component-tests',
|
|
||||||
render: () => 'Component tests',
|
|
||||||
sidebarContextMenu: () => <div>TEST_PROVIDER_CONTEXT_CONTENT</div>,
|
|
||||||
runnable: true,
|
|
||||||
},
|
|
||||||
'visual-tests': {
|
|
||||||
type: 'experimental_TEST_PROVIDER',
|
|
||||||
id: 'visual-tests',
|
|
||||||
render: () => 'Visual tests',
|
|
||||||
sidebarContextMenu: () => null,
|
|
||||||
runnable: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
api: {
|
api: {
|
||||||
on: fn().mockName('api::on'),
|
on: fn().mockName('api::on'),
|
||||||
off: fn().mockName('api::off'),
|
off: fn().mockName('api::off'),
|
||||||
emit: fn().mockName('api::emit'),
|
emit: fn().mockName('api::emit'),
|
||||||
getElements: fn(() => ({
|
getElements: fn(
|
||||||
'component-tests': {
|
() =>
|
||||||
type: 'experimental_TEST_PROVIDER',
|
({
|
||||||
id: 'component-tests',
|
'component-tests': {
|
||||||
render: () => 'Component tests',
|
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
||||||
sidebarContextMenu: () => <div>TEST_PROVIDER_CONTEXT_CONTENT</div>,
|
id: 'component-tests',
|
||||||
runnable: true,
|
render: () => 'Component tests',
|
||||||
},
|
sidebarContextMenu: () => <div>TEST_PROVIDER_CONTEXT_CONTENT</div>,
|
||||||
'visual-tests': {
|
},
|
||||||
type: 'experimental_TEST_PROVIDER',
|
'visual-tests': {
|
||||||
id: 'visual-tests',
|
type: Addon_TypesEnum.experimental_TEST_PROVIDER,
|
||||||
render: () => 'Visual tests',
|
id: 'visual-tests',
|
||||||
sidebarContextMenu: () => null,
|
render: () => 'Visual tests',
|
||||||
runnable: true,
|
sidebarContextMenu: () => null,
|
||||||
},
|
},
|
||||||
})),
|
}) satisfies Addon_Collection<Addon_TestProviderType>
|
||||||
|
),
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -588,11 +588,6 @@ export default {
|
|||||||
'STORY_THREW_EXCEPTION',
|
'STORY_THREW_EXCEPTION',
|
||||||
'STORY_UNCHANGED',
|
'STORY_UNCHANGED',
|
||||||
'TELEMETRY_ERROR',
|
'TELEMETRY_ERROR',
|
||||||
'TESTING_MODULE_CANCEL_TEST_RUN_REQUEST',
|
|
||||||
'TESTING_MODULE_CANCEL_TEST_RUN_RESPONSE',
|
|
||||||
'TESTING_MODULE_CRASH_REPORT',
|
|
||||||
'TESTING_MODULE_PROGRESS_REPORT',
|
|
||||||
'TESTING_MODULE_RUN_REQUEST',
|
|
||||||
'TOGGLE_WHATS_NEW_NOTIFICATIONS',
|
'TOGGLE_WHATS_NEW_NOTIFICATIONS',
|
||||||
'UNHANDLED_ERRORS_WHILE_PLAYING',
|
'UNHANDLED_ERRORS_WHILE_PLAYING',
|
||||||
'UPDATE_GLOBALS',
|
'UPDATE_GLOBALS',
|
||||||
@ -651,11 +646,6 @@ export default {
|
|||||||
'STORY_THREW_EXCEPTION',
|
'STORY_THREW_EXCEPTION',
|
||||||
'STORY_UNCHANGED',
|
'STORY_UNCHANGED',
|
||||||
'TELEMETRY_ERROR',
|
'TELEMETRY_ERROR',
|
||||||
'TESTING_MODULE_CANCEL_TEST_RUN_REQUEST',
|
|
||||||
'TESTING_MODULE_CANCEL_TEST_RUN_RESPONSE',
|
|
||||||
'TESTING_MODULE_CRASH_REPORT',
|
|
||||||
'TESTING_MODULE_PROGRESS_REPORT',
|
|
||||||
'TESTING_MODULE_RUN_REQUEST',
|
|
||||||
'TOGGLE_WHATS_NEW_NOTIFICATIONS',
|
'TOGGLE_WHATS_NEW_NOTIFICATIONS',
|
||||||
'UNHANDLED_ERRORS_WHILE_PLAYING',
|
'UNHANDLED_ERRORS_WHILE_PLAYING',
|
||||||
'UPDATE_GLOBALS',
|
'UPDATE_GLOBALS',
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
/* eslint-disable @typescript-eslint/naming-convention */
|
/* eslint-disable @typescript-eslint/naming-convention */
|
||||||
import type { FC, PropsWithChildren, ReactElement, ReactNode } from 'react';
|
import type { FC, PropsWithChildren, ReactElement, ReactNode } from 'react';
|
||||||
|
|
||||||
import type { TestProviderConfig, TestingModuleProgressReportProgress } from '../../core-events';
|
|
||||||
import type { RenderData as RouterData } from '../../router/types';
|
import type { RenderData as RouterData } from '../../router/types';
|
||||||
import type { ThemeVars } from '../../theming/types';
|
import type { ThemeVars } from '../../theming/types';
|
||||||
import type { API_SidebarOptions } from './api';
|
import type { API_SidebarOptions } from './api';
|
||||||
@ -324,7 +323,7 @@ export type Addon_Type =
|
|||||||
| Addon_BaseType
|
| Addon_BaseType
|
||||||
| Addon_PageType
|
| Addon_PageType
|
||||||
| Addon_WrapperType
|
| Addon_WrapperType
|
||||||
| Addon_TestProviderType<Addon_TestProviderState>;
|
| Addon_TestProviderType;
|
||||||
export interface Addon_BaseType {
|
export interface Addon_BaseType {
|
||||||
/**
|
/**
|
||||||
* The title of the addon. This can be a simple string, but it can also be a
|
* The title of the addon. This can be a simple string, but it can also be a
|
||||||
@ -440,44 +439,14 @@ export interface Addon_WrapperType {
|
|||||||
>;
|
>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Addon_TestProviderType<
|
export interface Addon_TestProviderType {
|
||||||
Details extends { [key: string]: any } = NonNullable<unknown>,
|
|
||||||
> {
|
|
||||||
type: Addon_TypesEnum.experimental_TEST_PROVIDER;
|
type: Addon_TypesEnum.experimental_TEST_PROVIDER;
|
||||||
/** The unique id of the test provider. */
|
/** The unique id of the test provider. */
|
||||||
id: string;
|
id: string;
|
||||||
name: string;
|
render: () => ReactNode;
|
||||||
/** @deprecated Use render instead */
|
sidebarContextMenu?: (options: { context: API_HashEntry }) => ReactNode;
|
||||||
title?: (state: TestProviderConfig & Addon_TestProviderState<Details>) => ReactNode;
|
|
||||||
/** @deprecated Use render instead */
|
|
||||||
description?: (state: TestProviderConfig & Addon_TestProviderState<Details>) => ReactNode;
|
|
||||||
render?: (state: TestProviderConfig & Addon_TestProviderState<Details>) => ReactNode;
|
|
||||||
sidebarContextMenu?: (options: {
|
|
||||||
context: API_HashEntry;
|
|
||||||
state: TestProviderConfig & Addon_TestProviderState<Details>;
|
|
||||||
}) => ReactNode;
|
|
||||||
stateUpdater?: (
|
|
||||||
state: TestProviderConfig & Addon_TestProviderState<Details>,
|
|
||||||
update: Partial<Addon_TestProviderState<Details>>
|
|
||||||
) => void | Partial<TestProviderConfig & Addon_TestProviderState<Details>>;
|
|
||||||
runnable?: boolean;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export type Addon_TestProviderState<Details extends { [key: string]: any } = NonNullable<unknown>> =
|
|
||||||
Pick<Addon_TestProviderType, 'runnable'> & {
|
|
||||||
progress?: TestingModuleProgressReportProgress;
|
|
||||||
details: Details;
|
|
||||||
cancellable: boolean;
|
|
||||||
cancelling: boolean;
|
|
||||||
running: boolean;
|
|
||||||
failed: boolean;
|
|
||||||
crashed: boolean;
|
|
||||||
error?: {
|
|
||||||
name: string;
|
|
||||||
message?: string;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
type Addon_TypeBaseNames = Exclude<
|
type Addon_TypeBaseNames = Exclude<
|
||||||
Addon_TypesEnum,
|
Addon_TypesEnum,
|
||||||
| Addon_TypesEnum.PREVIEW
|
| Addon_TypesEnum.PREVIEW
|
||||||
@ -488,7 +457,7 @@ type Addon_TypeBaseNames = Exclude<
|
|||||||
export interface Addon_TypesMapping extends Record<Addon_TypeBaseNames, Addon_BaseType> {
|
export interface Addon_TypesMapping extends Record<Addon_TypeBaseNames, Addon_BaseType> {
|
||||||
[Addon_TypesEnum.PREVIEW]: Addon_WrapperType;
|
[Addon_TypesEnum.PREVIEW]: Addon_WrapperType;
|
||||||
[Addon_TypesEnum.experimental_PAGE]: Addon_PageType;
|
[Addon_TypesEnum.experimental_PAGE]: Addon_PageType;
|
||||||
[Addon_TypesEnum.experimental_TEST_PROVIDER]: Addon_TestProviderType<Addon_TestProviderState>;
|
[Addon_TypesEnum.experimental_TEST_PROVIDER]: Addon_TestProviderType;
|
||||||
}
|
}
|
||||||
|
|
||||||
export type Addon_Loader<API> = (api: API) => void;
|
export type Addon_Loader<API> = (api: API) => void;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user