@gguf/claw
Version:
WhatsApp gateway CLI (Baileys web) with Pi RPC agent
279 lines (277 loc) • 8.48 kB
JavaScript
import { s as visibleWidth } from "./subsystem-46MXi6Ip.js";
import { o as displayString } from "./utils-Dg0Xbl6w.js";
//#region src/terminal/table.ts
function repeat(ch, n) {
if (n <= 0) return "";
return ch.repeat(n);
}
function padCell(text, width, align) {
const w = visibleWidth(text);
if (w >= width) return text;
const pad = width - w;
if (align === "right") return `${repeat(" ", pad)}${text}`;
if (align === "center") {
const left = Math.floor(pad / 2);
const right = pad - left;
return `${repeat(" ", left)}${text}${repeat(" ", right)}`;
}
return `${text}${repeat(" ", pad)}`;
}
function wrapLine(text, width) {
if (width <= 0) return [text];
const ESC = "\x1B";
const tokens = [];
for (let i = 0; i < text.length;) {
if (text[i] === ESC) {
if (text[i + 1] === "[") {
let j = i + 2;
while (j < text.length) {
const ch = text[j];
if (ch === "m") break;
if (ch && ch >= "0" && ch <= "9") {
j += 1;
continue;
}
if (ch === ";") {
j += 1;
continue;
}
break;
}
if (text[j] === "m") {
tokens.push({
kind: "ansi",
value: text.slice(i, j + 1)
});
i = j + 1;
continue;
}
}
if (text[i + 1] === "]" && text.slice(i + 2, i + 5) === "8;;") {
const st = text.indexOf(`${ESC}\\`, i + 5);
if (st >= 0) {
tokens.push({
kind: "ansi",
value: text.slice(i, st + 2)
});
i = st + 2;
continue;
}
}
}
const cp = text.codePointAt(i);
if (!cp) break;
const ch = String.fromCodePoint(cp);
tokens.push({
kind: "char",
value: ch
});
i += ch.length;
}
const firstCharIndex = tokens.findIndex((t) => t.kind === "char");
if (firstCharIndex < 0) return [text];
let lastCharIndex = -1;
for (let i = tokens.length - 1; i >= 0; i -= 1) if (tokens[i]?.kind === "char") {
lastCharIndex = i;
break;
}
const prefixAnsi = tokens.slice(0, firstCharIndex).filter((t) => t.kind === "ansi").map((t) => t.value).join("");
const suffixAnsi = tokens.slice(lastCharIndex + 1).filter((t) => t.kind === "ansi").map((t) => t.value).join("");
const coreTokens = tokens.slice(firstCharIndex, lastCharIndex + 1);
const lines = [];
const isBreakChar = (ch) => ch === " " || ch === " " || ch === "/" || ch === "-" || ch === "_" || ch === ".";
const isSpaceChar = (ch) => ch === " " || ch === " ";
let skipNextLf = false;
const buf = [];
let bufVisible = 0;
let lastBreakIndex = null;
const bufToString = (slice) => (slice ?? buf).map((t) => t.value).join("");
const bufVisibleWidth = (slice) => slice.reduce((acc, t) => acc + (t.kind === "char" ? 1 : 0), 0);
const pushLine = (value) => {
const cleaned = value.replace(/\s+$/, "");
if (cleaned.trim().length === 0) return;
lines.push(cleaned);
};
const flushAt = (breakAt) => {
if (buf.length === 0) return;
if (breakAt == null || breakAt <= 0) {
pushLine(bufToString());
buf.length = 0;
bufVisible = 0;
lastBreakIndex = null;
return;
}
const left = buf.slice(0, breakAt);
const rest = buf.slice(breakAt);
pushLine(bufToString(left));
while (rest.length > 0 && rest[0]?.kind === "char" && isSpaceChar(rest[0].value)) rest.shift();
buf.length = 0;
buf.push(...rest);
bufVisible = bufVisibleWidth(buf);
lastBreakIndex = null;
};
for (const token of coreTokens) {
if (token.kind === "ansi") {
buf.push(token);
continue;
}
const ch = token.value;
if (skipNextLf) {
skipNextLf = false;
if (ch === "\n") continue;
}
if (ch === "\n" || ch === "\r") {
flushAt(buf.length);
if (ch === "\r") skipNextLf = true;
continue;
}
if (bufVisible + 1 > width && bufVisible > 0) flushAt(lastBreakIndex);
buf.push(token);
bufVisible += 1;
if (isBreakChar(ch)) lastBreakIndex = buf.length;
}
flushAt(buf.length);
if (!lines.length) return [""];
if (!prefixAnsi && !suffixAnsi) return lines;
return lines.map((line) => {
if (!line) return line;
return `${prefixAnsi}${line}${suffixAnsi}`;
});
}
function normalizeWidth(n) {
if (n == null) return;
if (!Number.isFinite(n) || n <= 0) return;
return Math.floor(n);
}
function renderTable(opts) {
const rows = opts.rows.map((row) => {
const next = {};
for (const [key, value] of Object.entries(row)) next[key] = displayString(value);
return next;
});
const border = opts.border ?? "unicode";
if (border === "none") {
const columns = opts.columns;
return `${[columns.map((c) => c.header).join(" | "), ...rows.map((r) => columns.map((c) => r[c.key] ?? "").join(" | "))].join("\n")}\n`;
}
const padding = Math.max(0, opts.padding ?? 1);
const columns = opts.columns;
const metrics = columns.map((c) => {
return {
headerW: visibleWidth(c.header),
cellW: Math.max(0, ...rows.map((r) => visibleWidth(r[c.key] ?? "")))
};
});
const widths = columns.map((c, i) => {
const m = metrics[i];
const base = Math.max(m?.headerW ?? 0, m?.cellW ?? 0) + padding * 2;
const capped = c.maxWidth ? Math.min(base, c.maxWidth) : base;
return Math.max(c.minWidth ?? 3, capped);
});
const maxWidth = normalizeWidth(opts.width);
const sepCount = columns.length + 1;
const total = widths.reduce((a, b) => a + b, 0) + sepCount;
const preferredMinWidths = columns.map((c, i) => Math.max(c.minWidth ?? 3, (metrics[i]?.headerW ?? 0) + padding * 2, 3));
const absoluteMinWidths = columns.map((_c, i) => Math.max((metrics[i]?.headerW ?? 0) + padding * 2, 3));
if (maxWidth && total > maxWidth) {
let over = total - maxWidth;
const flexOrder = columns.map((_c, i) => ({
i,
w: widths[i] ?? 0
})).filter(({ i }) => Boolean(columns[i]?.flex)).toSorted((a, b) => b.w - a.w).map((x) => x.i);
const nonFlexOrder = columns.map((_c, i) => ({
i,
w: widths[i] ?? 0
})).filter(({ i }) => !columns[i]?.flex).toSorted((a, b) => b.w - a.w).map((x) => x.i);
const shrink = (order, minWidths) => {
while (over > 0) {
let progressed = false;
for (const i of order) {
if ((widths[i] ?? 0) <= (minWidths[i] ?? 0)) continue;
widths[i] = (widths[i] ?? 0) - 1;
over -= 1;
progressed = true;
if (over <= 0) break;
}
if (!progressed) break;
}
};
shrink(flexOrder, preferredMinWidths);
shrink(flexOrder, absoluteMinWidths);
shrink(nonFlexOrder, preferredMinWidths);
shrink(nonFlexOrder, absoluteMinWidths);
}
if (maxWidth) {
const sepCount = columns.length + 1;
let extra = maxWidth - (widths.reduce((a, b) => a + b, 0) + sepCount);
if (extra > 0) {
const flexCols = columns.map((c, i) => ({
c,
i
})).filter(({ c }) => Boolean(c.flex)).map(({ i }) => i);
if (flexCols.length > 0) {
const caps = columns.map((c) => typeof c.maxWidth === "number" && c.maxWidth > 0 ? Math.floor(c.maxWidth) : Number.POSITIVE_INFINITY);
while (extra > 0) {
let progressed = false;
for (const i of flexCols) {
if ((widths[i] ?? 0) >= (caps[i] ?? Number.POSITIVE_INFINITY)) continue;
widths[i] = (widths[i] ?? 0) + 1;
extra -= 1;
progressed = true;
if (extra <= 0) break;
}
if (!progressed) break;
}
}
}
}
const box = border === "ascii" ? {
tl: "+",
tr: "+",
bl: "+",
br: "+",
h: "-",
v: "|",
t: "+",
ml: "+",
m: "+",
mr: "+",
b: "+"
} : {
tl: "┌",
tr: "┐",
bl: "└",
br: "┘",
h: "─",
v: "│",
t: "┬",
ml: "├",
m: "┼",
mr: "┤",
b: "┴"
};
const hLine = (left, mid, right) => `${left}${widths.map((w) => repeat(box.h, w)).join(mid)}${right}`;
const contentWidthFor = (i) => Math.max(1, widths[i] - padding * 2);
const padStr = repeat(" ", padding);
const renderRow = (record, isHeader = false) => {
const wrapped = columns.map((c) => isHeader ? c.header : record[c.key] ?? "").map((cell, i) => wrapLine(cell, contentWidthFor(i)));
const height = Math.max(...wrapped.map((w) => w.length));
const out = [];
for (let li = 0; li < height; li += 1) {
const parts = wrapped.map((lines, i) => {
return `${padStr}${padCell(lines[li] ?? "", contentWidthFor(i), columns[i]?.align ?? "left")}${padStr}`;
});
out.push(`${box.v}${parts.join(box.v)}${box.v}`);
}
return out;
};
const lines = [];
lines.push(hLine(box.tl, box.t, box.tr));
lines.push(...renderRow({}, true));
lines.push(hLine(box.ml, box.m, box.mr));
for (const row of rows) lines.push(...renderRow(row, false));
lines.push(hLine(box.bl, box.b, box.br));
return `${lines.join("\n")}\n`;
}
//#endregion
export { renderTable as t };