This implements support for using Promises in WebIDL, executing promise callbacks in batches (equivalent to running all enqueued jobs from a setTimeout(0)), and attaching native callbacks to promise objects. This is the combined work of myself, @dati91, and @mmatyas based on the following prior work in Gecko: * Codegen.py * Promise.webidl * Promise.cpp/Promise.h * PromiseNativeHandler.h This does not implement microtasks per #4283. This allows us to make progress on testing code that requires the use of Promises right now; the microtasks work is more complicated, but also largely orthogonal to implement. Requires https://github.com/servo/mozjs/pull/89, https://github.com/servo/rust-mozjs/pull/287, and https://github.com/servo/rust-mozjs/pull/294. --- - [X] `./mach build -d` does not report any errors - [X] `./mach test-tidy` does not report any errors - [X] These changes fix #4282 - [X] There are tests for these changes Source-Repo: https://github.com/servo/servo Source-Revision: 2b1a39c2ae14b78e437551d05f0e691a13e5d183
50 lines
1.6 KiB
Rust
50 lines
1.6 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::Bindings::PromiseNativeHandlerBinding;
|
|
use dom::bindings::global::GlobalRef;
|
|
use dom::bindings::js::Root;
|
|
use dom::bindings::reflector::{Reflector, reflect_dom_object};
|
|
use dom::bindings::trace::JSTraceable;
|
|
use heapsize::HeapSizeOf;
|
|
use js::jsapi::{JSContext, HandleValue};
|
|
|
|
pub trait Callback: JSTraceable + HeapSizeOf {
|
|
fn callback(&self, cx: *mut JSContext, v: HandleValue);
|
|
}
|
|
|
|
#[dom_struct]
|
|
pub struct PromiseNativeHandler {
|
|
reflector: Reflector,
|
|
resolve: Option<Box<Callback>>,
|
|
reject: Option<Box<Callback>>,
|
|
}
|
|
|
|
impl PromiseNativeHandler {
|
|
pub fn new(global: GlobalRef,
|
|
resolve: Option<Box<Callback>>,
|
|
reject: Option<Box<Callback>>)
|
|
-> Root<PromiseNativeHandler> {
|
|
reflect_dom_object(box PromiseNativeHandler {
|
|
reflector: Reflector::new(),
|
|
resolve: resolve,
|
|
reject: reject,
|
|
}, global, PromiseNativeHandlerBinding::Wrap)
|
|
}
|
|
|
|
fn callback(callback: &Option<Box<Callback>>, cx: *mut JSContext, v: HandleValue) {
|
|
if let Some(ref callback) = *callback {
|
|
callback.callback(cx, v)
|
|
}
|
|
}
|
|
|
|
pub fn resolved_callback(&self, cx: *mut JSContext, v: HandleValue) {
|
|
PromiseNativeHandler::callback(&self.resolve, cx, v)
|
|
}
|
|
|
|
pub fn rejected_callback(&self, cx: *mut JSContext, v: HandleValue) {
|
|
PromiseNativeHandler::callback(&self.reject, cx, v)
|
|
}
|
|
}
|