#[allow(unused_imports, reason = "used by _js_method_str_alloc!")]
#[cfg(feature = "alloc")]
use devela::String;
use devela::{_js_doc, Distance, Extent, Float, offset_of};
#[allow(unused_imports, reason = "not(windows)")]
use devela::{
_js_extern, _js_method_str_alloc, Js, JsTimeout, WebDocument, js_bool, js_int32, js_number,
js_uint32,
};
#[doc = crate::_tags!(ui web)]
#[doc = crate::_doc_location!("sys/os/browser/web")]
#[repr(C)]
#[derive(Copy, Clone, Debug)]
pub struct WebWindow;
#[rustfmt::skip]
#[cfg(not(feature = "safe_lang"))]
#[cfg(all(feature = "unsafe_ffi", not(windows)))]
#[cfg_attr(nightly_doc, doc(cfg(feature = "unsafe_ffi")))]
#[cfg_attr(nightly_doc, doc(cfg(target_arch = "wasm32")))]
impl WebWindow {
#[doc = _js_doc!("Window", "document")]
pub fn document(&self) -> WebDocument { WebDocument }
pub fn state() -> WebWindowState { WebWindowState::new() }
#[doc = _js_doc!("Window", "closed")]
pub fn is_closed() -> js_bool { window_is_closed() }
#[doc = _js_doc!("Window", "crossOriginIsolated")]
pub fn is_coi() -> js_bool { window_is_coi() }
#[doc = _js_doc!("Window", "isSecureContext")]
pub fn is_secure() -> js_bool { window_is_secure() }
#[doc = _js_doc!("Window", "locationbar")]
pub fn is_popup() -> js_bool { window_is_popup() }
_js_method_str_alloc! {
#[doc = _js_doc!("Window", "name")]
name, window_name
}
#[doc = _js_doc!("Window", "name")]
pub fn set_name(name: &str) { unsafe { window_set_name(name.as_ptr(), name.len() as u32); } }
#[doc = _js_doc!("Window", "setTimeout")]
pub fn set_timeout(callback: extern "C" fn(), delay_ms: js_uint32) -> JsTimeout {
JsTimeout { id: unsafe { window_set_timeout(callback as usize, delay_ms) } } }
#[doc = _js_doc!("Window", "setInterval")]
pub fn set_interval(callback: extern "C" fn(), interval_ms: js_uint32) -> JsTimeout {
JsTimeout { id: unsafe { window_set_interval(callback as usize, interval_ms) } } }
#[doc = _js_doc!("Window", "clearTimeout")]
#[doc = _js_doc!("Window", "clearInterval")]
pub fn clear_timeout(id: JsTimeout) { window_clear_timeout(id.id()); }
pub fn eval(js_code: &str) { unsafe { window_eval(js_code.as_ptr(), js_code.len()); } }
#[doc = _js_doc!("Window", "setTimeout")]
pub fn eval_timeout(js_code: &str, delay_ms: js_uint32) -> JsTimeout { JsTimeout {
id: unsafe { window_eval_timeout(js_code.as_ptr(), js_code.len(), delay_ms) } } }
#[doc = _js_doc!("Window", "setInterval")]
pub fn eval_interval(js_code: &str, interval_ms: js_uint32) -> JsTimeout { JsTimeout {
id: unsafe { window_eval_interval(js_code.as_ptr(), js_code.len(), interval_ms) } } }
#[doc = _js_doc!("Window", "requestAnimationFrame")]
pub fn request_animation_frame(callback: extern "C" fn()) -> js_uint32 {
unsafe { window_request_animation_frame(callback as usize) } }
pub fn cancel_animation_frame(id: js_uint32) { window_cancel_animation_frame(id); }
}
_js_extern! {
[module: "api_window"]
unsafe fn window_state(data: *mut u8);
safe fn window_is_closed() -> js_bool;
safe fn window_is_coi() -> js_bool;
safe fn window_is_secure() -> js_bool;
safe fn window_is_popup() -> js_bool;
unsafe fn window_name(buf_ptr: *mut u8, max_len: js_uint32) -> js_int32;
unsafe fn window_set_name(str_ptr: *const u8, str_len: js_uint32);
unsafe fn window_set_timeout(callback_ptr: usize, delay_ms: js_uint32) -> js_uint32;
unsafe fn window_set_interval(callback_ptr: usize, interval_ms: js_uint32) -> js_uint32;
safe fn window_clear_timeout(timeout_id: js_uint32);
unsafe fn window_eval(js_code_ptr: *const u8, js_code_len: usize);
unsafe fn window_eval_timeout(js_code_ptr: *const u8, js_code_len: usize, delay_ms: js_uint32)
-> js_uint32;
unsafe fn window_eval_interval(js_code_ptr: *const u8, js_code_len: usize,
interval_ms: js_uint32) -> js_uint32;
unsafe fn window_request_animation_frame(callback_ptr: usize) -> js_uint32;
safe fn window_cancel_animation_frame(requestId: js_uint32);
}
#[doc = crate::_tags!(ui web)]
#[doc = crate::_doc_location!("sys/os/browser/web")]
#[repr(C)]
#[derive(Clone, Copy, Default, PartialEq)] pub struct WebWindowState {
#[doc = _js_doc!("Window", "innerWidth")]
#[doc = _js_doc!("Window", "innerHeight")]
pub inner_size: Extent<u32, 2>,
#[doc = _js_doc!("Window", "outerWidth")]
#[doc = _js_doc!("Window", "outerHeight")]
pub outer_size: Extent<u32, 2>,
#[doc = _js_doc!("Window", "screenLeft")]
#[doc = _js_doc!("Window", "screenTop")]
pub screen_offset: Distance<i32, 2>,
#[doc = _js_doc!("Screen", "width")]
#[doc = _js_doc!("Screen", "height")]
pub screen_size: Extent<u32, 2>,
#[doc = _js_doc!("Screen", "availWidth")]
#[doc = _js_doc!("Screen", "availHeight")]
pub screen_usable_size: Extent<u32, 2>,
#[doc = _js_doc!("Window", "devicePixelRatio")]
pub dpr: f32,
#[doc = _js_doc!("Screen", "colorDepth")]
pub bpp: u8,
_pad: [u8; 3],
}
impl WebWindowState {
const __ASSERT_FIELD_OFFSETS: () = const {
assert!(offset_of!(Self, inner_size) == 0);
assert!(offset_of!(Self, outer_size) == 8);
assert!(offset_of!(Self, screen_offset) == 16);
assert!(offset_of!(Self, screen_size) == 24);
assert!(offset_of!(Self, screen_usable_size) == 32);
assert!(offset_of!(Self, dpr) == 40);
assert!(offset_of!(Self, bpp) == 44);
};
#[cfg(not(feature = "safe_lang"))]
#[cfg(feature = "unsafe_ffi")]
#[cfg_attr(nightly_doc, doc(cfg(feature = "unsafe_ffi")))]
pub fn new() -> WebWindowState {
let mut state = WebWindowState::default();
unsafe {
window_state(&mut state as *mut WebWindowState as *mut u8);
}
state
}
#[cfg(not(feature = "safe_lang"))]
#[cfg(feature = "unsafe_ffi")]
#[cfg_attr(nightly_doc, doc(cfg(feature = "unsafe_ffi")))]
pub fn update(&mut self) {
unsafe { window_state(self as *mut Self as *mut u8) };
}
pub const fn is_valid(&self) -> bool {
let non_zero = self.inner_size.x() > 0
&& self.inner_size.y() > 0
&& self.outer_size.x() > 0
&& self.outer_size.y() > 0;
let inner_le_outer = self.inner_size.dim[0] <= self.outer_size.dim[0]
&& self.inner_size.dim[1] <= self.outer_size.dim[1];
let outer_le_screen = (self.outer_size.dim[0] <= self.screen_size.dim[0] + 10)
&& (self.outer_size.dim[1] <= self.screen_size.dim[1] + 10);
let sane_dpr = self.dpr >= 0.2 && self.dpr <= 10.0;
let sane_bpp = self.bpp >= 8 && self.bpp <= 64;
non_zero && inner_le_outer && outer_le_screen && sane_dpr && sane_bpp }
pub const fn chrome_size(&self) -> Extent<u32, 2> {
Extent::new([
self.outer_size.x() - self.inner_size.x(),
self.outer_size.y() - self.inner_size.y(),
])
}
pub const fn is_maximized(&self) -> bool {
self.outer_size.x() >= self.screen_usable_size.x()
&& self.outer_size.y() >= self.screen_usable_size.y()
}
pub const fn is_portrait(&self) -> bool {
self.inner_size.y() > self.inner_size.x()
}
pub const fn physical_size(&self) -> Extent<u32, 2> {
Extent::new([
(self.inner_size.x() as f32 * self.dpr) as u32,
(self.inner_size.y() as f32 * self.dpr) as u32,
])
}
pub const fn physical_size_rounded(&self) -> Extent<u32, 2> {
Extent::new([
Float(self.inner_size.x() as f32 * self.dpr).const_round().0 as u32,
Float(self.inner_size.y() as f32 * self.dpr).const_round().0 as u32,
])
}
pub const fn screen_margins(&self) -> [i32; 4] {
[
self.screen_offset.dim[0],
self.screen_offset.dim[1],
(self.screen_size.x() as i32)
- (self.screen_offset.dim[0] + self.outer_size.x() as i32),
(self.screen_size.y() as i32)
- (self.screen_offset.dim[1] + self.outer_size.y() as i32),
]
}
}
impl crate::Debug for WebWindowState {
fn fmt(&self, f: &mut crate::Formatter<'_>) -> crate::FmtResult<()> {
let mut state = f.debug_struct("WebWindowState");
state
.field("inner_size", &self.inner_size)
.field("outer_size", &self.outer_size)
.field("screen_offset", &self.screen_offset)
.field("screen_size", &self.screen_size)
.field("screen_usable_size", &self.screen_usable_size)
.field("dpr", &self.dpr)
.field("bpp", &self.bpp)
.field("chrome_size()", &self.chrome_size())
.field("is_maximized()", &self.is_maximized())
.field("is_portrait()", &self.is_portrait())
.field("is_valid()", &self.is_valid())
.field("physical_size()", &self.physical_size());
state.field("physical_size_rounded()", &self.physical_size_rounded());
state.field("screen_margins()", &self.screen_margins()).finish_non_exhaustive() }
}