Compare commits
5 Commits
9573291582
...
93df271f79
Author | SHA1 | Date | |
---|---|---|---|
93df271f79 | |||
cdeef54f68 | |||
90f1547e02 | |||
e5b173155a | |||
19eaf2d664 |
@ -30,7 +30,7 @@ jobs:
|
||||
uses: https://git.cyborggrizzly.com/bearmetal/ci-actions/deno-release@main
|
||||
with:
|
||||
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:
|
||||
GITEA_TOKEN: ${{ secrets.GIT_PAT }}
|
||||
|
||||
|
@ -1,6 +1,10 @@
|
||||
# Changelog
|
||||
|
||||
## v1.0.1 (2025-07-25)
|
||||
## v1.0.2 (2025-05-20)
|
||||
|
||||
<!-- auto-changelog -->
|
||||
|
||||
## v1.0.1 (2025-05-7)
|
||||
|
||||
<!-- auto-changelog -->
|
||||
|
||||
@ -8,7 +12,7 @@
|
||||
|
||||
- help flags can cause issues
|
||||
|
||||
## v1.0.0 (2025-07-25)
|
||||
## v1.0.0 (2025-05-7)
|
||||
|
||||
### Features
|
||||
|
||||
|
@ -20,7 +20,8 @@ Deno >=2.2 (not required if downloading .exe)
|
||||
|
||||
### 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
|
||||
|
||||
|
@ -111,6 +111,8 @@ export class TerminalBlock {
|
||||
private renderHeight: number = 0;
|
||||
private lastRenderRow = 1;
|
||||
|
||||
private lastRendered: string[] = [];
|
||||
|
||||
private preserveHistory = false;
|
||||
|
||||
constructor(private prepend: string = "") {}
|
||||
@ -193,27 +195,37 @@ export class TerminalBlock {
|
||||
}
|
||||
|
||||
renderInternal(startRow?: number) {
|
||||
this.lastRenderRow = startRow ?? this.lastRenderRow;
|
||||
this.clear(); // uses old renderedLineCount
|
||||
const outputLines: string[] = [];
|
||||
|
||||
const outputLines = this.renderLines.map((line) =>
|
||||
`${this.prepend}${line}\x1b[K`
|
||||
);
|
||||
const output = outputLines.join("\n");
|
||||
if (startRow !== undefined) {
|
||||
const moveCursor = `\x1b[${startRow};1H`;
|
||||
Deno.stdout.writeSync(new TextEncoder().encode(moveCursor + output));
|
||||
} else {
|
||||
Deno.stdout.writeSync(new TextEncoder().encode(output));
|
||||
for (let i = 0; i < this.renderLines.length; i++) {
|
||||
const line = `${this.prepend}${this.renderLines[i]}`;
|
||||
const previous = this.lastRendered[i];
|
||||
if (line !== previous) {
|
||||
const moveToLine = `\x1b[${(startRow ?? this.lastRenderRow) + i};1H`;
|
||||
outputLines.push(moveToLine + line + "\x1b[K");
|
||||
}
|
||||
}
|
||||
|
||||
// update rendered line count *after* rendering
|
||||
this.renderedLineCount = outputLines.reduce(
|
||||
(count, line) =>
|
||||
count +
|
||||
Math.ceil((line.length) / (Deno.consoleSize().columns || 80)),
|
||||
0,
|
||||
);
|
||||
if (this.lastRendered.length > this.renderLines.length) {
|
||||
const baseRow = startRow ?? this.lastRenderRow;
|
||||
for (let i = this.renderLines.length; i < this.lastRendered.length; i++) {
|
||||
const moveToLine = `\x1b[${baseRow + i};1H\x1b[2K`;
|
||||
Deno.stdout.writeSync(new TextEncoder().encode(moveToLine));
|
||||
}
|
||||
}
|
||||
|
||||
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[2K`;
|
||||
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() {
|
||||
|
@ -1,10 +1,10 @@
|
||||
{
|
||||
"name": "@bearmetal/pdf-tools",
|
||||
"version": "1.0.1",
|
||||
"version": "1.0.2",
|
||||
"license": "GPL 3.0",
|
||||
"tasks": {
|
||||
"dev": "deno run -A --env-file=.env --watch main.ts",
|
||||
"compile": "deno compile -o compare-form-fields.exe --target x86_64-pc-windows-msvc -R ./main.ts",
|
||||
"dev": "deno run -A --env-file=.env 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",
|
||||
"debug": "deno run -A --env-file=.env --inspect-wait --watch main.ts"
|
||||
},
|
||||
@ -12,7 +12,8 @@
|
||||
"@std/assert": "jsr:@std/assert@1",
|
||||
"@std/path": "jsr:@std/path@^1.0.9",
|
||||
"pdf-lib": "npm:pdf-lib@^1.17.1",
|
||||
"util/": "./util/"
|
||||
"util/": "./util/",
|
||||
"@/": "./"
|
||||
},
|
||||
"exports": {
|
||||
".": "./main.ts"
|
||||
|
@ -7,6 +7,7 @@ import { colorize } from "../cli/style.ts";
|
||||
import { cliAlert, cliLog, cliPrompt } from "../cli/prompts.ts";
|
||||
import { multiSelectMenuInteractive } from "../cli/selectMenu.ts";
|
||||
import type { callback, ITool } from "../types.ts";
|
||||
import { toCase } from "util/caseManagement.ts";
|
||||
|
||||
async function renameFields(
|
||||
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) {
|
||||
return change.replace(
|
||||
/\$(\d+)(i?)/g,
|
||||
(_, i, indexed) =>
|
||||
indexed
|
||||
? (parseInt(match[i]) ? (parseInt(match[i]) - 1).toString() : match[i])
|
||||
: match[i],
|
||||
/\$(\d+)([icslut]?)/g,
|
||||
(_, i, indexed) => {
|
||||
switch (indexed) {
|
||||
case "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,13 +133,19 @@ class RenameFields implements ITool {
|
||||
[pdfPath, pattern, change] = await forceArgs(
|
||||
[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 requested change:",
|
||||
],
|
||||
this.block,
|
||||
);
|
||||
|
||||
const paths = pdfPath.split(",");
|
||||
|
||||
for (const pdfPath of paths) {
|
||||
const patternRegex = new RegExp(pattern);
|
||||
|
||||
const pdf = await loadPdf(pdfPath);
|
||||
@ -149,6 +185,7 @@ class RenameFields implements ITool {
|
||||
await savePdf(pdf, path || pdfPath);
|
||||
}
|
||||
}
|
||||
}
|
||||
export default new RenameFields();
|
||||
|
||||
if (import.meta.main) {
|
||||
|
@ -1,12 +1,17 @@
|
||||
import { log } from "./logfile.ts";
|
||||
import { join } from "@std/path";
|
||||
|
||||
export async function getAsciiArt(art: string) {
|
||||
try {
|
||||
const artFilePath = Deno.env.get("BEARMETAL_ASCII_PATH") ||
|
||||
getBearmetalAsciiPath();
|
||||
if (!artFilePath) return art;
|
||||
join(import.meta.dirname || "", "../asciiart.txt");
|
||||
let artFileText: string;
|
||||
if (artFilePath.startsWith("http")) {
|
||||
if (artFilePath?.startsWith("http")) {
|
||||
artFileText = await fetch(artFilePath).then((res) => res.text());
|
||||
} else {
|
||||
artFileText = await Deno.readTextFile(artFilePath);
|
||||
artFileText = await Deno.readTextFile(
|
||||
artFilePath,
|
||||
);
|
||||
}
|
||||
const parserRX = /begin\s+(\w+)\s*\n([\s\S]*?)\s*end\s*/g;
|
||||
let result = parserRX.exec(artFileText);
|
||||
@ -16,15 +21,9 @@ export async function getAsciiArt(art: string) {
|
||||
if (name === art) return artText;
|
||||
result = parserRX.exec(artFileText);
|
||||
}
|
||||
} catch (e) {
|
||||
console.log(e);
|
||||
alert();
|
||||
}
|
||||
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";
|
||||
}
|
||||
|
@ -11,9 +11,45 @@ function lowerToTrainCase(str: string) {
|
||||
);
|
||||
}
|
||||
|
||||
function lowerToCamelCase(str: string) {
|
||||
return str.trim().replace(/(?:\s)\w/g, (match) => match.toUpperCase())
|
||||
.replaceAll(" ", "");
|
||||
/**
|
||||
* @param str
|
||||
* @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) {
|
||||
// We’ve 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) {
|
||||
@ -88,10 +124,10 @@ function coerceCaseToLower(str: string, caseType: CaseType) {
|
||||
case "macro":
|
||||
case "snake":
|
||||
case "upper":
|
||||
return str.replace("_", " ").toLowerCase();
|
||||
return str.replaceAll("_", " ").toLowerCase();
|
||||
case "train":
|
||||
case "kebab":
|
||||
return str.replace("-", " ").toLowerCase();
|
||||
return str.replaceAll("-", " ").toLowerCase();
|
||||
default:
|
||||
return str.toLowerCase();
|
||||
}
|
||||
@ -124,3 +160,7 @@ export function toCase(str: string, toCase: CaseType) {
|
||||
return str;
|
||||
}
|
||||
}
|
||||
|
||||
if (import.meta.main) {
|
||||
console.log(toCase("SSN", "camel"));
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user