2018-09-05 12:55:30 -07:00
|
|
|
//! Raw API bindings for Web APIs
|
|
|
|
//!
|
|
|
|
//! This is a procedurally generated crate from browser WebIDL which provides a
|
|
|
|
//! binding to all APIs that browser provide on the web.
|
|
|
|
//!
|
|
|
|
//! This crate by default contains very little when compiled as almost all of
|
|
|
|
//! its exposed APIs are gated by Cargo features. The exhaustive list of
|
|
|
|
//! features can be found in `crates/web-sys/Cargo.toml`, but the rule of thumb
|
|
|
|
//! for `web-sys` is that each type has its own cargo feature (named after the
|
|
|
|
//! type). Using an API requires enabling the features for all types used in the
|
|
|
|
//! API, and APIs should mention in the documentation what features they
|
|
|
|
//! require.
|
|
|
|
|
2018-07-19 14:57:04 -05:00
|
|
|
#![doc(html_root_url = "https://docs.rs/web-sys/0.2")]
|
2018-07-09 16:35:25 -07:00
|
|
|
|
|
|
|
extern crate wasm_bindgen;
|
2018-08-04 13:51:22 -07:00
|
|
|
extern crate js_sys;
|
2018-09-12 15:24:04 -07:00
|
|
|
use js_sys::Object;
|
2018-07-09 16:35:25 -07:00
|
|
|
|
2018-09-17 14:25:04 -07:00
|
|
|
#[cfg(feature = "Window")]
|
|
|
|
pub fn window() -> Option<Window> {
|
2018-09-17 17:39:20 -07:00
|
|
|
use std::sync::atomic::{AtomicUsize, ATOMIC_USIZE_INIT, Ordering::SeqCst};
|
2018-09-17 14:25:04 -07:00
|
|
|
use wasm_bindgen::{JsValue, JsCast};
|
|
|
|
|
2018-09-17 17:39:20 -07:00
|
|
|
// Cached `Box<JsValue>`, if we've already executed this.
|
|
|
|
//
|
|
|
|
// 0 = not calculated
|
|
|
|
// 1 = `None`
|
|
|
|
// n = Some(n) == Some(Box<JsValue>)
|
|
|
|
static WINDOW: AtomicUsize = ATOMIC_USIZE_INIT;
|
|
|
|
|
|
|
|
match WINDOW.load(SeqCst) {
|
|
|
|
0 => {}
|
|
|
|
1 => return None,
|
|
|
|
n => return unsafe { Some((*(n as *const JsValue)).clone().unchecked_into()) },
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ok we don't have a cached value, let's load one! Manufacture a function
|
|
|
|
// to get access to the `this` context and see if it's an instance of
|
|
|
|
// `Window`.
|
|
|
|
//
|
|
|
|
// Note that we avoid `unwrap()` on `call0` to avoid code size bloat, we
|
|
|
|
// just handle the `Err` case as returning `None`.
|
|
|
|
let window = js_sys::Function::new_no_args("return this")
|
2018-09-17 14:25:04 -07:00
|
|
|
.call0(&JsValue::undefined())
|
|
|
|
.ok()
|
2018-09-17 17:39:20 -07:00
|
|
|
.and_then(|w| w.dyn_into::<Window>().ok());
|
|
|
|
|
|
|
|
match &window {
|
|
|
|
None => WINDOW.store(1, SeqCst),
|
|
|
|
Some(window) => {
|
|
|
|
let window: &JsValue = window.as_ref();
|
|
|
|
let ptr: *mut JsValue = Box::into_raw(Box::new(window.clone()));
|
|
|
|
match WINDOW.compare_exchange(0, ptr as usize, SeqCst, SeqCst) {
|
|
|
|
// We stored out value, relinquishing ownership of `ptr`
|
|
|
|
Ok(_) => {}
|
|
|
|
// Another thread one, drop our value
|
|
|
|
Err(_) => unsafe { drop(Box::from_raw(ptr)) },
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
window
|
2018-09-17 14:25:04 -07:00
|
|
|
}
|
|
|
|
|
2018-07-09 16:35:25 -07:00
|
|
|
include!(concat!(env!("OUT_DIR"), "/bindings.rs"));
|