forked from mirrors/gecko-dev
		
	 c9176a14a6
			
		
	
	
		c9176a14a6
		
	
	
	
	
		
			
			Source-Repo: https://github.com/servo/servo Source-Revision: b655b54f8022d963460e510511ad774a1a1d9ccd
		
			
				
	
	
		
			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[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
 | |
|         }
 | |
|     }
 | |
| }
 |