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 +149 -0
- package/assets/charify.svg +12 -0
- package/bin/charify.js +205 -0
- package/package.json +36 -0
package/README.md
ADDED
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+

|
|
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
|
+
[](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
|
+
}
|