ruffle/core/src/string/avm.rs

119 lines
3.1 KiB
Rust

use gc_arena::{Collect, Gc, MutationContext};
use std::borrow::Cow;
use super::{BorrowWStr, WStr, WString};
#[derive(Clone, Copy, Collect)]
#[collect(no_drop)]
enum Source<'gc> {
Owned(Gc<'gc, WString>),
Static(WStr<'static>),
}
#[derive(Clone, Copy, Collect)]
#[collect(no_drop)]
pub struct AvmString<'gc> {
source: Source<'gc>,
}
impl<'gc> AvmString<'gc> {
pub fn new_utf8<'s, S: Into<Cow<'s, str>>>(
gc_context: MutationContext<'gc, '_>,
string: S,
) -> Self {
let buf = match string.into() {
Cow::Owned(utf8) => WString::from_utf8_owned(utf8),
Cow::Borrowed(utf8) => WString::from_utf8(utf8),
};
Self {
source: Source::Owned(Gc::allocate(gc_context, buf)),
}
}
pub fn new_utf8_bytes<'b, B: Into<Cow<'b, [u8]>>>(
gc_context: MutationContext<'gc, '_>,
bytes: B,
) -> Result<Self, std::str::Utf8Error> {
let utf8 = match bytes.into() {
Cow::Owned(b) => Cow::Owned(String::from_utf8(b).map_err(|e| e.utf8_error())?),
Cow::Borrowed(b) => Cow::Borrowed(std::str::from_utf8(b)?),
};
Ok(Self::new_utf8(gc_context, utf8))
}
pub fn new<S: Into<WString>>(gc_context: MutationContext<'gc, '_>, string: S) -> Self {
Self {
source: Source::Owned(Gc::allocate(gc_context, string.into())),
}
}
pub fn concat(
gc_context: MutationContext<'gc, '_>,
left: AvmString<'gc>,
right: AvmString<'gc>,
) -> AvmString<'gc> {
if left.is_empty() {
right
} else if right.is_empty() {
left
} else {
let mut out = WString::from(left.borrow());
out.push_str(right.borrow());
Self::new(gc_context, out)
}
}
#[inline]
pub fn ptr_eq(this: &Self, other: &Self) -> bool {
match (this.source, other.source) {
(Source::Owned(this), Source::Owned(other)) => Gc::ptr_eq(this, other),
(Source::Static(this), Source::Static(other)) => this.to_ptr() == other.to_ptr(),
_ => false,
}
}
impl_str_methods! {
lifetime: '_;
self: &Self;
deref: self.borrow();
pattern['a,]: 'a, &'a Self;
}
}
impl Default for AvmString<'_> {
fn default() -> Self {
Self {
source: Source::Static(WStr::empty()),
}
}
}
impl<'gc> From<&'static str> for AvmString<'gc> {
#[inline]
fn from(str: &'static str) -> Self {
// TODO(moulins): actually check that `str` is valid ASCII.
Self {
source: Source::Static(WStr::from_units(str.as_bytes())),
}
}
}
impl<'gc> From<WStr<'static>> for AvmString<'gc> {
#[inline]
fn from(str: WStr<'static>) -> Self {
Self {
source: Source::Static(str),
}
}
}
impl<'gc> BorrowWStr for AvmString<'gc> {
#[inline]
fn borrow(&self) -> WStr<'_> {
match &self.source {
Source::Owned(s) => s.borrow(),
Source::Static(s) => *s,
}
}
}