mirror of
https://github.com/aljazceru/opencode.git
synced 2025-12-22 02:04:22 +01:00
142 lines
3.8 KiB
TypeScript
142 lines
3.8 KiB
TypeScript
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<DiffViewProps> = (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 (
|
|
<div class={`${styles.diff} ${props.class ?? ""}`}>
|
|
<div class={styles.column}>
|
|
{rows().map((r) => (
|
|
<CodeBlock
|
|
code={r.left}
|
|
lang={props.lang}
|
|
data-section="cell"
|
|
data-diff-type={r.type === "removed" || r.type === "modified" ? "removed" : ""}
|
|
/>
|
|
))}
|
|
</div>
|
|
|
|
<div class={styles.column}>
|
|
{rows().map((r) => (
|
|
<CodeBlock
|
|
code={r.right}
|
|
lang={props.lang}
|
|
data-section="cell"
|
|
data-diff-type={r.type === "added" || r.type === "modified" ? "added" : ""}
|
|
/>
|
|
))}
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
export default DiffView
|