#[feature(proc_macro)]
#[macro_use]
extern crate syn;
#[macro_use]
extern crate quote;
#[macro_use]
extern crate lazy_static;
extern crate proc_macro;
use syn::Expr;
use proc_macro::TokenStream;
use proc_macro::TokenTree;
use std::env;
use std::fs;
use std::fs::File;
use std::fs::OpenOptions;
use std::io::Write;
use std::path::Path;
use std::sync::Mutex;
lazy_static! {
static ref CSS_COUNTER: Mutex<u32> = Mutex::new(0);
}
#[proc_macro]
pub fn css(input: TokenStream) -> TokenStream {
let mut css_counter = CSS_COUNTER.lock().unwrap();
let class = format!("_css_rs_{}", css_counter);
let css_file = env::vars().find(|(key, _)| key == "OUTPUT_CSS");
if css_file.is_some() {
let css_file = css_file.unwrap().1;
if *css_counter == 0 {
if Path::new(&css_file).exists() {
fs::remove_file(&css_file);
}
let mut css_file = OpenOptions::new()
.write(true)
.create_new(true)
.open(css_file)
.unwrap();
write_css_to_file(&mut css_file, &class, input);
} else {
let mut css_file = OpenOptions::new().append(true).open(css_file).unwrap();
write_css_to_file(&mut css_file, &class, input);
}
}
*css_counter += 1;
let expanded = quote! {
#class
};
expanded.into()
}
fn write_css_to_file(css_file: &mut File, class: &str, input: TokenStream) {
for css in input.into_iter() {
let mut css = css.to_string();
let first_quote_mark = css.find(r#"""#).unwrap();
let last_quote_mark = css.rfind(r#"""#).unwrap();
css.truncate(last_quote_mark);
let mut css = css.split_off(first_quote_mark + 1);
let css = css.replace(":host", &format!(".{}", class));
css_file.write(&css.into_bytes()).unwrap();
css_file.write("\n".as_bytes()).unwrap();
}
}