2024-07-26 20:40:07 +00:00
|
|
|
use syn::{Expr, Lit, Meta};
|
|
|
|
|
|
|
|
pub(crate) fn get_named_string(args: &[Meta], name: &str) -> Option<String> {
|
|
|
|
args.iter().find_map(|arg| {
|
|
|
|
let value = arg.require_name_value().ok()?;
|
|
|
|
let Expr::Lit(ref lit) = value.value else {
|
|
|
|
return None;
|
|
|
|
};
|
|
|
|
let Lit::Str(ref str) = lit.lit else {
|
|
|
|
return None;
|
|
|
|
};
|
|
|
|
value.path.is_ident(name).then_some(str.value())
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-07-24 03:55:01 +00:00
|
|
|
#[must_use]
|
|
|
|
pub(crate) fn camel_to_snake_string(s: &str) -> String {
|
|
|
|
let mut output = String::with_capacity(
|
|
|
|
s.chars()
|
|
|
|
.fold(s.len(), |a, ch| a.saturating_add(usize::from(ch.is_ascii_uppercase()))),
|
|
|
|
);
|
|
|
|
|
|
|
|
let mut state = false;
|
|
|
|
s.chars().for_each(|ch| {
|
|
|
|
let m = ch.is_ascii_uppercase();
|
|
|
|
let s = exchange(&mut state, !m);
|
|
|
|
if m && s {
|
|
|
|
output.push('_');
|
|
|
|
}
|
|
|
|
output.push(ch.to_ascii_lowercase());
|
|
|
|
});
|
|
|
|
|
|
|
|
output
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(crate) fn exchange<T: Clone>(state: &mut T, source: T) -> T {
|
|
|
|
let ret = state.clone();
|
|
|
|
*state = source;
|
|
|
|
ret
|
|
|
|
}
|