UNPKG

next

Version:

The React Framework

275 lines (274 loc) • 11.4 kB
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime"; import { createElement as _createElement } from "react"; import { useMemo, Fragment, useState } from 'react'; import { CollapseIcon } from '../../icons/CollapseIcon'; function getAdjacentProps(isAdj) { return { 'data-nextjs-container-errors-pseudo-html--tag-adjacent': isAdj }; } /** * * Format component stack into pseudo HTML * component stack is an array of strings, e.g.: ['p', 'p', 'Page', ...] * * For html tags mismatch, it will render it for the code block * * ``` * <pre> * <code>{` * <Page> * <p red> * <p red> * `}</code> * </pre> * ``` * * For text mismatch, it will render it for the code block * * ``` * <pre> * <code>{` * <Page> * <p> * "Server Text" (green) * "Client Text" (red) * </p> * </Page> * `}</code> * ``` * * For bad text under a tag it will render it for the code block, * e.g. "Mismatched Text" under <p> * * ``` * <pre> * <code>{` * <Page> * <div> * <p> * "Mismatched Text" (red) * </p> * </div> * </Page> * `}</code> * ``` * */ export function PseudoHtmlDiff(param) { let { componentStackFrames, firstContent, secondContent, hydrationMismatchType, reactOutputComponentDiff, ...props } = param; const isHtmlTagsWarning = hydrationMismatchType === 'tag'; const isReactHydrationDiff = !!reactOutputComponentDiff; // For text mismatch, mismatched text will take 2 rows, so we display 4 rows of component stack const MAX_NON_COLLAPSED_FRAMES = isHtmlTagsWarning ? 6 : 4; const [isHtmlCollapsed, toggleCollapseHtml] = useState(true); const htmlComponents = useMemo(()=>{ const componentStacks = []; // React 19 unified mismatch if (isReactHydrationDiff) { let currentComponentIndex = componentStackFrames.length - 1; const reactComponentDiffLines = reactOutputComponentDiff.split('\n'); const diffHtmlStack = []; reactComponentDiffLines.forEach((line, index)=>{ let trimmedLine = line.trim(); const isDiffLine = trimmedLine[0] === '+' || trimmedLine[0] === '-'; const spaces = ' '.repeat(Math.max(componentStacks.length * 2, 1)); if (isDiffLine) { const sign = trimmedLine[0]; trimmedLine = trimmedLine.slice(1).trim() // trim spaces after sign ; diffHtmlStack.push(/*#__PURE__*/ _jsxs("span", { "data-nextjs-container-errors-pseudo-html--diff": sign === '+' ? 'add' : 'remove', children: [ sign, spaces, trimmedLine, '\n' ] }, 'comp-diff' + index)); } else if (currentComponentIndex >= 0) { const isUserLandComponent = trimmedLine.startsWith('<' + componentStackFrames[currentComponentIndex].component); // If it's matched userland component or it's ... we will keep the component stack in diff if (isUserLandComponent || trimmedLine === '...') { currentComponentIndex--; componentStacks.push(/*#__PURE__*/ _jsxs("span", { children: [ spaces, trimmedLine, '\n' ] }, 'comp-diff' + index)); } else if (!isHtmlCollapsed) { componentStacks.push(/*#__PURE__*/ _jsxs("span", { children: [ spaces, trimmedLine, '\n' ] }, 'comp-diff' + index)); } } else if (!isHtmlCollapsed) { // In general, if it's not collapsed, show the whole diff componentStacks.push(/*#__PURE__*/ _jsxs("span", { children: [ spaces, trimmedLine, '\n' ] }, 'comp-diff' + index)); } }); return componentStacks.concat(diffHtmlStack); } const nestedHtmlStack = []; const tagNames = isHtmlTagsWarning ? [ firstContent.replace(/<|>/g, ''), secondContent.replace(/<|>/g, '') ] : []; let lastText = ''; const componentStack = componentStackFrames.map((frame)=>frame.component).reverse(); // [child index, parent index] const matchedIndex = [ -1, -1 ]; if (isHtmlTagsWarning) { // Reverse search for the child tag for(let i = componentStack.length - 1; i >= 0; i--){ if (componentStack[i] === tagNames[0]) { matchedIndex[0] = i; break; } } // Start searching parent tag from child tag above for(let i = matchedIndex[0] - 1; i >= 0; i--){ if (componentStack[i] === tagNames[1]) { matchedIndex[1] = i; break; } } } componentStack.forEach((component, index, componentList)=>{ const spaces = ' '.repeat(nestedHtmlStack.length * 2); // When component is the server or client tag name, highlight it const isHighlightedTag = isHtmlTagsWarning ? index === matchedIndex[0] || index === matchedIndex[1] : tagNames.includes(component); const isAdjacentTag = isHighlightedTag || Math.abs(index - matchedIndex[0]) <= 1 || Math.abs(index - matchedIndex[1]) <= 1; const isLastFewFrames = !isHtmlTagsWarning && index >= componentList.length - 6; const adjProps = getAdjacentProps(isAdjacentTag); if (isHtmlTagsWarning && isAdjacentTag || isLastFewFrames) { const codeLine = /*#__PURE__*/ _jsxs("span", { children: [ spaces, /*#__PURE__*/ _jsx("span", { ...adjProps, ...isHighlightedTag ? { 'data-nextjs-container-errors-pseudo-html--tag-error': true } : undefined, children: "<" + component + ">\n" }) ] }); lastText = component; const wrappedCodeLine = /*#__PURE__*/ _jsxs(Fragment, { children: [ codeLine, isHighlightedTag && /*#__PURE__*/ _jsx("span", { "data-nextjs-container-errors-pseudo-html--hint": true, children: spaces + '^'.repeat(component.length + 2) + '\n' }) ] }, nestedHtmlStack.length); nestedHtmlStack.push(wrappedCodeLine); } else { if (nestedHtmlStack.length >= MAX_NON_COLLAPSED_FRAMES && isHtmlCollapsed) { return; } if (!isHtmlCollapsed || isLastFewFrames) { nestedHtmlStack.push(/*#__PURE__*/ _createElement("span", { ...adjProps, key: nestedHtmlStack.length, children: [ spaces, '<' + component + '>\n' ] })); } else if (isHtmlCollapsed && lastText !== '...') { lastText = '...'; nestedHtmlStack.push(/*#__PURE__*/ _createElement("span", { ...adjProps, key: nestedHtmlStack.length, children: [ spaces, '...\n' ] })); } } }); // Hydration mismatch: text or text-tag if (!isHtmlTagsWarning) { const spaces = ' '.repeat(nestedHtmlStack.length * 2); let wrappedCodeLine; if (hydrationMismatchType === 'text') { // hydration type is "text", represent [server content, client content] wrappedCodeLine = /*#__PURE__*/ _jsxs(Fragment, { children: [ /*#__PURE__*/ _jsx("span", { "data-nextjs-container-errors-pseudo-html--diff": "remove", children: spaces + ('"' + firstContent + '"\n') }), /*#__PURE__*/ _jsx("span", { "data-nextjs-container-errors-pseudo-html--diff": "add", children: spaces + ('"' + secondContent + '"\n') }) ] }, nestedHtmlStack.length); } else if (hydrationMismatchType === 'text-in-tag') { // hydration type is "text-in-tag", represent [parent tag, mismatch content] wrappedCodeLine = /*#__PURE__*/ _jsxs(Fragment, { children: [ /*#__PURE__*/ _jsx("span", { "data-nextjs-container-errors-pseudo-html--tag-adjacent": true, children: spaces + ("<" + secondContent + ">\n") }), /*#__PURE__*/ _jsx("span", { "data-nextjs-container-errors-pseudo-html--diff": "remove", children: spaces + (' "' + firstContent + '"\n') }) ] }, nestedHtmlStack.length); } nestedHtmlStack.push(wrappedCodeLine); } return nestedHtmlStack; }, [ componentStackFrames, isHtmlCollapsed, firstContent, secondContent, isHtmlTagsWarning, hydrationMismatchType, MAX_NON_COLLAPSED_FRAMES, isReactHydrationDiff, reactOutputComponentDiff ]); return /*#__PURE__*/ _jsxs("div", { "data-nextjs-container-errors-pseudo-html": true, children: [ /*#__PURE__*/ _jsx("button", { tabIndex: 10, "data-nextjs-container-errors-pseudo-html-collapse": true, onClick: ()=>toggleCollapseHtml(!isHtmlCollapsed), children: /*#__PURE__*/ _jsx(CollapseIcon, { collapsed: isHtmlCollapsed }) }), /*#__PURE__*/ _jsx("pre", { ...props, children: /*#__PURE__*/ _jsx("code", { children: htmlComponents }) }) ] }); } //# sourceMappingURL=component-stack-pseudo-html.js.map