Compare commits

...

12 Commits

Author SHA1 Message Date
waveplate
3577cf95a2
fix formatting 2024-12-16 17:04:38 -08:00
waveplate
148f809623
add braille output example 2024-12-15 21:05:40 -08:00
Waveplate
ed8866abe2 update link to latest release 2024-12-15 18:49:01 -08:00
Waveplate
fb61052438 fix quarterblock rendering bug 2024-12-15 14:19:19 -08:00
Waveplate
2dfac15db6 update README 2024-12-15 07:35:24 -08:00
Waveplate
f146a613e6 update README 2024-12-15 07:33:48 -08:00
Waveplate
7cadbf8aad apply luma-only effects 2024-12-15 07:33:48 -08:00
Waveplate
bb0ba0e990 add image transformations including crop, rotate, flip horizontal/vertical, filter, scale, and aspect ratio 2024-12-15 07:33:48 -08:00
Waveplate
34f2a1ef13 improve quarterblock rendering and add braille output mode with luma-only color manipulation 2024-12-15 07:33:48 -08:00
Waveplate
703000e68a add new CLI options for crop, rotate, flip, filter, scale, and aspect ratio 2024-12-15 07:33:48 -08:00
Waveplate
492079af6c bump version to 1.1.0 and update dependencies 2024-12-15 07:33:48 -08:00
Waveplate
4c1d23f365 remove unnecessary .cargo/config.toml file 2024-12-15 07:33:48 -08:00
7 changed files with 1110 additions and 446 deletions

View File

@ -1,3 +0,0 @@
[target.x86_64-unknown-linux-musl]
linker = "musl-gcc"

View File

@ -1,6 +1,6 @@
[package]
name = "img2irc-rs"
version = "1.0.6"
version = "1.1.0"
authors = ["waveplate"]
github = "https://github.com/waveplate/img2irc"
repository = "https://github.com/waveplate/img2irc"
@ -15,16 +15,21 @@ codegen-units = 1
opt-level = "z"
strip = true # Automatically strip symbols from the binary.
[target.x86_64-unknown-linux-musl]
linker = "musl-gcc"
[[bin]]
name = "img2irc"
path = "src/main.rs"
[dependencies]
tokio = { version = "1", features = ["rt", "rt-multi-thread", "macros"] }
reqwest = { version = "0.11.14", default-features = false, features = ["rustls-tls"] }
reqwest = "0.11.14"
photon-rs = { version = "0.3.2", default-features = false, git = "https://github.com/silvia-odwyer/photon", rev = "3b72d357848cd76be9363e87ad0cd02a19b988d2" }
#photon-rs = "0.3.2"
clap = { version = "4.2.0", features = ["cargo", "derive"] }
url = "2.3.1"
atty = "0.2.14"
[build-dependencies]
cc = "1.0"
[target.'cfg(target_env = "musl")'.dependencies]
openssl = { version = "0.10", features = ["vendored"] }

155
README.md
View File

@ -1,13 +1,9 @@
# img2irc (1.0.6)
![img2irc preview](https://i.imgur.com/oetHhMB.png)
# img2irc (1.1.0)
img2irc is a utility which converts images to half or quarterblock irc/ansi art, with a lot of post-processing filters
![img2irc braille example](https://i.imgur.com/ZEJwuOb.png)
![img2irc preview](https://i.imgur.com/0omljq5.png)
*halfblock* means that each row will contain two rows worth of pixels, effectively doubling the vertical resolution
*quarterblock* (experimental) means that each row will contain two rows worth of pixels, and each column will contain two columns worth of pixels, quadrupling the resolution
the `irc` mode has 99 colours, the `ansi` mode has 256, `ansi24` has 16777216
*img2irc* is a premiere command-line utility which converts images to irc/ansi art, with a lot of post-processing filters
# how to install
@ -16,18 +12,18 @@ the `irc` mode has 99 colours, the `ansi` mode has 256, `ansi24` has 16777216
statically linked with musl, works on all x86_64 linux platforms
cd /tmp
wget https://github.com/waveplate/img2irc/releases/download/v1.0.4/img2irc-1.0.4-linux-x86_64.tar.gz
sudo tar -xzf img2irc-1.0.4-linux-x86_64.tar.gz -C /usr/local/bin --strip-components=1 img2irc-1.0.4/img2irc
rm -rf img2irc-1.0.4-linux-x86_64.tar.gz
wget https://github.com/waveplate/img2irc/releases/download/v1.1.0/img2irc-1.1.0-linux-x86_64.tar.gz
sudo tar -xzf img2irc-1.1.0-linux-x86_64.tar.gz -C /usr/local/bin --strip-components=1 img2irc-1.1.0/img2irc
rm -rf img2irc-1.1.0-linux-x86_64.tar.gz
- ### install with `yay` (arch linux)
>[!NOTE]
>if you like this project, i would appreciate you giving it a vote on the [aur](https://aur.archlinux.org/packages/img2irc)!
yay -S img2irc
> [!NOTE]
> if you like this project, i would appreciate you giving it a vote on the [aur](https://aur.archlinux.org/packages/img2irc)!
- ### install with `cargo`
cargo install img2irc-rs
@ -38,46 +34,91 @@ the `irc` mode has 99 colours, the `ansi` mode has 256, `ansi24` has 16777216
`img2irc <URL or PATH> [OPTIONS]`
| option | description | default value |
| ------ | ----------- | ------------- |
| `<IMAGE>` | image url or file path | none |
| `--irc` | irc render type | true |
| `--ansi` | 8-bit ansi render type | false |
| `--ansi24` | 24-bit ansi render type | false |
| `--qb` | use quarterblocks (experimental) | false |
| `-w, --width <WIDTH>` | output image width in columns | 50 |
| `-b, --brightness=<BRIGHTNESS>` | adjust brightness (-255 to 255) | 0 |
| `-c, --contrast=<CONTRAST>` | adjust contrast (-255 to 255) | 0 |
| `-s, --saturation=<SATURATION>` | adjust saturation (-255 to 255) | 0 |
| `-H, --hue <HUE>` | rotate hue (0 to 360) | 0 |
| `-g, --gamma <GAMMA>` | adjust gamma (0 to 255) | 0 |
| `--dither <DITHER>` | dithering (1 to 8) | 0 |
| `--pixelize <PIXELIZE>` | pixelize pixel size | 0 |
| `--gaussian-blur <GAUSSIAN_BLUR>` | gaussian blur radius | 0 |
| `--oil <OIL>` | oil ("[RADIUS],[INTENSITY]") | |
| `--grayscale` | converts image to black and white |
| `--nograyscale` | exclude grayscale colours from the palette |
| `--halftone` | made up of small dots creating a continuous-tone illusion |
| `--sepia` | brownish, aged appearance like old photographs |
| `--normalize` | adjusts brightness and contrast for better image quality |
| `--noise` | random variations in brightness and color like film grain |
| `--emboss` | gives a raised, 3d appearance |
| `--box-blur` | smoothed appearance like frosted glass |
| `--identity` | no modifications, unchanged image |
| `--laplace` | enhances edges and boundaries in an image |
| `--noise-reduction` | reduces noise for a cleaner, clearer image |
| `--sharpen` | increases clarity and definition, making edges and details more distinct |
| `--cali` | cool blue tone with increased contrast |
| `--dramatic` | high contrast and vivid colors for a dramatic effect |
| `--firenze` | warm, earthy tones reminiscent of tuscan landscapes |
| `--golden` | warm, golden glow like sunset light |
| `--lix` | high-contrast black and white appearance with increased sharpness |
| `--lofi` | low-fidelity, retro appearance like old photographs or film |
| `--neue` | clean, modern appearance with neutral colors and simple design |
| `--obsidian` | dark, monochromatic appearance with black and gray shades |
| `--pastel-pink` | soft, delicate pink tint like pastel colors |
| `--ryo` | bright, high-contrast appearance with vivid colors and sharp details |
| `--invert` | colors are inverted, opposite on the color wheel |
| `--frosted-glass` | blurred, frosted appearance as if viewed through semi-transparent surface |
| `--solarize` | strange, otherworldly appearance with inverted colors and surreal atmosphere |
| `--edge-detection` | highlights edges and boundaries in an image |
| option | description | default value |
|----------------------------------------|---------------------------------------------------------------|---------------|
| image | image url or file path | required |
| -w, --width | output image width in columns | auto |
| -h, --height | output image height in rows | auto |
| --scale | scaling factors (x:y, e.g., "2:2") | none |
| --aspect | final aspect ratio (x:y, e.g., "2:1") | none |
| --crop | crop image ("x1,y1,x2,y2") | none |
| --filter | sampling filter | nearest |
| --rotate | rotate degrees | 0 |
| --fliph | flip horizontal | false |
| --flipv | flip vertical | false |
### colours rendering options (select one)
`irc` mode has 99 colours, (6.62-bit)
`ansi` mode has 256 colours (8-bit)
`ansi24` has 16777216 colours (24-bit)
| option | description |
|----------------------------------------|---------------------------------------------------------------|
| --irc | use irc99 colours |
| --ansi | use 8-bit ansi colours |
| --ansi24 | use 24-bit ansi colours |
### pixel rendering options (select one)
`halfblock` mode increases the vertical resolution, doubling the total resolution for a given size
`quarterblock` mode increases both the vertical and horizontal resolution by twofold, quadrupling the total resolution for a given size
`braille` mode uses 2x4 dot patterns to represent pixels, increasing resolution eightfold
| option | description |
|----------------------------------------|---------------------------------------------------------------|
| --braille | use braille pixels |
| --hb, --halfblock | use halfblock pixels |
| --qb, --quarterblock | use quarterblocks pixels |
### image processing options
| option | description | default value |
|----------------------------------------|---------------------------------------------------------------|---------------|
| -b, --brightness | adjust brightness (0 = no change) | 0.0 |
| -c, --contrast | adjust contrast (0 = no change) | 0.0 |
| -g, --gamma | adjust gamma (0 to 255) | 0.0 |
| -s, --saturation | adjust saturation (0 = no change) | 0.0 |
| -u, --hue | rotate hue (0 to 360) | 0.0 |
| -i, --invert | colors are inverted, opposite on the color wheel | false |
| -d, --dither | dithering (1 to 8) | 0 |
| -B, --luma-brightness | adjust luma brightness (braille only) | 0.0 |
| -C, --luma-contrast | adjust luma contrast (braille only) | 0.0 |
| -G, --luma-gamma | adjust luma gamma (braille only) | 0.0 |
| -S, --luma-saturation | adjust luma saturation (braille only) | 0.0 |
| -I, --luma-invert | luminance is inverted (braille only) | false |
| --colorspace | colourspace (hsl, hsv, hsluv, lch) | hsv |
| --grayscale | converts image to black and white | false |
| --nograyscale | exclude grayscale colours from the palette | false |
| --pixelize | pixelize pixel size | 0 |
| --boxblur | simple average of all the neighboring pixels surrounding one | false |
| --gaussianblur | gaussian blur radius | 0 |
| --oil | oil filter ("[radius],[intensity]") | none |
| --halftone | made up of small dots creating a continuous-tone illusion | false |
| --sepia | brownish, aged appearance like old photographs | false |
| --normalize | adjusts brightness and contrast for better image quality | false |
| --noise | random variations in brightness and color like film grain | false |
| --emboss | gives a raised, 3d appearance | false |
| --identity | no modifications, unchanged image | false |
| --laplace | enhances edges and boundaries in an image | false |
| --denoise | reduces noise for a cleaner, clearer image | false |
| --sharpen | increases clarity and definition, making edges and details more distinct | false |
| --cali | cool blue tone with increased contrast | false |
| --dramatic | high contrast and vivid colors for a dramatic effect | false |
| --firenze | warm, earthy tones reminiscent of tuscan landscapes | false |
| --golden | warm, golden glow like sunset light | false |
| --lix | high-contrast black and white appearance with increased sharpness | false |
| --lofi | low-fidelity, retro appearance like old photographs or film | false |
| --neue | clean, modern appearance with neutral colors and simple design | false |
| --obsidian | dark, monochromatic appearance with black and gray shades | false |
| --pastelpink | soft, delicate pink tint like pastel colors | false |
| --ryo | bright, high-contrast appearance with vivid colors and sharp details | false |
| --frostedglass | blurred, frosted appearance as if viewed through semi-transparent surface | false |
| --solarize | strange, otherworldly appearance with inverted colors and surreal atmosphere | false |
| --edgedetection | highlights edges and boundaries in an image | false |
![img2irc braille example](https://i.imgur.com/MxroWUb.png)

View File

@ -1,173 +1,300 @@
use clap::Parser;
#[derive(Parser, Debug)]
#[derive(clap::ValueEnum, Clone, Debug)]
pub enum SamplingFilter {
Nearest,
Triangle,
CatmullRom,
Gaussian,
Lanczos3,
}
#[derive(clap::ValueEnum, Clone, Debug)]
pub enum ColourSpace {
HSL,
HSV,
HSLUV,
LCH,
}
#[derive(Parser, Clone, Debug)]
#[command(author, version, about, long_about = None)]
pub struct Args {
/// image url or file path
#[arg(index = 1)]
pub image: String,
/// irc
/// output image width in columns
#[arg(short, long)]
pub width: Option<u32>,
/// output image height in rows
#[arg(short, long)]
pub height: Option<u32>,
/// scaling factors (x:y, e.g., "2:2")
#[arg(long, value_parser = parse_xy_pair)]
pub scale: Option<(f32, f32)>,
/// final aspect ratio (x:y, e.g., "2:1")
#[arg(long, value_parser = parse_xy_pair)]
pub aspect: Option<(f32, f32)>,
/// crop image (x1,y1,x2,y2)
#[arg(long, value_parser = parse_crop_coordinates)]
pub crop: Option<(u32, u32, u32, u32)>,
/// sampling filter
#[arg(long, value_enum, default_value_t = SamplingFilter::Nearest)]
pub filter: SamplingFilter,
/// rotate degrees
#[arg(long, default_value_t = 0)]
pub rotate: i32,
/// flip horizontal
#[arg(long, default_value_t = false)]
pub fliph: bool,
/// flip vertical
#[arg(long, default_value_t = false)]
pub flipv: bool,
/// use IRC99 colours
#[arg(long, default_value_t = false, group = "colour", required_unless_present_any = ["ansi", "ansi24"])]
pub irc: bool,
/// 8-bit ansi
#[arg(long, default_value_t = false)]
/// use 8-bit ANSI colours
#[arg(long, default_value_t = false, group = "colour", required_unless_present_any = ["irc", "ansi24"])]
pub ansi: bool,
/// 24-bit ansi
#[arg(long, default_value_t = false)]
/// use 24-bit ANSI colours
#[arg(long, default_value_t = false, group = "colour", required_unless_present_any = ["irc", "ansi"])]
pub ansi24: bool,
/// quarterblock
#[arg(long, default_value_t = false)]
pub qb: bool,
/// use braille pixels
#[arg(long, default_value_t = false, group = "pixel")]
pub braille: bool,
/// image width to resize to
#[arg(short, long, default_value_t = 50)]
pub width: u32,
/// use halfblock pixels
#[arg(long, alias = "halfblock", default_value_t = true, group = "pixel")]
pub hb: bool,
/// brightness (-255 to 255)
#[arg(short, long, require_equals = true, default_value_t = 0.0)]
/// use quarterblocks pixels
#[arg(long, alias = "quarterblock", default_value_t = false, group = "pixel")]
pub qb: bool,
/// adjust brightness (0 = no change)
#[arg(short = 'b', long, default_value_t = 0.0, allow_hyphen_values = true)]
pub brightness: f32,
/// contrast (-255 to 255)
#[arg(short, long, require_equals = true, default_value_t = 0.0)]
/// adjust contrast (0 = no change)
#[arg(short = 'c', long, default_value_t = 0.0, allow_hyphen_values = true)]
pub contrast: f32,
/// saturation (-255 to 255)
#[arg(short, long, require_equals = true, default_value_t = 0.0)]
pub saturation: f32,
/// hue (0 to 360)
#[arg(short = 'H', long, default_value_t = 0.0)]
pub hue: f32,
/// gamma (0 to 255)
#[arg(short, long, default_value_t = 0.0)]
/// adjust gamma (0 to 255)
#[arg(short = 'g', long, default_value_t = 0.0, allow_hyphen_values = true)]
pub gamma: f32,
/// dither (1 to 8)
#[arg(long, default_value_t = 0)]
/// adjust saturation (0 = no change)
#[arg(short = 's', long, default_value_t = 0.0, allow_hyphen_values = true)]
pub saturation: f32,
/// rotate hue (0 to 360)
#[arg(short = 'u', long, default_value_t = 0.0)]
pub hue: f32,
/// colors are inverted, opposite on the color wheel
#[arg(short = 'i', long, default_value_t = false)]
pub invert: bool,
/// dithering (1 to 8)
#[arg(long, short = 'd', long, default_value_t = 0)]
pub dither: u32,
/// pixelize size
/// adjust luma brightness (braille only)
#[arg(short = 'B', long, default_value_t = 0.0, allow_hyphen_values = true, requires = "braille")]
pub luma_brightness: f32,
/// adjust luma contrast (braille only)
#[arg(short = 'C', long, default_value_t = 0.0, allow_hyphen_values = true, requires = "braille")]
pub luma_contrast: f32,
/// adjust luma gamma (braille only)
#[arg(short = 'G', long, default_value_t = 0.0, allow_hyphen_values = true, requires = "braille")]
pub luma_gamma: f32,
/// adjust luma saturation (braille only)
#[arg(short = 'S', long, default_value_t = 0.0, allow_hyphen_values = true, requires = "braille")]
pub luma_saturation: f32,
/// luminance is inverted
#[arg(short = 'I', long, default_value_t = false, requires = "braille")]
pub luma_invert: bool,
/// colour space
#[arg(long, value_enum, default_value_t = ColourSpace::HSV)]
pub colorspace: ColourSpace,
/// converts image to black and white
#[arg(long, default_value_t = false, group = "grayscale_opts")]
pub grayscale: bool,
/// exclude grayscale colours from the palette
#[arg(long, default_value_t = false, group = "grayscale_opts")]
pub nograyscale: bool,
/// pixelize pixel size
#[arg(long, default_value_t = 0)]
pub pixelize: i32,
/// simple average of all the neighboring pixels surrounding a given pixel
#[arg(long="boxblur", default_value_t = false)]
pub box_blur: bool,
/// gaussian blur radius
#[arg(long, default_value_t = 0)]
#[arg(long="gaussianblur", default_value_t = 0)]
pub gaussian_blur: i32,
/// oil ("<radius>,<intensity>")
/// oil filter ("[radius],[intensity]")
#[arg(long)]
pub oil: Option<String>,
/// grayscale
#[arg(long, default_value_t = false)]
pub grayscale: bool,
/// no grayscale
#[arg(long, default_value_t = false)]
pub nograyscale: bool,
/// halftone
/// made up of small dots creating a continuous-tone illusion
#[arg(long, default_value_t = false)]
pub halftone: bool,
/// sepia
/// brownish, aged appearance like old photographs
#[arg(long, default_value_t = false)]
pub sepia: bool,
/// normalize
/// adjusts brightness and contrast for better image quality
#[arg(long, default_value_t = false)]
pub normalize: bool,
/// noise
/// random variations in brightness and color like film grain
#[arg(long, default_value_t = false)]
pub noise: bool,
/// emboss
/// gives a raised, 3d appearance
#[arg(long, default_value_t = false)]
pub emboss: bool,
/// box_blur
#[arg(long, default_value_t = false)]
pub box_blur: bool,
/// identity
/// no modifications, unchanged image
#[arg(long, default_value_t = false)]
pub identity: bool,
/// laplace
/// enhances edges and boundaries in an image
#[arg(long, default_value_t = false)]
pub laplace: bool,
/// noise reduction
#[arg(long, default_value_t = false)]
/// reduces noise for a cleaner, clearer image
#[arg(long="denoise", default_value_t = false)]
pub noise_reduction: bool,
/// sharpen
/// increases clarity and definition, making edges and details more distinct
#[arg(long, default_value_t = false)]
pub sharpen: bool,
/// cali
/// cool blue tone with increased contrast
#[arg(long, default_value_t = false)]
pub cali: bool,
/// dramatic
/// high contrast and vivid colors for a dramatic effect
#[arg(long, default_value_t = false)]
pub dramatic: bool,
/// firenze
/// warm, earthy tones reminiscent of tuscan landscapes
#[arg(long, default_value_t = false)]
pub firenze: bool,
/// golden
/// warm, golden glow like sunset light
#[arg(long, default_value_t = false)]
pub golden: bool,
/// lix
/// high-contrast black and white appearance with increased sharpness
#[arg(long, default_value_t = false)]
pub lix: bool,
/// lofi
/// low-fidelity, retro appearance like old photographs or film
#[arg(long, default_value_t = false)]
pub lofi: bool,
/// neue
/// clean, modern appearance with neutral colors and simple design
#[arg(long, default_value_t = false)]
pub neue: bool,
/// obsidian
/// dark, monochromatic appearance with black and gray shades
#[arg(long, default_value_t = false)]
pub obsidian: bool,
/// pastel_pink
#[arg(long, default_value_t = false)]
/// soft, delicate pink tint like pastel colors
#[arg(long="pastelpink", default_value_t = false)]
pub pastel_pink: bool,
/// ryo
/// bright, high-contrast appearance with vivid colors and sharp details
#[arg(long, default_value_t = false)]
pub ryo: bool,
/// invert
#[arg(long, default_value_t = false)]
pub invert: bool,
/// frosted glass
#[arg(long, default_value_t = false)]
/// blurred, frosted appearance as if viewed through semi-transparent surface
#[arg(long="frostedglass", default_value_t = false)]
pub frosted_glass: bool,
/// solarize
/// strange, otherworldly appearance with inverted colors and surreal atmosphere
#[arg(long, default_value_t = false)]
pub solarize: bool,
/// edge detection
#[arg(long, default_value_t = false)]
/// highlights edges and boundaries in an image
#[arg(long="edgedetection", default_value_t = false)]
pub edge_detection: bool,
}
pub fn parse_args() -> Args {
Args::parse()
}
}
fn parse_xy_pair(s: &str) -> Result<(f32, f32), String> {
let parts: Vec<&str> = s.split(':').collect();
if parts.len() != 2 {
return Err(format!(
"Invalid format. Expected 'value1:value2', got '{}'",
s
));
}
let first = parts[0].parse::<f32>().map_err(|e| {
format!(
"Failed to parse the first value ('{}') as f32: {}",
parts[0], e
)
})?;
let second = parts[1].parse::<f32>().map_err(|e| {
format!(
"Failed to parse the second value ('{}') as f32: {}",
parts[1], e
)
})?;
if first <= 0.0 || second <= 0.0 {
return Err("Both values must be positive numbers.".to_string());
}
Ok((first, second))
}
fn parse_crop_coordinates(s: &str) -> Result<(u32, u32, u32, u32), String> {
let coords: Vec<&str> = s.split(',').collect();
if coords.len() != 4 {
return Err(format!(
"Invalid crop format '{}'. Expected 'x1,y1,x2,y2' (e.g., '50,50,200,200').",
s
));
}
let x1 = coords[0].parse::<u32>().map_err(|_| format!("Invalid x1 value '{}'.", coords[0]))?;
let y1 = coords[1].parse::<u32>().map_err(|_| format!("Invalid y1 value '{}'.", coords[1]))?;
let x2 = coords[2].parse::<u32>().map_err(|_| format!("Invalid x2 value '{}'.", coords[2]))?;
let y2 = coords[3].parse::<u32>().map_err(|_| format!("Invalid y2 value '{}'.", coords[3]))?;
Ok((x1, y1, x2, y2))
}

View File

@ -1,45 +1,45 @@
use crate::args;
use crate::palette::{RGB99, RGB88, ANSI232, ANSI256, nearest_hex_color};
use photon_rs::PhotonImage;
use std::collections::HashMap;
// █ full
const FULL: &str = "\u{2588}";
// ▄ down
const UP: &str = "\u{2580}";
// ▀ up
const DOWN: &str = "\u{2584}";
static QUARTER_BLOCKS: [&str; 16] = [
" ", // 0b0000
"\u{2597}", // ▗ Quadrant lower right
"\u{2596}", // ▖ Quadrant lower left
"\u{2584}", // ▄ Lower half block
"\u{259D}", // ▝ Quadrant upper right
"\u{2590}", // ▐ Right half block
"\u{259E}", // ▞ Quadrant upper right and lower left
"\u{259F}", // ▟ Quadrant upper right, lower left, lower right
"\u{2598}", // ▘ Quadrant upper left
"\u{259A}", // ▚ Quadrant upper left and lower right
"\u{258C}", // ▌ Left half block
"\u{2599}", // ▙ Quadrant upper left, lower left, lower right
"\u{2580}", // ▀ Upper half block
"\u{259C}", // ▜ Quadrant upper left, upper right, lower right
"\u{259B}", // ▛ Quadrant upper left, upper right, lower left
"\u{2588}", // █ Full block
];
// ▌ left
const LEFT: &str = "\u{258C}";
// ▐ right
const RIGHT: &str = "\u{2590}";
// ▞ diag_right
const DIAG_RIGHT: &str = "\u{259E}";
// ▚ diag_left
const DIAG_LEFT: &str = "\u{259A}";
// ▙ down_left (2596 prev)
const DOWN_LEFT: &str = "\u{2599}";
// ▟ down_right
const DOWN_RIGHT: &str = "\u{259F}";
// ▛ top_left
const UP_LEFT: &str = "\u{259B}";
// ▜ top_right
const UP_RIGHT: &str = "\u{259C}";
const POSITIONS: [(usize, usize, u32); 8] = [
(0, 0, 0x01), // dot 1
(0, 1, 0x02), // dot 2
(0, 2, 0x04), // dot 3
(1, 0, 0x08), // dot 4
(1, 1, 0x10), // dot 5
(1, 2, 0x20), // dot 6
(0, 3, 0x40), // dot 7
(1, 3, 0x80), // dot 8
];
#[derive(Debug, Clone)]
pub struct AnsiImage {
pub image: PhotonImage,
pub bitmap: Vec<Vec<u32>>,
pub halfblock: Vec<Vec<AnsiPixelPair>>,
pub quarterblock: Vec<Vec<AnsiPixelQuad>>,
}
#[derive(Debug, Clone, Copy)]
@ -57,6 +57,14 @@ pub struct AnsiPixel {
pub irc88: u8,
}
#[derive(Debug, Clone, Copy)]
pub struct AnsiPixelQuad {
pub top_left: AnsiPixel,
pub top_right: AnsiPixel,
pub bottom_left: AnsiPixel,
pub bottom_right: AnsiPixel,
}
impl AnsiPixel {
pub fn new(pixel: &u32) -> AnsiPixel {
let irc = nearest_hex_color(*pixel, RGB99.to_vec());
@ -65,17 +73,18 @@ impl AnsiPixel {
let ansi232 = nearest_hex_color(*pixel, ANSI232.to_vec());
AnsiPixel {
orig: *pixel,
ansi: ansi,
ansi232: ansi232,
irc: irc,
irc88: irc88,
ansi,
ansi232,
irc,
irc88,
}
}
}
impl AnsiImage {
pub fn new(image: PhotonImage) -> AnsiImage {
let mut bitmap = image.get_raw_pixels()
let mut bitmap = image
.get_raw_pixels()
.chunks(4)
.map(|x| make_rgb_u32(x.to_vec()))
.collect::<Vec<u32>>()
@ -87,12 +96,19 @@ impl AnsiImage {
bitmap.push(vec![0; image.get_width() as usize]);
}
for row in &mut bitmap {
if row.len() % 2 != 0 {
row.push(0);
}
}
let halfblock = halfblock_bitmap(&bitmap);
return AnsiImage {
image: image,
bitmap: bitmap,
halfblock: halfblock,
let quarterblock = quarterblock_bitmap(&bitmap);
AnsiImage {
bitmap,
halfblock,
quarterblock,
}
}
}
@ -102,39 +118,72 @@ pub fn make_rgb_u8(rgb: u32) -> [u8; 3] {
let g = (rgb >> 8) as u8;
let b = rgb as u8;
return [r, g, b]
[r, g, b]
}
pub fn make_rgb_u32(rgb: Vec<u8>) -> u32 {
let r = *rgb.get(0).unwrap() as u32;
let g = *rgb.get(1).unwrap() as u32;
let b = *rgb.get(2).unwrap() as u32;
let r = rgb[0] as u32;
let g = rgb[1] as u32;
let b = rgb[2] as u32;
let rgb = (r << 16) + (g << 8) + b;
(r << 16) + (g << 8) + b
}
return rgb
pub fn quarterblock_bitmap(bitmap: &Vec<Vec<u32>>) -> Vec<Vec<AnsiPixelQuad>> {
let ansi_bitmap = bitmap
.iter()
.map(|x| x.iter().map(|y| AnsiPixel::new(y)).collect::<Vec<AnsiPixel>>())
.collect::<Vec<Vec<AnsiPixel>>>();
let mut ansi_canvas: Vec<Vec<AnsiPixelQuad>> = Vec::new();
for two_rows in ansi_bitmap.chunks(2) {
let top_row = &two_rows[0];
let bottom_row = &two_rows[1];
let mut ansi_row: Vec<AnsiPixelQuad> = Vec::new();
for i in (0..top_row.len()).step_by(2) {
let default_pixel = AnsiPixel::new(&0);
let top_left_pixel = top_row.get(i).unwrap_or(&default_pixel);
let top_right_pixel = top_row.get(i + 1).unwrap_or(&default_pixel);
let bottom_left_pixel = bottom_row.get(i).unwrap_or(&default_pixel);
let bottom_right_pixel = bottom_row.get(i + 1).unwrap_or(&default_pixel);
let pixel_quad = AnsiPixelQuad {
top_left: *top_left_pixel,
top_right: *top_right_pixel,
bottom_left: *bottom_left_pixel,
bottom_right: *bottom_right_pixel,
};
ansi_row.push(pixel_quad);
}
ansi_canvas.push(ansi_row);
}
ansi_canvas
}
pub fn halfblock_bitmap(bitmap: &Vec<Vec<u32>>) -> Vec<Vec<AnsiPixelPair>> {
let ansi_bitmap = bitmap
.iter()
.map(|x| {
x.iter().map(|y| AnsiPixel::new(y)).collect::<Vec<AnsiPixel>>()
})
.collect::<Vec<Vec<AnsiPixel>>>();
.iter()
.map(|x| x.iter().map(|y| AnsiPixel::new(y)).collect::<Vec<AnsiPixel>>())
.collect::<Vec<Vec<AnsiPixel>>>();
let mut ansi_canvas: Vec<Vec<AnsiPixelPair>> = Vec::new();
for two_rows in ansi_bitmap.chunks(2) {
let rows = two_rows.to_vec();
let top_row = rows.get(0).unwrap();
let bottom_row = rows.get(1).unwrap();
let top_row = &two_rows[0];
let bottom_row = &two_rows[1];
let mut ansi_row: Vec<AnsiPixelPair> = Vec::new();
for i in 0..bitmap.get(0).unwrap().len() {
let top_pixel = top_row.get(i as usize).unwrap();
let bottom_pixel = bottom_row.get(i as usize).unwrap();
for i in 0..top_row.len() {
let default_pixel = AnsiPixel::new(&0);
let top_pixel = top_row.get(i).unwrap_or(&default_pixel);
let bottom_pixel = bottom_row.get(i).unwrap_or(&default_pixel);
let pixel_pair = AnsiPixelPair {
top: *top_pixel,
@ -150,231 +199,591 @@ pub fn halfblock_bitmap(bitmap: &Vec<Vec<u32>>) -> Vec<Vec<AnsiPixelPair>> {
ansi_canvas
}
fn get_qb_char(pixel_pairs: &[AnsiPixelPair]) -> &str {
let (pair0_top, pair0_bottom) = (&pixel_pairs[0].top.irc, &pixel_pairs[0].bottom.irc);
let (pair1_top, pair1_bottom) = (&pixel_pairs[1].top.irc, &pixel_pairs[1].bottom.irc);
let ups_equal = pair0_top == pair1_top;
let downs_equal = pair0_bottom == pair1_bottom;
let lefts_equal = pair0_top == pair0_bottom;
let rights_equal = pair1_top == pair1_bottom;
let left_diag = pair0_top == pair1_bottom;
let right_diag = pair1_top == pair0_bottom;
match (ups_equal, downs_equal, lefts_equal, rights_equal, left_diag, right_diag) {
(true, _, true, true, _, _) => FULL,
(true, _, true, _, _, _) => UP_LEFT,
(true, _, _, true, _, _) => UP_RIGHT,
(true, _, _, _, _, _) => UP,
(_, true, true, _, _, _) => DOWN_LEFT,
(_, true, _, true, _, _) => DOWN_RIGHT,
(_, true, _, _, _, _) => DOWN,
(_, _, true, false, _, _) => LEFT,
(_, _, false, true, _, _) => RIGHT,
(_, _, _, _, true, _) => DIAG_LEFT,
(_, _, _, _, _, true) => DIAG_RIGHT,
_ => UP,
fn get_qb_char<T: PartialEq>(pixels: &[T; 4], fg_color: &T) -> &'static str {
let mut pattern = 0;
if pixels[2] == *fg_color {
pattern |= 1 << 0; // bit 0 (bottom-left)
}
if pixels[3] == *fg_color {
pattern |= 1 << 1; // bit 1 (bottom-right)
}
if pixels[0] == *fg_color {
pattern |= 1 << 2; // bit 2 (top-left)
}
if pixels[1] == *fg_color {
pattern |= 1 << 3; // bit 3 (top-right)
}
QUARTER_BLOCKS[pattern as usize]
}
pub fn ansi_draw_24bit(image: AnsiImage) -> String {
pub fn irc_draw_qb(image: &AnsiImage, args: &args::Args) -> String {
let mut out: String = String::new();
for (y, row) in image.halfblock.iter().enumerate() {
for (y, row) in image.quarterblock.iter().enumerate() {
let mut last_fg: u8 = 255;
let mut last_bg: u8 = 255;
for (x, pixel_quad) in row.iter().enumerate() {
let c0 = if args.nograyscale {
pixel_quad.top_right.irc88
} else {
pixel_quad.top_right.irc
};
let c1 = if args.nograyscale {
pixel_quad.top_left.irc88
} else {
pixel_quad.top_left.irc
};
let c2 = if args.nograyscale {
pixel_quad.bottom_right.irc88
} else {
pixel_quad.bottom_right.irc
};
let c3 = if args.nograyscale {
pixel_quad.bottom_left.irc88
} else {
pixel_quad.bottom_left.irc
};
let mut color_counts = HashMap::new();
for &color in &[c0, c1, c2, c3] {
*color_counts.entry(color).or_insert(0) += 1;
}
let bg_color = *color_counts
.iter()
.max_by_key(|entry| entry.1)
.map(|(color, _)| color)
.unwrap_or(&0);
let fg_color = *color_counts
.iter()
.filter(|&(color, _)| color != &bg_color)
.map(|(color, _)| color)
.next()
.unwrap_or(&bg_color);
let pixels = [c0, c1, c2, c3];
let char = get_qb_char(&pixels, &fg_color);
if x == 0 || fg_color != last_fg || bg_color != last_bg {
if fg_color == bg_color {
out.push_str(&format!("\x03{}{}", fg_color, char));
} else {
out.push_str(&format!("\x03{},{}{}", fg_color, bg_color, char));
}
} else {
out.push_str(&char);
}
last_fg = fg_color;
last_bg = bg_color;
}
out.push_str("\x0f");
if y != image.quarterblock.len() - 1 {
out.push_str("\n");
}
}
out.trim_end().to_string()
}
pub fn ansi_draw_24bit_qb(image: &AnsiImage) -> String {
let mut out: String = String::new();
for row in &image.quarterblock {
for pixel_quad in row.iter() {
let c0_rgb = make_rgb_u8(pixel_quad.top_right.orig);
let c1_rgb = make_rgb_u8(pixel_quad.top_left.orig);
let c2_rgb = make_rgb_u8(pixel_quad.bottom_right.orig);
let c3_rgb = make_rgb_u8(pixel_quad.bottom_left.orig);
let mut color_counts = HashMap::new();
for &rgb in &[c0_rgb, c1_rgb, c2_rgb, c3_rgb] {
*color_counts.entry(rgb).or_insert(0) += 1;
}
let bg_color = *color_counts
.iter()
.max_by_key(|entry| entry.1)
.map(|(rgb, _)| rgb)
.unwrap_or(&[0, 0, 0]);
let fg_color = *color_counts
.iter()
.filter(|&(rgb, _)| rgb != &bg_color)
.map(|(rgb, _)| rgb)
.next()
.unwrap_or(&bg_color);
let pixels = [c0_rgb, c1_rgb, c2_rgb, c3_rgb];
let char = get_qb_char(&pixels, &fg_color);
out.push_str(&format!(
"\x1b[38;2;{};{};{}m\x1b[48;2;{};{};{}m{}",
fg_color[0],
fg_color[1],
fg_color[2],
bg_color[0],
bg_color[1],
bg_color[2],
char
));
}
out.push_str("\x1b[0m\n");
}
out.trim_end().to_string()
}
pub fn ansi_draw_8bit_qb(image: &AnsiImage, args: &args::Args) -> String {
let mut out: String = String::new();
for row in &image.quarterblock {
for pixel_quad in row.iter() {
let c0 = if args.nograyscale {
pixel_quad.top_right.ansi232
} else {
pixel_quad.top_right.ansi
};
let c1 = if args.nograyscale {
pixel_quad.top_left.ansi232
} else {
pixel_quad.top_left.ansi
};
let c2 = if args.nograyscale {
pixel_quad.bottom_right.ansi232
} else {
pixel_quad.bottom_right.ansi
};
let c3 = if args.nograyscale {
pixel_quad.bottom_left.ansi232
} else {
pixel_quad.bottom_left.ansi
};
let mut color_counts = HashMap::new();
for &color in &[c0, c1, c2, c3] {
*color_counts.entry(color).or_insert(0) += 1;
}
let bg_color = *color_counts
.iter()
.max_by_key(|entry| entry.1)
.map(|(color, _)| color)
.unwrap_or(&0);
let fg_color = *color_counts
.iter()
.filter(|&(color, _)| color != &bg_color)
.map(|(color, _)| color)
.next()
.unwrap_or(&bg_color);
let pixels = [c0, c1, c2, c3];
let char = get_qb_char(&pixels, &fg_color);
out.push_str(&format!(
"\x1b[38;5;{}m\x1b[48;5;{}m{}",
fg_color,
bg_color,
char
));
}
out.push_str("\x1b[0m\n");
}
out.trim_end().to_string()
}
pub fn ansi_draw_24bit(image: &AnsiImage) -> String {
let mut out: String = String::new();
for row in &image.halfblock {
for pixel_pair in row.iter() {
let fg = make_rgb_u8(pixel_pair.top.orig)
.to_vec()
.iter()
.map(|x| x.to_string())
.collect::<Vec<String>>();
let bg = make_rgb_u8(pixel_pair.bottom.orig)
.to_vec()
.iter()
.map(|x| x.to_string())
.collect::<Vec<String>>();
out.push_str(format!("\x1b[38;2;{}m\x1b[48;2;{}m{}", fg.join(";"), bg.join(";"), UP).as_str());
}
out.push_str("\x1b[0m");
if y != image.halfblock.len() - 1 {
out.push_str("\n");
out.push_str(
format!(
"\x1b[38;2;{}m\x1b[48;2;{}m{}",
fg.join(";"),
bg.join(";"),
UP
)
.as_str(),
);
}
out.push_str("\x1b[0m\n");
}
return out
out.trim_end().to_string()
}
pub fn ansi_draw_24bit_qb(image: AnsiImage) -> String {
pub fn ansi_draw_8bit(image: &AnsiImage, args: &args::Args) -> String {
let mut out: String = String::new();
for (y, row) in image.halfblock.iter().enumerate() {
for pixel_pairs in row.chunks(2) {
let fg = make_rgb_u8(pixel_pairs[0].top.orig)
.to_vec()
.iter()
.map(|x| x.to_string())
.collect::<Vec<String>>();
let bg = make_rgb_u8(pixel_pairs[0].bottom.orig)
.to_vec()
.iter()
.map(|x| x.to_string())
.collect::<Vec<String>>();
let char = match y {
_ if y == image.halfblock.len() - 1 => UP,
_ => get_qb_char(pixel_pairs),
};
out.push_str(format!("\x1b[38;2;{}m\x1b[48;2;{}m{}", fg.join(";"), bg.join(";"), char).as_str());
}
out.push_str("\x1b[0m");
if y != image.halfblock.len() - 1 {
out.push_str("\n");
}
}
return out
}
pub fn ansi_draw_8bit(image: AnsiImage, args: &args::Args) -> String {
let mut out: String = String::new();
for (y, row) in image.halfblock.iter().enumerate() {
for row in &image.halfblock {
for pixel_pair in row.iter() {
let fg = match args.nograyscale {
true => pixel_pair.top.ansi232,
false => pixel_pair.top.ansi,
let fg = if args.nograyscale {
pixel_pair.top.ansi232
} else {
pixel_pair.top.ansi
};
let bg = match args.nograyscale {
true => pixel_pair.bottom.ansi232,
false => pixel_pair.bottom.ansi,
let bg = if args.nograyscale {
pixel_pair.bottom.ansi232
} else {
pixel_pair.bottom.ansi
};
out.push_str(format!("\x1b[38;5;{}m\x1b[48;5;{}m{}", fg, bg, UP).as_str());
}
out.push_str("\x1b[0m");
if y != image.halfblock.len() - 1 {
out.push_str("\n");
}
out.push_str("\x1b[0m\n");
}
return out
out.trim_end().to_string()
}
pub fn ansi_draw_8bit_qb(image: AnsiImage, args: &args::Args) -> String {
pub fn irc_draw(image: &AnsiImage, args: &args::Args) -> String {
let mut out: String = String::new();
for (y, row) in image.halfblock.iter().enumerate() {
for pixel_pairs in row.chunks(2) {
let fg = match args.nograyscale {
true => pixel_pairs[0].top.ansi232,
false => pixel_pairs[0].top.ansi,
};
let bg = match args.nograyscale {
true => pixel_pairs[0].bottom.ansi232,
false => pixel_pairs[0].bottom.ansi,
};
let char = match y {
_ if y == image.halfblock.len() - 1 => UP,
_ => get_qb_char(pixel_pairs),
};
out.push_str(format!("\x1b[38;5;{}m\x1b[48;5;{}m{}", fg, bg, char).as_str());
}
out.push_str("\x1b[0m");
if y != image.halfblock.len() - 1 {
out.push_str("\n");
}
}
return out
}
pub fn irc_draw(image: AnsiImage, args: &args::Args) -> String {
let mut out: String = String::new();
for (y, row) in image.halfblock.iter().enumerate() {
let mut last_fg: u8 = 0;
let mut last_bg: u8 = 0;
for row in &image.halfblock {
let mut last_fg: u8 = 255;
let mut last_bg: u8 = 255;
for (x, pixel_pair) in row.iter().enumerate() {
let fg = match args.nograyscale {
true => pixel_pair.top.irc88,
false => pixel_pair.top.irc,
let fg = if args.nograyscale {
pixel_pair.top.irc88
} else {
pixel_pair.top.irc
};
let bg = match args.nograyscale {
true => pixel_pair.bottom.irc88,
false => pixel_pair.bottom.irc,
let bg = if args.nograyscale {
pixel_pair.bottom.irc88
} else {
pixel_pair.bottom.irc
};
if x != 0 {
if fg == last_fg && bg == last_bg {
out.push_str(&format!("{}", UP));
} else if bg == last_bg {
if x == 0 || fg != last_fg || bg != last_bg {
if last_bg == bg {
out.push_str(&format!("\x03{}{}", fg, UP));
} else {
out.push_str(&format!("\x03{},{}{}", fg, bg, UP));
}
} else {
out.push_str(&format!("\x03{},{}{}", fg, bg, UP));
out.push_str(&UP);
}
last_fg = fg;
last_bg = bg;
}
out.push_str("\x0f");
if y != image.halfblock.len() - 1 {
out.push_str("\n");
}
out.push_str("\x0f\n");
}
return out
out.trim_end().to_string()
}
pub fn irc_draw_qb(image: AnsiImage, args: &args::Args) -> String {
pub fn luma(rgb: &[u8; 3]) -> u8 {
let r = rgb[0] as u16;
let g = rgb[1] as u16;
let b = rgb[2] as u16;
((0.299 * r as f32 + 0.587 * g as f32 + 0.114 * b as f32).round()) as u8
}
pub fn ansi_draw_braille_24bit(image_luma: &AnsiImage, image_chroma: &AnsiImage) -> String {
let mut out: String = String::new();
for (y, row) in image.halfblock.iter().enumerate() {
let mut last_fg: u8 = 0;
let mut last_bg: u8 = 0;
for (x, pixel_pairs) in row.chunks(2).enumerate() {
let fg = match args.nograyscale {
true => pixel_pairs[0].top.irc88,
false => pixel_pairs[0].top.irc,
};
let bitmap_luma = &image_luma.bitmap;
let bitmap_chroma = &image_chroma.bitmap;
let height = bitmap_luma.len();
let width = bitmap_luma[0].len();
let bg = match args.nograyscale {
true => pixel_pairs[0].bottom.irc88,
false => pixel_pairs[0].bottom.irc,
};
let mut error_matrix = vec![vec![0.0; width]; height];
let mut min_luma = 255u32;
let mut max_luma = 0u32;
for row in bitmap_luma.iter() {
for &pixel in row.iter() {
let l = luma(&make_rgb_u8(pixel)) as u32;
min_luma = min_luma.min(l);
max_luma = max_luma.max(l);
}
}
let char = match y {
_ if y == image.halfblock.len() - 1 => UP,
_ => get_qb_char(pixel_pairs),
};
let luma_range = max_luma.saturating_sub(min_luma).max(1);
if x == 0 {
out.push_str(&format!("\x03{},{}{}", fg, bg, char));
} else {
if fg == last_fg && bg == last_bg {
out.push_str(&format!("{}", char));
} else if bg == last_bg {
out.push_str(&format!("\x03{}{}", fg, char));
} else {
out.push_str(&format!("\x03{},{}{}", fg, bg, char));
let scaled_threshold = min_luma + (luma_range / 2);
for y in (0..height).step_by(4) {
let mut last_fg = [0u8; 3];
let mut first = true;
for x in (0..width).step_by(2) {
let mut braille_char = 0x2800;
let mut color_counts: HashMap<[u8; 3], u32> = HashMap::new();
for &(dx, dy, bit) in &POSITIONS {
let current_y = y + dy;
let current_x = x + dx;
if current_y < height && current_x < width {
let original_pixel = bitmap_luma[current_y][current_x];
let rgb = make_rgb_u8(original_pixel);
let current_luma = luma(&rgb) as f64;
let mut pixel_luma = current_luma + error_matrix[current_y][current_x];
pixel_luma = pixel_luma.clamp(0.0, 255.0);
let new_pixel = if pixel_luma > scaled_threshold as f64 {
255.0
} else {
0.0
};
let error = pixel_luma - new_pixel;
if new_pixel == 255.0 {
braille_char |= bit;
let px_chroma = bitmap_chroma[current_y][current_x];
let rgb_chroma = make_rgb_u8(px_chroma);
*color_counts.entry(rgb_chroma).or_insert(0) += 1;
}
if current_x + 1 < width {
error_matrix[current_y][current_x + 1] += error * 7.0 / 16.0;
}
if current_y + 1 < height {
if current_x > 0 {
error_matrix[current_y + 1][current_x - 1] += error * 3.0 / 16.0;
}
error_matrix[current_y + 1][current_x] += error * 5.0 / 16.0;
if current_x + 1 < width {
error_matrix[current_y + 1][current_x + 1] += error * 1.0 / 16.0;
}
}
}
}
last_fg = fg;
last_bg = bg;
let fg_color = color_counts
.iter()
.max_by_key(|entry| entry.1)
.map(|(color, _)| *color)
.unwrap_or([0, 0, 0]);
let braille_char = char::from_u32(braille_char).unwrap_or(' ');
if first || last_fg != fg_color {
out.push_str(&format!(
"\x1b[38;2;{};{};{}m{}",
fg_color[0], fg_color[1], fg_color[2], braille_char
));
} else {
out.push(braille_char);
}
last_fg = fg_color;
first = false;
}
out.push_str("\x1b[0m\n");
}
out.trim_end().to_string()
}
out.push_str("\x0f");
if y != image.halfblock.len() - 1 {
out.push_str("\n");
pub fn ansi_draw_braille_8bit(
image_luma: &AnsiImage,
image_chroma: &AnsiImage,
args: &args::Args,
) -> String {
let mut out: String = String::new();
let bitmap_luma = &image_luma.bitmap;
let bitmap_chroma = &image_chroma.bitmap;
let height = bitmap_luma.len();
let width = bitmap_luma[0].len();
let mut error_matrix = vec![vec![0.0; width]; height];
let mut min_luma = 255u32;
let mut max_luma = 0u32;
for row in bitmap_luma.iter() {
for &pixel in row.iter() {
let l = luma(&make_rgb_u8(pixel)) as u32;
min_luma = min_luma.min(l);
max_luma = max_luma.max(l);
}
}
return out
}
let luma_range = max_luma.saturating_sub(min_luma).max(1);
let scaled_threshold = min_luma + (luma_range / 2);
for y in (0..height).step_by(4) {
let mut last_fg = 255u8; // Initialize with a default color
let mut first = true;
for x in (0..width).step_by(2) {
let mut braille_char = 0x2800; // Base Unicode braille character
let mut color_counts: HashMap<u8, usize> = HashMap::new(); // Color counting
for &(dx, dy, bit) in &POSITIONS {
let current_y = y + dy;
let current_x = x + dx;
if current_y < height && current_x < width {
let original_pixel = bitmap_luma[current_y][current_x];
let rgb = make_rgb_u8(original_pixel);
let current_luma = luma(&rgb) as f64;
let mut pixel_luma = current_luma + error_matrix[current_y][current_x];
pixel_luma = pixel_luma.clamp(0.0, 255.0);
let new_pixel = if pixel_luma > scaled_threshold as f64 {
255.0
} else {
0.0
};
let error = pixel_luma - new_pixel;
if new_pixel == 255.0 {
braille_char |= bit;
let px_chroma = bitmap_chroma[current_y][current_x];
let color = if args.nograyscale {
nearest_hex_color(px_chroma, ANSI232.to_vec())
} else {
nearest_hex_color(px_chroma, ANSI256.to_vec())
};
*color_counts.entry(color).or_insert(0) += 1;
}
if current_x + 1 < width {
error_matrix[current_y][current_x + 1] += error * 7.0 / 16.0;
}
if current_y + 1 < height {
if current_x > 0 {
error_matrix[current_y + 1][current_x - 1] += error * 3.0 / 16.0;
}
error_matrix[current_y + 1][current_x] += error * 5.0 / 16.0;
if current_x + 1 < width {
error_matrix[current_y + 1][current_x + 1] += error * 1.0 / 16.0;
}
}
}
}
let fg_color = *color_counts
.iter()
.max_by_key(|entry| entry.1)
.map(|(color, _)| color)
.unwrap_or(&last_fg); // Default to last_fg if no colors are counted
let braille_char = char::from_u32(braille_char).unwrap_or(' ');
if first || last_fg != fg_color {
out.push_str(&format!("\x1b[38;5;{}m{}", fg_color, braille_char));
} else {
out.push(braille_char);
}
last_fg = fg_color;
first = false;
}
out.push_str("\x1b[0m\n"); // Reset colors and move to the next line
}
out.trim_end().to_string()
}
pub fn irc_draw_braille(image_luma: &AnsiImage, image_chroma: &AnsiImage, args: &args::Args) -> String {
let mut out: String = String::new();
let bitmap_luma = &image_luma.bitmap;
let bitmap_chroma = &image_chroma.bitmap;
let height = bitmap_luma.len();
let width = bitmap_luma[0].len();
let mut error_matrix = vec![vec![0.0; width]; height];
let mut min_luma = 255u32;
let mut max_luma = 0u32;
for row in bitmap_luma.iter() {
for &pixel in row.iter() {
let l = luma(&make_rgb_u8(pixel)) as u32;
min_luma = min_luma.min(l);
max_luma = max_luma.max(l);
}
}
let luma_range = max_luma.saturating_sub(min_luma).max(1);
let scaled_threshold = min_luma + (luma_range / 2);
for y in (0..height).step_by(4) {
let mut last_fg = 255u8;
let mut first = true;
for x in (0..width).step_by(2) {
let mut braille_char = 0x2800;
let mut color_counts = HashMap::new();
for &(dx, dy, bit) in &POSITIONS {
let current_y = y + dy;
let current_x = x + dx;
if current_y < height && current_x < width {
let original_pixel = bitmap_luma[current_y][current_x];
let rgb = make_rgb_u8(original_pixel);
let current_luma = luma(&rgb) as f64;
let mut pixel_luma = current_luma + error_matrix[current_y][current_x];
pixel_luma = pixel_luma.clamp(0.0, 255.0);
let new_pixel = if pixel_luma > scaled_threshold as f64 {
255.0
} else {
0.0
};
let error = pixel_luma - new_pixel;
if new_pixel == 255.0 {
braille_char |= bit;
let px_chroma = bitmap_chroma[current_y][current_x];
let color = if args.nograyscale {
nearest_hex_color(px_chroma, RGB88.to_vec())
} else {
nearest_hex_color(px_chroma, RGB99.to_vec())
};
*color_counts.entry(color).or_insert(0) += 1;
}
if current_x + 1 < width {
error_matrix[current_y][current_x + 1] += error * 7.0 / 16.0;
}
if current_y + 1 < height {
if current_x > 0 {
error_matrix[current_y + 1][current_x - 1] += error * 3.0 / 16.0;
}
error_matrix[current_y + 1][current_x] += error * 5.0 / 16.0;
if current_x + 1 < width {
error_matrix[current_y + 1][current_x + 1] += error * 1.0 / 16.0;
}
}
}
}
let fg_color = *color_counts
.iter()
.max_by_key(|entry| entry.1)
.map(|(color, _)| color)
.unwrap_or(&last_fg);
let braille_char = char::from_u32(braille_char).unwrap_or(' ');
if first || fg_color != last_fg {
out.push_str(&format!("\x03{}{}", fg_color, braille_char));
} else {
out.push(braille_char);
}
last_fg = fg_color;
first = false;
}
out.push_str("\x0f\n");
}
out.trim_end().to_string()
}

View File

@ -1,213 +1,238 @@
use crate::args;
use photon_rs::{colour_spaces};
use photon_rs::{channels, conv, effects, filters, monochrome, noise};
use photon_rs::transform::{resize, SamplingFilter};
use photon_rs::{colour_spaces, channels, conv, effects, filters, monochrome, noise};
use photon_rs::transform::{SamplingFilter, resize, crop, rotate, flipv, fliph};
use photon_rs::PhotonImage;
fn calculate_dimensions(args: &args::Args, image: &PhotonImage) -> (u32, u32) {
let original_width = image.get_width() as f32;
let original_height = image.get_height() as f32;
let original_aspect = original_width / original_height;
let base_width;
let base_height;
let aspect_ratio = if let Some(aspect) = args.aspect {
aspect.0 / aspect.1
} else {
original_aspect
};
if args.width.is_none() && args.height.is_none() {
base_width = original_width;
base_height = original_height;
} else if args.width.is_none() {
let provided_height = args.height.unwrap();
base_height = provided_height as f32;
base_width = base_height * aspect_ratio;
} else if args.height.is_none() {
let provided_width = args.width.unwrap();
base_width = provided_width as f32;
base_height = base_width / aspect_ratio;
} else {
base_width = args.width.unwrap() as f32;
base_height = args.height.unwrap() as f32;
}
let (scaled_width, scaled_height) = if let Some(scale) = args.scale {
(base_width * scale.0, base_height * scale.1)
} else {
(base_width, base_height)
};
// Step 4: Round the scaled dimensions to the nearest integer and ensure a minimum size of 1.
let final_width = scaled_width.round().max(1.0) as u32;
let final_height = scaled_height.round().max(1.0) as u32;
(final_width, final_height)
}
pub fn apply_effects(
args: &args::Args,
mut photon_image: PhotonImage,
) -> PhotonImage {
let (width, height) = calculate_dimensions(args, &photon_image);
if args.rotate != 0 {
photon_image = rotate(&photon_image, args.rotate as i32);
}
// Resize to width
let height =
(args.width as f32 / photon_image.get_width() as f32 * photon_image.get_height() as f32) as u32;
if args.fliph {
fliph(&mut photon_image);
}
let width = match args.qb {
true => args.width * 2,
_ => args.width,
if args.flipv {
flipv(&mut photon_image);
}
photon_image = resize(&photon_image, width, height, match args.filter {
args::SamplingFilter::Nearest => SamplingFilter::Nearest,
args::SamplingFilter::Triangle => SamplingFilter::Triangle,
args::SamplingFilter::CatmullRom => SamplingFilter::CatmullRom,
args::SamplingFilter::Gaussian => SamplingFilter::Gaussian,
args::SamplingFilter::Lanczos3 => SamplingFilter::Lanczos3,
});
type ColourFunc = fn(&mut PhotonImage, &str, f32);
let colour_func: ColourFunc = match args.colorspace {
args::ColourSpace::HSL => colour_spaces::hsl,
args::ColourSpace::HSV => colour_spaces::hsv,
args::ColourSpace::HSLUV => colour_spaces::hsluv,
args::ColourSpace::LCH => colour_spaces::lch,
};
photon_image = match args.qb {
true => resize(&photon_image, width, height, SamplingFilter::Lanczos3),
_ => resize(&mut photon_image, width, height, SamplingFilter::Lanczos3),
};
if args.dither > 0 {
effects::dither(&mut photon_image, args.dither);
}
// Adjust brightness
match args.brightness {
x if x > 0.0 => {
colour_spaces::hsv(&mut photon_image, "brighten", args.brightness/255.0);
colour_func(&mut photon_image, "lighten", args.brightness / 100.0);
}
x if x < 0.0 => {
colour_spaces::hsv(&mut photon_image, "darken", args.brightness.abs()/255.0);
colour_func(&mut photon_image, "darken", args.brightness.abs() / 100.0);
},
_ => {}
}
// Adjust hue
if args.hue > 0.0 {
colour_spaces::hsv(&mut photon_image, "shift_hue", args.hue/360.0);
}
// Adjust contrast
if args.contrast != 0.0 {
effects::adjust_contrast(&mut photon_image, args.contrast);
}
// Adjust saturation
match args.saturation {
x if x > 0.0 => {
colour_spaces::hsv(&mut photon_image, "saturate", args.saturation/255.0);
colour_func(&mut photon_image, "saturate", args.saturation/100.0);
}
x if x < 0.0 => {
colour_spaces::hsv(&mut photon_image, "desaturate", args.saturation.abs()/255.0);
colour_func(&mut photon_image, "desaturate", args.saturation.abs()/100.0);
}
_ => {}
}
// Adjust gamma
if args.contrast != 0.0 {
effects::adjust_contrast(&mut photon_image, args.contrast);
}
if args.hue > 0.0 {
colour_func(&mut photon_image, "shift_hue", args.hue/360.0);
}
if args.gamma != 0.0 {
let gamma_value = 1.0 - args.gamma/255.0;
colour_spaces::gamma_correction(&mut photon_image, gamma_value, gamma_value, gamma_value);
}
// Adjust dither
if args.dither > 0 {
effects::dither(&mut photon_image, args.dither);
if args.crop.is_some() {
let crop_args = args.crop.unwrap();
photon_image = crop(&mut photon_image, crop_args.0, crop_args.1, crop_args.2, crop_args.3);
}
// Adjust gaussian_blur
if args.gaussian_blur > 0 {
conv::gaussian_blur(&mut photon_image, args.gaussian_blur);
}
// Adjust pixelize
if args.pixelize > 0 {
effects::pixelize(&mut photon_image, args.pixelize);
}
// Adjust halftone
if args.halftone {
effects::halftone(&mut photon_image);
}
// Adjust invert
if args.invert {
channels::invert(&mut photon_image);
}
// Adjust sepia
if args.sepia {
monochrome::sepia(&mut photon_image);
}
// Adjust solarize
if args.solarize {
effects::solarize(&mut photon_image);
}
// Adjust normalize
if args.normalize {
effects::normalize(&mut photon_image);
}
// Adjust noise
if args.noise {
noise::add_noise_rand(&mut photon_image);
}
// Adjust sharpen
if args.sharpen {
conv::sharpen(&mut photon_image);
}
// Adjust edge_detection
if args.edge_detection {
conv::edge_detection(&mut photon_image);
}
// Adjust emboss
if args.emboss {
conv::emboss(&mut photon_image);
}
// Adjust frosted_glass
if args.frosted_glass {
effects::frosted_glass(&mut photon_image);
}
// Adjust box_blur
if args.box_blur {
conv::box_blur(&mut photon_image);
}
// Adjust grayscale
if args.grayscale {
monochrome::grayscale(&mut photon_image);
}
// Adjust identity
if args.identity {
conv::identity(&mut photon_image);
}
// Adjust laplace
if args.laplace {
conv::laplace(&mut photon_image);
}
// Adjust cali
if args.cali {
filters::cali(&mut photon_image);
}
// Adjust dramatic
if args.dramatic {
filters::dramatic(&mut photon_image);
}
// Adjust firenze
if args.firenze {
filters::firenze(&mut photon_image);
}
// Adjust golden
if args.golden {
filters::golden(&mut photon_image);
}
// Adjust lix
if args.lix {
filters::lix(&mut photon_image);
}
// Adjust lofi
if args.lofi {
filters::lofi(&mut photon_image);
}
// Adjust neue
if args.neue {
filters::neue(&mut photon_image);
}
// Adjust obsidian
if args.obsidian {
filters::obsidian(&mut photon_image);
}
// Adjust pastel_pink
if args.pastel_pink {
filters::pastel_pink(&mut photon_image);
}
// Adjust ryo
if args.ryo {
filters::ryo(&mut photon_image);
}
// Adjust oil
match &args.oil {
Some(oil) => {
// split oil at comma
let vals: Vec<&str> = oil.split(",").collect();
// check if args.oil has 2 values
if vals.len() == 2 {
// convert oil values to i32 and f64
let radius: i32 = vals.get(0).unwrap().parse::<i32>().unwrap();
let intensity: f64 = vals.get(1).unwrap().parse::<f64>().unwrap();
effects::oil(&mut photon_image, radius, intensity);
}
}
@ -216,3 +241,64 @@ pub fn apply_effects(
photon_image
}
pub fn apply_luma_effects(args: &args::Args, mut photon_image: PhotonImage) -> PhotonImage {
let (width, height) = calculate_dimensions(args, &photon_image);
if args.rotate != 0 {
photon_image = rotate(&photon_image, args.rotate);
}
if args.fliph {
fliph(&mut photon_image);
}
if args.flipv {
flipv(&mut photon_image);
}
photon_image = resize(&photon_image, width, height, match args.filter {
args::SamplingFilter::Nearest => SamplingFilter::Nearest,
args::SamplingFilter::Triangle => SamplingFilter::Triangle,
args::SamplingFilter::CatmullRom => SamplingFilter::CatmullRom,
args::SamplingFilter::Gaussian => SamplingFilter::Gaussian,
args::SamplingFilter::Lanczos3 => SamplingFilter::Lanczos3,
});
type ColourFunc = fn(&mut PhotonImage, &str, f32);
let colour_func: ColourFunc = match args.colorspace {
args::ColourSpace::HSL => colour_spaces::hsl,
args::ColourSpace::HSV => colour_spaces::hsv,
args::ColourSpace::HSLUV => colour_spaces::hsluv,
args::ColourSpace::LCH => colour_spaces::lch,
};
if args.luma_invert {
channels::invert(&mut photon_image);
}
if args.luma_contrast != 0.0 {
effects::adjust_contrast(&mut photon_image, args.luma_contrast);
}
if args.luma_gamma != 0.0 {
let gamma_value = 1.0 - args.luma_gamma/255.0;
colour_spaces::gamma_correction(&mut photon_image, gamma_value, gamma_value, gamma_value);
}
if args.luma_brightness > 0.0 {
colour_func(&mut photon_image, "lighten", args.luma_brightness/100.0);
} else if args.luma_brightness < 0.0 {
colour_func(&mut photon_image, "darken", args.luma_brightness.abs()/100.0);
}
if args.luma_saturation < 0.0 {
colour_func(&mut photon_image, "saturate", args.luma_saturation.abs()/100.0);
} else if args.luma_saturation > 0.0 {
colour_func(&mut photon_image, "desaturate", args.luma_saturation/100.0);
}
photon_image
}

View File

@ -5,36 +5,37 @@ mod effects;
use reqwest;
use url::Url;
use photon_rs::PhotonImage;
use std::{error::Error, io::Cursor, process::exit};
#[tokio::main]
async fn main() {
let args = args::parse_args();
match load_image_from_url_or_path(args.image.as_str()).await {
Ok(mut image) => {
image = effects::apply_effects(
&args,
image,
);
Ok(image) => {
let image_luma = effects::apply_luma_effects(&args, image.clone());
let image_chroma = effects::apply_effects(&args, image.clone());
let canvas = draw::AnsiImage::new(image);
let canvas_luma = draw::AnsiImage::new(image_luma.clone());
let canvas_chroma = draw::AnsiImage::new(image_chroma.clone());
match (args.irc, args.ansi, args.ansi24, args.qb) {
(true, _, _, true) => println!("{}", draw::irc_draw_qb(canvas, &args).as_str()),
(true, _, _, false) => println!("{}", draw::irc_draw(canvas, &args).as_str()),
(_, true, _, true) => println!("{}", draw::ansi_draw_8bit_qb(canvas, &args).as_str()),
(_, true, _, false) => println!("{}", draw::ansi_draw_8bit(canvas, &args).as_str()),
(_, _, true, true) => println!("{}", draw::ansi_draw_24bit_qb(canvas).as_str()),
(_, _, true, false) => println!("{}", draw::ansi_draw_24bit(canvas).as_str()),
(_, _, _, true) => println!("{}", draw::irc_draw_qb(canvas, &args).as_str()),
_ => println!("{}", draw::irc_draw(canvas, &args).as_str()),
match (args.irc, args.ansi, args.ansi24, args.qb, args.braille) {
(true, _, _, true, false) => println!("{}", draw::irc_draw_qb(&canvas_chroma, &args)),
(true, _, _, false, false) => println!("{}", draw::irc_draw(&canvas_chroma, &args)),
(true, _, _, _, true) => println!("{}", draw::irc_draw_braille(&canvas_luma, &canvas_chroma, &args)),
(_, true, _, true, false) => println!("{}", draw::ansi_draw_8bit_qb(&canvas_chroma, &args)),
(_, true, _, false, false) => println!("{}", draw::ansi_draw_8bit(&canvas_chroma, &args)),
(_, true, _, _, true) => println!("{}", draw::ansi_draw_braille_8bit(&canvas_luma, &canvas_chroma, &args)),
(_, _, true, true, false) => println!("{}", draw::ansi_draw_24bit_qb(&canvas_chroma)),
(_, _, true, false, false) => println!("{}", draw::ansi_draw_24bit(&canvas_chroma)),
(_, _, true, _, true) => println!("{}", draw::ansi_draw_braille_24bit(&canvas_luma, &canvas_chroma)),
(_, _, _, true, false) => println!("{}", draw::irc_draw_qb(&canvas_chroma, &args)),
(_, _, _, _, true) => println!("{}", draw::irc_draw_braille(&canvas_luma, &canvas_chroma, &args)),
_ => println!("{}", draw::irc_draw(&canvas_chroma, &args)),
}
}
Err(e) => {
eprintln!("Error: {}", e);
exit(1);
@ -47,7 +48,6 @@ async fn load_image_from_url_or_path(image: &str) -> Result<PhotonImage, Box<dyn
Ok(url) => {
let response = reqwest::get(url).await?;
let bytes = response.bytes().await?;
let image_data = Cursor::new(bytes);
match photon_rs::native::open_image_from_bytes(image_data.into_inner().as_ref()) {
Ok(image) => Ok(image),
@ -62,4 +62,3 @@ async fn load_image_from_url_or_path(image: &str) -> Result<PhotonImage, Box<dyn
}
}
}