mirror of
https://github.com/servo/servo.git
synced 2025-08-04 13:10:20 +01:00
Partial implementation of FormData and File
This commit is contained in:
parent
0547609589
commit
06eb08ab70
7 changed files with 160 additions and 14 deletions
|
@ -40,6 +40,7 @@ DOMInterfaces = {
|
||||||
'nativeType': 'EventListenerBinding::EventListener',
|
'nativeType': 'EventListenerBinding::EventListener',
|
||||||
},
|
},
|
||||||
'EventTarget': {},
|
'EventTarget': {},
|
||||||
|
'File': {},
|
||||||
'FormData': {},
|
'FormData': {},
|
||||||
'HTMLAnchorElement': {},
|
'HTMLAnchorElement': {},
|
||||||
'HTMLAppletElement': {},
|
'HTMLAppletElement': {},
|
||||||
|
|
|
@ -2,23 +2,32 @@
|
||||||
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
* 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/. */
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
||||||
|
|
||||||
|
use dom::bindings::codegen::InheritTypes::FileDerived;
|
||||||
use dom::bindings::js::{JS, JSRef, Temporary};
|
use dom::bindings::js::{JS, JSRef, Temporary};
|
||||||
use dom::bindings::utils::{Reflectable, Reflector, reflect_dom_object};
|
use dom::bindings::utils::{Reflectable, Reflector, reflect_dom_object};
|
||||||
use dom::bindings::error::Fallible;
|
use dom::bindings::error::Fallible;
|
||||||
use dom::bindings::codegen::Bindings::BlobBinding;
|
use dom::bindings::codegen::Bindings::BlobBinding;
|
||||||
use dom::window::Window;
|
use dom::window::Window;
|
||||||
|
|
||||||
|
#[deriving(Encodable)]
|
||||||
|
pub enum BlobType {
|
||||||
|
BlobTypeId,
|
||||||
|
FileTypeId
|
||||||
|
}
|
||||||
|
|
||||||
#[deriving(Encodable)]
|
#[deriving(Encodable)]
|
||||||
pub struct Blob {
|
pub struct Blob {
|
||||||
pub reflector_: Reflector,
|
pub reflector_: Reflector,
|
||||||
pub window: JS<Window>
|
pub window: JS<Window>,
|
||||||
|
pub type_: BlobType
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Blob {
|
impl Blob {
|
||||||
pub fn new_inherited(window: &JSRef<Window>) -> Blob {
|
pub fn new_inherited(window: &JSRef<Window>) -> Blob {
|
||||||
Blob {
|
Blob {
|
||||||
reflector_: Reflector::new(),
|
reflector_: Reflector::new(),
|
||||||
window: JS::from_rooted(window)
|
window: JS::from_rooted(window),
|
||||||
|
type_: BlobTypeId
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -41,3 +50,12 @@ impl Reflectable for Blob {
|
||||||
&self.reflector_
|
&self.reflector_
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl FileDerived for Blob {
|
||||||
|
fn is_file(&self) -> bool {
|
||||||
|
match self.type_ {
|
||||||
|
FileTypeId => true,
|
||||||
|
_ => false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
53
src/components/script/dom/file.rs
Normal file
53
src/components/script/dom/file.rs
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
/* 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::js::{JS, JSRef, Temporary};
|
||||||
|
use dom::bindings::utils::{Reflectable, Reflector, reflect_dom_object};
|
||||||
|
use dom::bindings::codegen::Bindings::FileBinding;
|
||||||
|
use dom::blob::{Blob, BlobType, FileTypeId};
|
||||||
|
use dom::window::Window;
|
||||||
|
use servo_util::str::DOMString;
|
||||||
|
|
||||||
|
#[deriving(Encodable)]
|
||||||
|
pub struct File {
|
||||||
|
pub blob: Blob,
|
||||||
|
pub name: DOMString,
|
||||||
|
pub window: JS<Window>,
|
||||||
|
pub type_: BlobType
|
||||||
|
}
|
||||||
|
|
||||||
|
impl File {
|
||||||
|
pub fn new_inherited(window: &JSRef<Window>, _file_bits: &JSRef<Blob>, name: DOMString) -> File {
|
||||||
|
File {
|
||||||
|
blob: Blob::new_inherited(window),
|
||||||
|
name: name,
|
||||||
|
window: JS::from_rooted(window),
|
||||||
|
type_: FileTypeId
|
||||||
|
}
|
||||||
|
// XXXManishearth Once Blob is able to store data
|
||||||
|
// the relevant subfields of file_bits should be copied over
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn new(window: &JSRef<Window>, file_bits: &JSRef<Blob>, name: DOMString) -> Temporary<File> {
|
||||||
|
reflect_dom_object(box File::new_inherited(window, file_bits, name),
|
||||||
|
window,
|
||||||
|
FileBinding::Wrap)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait FileMethods {
|
||||||
|
fn Name(&self) -> DOMString;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FileMethods for File {
|
||||||
|
fn Name(&self) -> DOMString {
|
||||||
|
self.name.clone()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Reflectable for File {
|
||||||
|
fn reflector<'a>(&'a self) -> &'a Reflector {
|
||||||
|
self.blob.reflector()
|
||||||
|
}
|
||||||
|
}
|
|
@ -3,26 +3,29 @@
|
||||||
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
||||||
|
|
||||||
use dom::bindings::codegen::Bindings::FormDataBinding;
|
use dom::bindings::codegen::Bindings::FormDataBinding;
|
||||||
|
use dom::bindings::codegen::InheritTypes::FileCast;
|
||||||
|
use dom::bindings::codegen::UnionTypes::FileOrString::{FileOrString, eFile, eString};
|
||||||
use dom::bindings::error::{Fallible};
|
use dom::bindings::error::{Fallible};
|
||||||
use dom::bindings::js::{JS, JSRef, Temporary, OptionalUnrootable};
|
use dom::bindings::js::{JS, JSRef, Temporary};
|
||||||
use dom::bindings::trace::Traceable;
|
use dom::bindings::trace::Traceable;
|
||||||
use dom::bindings::utils::{Reflectable, Reflector, reflect_dom_object};
|
use dom::bindings::utils::{Reflectable, Reflector, reflect_dom_object};
|
||||||
use dom::blob::Blob;
|
use dom::blob::Blob;
|
||||||
|
use dom::file::File;
|
||||||
use dom::htmlformelement::HTMLFormElement;
|
use dom::htmlformelement::HTMLFormElement;
|
||||||
use dom::window::Window;
|
use dom::window::Window;
|
||||||
use servo_util::str::DOMString;
|
use servo_util::str::DOMString;
|
||||||
use collections::hashmap::HashMap;
|
|
||||||
use std::cell::RefCell;
|
use std::cell::RefCell;
|
||||||
|
use collections::hashmap::HashMap;
|
||||||
|
|
||||||
#[deriving(Encodable)]
|
#[deriving(Encodable, Clone)]
|
||||||
pub enum FormDatum {
|
pub enum FormDatum {
|
||||||
StringData(DOMString),
|
StringData(DOMString),
|
||||||
BlobData { blob: JS<Blob>, name: DOMString }
|
FileData(JS<File>)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[deriving(Encodable)]
|
#[deriving(Encodable)]
|
||||||
pub struct FormData {
|
pub struct FormData {
|
||||||
pub data: Traceable<RefCell<HashMap<DOMString, FormDatum>>>,
|
pub data: Traceable<RefCell<HashMap<DOMString, Vec<FormDatum>>>>,
|
||||||
pub reflector_: Reflector,
|
pub reflector_: Reflector,
|
||||||
pub window: JS<Window>,
|
pub window: JS<Window>,
|
||||||
pub form: Option<JS<HTMLFormElement>>
|
pub form: Option<JS<HTMLFormElement>>
|
||||||
|
@ -34,7 +37,7 @@ impl FormData {
|
||||||
data: Traceable::new(RefCell::new(HashMap::new())),
|
data: Traceable::new(RefCell::new(HashMap::new())),
|
||||||
reflector_: Reflector::new(),
|
reflector_: Reflector::new(),
|
||||||
window: JS::from_rooted(window),
|
window: JS::from_rooted(window),
|
||||||
form: form.unrooted(),
|
form: form.map(|f| JS::from_rooted(&f)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -50,19 +53,53 @@ impl FormData {
|
||||||
pub trait FormDataMethods {
|
pub trait FormDataMethods {
|
||||||
fn Append(&self, name: DOMString, value: &JSRef<Blob>, filename: Option<DOMString>);
|
fn Append(&self, name: DOMString, value: &JSRef<Blob>, filename: Option<DOMString>);
|
||||||
fn Append_(&self, name: DOMString, value: DOMString);
|
fn Append_(&self, name: DOMString, value: DOMString);
|
||||||
|
fn Delete(&self, name: DOMString);
|
||||||
|
fn Get(&self, name: DOMString) -> Option<FileOrString>;
|
||||||
|
fn Has(&self, name: DOMString) -> bool;
|
||||||
|
fn Set(&self, name: DOMString, value: &JSRef<Blob>, filename: Option<DOMString>);
|
||||||
|
fn Set_(&self, name: DOMString, value: DOMString);
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> FormDataMethods for JSRef<'a, FormData> {
|
impl<'a> FormDataMethods for JSRef<'a, FormData> {
|
||||||
fn Append(&self, name: DOMString, value: &JSRef<Blob>, filename: Option<DOMString>) {
|
fn Append(&self, name: DOMString, value: &JSRef<Blob>, filename: Option<DOMString>) {
|
||||||
let blob = BlobData {
|
let file = FileData(JS::from_rooted(&self.get_file_from_blob(value, filename)));
|
||||||
blob: JS::from_rooted(value),
|
self.data.deref().borrow_mut().insert_or_update_with(name.clone(), vec!(file.clone()),
|
||||||
name: filename.unwrap_or("default".to_string())
|
|_k, v| {v.push(file.clone());});
|
||||||
};
|
|
||||||
self.data.deref().borrow_mut().insert(name.clone(), blob);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn Append_(&self, name: DOMString, value: DOMString) {
|
fn Append_(&self, name: DOMString, value: DOMString) {
|
||||||
self.data.deref().borrow_mut().insert(name, StringData(value));
|
self.data.deref().borrow_mut().insert_or_update_with(name, vec!(StringData(value.clone())),
|
||||||
|
|_k, v| {v.push(StringData(value.clone()));});
|
||||||
|
}
|
||||||
|
|
||||||
|
fn Delete(&self, name: DOMString) {
|
||||||
|
self.data.deref().borrow_mut().remove(&name);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn Get(&self, name: DOMString) -> Option<FileOrString> {
|
||||||
|
if self.data.deref().borrow().contains_key_equiv(&name) {
|
||||||
|
match self.data.deref().borrow().get(&name).get(0).clone() {
|
||||||
|
StringData(ref s) => Some(eString(s.clone())),
|
||||||
|
FileData(ref f) => {
|
||||||
|
Some(eFile(f.clone()))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn Has(&self, name: DOMString) -> bool {
|
||||||
|
self.data.deref().borrow().contains_key_equiv(&name)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn Set(&self, name: DOMString, value: &JSRef<Blob>, filename: Option<DOMString>) {
|
||||||
|
let file = FileData(JS::from_rooted(&self.get_file_from_blob(value, filename)));
|
||||||
|
self.data.deref().borrow_mut().insert(name, vec!(file));
|
||||||
|
}
|
||||||
|
|
||||||
|
fn Set_(&self, name: DOMString, value: DOMString) {
|
||||||
|
self.data.deref().borrow_mut().insert(name, vec!(StringData(value)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -71,3 +108,16 @@ impl Reflectable for FormData {
|
||||||
&self.reflector_
|
&self.reflector_
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
trait PrivateFormDataHelpers{
|
||||||
|
fn get_file_from_blob(&self, value: &JSRef<Blob>, filename: Option<DOMString>) -> Temporary<File>;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl PrivateFormDataHelpers for FormData {
|
||||||
|
fn get_file_from_blob(&self, value: &JSRef<Blob>, filename: Option<DOMString>) -> Temporary<File> {
|
||||||
|
let global = self.window.root();
|
||||||
|
let f: Option<&JSRef<File>> = FileCast::to_ref(value);
|
||||||
|
let name = filename.unwrap_or(f.map(|inner| inner.name.clone()).unwrap_or("blob".to_string()));
|
||||||
|
File::new(&*global, value, name)
|
||||||
|
}
|
||||||
|
}
|
15
src/components/script/dom/webidls/File.webidl
Normal file
15
src/components/script/dom/webidls/File.webidl
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
/* -*- Mode: IDL; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
||||||
|
/* 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/. */
|
||||||
|
|
||||||
|
// http://dev.w3.org/2006/webapi/FileAPI/#dfn-file
|
||||||
|
|
||||||
|
// [Constructor(sequence<(Blob or DOMString or ArrayBufferView or ArrayBuffer)> fileBits,
|
||||||
|
// [EnsureUTF16] DOMString fileName, optional FilePropertyBag options)]
|
||||||
|
interface File : Blob {
|
||||||
|
|
||||||
|
readonly attribute DOMString name;
|
||||||
|
// readonly attribute Date lastModifiedDate;
|
||||||
|
|
||||||
|
};
|
|
@ -7,8 +7,16 @@
|
||||||
* http://xhr.spec.whatwg.org
|
* http://xhr.spec.whatwg.org
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
typedef (File or DOMString) FormDataEntryValue;
|
||||||
|
|
||||||
[Constructor(optional HTMLFormElement form)]
|
[Constructor(optional HTMLFormElement form)]
|
||||||
interface FormData {
|
interface FormData {
|
||||||
void append(DOMString name, Blob value, optional DOMString filename);
|
void append(DOMString name, Blob value, optional DOMString filename);
|
||||||
void append(DOMString name, DOMString value);
|
void append(DOMString name, DOMString value);
|
||||||
|
void delete(DOMString name);
|
||||||
|
FormDataEntryValue? get(DOMString name);
|
||||||
|
// sequence<FormDataEntryValue> getAll(DOMString name);
|
||||||
|
boolean has(DOMString name);
|
||||||
|
void set(DOMString name, Blob value, optional DOMString filename);
|
||||||
|
void set(DOMString name, DOMString value);
|
||||||
};
|
};
|
||||||
|
|
|
@ -83,6 +83,7 @@ pub mod dom {
|
||||||
pub mod event;
|
pub mod event;
|
||||||
pub mod eventdispatcher;
|
pub mod eventdispatcher;
|
||||||
pub mod eventtarget;
|
pub mod eventtarget;
|
||||||
|
pub mod file;
|
||||||
pub mod formdata;
|
pub mod formdata;
|
||||||
pub mod htmlanchorelement;
|
pub mod htmlanchorelement;
|
||||||
pub mod htmlappletelement;
|
pub mod htmlappletelement;
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue