mirror of
https://github.com/servo/servo.git
synced 2025-08-06 14:10:11 +01:00
Implement Window.sessionStorage: Storage Task, Storage Methods (excluding Storage event, QuotaExceededError)
This commit is contained in:
parent
68c90e2797
commit
f6fb9f862d
31 changed files with 347 additions and 109 deletions
|
@ -4,58 +4,117 @@
|
|||
|
||||
use dom::bindings::codegen::Bindings::StorageBinding;
|
||||
use dom::bindings::codegen::Bindings::StorageBinding::StorageMethods;
|
||||
use dom::bindings::global::GlobalRef;
|
||||
use dom::bindings::global::{GlobalRef, GlobalField};
|
||||
use dom::bindings::js::{JSRef, Temporary};
|
||||
use dom::bindings::utils::{Reflectable, Reflector, reflect_dom_object};
|
||||
use dom::bindings::error::Fallible;
|
||||
use servo_util::str::DOMString;
|
||||
use servo_net::storage_task::StorageTask;
|
||||
use servo_net::storage_task::StorageTaskMsg;
|
||||
use std::comm::channel;
|
||||
use url::Url;
|
||||
|
||||
#[dom_struct]
|
||||
pub struct Storage {
|
||||
reflector_: Reflector,
|
||||
global: GlobalField,
|
||||
}
|
||||
|
||||
impl Storage {
|
||||
fn new_inherited() -> Storage {
|
||||
fn new_inherited(global: &GlobalRef) -> Storage {
|
||||
Storage {
|
||||
reflector_: Reflector::new(),
|
||||
global: GlobalField::from_rooted(global),
|
||||
}
|
||||
}
|
||||
|
||||
pub fn new(global: GlobalRef) -> Temporary<Storage> {
|
||||
reflect_dom_object(box Storage::new_inherited(), global, StorageBinding::Wrap)
|
||||
pub fn new(global: &GlobalRef) -> Temporary<Storage> {
|
||||
reflect_dom_object(box Storage::new_inherited(global), *global, StorageBinding::Wrap)
|
||||
}
|
||||
|
||||
pub fn Constructor(global: &GlobalRef) -> Fallible<Temporary<Storage>> {
|
||||
Ok(Storage::new(global))
|
||||
}
|
||||
|
||||
fn get_url(&self) -> Url {
|
||||
let global_root = self.global.root();
|
||||
let global_ref = global_root.root_ref();
|
||||
global_ref.get_url()
|
||||
}
|
||||
|
||||
fn get_storage_task(&self) -> StorageTask {
|
||||
let global_root = self.global.root();
|
||||
let global_ref = global_root.root_ref();
|
||||
global_ref.as_window().storage_task()
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
impl<'a> StorageMethods for JSRef<'a, Storage> {
|
||||
fn Length(self) -> u32 {
|
||||
0
|
||||
let (sender, receiver) = channel();
|
||||
|
||||
self.get_storage_task().send(StorageTaskMsg::Length(sender, self.get_url()));
|
||||
receiver.recv()
|
||||
}
|
||||
|
||||
fn Key(self, index: u32) -> Option<DOMString> {
|
||||
let (sender, receiver) = channel();
|
||||
|
||||
//Return null for out of range index
|
||||
if index >= self.Length() {
|
||||
return None;
|
||||
}
|
||||
|
||||
return None;
|
||||
self.get_storage_task().send(StorageTaskMsg::Key(sender, self.get_url(), index));
|
||||
receiver.recv()
|
||||
}
|
||||
|
||||
fn GetItem(self, key: DOMString) -> Option<DOMString> {
|
||||
if key.is_empty() {
|
||||
return None;
|
||||
}
|
||||
fn GetItem(self, name: DOMString) -> Option<DOMString> {
|
||||
let (sender, receiver) = channel();
|
||||
|
||||
return None;
|
||||
self.get_storage_task().send(StorageTaskMsg::GetItem(sender, self.get_url(), name));
|
||||
receiver.recv()
|
||||
}
|
||||
|
||||
fn NamedGetter(self, key: DOMString, found: &mut bool) -> Option<DOMString> {
|
||||
let item = self.GetItem(key);
|
||||
fn NamedGetter(self, name: DOMString, found: &mut bool) -> Option<DOMString> {
|
||||
let item = self.GetItem(name);
|
||||
*found = item.is_some();
|
||||
item
|
||||
}
|
||||
|
||||
fn SetItem(self, name: DOMString, value: DOMString) {
|
||||
let (sender, receiver) = channel();
|
||||
|
||||
self.get_storage_task().send(StorageTaskMsg::SetItem(sender, self.get_url(), name, value));
|
||||
if receiver.recv() {
|
||||
//TODO send notification
|
||||
}
|
||||
}
|
||||
|
||||
fn NamedSetter(self, name: DOMString, value: DOMString) {
|
||||
self.SetItem(name, value);
|
||||
}
|
||||
|
||||
fn NamedCreator(self, name: DOMString, value: DOMString) {
|
||||
self.SetItem(name, value);
|
||||
}
|
||||
|
||||
fn RemoveItem(self, name: DOMString) {
|
||||
let (sender, receiver) = channel();
|
||||
|
||||
self.get_storage_task().send(StorageTaskMsg::RemoveItem(sender, self.get_url(), name));
|
||||
if receiver.recv() {
|
||||
//TODO send notification
|
||||
}
|
||||
}
|
||||
|
||||
fn NamedDeleter(self, name: DOMString) {
|
||||
self.RemoveItem(name);
|
||||
}
|
||||
|
||||
fn Clear(self) {
|
||||
let (sender, receiver) = channel();
|
||||
|
||||
self.get_storage_task().send(StorageTaskMsg::Clear(sender, self.get_url()));
|
||||
if receiver.recv() {
|
||||
//TODO send notification
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -16,10 +16,9 @@ interface Storage {
|
|||
|
||||
getter DOMString? getItem(DOMString name);
|
||||
|
||||
//setter creator void setItem(DOMString name, DOMString value);
|
||||
setter creator void setItem(DOMString name, DOMString value);
|
||||
|
||||
//deleter not supported yet
|
||||
//deleter void removeItem(DOMString name);
|
||||
deleter void removeItem(DOMString name);
|
||||
|
||||
void clear();
|
||||
};
|
||||
|
|
|
@ -127,3 +127,10 @@ partial interface Window {
|
|||
void gc();
|
||||
};
|
||||
Window implements OnErrorEventHandlerForWindow;
|
||||
|
||||
// https://html.spec.whatwg.org/multipage/webstorage.html#dom-sessionstorage
|
||||
[NoInterfaceObject]
|
||||
interface WindowSessionStorage {
|
||||
readonly attribute Storage sessionStorage;
|
||||
};
|
||||
Window implements WindowSessionStorage;
|
||||
|
|
|
@ -20,6 +20,7 @@ use dom::location::Location;
|
|||
use dom::navigator::Navigator;
|
||||
use dom::performance::Performance;
|
||||
use dom::screen::Screen;
|
||||
use dom::storage::Storage;
|
||||
use layout_interface::NoQuery;
|
||||
use page::Page;
|
||||
use script_task::{ExitWindowMsg, ScriptChan, TriggerLoadMsg, TriggerFragmentMsg};
|
||||
|
@ -30,6 +31,7 @@ use timers::{Interval, NonInterval, TimerId, TimerManager};
|
|||
use servo_msg::compositor_msg::ScriptListener;
|
||||
use servo_msg::constellation_msg::LoadData;
|
||||
use servo_net::image_cache_task::ImageCacheTask;
|
||||
use servo_net::storage_task::StorageTask;
|
||||
use servo_util::str::{DOMString,HTML_SPACE_CHARACTERS};
|
||||
|
||||
use js::jsapi::JS_EvaluateUCScript;
|
||||
|
@ -62,6 +64,7 @@ pub struct Window {
|
|||
navigation_start: u64,
|
||||
navigation_start_precise: f64,
|
||||
screen: MutNullableJS<Screen>,
|
||||
session_storage: MutNullableJS<Storage>,
|
||||
timers: TimerManager
|
||||
}
|
||||
|
||||
|
@ -106,6 +109,10 @@ impl Window {
|
|||
pub fn get_url(&self) -> Url {
|
||||
self.page().get_url()
|
||||
}
|
||||
|
||||
pub fn storage_task(&self) -> StorageTask {
|
||||
self.page().storage_task.clone()
|
||||
}
|
||||
}
|
||||
|
||||
// http://www.whatwg.org/html/#atob
|
||||
|
@ -208,6 +215,14 @@ impl<'a> WindowMethods for JSRef<'a, Window> {
|
|||
self.location.get().unwrap()
|
||||
}
|
||||
|
||||
fn SessionStorage(self) -> Temporary<Storage> {
|
||||
if self.session_storage.get().is_none() {
|
||||
let session_storage = Storage::new(&global::Window(self));
|
||||
self.session_storage.assign(Some(session_storage));
|
||||
}
|
||||
self.session_storage.get().unwrap()
|
||||
}
|
||||
|
||||
fn Console(self) -> Temporary<Console> {
|
||||
if self.console.get().is_none() {
|
||||
let console = Console::new(global::Window(self));
|
||||
|
@ -412,6 +427,7 @@ impl Window {
|
|||
navigation_start: time::get_time().sec as u64,
|
||||
navigation_start_precise: time::precise_time_s(),
|
||||
screen: Default::default(),
|
||||
session_storage: Default::default(),
|
||||
timers: TimerManager::new()
|
||||
};
|
||||
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue