forked from mirrors/gecko-dev
		
	This used to conflict with the util crate from the standard library, which has long since been removed. The import in layout has not been changed because of a conflict with the util mod there. Source-Repo: https://github.com/servo/servo Source-Revision: 27e0f16407629422b5e047e067d458142372c97e
		
			
				
	
	
		
			154 lines
		
	
	
	
		
			4.7 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			154 lines
		
	
	
	
		
			4.7 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
/* This Source Code Form is subject to the terms of the Mozilla Public
 | 
						|
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 | 
						|
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 | 
						|
 | 
						|
use dom::bindings::codegen::InheritTypes::FileDerived;
 | 
						|
use dom::bindings::global::{GlobalRef, GlobalField};
 | 
						|
use dom::bindings::js::{JSRef, Temporary};
 | 
						|
use dom::bindings::utils::{Reflector, reflect_dom_object};
 | 
						|
use dom::bindings::error::Fallible;
 | 
						|
use dom::bindings::codegen::Bindings::BlobBinding;
 | 
						|
use dom::bindings::codegen::Bindings::BlobBinding::BlobMethods;
 | 
						|
 | 
						|
use util::str::DOMString;
 | 
						|
 | 
						|
use std::ascii::AsciiExt;
 | 
						|
use std::borrow::ToOwned;
 | 
						|
use std::cmp::{min, max};
 | 
						|
use std::num::ToPrimitive;
 | 
						|
 | 
						|
#[jstraceable]
 | 
						|
pub enum BlobTypeId {
 | 
						|
    Blob,
 | 
						|
    File,
 | 
						|
}
 | 
						|
 | 
						|
#[dom_struct]
 | 
						|
pub struct Blob {
 | 
						|
    reflector_: Reflector,
 | 
						|
    type_: BlobTypeId,
 | 
						|
    bytes: Option<Vec<u8>>,
 | 
						|
    typeString: DOMString,
 | 
						|
    global: GlobalField
 | 
						|
    // isClosed_: bool
 | 
						|
}
 | 
						|
 | 
						|
fn is_ascii_printable(string: &DOMString) -> bool{
 | 
						|
    // Step 5.1 in Sec 5.1 of File API spec
 | 
						|
    // http://dev.w3.org/2006/webapi/FileAPI/#constructorBlob
 | 
						|
    return string.chars().all(|c| { c >= '\x20' && c <= '\x7E' })
 | 
						|
}
 | 
						|
 | 
						|
impl Blob {
 | 
						|
    pub fn new_inherited(global: GlobalRef, type_: BlobTypeId,
 | 
						|
                         bytes: Option<Vec<u8>>, typeString: &str) -> Blob {
 | 
						|
        Blob {
 | 
						|
            reflector_: Reflector::new(),
 | 
						|
            type_: type_,
 | 
						|
            bytes: bytes,
 | 
						|
            typeString: typeString.to_owned(),
 | 
						|
            global: GlobalField::from_rooted(&global)
 | 
						|
            //isClosed_: false
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    pub fn new(global: GlobalRef, bytes: Option<Vec<u8>>,
 | 
						|
               typeString: &str) -> Temporary<Blob> {
 | 
						|
        reflect_dom_object(box Blob::new_inherited(global, BlobTypeId::Blob, bytes, typeString),
 | 
						|
                           global,
 | 
						|
                           BlobBinding::Wrap)
 | 
						|
    }
 | 
						|
 | 
						|
    pub fn Constructor(global: GlobalRef) -> Fallible<Temporary<Blob>> {
 | 
						|
        Ok(Blob::new(global, None, ""))
 | 
						|
    }
 | 
						|
 | 
						|
    pub fn Constructor_(global: GlobalRef, blobParts: DOMString, blobPropertyBag: &BlobBinding::BlobPropertyBag) -> Fallible<Temporary<Blob>> {
 | 
						|
        //TODO: accept other blobParts types - ArrayBuffer or ArrayBufferView or Blob
 | 
						|
        let bytes: Option<Vec<u8>> = Some(blobParts.into_bytes());
 | 
						|
        let typeString = if is_ascii_printable(&blobPropertyBag.type_) {
 | 
						|
            blobPropertyBag.type_.as_slice()
 | 
						|
        } else {
 | 
						|
            ""
 | 
						|
        };
 | 
						|
        let typeStrLower = typeString.as_slice().to_ascii_lowercase();
 | 
						|
        Ok(Blob::new(global, bytes, typeStrLower.as_slice()))
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
impl<'a> BlobMethods for JSRef<'a, Blob> {
 | 
						|
    fn Size(self) -> u64{
 | 
						|
        match self.bytes {
 | 
						|
            None => 0,
 | 
						|
            Some(ref bytes) => bytes.len() as u64
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    fn Type(self) -> DOMString {
 | 
						|
        self.typeString.clone()
 | 
						|
    }
 | 
						|
 | 
						|
    fn Slice(self, start: Option<i64>, end: Option<i64>,
 | 
						|
             contentType: Option<DOMString>) -> Temporary<Blob> {
 | 
						|
        let size: i64 = self.Size().to_i64().unwrap();
 | 
						|
        let relativeStart: i64 = match start {
 | 
						|
            None => 0,
 | 
						|
            Some(start) => {
 | 
						|
                if start < 0 {
 | 
						|
                    max(size.to_i64().unwrap() + start, 0)
 | 
						|
                } else {
 | 
						|
                    min(start, size)
 | 
						|
                }
 | 
						|
            }
 | 
						|
        };
 | 
						|
        let relativeEnd: i64 = match end {
 | 
						|
            None => size,
 | 
						|
            Some(end) => {
 | 
						|
                if end < 0 {
 | 
						|
                    max(size + end, 0)
 | 
						|
                } else {
 | 
						|
                    min(end, size)
 | 
						|
                }
 | 
						|
            }
 | 
						|
        };
 | 
						|
        let relativeContentType = match contentType {
 | 
						|
            None => "".to_owned(),
 | 
						|
            Some(str) => {
 | 
						|
                if is_ascii_printable(&str) {
 | 
						|
                    str.as_slice().to_ascii_lowercase().to_owned()
 | 
						|
                } else {
 | 
						|
                    "".to_owned()
 | 
						|
                }
 | 
						|
            }
 | 
						|
        };
 | 
						|
        let span: i64 = max(relativeEnd - relativeStart, 0);
 | 
						|
        let global = self.global.root();
 | 
						|
        match self.bytes {
 | 
						|
            None => Blob::new(global.r(), None, relativeContentType.as_slice()),
 | 
						|
            Some(ref vec) => {
 | 
						|
                let start = relativeStart.to_uint().unwrap();
 | 
						|
                let end = (relativeStart + span).to_uint().unwrap();
 | 
						|
                let mut bytes: Vec<u8> = Vec::new();
 | 
						|
                bytes.push_all(vec.slice(start, end));
 | 
						|
                Blob::new(global.r(), Some(bytes), relativeContentType.as_slice())
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    //fn IsClosed(self) -> bool {
 | 
						|
    //    self.isClosed_.clone()
 | 
						|
    //}
 | 
						|
 | 
						|
    //fn Close(self) {
 | 
						|
    //    TODO
 | 
						|
    //}
 | 
						|
}
 | 
						|
 | 
						|
impl FileDerived for Blob {
 | 
						|
    fn is_file(&self) -> bool {
 | 
						|
        match self.type_ {
 | 
						|
            BlobTypeId::File => true,
 | 
						|
            _ => false
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 |