You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

246 lines
7.6 KiB

#[macro_use]
extern crate log;
use std::env;
use std::fs::File;
use std::io::{self, Read};
use std::path::PathBuf;
use g_code::parse::{ast::Snippet, lexer::Lexer, ParseError, SnippetParser};
use structopt::StructOpt;
/// Converts an SVG to GCode in an internal representation
mod converter;
/// Emulates the state of an arbitrary machine that can run GCode
mod machine;
/// Operations that are easier to implement after GCode is generated, or would
/// over-complicate SVG conversion
mod postprocess;
/// Provides an interface for drawing lines in GCode
/// This concept is referred to as [Turtle graphics](https://en.wikipedia.org/wiki/Turtle_graphics).
mod turtle;
use converter::ProgramOptions;
use machine::Machine;
#[derive(Debug, StructOpt)]
#[structopt(name = "svg2gcode", author, about)]
struct Opt {
/// Curve interpolation tolerance
#[structopt(long, default_value = "0.002")]
tolerance: f64,
/// Machine feed rate in mm/min
#[structopt(long, default_value = "300")]
feedrate: f64,
/// Dots per inch (DPI) for pixels, points, picas, etc.
#[structopt(long, default_value = "96")]
dpi: f64,
#[structopt(alias = "tool_on_sequence", long = "on")]
/// Tool on GCode sequence
tool_on_sequence: Option<String>,
#[structopt(alias = "tool_off_sequence", long = "off")]
/// Tool off GCode sequence
tool_off_sequence: Option<String>,
/// Optional GCode begin sequence (i.e. change to a cutter tool)
#[structopt(alias = "begin_sequence", long = "begin")]
begin_sequence: Option<String>,
/// Optional GCode end sequence, prior to program end (i.e. put away a cutter tool)
#[structopt(alias = "end_sequence", long = "end")]
end_sequence: Option<String>,
/// A file path for an SVG, else reads from stdin
file: Option<PathBuf>,
/// Output file path (overwrites old files), else writes to stdout
#[structopt(short, long)]
out: Option<PathBuf>,
/// Set where the bottom left corner of the SVG will be placed
#[structopt(long, default_value = "0,0")]
origin: String,
}
fn main() -> io::Result<()> {
if env::var("RUST_LOG").is_err() {
env::set_var("RUST_LOG", "svg2gcode=info")
}
env_logger::init();
let opt = Opt::from_args();
let input = match opt.file {
Some(filename) => {
let mut f = File::open(filename)?;
let len = f.metadata()?.len();
let mut input = String::with_capacity(len as usize + 1);
f.read_to_string(&mut input)?;
input
}
None => {
info!("Reading from stdin");
let mut input = String::new();
io::stdin().read_to_string(&mut input)?;
input
}
};
let mut options = ProgramOptions::default();
options.tolerance = opt.tolerance;
options.feedrate = opt.feedrate;
options.dpi = opt.dpi;
let snippets = [
opt.tool_on_sequence.as_ref().map(parse_snippet).transpose(),
opt.tool_off_sequence
.as_ref()
.map(parse_snippet)
.transpose(),
opt.begin_sequence.as_ref().map(parse_snippet).transpose(),
opt.end_sequence.as_ref().map(parse_snippet).transpose(),
];
let machine = if let [Ok(tool_on_action), Ok(tool_off_action), Ok(program_begin_sequence), Ok(program_end_sequence)] =
snippets
{
Machine {
tool_on_action,
tool_off_action,
program_begin_sequence,
program_end_sequence,
tool_state: None,
distance_mode: None,
}
} else {
use codespan_reporting::term::{
emit,
termcolor::{ColorChoice, StandardStream},
};
let mut writer = StandardStream::stderr(ColorChoice::Auto);
let config = codespan_reporting::term::Config::default();
for (i, (filename, gcode)) in [
("tool_on_sequence", &opt.tool_on_sequence),
("tool_off_sequence", &opt.tool_off_sequence),
("begin_sequence", &opt.begin_sequence),
("end_sequence", &opt.end_sequence),
]
.iter()
.enumerate()
{
if let Err(err) = &snippets[i] {
emit(
&mut writer,
&config,
&codespan_reporting::files::SimpleFile::new(filename, gcode.as_ref().unwrap()),
&g_code::parse::into_diagnostic(&err),
)
.unwrap();
}
}
std::process::exit(1)
};
let document = roxmltree::Document::parse(&input).expect("Invalid or unsupported SVG file");
let mut program = converter::svg2program(&document, options, machine);
let origin = opt
.origin
.split(',')
.map(|point| point.parse().expect("could not parse coordinate"))
.collect::<Vec<f64>>();
postprocess::set_origin(&mut program, lyon_geom::point(origin[0], origin[1]));
if let Some(out_path) = opt.out {
tokens_into_gcode(program, File::create(out_path)?)
} else {
tokens_into_gcode(program, std::io::stdout())
}
}
fn parse_snippet<'input>(gcode: &'input String) -> Result<Snippet<'input>, ParseError<'input>> {
SnippetParser::new().parse(gcode, Lexer::new(gcode))
}
fn tokens_into_gcode<W: std::io::Write>(
program: Vec<g_code::emit::Token>,
mut w: W,
) -> io::Result<()> {
use g_code::emit::Token::*;
let mut preceded_by_newline = true;
for token in program {
match token {
Field(f) => {
if !preceded_by_newline {
if matches!(f.letters.as_str(), "G" | "M") {
writeln!(w, "")?;
} else {
write!(w, " ")?;
}
}
write!(w, "{}", f)?;
preceded_by_newline = false;
}
Comment {
is_inline: true,
inner,
} => {
write!(w, "({})", inner)?;
preceded_by_newline = false;
}
Comment {
is_inline: false,
inner,
} => {
writeln!(w, ";{}", inner)?;
preceded_by_newline = true;
}
_ => {}
}
}
// Ensure presence of trailing newline
if !preceded_by_newline {
writeln!(w, "")?;
}
Ok(())
}
#[cfg(test)]
mod test {
use super::*;
use pretty_assertions::assert_eq;
fn get_actual(input: &str) -> String {
let options = ProgramOptions::default();
let machine = Machine {
tool_state: None,
distance_mode: None,
tool_on_action: None,
tool_off_action: None,
program_begin_sequence: None,
program_end_sequence: None,
};
let document = roxmltree::Document::parse(input).unwrap();
let mut program = converter::svg2program(&document, options, machine);
postprocess::set_origin(&mut program, lyon_geom::point(0., 0.));
let mut actual = vec![];
assert!(tokens_into_gcode(program, &mut actual).is_ok());
String::from_utf8(actual).unwrap()
}
#[test]
fn square_produces_expected_gcode() {
let square = include_str!("../tests/square.svg");
let actual = get_actual(square);
assert_eq!(actual, include_str!("../tests/square.gcode"))
}
#[test]
fn square_transformed_produces_expected_gcode() {
let square_transformed = include_str!("../tests/square_transformed.svg");
let actual = get_actual(square_transformed);
assert_eq!(actual, include_str!("../tests/square_transformed.gcode"))
}
}