2024-12-16 12:45:42 +00:00
|
|
|
import * as data from "./data.ts";
|
|
|
|
|
|
|
|
type Color = { r: number; g: number; b: number };
|
|
|
|
|
|
|
|
function lerp2(ratio: number, start: number, end: number) {
|
|
|
|
return (1 - ratio) * start + ratio * end;
|
|
|
|
}
|
|
|
|
|
|
|
|
function lerp3(ratio: number, start: number, middle: number, end: number) {
|
|
|
|
return (1 - ratio) * lerp2(ratio, start, middle) +
|
|
|
|
ratio * lerp2(ratio, middle, end);
|
|
|
|
}
|
|
|
|
|
|
|
|
function colorLerp(
|
|
|
|
ratio: number,
|
|
|
|
start: Color,
|
|
|
|
middle: Color,
|
|
|
|
end: Color,
|
|
|
|
): Color {
|
|
|
|
return {
|
|
|
|
r: lerp3(ratio, start.r, middle.r, end.r),
|
|
|
|
g: lerp3(ratio, start.g, middle.g, end.g),
|
|
|
|
b: lerp3(ratio, start.b, middle.b, end.b),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2024-12-16 14:33:12 +00:00
|
|
|
function colorToString(color: Color): string {
|
|
|
|
return `rgb(${color.r}, ${color.g}, ${color.b})`;
|
|
|
|
}
|
|
|
|
|
|
|
|
const GREEN = { r: 42, g: 121, b: 82 };
|
|
|
|
const YELLOW = {
|
|
|
|
r: 247,
|
|
|
|
g: 203,
|
|
|
|
b: 21,
|
|
|
|
};
|
|
|
|
const RED = {
|
|
|
|
r: 167,
|
|
|
|
g: 29,
|
|
|
|
b: 49,
|
|
|
|
};
|
|
|
|
|
|
|
|
export type CodeCovRender = "performance" | "test-coverage";
|
|
|
|
|
2024-12-16 12:45:42 +00:00
|
|
|
export function loadCodeCoverage(
|
|
|
|
text: string,
|
|
|
|
input: data.CodeCovEntry[],
|
|
|
|
tooltip: HTMLElement,
|
2024-12-16 14:33:12 +00:00
|
|
|
mode: CodeCovRender,
|
|
|
|
): HTMLPreElement {
|
|
|
|
const container = document.createElement("pre");
|
|
|
|
container.classList.add("code-source");
|
2024-12-16 12:45:42 +00:00
|
|
|
if (input.length === 0) {
|
2024-12-16 14:33:12 +00:00
|
|
|
return container;
|
2024-12-16 12:45:42 +00:00
|
|
|
}
|
|
|
|
const entries = input.toSorted((
|
|
|
|
a: data.CodeCovEntry,
|
|
|
|
b: data.CodeCovEntry,
|
|
|
|
) => b.index - a.index);
|
|
|
|
const charEntries: { [key: string]: data.CodeCovEntry } = {};
|
|
|
|
const elements: HTMLElement[] = [];
|
|
|
|
let line = 1;
|
|
|
|
let col = 1;
|
2024-12-16 14:33:12 +00:00
|
|
|
const maxPerfCovers = entries.map((v) => v.covers).reduce((acc, v) =>
|
2024-12-16 12:45:42 +00:00
|
|
|
acc > Math.log10(v) ? acc : Math.log10(v)
|
|
|
|
);
|
|
|
|
for (let index = 0; index < text.length; ++index) {
|
|
|
|
if (text[index] === "\n") {
|
|
|
|
col = 1;
|
|
|
|
line += 1;
|
|
|
|
const newlineSpan = document.createElement("span");
|
|
|
|
newlineSpan.innerText = "\n";
|
|
|
|
elements.push(newlineSpan);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const entry = entries.find((entry) => index >= entry.index);
|
|
|
|
if (!entry) {
|
|
|
|
throw new Error("unreachable");
|
|
|
|
}
|
|
|
|
charEntries[`${line}-${col}`] = entry;
|
|
|
|
|
2024-12-16 14:33:12 +00:00
|
|
|
const perfColor = (ratio: number) => {
|
|
|
|
const clr = colorLerp(ratio, GREEN, YELLOW, RED);
|
|
|
|
return colorToString(clr);
|
2024-12-16 12:45:42 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
const span = document.createElement("span");
|
2024-12-16 14:33:12 +00:00
|
|
|
span.style.backgroundColor = mode === "performance"
|
|
|
|
? perfColor(
|
|
|
|
Math.log10(entry.covers) / maxPerfCovers,
|
|
|
|
)
|
|
|
|
: entry.covers > 0
|
|
|
|
? colorToString(GREEN)
|
|
|
|
: colorToString(RED);
|
2024-12-16 12:45:42 +00:00
|
|
|
span.innerText = text[index];
|
|
|
|
span.dataset.covers = entry.covers.toString();
|
|
|
|
elements.push(span);
|
|
|
|
col += 1;
|
|
|
|
}
|
|
|
|
function positionInBox(
|
|
|
|
position: [number, number],
|
|
|
|
boundingRect: {
|
|
|
|
left: number;
|
|
|
|
top: number;
|
|
|
|
right: number;
|
|
|
|
bottom: number;
|
|
|
|
},
|
|
|
|
) {
|
|
|
|
const [x, y] = position;
|
|
|
|
const outside = x < boundingRect.left ||
|
|
|
|
x >= boundingRect.right || y < boundingRect.top ||
|
|
|
|
y >= boundingRect.bottom;
|
|
|
|
return !outside;
|
|
|
|
}
|
|
|
|
container.append(...elements);
|
|
|
|
document.addEventListener("mousemove", (event) => {
|
|
|
|
const [x, y] = [event.clientX, event.clientY];
|
|
|
|
const outerBox = container.getBoundingClientRect();
|
|
|
|
if (!positionInBox([x, y], outerBox)) {
|
|
|
|
tooltip.hidden = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const element = elements.find((element) => {
|
|
|
|
if (typeof element === "string") {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (!element.dataset.covers) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const isIn = positionInBox([x, y], element.getBoundingClientRect());
|
|
|
|
return isIn;
|
|
|
|
});
|
|
|
|
if (!element) {
|
|
|
|
tooltip.hidden = true;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const maybeCovers = element.dataset.covers;
|
|
|
|
if (!maybeCovers) {
|
|
|
|
throw new Error("unreachable");
|
|
|
|
}
|
|
|
|
const covers = parseInt(maybeCovers);
|
|
|
|
tooltip.hidden = false;
|
|
|
|
tooltip.style.left = `${event.clientX + 20}px`;
|
|
|
|
tooltip.style.top = `${event.clientY + 20}px`;
|
|
|
|
tooltip.innerText = `Ran ${covers} time${covers !== 1 ? "s" : ""}`;
|
|
|
|
});
|
2024-12-16 14:33:12 +00:00
|
|
|
return container;
|
2024-12-16 12:45:42 +00:00
|
|
|
}
|