use std::ptr;
use dom_struct::dom_struct;
use js::gc::CustomAutoRooterGuard;
use js::jsapi::JSObject;
use js::rust::HandleObject;
use js::typedarray;
use js::typedarray::Uint8Array;
use crate::dom::bindings::buffer_source::create_buffer_source;
use crate::dom::bindings::codegen::Bindings::TextEncoderBinding::{
TextEncoderEncodeIntoResult, TextEncoderMethods,
};
use crate::dom::bindings::error::Fallible;
use crate::dom::bindings::reflector::{reflect_dom_object_with_proto, Reflector};
use crate::dom::bindings::root::DomRoot;
use crate::dom::bindings::str::{DOMString, USVString};
use crate::dom::globalscope::GlobalScope;
use crate::script_runtime::{CanGc, JSContext};
#[dom_struct]
pub struct TextEncoder {
reflector_: Reflector,
}
impl TextEncoder {
fn new_inherited() -> TextEncoder {
TextEncoder {
reflector_: Reflector::new(),
}
}
fn new(
global: &GlobalScope,
proto: Option<HandleObject>,
can_gc: CanGc,
) -> DomRoot<TextEncoder> {
reflect_dom_object_with_proto(
Box::new(TextEncoder::new_inherited()),
global,
proto,
can_gc,
)
}
#[allow(non_snake_case)]
pub fn Constructor(
global: &GlobalScope,
proto: Option<HandleObject>,
can_gc: CanGc,
) -> Fallible<DomRoot<TextEncoder>> {
Ok(TextEncoder::new(global, proto, can_gc))
}
}
impl TextEncoderMethods for TextEncoder {
fn Encoding(&self) -> DOMString {
DOMString::from("utf-8")
}
fn Encode(&self, cx: JSContext, input: USVString) -> Uint8Array {
let encoded = input.0.as_bytes();
rooted!(in(*cx) let mut js_object = ptr::null_mut::<JSObject>());
create_buffer_source(cx, encoded, js_object.handle_mut())
.expect("Converting input to uint8 array should never fail")
}
#[allow(unsafe_code)]
fn EncodeInto(
&self,
source: USVString,
mut destination: CustomAutoRooterGuard<typedarray::Uint8Array>,
) -> TextEncoderEncodeIntoResult {
let available = destination.len();
if available == 0 {
return TextEncoderEncodeIntoResult {
read: Some(0),
written: Some(0),
};
}
let mut read = 0;
let mut written = 0;
let dest = unsafe { destination.as_mut_slice() };
for result in source.0.chars() {
let utf8_len = result.len_utf8();
if available - written >= utf8_len {
read += if result > '\u{FFFF}' { 2 } else { 1 };
let target = &mut dest[written..written + utf8_len];
result.encode_utf8(target);
written += utf8_len;
} else {
break;
}
}
TextEncoderEncodeIntoResult {
read: Some(read),
written: Some(written as _),
}
}
}