Compare commits

...

5 Commits

Author SHA1 Message Date
93df271f79 Merge pull request '1.0.2' (#9) from 1.0.2 into main
All checks were successful
Create Version Tag / version-check (push) Successful in 22s
Create Version Tag / build-release (push) Successful in 2m58s
Create Version Tag / publish-release (push) Successful in 32s
Reviewed-on: #9
2025-05-20 12:46:29 -07:00
cdeef54f68 fix: local ASCII Art inclusion 2025-05-20 13:45:40 -06:00
90f1547e02 fix: flickering eyebleed 2025-05-20 10:53:37 -06:00
e5b173155a feat: change evaluation now adds case transformation for capture groups 2025-05-20 10:22:49 -06:00
19eaf2d664 feat: field rename multiple files 2025-05-20 09:55:52 -06:00
8 changed files with 187 additions and 93 deletions

View File

@ -30,7 +30,7 @@ jobs:
uses: https://git.cyborggrizzly.com/bearmetal/ci-actions/deno-release@main uses: https://git.cyborggrizzly.com/bearmetal/ci-actions/deno-release@main
with: with:
entrypoint: main.ts entrypoint: main.ts
compile-flags: "--allow-read --allow-write --allow-env --allow-net" compile-flags: "--allow-read --allow-write --allow-env --allow-net --include asciiart.txt"
env: env:
GITEA_TOKEN: ${{ secrets.GIT_PAT }} GITEA_TOKEN: ${{ secrets.GIT_PAT }}

View File

@ -1,6 +1,10 @@
# Changelog # Changelog
## v1.0.1 (2025-07-25) ## v1.0.2 (2025-05-20)
<!-- auto-changelog -->
## v1.0.1 (2025-05-7)
<!-- auto-changelog --> <!-- auto-changelog -->
@ -8,7 +12,7 @@
- help flags can cause issues - help flags can cause issues
## v1.0.0 (2025-07-25) ## v1.0.0 (2025-05-7)
### Features ### Features

View File

@ -20,7 +20,8 @@ Deno >=2.2 (not required if downloading .exe)
### Deno install ### Deno install
`deno task install` -> installs as global command `checkfields` `deno install -g --allow-read --allow-write --allow-net --allow-env jsr:@bearmetal/pdf-tools`
-> installs as global command `pdf-tools`
### Compile ### Compile

View File

@ -111,6 +111,8 @@ export class TerminalBlock {
private renderHeight: number = 0; private renderHeight: number = 0;
private lastRenderRow = 1; private lastRenderRow = 1;
private lastRendered: string[] = [];
private preserveHistory = false; private preserveHistory = false;
constructor(private prepend: string = "") {} constructor(private prepend: string = "") {}
@ -193,27 +195,37 @@ export class TerminalBlock {
} }
renderInternal(startRow?: number) { renderInternal(startRow?: number) {
this.lastRenderRow = startRow ?? this.lastRenderRow; const outputLines: string[] = [];
this.clear(); // uses old renderedLineCount
const outputLines = this.renderLines.map((line) => for (let i = 0; i < this.renderLines.length; i++) {
`${this.prepend}${line}\x1b[K` const line = `${this.prepend}${this.renderLines[i]}`;
); const previous = this.lastRendered[i];
const output = outputLines.join("\n"); if (line !== previous) {
if (startRow !== undefined) { const moveToLine = `\x1b[${(startRow ?? this.lastRenderRow) + i};1H`;
const moveCursor = `\x1b[${startRow};1H`; outputLines.push(moveToLine + line + "\x1b[K");
Deno.stdout.writeSync(new TextEncoder().encode(moveCursor + output)); }
} else {
Deno.stdout.writeSync(new TextEncoder().encode(output));
} }
// update rendered line count *after* rendering if (this.lastRendered.length > this.renderLines.length) {
this.renderedLineCount = outputLines.reduce( const baseRow = startRow ?? this.lastRenderRow;
(count, line) => for (let i = this.renderLines.length; i < this.lastRendered.length; i++) {
count + const moveToLine = `\x1b[${baseRow + i};1H\x1b[2K`;
Math.ceil((line.length) / (Deno.consoleSize().columns || 80)), Deno.stdout.writeSync(new TextEncoder().encode(moveToLine));
0, }
); }
const baseRow = startRow ?? this.lastRenderRow;
const excessLines = this.renderHeight - this.renderLines.length;
for (let i = 0; i < excessLines; i++) {
const moveToLine = `[${baseRow + this.renderLines.length + i};1H`;
Deno.stdout.writeSync(new TextEncoder().encode(moveToLine));
}
this.lastRendered = [...this.renderLines];
this.renderedLineCount = this.renderHeight;
this.lastRenderRow = baseRow;
const output = outputLines.join("\n");
Deno.stdout.writeSync(new TextEncoder().encode(output));
} }
clear() { clear() {

View File

@ -1,10 +1,10 @@
{ {
"name": "@bearmetal/pdf-tools", "name": "@bearmetal/pdf-tools",
"version": "1.0.1", "version": "1.0.2",
"license": "GPL 3.0", "license": "GPL 3.0",
"tasks": { "tasks": {
"dev": "deno run -A --env-file=.env --watch main.ts", "dev": "deno run -A --env-file=.env main.ts",
"compile": "deno compile -o compare-form-fields.exe --target x86_64-pc-windows-msvc -R ./main.ts", "compile": "deno compile -o pdf-tools.exe --target x86_64-pc-windows-msvc --include ./asciiart.txt -A ./main.ts",
"install": "deno install -fgq --import-map ./deno.json -n checkfields -R ./main.ts", "install": "deno install -fgq --import-map ./deno.json -n checkfields -R ./main.ts",
"debug": "deno run -A --env-file=.env --inspect-wait --watch main.ts" "debug": "deno run -A --env-file=.env --inspect-wait --watch main.ts"
}, },
@ -12,7 +12,8 @@
"@std/assert": "jsr:@std/assert@1", "@std/assert": "jsr:@std/assert@1",
"@std/path": "jsr:@std/path@^1.0.9", "@std/path": "jsr:@std/path@^1.0.9",
"pdf-lib": "npm:pdf-lib@^1.17.1", "pdf-lib": "npm:pdf-lib@^1.17.1",
"util/": "./util/" "util/": "./util/",
"@/": "./"
}, },
"exports": { "exports": {
".": "./main.ts" ".": "./main.ts"

View File

@ -7,6 +7,7 @@ import { colorize } from "../cli/style.ts";
import { cliAlert, cliLog, cliPrompt } from "../cli/prompts.ts"; import { cliAlert, cliLog, cliPrompt } from "../cli/prompts.ts";
import { multiSelectMenuInteractive } from "../cli/selectMenu.ts"; 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";
async function renameFields( async function renameFields(
path: string, path: string,
@ -69,13 +70,42 @@ function applyRename(
} }
} }
/***
* Evaluates the change string with the match array
*
* @description The change string can include the following variables:
*
* - $<int> - capture groups, indexed from 1
* - $<int>i - capture groups, indexed from 1, transforming an integer to an index
* - $<int>s - capture groups, indexed from 1, transforming a string to snake case
* - $<int>c - capture groups, indexed from 1, transforming a string to camel case
* - $<int>l - capture groups, indexed from 1, transforming a string to lower 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
*/
function evaluateChange(change: string, match: RegExpExecArray) { function evaluateChange(change: string, match: RegExpExecArray) {
return change.replace( return change.replace(
/\$(\d+)(i?)/g, /\$(\d+)([icslut]?)/g,
(_, i, indexed) => (_, i, indexed) => {
indexed switch (indexed) {
? (parseInt(match[i]) ? (parseInt(match[i]) - 1).toString() : match[i]) case "i":
: match[i], return (parseInt(match[i])
? (parseInt(match[i]) - 1).toString()
: match[i]);
case "s":
return toCase(match[i], "snake");
case "c":
return toCase(match[i], "camel");
case "t":
return toCase(match[i], "title");
case "l":
return match[i].toLowerCase();
case "u":
return match[i].toUpperCase();
default:
return match[i];
}
},
); );
} }
@ -103,50 +133,57 @@ class RenameFields implements ITool {
[pdfPath, pattern, change] = await forceArgs( [pdfPath, pattern, change] = await forceArgs(
[pdfPath, pattern, change], [pdfPath, pattern, change],
[ [
["Please provide path to PDF:", (p) => !!p && p.endsWith(".pdf")], [
"Please provide path to PDF (comma separated for multiple):",
(p) => !!p && p.endsWith(".pdf"),
],
"Please provide search string:", "Please provide search string:",
"Please provide requested change:", "Please provide requested change:",
], ],
this.block, this.block,
); );
const patternRegex = new RegExp(pattern); const paths = pdfPath.split(",");
const pdf = await loadPdf(pdfPath); for (const pdfPath of paths) {
const form = pdf.getForm(); const patternRegex = new RegExp(pattern);
const fields = form.getFields();
const foundUpdates: [string, callback][] = []; const pdf = await loadPdf(pdfPath);
const form = pdf.getForm();
const fields = form.getFields();
for (const field of fields) { const foundUpdates: [string, callback][] = [];
const name = field.getName();
const match = patternRegex.exec(name); for (const field of fields) {
if (match) { const name = field.getName();
const toChange = evaluateChange(change, match); const match = patternRegex.exec(name);
const preview = name.replace(new RegExp(patternRegex), toChange); if (match) {
foundUpdates.push([ const toChange = evaluateChange(change, match);
`${colorize(name, "red")} -> ${colorize(preview, "green")}`, const preview = name.replace(new RegExp(patternRegex), toChange);
() => { foundUpdates.push([
applyRename(field, name, patternRegex, toChange); `${colorize(name, "red")} -> ${colorize(preview, "green")}`,
}, () => {
]); applyRename(field, name, patternRegex, toChange);
},
]);
}
} }
}
if (foundUpdates.length) { if (foundUpdates.length) {
cliLog("Found updates:", this.block); cliLog("Found updates:", this.block);
await multiSelectMenuInteractive( await multiSelectMenuInteractive(
"Please select an option to apply", "Please select an option to apply",
foundUpdates, foundUpdates,
{ terminalBlock: this.block }, { terminalBlock: this.block },
);
}
const path = await cliPrompt(
"Save to path (or hit enter to keep current):",
this.block,
); );
await savePdf(pdf, path || pdfPath);
} }
const path = await cliPrompt(
"Save to path (or hit enter to keep current):",
this.block,
);
await savePdf(pdf, path || pdfPath);
} }
} }
export default new RenameFields(); export default new RenameFields();

View File

@ -1,30 +1,29 @@
export async function getAsciiArt(art: string) { import { log } from "./logfile.ts";
const artFilePath = Deno.env.get("BEARMETAL_ASCII_PATH") || import { join } from "@std/path";
getBearmetalAsciiPath();
if (!artFilePath) return art;
let artFileText: string;
if (artFilePath.startsWith("http")) {
artFileText = await fetch(artFilePath).then((res) => res.text());
} else {
artFileText = await Deno.readTextFile(artFilePath);
}
const parserRX = /begin\s+(\w+)\s*\n([\s\S]*?)\s*end\s*/g;
let result = parserRX.exec(artFileText);
while (result !== null) { export async function getAsciiArt(art: string) {
const [_, name, artText] = result; try {
if (name === art) return artText; const artFilePath = Deno.env.get("BEARMETAL_ASCII_PATH") ||
result = parserRX.exec(artFileText); join(import.meta.dirname || "", "../asciiart.txt");
let artFileText: string;
if (artFilePath?.startsWith("http")) {
artFileText = await fetch(artFilePath).then((res) => res.text());
} else {
artFileText = await Deno.readTextFile(
artFilePath,
);
}
const parserRX = /begin\s+(\w+)\s*\n([\s\S]*?)\s*end\s*/g;
let result = parserRX.exec(artFileText);
while (result !== null) {
const [_, name, artText] = result;
if (name === art) return artText;
result = parserRX.exec(artFileText);
}
} catch (e) {
console.log(e);
alert();
} }
return art; return art;
} }
function getBearmetalAsciiPath() {
const filenameRX = /asciiarts?\.txt$/;
for (const filename of Deno.readDirSync(".")) {
if (filename.isFile && filenameRX.test(filename.name)) {
return filename.name;
}
}
return "https://git.cyborggrizzly.com/BearMetal/pdf-tools/raw/branch/main/asciiart.txt";
}

View File

@ -11,9 +11,45 @@ function lowerToTrainCase(str: string) {
); );
} }
function lowerToCamelCase(str: string) { /**
return str.trim().replace(/(?:\s)\w/g, (match) => match.toUpperCase()) * @param str
.replaceAll(" ", ""); * @returns camelCased string (single letter words are lower cased, e.g. SSN -> ssn)
*/
function lowerToCamelCase(str: string): string {
const words = str.trim().split(/\s+/);
const result: string[] = [];
let i = 0;
while (i < words.length) {
if (words[i].length === 1) {
// Weve hit the start of a chain of single-letter words
let j = i;
while (j < words.length && words[j].length === 1) {
j++;
}
const chainIsAtStart = i === 0;
// Process that entire chain
for (let k = i; k < j; k++) {
result[k] = chainIsAtStart
? words[k].toLowerCase()
: words[k].toUpperCase();
}
i = j;
} else {
// Normal multi-letter word
if (i === 0) {
// first word: all lower
result[i] = words[i].toLowerCase();
} else {
// subsequent words: capitalize first letter
result[i] = words[i][0].toUpperCase() +
words[i].slice(1).toLowerCase();
}
i++;
}
}
return result.join("");
} }
function lowerToSnakeCase(str: string) { function lowerToSnakeCase(str: string) {
@ -88,10 +124,10 @@ function coerceCaseToLower(str: string, caseType: CaseType) {
case "macro": case "macro":
case "snake": case "snake":
case "upper": case "upper":
return str.replace("_", " ").toLowerCase(); return str.replaceAll("_", " ").toLowerCase();
case "train": case "train":
case "kebab": case "kebab":
return str.replace("-", " ").toLowerCase(); return str.replaceAll("-", " ").toLowerCase();
default: default:
return str.toLowerCase(); return str.toLowerCase();
} }
@ -124,3 +160,7 @@ export function toCase(str: string, toCase: CaseType) {
return str; return str;
} }
} }
if (import.meta.main) {
console.log(toCase("SSN", "camel"));
}