1
1
Fork 0
mirror of https://github.com/lune-org/lune.git synced 2025-04-04 10:30:54 +01:00
lune/crates/lune-utils/src/fmt/value/basic.rs

99 lines
3.5 KiB
Rust

use mlua::prelude::*;
use crate::fmt::ErrorComponents;
use super::{
metamethods::{
call_table_tostring_metamethod, call_userdata_tostring_metamethod,
get_table_type_metavalue, get_userdata_type_metavalue,
},
style::{COLOR_CYAN, COLOR_GREEN, COLOR_MAGENTA, COLOR_YELLOW},
};
const STRING_REPLACEMENTS: &[(&str, &str)] =
&[("\"", r#"\""#), ("\t", r"\t"), ("\r", r"\r"), ("\n", r"\n")];
/**
Tries to return the given value as a plain string key.
A plain string key must:
- Start with an alphabetic character.
- Only contain alphanumeric characters and underscores.
*/
pub(crate) fn lua_value_as_plain_string_key(value: &LuaValue) -> Option<String> {
if let LuaValue::String(s) = value {
if let Ok(s) = s.to_str() {
let first_valid = s.chars().next().is_some_and(|c| c.is_ascii_alphabetic());
let all_valid = s.chars().all(|c| c.is_ascii_alphanumeric() || c == '_');
if first_valid && all_valid {
return Some(s.to_string());
}
}
}
None
}
/**
Formats a Lua value into a pretty string.
This does not recursively format tables.
*/
pub(crate) fn format_value_styled(value: &LuaValue, prefer_plain: bool) -> String {
match value {
LuaValue::Nil => COLOR_YELLOW.apply_to("nil").to_string(),
LuaValue::Boolean(true) => COLOR_YELLOW.apply_to("true").to_string(),
LuaValue::Boolean(false) => COLOR_YELLOW.apply_to("false").to_string(),
LuaValue::Number(n) => COLOR_CYAN.apply_to(n).to_string(),
LuaValue::Integer(i) => COLOR_CYAN.apply_to(i).to_string(),
LuaValue::String(s) if prefer_plain => s.to_string_lossy().to_string(),
LuaValue::String(s) => COLOR_GREEN
.apply_to({
let mut s = s.to_string_lossy().to_string();
for (from, to) in STRING_REPLACEMENTS {
s = s.replace(from, to);
}
format!(r#""{s}""#)
})
.to_string(),
LuaValue::Vector(_) => COLOR_MAGENTA.apply_to("<vector>").to_string(),
LuaValue::Thread(_) => COLOR_MAGENTA.apply_to("<thread>").to_string(),
LuaValue::Function(_) => COLOR_MAGENTA.apply_to("<function>").to_string(),
LuaValue::LightUserData(_) => COLOR_MAGENTA.apply_to("<pointer>").to_string(),
LuaValue::UserData(u) => {
let formatted = format_typename_and_tostringed(
"userdata",
get_userdata_type_metavalue(u),
call_userdata_tostring_metamethod(u),
);
COLOR_MAGENTA.apply_to(formatted).to_string()
}
LuaValue::Table(t) => {
let formatted = format_typename_and_tostringed(
"table",
get_table_type_metavalue(t),
call_table_tostring_metamethod(t),
);
COLOR_MAGENTA.apply_to(formatted).to_string()
}
LuaValue::Error(e) => COLOR_MAGENTA
.apply_to(format!(
"<LuaError(\n{})>",
ErrorComponents::from(e.clone())
))
.to_string(),
}
}
fn format_typename_and_tostringed(
fallback: &'static str,
typename: Option<String>,
tostringed: Option<String>,
) -> String {
match (typename, tostringed) {
(Some(typename), Some(tostringed)) => format!("<{typename}({tostringed})>"),
(Some(typename), None) => format!("<{typename}>"),
(None, Some(tostringed)) => format!("<{tostringed}>"),
(None, None) => format!("<{fallback}>"),
}
}