#![deny(missing_docs)]
use log::{Level, Log, Metadata, Record};
use wasm_bindgen::prelude::*;
use web_sys::console;
pub struct Config {
level: Level,
module_prefix: Option<String>,
message_location: MessageLocation,
}
pub enum MessageLocation {
SameLine,
NewLine,
}
impl Default for Config {
fn default() -> Self {
Self {
level: Level::Debug,
module_prefix: None,
message_location: MessageLocation::SameLine,
}
}
}
impl Config {
pub fn new(level: Level) -> Self {
Self {
level,
module_prefix: None,
message_location: MessageLocation::SameLine,
}
}
pub fn module_prefix(mut self, module_prefix: &str) -> Self {
self.module_prefix = Some(module_prefix.to_string());
self
}
pub fn message_on_new_line(mut self) -> Self {
self.message_location = MessageLocation::NewLine;
self
}
}
struct Style {
lvl_trace: String,
lvl_debug: String,
lvl_info: String,
lvl_warn: String,
lvl_error: String,
tgt: String,
args: String,
}
impl Style {
fn new() -> Style {
let base = String::from("color: white; padding: 0 3px; background:");
Style {
lvl_trace: format!("{} gray;", base),
lvl_debug: format!("{} blue;", base),
lvl_info: format!("{} green;", base),
lvl_warn: format!("{} orange;", base),
lvl_error: format!("{} darkred;", base),
tgt: String::from("font-weight: bold; color: inherit"),
args: String::from("background: inherit; color: inherit"),
}
}
}
struct WasmLogger {
config: Config,
style: Style,
}
impl Log for WasmLogger {
fn enabled(&self, metadata: &Metadata<'_>) -> bool {
if let Some(ref prefix) = self.config.module_prefix {
metadata.target().starts_with(prefix)
} else {
true
}
}
fn log(&self, record: &Record<'_>) {
if self.enabled(record.metadata()) {
let style = &self.style;
let message_separator = match self.config.message_location {
MessageLocation::NewLine => "\n",
MessageLocation::SameLine => " ",
};
let s = format!(
"%c{}%c {}:{}%c{}{}",
record.level(),
record.file().unwrap_or_else(|| record.target()),
record
.line()
.map_or_else(|| "[Unknown]".to_string(), |line| line.to_string()),
message_separator,
record.args(),
);
let s = JsValue::from_str(&s);
let tgt_style = JsValue::from_str(&style.tgt);
let args_style = JsValue::from_str(&style.args);
match record.level() {
Level::Trace => console::debug_4(
&s,
&JsValue::from(&style.lvl_trace),
&tgt_style,
&args_style,
),
Level::Debug => console::log_4(
&s,
&JsValue::from(&style.lvl_debug),
&tgt_style,
&args_style,
),
Level::Info => {
console::info_4(&s, &JsValue::from(&style.lvl_info), &tgt_style, &args_style)
}
Level::Warn => {
console::warn_4(&s, &JsValue::from(&style.lvl_warn), &tgt_style, &args_style)
}
Level::Error => console::error_4(
&s,
&JsValue::from(&style.lvl_error),
&tgt_style,
&args_style,
),
}
}
}
fn flush(&self) {}
}
pub fn init(config: Config) {
let max_level = config.level;
let wl = WasmLogger {
config,
style: Style::new(),
};
match log::set_boxed_logger(Box::new(wl)) {
Ok(_) => log::set_max_level(max_level.to_level_filter()),
Err(e) => console::error_1(&JsValue::from(e.to_string())),
}
}