blased web

This commit is contained in:
Theis Pieter Hollebeek 2024-12-06 12:54:43 +01:00
parent 8771d887c0
commit fd072ce0e9
3 changed files with 62 additions and 18 deletions

View File

@ -30,7 +30,6 @@
<pre id="code-coverage"></pre> <pre id="code-coverage"></pre>
<div id="cover">Process is currently running</div> <div id="cover">Process is currently running</div>
</main> </main>
<span id="covers-tooltip"></span>
</div> </div>
</div> </div>
</body> </body>

View File

@ -7,13 +7,11 @@ type CodeCovEntry = {
function loadCodeCoverage( function loadCodeCoverage(
text: string, text: string,
codeCoverageData: CodeCovEntry[], data: CodeCovEntry[],
codeCoverageDiv: HTMLPreElement, container: HTMLPreElement,
tooltip: HTMLElement,
) { ) {
const tooltip = document.createElement("span"); const entries = data.toSorted((
tooltip.id = "covers-tooltip";
codeCoverageDiv.append(tooltip);
const entries = codeCoverageData.toSorted((
a: CodeCovEntry, a: CodeCovEntry,
b: CodeCovEntry, b: CodeCovEntry,
) => b.index - a.index); ) => b.index - a.index);
@ -61,11 +59,12 @@ function loadCodeCoverage(
y >= boundingRect.bottom; y >= boundingRect.bottom;
return !outside; return !outside;
} }
codeCoverageDiv.append(...elements); container.append(...elements);
document.addEventListener("mousemove", (event) => { document.addEventListener("mousemove", (event) => {
const [x, y] = [event.clientX, event.clientY]; const [x, y] = [event.clientX, event.clientY];
const outerBox = codeCoverageDiv.getBoundingClientRect(); const outerBox = container.getBoundingClientRect();
if (!positionInBox([x, y], outerBox)) { if (!positionInBox([x, y], outerBox)) {
tooltip.hidden = true;
return; return;
} }
const element = elements.find((element) => { const element = elements.find((element) => {
@ -82,7 +81,11 @@ function loadCodeCoverage(
tooltip.hidden = true; tooltip.hidden = true;
return; return;
} }
const covers = parseInt(element.dataset.covers); const maybeCovers = element.dataset.covers;
if (!maybeCovers) {
throw new Error("unreachable");
}
const covers = parseInt(maybeCovers);
tooltip.hidden = false; tooltip.hidden = false;
tooltip.style.left = `${event.clientX + 20}px`; tooltip.style.left = `${event.clientX + 20}px`;
tooltip.style.top = `${event.clientY + 20}px`; tooltip.style.top = `${event.clientY + 20}px`;
@ -213,18 +216,51 @@ loop {
"flame-graph": () => void; "flame-graph": () => void;
}; };
function countLines(code: string) {
let lines = 0;
for (const char of code) {
if (char === "\n") lines += 1;
}
return lines;
}
function createLineElement(code: string): HTMLPreElement {
const lines = countLines(code) + 1;
const maxLineWidth = lines.toString().length;
let text = "";
for (let i = 1; i < lines; ++i) {
const node = i.toString().padStart(maxLineWidth);
text += node;
text += "\n";
}
const lineElement = document.createElement("pre");
lineElement.classList.add("code-lines");
lineElement.innerText = text;
return lineElement;
}
const view = document.querySelector("#view")!; const view = document.querySelector("#view")!;
const renderFunctions: RenderFns = { const renderFunctions: RenderFns = {
"source-code": () => { "source-code": () => {
const container = document.createElement("div");
container.classList.add("code-container");
const lines = createLineElement(codeData);
const code = document.createElement("pre"); const code = document.createElement("pre");
code.innerHTML = codeData; code.innerText = codeData;
view.replaceChildren(code); container.append(lines, code);
view.replaceChildren(container);
}, },
"code-coverage": () => { "code-coverage": () => {
const codeCoverageElement = document.createElement("pre"); const container = document.createElement("div");
loadCodeCoverage(codeData, codeCoverageData, codeCoverageElement); container.classList.add("code-container");
const tooltip = document.createElement("div");
tooltip.id = "covers-tooltip";
const code = document.createElement("pre");
loadCodeCoverage(codeData, codeCoverageData, code, tooltip);
const lines = createLineElement(codeData);
container.append(lines, code);
const view = document.querySelector("#view")!; const view = document.querySelector("#view")!;
view.replaceChildren(codeCoverageElement); view.replaceChildren(container, tooltip);
}, },
"flame-graph": () => {}, "flame-graph": () => {},
}; };

View File

@ -88,10 +88,18 @@ main #cover {
cursor: pointer; cursor: pointer;
} }
#view pre { #view .code-container {
display: flex;
font-size: 1rem; font-size: 1rem;
gap: 1ch;
} }
#view .code-lines {
border-right: 1px solid var(--white);
padding-right: 0.5ch;
}
#views-nav input:checked + label { #views-nav input:checked + label {
background-color: var(--code-status); background-color: var(--code-status);
color: var(--black); color: var(--black);
@ -113,8 +121,9 @@ main #cover {
left: 0; left: 0;
padding: 3px; padding: 3px;
border-radius: 3px; border-radius: 3px;
background-color: var(--bg-2); background-color: var(--black);
box-shadow: 2px 2px 2px black; border: 2px solid var(--code-status);
border-radius: 0.25rem;
color: #eee; color: #eee;
} }