charify 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md ADDED
@@ -0,0 +1,149 @@
1
+ ![charify logo](./assets/charify.svg)
2
+
3
+ High-quality image to ASCII art converter for the command line.
4
+ npm-ready. No dependencies on Unix tools.
5
+
6
+ ---
7
+
8
+ ## Install
9
+
10
+ Global install:
11
+
12
+ ```bash
13
+ npm install -g charify
14
+ ```
15
+
16
+ Run without installing:
17
+
18
+ ```bash
19
+ npx charify image.png
20
+ ```
21
+
22
+ ---
23
+
24
+ ## Usage
25
+
26
+ ```bash
27
+ charify image.png
28
+ charify image.jpg -w 120
29
+ charify image.png -o output.txt
30
+ charify image.png --html -o output.html
31
+ charify --link https://example.com/image.jpg
32
+ ```
33
+
34
+ ---
35
+
36
+ ## Options
37
+
38
+ | Option | Description |
39
+ | ------------ | ---------------------------------------------------------- |
40
+ | -w, --width | Output width (default: 100) |
41
+ | --ratio | Height/width ratio (default: 0.55) |
42
+ | --gamma | Gamma correction (default: 2.2) |
43
+ | --invert | Invert brightness |
44
+ | --html | Export as HTML (only works if output file ends with .html) |
45
+ | -o, --output | Output file (.txt or .html) |
46
+ | --link | Load image from URL |
47
+ | --charset | Custom ASCII charset |
48
+ | --preset | Use a preset: dense, light, blocks, minimal |
49
+
50
+ ---
51
+
52
+ ## Presets
53
+
54
+ | Name | Description |
55
+ | ------- | ---------------------- |
56
+ | dense | Best quality (default) |
57
+ | light | Softer output |
58
+ | blocks | Block-style ASCII |
59
+ | minimal | Very simple |
60
+
61
+ Example:
62
+
63
+ ```bash
64
+ charify image.png --preset blocks
65
+ ```
66
+
67
+ ---
68
+
69
+ ## Custom Charset
70
+
71
+ Specify your own charset to control ASCII output characters:
72
+
73
+ ```bash
74
+ charify image.png --charset " .:-=+*#%@"
75
+ ```
76
+
77
+ ---
78
+
79
+ ## HTML Export
80
+
81
+ To export ASCII art wrapped in an HTML page, use `--html` **with** an output file ending in `.html`:
82
+
83
+ ```bash
84
+ charify image.png --html -o art.html
85
+ ```
86
+
87
+ If you use `--html` without specifying an output file ending with `.html`, it will be ignored and plain ASCII will be output instead.
88
+
89
+ ---
90
+
91
+ ## Input from URL
92
+
93
+ Load image from the internet:
94
+
95
+ ```bash
96
+ charify --link https://example.com/image.jpg
97
+ ```
98
+
99
+ You can combine with other options like `--html` and `-o`:
100
+
101
+ ```bash
102
+ charify --link https://example.com/image.jpg --html -o output.html
103
+ ```
104
+
105
+ ---
106
+
107
+ ## Quality Tuning
108
+
109
+ Adjust gamma correction and character aspect ratio:
110
+
111
+ ```bash
112
+ charify image.png --gamma 2.0 --ratio 0.6
113
+ ```
114
+
115
+ ---
116
+
117
+ ## Invert Brightness
118
+
119
+ Invert the brightness mapping:
120
+
121
+ ```bash
122
+ charify image.png --invert
123
+ ```
124
+
125
+ ---
126
+
127
+ ## Local Development
128
+
129
+ Run directly without installing:
130
+
131
+ ```bash
132
+ node bin/charify.js image.png
133
+ ```
134
+
135
+ ---
136
+
137
+ ## Donate
138
+
139
+ If you find **charify** useful and want to support development, you can donate any amount here:
140
+
141
+ [![Donate](https://img.shields.io/badge/Donate-00457C?logo=paypal)](https://paypal.me/Abdoelsayd81)
142
+
143
+ Thank you for your support!
144
+
145
+ ---
146
+
147
+ ## License
148
+
149
+ MIT
@@ -0,0 +1,12 @@
1
+ <?xml version="1.0" encoding="UTF-8" standalone="no"?>
2
+ <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
3
+ <svg width="100%" height="100%" viewBox="0 0 512 127" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" xml:space="preserve" xmlns:serif="http://www.serif.com/" style="fill-rule:evenodd;clip-rule:evenodd;stroke-linejoin:round;stroke-miterlimit:2;">
4
+ <g transform="matrix(1,0,0,1,0,-6.755875)">
5
+ <g transform="matrix(1,-0,-0,1,-0,6.755875)">
6
+ <use xlink:href="#_Image1" x="0" y="0" width="512px" height="127px"/>
7
+ </g>
8
+ </g>
9
+ <defs>
10
+ <image id="_Image1" width="512px" height="127px" xlink:href=""/>
11
+ </defs>
12
+ </svg>
package/bin/charify.js ADDED
@@ -0,0 +1,205 @@
1
+ #!/usr/bin/env node
2
+
3
+ import fs from "fs";
4
+ import path from "path";
5
+ import https from "https";
6
+ import sharp from "sharp";
7
+ import { Command } from "commander";
8
+
9
+ /* ================= CONFIG ================= */
10
+
11
+ const PRESETS = {
12
+ dense: "█▓▒░@%#*+=-:. ",
13
+ light: "#*+=-:. ",
14
+ blocks: "█▓▒░ ",
15
+ minimal: "#.+ ",
16
+ };
17
+
18
+ const DEFAULT_PRESET = "dense";
19
+ const DEFAULT_WIDTH = 100;
20
+ const DEFAULT_RATIO = 0.55;
21
+ const DEFAULT_GAMMA = 2.2;
22
+
23
+ /* ================= CLI ================= */
24
+
25
+ const program = new Command();
26
+
27
+ program
28
+ .name("charify")
29
+ .description("Convert images to ASCII art")
30
+ .argument("[image]", "local image path")
31
+ .option("--link <url>", "image URL")
32
+ .option("-w, --width <number>", "output width", String(DEFAULT_WIDTH))
33
+ .option("--ratio <number>", "height/width ratio", String(DEFAULT_RATIO))
34
+ .option("--gamma <number>", "gamma correction", String(DEFAULT_GAMMA))
35
+ .option("--charset <chars>", "custom ASCII charset")
36
+ .option("--preset <name>", "preset: dense | light | blocks | minimal")
37
+ .option("--invert", "invert brightness")
38
+ .option(
39
+ "--html",
40
+ "export as HTML (only works if output file ends with .html)"
41
+ )
42
+ .option("-o, --output <file>", "output file (.txt or .html)");
43
+ program
44
+ .addHelpText(
45
+ "before",
46
+ `
47
+ _ _ __
48
+ ___| |__ __ _ _ __(_)/ _|_ _
49
+ / __| '_ \\ / _' | '__| | |_| | | |
50
+ | (__| | | | (_| | | | | _| |_| |
51
+ \\___|_| |_|\\__,_|_| |_|_| \\__, |
52
+ |___/
53
+ `
54
+ )
55
+ .parse(process.argv);
56
+
57
+ const opts = program.opts();
58
+ const inputPath = program.args[0];
59
+
60
+ if (!opts.link && !inputPath) {
61
+ console.error("❌ Provide an image path or --link URL");
62
+ process.exit(1);
63
+ }
64
+
65
+ /* ================= HELPERS ================= */
66
+
67
+ function fetchImage(url) {
68
+ return new Promise((resolve, reject) => {
69
+ https.get(url, (res) => {
70
+ const chunks = [];
71
+ res.on("data", (d) => chunks.push(d));
72
+ res.on("end", () => resolve(Buffer.concat(chunks)));
73
+ res.on("error", reject);
74
+ });
75
+ });
76
+ }
77
+
78
+ function gammaCorrect(v, gamma) {
79
+ return Math.pow(v / 255, 1 / gamma) * 255;
80
+ }
81
+
82
+ function normalizeBuffer(buffer) {
83
+ let min = 255,
84
+ max = 0;
85
+ for (const v of buffer) {
86
+ if (v < min) min = v;
87
+ if (v > max) max = v;
88
+ }
89
+ if (max === min) return buffer;
90
+
91
+ return Buffer.from(
92
+ buffer.map((v) => Math.round(((v - min) / (max - min)) * 255))
93
+ );
94
+ }
95
+
96
+ function toAscii(buffer, width, charset, invert, gamma) {
97
+ let out = "";
98
+ const len = charset.length - 1;
99
+
100
+ for (let i = 0; i < buffer.length; i += width) {
101
+ for (let x = 0; x < width; x++) {
102
+ let v = buffer[i + x];
103
+ v = gammaCorrect(v, gamma);
104
+ const idx = Math.floor((v / 255) * len);
105
+ out += invert ? charset[len - idx] : charset[idx];
106
+ }
107
+ out += "\n";
108
+ }
109
+ return out;
110
+ }
111
+
112
+ function wrapHtml(ascii) {
113
+ return `<!doctype html>
114
+ <html>
115
+ <head>
116
+ <meta charset="utf-8">
117
+ <title>charify</title>
118
+ <style>
119
+ body {
120
+ background: #000;
121
+ color: #0f0;
122
+ font-family: monospace;
123
+ white-space: pre;
124
+ line-height: 1;
125
+ }
126
+ </style>
127
+ </head>
128
+ <body>${ascii}</body>
129
+ </html>`;
130
+ }
131
+
132
+ /* ================= MAIN ================= */
133
+
134
+ (async () => {
135
+ try {
136
+ const width = Number(opts.width);
137
+ const ratio = Number(opts.ratio);
138
+ const gamma = Number(opts.gamma);
139
+
140
+ const charset =
141
+ opts.charset || PRESETS[opts.preset] || PRESETS[DEFAULT_PRESET];
142
+
143
+ const inputBuffer = opts.link
144
+ ? await fetchImage(opts.link)
145
+ : fs.readFileSync(path.resolve(inputPath));
146
+
147
+ let sharpInstance = sharp(inputBuffer)
148
+ .resize({
149
+ width,
150
+ fit: sharp.fit.inside,
151
+ withoutEnlargement: true,
152
+ })
153
+ .grayscale();
154
+
155
+ const metadata = await sharpInstance.metadata();
156
+
157
+ const adjustedHeight = Math.round(metadata.height * ratio);
158
+
159
+ sharpInstance = sharpInstance.resize(width, adjustedHeight);
160
+
161
+ const { data, info } = await sharpInstance
162
+ .raw()
163
+ .toBuffer({ resolveWithObject: true });
164
+
165
+ const normalizedData = normalizeBuffer(data);
166
+
167
+ const ascii = toAscii(
168
+ normalizedData,
169
+ info.width,
170
+ charset,
171
+ opts.invert,
172
+ gamma
173
+ );
174
+
175
+ let output = ascii;
176
+ const wantHtml =
177
+ opts.html && opts.output && opts.output.toLowerCase().endsWith(".html");
178
+
179
+ if (wantHtml) {
180
+ output = wrapHtml(ascii);
181
+ } else if (
182
+ opts.html &&
183
+ (!opts.output || !opts.output.toLowerCase().endsWith(".html"))
184
+ ) {
185
+ if (!opts.output) {
186
+ output = ascii;
187
+ } else {
188
+ console.warn(
189
+ "⚠ Warning: --html ignored because output file extension is not .html"
190
+ );
191
+ output = ascii;
192
+ }
193
+ }
194
+
195
+ if (opts.output) {
196
+ fs.writeFileSync(opts.output, output, "utf8");
197
+ console.log("✔ Saved:", opts.output);
198
+ } else {
199
+ process.stdout.write(output);
200
+ }
201
+ } catch (err) {
202
+ console.error("❌ Error:", err.message);
203
+ process.exit(1);
204
+ }
205
+ })();
package/package.json ADDED
@@ -0,0 +1,36 @@
1
+ {
2
+ "name": "charify",
3
+ "version": "1.0.0",
4
+ "description": "High-quality image to ASCII art CLI",
5
+ "keywords": [
6
+ "ascii",
7
+ "ascii",
8
+ "art"
9
+ ],
10
+ "homepage": "https://github.com/abdodev/charify#readme",
11
+ "bugs": {
12
+ "url": "https://github.com/abdodev/charify/issues"
13
+ },
14
+ "repository": {
15
+ "type": "git",
16
+ "url": "git+https://github.com/abdodev/charify.git"
17
+ },
18
+ "license": "MIT",
19
+ "author": "AbdoDev",
20
+ "type": "module",
21
+ "main": "/bin/charify.js",
22
+ "bin": {
23
+ "charify": "bin/charify.js"
24
+ },
25
+ "scripts": {
26
+ "test": "echo \"Error: no test specified\" && exit 1"
27
+ },
28
+ "dependencies": {
29
+ "commander": "^12.0.0",
30
+ "sharp": "^0.33.0"
31
+ },
32
+ "devDependencies": {},
33
+ "engines": {
34
+ "node": ">=18"
35
+ }
36
+ }