charify 1.0.0 → 2.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/LICENSE +21 -0
- package/README.md +155 -68
- package/package.json +14 -9
- package/src/charify.js +65 -0
- package/src/cli.js +79 -0
- package/src/constants.js +13 -0
- package/src/errors.js +8 -0
- package/src/index.js +7 -0
- package/src/utils.js +38 -0
- package/assets/charify.svg +0 -12
- package/bin/charify.js +0 -205
package/LICENSE
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) 2025 CrazyBrain Labs
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
+
in the Software without restriction, including without limitation the rights
|
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
+
furnished to do so, subject to the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
|
13
|
+
copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
+
SOFTWARE.
|
package/README.md
CHANGED
|
@@ -1,146 +1,233 @@
|
|
|
1
1
|

|
|
2
2
|
|
|
3
|
-
|
|
4
|
-
|
|
3
|
+
# charify
|
|
4
|
+
|
|
5
|
+
High-quality image to ASCII art converter – CLI tool and npm library.
|
|
6
|
+
|
|
7
|
+
Powered by `sharp` for excellent image processing. No external Unix tools required.
|
|
5
8
|
|
|
6
9
|
---
|
|
7
10
|
|
|
8
11
|
## Install
|
|
9
12
|
|
|
10
|
-
Global install:
|
|
13
|
+
Global install (recommended for CLI usage):
|
|
11
14
|
|
|
12
|
-
```
|
|
15
|
+
```batch
|
|
13
16
|
npm install -g charify
|
|
14
17
|
```
|
|
15
18
|
|
|
16
19
|
Run without installing:
|
|
17
20
|
|
|
18
|
-
```
|
|
21
|
+
```batch
|
|
19
22
|
npx charify image.png
|
|
20
23
|
```
|
|
21
24
|
|
|
25
|
+
Install as a library in your project:
|
|
26
|
+
|
|
27
|
+
```batch
|
|
28
|
+
npm install charify
|
|
29
|
+
```
|
|
30
|
+
|
|
22
31
|
---
|
|
23
32
|
|
|
24
|
-
## Usage
|
|
33
|
+
## CLI Usage
|
|
25
34
|
|
|
26
|
-
```
|
|
27
|
-
charify image
|
|
28
|
-
charify image
|
|
29
|
-
charify image.png -o output.txt
|
|
30
|
-
charify image.png --html -o output.html
|
|
31
|
-
charify --link https://example.com/image.jpg
|
|
35
|
+
```batch
|
|
36
|
+
charify <image> [options]
|
|
37
|
+
charify --url <image-url> [options]
|
|
32
38
|
```
|
|
33
39
|
|
|
34
|
-
|
|
40
|
+
## Examples
|
|
35
41
|
|
|
36
|
-
|
|
42
|
+
### Basic conversion (prints to terminal)
|
|
37
43
|
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
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 |
|
|
44
|
+
```batch
|
|
45
|
+
charify photo.jpg
|
|
46
|
+
```
|
|
49
47
|
|
|
50
|
-
|
|
48
|
+
### Wider output
|
|
51
49
|
|
|
52
|
-
|
|
50
|
+
```batch
|
|
51
|
+
charify photo.jpg -w 140
|
|
52
|
+
```
|
|
53
53
|
|
|
54
|
-
|
|
55
|
-
| ------- | ---------------------- |
|
|
56
|
-
| dense | Best quality (default) |
|
|
57
|
-
| light | Softer output |
|
|
58
|
-
| blocks | Block-style ASCII |
|
|
59
|
-
| minimal | Very simple |
|
|
54
|
+
### Save to file
|
|
60
55
|
|
|
61
|
-
|
|
56
|
+
```batch
|
|
57
|
+
charify photo.jpg -o art.txt
|
|
58
|
+
```
|
|
59
|
+
|
|
60
|
+
### Export as standalone HTML
|
|
62
61
|
|
|
63
|
-
```
|
|
64
|
-
charify
|
|
62
|
+
```batch
|
|
63
|
+
charify photo.jpg --html -o art.html
|
|
65
64
|
```
|
|
66
65
|
|
|
67
|
-
|
|
66
|
+
### From URL
|
|
68
67
|
|
|
69
|
-
|
|
68
|
+
```batch
|
|
69
|
+
charify --url https://example.com/photo.jpg --preset blocks -o blocks.html
|
|
70
|
+
```
|
|
70
71
|
|
|
71
|
-
|
|
72
|
+
### Invert colors with custom preset
|
|
72
73
|
|
|
73
|
-
```
|
|
74
|
-
charify
|
|
74
|
+
```batch
|
|
75
|
+
charify photo.jpg --invert --preset light
|
|
75
76
|
```
|
|
76
77
|
|
|
78
|
+
### CLI Options
|
|
79
|
+
|
|
80
|
+
| Option | Description |
|
|
81
|
+
| --------------------- | ---------------------------------------------------------- |
|
|
82
|
+
| `<image>` | Path to local image file |
|
|
83
|
+
| `-u, --url <url>` | Fetch image from URL |
|
|
84
|
+
| `-w, --width <n>` | Output width in characters (default: 100) |
|
|
85
|
+
| `-r, --ratio <n>` | Height-to-width ratio for character aspect (default: 0.55) |
|
|
86
|
+
| `-g, --gamma <n>` | Gamma correction (default: 2.2) |
|
|
87
|
+
| `-p, --preset <name>` | Preset charset: dense (default), light, blocks, minimal |
|
|
88
|
+
| `-c, --charset <str>` | Custom character ramp (e.g. " .:-=+\*#%@") |
|
|
89
|
+
| `-i, --invert` | Invert brightness mapping |
|
|
90
|
+
| `--html` | Output as standalone HTML page |
|
|
91
|
+
| `-o, --output <file>` | Write result to file (.txt for plain, .html for HTML) |
|
|
92
|
+
|
|
93
|
+
Note: --html is automatically enabled if the output filename ends with .html.
|
|
94
|
+
|
|
77
95
|
---
|
|
78
96
|
|
|
79
|
-
##
|
|
97
|
+
## Library Usage
|
|
80
98
|
|
|
81
|
-
|
|
99
|
+
Import and use directly in Node.js projects:
|
|
82
100
|
|
|
83
|
-
```
|
|
84
|
-
charify
|
|
101
|
+
```js
|
|
102
|
+
import charify, { PRESETS, wrapAsHtml } from "charify";
|
|
103
|
+
import fs from "fs";
|
|
104
|
+
|
|
105
|
+
const buffer = fs.readFileSync("photo.jpg");
|
|
106
|
+
|
|
107
|
+
const ascii = await charify(buffer, {
|
|
108
|
+
width: 140,
|
|
109
|
+
preset: "blocks",
|
|
110
|
+
invert: true,
|
|
111
|
+
gamma: 2.0,
|
|
112
|
+
ratio: 0.6,
|
|
113
|
+
});
|
|
114
|
+
|
|
115
|
+
console.log(ascii);
|
|
116
|
+
|
|
117
|
+
// Or generate HTML manually
|
|
118
|
+
const html = wrapAsHtml(ascii);
|
|
119
|
+
fs.writeFileSync("art.html", html);
|
|
85
120
|
```
|
|
86
121
|
|
|
87
|
-
|
|
122
|
+
### Exported API
|
|
123
|
+
|
|
124
|
+
- `charify(buffer, options?)` – main function (default export)
|
|
125
|
+
- `PRESETS` – object with built-in charsets (dense, light, blocks, minimal)
|
|
126
|
+
- `wrapAsHtml(asciiString)` – utility to wrap plain ASCII in a minimal standalone HTML page
|
|
127
|
+
|
|
128
|
+
Requires sharp at runtime (peer dependency). Install it in your project:
|
|
129
|
+
|
|
130
|
+
```batch
|
|
131
|
+
npm install sharp
|
|
132
|
+
```
|
|
88
133
|
|
|
89
134
|
---
|
|
90
135
|
|
|
91
|
-
##
|
|
136
|
+
## Presets
|
|
137
|
+
|
|
138
|
+
| Preset | Charset | Best for |
|
|
139
|
+
| ------- | -------------- | ---------------------------- |
|
|
140
|
+
| dense | █▓▒░@%#\*+=-:. | Highest detail (default) |
|
|
141
|
+
| light | #\*+=-:. | Softer, less dense output |
|
|
142
|
+
| blocks | █▓▒░ | Classic blocky ASCII style |
|
|
143
|
+
| minimal | #.+ | Extremely simple, clean look |
|
|
92
144
|
|
|
93
|
-
|
|
145
|
+
Example:
|
|
94
146
|
|
|
95
|
-
```
|
|
96
|
-
charify --
|
|
147
|
+
```batch
|
|
148
|
+
charify photo.jpg --preset blocks
|
|
97
149
|
```
|
|
98
150
|
|
|
99
|
-
|
|
151
|
+
### Custom Charset
|
|
152
|
+
|
|
153
|
+
Override with your own ramp (dark → bright):
|
|
100
154
|
|
|
101
|
-
```
|
|
102
|
-
charify
|
|
155
|
+
```batch
|
|
156
|
+
charify photo.jpg --charset " .:-=+\*#%@"
|
|
103
157
|
```
|
|
104
158
|
|
|
105
159
|
---
|
|
106
160
|
|
|
107
|
-
##
|
|
161
|
+
## HTML Export
|
|
162
|
+
|
|
163
|
+
Creates a ready-to-open HTML file with green-on-black terminal styling:
|
|
164
|
+
|
|
165
|
+
```batch
|
|
166
|
+
charify photo.jpg --html -o my-art.html
|
|
167
|
+
```
|
|
108
168
|
|
|
109
|
-
|
|
169
|
+
# or simply
|
|
110
170
|
|
|
111
|
-
```
|
|
112
|
-
charify
|
|
171
|
+
```batch
|
|
172
|
+
charify photo.jpg -o my-art.html # auto-detects .html extension
|
|
113
173
|
```
|
|
114
174
|
|
|
115
175
|
---
|
|
116
176
|
|
|
117
|
-
##
|
|
177
|
+
## Browser Demo
|
|
118
178
|
|
|
119
|
-
|
|
179
|
+
Try charify instantly in the browser!
|
|
120
180
|
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
181
|
+
Open `demo/index.html` (included in the package):
|
|
182
|
+
|
|
183
|
+
- Drag & drop images
|
|
184
|
+
- Live preview
|
|
185
|
+
- Adjustable width, preset, invert
|
|
186
|
+
- Download as HTML
|
|
187
|
+
|
|
188
|
+
No build step required – uses Tailwind CSS via CDN.
|
|
124
189
|
|
|
125
190
|
---
|
|
126
191
|
|
|
127
192
|
## Local Development
|
|
128
193
|
|
|
129
|
-
|
|
194
|
+
```batch
|
|
195
|
+
git clone https://github.com/abdodev/charify.git
|
|
196
|
+
cd charify
|
|
197
|
+
npm install
|
|
198
|
+
```
|
|
199
|
+
|
|
200
|
+
# Run CLI directly
|
|
201
|
+
|
|
202
|
+
```batch
|
|
203
|
+
node src/cli.js image.png
|
|
204
|
+
```
|
|
205
|
+
|
|
206
|
+
# Test library
|
|
130
207
|
|
|
131
|
-
```
|
|
132
|
-
node
|
|
208
|
+
```batch
|
|
209
|
+
node -e "import { charify } from './src/index.js'; /_ your test code _/"
|
|
133
210
|
```
|
|
134
211
|
|
|
135
212
|
---
|
|
136
213
|
|
|
214
|
+
## ⭐ Star on GitHub
|
|
215
|
+
|
|
216
|
+
If you like charify, please star the repo!
|
|
217
|
+
|
|
218
|
+
[](https://github.com/abdodev/charify)
|
|
219
|
+
|
|
220
|
+
It helps others discover the project.
|
|
221
|
+
|
|
222
|
+
---
|
|
223
|
+
|
|
137
224
|
## Donate
|
|
138
225
|
|
|
139
|
-
If
|
|
226
|
+
If charify saves you time or brings you joy, consider supporting development:
|
|
140
227
|
|
|
141
|
-
[](https://paypal.me/Abdoelsayd81)
|
|
228
|
+
[](https://paypal.me/Abdoelsayd81)
|
|
142
229
|
|
|
143
|
-
Thank you
|
|
230
|
+
Thank you ❤️
|
|
144
231
|
|
|
145
232
|
---
|
|
146
233
|
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "charify",
|
|
3
|
-
"version": "
|
|
3
|
+
"version": "2.0.0",
|
|
4
4
|
"description": "High-quality image to ASCII art CLI",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"ascii",
|
|
@@ -18,19 +18,24 @@
|
|
|
18
18
|
"license": "MIT",
|
|
19
19
|
"author": "AbdoDev",
|
|
20
20
|
"type": "module",
|
|
21
|
-
"main": "/bin/charify.js",
|
|
22
21
|
"bin": {
|
|
23
|
-
"charify": "
|
|
22
|
+
"charify": "./src/cli.js"
|
|
24
23
|
},
|
|
24
|
+
"files": [
|
|
25
|
+
"src"
|
|
26
|
+
],
|
|
25
27
|
"scripts": {
|
|
26
|
-
"
|
|
28
|
+
"start": "node src/cli.js"
|
|
27
29
|
},
|
|
28
30
|
"dependencies": {
|
|
29
|
-
"commander": "^12.
|
|
30
|
-
|
|
31
|
+
"commander": "^12.1.0"
|
|
32
|
+
},
|
|
33
|
+
"peerDependencies": {
|
|
34
|
+
"sharp": "^0.33.5"
|
|
31
35
|
},
|
|
32
|
-
"
|
|
33
|
-
|
|
34
|
-
|
|
36
|
+
"peerDependenciesMeta": {
|
|
37
|
+
"sharp": {
|
|
38
|
+
"optional": true
|
|
39
|
+
}
|
|
35
40
|
}
|
|
36
41
|
}
|
package/src/charify.js
ADDED
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
import sharp from "sharp";
|
|
2
|
+
import { PRESETS, DEFAULTS } from "./constants.js";
|
|
3
|
+
import { SharpMissingError } from "./errors.js";
|
|
4
|
+
|
|
5
|
+
export async function convertToAscii(buffer, options = {}) {
|
|
6
|
+
if (!sharp) {
|
|
7
|
+
throw new SharpMissingError();
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
const {
|
|
11
|
+
width = DEFAULTS.width,
|
|
12
|
+
ratio = DEFAULTS.ratio,
|
|
13
|
+
gamma = DEFAULTS.gamma,
|
|
14
|
+
preset = DEFAULTS.preset,
|
|
15
|
+
charset,
|
|
16
|
+
invert = false,
|
|
17
|
+
} = options;
|
|
18
|
+
|
|
19
|
+
const chars = charset || PRESETS[preset] || PRESETS.dense;
|
|
20
|
+
const rampLength = chars.length - 1;
|
|
21
|
+
|
|
22
|
+
let image = sharp(buffer)
|
|
23
|
+
.resize({ width, fit: "inside", withoutEnlargement: true })
|
|
24
|
+
.grayscale();
|
|
25
|
+
|
|
26
|
+
const { height: originalHeight } = await image.metadata();
|
|
27
|
+
const targetHeight = Math.round(originalHeight * ratio);
|
|
28
|
+
|
|
29
|
+
image = image.resize(width, targetHeight);
|
|
30
|
+
|
|
31
|
+
const { data, info } = await image
|
|
32
|
+
.raw()
|
|
33
|
+
.toBuffer({ resolveWithObject: true });
|
|
34
|
+
|
|
35
|
+
// Normalize contrast
|
|
36
|
+
let min = 255;
|
|
37
|
+
let max = 0;
|
|
38
|
+
for (const value of data) {
|
|
39
|
+
if (value < min) min = value;
|
|
40
|
+
if (value > max) max = value;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
const range = max - min || 1;
|
|
44
|
+
const normalized = Buffer.from(
|
|
45
|
+
data.map((v) => Math.round(((v - min) / range) * 255))
|
|
46
|
+
);
|
|
47
|
+
|
|
48
|
+
// Build ASCII
|
|
49
|
+
let result = "";
|
|
50
|
+
for (let y = 0; y < info.height; y++) {
|
|
51
|
+
for (let x = 0; x < info.width; x++) {
|
|
52
|
+
const idx = y * info.width + x;
|
|
53
|
+
let intensity = normalized[idx];
|
|
54
|
+
intensity = Math.pow(intensity / 255, 1 / gamma) * 255;
|
|
55
|
+
|
|
56
|
+
const charIndex = Math.floor((intensity / 255) * rampLength);
|
|
57
|
+
const finalIndex = invert ? rampLength - charIndex : charIndex;
|
|
58
|
+
|
|
59
|
+
result += chars[finalIndex];
|
|
60
|
+
}
|
|
61
|
+
result += "\n";
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
return result;
|
|
65
|
+
}
|
package/src/cli.js
ADDED
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
import { Command } from "commander";
|
|
4
|
+
import fs from "fs";
|
|
5
|
+
import path from "path";
|
|
6
|
+
import { charify, PRESETS, wrapAsHtml } from "./index.js";
|
|
7
|
+
import { fetchImage } from "./utils.js";
|
|
8
|
+
|
|
9
|
+
const program = new Command();
|
|
10
|
+
|
|
11
|
+
program
|
|
12
|
+
.name("charify")
|
|
13
|
+
.description("Convert images to ASCII art")
|
|
14
|
+
.argument("[image]", "path to image file")
|
|
15
|
+
.option("-u, --url <url>", "fetch image from URL")
|
|
16
|
+
.option("-w, --width <number>", "output width (characters)", parseInt)
|
|
17
|
+
.option("-r, --ratio <number>", "height scaling ratio", parseFloat)
|
|
18
|
+
.option("-g, --gamma <number>", "gamma correction", parseFloat)
|
|
19
|
+
.option(
|
|
20
|
+
"-p, --preset <name>",
|
|
21
|
+
"character preset: dense, light, blocks, minimal"
|
|
22
|
+
)
|
|
23
|
+
.option("-c, --charset <string>", "custom character ramp")
|
|
24
|
+
.option("-i, --invert", "invert brightness")
|
|
25
|
+
.option("--html", "output as standalone HTML")
|
|
26
|
+
.option("-o, --output <file>", "write to file instead of stdout")
|
|
27
|
+
.addHelpText(
|
|
28
|
+
"before",
|
|
29
|
+
`
|
|
30
|
+
_ _ __
|
|
31
|
+
___| |__ __ _ _ __(_)/ _|_ _
|
|
32
|
+
/ __| '_ \\ / _' | '__| | |_| | | |
|
|
33
|
+
| (__| | | | (_| | | | | _| |_| |
|
|
34
|
+
\\___|_| |_|\\__,_|_| |_|_| \\__, |
|
|
35
|
+
|___/
|
|
36
|
+
`
|
|
37
|
+
);
|
|
38
|
+
|
|
39
|
+
program.parse();
|
|
40
|
+
|
|
41
|
+
const opts = program.opts();
|
|
42
|
+
const inputPath = program.args[0];
|
|
43
|
+
|
|
44
|
+
if (!inputPath && !opts.url) {
|
|
45
|
+
console.error("Error: provide an image path or --url");
|
|
46
|
+
process.exit(1);
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
(async () => {
|
|
50
|
+
try {
|
|
51
|
+
const buffer = opts.url
|
|
52
|
+
? await fetchImage(opts.url)
|
|
53
|
+
: fs.readFileSync(path.resolve(inputPath));
|
|
54
|
+
|
|
55
|
+
const ascii = await charify(buffer, {
|
|
56
|
+
width: opts.width,
|
|
57
|
+
ratio: opts.ratio,
|
|
58
|
+
gamma: opts.gamma,
|
|
59
|
+
preset: opts.preset,
|
|
60
|
+
charset: opts.charset,
|
|
61
|
+
invert: opts.invert,
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
const output =
|
|
65
|
+
opts.html || (opts.output && opts.output.endsWith(".html"))
|
|
66
|
+
? wrapAsHtml(ascii)
|
|
67
|
+
: ascii;
|
|
68
|
+
|
|
69
|
+
if (opts.output) {
|
|
70
|
+
fs.writeFileSync(opts.output, output, "utf8");
|
|
71
|
+
console.log(`Saved to ${opts.output}`);
|
|
72
|
+
} else {
|
|
73
|
+
process.stdout.write(output);
|
|
74
|
+
}
|
|
75
|
+
} catch (err) {
|
|
76
|
+
console.error(`Error: ${err.message}`);
|
|
77
|
+
process.exit(1);
|
|
78
|
+
}
|
|
79
|
+
})();
|
package/src/constants.js
ADDED
package/src/errors.js
ADDED
package/src/index.js
ADDED
package/src/utils.js
ADDED
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import https from "https";
|
|
2
|
+
|
|
3
|
+
export function fetchImage(url) {
|
|
4
|
+
return new Promise((resolve, reject) => {
|
|
5
|
+
https
|
|
6
|
+
.get(url, (res) => {
|
|
7
|
+
if (res.statusCode !== 200) {
|
|
8
|
+
reject(new Error(`HTTP ${res.statusCode}`));
|
|
9
|
+
return;
|
|
10
|
+
}
|
|
11
|
+
const chunks = [];
|
|
12
|
+
res.on("data", (chunk) => chunks.push(chunk));
|
|
13
|
+
res.on("end", () => resolve(Buffer.concat(chunks)));
|
|
14
|
+
})
|
|
15
|
+
.on("error", reject);
|
|
16
|
+
});
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export function wrapAsHtml(ascii) {
|
|
20
|
+
return `<!DOCTYPE html>
|
|
21
|
+
<html lang="en">
|
|
22
|
+
<head>
|
|
23
|
+
<meta charset="utf-8">
|
|
24
|
+
<title>ASCII Art</title>
|
|
25
|
+
<style>
|
|
26
|
+
body {
|
|
27
|
+
margin: 20px;
|
|
28
|
+
background: #000;
|
|
29
|
+
color: #0f0;
|
|
30
|
+
font-family: monospace;
|
|
31
|
+
white-space: pre;
|
|
32
|
+
line-height: 1;
|
|
33
|
+
}
|
|
34
|
+
</style>
|
|
35
|
+
</head>
|
|
36
|
+
<body>${ascii}</body>
|
|
37
|
+
</html>`;
|
|
38
|
+
}
|
package/assets/charify.svg
DELETED
|
@@ -1,12 +0,0 @@
|
|
|
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
DELETED
|
@@ -1,205 +0,0 @@
|
|
|
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
|
-
})();
|