rhai/src/packages/string_basic.rs

405 lines
14 KiB
Rust
Raw Normal View History

2020-08-15 06:57:47 +02:00
use crate::plugin::*;
2021-06-16 17:49:18 +02:00
use crate::{def_package, FnPtr, INT};
2022-04-09 07:37:43 +02:00
use std::any::TypeId;
2021-06-16 17:49:18 +02:00
use std::fmt::{Binary, LowerHex, Octal};
2021-04-17 09:15:54 +02:00
#[cfg(feature = "no_std")]
use std::prelude::v1::*;
#[cfg(not(feature = "no_index"))]
2020-11-16 09:28:04 +01:00
use crate::Array;
#[cfg(not(feature = "no_object"))]
2020-11-16 09:28:04 +01:00
use crate::Map;
2021-07-24 08:11:16 +02:00
pub const FUNC_TO_STRING: &str = "to_string";
pub const FUNC_TO_DEBUG: &str = "to_debug";
2020-11-30 04:20:51 +01:00
2021-12-20 04:42:39 +01:00
def_package! {
/// Package of basic string utilities (e.g. printing)
2022-02-10 05:33:48 +01:00
pub BasicStringPackage(lib) {
2021-12-20 04:42:39 +01:00
lib.standard = true;
2021-12-20 04:42:39 +01:00
combine_with_exported_module!(lib, "print_debug", print_debug_functions);
combine_with_exported_module!(lib, "number_formatting", number_formatting);
2022-04-09 07:37:43 +02:00
// Register characters iterator
#[cfg(not(feature = "no_index"))]
lib.set_iter(TypeId::of::<ImmutableString>(), |value| Box::new(
value.cast::<ImmutableString>().chars().map(Into::into).collect::<crate::Array>().into_iter()
));
2021-12-20 04:42:39 +01:00
}
}
2020-08-15 06:57:47 +02:00
2020-11-30 04:20:51 +01:00
// Register print and debug
#[inline]
pub fn print_with_func(
2021-03-31 04:16:38 +02:00
fn_name: &str,
ctx: &NativeCallContext,
value: &mut Dynamic,
) -> crate::ImmutableString {
match ctx.call_fn_raw(fn_name, true, false, &mut [value]) {
2021-11-13 05:23:35 +01:00
Ok(result) if result.is::<crate::ImmutableString>() => {
result.into_immutable_string().expect("`ImmutableString`")
}
2020-11-30 04:20:51 +01:00
Ok(result) => ctx.engine().map_type_name(result.type_name()).into(),
Err(_) => ctx.engine().map_type_name(value.type_name()).into(),
}
}
2020-08-15 06:57:47 +02:00
2020-10-19 17:49:01 +02:00
#[export_module]
mod print_debug_functions {
use crate::ImmutableString;
/// Convert the value of the `item` into a string.
#[rhai_fn(name = "print", pure)]
pub fn print_generic(ctx: NativeCallContext, item: &mut Dynamic) -> ImmutableString {
print_with_func(FUNC_TO_STRING, &ctx, item)
}
/// Convert the value of the `item` into a string.
#[rhai_fn(name = "to_string", pure)]
pub fn to_string_generic(ctx: NativeCallContext, item: &mut Dynamic) -> ImmutableString {
ctx.engine().map_type_name(&item.to_string()).into()
}
/// Convert the value of the `item` into a string in debug format.
#[rhai_fn(name = "debug", pure)]
pub fn debug_generic(ctx: NativeCallContext, item: &mut Dynamic) -> ImmutableString {
print_with_func(FUNC_TO_DEBUG, &ctx, item)
}
/// Convert the value of the `item` into a string in debug format.
#[rhai_fn(name = "to_debug", pure)]
pub fn to_debug_generic(ctx: NativeCallContext, item: &mut Dynamic) -> ImmutableString {
2022-08-11 13:01:23 +02:00
ctx.engine().map_type_name(&format!("{item:?}")).into()
}
/// Return the empty string.
2020-10-19 17:49:01 +02:00
#[rhai_fn(name = "print", name = "debug")]
pub fn print_empty_string(ctx: NativeCallContext) -> ImmutableString {
2022-08-12 10:34:57 +02:00
ctx.engine().get_interned_string("")
2020-10-19 17:49:01 +02:00
}
/// Return the `string`.
2020-10-19 17:49:01 +02:00
#[rhai_fn(name = "print", name = "to_string")]
pub fn print_string(string: ImmutableString) -> ImmutableString {
string
2020-10-19 17:49:01 +02:00
}
/// Convert the string into debug format.
#[rhai_fn(name = "debug", name = "to_debug", pure)]
pub fn debug_string(string: &mut ImmutableString) -> ImmutableString {
2022-08-11 13:01:23 +02:00
format!("{string:?}").into()
}
/// Return the character into a string.
#[rhai_fn(name = "print", name = "to_string")]
pub fn print_char(character: char) -> ImmutableString {
character.to_string().into()
}
/// Convert the string into debug format.
#[rhai_fn(name = "debug", name = "to_debug")]
pub fn debug_char(character: char) -> ImmutableString {
2022-08-11 13:01:23 +02:00
format!("{character:?}").into()
}
/// Convert the function pointer into a string in debug format.
2021-02-24 04:05:39 +01:00
#[rhai_fn(name = "debug", name = "to_debug", pure)]
2020-10-19 17:49:01 +02:00
pub fn debug_fn_ptr(f: &mut FnPtr) -> ImmutableString {
f.to_string().into()
}
/// Return the boolean value into a string.
#[rhai_fn(name = "print", name = "to_string")]
pub fn print_bool(value: bool) -> ImmutableString {
2022-08-11 13:01:23 +02:00
value.to_string().into()
}
/// Convert the boolean value into a string in debug format.
#[rhai_fn(name = "debug", name = "to_debug")]
pub fn debug_bool(value: bool) -> ImmutableString {
2022-08-11 13:01:23 +02:00
format!("{value:?}").into()
}
/// Return the empty string.
#[rhai_fn(name = "print", name = "to_string")]
pub fn print_unit(ctx: NativeCallContext, unit: ()) -> ImmutableString {
let _ = unit;
2022-08-12 10:34:57 +02:00
ctx.engine().get_interned_string("")
}
/// Convert the unit into a string in debug format.
#[rhai_fn(name = "debug", name = "to_debug")]
pub fn debug_unit(unit: ()) -> ImmutableString {
let _ = unit;
"()".into()
}
/// Convert the value of `number` into a string.
#[cfg(not(feature = "no_float"))]
2021-10-20 10:22:12 +02:00
#[rhai_fn(name = "print", name = "to_string")]
pub fn print_f64(number: f64) -> ImmutableString {
crate::ast::FloatWrapper::new(number).to_string().into()
}
/// Convert the value of `number` into a string.
2021-10-20 10:22:12 +02:00
#[cfg(not(feature = "no_float"))]
#[rhai_fn(name = "print", name = "to_string")]
pub fn print_f32(number: f32) -> ImmutableString {
crate::ast::FloatWrapper::new(number).to_string().into()
}
/// Convert the value of `number` into a string.
2021-10-20 10:22:12 +02:00
#[cfg(not(feature = "no_float"))]
#[rhai_fn(name = "debug", name = "to_debug")]
pub fn debug_f64(number: f64) -> ImmutableString {
2022-08-11 13:01:23 +02:00
let number = crate::ast::FloatWrapper::new(number);
format!("{number:?}").into()
2021-10-20 10:22:12 +02:00
}
/// Convert the value of `number` into a string.
2021-10-20 10:22:12 +02:00
#[cfg(not(feature = "no_float"))]
#[rhai_fn(name = "debug", name = "to_debug")]
pub fn debug_f32(number: f32) -> ImmutableString {
2022-08-11 13:01:23 +02:00
let number = crate::ast::FloatWrapper::new(number);
format!("{number:?}").into()
2020-10-19 17:49:01 +02:00
}
2020-09-23 06:00:03 +02:00
/// Convert the array into a string.
2020-11-30 04:20:51 +01:00
#[cfg(not(feature = "no_index"))]
2021-10-20 10:22:12 +02:00
#[rhai_fn(
name = "print",
name = "to_string",
name = "debug",
name = "to_debug",
pure
)]
pub fn format_array(ctx: NativeCallContext, array: &mut Array) -> ImmutableString {
let len = array.len();
let mut result = String::with_capacity(len * 5 + 2);
result.push('[');
array.iter_mut().enumerate().for_each(|(i, x)| {
result.push_str(&print_with_func(FUNC_TO_DEBUG, &ctx, x));
if i < len - 1 {
result.push_str(", ");
}
});
result.push(']');
result.into()
2020-11-30 04:20:51 +01:00
}
/// Convert the object map into a string.
2020-10-19 17:49:01 +02:00
#[cfg(not(feature = "no_object"))]
2021-10-20 10:22:12 +02:00
#[rhai_fn(
name = "print",
name = "to_string",
name = "debug",
name = "to_debug",
pure
)]
pub fn format_map(ctx: NativeCallContext, map: &mut Map) -> ImmutableString {
let len = map.len();
let mut result = String::with_capacity(len * 5 + 3);
result.push_str("#{");
map.iter_mut().enumerate().for_each(|(i, (k, v))| {
2022-05-07 10:29:20 +02:00
use std::fmt::Write;
write!(
result,
2021-10-20 10:22:12 +02:00
"{:?}: {}{}",
k,
&print_with_func(FUNC_TO_DEBUG, &ctx, v),
if i < len - 1 { ", " } else { "" }
2022-05-07 10:29:20 +02:00
)
.unwrap();
2021-10-20 10:22:12 +02:00
});
result.push('}');
result.into()
}
2020-08-15 06:57:47 +02:00
}
2021-06-16 17:49:18 +02:00
#[export_module]
mod number_formatting {
fn to_hex<T: LowerHex>(value: T) -> ImmutableString {
2022-08-11 13:01:23 +02:00
format!("{value:x}").into()
2021-06-16 17:49:18 +02:00
}
fn to_octal<T: Octal>(value: T) -> ImmutableString {
2022-08-11 13:01:23 +02:00
format!("{value:o}").into()
2021-06-16 17:49:18 +02:00
}
fn to_binary<T: Binary>(value: T) -> ImmutableString {
2022-08-11 13:01:23 +02:00
format!("{value:b}").into()
2021-06-16 17:49:18 +02:00
}
/// Convert the `value` into a string in hex format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_hex")]
pub fn int_to_hex(value: INT) -> ImmutableString {
to_hex(value)
}
/// Convert the `value` into a string in octal format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_octal")]
pub fn int_to_octal(value: INT) -> ImmutableString {
to_octal(value)
}
/// Convert the `value` into a string in binary format.
2021-07-04 10:51:05 +02:00
#[rhai_fn(name = "to_binary")]
pub fn int_to_binary(value: INT) -> ImmutableString {
to_binary(value)
}
2021-06-16 17:49:18 +02:00
#[cfg(not(feature = "only_i32"))]
#[cfg(not(feature = "only_i64"))]
pub mod numbers {
/// Convert the `value` into a string in hex format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_hex")]
pub fn u8_to_hex(value: u8) -> ImmutableString {
to_hex(value)
}
/// Convert the `value` into a string in hex format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_hex")]
pub fn u16_to_hex(value: u16) -> ImmutableString {
to_hex(value)
}
/// Convert the `value` into a string in hex format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_hex")]
pub fn u32_to_hex(value: u32) -> ImmutableString {
to_hex(value)
}
/// Convert the `value` into a string in hex format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_hex")]
pub fn u64_to_hex(value: u64) -> ImmutableString {
to_hex(value)
}
/// Convert the `value` into a string in hex format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_hex")]
pub fn i8_to_hex(value: i8) -> ImmutableString {
to_hex(value)
}
/// Convert the `value` into a string in hex format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_hex")]
pub fn i16_to_hex(value: i16) -> ImmutableString {
to_hex(value)
}
/// Convert the `value` into a string in hex format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_hex")]
pub fn i32_to_hex(value: i32) -> ImmutableString {
to_hex(value)
}
/// Convert the `value` into a string in hex format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_hex")]
pub fn i64_to_hex(value: i64) -> ImmutableString {
to_hex(value)
}
/// Convert the `value` into a string in octal format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_octal")]
pub fn u8_to_octal(value: u8) -> ImmutableString {
to_octal(value)
}
/// Convert the `value` into a string in octal format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_octal")]
pub fn u16_to_octal(value: u16) -> ImmutableString {
to_octal(value)
}
/// Convert the `value` into a string in octal format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_octal")]
pub fn u32_to_octal(value: u32) -> ImmutableString {
to_octal(value)
}
/// Convert the `value` into a string in octal format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_octal")]
pub fn u64_to_octal(value: u64) -> ImmutableString {
to_octal(value)
}
/// Convert the `value` into a string in octal format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_octal")]
pub fn i8_to_octal(value: i8) -> ImmutableString {
to_octal(value)
}
/// Convert the `value` into a string in octal format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_octal")]
pub fn i16_to_octal(value: i16) -> ImmutableString {
to_octal(value)
}
/// Convert the `value` into a string in octal format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_octal")]
pub fn i32_to_octal(value: i32) -> ImmutableString {
to_octal(value)
}
/// Convert the `value` into a string in octal format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_octal")]
pub fn i64_to_octal(value: i64) -> ImmutableString {
to_octal(value)
}
/// Convert the `value` into a string in binary format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_binary")]
pub fn u8_to_binary(value: u8) -> ImmutableString {
to_binary(value)
}
/// Convert the `value` into a string in binary format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_binary")]
pub fn u16_to_binary(value: u16) -> ImmutableString {
to_binary(value)
}
/// Convert the `value` into a string in binary format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_binary")]
pub fn u32_to_binary(value: u32) -> ImmutableString {
to_binary(value)
}
/// Convert the `value` into a string in binary format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_binary")]
pub fn u64_to_binary(value: u64) -> ImmutableString {
to_binary(value)
}
/// Convert the `value` into a string in binary format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_binary")]
pub fn i8_to_binary(value: i8) -> ImmutableString {
to_binary(value)
}
/// Convert the `value` into a string in binary format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_binary")]
pub fn i16_to_binary(value: i16) -> ImmutableString {
to_binary(value)
}
/// Convert the `value` into a string in binary format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_binary")]
pub fn i32_to_binary(value: i32) -> ImmutableString {
to_binary(value)
}
/// Convert the `value` into a string in binary format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_binary")]
pub fn i64_to_binary(value: i64) -> ImmutableString {
to_binary(value)
}
2022-01-12 01:12:28 +01:00
#[cfg(not(target_family = "wasm"))]
2021-06-16 17:49:18 +02:00
pub mod num_128 {
/// Convert the `value` into a string in hex format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_hex")]
pub fn u128_to_hex(value: u128) -> ImmutableString {
to_hex(value)
}
/// Convert the `value` into a string in hex format.
#[rhai_fn(name = "to_hex")]
pub fn i128_to_hex(value: i128) -> ImmutableString {
to_hex(value)
}
/// Convert the `value` into a string in octal format.
#[rhai_fn(name = "to_octal")]
pub fn u128_to_octal(value: u128) -> ImmutableString {
to_octal(value)
}
/// Convert the `value` into a string in octal format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_octal")]
pub fn i128_to_octal(value: i128) -> ImmutableString {
to_octal(value)
}
/// Convert the `value` into a string in binary format.
#[rhai_fn(name = "to_binary")]
pub fn u128_to_binary(value: u128) -> ImmutableString {
to_binary(value)
}
/// Convert the `value` into a string in binary format.
2021-06-16 17:49:18 +02:00
#[rhai_fn(name = "to_binary")]
pub fn i128_to_binary(value: i128) -> ImmutableString {
to_binary(value)
}
}
}
}