import { type Component, createMemo } from "solid-js" import { parsePatch } from "diff" import CodeBlock from "./CodeBlock" import styles from "./diffview.module.css" type DiffRow = { left: string right: string type: "added" | "removed" | "unchanged" | "modified" } interface DiffViewProps { diff: string lang?: string class?: string } const DiffView: Component = (props) => { const rows = createMemo(() => { const diffRows: DiffRow[] = [] try { const patches = parsePatch(props.diff) for (const patch of patches) { for (const hunk of patch.hunks) { const lines = hunk.lines let i = 0 while (i < lines.length) { const line = lines[i] const content = line.slice(1) const prefix = line[0] if (prefix === '-') { // Look ahead for consecutive additions to pair with removals const removals: string[] = [content] let j = i + 1 // Collect all consecutive removals while (j < lines.length && lines[j][0] === '-') { removals.push(lines[j].slice(1)) j++ } // Collect all consecutive additions that follow const additions: string[] = [] while (j < lines.length && lines[j][0] === '+') { additions.push(lines[j].slice(1)) j++ } // Pair removals with additions const maxLength = Math.max(removals.length, additions.length) for (let k = 0; k < maxLength; k++) { const hasLeft = !!removals[k] const hasRight = !!additions[k] if (hasLeft && hasRight) { // Replacement - left is removed, right is added diffRows.push({ left: removals[k], right: additions[k], type: "modified" }) } else if (hasLeft) { // Pure removal diffRows.push({ left: removals[k], right: "", type: "removed" }) } else { // Pure addition diffRows.push({ left: "", right: additions[k], type: "added" }) } } i = j } else if (prefix === '+') { // Standalone addition (not paired with removal) diffRows.push({ left: "", right: content, type: "added" }) i++ } else if (prefix === ' ') { diffRows.push({ left: content, right: content, type: "unchanged" }) i++ } else { i++ } } } } } catch (error) { console.error("Failed to parse patch:", error) return [] } return diffRows }) return (
{rows().map((r) => ( ))}
{rows().map((r) => ( ))}
) } export default DiffView