change: selects now use inputmanager

fix: bad exit logic
feat: field rename now supports renaming things with multiple widgets
This commit is contained in:
Emmaline Autumn 2025-05-27 12:44:45 -06:00
parent 7a394c642a
commit 0f9c377853
12 changed files with 543 additions and 161 deletions

2
.gitignore vendored
View File

@ -4,3 +4,5 @@
log.txt log.txt
log log
test2.pdf

View File

@ -1,4 +1,5 @@
import { Cursor } from "./cursor.ts"; import { Cursor } from "./cursor.ts";
import { InputManager } from "./InputManager.ts";
export class TerminalLayout { export class TerminalLayout {
private static ALT_BUFFER_ENABLE = "\x1b[?1049h"; private static ALT_BUFFER_ENABLE = "\x1b[?1049h";
@ -22,7 +23,7 @@ export class TerminalLayout {
Deno.addSignalListener("SIGINT", () => { Deno.addSignalListener("SIGINT", () => {
this.clearAll(); this.clearAll();
Deno.exit(0); // Deno.exit(0);
}); });
} }
@ -121,7 +122,8 @@ export class TerminalBlock {
private preserveHistory = false; private preserveHistory = false;
constructor(private prepend: string = "") {} constructor(private prepend: string = "") {
}
setPreserveHistory(preserveHistory: boolean) { setPreserveHistory(preserveHistory: boolean) {
this.preserveHistory = preserveHistory; this.preserveHistory = preserveHistory;

View File

@ -6,6 +6,7 @@ import { selectMenuInteractive } from "./selectMenu.ts";
import { TerminalBlock, TerminalLayout } from "./TerminalLayout.ts"; import { TerminalBlock, TerminalLayout } from "./TerminalLayout.ts";
import { cliAlert, cliLog } from "./prompts.ts"; import { cliAlert, cliLog } from "./prompts.ts";
import type { ITool } from "../types.ts"; import type { ITool } from "../types.ts";
import { InputManager } from "./InputManager.ts";
// Register tools here (filename, no extension) // Register tools here (filename, no extension)
const toolRegistry: [string, Promise<{ default: ITool }>][] = [ const toolRegistry: [string, Promise<{ default: ITool }>][] = [
@ -55,6 +56,12 @@ export class PdfToolsCli {
} }
public async run() { public async run() {
const im = InputManager.getInstance();
im.activate();
im.addEventListener("exit", () => {
this.closeMessage = "Exiting...";
this.cleanup();
});
try { try {
await this.importTools(); await this.importTools();
const titleBlock = new TerminalBlock(); const titleBlock = new TerminalBlock();
@ -78,11 +85,13 @@ export class PdfToolsCli {
} }
} finally { } finally {
this.cleanup(); this.cleanup();
Deno.exit(0);
} }
} }
private cleanup() { private cleanup() {
this.terminalLayout.clearAll(); this.terminalLayout.clearAll();
InputManager.getInstance().deactivate();
Deno.stdin.setRaw(false); Deno.stdin.setRaw(false);
if (this.closeMessage) console.log(this.closeMessage); if (this.closeMessage) console.log(this.closeMessage);
} }

View File

@ -295,7 +295,7 @@ if (import.meta.main) {
Deno.addSignalListener("SIGINT", () => { Deno.addSignalListener("SIGINT", () => {
layout.clearAll(); layout.clearAll();
// console.clear(); // console.clear();
Deno.exit(0); // Deno.exit(0);
}); });
const name = await cliPrompt("Enter your name:", block); const name = await cliPrompt("Enter your name:", block);
cliLog(`Hello, ${name}!`, block); cliLog(`Hello, ${name}!`, block);

View File

@ -1,6 +1,7 @@
import type { callback } from "../types.ts"; import type { callback } from "../types.ts";
import { type CLICharEvent, InputManager } from "./InputManager.ts";
import { colorize } from "./style.ts"; import { colorize } from "./style.ts";
import { TerminalBlock } from "./TerminalLayout.ts"; import { TerminalBlock, TerminalLayout } from "./TerminalLayout.ts";
interface ISelectMenuConfig { interface ISelectMenuConfig {
terminalBlock?: TerminalBlock; terminalBlock?: TerminalBlock;
@ -62,23 +63,37 @@ export async function selectMenuInteractive(
let inputBuffer = ""; let inputBuffer = "";
// Function to handle input const im = InputManager.getInstance();
async function handleInput() { im.activate();
const buf = new Uint8Array(3); // arrow keys send 3 bytes
while (true) { const onUp = (e: Event) => {
e.stopImmediatePropagation();
selected = (selected - 1 + options.length) % options.length;
renderMenu(); renderMenu();
const n = await Deno.stdin.read(buf); };
if (n === null) break;
const [a, b, c] = buf; const onDown = (e: Event) => {
e.stopImmediatePropagation();
selected = (selected + 1) % options.length;
renderMenu();
};
if (a === 3) { const onKey = (e: CLICharEvent) => {
Deno.stdin.setRaw(false); e.stopImmediatePropagation();
terminalBlock?.["layout"]?.clearAll(); const ke = e.detail;
Deno.exit(130); const char = String.fromCharCode(ke.key);
} inputBuffer += char;
};
if (a === 13) { // Enter key const onBackspace = (e: Event) => {
e.stopImmediatePropagation();
inputBuffer = inputBuffer.slice(0, -1);
};
let resolve: null | ((value: string) => void) = null;
const onEnter = (e: Event) => {
e.stopImmediatePropagation();
if (inputBuffer) { if (inputBuffer) {
const parsed = parseInt(inputBuffer); const parsed = parseInt(inputBuffer);
if (!isNaN(parsed)) { if (!isNaN(parsed)) {
@ -86,26 +101,27 @@ export async function selectMenuInteractive(
} }
inputBuffer = ""; inputBuffer = "";
} }
break; im.removeEventListener("arrow-up", onUp);
} else if (a === 27 && b === 91) { // Arrow keys im.removeEventListener("arrow-down", onDown);
inputBuffer = ""; im.removeEventListener("char", onKey);
if (c === 65) { // Up im.removeEventListener("backspace", onBackspace);
selected = (selected - 1 + options.length) % options.length; im.removeEventListener("enter", onEnter);
} else if (c === 66) { // Down resolve?.(options[selected]);
selected = (selected + 1) % options.length; };
}
} else if (a >= 48 && a <= 57) { renderMenu();
inputBuffer += String.fromCharCode(a); await new Promise<string>((res) => {
} else if (a === 8) { resolve = res;
inputBuffer = inputBuffer.slice(0, -1); im.addEventListener("char", onKey);
} im.addEventListener("backspace", onBackspace);
} im.addEventListener("enter", onEnter);
im.addEventListener("arrow-up", onUp);
im.addEventListener("arrow-down", onDown);
});
Deno.stdin.setRaw(false);
return options[selected];
}
terminalBlock.setLines(["Selected: " + options[selected]], range); terminalBlock.setLines(["Selected: " + options[selected]], range);
return await handleInput();
return options[selected];
} }
export async function multiSelectMenuInteractive( export async function multiSelectMenuInteractive(
@ -117,9 +133,7 @@ export async function multiSelectMenuInteractive(
let selected = 0; let selected = 0;
let selectedOptions: number[] = config?.initialSelections || []; let selectedOptions: number[] = config?.initialSelections || [];
const rawValues = new Set( const rawValues = options.map((i) => typeof i === "string" ? i : i[0]);
options.map((i) => typeof i === "string" ? i : i[0]),
).values().toArray();
if (rawValues.length !== options.length) { if (rawValues.length !== options.length) {
throw new Error("Duplicate options in multi-select menu"); throw new Error("Duplicate options in multi-select menu");
@ -158,44 +172,52 @@ export async function multiSelectMenuInteractive(
range = terminalBlock.setLines(lines, range); range = terminalBlock.setLines(lines, range);
} }
// Function to handle input const im = InputManager.getInstance();
async function handleInput() { im.activate();
const buf = new Uint8Array(3); // arrow keys send 3 bytes
while (true) {
renderMenu();
const n = await Deno.stdin.read(buf);
if (n === null) break;
const [a, b, c] = buf; let resolve = null as null | ((value: number[]) => void);
if (a === 3) { const onUp = (e: Event) => {
Deno.stdin.setRaw(false); e.stopImmediatePropagation();
terminalBlock?.["layout"]?.clearAll();
Deno.exit(130);
}
if (a === 13) { // Enter key
break;
} else if (a === 27 && b === 91) { // Arrow keys
if (c === 65) { // Up
selected = (selected - 1 + options.length) % options.length; selected = (selected - 1 + options.length) % options.length;
} else if (c === 66) { // Down renderMenu();
};
const onDown = (e: Event) => {
e.stopImmediatePropagation();
selected = (selected + 1) % options.length; selected = (selected + 1) % options.length;
} renderMenu();
} else if (a === 32) { // Space };
Deno.stdout.writeSync(new TextEncoder().encode("\x07"));
const onSpace = (e: CLICharEvent) => {
if (e.detail.char !== " ") return;
e.stopImmediatePropagation();
if (selectedOptions.includes(selected)) { if (selectedOptions.includes(selected)) {
selectedOptions = selectedOptions.filter((i) => i !== selected); selectedOptions = selectedOptions.filter((i) => i !== selected);
} else { } else {
selectedOptions.push(selected); selectedOptions.push(selected);
} }
} renderMenu();
} };
Deno.stdin.setRaw(false); const onEnter = (e: Event) => {
return selectedOptions; e.stopImmediatePropagation();
} resolve?.(selectedOptions);
const selections = await handleInput(); im.removeEventListener("arrow-up", onUp);
im.removeEventListener("arrow-down", onDown);
im.removeEventListener("char", onSpace);
im.removeEventListener("enter", onEnter);
};
renderMenu();
const selections = await new Promise<number[]>((res) => {
resolve = res;
im.addEventListener("arrow-up", onUp);
im.addEventListener("arrow-down", onDown);
im.addEventListener("char", onSpace);
im.addEventListener("enter", onEnter);
});
for (const optionI of selections) { for (const optionI of selections) {
const option = options[optionI]; const option = options[optionI];
if (Array.isArray(option)) { if (Array.isArray(option)) {
@ -208,17 +230,17 @@ export async function multiSelectMenuInteractive(
} }
if (import.meta.main) { if (import.meta.main) {
// const layout = new TerminalLayout(); const layout = new TerminalLayout();
// const block = new TerminalBlock(); const block = new TerminalBlock();
// const titleBlock = new TerminalBlock(); const titleBlock = new TerminalBlock();
// const postBlock = new TerminalBlock(); const postBlock = new TerminalBlock();
// titleBlock.setLines(["An incredible fruit menu!"]); InputManager.addEventListener("exit", () => layout.clearAll());
// postBlock.setLines(["I'm here too!"]); titleBlock.setLines(["An incredible fruit menu!"]);
// titleBlock.setFixedHeight(1); postBlock.setLines(["I'm here too!"]);
// postBlock.setFixedHeight(1); titleBlock.setFixedHeight(1);
// layout.register("title", titleBlock); postBlock.setFixedHeight(1);
// layout.register("block", block); layout.register("title", titleBlock);
// layout.register("post", postBlock); layout.register("block", block);
// const val = await selectMenuInteractive("choose a fruit", [ // const val = await selectMenuInteractive("choose a fruit", [
// "apple", // "apple",
@ -276,8 +298,8 @@ if (import.meta.main) {
"ximenia", "ximenia",
"yuzu", "yuzu",
"zucchini", "zucchini",
]); ], { terminalBlock: block });
console.log(val); // console.log(val);
// Deno.stdout.writeSync(new TextEncoder().encode("\x07")); // Deno.stdout.writeSync(new TextEncoder().encode("\x07"));
} }

View File

@ -1,6 +1,6 @@
{ {
"name": "@bearmetal/pdf-tools", "name": "@bearmetal/pdf-tools",
"version": "1.0.8-a", "version": "1.0.8-h",
"license": "GPL 3.0", "license": "GPL 3.0",
"tasks": { "tasks": {
"dev": "deno run -A --env-file=.env main.ts", "dev": "deno run -A --env-file=.env main.ts",

View File

@ -1,5 +1,10 @@
/// <reference types="./types.ts" /> /// <reference types="./types.ts" />
import { PdfToolsCli } from "./cli/index.ts"; import { PdfToolsCli } from "./cli/index.ts";
// import { log } from "util/logfile.ts";
// try {
const app = new PdfToolsCli(); const app = new PdfToolsCli();
app.run(); app.run();
// } catch (e) {
// // log(e);
// }

Binary file not shown.

View File

@ -1,6 +1,18 @@
import { type PDFAcroField, type PDFField, PDFName, PDFString } from "pdf-lib"; import {
import { loadPdf, loadPdfForm, savePdf } from "util/saveLoadPdf.ts"; type PDFAcroField,
import { callWithArgPrompt } from "util/call.ts"; PDFArray,
PDFCheckBox,
type PDFDocument,
type PDFField,
PDFName,
PDFNumber,
PDFRadioGroup,
type PDFRef,
PDFString,
PDFTextField,
type PDFWidgetAnnotation,
} from "pdf-lib";
import { loadPdf, savePdf } from "util/saveLoadPdf.ts";
import { TerminalBlock } from "../cli/TerminalLayout.ts"; import { TerminalBlock } from "../cli/TerminalLayout.ts";
import { forceArgs } from "../cli/forceArgs.ts"; import { forceArgs } from "../cli/forceArgs.ts";
import { colorize } from "../cli/style.ts"; import { colorize } from "../cli/style.ts";
@ -9,42 +21,6 @@ import { multiSelectMenuInteractive } from "../cli/selectMenu.ts";
import type { callback, ITool } from "../types.ts"; import type { callback, ITool } from "../types.ts";
import { toCase } from "util/caseManagement.ts"; import { toCase } from "util/caseManagement.ts";
async function renameFields(
path: string,
pattern: string | RegExp,
change: string,
) {
if (typeof pattern === "string") pattern = new RegExp(pattern);
const form = await loadPdfForm(path);
const fields = form.getFields();
let changesMade = false;
for (const field of fields) {
const name = field.getName();
if (pattern.test(name)) {
console.log(name + " %cfound", "color: red");
const segments = name.split(".");
const matchingSegments = segments.filter((s) => pattern.test(s));
let cField: PDFAcroField | undefined = field.acroField;
while (cField) {
if (
cField.getPartialName() &&
matchingSegments.includes(cField.getPartialName()!)
) {
const mName = cField.getPartialName()?.replace(pattern, change);
if (mName) {
changesMade = true;
cField.dict.set(PDFName.of("T"), PDFString.of(mName));
}
}
cField = cField.getParent();
}
}
}
if (changesMade) {
savePdf(form.doc, path);
}
}
function applyRename( function applyRename(
field: PDFField, field: PDFField,
name: string, name: string,
@ -70,6 +46,294 @@ function applyRename(
} }
} }
// function applyWidgetRename(
// doc: PDFDocument,
// field: PDFField,
// widget: PDFWidgetAnnotation,
// name: string,
// pattern: RegExp,
// change: string,
// ) {
// if (field.acroField.getWidgets().length > 1) {
// const widgets = field.acroField.getWidgets();
// const widgetIndex = widgets.indexOf(widget);
// widgets.splice(widgetIndex, 1);
// const pdfDocContext = doc.context;
// const originalRef = field.acroField.ref;
// const originalFieldDict = pdfDocContext.lookup(originalRef);
// if (!originalFieldDict) return;
// const newFieldDict = pdfDocContext.obj({
// ...originalFieldDict,
// T: PDFString.of(name.replace(pattern, change)),
// Kids: [getWidgetRef(widget, doc.getPages())],
// });
// const newField = pdfDocContext.register(newFieldDict);
// const acroForm = doc.catalog.lookup(PDFName.of("AcroForm"), PDFDict);
// const fields = acroForm.lookup(PDFName.of("Fields"), PDFArray);
// fields.push(newField);
// }
// }
function findPageForWidget(
doc: PDFDocument,
widget: PDFWidgetAnnotation,
) {
const pages = doc.getPages();
for (const page of pages) {
const annots = page.node.Annots();
if (!annots) continue;
const annotRefs = annots.asArray();
for (const ref of annotRefs) {
const annot = doc.context.lookup(ref);
if (annot === widget.dict) {
return page;
}
}
}
return undefined;
}
function detectFieldType(field: PDFField): string | undefined {
const ft = field.acroField.dict.get(PDFName.of("FT"));
return ft instanceof PDFName ? ft.asString() : undefined;
}
function getFlag(field: PDFField, bit: number): boolean {
const ff = field.acroField.dict.get(PDFName.of("Ff"));
return ff instanceof PDFNumber ? (ff.asNumber() & (1 << bit)) !== 0 : false;
}
function getWidgetRef(
widget: PDFWidgetAnnotation,
doc: PDFDocument,
): PDFRef | undefined {
for (const page of doc.getPages()) {
const annots = page.node.Annots()?.asArray() ?? [];
for (const ref of annots) {
const maybeDict = doc.context.lookup(ref);
if (maybeDict === widget.dict) {
return ref as PDFRef;
}
}
}
return undefined;
}
function applyWidgetRename(
doc: PDFDocument,
field: PDFField,
widget: PDFWidgetAnnotation,
newName: string,
pattern: RegExp,
change: string,
) {
try {
const form = doc.getForm();
const widgets = field.acroField.getWidgets();
if (widgets.length <= 1) return;
const widgetDict = widget.dict;
const widgetIndex = widgets.findIndex((w) => w.dict === widgetDict);
if (widgetIndex === -1) return;
widgets.splice(widgetIndex, 1);
const kids = field.acroField.dict.lookup(PDFName.of("Kids"), PDFArray);
if (kids) {
const updatedKids = kids.asArray().filter((ref) => {
const maybeDict = doc.context.lookup(ref);
return maybeDict !== widget.dict;
});
field.acroField.dict.set(
PDFName.of("Kids"),
doc.context.obj(updatedKids),
);
}
const page = findPageForWidget(doc, widget);
if (!page) throw new Error("Widget page not found");
const rect = widget.getRectangle();
if (!rect) throw new Error("Widget has no rectangle");
const finalName = newName.replace(pattern, change);
// Try to get existing field with the new name
let targetField: PDFField | undefined;
try {
targetField = form.getField(finalName);
} catch {
// Field doesn't exist — that's fine
}
// Compare field types if field exists
if (targetField) {
const sourceType = detectFieldType(field);
const targetType = detectFieldType(targetField);
if (sourceType !== targetType) {
throw new Error(
`Field "${finalName}" already exists with a different type (${targetType} vs ${sourceType})`,
);
}
// ✅ Same type — attach widget to the existing field
// const targetFieldWidgets = targetField.acroField.getWidgets();
const targetKidsArray = targetField.acroField.dict.lookup(
PDFName.of("Kids"),
PDFArray,
);
// Set /Parent on the widget to point to the existing field
widget.dict.set(PDFName.of("Parent"), targetField.acroField.ref);
// Add the widget to the field's /Kids array
const widgetRef = getWidgetRef(widget, doc);
if (!widgetRef) throw new Error("Widget ref not found");
if (targetKidsArray) {
targetKidsArray.push(widgetRef);
} else {
targetField.acroField.dict.set(
PDFName.of("Kids"),
doc.context.obj([widgetRef]),
);
}
// Also ensure widget is attached to a page
const page = findPageForWidget(doc, widget);
if (!page) throw new Error("Widget's page not found");
const pageAnnots = page.node.Annots();
const refs = pageAnnots?.asArray() ?? [];
if (!refs.includes(widgetRef)) {
refs.push(widgetRef);
page.node.set(PDFName.of("Annots"), doc.context.obj(refs));
}
return; // Done
}
removeWidgetFromPage(widget, doc);
const fieldType = detectFieldType(field);
let newField: PDFField;
switch (fieldType) {
case "/Tx": {
const tf = form.createTextField(finalName);
if (field instanceof PDFTextField) {
const val = field.getText();
if (val) tf.setText(val);
}
newField = tf;
break;
}
case "/Btn": {
const isRadio = getFlag(field, 15);
if (isRadio) {
const rf = form.createRadioGroup(finalName);
rf.addOptionToPage(finalName, page, {
x: rect.x,
y: rect.y,
width: rect.width,
height: rect.height,
});
if (field instanceof PDFRadioGroup) {
const selected = field.getSelected();
if (selected) rf.select(selected);
}
return;
} else {
const cb = form.createCheckBox(finalName);
cb.addToPage(page, {
x: rect.x,
y: rect.y,
width: rect.width,
height: rect.height,
});
if (field instanceof PDFCheckBox && field.isChecked()) {
cb.check();
}
return;
}
}
default:
throw new Error(`Unsupported field type: ${fieldType}`);
}
// Attach the new field to the page if necessary
if (
newField instanceof PDFTextField ||
newField instanceof PDFCheckBox
) {
newField.addToPage(page, {
x: rect.x,
y: rect.y,
width: rect.width,
height: rect.height,
});
}
} catch {
// log(e);
}
}
function removeWidgetFromPage(widget: PDFWidgetAnnotation, doc: PDFDocument) {
const pages = doc.getPages();
for (const page of pages) {
const annotsArray = page.node.Annots();
if (!annotsArray) continue;
const refs = annotsArray.asArray();
const newRefs = refs.filter((ref) => {
const maybeDict = doc.context.lookup(ref);
return maybeDict !== widget.dict;
});
// Replace /Annots with updated array
if (newRefs.length === refs.length) continue;
page.node.set(PDFName.of("Annots"), doc.context.obj(newRefs));
}
}
// function getWidgetRef(
// widget: PDFWidgetAnnotation,
// pages: PDFPage[],
// ): PDFRef | undefined {
// const widgetRect = (widget?.dict?.get(PDFName.of("Rect")) as PDFArray)
// ?.asArray();
// const widgetFT = (widget?.dict?.get(PDFName.of("FT")) as PDFString)
// ?.["value"];
// for (const page of pages) {
// const annotsArray = page.node.Annots()?.asArray();
// if (!annotsArray) continue;
// for (const annotRef of annotsArray) {
// const annotDict = page.doc.context.lookup(annotRef);
// if (!annotDict) continue;
// if (!(annotDict instanceof PDFDict)) continue;
// const rect = (annotDict.get(PDFName.of("Rect")) as PDFArray)?.asArray();
// const ft = (annotDict.get(PDFName.of("FT")) as PDFString)?.["value"];
// // rudimentary match (you can add more checks like /T, /Subtype, etc.)
// if (rect?.toString() === widgetRect?.toString() && ft === widgetFT) {
// return annotRef as PDFRef;
// }
// }
// }
// return undefined;
// }
/*** /***
* Evaluates the change string with the match array * Evaluates the change string with the match array
* *
@ -83,7 +347,7 @@ function applyRename(
* - $<int>u - capture groups, indexed from 1, transforming a string to upper case * - $<int>u - capture groups, indexed from 1, transforming a string to upper case
* - $<int>t - capture groups, indexed from 1, transforming a string to title case * - $<int>t - capture groups, indexed from 1, transforming a string to title case
*/ */
function evaluateChange(change: string, match: RegExpExecArray) { function evaluateChange(change: string, match: RegExpExecArray, index: number) {
return change.replace( return change.replace(
/\$(\d+)([icslut]?)/g, /\$(\d+)([icslut]?)/g,
(_, i, indexed) => { (_, i, indexed) => {
@ -106,6 +370,18 @@ function evaluateChange(change: string, match: RegExpExecArray) {
return match[i]; return match[i];
} }
}, },
)
.replace(
/\$I{((\w+,?)+)}/,
(_, offset) => {
const options = offset.split(",");
return options[index % options.length];
},
)
.replace(
/\$I(-?\d+)?/,
(_, offset) =>
(parseInt(offset) ? index + parseInt(offset) : index).toString(),
); );
} }
@ -150,29 +426,92 @@ class RenameFields implements ITool {
const pdf = await loadPdf(pdfPath); const pdf = await loadPdf(pdfPath);
const form = pdf.getForm(); const form = pdf.getForm();
const fields = form.getFields(); const fields = form.getFields().sort((a, b) => {
const aWidgets = a.acroField.getWidgets();
const bWidgets = b.acroField.getWidgets();
const aWidget = aWidgets[0];
const bWidget = bWidgets[0];
const aPage = a.doc.findPageForAnnotationRef(a.acroField.ref);
const bPage = b.doc.findPageForAnnotationRef(b.acroField.ref);
if (aPage && bPage && aPage !== bPage) {
const pages = a.doc.getPages();
const aPageIndex = pages.indexOf(aPage);
const bPageIndex = pages.indexOf(bPage);
if (aPageIndex !== bPageIndex) return aPageIndex - bPageIndex;
}
const aRect = aWidget.Rect()?.asRectangle();
const bRect = bWidget.Rect()?.asRectangle();
if (aRect && bRect) {
const dy = bRect.y - aRect.y;
if (Math.abs(dy) > 5) return dy;
return aRect.x - bRect.x;
}
return a.getName().localeCompare(b.getName());
});
let badFields = 0;
for (const field of fields) {
if (field.acroField.getWidgets().length > 1) {
badFields++;
}
}
badFields && await cliLog(
colorize(
`Warning, ${badFields} fields with shared widgets found`,
"yellow",
),
this.block,
);
const foundUpdates: [string, callback][] = []; const foundUpdates: [string, callback][] = [];
let changesMade = false; let changesMade = false;
let i = 0;
for (const field of fields) { for (const field of fields) {
const name = field.getName(); const name = field.getName();
const match = patternRegex.exec(name); const match = patternRegex.exec(name);
if (match) { if (match) {
const toChange = evaluateChange(change, match); foundUpdates.push(
const preview = name.replace(new RegExp(patternRegex), toChange); ...field.acroField.getWidgets()?.map<[string, callback]>((
foundUpdates.push([ widget,
) => {
const toChange = evaluateChange(change, match, i);
const preview = name.replace(
new RegExp(patternRegex),
toChange,
);
i++;
return [
`${colorize(name, "red")} -> ${colorize(preview, "green")}`, `${colorize(name, "red")} -> ${colorize(preview, "green")}`,
() => { () => {
applyRename(field, name, patternRegex, toChange); field.acroField.getWidgets().length > 1
? applyWidgetRename(
pdf,
field,
widget,
name,
new RegExp(patternRegex),
toChange,
)
: applyRename(field, name, patternRegex, toChange);
changesMade = true; changesMade = true;
}, },
]); ];
}),
);
} }
} }
if (foundUpdates.length) { if (foundUpdates.length) {
cliLog("Found updates:", this.block); await cliLog("Found updates:", this.block);
await multiSelectMenuInteractive( await multiSelectMenuInteractive(
"Please select an option to apply", "Please select an option to apply",
foundUpdates, foundUpdates,
@ -185,7 +524,11 @@ class RenameFields implements ITool {
"Save to path (or hit enter to keep current):", "Save to path (or hit enter to keep current):",
this.block, this.block,
); );
try {
await savePdf(pdf, path || pdfPath); await savePdf(pdf, path || pdfPath);
} catch {
// log(e);
}
} else { } else {
cliLog("No changes made, skipping", this.block); cliLog("No changes made, skipping", this.block);
} }
@ -194,14 +537,14 @@ class RenameFields implements ITool {
} }
export default new RenameFields(); export default new RenameFields();
if (import.meta.main) { // if (import.meta.main) {
// await call(renameFields) // // await call(renameFields)
// while (!path || !path.endsWith('.pdf')) path = prompt("Please provide path to PDF:") || ''; // // while (!path || !path.endsWith('.pdf')) path = prompt("Please provide path to PDF:") || '';
// while (!pattern) pattern = prompt("Please provide search string:") || ''; // // while (!pattern) pattern = prompt("Please provide search string:") || '';
// while (!change) change = prompt("Please provide requested change:") || ''; // // while (!change) change = prompt("Please provide requested change:") || '';
await callWithArgPrompt(renameFields, [ // await callWithArgPrompt(renameFields, [
["Please provide path to PDF:", (p) => !!p && p.endsWith(".pdf")], // ["Please provide path to PDF:", (p) => !!p && p.endsWith(".pdf")],
"Please provide search string:", // "Please provide search string:",
"Please provide requested change:", // "Please provide requested change:",
]); // ]);
} // }

View File

@ -1,4 +1,3 @@
import { log } from "./logfile.ts";
import { join } from "@std/path"; import { join } from "@std/path";
export async function getAsciiArt(art: string) { export async function getAsciiArt(art: string) {

View File

@ -9,7 +9,7 @@ logFile.truncateSync(0);
export function log(message: any) { export function log(message: any) {
if (typeof message === "object") { if (typeof message === "object") {
message = JSON.stringify(message); message = Deno.inspect(message);
} }
logFile.writeSync(new TextEncoder().encode(message + "\n")); logFile.writeSync(new TextEncoder().encode(message + "\n"));
} }

View File

@ -15,10 +15,10 @@ export async function loadPdf(path: string) {
export async function savePdf(doc: PDFDocument, path: string) { export async function savePdf(doc: PDFDocument, path: string) {
doc.getForm().getFields().forEach((field) => { doc.getForm().getFields().forEach((field) => {
if (field instanceof PDFTextField) { if (field instanceof PDFTextField) {
field.disableRichFormatting(); field.disableRichFormatting?.();
} }
}); });
const pdfBytes = await doc.save(); const pdfBytes = await doc.save({ updateFieldAppearances: true });
if (Deno.env.get("DRYRUN") || path.includes("dryrun")) return; if (Deno.env.get("DRYRUN") || path.includes("dryrun")) return;
await Deno.writeFile(path, pdfBytes); await Deno.writeFile(path, pdfBytes);
} }