the aliasing svg renderer
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
use anyhow::{anyhow, Result};
use clap::Parser;
use log::Level as RLevel;
use psvg::tree::Tree;
use std::{path::PathBuf, str::FromStr};
mod logger;

#[derive(Parser)]

/// PSVG: the curveless aliasing svg renderer
struct Args {
    #[arg(short = 's')]
    /// Specify the size of the output png.
    /// If not supplied, will render at the svg's set width and height.
    /// Specify as: '144x124'
    size: Option<Size>,
    /// Log level.
    #[arg(default_value = "info", long = "log")]
    log: Level,
    /// Svg to render.
    file: PathBuf,
    /// File to output rendered svg (png).
    out: PathBuf,
}

#[repr(usize)]
#[derive(clap::ValueEnum, Clone, Copy)]
enum Level {
    Error = 1,
    Warn,
    Info,
    Debug,
    Trace,
}

impl From<Level> for RLevel {
    fn from(value: Level) -> Self {
        // SAFETY: same
        unsafe { std::mem::transmute(value) }
    }
}

#[derive(Copy, Clone, Debug, Default)]
struct Size {
    w: u32,
    h: u32,
}

impl FromStr for Size {
    type Err = anyhow::Error;

    fn from_str(s: &str) -> Result<Self, Self::Err> {
        let (w, h) = s.split_once('x').ok_or(anyhow!(
            "please delimit width and height witih a 'x': 128x142"
        ))?;
        Ok(Size {
            w: w.parse()?,
            h: h.parse()?,
        })
    }
}

fn main() -> Result<()> {
    let args = Args::parse();
    logger::init(args.log.into());
    let (mut svg, size) = Tree::new(&std::fs::read_to_string(args.file)?)?;
    match args.size {
        Some(Size { w, h }) => {
            svg.resize(w as f32, h as f32);
        }
        None => svg.resize(size.width(), size.height()),
    }
    svg.render().save(args.out);
    Ok(())
}