Gert Hengeveld 1505bbc9b6 Many fixes
2022-06-13 23:32:08 +02:00

124 lines
4.1 KiB
TypeScript

import React from 'react';
import { styled, typography } from '@storybook/theming';
import { Node } from './MethodCall';
const getParams = (line: string, fromIndex = 0): string => {
for (let i = fromIndex, depth = 1; i < line.length; i += 1) {
if (line[i] === '(') depth += 1;
else if (line[i] === ')') depth -= 1;
if (depth === 0) return line.slice(fromIndex, i);
}
return '';
};
const parseValue = (value: string): any => {
try {
return value === 'undefined' ? undefined : JSON.parse(value);
} catch (e) {
return value;
}
};
const StyledExpected = styled.span(({ theme }) => ({
color: theme.color.positive,
}));
const StyledReceived = styled.span(({ theme }) => ({
color: theme.color.negative,
}));
export const Received = ({ value, parsed }: { value: any; parsed?: boolean }) =>
parsed ? (
<Node showObjectInspector value={value} style={{ color: '#D43900' }} />
) : (
<StyledReceived>{value}</StyledReceived>
);
export const Expected = ({ value, parsed }: { value: any; parsed?: boolean }) => {
if (parsed) {
if (typeof value === 'string' && value.startsWith('called with')) {
return <>{value}</>;
}
return <Node showObjectInspector value={value} style={{ color: '#16B242' }} />;
}
return <StyledExpected>{value}</StyledExpected>;
};
export const MatcherResult = ({ message }: { message: string }) => {
const lines = message.split('\n');
return (
<pre
style={{
margin: 0,
padding: '8px 10px 8px 36px',
fontSize: typography.size.s1,
}}
>
{lines.flatMap((line: string, index: number) => {
if (line.startsWith('expect(')) {
const received = getParams(line, 7);
const remainderIndex = received && 7 + received.length;
const matcher = received && line.slice(remainderIndex).match(/\.(to|last|nth)[A-Z]\w+\(/);
if (matcher) {
const expectedIndex = remainderIndex + matcher.index + matcher[0].length;
const expected = getParams(line, expectedIndex);
if (expected) {
return [
'expect(',
<Received key={`received_${received}`} value={received} />,
line.slice(remainderIndex, expectedIndex),
<Expected key={`expected_${expected}`} value={expected} />,
line.slice(expectedIndex + expected.length),
<br key={`br${index}`} />,
];
}
}
}
if (line.match(/^\s*- /)) {
return [<Expected key={line + index} value={line} />, <br key={`br${index}`} />];
}
if (line.match(/^\s*\+ /)) {
return [<Received key={line + index} value={line} />, <br key={`br${index}`} />];
}
const [, assertionLabel, assertionValue] = line.match(/^(Expected|Received): (.*)$/) || [];
if (assertionLabel && assertionValue) {
return assertionLabel === 'Expected'
? [
'Expected: ',
<Expected key={line + index} value={parseValue(assertionValue)} parsed />,
<br key={`br${index}`} />,
]
: [
'Received: ',
<Received key={line + index} value={parseValue(assertionValue)} parsed />,
<br key={`br${index}`} />,
];
}
const [, prefix, numberOfCalls] =
line.match(/(Expected number|Received number|Number) of calls: (\d+)$/i) || [];
if (prefix && numberOfCalls) {
return [
`${prefix} of calls: `,
<Node key={line + index} value={Number(numberOfCalls)} />,
<br key={`br${index}`} />,
];
}
const [, receivedValue] = line.match(/^Received has value: (.+)$/) || [];
if (receivedValue) {
return [
'Received has value: ',
<Node key={line + index} value={parseValue(receivedValue)} />,
<br key={`br${index}`} />,
];
}
return [<span key={line + index}>{line}</span>, <br key={`br${index}`} />];
})}
</pre>
);
};