fix(app): line selection waits on ready
This commit is contained in:
@@ -128,20 +128,56 @@ export function Code<T>(props: CodeProps<T>) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const notifyRendered = () => {
|
const lineCount = () => {
|
||||||
if (!local.onRendered) return
|
const text = local.file.contents
|
||||||
|
const total = text.split("\n").length - (text.endsWith("\n") ? 1 : 0)
|
||||||
|
return Math.max(1, total)
|
||||||
|
}
|
||||||
|
|
||||||
|
const applySelection = (range: SelectedLineRange | null) => {
|
||||||
|
const root = getRoot()
|
||||||
|
if (!root) return false
|
||||||
|
|
||||||
|
const lines = lineCount()
|
||||||
|
if (root.querySelectorAll("[data-line]").length < lines) return false
|
||||||
|
|
||||||
|
if (!range) {
|
||||||
|
file().setSelectedLines(null)
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
const start = Math.min(range.start, range.end)
|
||||||
|
const end = Math.max(range.start, range.end)
|
||||||
|
|
||||||
|
if (start < 1 || end > lines) {
|
||||||
|
file().setSelectedLines(null)
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!root.querySelector(`[data-line="${start}"]`) || !root.querySelector(`[data-line="${end}"]`)) {
|
||||||
|
file().setSelectedLines(null)
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
const normalized = (() => {
|
||||||
|
if (range.endSide != null) return { start: range.start, end: range.end }
|
||||||
|
if (range.side !== "deletions") return range
|
||||||
|
if (root.querySelector("[data-deletions]") != null) return range
|
||||||
|
return { start: range.start, end: range.end }
|
||||||
|
})()
|
||||||
|
|
||||||
|
file().setSelectedLines(normalized)
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
const notifyRendered = () => {
|
||||||
observer?.disconnect()
|
observer?.disconnect()
|
||||||
observer = undefined
|
observer = undefined
|
||||||
renderToken++
|
renderToken++
|
||||||
|
|
||||||
const token = renderToken
|
const token = renderToken
|
||||||
|
|
||||||
const lines = (() => {
|
const lines = lineCount()
|
||||||
const text = local.file.contents
|
|
||||||
const total = text.split("\n").length - (text.endsWith("\n") ? 1 : 0)
|
|
||||||
return Math.max(1, total)
|
|
||||||
})()
|
|
||||||
|
|
||||||
const isReady = (root: ShadowRoot) => root.querySelectorAll("[data-line]").length >= lines
|
const isReady = (root: ShadowRoot) => root.querySelectorAll("[data-line]").length >= lines
|
||||||
|
|
||||||
@@ -152,6 +188,7 @@ export function Code<T>(props: CodeProps<T>) {
|
|||||||
observer = undefined
|
observer = undefined
|
||||||
requestAnimationFrame(() => {
|
requestAnimationFrame(() => {
|
||||||
if (token !== renderToken) return
|
if (token !== renderToken) return
|
||||||
|
applySelection(lastSelection)
|
||||||
local.onRendered?.()
|
local.onRendered?.()
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@@ -241,7 +278,7 @@ export function Code<T>(props: CodeProps<T>) {
|
|||||||
|
|
||||||
const setSelectedLines = (range: SelectedLineRange | null) => {
|
const setSelectedLines = (range: SelectedLineRange | null) => {
|
||||||
lastSelection = range
|
lastSelection = range
|
||||||
file().setSelectedLines(range)
|
applySelection(range)
|
||||||
}
|
}
|
||||||
|
|
||||||
const scheduleSelectionUpdate = () => {
|
const scheduleSelectionUpdate = () => {
|
||||||
|
|||||||
@@ -38,6 +38,77 @@ export function Diff<T>(props: SSRDiffProps<T>) {
|
|||||||
fileDiffRef.removeAttribute("data-color-scheme")
|
fileDiffRef.removeAttribute("data-color-scheme")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const lineIndex = (split: boolean, element: HTMLElement) => {
|
||||||
|
const raw = element.dataset.lineIndex
|
||||||
|
if (!raw) return
|
||||||
|
const values = raw
|
||||||
|
.split(",")
|
||||||
|
.map((value) => parseInt(value, 10))
|
||||||
|
.filter((value) => !Number.isNaN(value))
|
||||||
|
if (values.length === 0) return
|
||||||
|
if (!split) return values[0]
|
||||||
|
if (values.length === 2) return values[1]
|
||||||
|
return values[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
const rowIndex = (root: ShadowRoot, split: boolean, line: number, side: "additions" | "deletions" | undefined) => {
|
||||||
|
const nodes = Array.from(root.querySelectorAll(`[data-line="${line}"], [data-alt-line="${line}"]`)).filter(
|
||||||
|
(node): node is HTMLElement => node instanceof HTMLElement,
|
||||||
|
)
|
||||||
|
if (nodes.length === 0) return
|
||||||
|
|
||||||
|
const targetSide = side ?? "additions"
|
||||||
|
|
||||||
|
for (const node of nodes) {
|
||||||
|
if (findSide(node) === targetSide) return lineIndex(split, node)
|
||||||
|
if (parseInt(node.dataset.altLine ?? "", 10) === line) return lineIndex(split, node)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const fixSelection = (range: SelectedLineRange | null) => {
|
||||||
|
if (!range) return range
|
||||||
|
const root = getRoot()
|
||||||
|
if (!root) return
|
||||||
|
|
||||||
|
const diffs = root.querySelector("[data-diffs]")
|
||||||
|
if (!(diffs instanceof HTMLElement)) return
|
||||||
|
|
||||||
|
const split = diffs.dataset.type === "split"
|
||||||
|
|
||||||
|
const start = rowIndex(root, split, range.start, range.side)
|
||||||
|
const end = rowIndex(root, split, range.end, range.endSide ?? range.side)
|
||||||
|
|
||||||
|
if (start === undefined || end === undefined) {
|
||||||
|
if (root.querySelector("[data-line], [data-alt-line]") == null) return
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
if (start <= end) return range
|
||||||
|
|
||||||
|
const side = range.endSide ?? range.side
|
||||||
|
const swapped: SelectedLineRange = {
|
||||||
|
start: range.end,
|
||||||
|
end: range.start,
|
||||||
|
}
|
||||||
|
if (side) swapped.side = side
|
||||||
|
if (range.endSide && range.side) swapped.endSide = range.side
|
||||||
|
|
||||||
|
return swapped
|
||||||
|
}
|
||||||
|
|
||||||
|
const setSelectedLines = (range: SelectedLineRange | null, attempt = 0) => {
|
||||||
|
const diff = fileDiffInstance
|
||||||
|
if (!diff) return
|
||||||
|
|
||||||
|
const fixed = fixSelection(range)
|
||||||
|
if (fixed === undefined) {
|
||||||
|
if (attempt >= 120) return
|
||||||
|
requestAnimationFrame(() => setSelectedLines(range, attempt + 1))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
diff.setSelectedLines(fixed)
|
||||||
|
}
|
||||||
|
|
||||||
const findSide = (element: HTMLElement): "additions" | "deletions" => {
|
const findSide = (element: HTMLElement): "additions" | "deletions" => {
|
||||||
const line = element.closest("[data-line], [data-alt-line]")
|
const line = element.closest("[data-line], [data-alt-line]")
|
||||||
if (line instanceof HTMLElement) {
|
if (line instanceof HTMLElement) {
|
||||||
@@ -159,14 +230,14 @@ export function Diff<T>(props: SSRDiffProps<T>) {
|
|||||||
containerWrapper: container,
|
containerWrapper: container,
|
||||||
})
|
})
|
||||||
|
|
||||||
fileDiffInstance.setSelectedLines(local.selectedLines ?? null)
|
setSelectedLines(local.selectedLines ?? null)
|
||||||
|
|
||||||
createEffect(() => {
|
createEffect(() => {
|
||||||
fileDiffInstance?.setLineAnnotations(local.annotations ?? [])
|
fileDiffInstance?.setLineAnnotations(local.annotations ?? [])
|
||||||
})
|
})
|
||||||
|
|
||||||
createEffect(() => {
|
createEffect(() => {
|
||||||
fileDiffInstance?.setSelectedLines(local.selectedLines ?? null)
|
setSelectedLines(local.selectedLines ?? null)
|
||||||
})
|
})
|
||||||
|
|
||||||
createEffect(() => {
|
createEffect(() => {
|
||||||
|
|||||||
@@ -115,9 +115,64 @@ export function Diff<T>(props: DiffProps<T>) {
|
|||||||
host.removeAttribute("data-color-scheme")
|
host.removeAttribute("data-color-scheme")
|
||||||
}
|
}
|
||||||
|
|
||||||
const notifyRendered = () => {
|
const lineIndex = (split: boolean, element: HTMLElement) => {
|
||||||
if (!local.onRendered) return
|
const raw = element.dataset.lineIndex
|
||||||
|
if (!raw) return
|
||||||
|
const values = raw
|
||||||
|
.split(",")
|
||||||
|
.map((value) => parseInt(value, 10))
|
||||||
|
.filter((value) => !Number.isNaN(value))
|
||||||
|
if (values.length === 0) return
|
||||||
|
if (!split) return values[0]
|
||||||
|
if (values.length === 2) return values[1]
|
||||||
|
return values[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
const rowIndex = (root: ShadowRoot, split: boolean, line: number, side: SelectionSide | undefined) => {
|
||||||
|
const nodes = Array.from(root.querySelectorAll(`[data-line="${line}"], [data-alt-line="${line}"]`)).filter(
|
||||||
|
(node): node is HTMLElement => node instanceof HTMLElement,
|
||||||
|
)
|
||||||
|
if (nodes.length === 0) return
|
||||||
|
|
||||||
|
const targetSide = side ?? "additions"
|
||||||
|
|
||||||
|
for (const node of nodes) {
|
||||||
|
if (findSide(node) === targetSide) return lineIndex(split, node)
|
||||||
|
if (parseInt(node.dataset.altLine ?? "", 10) === line) return lineIndex(split, node)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const fixSelection = (range: SelectedLineRange | null) => {
|
||||||
|
if (!range) return range
|
||||||
|
const root = getRoot()
|
||||||
|
if (!root) return
|
||||||
|
|
||||||
|
const diffs = root.querySelector("[data-diffs]")
|
||||||
|
if (!(diffs instanceof HTMLElement)) return
|
||||||
|
|
||||||
|
const split = diffs.dataset.type === "split"
|
||||||
|
|
||||||
|
const start = rowIndex(root, split, range.start, range.side)
|
||||||
|
const end = rowIndex(root, split, range.end, range.endSide ?? range.side)
|
||||||
|
if (start === undefined || end === undefined) {
|
||||||
|
if (root.querySelector("[data-line], [data-alt-line]") == null) return
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
if (start <= end) return range
|
||||||
|
|
||||||
|
const side = range.endSide ?? range.side
|
||||||
|
const swapped: SelectedLineRange = {
|
||||||
|
start: range.end,
|
||||||
|
end: range.start,
|
||||||
|
}
|
||||||
|
|
||||||
|
if (side) swapped.side = side
|
||||||
|
if (range.endSide && range.side) swapped.endSide = range.side
|
||||||
|
|
||||||
|
return swapped
|
||||||
|
}
|
||||||
|
|
||||||
|
const notifyRendered = () => {
|
||||||
observer?.disconnect()
|
observer?.disconnect()
|
||||||
observer = undefined
|
observer = undefined
|
||||||
renderToken++
|
renderToken++
|
||||||
@@ -134,6 +189,7 @@ export function Diff<T>(props: DiffProps<T>) {
|
|||||||
observer = undefined
|
observer = undefined
|
||||||
requestAnimationFrame(() => {
|
requestAnimationFrame(() => {
|
||||||
if (token !== renderToken) return
|
if (token !== renderToken) return
|
||||||
|
setSelectedLines(lastSelection)
|
||||||
local.onRendered?.()
|
local.onRendered?.()
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@@ -173,7 +229,8 @@ export function Diff<T>(props: DiffProps<T>) {
|
|||||||
const root = getRoot()
|
const root = getRoot()
|
||||||
if (typeof MutationObserver === "undefined") {
|
if (typeof MutationObserver === "undefined") {
|
||||||
if (!root || !isReady(root)) return
|
if (!root || !isReady(root)) return
|
||||||
local.onRendered()
|
setSelectedLines(lastSelection)
|
||||||
|
local.onRendered?.()
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -214,41 +271,14 @@ export function Diff<T>(props: DiffProps<T>) {
|
|||||||
)
|
)
|
||||||
if (code.length === 0) return
|
if (code.length === 0) return
|
||||||
|
|
||||||
const lineIndex = (element: HTMLElement) => {
|
|
||||||
const raw = element.dataset.lineIndex
|
|
||||||
if (!raw) return
|
|
||||||
const values = raw
|
|
||||||
.split(",")
|
|
||||||
.map((value) => parseInt(value, 10))
|
|
||||||
.filter((value) => !Number.isNaN(value))
|
|
||||||
if (values.length === 0) return
|
|
||||||
if (!split) return values[0]
|
|
||||||
if (values.length === 2) return values[1]
|
|
||||||
return values[0]
|
|
||||||
}
|
|
||||||
|
|
||||||
const rowIndex = (line: number, side: SelectionSide | undefined) => {
|
|
||||||
const nodes = Array.from(root.querySelectorAll(`[data-line="${line}"], [data-alt-line="${line}"]`)).filter(
|
|
||||||
(node): node is HTMLElement => node instanceof HTMLElement,
|
|
||||||
)
|
|
||||||
if (nodes.length === 0) return
|
|
||||||
|
|
||||||
const targetSide = side ?? "additions"
|
|
||||||
|
|
||||||
for (const node of nodes) {
|
|
||||||
if (findSide(node) === targetSide) return lineIndex(node)
|
|
||||||
if (parseInt(node.dataset.altLine ?? "", 10) === line) return lineIndex(node)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const range of ranges) {
|
for (const range of ranges) {
|
||||||
const start = rowIndex(range.start, range.side)
|
const start = rowIndex(root, split, range.start, range.side)
|
||||||
if (start === undefined) continue
|
if (start === undefined) continue
|
||||||
|
|
||||||
const end = (() => {
|
const end = (() => {
|
||||||
const same = range.end === range.start && (range.endSide == null || range.endSide === range.side)
|
const same = range.end === range.start && (range.endSide == null || range.endSide === range.side)
|
||||||
if (same) return start
|
if (same) return start
|
||||||
return rowIndex(range.end, range.endSide ?? range.side)
|
return rowIndex(root, split, range.end, range.endSide ?? range.side)
|
||||||
})()
|
})()
|
||||||
if (end === undefined) continue
|
if (end === undefined) continue
|
||||||
|
|
||||||
@@ -258,7 +288,7 @@ export function Diff<T>(props: DiffProps<T>) {
|
|||||||
for (const block of code) {
|
for (const block of code) {
|
||||||
for (const element of Array.from(block.children)) {
|
for (const element of Array.from(block.children)) {
|
||||||
if (!(element instanceof HTMLElement)) continue
|
if (!(element instanceof HTMLElement)) continue
|
||||||
const idx = lineIndex(element)
|
const idx = lineIndex(split, element)
|
||||||
if (idx === undefined) continue
|
if (idx === undefined) continue
|
||||||
if (idx > last) break
|
if (idx > last) break
|
||||||
if (idx < first) continue
|
if (idx < first) continue
|
||||||
@@ -275,8 +305,15 @@ export function Diff<T>(props: DiffProps<T>) {
|
|||||||
const setSelectedLines = (range: SelectedLineRange | null) => {
|
const setSelectedLines = (range: SelectedLineRange | null) => {
|
||||||
const active = current()
|
const active = current()
|
||||||
if (!active) return
|
if (!active) return
|
||||||
lastSelection = range
|
|
||||||
active.setSelectedLines(range)
|
const fixed = fixSelection(range)
|
||||||
|
if (fixed === undefined) {
|
||||||
|
lastSelection = range
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
lastSelection = fixed
|
||||||
|
active.setSelectedLines(fixed)
|
||||||
}
|
}
|
||||||
|
|
||||||
const updateSelection = () => {
|
const updateSelection = () => {
|
||||||
|
|||||||
Reference in New Issue
Block a user