mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-06 15:31:16 +08:00
Add discrepancy handling to A11yPanel
This commit is contained in:
parent
d3923d25d0
commit
2eebee1264
@ -8,6 +8,7 @@ import { CheckIcon, SyncIcon } from '@storybook/icons';
|
|||||||
import { useA11yContext } from './A11yContext';
|
import { useA11yContext } from './A11yContext';
|
||||||
import { Report } from './Report';
|
import { Report } from './Report';
|
||||||
import { Tabs } from './Tabs';
|
import { Tabs } from './Tabs';
|
||||||
|
import { TestDiscrepancyMessage } from './TestDiscrepancyMessage';
|
||||||
|
|
||||||
export enum RuleType {
|
export enum RuleType {
|
||||||
VIOLATION,
|
VIOLATION,
|
||||||
@ -43,7 +44,7 @@ const Centered = styled.span({
|
|||||||
});
|
});
|
||||||
|
|
||||||
export const A11YPanel: React.FC = () => {
|
export const A11YPanel: React.FC = () => {
|
||||||
const { results, status, handleManual, error } = useA11yContext();
|
const { results, status, handleManual, error, discrepancy } = useA11yContext();
|
||||||
|
|
||||||
const manualActionItems = useMemo(
|
const manualActionItems = useMemo(
|
||||||
() => [{ title: 'Run test', onClick: handleManual }],
|
() => [{ title: 'Run test', onClick: handleManual }],
|
||||||
@ -106,31 +107,35 @@ export const A11YPanel: React.FC = () => {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{status === 'initial' && <Centered>Initializing...</Centered>}
|
{discrepancy && <TestDiscrepancyMessage discrepancy={discrepancy} />}
|
||||||
{status === 'manual' && (
|
{status === 'ready' || status === 'ran' ? (
|
||||||
<>
|
|
||||||
<Centered>Manually run the accessibility scan.</Centered>
|
|
||||||
<ActionBar key="actionbar" actionItems={manualActionItems} />
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
{status === 'running' && (
|
|
||||||
<Centered>
|
|
||||||
<RotatingIcon size={12} /> Please wait while the accessibility scan is running ...
|
|
||||||
</Centered>
|
|
||||||
)}
|
|
||||||
{(status === 'ready' || status === 'ran') && (
|
|
||||||
<>
|
<>
|
||||||
<ScrollArea vertical horizontal>
|
<ScrollArea vertical horizontal>
|
||||||
<Tabs key="tabs" tabs={tabs} />
|
<Tabs key="tabs" tabs={tabs} />
|
||||||
</ScrollArea>
|
</ScrollArea>
|
||||||
<ActionBar key="actionbar" actionItems={readyActionItems} />
|
<ActionBar key="actionbar" actionItems={readyActionItems} />
|
||||||
</>
|
</>
|
||||||
)}
|
) : (
|
||||||
{status === 'error' && (
|
<Centered style={{ marginTop: discrepancy ? '1em' : 0 }}>
|
||||||
<Centered>
|
{status === 'initial' && 'Initializing...'}
|
||||||
The accessibility scan encountered an error.
|
{status === 'manual' && (
|
||||||
<br />
|
<>
|
||||||
{typeof error === 'string' ? error : JSON.stringify(error)}
|
<>Manually run the accessibility scan.</>
|
||||||
|
<ActionBar key="actionbar" actionItems={manualActionItems} />
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
{status === 'running' && (
|
||||||
|
<>
|
||||||
|
<RotatingIcon size={12} /> Please wait while the accessibility scan is running ...
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
{status === 'error' && (
|
||||||
|
<>
|
||||||
|
The accessibility scan encountered an error.
|
||||||
|
<br />
|
||||||
|
{typeof error === 'string' ? error : JSON.stringify(error)}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
</Centered>
|
</Centered>
|
||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import type { FC, PropsWithChildren } from 'react';
|
import type { FC, PropsWithChildren } from 'react';
|
||||||
import React, { createContext, useCallback, useContext, useEffect, useState } from 'react';
|
import React, { createContext, useCallback, useContext, useEffect, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
STORY_FINISHED,
|
STORY_FINISHED,
|
||||||
@ -10,6 +10,7 @@ import {
|
|||||||
useAddonState,
|
useAddonState,
|
||||||
useChannel,
|
useChannel,
|
||||||
useParameter,
|
useParameter,
|
||||||
|
useStorybookApi,
|
||||||
useStorybookState,
|
useStorybookState,
|
||||||
} from 'storybook/internal/manager-api';
|
} from 'storybook/internal/manager-api';
|
||||||
import type { Report } from 'storybook/internal/preview-api';
|
import type { Report } from 'storybook/internal/preview-api';
|
||||||
@ -19,9 +20,10 @@ import { HIGHLIGHT } from '@storybook/addon-highlight';
|
|||||||
|
|
||||||
import type { AxeResults, Result } from 'axe-core';
|
import type { AxeResults, Result } from 'axe-core';
|
||||||
|
|
||||||
import { ADDON_ID, EVENTS } from '../constants';
|
import { ADDON_ID, EVENTS, TEST_PROVIDER_ID } from '../constants';
|
||||||
import type { A11yParameters } from '../params';
|
import type { A11yParameters } from '../params';
|
||||||
import type { A11YReport } from '../types';
|
import type { A11YReport } from '../types';
|
||||||
|
import type { TestDiscrepancy } from './TestDiscrepancyMessage';
|
||||||
|
|
||||||
export interface Results {
|
export interface Results {
|
||||||
passes: Result[];
|
passes: Result[];
|
||||||
@ -40,6 +42,7 @@ export interface A11yContextStore {
|
|||||||
setStatus: (status: Status) => void;
|
setStatus: (status: Status) => void;
|
||||||
error: unknown;
|
error: unknown;
|
||||||
handleManual: () => void;
|
handleManual: () => void;
|
||||||
|
discrepancy: TestDiscrepancy;
|
||||||
}
|
}
|
||||||
|
|
||||||
const colorsByType = [
|
const colorsByType = [
|
||||||
@ -63,6 +66,7 @@ export const A11yContext = createContext<A11yContextStore>({
|
|||||||
status: 'initial',
|
status: 'initial',
|
||||||
error: undefined,
|
error: undefined,
|
||||||
handleManual: () => {},
|
handleManual: () => {},
|
||||||
|
discrepancy: null,
|
||||||
});
|
});
|
||||||
|
|
||||||
const defaultResult = {
|
const defaultResult = {
|
||||||
@ -80,12 +84,15 @@ export const A11yContextProvider: FC<PropsWithChildren> = (props) => {
|
|||||||
|
|
||||||
const getInitialStatus = useCallback((manual = false) => (manual ? 'manual' : 'initial'), []);
|
const getInitialStatus = useCallback((manual = false) => (manual ? 'manual' : 'initial'), []);
|
||||||
|
|
||||||
|
const api = useStorybookApi();
|
||||||
const [results, setResults] = useAddonState<Results>(ADDON_ID, defaultResult);
|
const [results, setResults] = useAddonState<Results>(ADDON_ID, defaultResult);
|
||||||
const [tab, setTab] = useState(0);
|
const [tab, setTab] = useState(0);
|
||||||
const [error, setError] = React.useState<unknown>(undefined);
|
const [error, setError] = React.useState<unknown>(undefined);
|
||||||
const [status, setStatus] = useState<Status>(getInitialStatus(parameters.manual!));
|
const [status, setStatus] = useState<Status>(getInitialStatus(parameters.manual!));
|
||||||
const [highlighted, setHighlighted] = useState<string[]>([]);
|
const [highlighted, setHighlighted] = useState<string[]>([]);
|
||||||
|
|
||||||
const { storyId } = useStorybookState();
|
const { storyId } = useStorybookState();
|
||||||
|
const storyStatus = api.getCurrentStoryStatus();
|
||||||
|
|
||||||
const handleToggleHighlight = useCallback((target: string[], highlight: boolean) => {
|
const handleToggleHighlight = useCallback((target: string[], highlight: boolean) => {
|
||||||
setHighlighted((prevHighlighted) =>
|
setHighlighted((prevHighlighted) =>
|
||||||
@ -178,6 +185,28 @@ export const A11yContextProvider: FC<PropsWithChildren> = (props) => {
|
|||||||
emit(HIGHLIGHT, { elements: highlighted, color: colorsByType[tab] });
|
emit(HIGHLIGHT, { elements: highlighted, color: colorsByType[tab] });
|
||||||
}, [emit, highlighted, tab]);
|
}, [emit, highlighted, tab]);
|
||||||
|
|
||||||
|
const discrepancy: TestDiscrepancy = useMemo(() => {
|
||||||
|
const storyStatusA11y = storyStatus[TEST_PROVIDER_ID]?.status;
|
||||||
|
|
||||||
|
if (storyStatusA11y) {
|
||||||
|
if (storyStatusA11y === 'success' && results.violations.length > 0) {
|
||||||
|
return 'cliPassedBrowserFailed';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (storyStatusA11y === 'error' && results.violations.length === 0) {
|
||||||
|
if (status === 'ready' || status === 'ran') {
|
||||||
|
return 'browserPassedCliFailed';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (status === 'manual') {
|
||||||
|
return 'cliFailedButModeManual';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}, [results.violations.length, status, storyStatus]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<A11yContext.Provider
|
<A11yContext.Provider
|
||||||
value={{
|
value={{
|
||||||
@ -191,6 +220,7 @@ export const A11yContextProvider: FC<PropsWithChildren> = (props) => {
|
|||||||
setStatus,
|
setStatus,
|
||||||
error,
|
error,
|
||||||
handleManual,
|
handleManual,
|
||||||
|
discrepancy,
|
||||||
}}
|
}}
|
||||||
{...props}
|
{...props}
|
||||||
/>
|
/>
|
||||||
|
73
code/addons/a11y/src/components/TestDiscrepancyMessage.tsx
Normal file
73
code/addons/a11y/src/components/TestDiscrepancyMessage.tsx
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
import React, { useMemo } from 'react';
|
||||||
|
|
||||||
|
import { Link } from 'storybook/internal/components';
|
||||||
|
import { useStorybookApi } from 'storybook/internal/manager-api';
|
||||||
|
import { styled } from 'storybook/internal/theming';
|
||||||
|
|
||||||
|
import { DOCUMENTATION_DISCREPANCY_LINK } from '../constants';
|
||||||
|
|
||||||
|
const Wrapper = styled.div(({ theme: { color, typography, background } }) => ({
|
||||||
|
textAlign: 'start',
|
||||||
|
padding: '11px 15px',
|
||||||
|
fontSize: `${typography.size.s2}px`,
|
||||||
|
fontWeight: typography.weight.regular,
|
||||||
|
lineHeight: '1rem',
|
||||||
|
background: background.app,
|
||||||
|
borderBottom: `1px solid ${color.border}`,
|
||||||
|
color: color.defaultText,
|
||||||
|
backgroundClip: 'padding-box',
|
||||||
|
position: 'relative',
|
||||||
|
code: {
|
||||||
|
fontSize: `${typography.size.s1 - 1}px`,
|
||||||
|
color: 'inherit',
|
||||||
|
margin: '0 0.2em',
|
||||||
|
padding: '0 0.2em',
|
||||||
|
background: 'rgba(255, 255, 255, 0.8)',
|
||||||
|
borderRadius: '2px',
|
||||||
|
boxShadow: '0 0 0 1px rgba(0, 0, 0, 0.1)',
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
export type TestDiscrepancy =
|
||||||
|
| 'browserPassedCliFailed'
|
||||||
|
| 'cliPassedBrowserFailed'
|
||||||
|
| 'cliFailedButModeManual'
|
||||||
|
| null;
|
||||||
|
|
||||||
|
interface TestDiscrepancyMessageProps {
|
||||||
|
discrepancy: TestDiscrepancy;
|
||||||
|
}
|
||||||
|
export const TestDiscrepancyMessage = ({ discrepancy }: TestDiscrepancyMessageProps) => {
|
||||||
|
const api = useStorybookApi();
|
||||||
|
const docsUrl = api.getDocsUrl({
|
||||||
|
subpath: DOCUMENTATION_DISCREPANCY_LINK,
|
||||||
|
versioned: true,
|
||||||
|
renderer: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
const message = useMemo(() => {
|
||||||
|
switch (discrepancy) {
|
||||||
|
case 'browserPassedCliFailed':
|
||||||
|
return 'Accessibility checks passed in this browser but failed in the CLI.';
|
||||||
|
case 'cliPassedBrowserFailed':
|
||||||
|
return 'Accessibility checks passed in the CLI but failed in this browser.';
|
||||||
|
case 'cliFailedButModeManual':
|
||||||
|
return 'Accessibility checks failed in the CLI. Run the tests manually to see the results.';
|
||||||
|
default:
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}, [discrepancy]);
|
||||||
|
|
||||||
|
if (!message) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Wrapper>
|
||||||
|
{message}{' '}
|
||||||
|
<Link href={docsUrl} target="_blank" withArrow>
|
||||||
|
Learn what could cause this
|
||||||
|
</Link>
|
||||||
|
</Wrapper>
|
||||||
|
);
|
||||||
|
};
|
@ -7,4 +7,9 @@ const RUNNING = `${ADDON_ID}/running`;
|
|||||||
const ERROR = `${ADDON_ID}/error`;
|
const ERROR = `${ADDON_ID}/error`;
|
||||||
const MANUAL = `${ADDON_ID}/manual`;
|
const MANUAL = `${ADDON_ID}/manual`;
|
||||||
|
|
||||||
|
export const DOCUMENTATION_LINK = 'writing-tests/accessibility-testing';
|
||||||
|
export const DOCUMENTATION_DISCREPANCY_LINK = `${DOCUMENTATION_LINK}#what-happens-when-there-are-different-results-in-multiple-environments`;
|
||||||
|
|
||||||
|
export const TEST_PROVIDER_ID = 'storybook/addon-a11y/test-provider';
|
||||||
|
|
||||||
export const EVENTS = { RESULT, REQUEST, RUNNING, ERROR, MANUAL };
|
export const EVENTS = { RESULT, REQUEST, RUNNING, ERROR, MANUAL };
|
||||||
|
@ -1,22 +1,34 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
|
||||||
|
import { ManagerContext } from 'storybook/internal/manager-api';
|
||||||
import { ThemeProvider, convert, themes } from 'storybook/internal/theming';
|
import { ThemeProvider, convert, themes } from 'storybook/internal/theming';
|
||||||
|
|
||||||
import type { Meta, StoryObj } from '@storybook/react';
|
import type { Meta, StoryObj } from '@storybook/react';
|
||||||
import { fn } from '@storybook/test';
|
import { fn } from '@storybook/test';
|
||||||
|
|
||||||
|
import type axe from 'axe-core';
|
||||||
|
|
||||||
import { A11YPanel } from '../../src/components/A11YPanel';
|
import { A11YPanel } from '../../src/components/A11YPanel';
|
||||||
import { A11yContext } from '../../src/components/A11yContext';
|
import { A11yContext } from '../../src/components/A11yContext';
|
||||||
import type { A11yContextStore } from '../../src/components/A11yContext';
|
import type { A11yContextStore } from '../../src/components/A11yContext';
|
||||||
|
|
||||||
|
const managerContext: any = {
|
||||||
|
state: {},
|
||||||
|
api: {
|
||||||
|
getDocsUrl: fn().mockName('api::getDocsUrl'),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
const meta: Meta = {
|
const meta: Meta = {
|
||||||
title: 'A11YPanel',
|
title: 'A11YPanel',
|
||||||
component: A11YPanel,
|
component: A11YPanel,
|
||||||
decorators: [
|
decorators: [
|
||||||
(Story) => (
|
(Story) => (
|
||||||
<ThemeProvider theme={convert(themes.light)}>
|
<ManagerContext.Provider value={managerContext}>
|
||||||
<Story />
|
<ThemeProvider theme={convert(themes.light)}>
|
||||||
</ThemeProvider>
|
<Story />
|
||||||
|
</ThemeProvider>
|
||||||
|
</ManagerContext.Provider>
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
} satisfies Meta<typeof A11YPanel>;
|
} satisfies Meta<typeof A11YPanel>;
|
||||||
@ -25,7 +37,62 @@ export default meta;
|
|||||||
|
|
||||||
type Story = StoryObj<typeof meta>;
|
type Story = StoryObj<typeof meta>;
|
||||||
|
|
||||||
const Template = (args: Pick<A11yContextStore, 'results' | 'error' | 'status'>) => (
|
const violations: axe.Result[] = [
|
||||||
|
{
|
||||||
|
id: 'aria-command-name',
|
||||||
|
impact: 'serious',
|
||||||
|
tags: ['cat.aria', 'wcag2a', 'wcag412', 'TTv5', 'TT6.a', 'EN-301-549', 'EN-9.4.1.2', 'ACT'],
|
||||||
|
description: 'Ensures every ARIA button, link and menuitem has an accessible name',
|
||||||
|
help: 'ARIA commands must have an accessible name',
|
||||||
|
helpUrl: 'https://dequeuniversity.com/rules/axe/4.8/aria-command-name?application=axeAPI',
|
||||||
|
nodes: [
|
||||||
|
{
|
||||||
|
any: [
|
||||||
|
{
|
||||||
|
id: 'has-visible-text',
|
||||||
|
data: null,
|
||||||
|
relatedNodes: [],
|
||||||
|
impact: 'serious',
|
||||||
|
message: 'Element does not have text that is visible to screen readers',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'aria-label',
|
||||||
|
data: null,
|
||||||
|
relatedNodes: [],
|
||||||
|
impact: 'serious',
|
||||||
|
message: 'aria-label attribute does not exist or is empty',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'aria-labelledby',
|
||||||
|
data: null,
|
||||||
|
relatedNodes: [],
|
||||||
|
impact: 'serious',
|
||||||
|
message:
|
||||||
|
'aria-labelledby attribute does not exist, references elements that do not exist or references elements that are empty',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'non-empty-title',
|
||||||
|
data: {
|
||||||
|
messageKey: 'noAttr',
|
||||||
|
},
|
||||||
|
relatedNodes: [],
|
||||||
|
impact: 'serious',
|
||||||
|
message: 'Element has no title attribute',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
all: [],
|
||||||
|
none: [],
|
||||||
|
impact: 'serious',
|
||||||
|
html: '<div role="button" class="css-12jpz5t">',
|
||||||
|
target: ['.css-12jpz5t'],
|
||||||
|
failureSummary:
|
||||||
|
'Fix any of the following:\n Element does not have text that is visible to screen readers\n aria-label attribute does not exist or is empty\n aria-labelledby attribute does not exist, references elements that do not exist or references elements that are empty\n Element has no title attribute',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const Template = (args: Pick<A11yContextStore, 'results' | 'error' | 'status' | 'discrepancy'>) => (
|
||||||
<A11yContext.Provider
|
<A11yContext.Provider
|
||||||
value={{
|
value={{
|
||||||
handleManual: fn(),
|
handleManual: fn(),
|
||||||
@ -49,6 +116,7 @@ export const Initializing: Story = {
|
|||||||
results={{ passes: [], incomplete: [], violations: [] }}
|
results={{ passes: [], incomplete: [], violations: [] }}
|
||||||
status="initial"
|
status="initial"
|
||||||
error={null}
|
error={null}
|
||||||
|
discrepancy={null}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@ -61,6 +129,20 @@ export const Manual: Story = {
|
|||||||
results={{ passes: [], incomplete: [], violations: [] }}
|
results={{ passes: [], incomplete: [], violations: [] }}
|
||||||
status="manual"
|
status="manual"
|
||||||
error={null}
|
error={null}
|
||||||
|
discrepancy={null}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ManualWithDiscrepancy: Story = {
|
||||||
|
render: () => {
|
||||||
|
return (
|
||||||
|
<Template
|
||||||
|
results={{ passes: [], incomplete: [], violations: [] }}
|
||||||
|
status="manual"
|
||||||
|
error={null}
|
||||||
|
discrepancy={'cliFailedButModeManual'}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@ -73,6 +155,7 @@ export const Running: Story = {
|
|||||||
results={{ passes: [], incomplete: [], violations: [] }}
|
results={{ passes: [], incomplete: [], violations: [] }}
|
||||||
status="running"
|
status="running"
|
||||||
error={null}
|
error={null}
|
||||||
|
discrepancy={null}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@ -85,73 +168,28 @@ export const ReadyWithResults: Story = {
|
|||||||
results={{
|
results={{
|
||||||
passes: [],
|
passes: [],
|
||||||
incomplete: [],
|
incomplete: [],
|
||||||
violations: [
|
violations,
|
||||||
{
|
|
||||||
id: 'aria-command-name',
|
|
||||||
impact: 'serious',
|
|
||||||
tags: [
|
|
||||||
'cat.aria',
|
|
||||||
'wcag2a',
|
|
||||||
'wcag412',
|
|
||||||
'TTv5',
|
|
||||||
'TT6.a',
|
|
||||||
'EN-301-549',
|
|
||||||
'EN-9.4.1.2',
|
|
||||||
'ACT',
|
|
||||||
],
|
|
||||||
description: 'Ensures every ARIA button, link and menuitem has an accessible name',
|
|
||||||
help: 'ARIA commands must have an accessible name',
|
|
||||||
helpUrl:
|
|
||||||
'https://dequeuniversity.com/rules/axe/4.8/aria-command-name?application=axeAPI',
|
|
||||||
nodes: [
|
|
||||||
{
|
|
||||||
any: [
|
|
||||||
{
|
|
||||||
id: 'has-visible-text',
|
|
||||||
data: null,
|
|
||||||
relatedNodes: [],
|
|
||||||
impact: 'serious',
|
|
||||||
message: 'Element does not have text that is visible to screen readers',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 'aria-label',
|
|
||||||
data: null,
|
|
||||||
relatedNodes: [],
|
|
||||||
impact: 'serious',
|
|
||||||
message: 'aria-label attribute does not exist or is empty',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 'aria-labelledby',
|
|
||||||
data: null,
|
|
||||||
relatedNodes: [],
|
|
||||||
impact: 'serious',
|
|
||||||
message:
|
|
||||||
'aria-labelledby attribute does not exist, references elements that do not exist or references elements that are empty',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 'non-empty-title',
|
|
||||||
data: {
|
|
||||||
messageKey: 'noAttr',
|
|
||||||
},
|
|
||||||
relatedNodes: [],
|
|
||||||
impact: 'serious',
|
|
||||||
message: 'Element has no title attribute',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
all: [],
|
|
||||||
none: [],
|
|
||||||
impact: 'serious',
|
|
||||||
html: '<div role="button" class="css-12jpz5t">',
|
|
||||||
target: ['.css-12jpz5t'],
|
|
||||||
failureSummary:
|
|
||||||
'Fix any of the following:\n Element does not have text that is visible to screen readers\n aria-label attribute does not exist or is empty\n aria-labelledby attribute does not exist, references elements that do not exist or references elements that are empty\n Element has no title attribute',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
}}
|
}}
|
||||||
status="ready"
|
status="ready"
|
||||||
error={null}
|
error={null}
|
||||||
|
discrepancy={null}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ReadyWithResultsDiscrepancyCLIPassedBrowserFailed: Story = {
|
||||||
|
render: () => {
|
||||||
|
return (
|
||||||
|
<Template
|
||||||
|
results={{
|
||||||
|
passes: [],
|
||||||
|
incomplete: [],
|
||||||
|
violations,
|
||||||
|
}}
|
||||||
|
status="ready"
|
||||||
|
error={null}
|
||||||
|
discrepancy={'cliPassedBrowserFailed'}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@ -164,6 +202,7 @@ export const Error: Story = {
|
|||||||
results={{ passes: [], incomplete: [], violations: [] }}
|
results={{ passes: [], incomplete: [], violations: [] }}
|
||||||
status="error"
|
status="error"
|
||||||
error="Test error message"
|
error="Test error message"
|
||||||
|
discrepancy={null}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@ -176,6 +215,7 @@ export const ErrorStateWithObject: Story = {
|
|||||||
results={{ passes: [], incomplete: [], violations: [] }}
|
results={{ passes: [], incomplete: [], violations: [] }}
|
||||||
status="error"
|
status="error"
|
||||||
error={{ message: 'Test error object message' }}
|
error={{ message: 'Test error object message' }}
|
||||||
|
discrepancy={null}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
|
@ -0,0 +1,51 @@
|
|||||||
|
import React from 'react';
|
||||||
|
|
||||||
|
import { ManagerContext } from 'storybook/internal/manager-api';
|
||||||
|
|
||||||
|
import type { Meta, StoryObj } from '@storybook/react';
|
||||||
|
import { fn } from '@storybook/test';
|
||||||
|
|
||||||
|
import { TestDiscrepancyMessage } from '../../src/components/TestDiscrepancyMessage';
|
||||||
|
|
||||||
|
type Story = StoryObj<typeof TestDiscrepancyMessage>;
|
||||||
|
|
||||||
|
const managerContext: any = {
|
||||||
|
state: {},
|
||||||
|
api: {
|
||||||
|
getDocsUrl: fn().mockName('api::getDocsUrl'),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default {
|
||||||
|
title: 'TestDiscrepancyMessage',
|
||||||
|
component: TestDiscrepancyMessage,
|
||||||
|
parameters: {
|
||||||
|
layout: 'fullscreen',
|
||||||
|
},
|
||||||
|
args: {
|
||||||
|
storyId: 'story-id',
|
||||||
|
},
|
||||||
|
decorators: [
|
||||||
|
(storyFn) => (
|
||||||
|
<ManagerContext.Provider value={managerContext}>{storyFn()}</ManagerContext.Provider>
|
||||||
|
),
|
||||||
|
],
|
||||||
|
} as Meta<typeof TestDiscrepancyMessage>;
|
||||||
|
|
||||||
|
export const BrowserPassedCliFailed: Story = {
|
||||||
|
args: {
|
||||||
|
discrepancy: 'browserPassedCliFailed',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const CliPassedBrowserFailed: Story = {
|
||||||
|
args: {
|
||||||
|
discrepancy: 'cliPassedBrowserFailed',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const CliFailedButModeManual: Story = {
|
||||||
|
args: {
|
||||||
|
discrepancy: 'cliFailedButModeManual',
|
||||||
|
},
|
||||||
|
};
|
@ -7,6 +7,7 @@ import type {
|
|||||||
API_LeafEntry,
|
API_LeafEntry,
|
||||||
API_LoadedRefData,
|
API_LoadedRefData,
|
||||||
API_PreparedStoryIndex,
|
API_PreparedStoryIndex,
|
||||||
|
API_StatusObject,
|
||||||
API_StatusState,
|
API_StatusState,
|
||||||
API_StatusUpdate,
|
API_StatusUpdate,
|
||||||
API_StoryEntry,
|
API_StoryEntry,
|
||||||
@ -268,6 +269,12 @@ export interface SubAPI {
|
|||||||
* @returns {Promise<void>} A promise that resolves when the preview has been set as initialized.
|
* @returns {Promise<void>} A promise that resolves when the preview has been set as initialized.
|
||||||
*/
|
*/
|
||||||
setPreviewInitialized: (ref?: ComposedRef) => Promise<void>;
|
setPreviewInitialized: (ref?: ComposedRef) => Promise<void>;
|
||||||
|
/**
|
||||||
|
* Returns the current status of the stories.
|
||||||
|
*
|
||||||
|
* @returns {API_StatusState} The current status of the stories.
|
||||||
|
*/
|
||||||
|
getCurrentStoryStatus: () => Record<string, API_StatusObject>;
|
||||||
/**
|
/**
|
||||||
* Updates the status of a collection of stories.
|
* Updates the status of a collection of stories.
|
||||||
*
|
*
|
||||||
@ -630,6 +637,11 @@ export const init: ModuleFn<SubAPI, SubState> = ({
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
getCurrentStoryStatus: () => {
|
||||||
|
const { status, storyId } = store.getState();
|
||||||
|
return status[storyId as StoryId];
|
||||||
|
},
|
||||||
|
|
||||||
/* EXPERIMENTAL APIs */
|
/* EXPERIMENTAL APIs */
|
||||||
experimental_updateStatus: async (id, input) => {
|
experimental_updateStatus: async (id, input) => {
|
||||||
const { status, internal_index: index } = store.getState();
|
const { status, internal_index: index } = store.getState();
|
||||||
|
Loading…
x
Reference in New Issue
Block a user