dioxus/packages/web/src/new.rs

599 lines
20 KiB
Rust
Raw Normal View History

2021-06-23 05:44:48 +00:00
use std::{collections::HashMap, rc::Rc, sync::Arc};
2021-06-23 05:44:48 +00:00
use dioxus_core::{
events::{EventTrigger, VirtualEvent},
2021-07-09 16:47:41 +00:00
prelude::{RealDomNode, ScopeIdx},
2021-06-23 05:44:48 +00:00
};
use fxhash::FxHashMap;
2021-06-27 02:13:57 +00:00
use slotmap::{DefaultKey, Key, KeyData};
use wasm_bindgen::{closure::Closure, JsCast};
use web_sys::{
window, Document, Element, Event, HtmlElement, HtmlInputElement, HtmlOptionElement, Node,
};
pub struct WebsysDom {
pub stack: Stack,
2021-06-27 02:13:57 +00:00
nodes: slotmap::SlotMap<DefaultKey, Node>,
document: Document,
root: Element,
2021-06-23 05:44:48 +00:00
event_receiver: async_channel::Receiver<EventTrigger>,
2021-06-30 18:08:12 +00:00
trigger: Arc<dyn Fn(EventTrigger)>,
2021-06-23 05:44:48 +00:00
// map of listener types to number of those listeners
2021-06-30 18:08:12 +00:00
// This is roughly a delegater
// TODO: check how infero delegates its events - some are more performant
listeners: FxHashMap<&'static str, (usize, Closure<dyn FnMut(&Event)>)>,
2021-06-23 05:44:48 +00:00
// We need to make sure to add comments between text nodes
// We ensure that the text siblings are patched by preventing the browser from merging
// neighboring text nodes. Originally inspired by some of React's work from 2016.
// -> https://reactjs.org/blog/2016/04/07/react-v15.html#major-changes
// -> https://github.com/facebook/react/pull/5753
last_node_was_text: bool,
}
impl WebsysDom {
pub fn new(root: Element) -> Self {
let document = window()
.expect("must have access to the window")
.document()
.expect("must have access to the Document");
2021-06-30 18:08:12 +00:00
let (sender, receiver) = async_channel::unbounded::<EventTrigger>();
2021-06-23 05:44:48 +00:00
let sender_callback = Arc::new(move |ev| {
2021-06-27 02:13:57 +00:00
let c = sender.clone();
2021-06-23 05:44:48 +00:00
wasm_bindgen_futures::spawn_local(async move {
c.send(ev).await.unwrap();
});
});
2021-06-30 18:08:12 +00:00
let mut nodes = slotmap::SlotMap::with_capacity(1000);
2021-06-23 05:44:48 +00:00
2021-06-27 02:13:57 +00:00
let root_id = nodes.insert(root.clone().dyn_into::<Node>().unwrap());
Self {
stack: Stack::with_capacity(10),
2021-06-23 05:44:48 +00:00
nodes,
listeners: FxHashMap::default(),
document,
2021-06-23 05:44:48 +00:00
event_receiver: receiver,
trigger: sender_callback,
root,
last_node_was_text: false,
}
}
2021-06-23 05:44:48 +00:00
pub async fn wait_for_event(&mut self) -> Option<EventTrigger> {
let v = self.event_receiver.recv().await.unwrap();
Some(v)
}
}
2021-06-30 02:44:21 +00:00
impl<'a> dioxus_core::diff::RealDom<'a> for WebsysDom {
2021-07-09 16:47:41 +00:00
fn push_root(&mut self, root: RealDomNode) {
2021-06-27 02:13:57 +00:00
log::debug!("Called [push_root] {:?}", root);
let key: DefaultKey = KeyData::from_ffi(root.0).into();
let domnode = self.nodes.get(key).expect("Failed to pop know root");
self.stack.push(domnode.clone());
}
fn append_child(&mut self) {
2021-06-23 05:44:48 +00:00
log::debug!("Called [`append_child`]");
let child = self.stack.pop();
if child.dyn_ref::<web_sys::Text>().is_some() {
if self.last_node_was_text {
let comment_node = self
.document
.create_comment("dioxus")
.dyn_into::<Node>()
.unwrap();
self.stack.top().append_child(&comment_node).unwrap();
}
self.last_node_was_text = true;
} else {
self.last_node_was_text = false;
}
self.stack.top().append_child(&child).unwrap();
}
fn replace_with(&mut self) {
2021-06-23 05:44:48 +00:00
log::debug!("Called [`replace_with`]");
let new_node = self.stack.pop();
let old_node = self.stack.pop();
if old_node.has_type::<Element>() {
old_node
.dyn_ref::<Element>()
.unwrap()
.replace_with_with_node_1(&new_node)
.unwrap();
} else if old_node.has_type::<web_sys::CharacterData>() {
old_node
.dyn_ref::<web_sys::CharacterData>()
.unwrap()
.replace_with_with_node_1(&new_node)
.unwrap();
} else if old_node.has_type::<web_sys::DocumentType>() {
old_node
.dyn_ref::<web_sys::DocumentType>()
.unwrap()
.replace_with_with_node_1(&new_node)
.unwrap();
} else {
panic!("Cannot replace node: {:?}", old_node);
}
// // poc to see if this is a valid solution
// if let Some(id) = self.current_known {
// // update mapping
// self.known_roots.insert(id, new_node.clone());
// self.current_known = None;
// }
self.stack.push(new_node);
}
fn remove(&mut self) {
2021-06-23 05:44:48 +00:00
log::debug!("Called [`remove`]");
todo!()
}
fn remove_all_children(&mut self) {
2021-06-23 05:44:48 +00:00
log::debug!("Called [`remove_all_children`]");
todo!()
}
2021-06-27 02:13:57 +00:00
fn create_placeholder(&mut self) -> RealDomNode {
2021-06-30 02:44:21 +00:00
self.create_element("pre", None)
2021-06-27 02:13:57 +00:00
}
2021-07-09 16:47:41 +00:00
fn create_text_node(&mut self, text: &str) -> RealDomNode {
2021-06-27 02:13:57 +00:00
// let nid = self.node_counter.next();
let textnode = self
.document
.create_text_node(text)
.dyn_into::<Node>()
.unwrap();
self.stack.push(textnode.clone());
2021-06-27 02:13:57 +00:00
let nid = self.nodes.insert(textnode);
let nid = nid.data().as_ffi();
2021-06-23 05:44:48 +00:00
log::debug!("Called [`create_text_node`]: {}, {}", text, nid);
RealDomNode::new(nid)
}
2021-07-09 16:47:41 +00:00
fn create_element(&mut self, tag: &str, ns: Option<&'static str>) -> RealDomNode {
2021-07-05 05:11:49 +00:00
let tag = wasm_bindgen::intern(tag);
2021-06-30 02:44:21 +00:00
let el = match ns {
Some(ns) => self
.document
.create_element_ns(Some(ns), tag)
.unwrap()
.dyn_into::<Node>()
.unwrap(),
None => self
.document
.create_element(tag)
.unwrap()
.dyn_into::<Node>()
.unwrap(),
};
self.stack.push(el.clone());
2021-06-30 02:44:21 +00:00
// let nid = self.node_counter.?next();
2021-06-27 02:13:57 +00:00
let nid = self.nodes.insert(el).data().as_ffi();
2021-06-30 02:44:21 +00:00
log::debug!("Called [`create_element`]: {}, {:?}", tag, nid);
RealDomNode::new(nid)
}
fn new_event_listener(
&mut self,
2021-06-30 18:08:12 +00:00
event: &'static str,
scope: dioxus_core::prelude::ScopeIdx,
2021-06-23 05:44:48 +00:00
el_id: usize,
real_id: RealDomNode,
) {
2021-07-05 05:11:49 +00:00
let event = wasm_bindgen::intern(event);
2021-06-23 05:44:48 +00:00
log::debug!(
"Called [`new_event_listener`]: {}, {:?}, {}, {:?}",
event,
scope,
el_id,
real_id
);
// attach the correct attributes to the element
// these will be used by accessing the event's target
// This ensures we only ever have one handler attached to the root, but decide
// dynamically when we want to call a listener.
let el = self.stack.top();
let el = el
.dyn_ref::<Element>()
.expect(&format!("not an element: {:?}", el));
2021-06-30 18:08:12 +00:00
let gi_id = scope.data().as_ffi();
2021-06-23 05:44:48 +00:00
el.set_attribute(
&format!("dioxus-event-{}", event),
2021-06-30 18:08:12 +00:00
&format!("{}.{}.{}", gi_id, el_id, real_id.0),
2021-06-23 05:44:48 +00:00
)
.unwrap();
// Register the callback to decode
if let Some(entry) = self.listeners.get_mut(event) {
entry.0 += 1;
} else {
let trigger = self.trigger.clone();
2021-06-30 02:44:21 +00:00
2021-06-23 05:44:48 +00:00
let handler = Closure::wrap(Box::new(move |event: &web_sys::Event| {
// "Result" cannot be received from JS
// Instead, we just build and immediately execute a closure that returns result
2021-06-30 02:44:21 +00:00
match decode_trigger(event) {
2021-06-23 05:44:48 +00:00
Ok(synthetic_event) => trigger.as_ref()(synthetic_event),
2021-07-05 05:11:49 +00:00
Err(e) => log::error!("Error decoding Dioxus event attribute. {:#?}", e),
2021-06-23 05:44:48 +00:00
};
}) as Box<dyn FnMut(&Event)>);
self.root
.add_event_listener_with_callback(event, (&handler).as_ref().unchecked_ref())
.unwrap();
// Increment the listeners
self.listeners.insert(event.into(), (1, handler));
}
}
fn remove_event_listener(&mut self, event: &str) {
2021-06-23 05:44:48 +00:00
log::debug!("Called [`remove_event_listener`]: {}", event);
todo!()
}
fn set_text(&mut self, text: &str) {
2021-06-23 05:44:48 +00:00
log::debug!("Called [`set_text`]: {}", text);
self.stack.top().set_text_content(Some(text))
}
2021-06-30 02:44:21 +00:00
fn set_attribute(&mut self, name: &str, value: &str, ns: Option<&str>) {
2021-06-23 05:44:48 +00:00
log::debug!("Called [`set_attribute`]: {}, {}", name, value);
if name == "class" {
if let Some(el) = self.stack.top().dyn_ref::<Element>() {
el.set_class_name(value);
}
} else {
2021-06-23 05:44:48 +00:00
if let Some(el) = self.stack.top().dyn_ref::<Element>() {
el.set_attribute(name, value).unwrap();
}
}
}
fn remove_attribute(&mut self, name: &str) {
2021-06-23 05:44:48 +00:00
log::debug!("Called [`remove_attribute`]: {}", name);
let node = self.stack.top();
if let Some(node) = node.dyn_ref::<web_sys::Element>() {
node.remove_attribute(name).unwrap();
}
if let Some(node) = node.dyn_ref::<HtmlInputElement>() {
// Some attributes are "volatile" and don't work through `removeAttribute`.
if name == "value" {
node.set_value("");
}
if name == "checked" {
node.set_checked(false);
}
}
if let Some(node) = node.dyn_ref::<HtmlOptionElement>() {
if name == "selected" {
node.set_selected(true);
}
}
}
fn raw_node_as_any_mut(&self) -> &mut dyn std::any::Any {
2021-06-23 05:44:48 +00:00
log::debug!("Called [`raw_node_as_any_mut`]");
todo!()
}
}
2021-06-23 05:44:48 +00:00
#[derive(Debug, Default)]
pub struct Stack {
list: Vec<Node>,
}
impl Stack {
pub fn with_capacity(cap: usize) -> Self {
Stack {
list: Vec::with_capacity(cap),
}
}
pub fn push(&mut self, node: Node) {
// debug!("stack-push: {:?}", node);
self.list.push(node);
}
pub fn pop(&mut self) -> Node {
let res = self.list.pop().unwrap();
res
}
pub fn clear(&mut self) {
self.list.clear();
}
pub fn top(&self) -> &Node {
match self.list.last() {
Some(a) => a,
None => panic!("Called 'top' of an empty stack, make sure to push the root first"),
}
}
}
fn virtual_event_from_websys_event(event: &web_sys::Event) -> VirtualEvent {
use dioxus_core::events::on::*;
match event.type_().as_str() {
"copy" | "cut" | "paste" => {
// let evt: web_sys::ClipboardEvent = event.clone().dyn_into().unwrap();
todo!()
}
"compositionend" | "compositionstart" | "compositionupdate" => {
let evt: web_sys::CompositionEvent = event.clone().dyn_into().unwrap();
todo!()
}
"keydown" | "keypress" | "keyup" => {
let evt: web_sys::KeyboardEvent = event.clone().dyn_into().unwrap();
todo!()
}
"focus" | "blur" => {
let evt: web_sys::FocusEvent = event.clone().dyn_into().unwrap();
todo!()
}
"change" => {
let evt: web_sys::Event = event.clone().dyn_into().expect("wrong error typ");
todo!()
// VirtualEvent::FormEvent(FormEvent {value:})
}
"input" | "invalid" | "reset" | "submit" => {
// is a special react events
let evt: web_sys::InputEvent = event.clone().dyn_into().expect("wrong event type");
let this: web_sys::EventTarget = evt.target().unwrap();
let value = (&this)
.dyn_ref()
.map(|input: &web_sys::HtmlInputElement| input.value())
.or_else(|| {
(&this)
.dyn_ref()
.map(|input: &web_sys::HtmlTextAreaElement| input.value())
})
.or_else(|| {
(&this)
.dyn_ref::<web_sys::HtmlElement>()
.unwrap()
.text_content()
})
.expect("only an InputElement or TextAreaElement or an element with contenteditable=true can have an oninput event listener");
// let p2 = evt.data_transfer();
// let value: Option<String> = (&evt).data();
// let value = val;
// let value = value.unwrap_or_default();
// let value = (&evt).data().expect("No data to unwrap");
// todo - this needs to be a "controlled" event
// these events won't carry the right data with them
todo!()
// VirtualEvent::FormEvent(FormEvent { value })
}
"click" | "contextmenu" | "doubleclick" | "drag" | "dragend" | "dragenter" | "dragexit"
| "dragleave" | "dragover" | "dragstart" | "drop" | "mousedown" | "mouseenter"
| "mouseleave" | "mousemove" | "mouseout" | "mouseover" | "mouseup" => {
let evt: web_sys::MouseEvent = event.clone().dyn_into().unwrap();
#[derive(Debug)]
pub struct CustomMouseEvent(web_sys::MouseEvent);
2021-07-05 05:11:49 +00:00
impl dioxus_core::events::on::MouseEventInner for CustomMouseEvent {
2021-06-23 05:44:48 +00:00
fn alt_key(&self) -> bool {
self.0.alt_key()
2021-06-23 05:44:48 +00:00
}
fn button(&self) -> i16 {
self.0.button()
2021-06-23 05:44:48 +00:00
}
fn buttons(&self) -> u16 {
self.0.buttons()
2021-06-23 05:44:48 +00:00
}
fn client_x(&self) -> i32 {
self.0.client_x()
2021-06-23 05:44:48 +00:00
}
fn client_y(&self) -> i32 {
self.0.client_y()
2021-06-23 05:44:48 +00:00
}
fn ctrl_key(&self) -> bool {
self.0.ctrl_key()
2021-06-23 05:44:48 +00:00
}
fn meta_key(&self) -> bool {
self.0.meta_key()
2021-06-23 05:44:48 +00:00
}
fn page_x(&self) -> i32 {
self.0.page_x()
2021-06-23 05:44:48 +00:00
}
fn page_y(&self) -> i32 {
self.0.page_y()
2021-06-23 05:44:48 +00:00
}
fn screen_x(&self) -> i32 {
self.0.screen_x()
2021-06-23 05:44:48 +00:00
}
fn screen_y(&self) -> i32 {
self.0.screen_y()
2021-06-23 05:44:48 +00:00
}
fn shift_key(&self) -> bool {
self.0.shift_key()
2021-06-23 05:44:48 +00:00
}
// yikes
// https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/key/Key_Values
fn get_modifier_state(&self, key_code: &str) -> bool {
self.0.get_modifier_state(key_code)
2021-06-23 05:44:48 +00:00
}
}
2021-07-05 05:11:49 +00:00
VirtualEvent::MouseEvent(MouseEvent(Rc::new(CustomMouseEvent(evt))))
2021-06-23 05:44:48 +00:00
// MouseEvent(Box::new(RawMouseEvent {
// alt_key: evt.alt_key(),
// button: evt.button() as i32,
// buttons: evt.buttons() as i32,
// client_x: evt.client_x(),
// client_y: evt.client_y(),
// ctrl_key: evt.ctrl_key(),
// meta_key: evt.meta_key(),
// page_x: evt.page_x(),
// page_y: evt.page_y(),
// screen_x: evt.screen_x(),
// screen_y: evt.screen_y(),
// shift_key: evt.shift_key(),
// get_modifier_state: GetModifierKey(Box::new(|f| {
// // evt.get_modifier_state(f)
// todo!("This is not yet implemented properly, sorry :(");
// })),
// }))
// todo!()
// VirtualEvent::MouseEvent()
}
"pointerdown" | "pointermove" | "pointerup" | "pointercancel" | "gotpointercapture"
| "lostpointercapture" | "pointerenter" | "pointerleave" | "pointerover" | "pointerout" => {
let evt: web_sys::PointerEvent = event.clone().dyn_into().unwrap();
todo!()
}
"select" => {
// let evt: web_sys::SelectionEvent = event.clone().dyn_into().unwrap();
// not required to construct anything special beyond standard event stuff
todo!()
}
"touchcancel" | "touchend" | "touchmove" | "touchstart" => {
let evt: web_sys::TouchEvent = event.clone().dyn_into().unwrap();
todo!()
}
"scroll" => {
// let evt: web_sys::UIEvent = event.clone().dyn_into().unwrap();
todo!()
}
"wheel" => {
let evt: web_sys::WheelEvent = event.clone().dyn_into().unwrap();
todo!()
}
"abort" | "canplay" | "canplaythrough" | "durationchange" | "emptied" | "encrypted"
| "ended" | "error" | "loadeddata" | "loadedmetadata" | "loadstart" | "pause" | "play"
| "playing" | "progress" | "ratechange" | "seeked" | "seeking" | "stalled" | "suspend"
| "timeupdate" | "volumechange" | "waiting" => {
// not required to construct anything special beyond standard event stuff
// let evt: web_sys::MediaEvent = event.clone().dyn_into().unwrap();
// let evt: web_sys::MediaEvent = event.clone().dyn_into().unwrap();
todo!()
}
"animationstart" | "animationend" | "animationiteration" => {
let evt: web_sys::AnimationEvent = event.clone().dyn_into().unwrap();
todo!()
}
"transitionend" => {
let evt: web_sys::TransitionEvent = event.clone().dyn_into().unwrap();
todo!()
}
"toggle" => {
// not required to construct anything special beyond standard event stuff (target)
// let evt: web_sys::ToggleEvent = event.clone().dyn_into().unwrap();
todo!()
}
_ => VirtualEvent::OtherEvent,
}
}
2021-06-30 02:44:21 +00:00
/// This function decodes a websys event and produces an EventTrigger
/// With the websys implementation, we attach a unique key to the nodes
fn decode_trigger(event: &web_sys::Event) -> anyhow::Result<EventTrigger> {
log::debug!("Handling event!");
let target = event
.target()
.expect("missing target")
.dyn_into::<Element>()
.expect("not a valid element");
let typ = event.type_();
use anyhow::Context;
2021-07-05 05:11:49 +00:00
// for attr in {
let attrs = target.attributes();
for x in 0..attrs.length() {
let attr = attrs.item(x).unwrap();
log::debug!("attrs include: {:#?}", attr);
}
// }
// for attr in target.attributes() {
// log::debug!("attrs include: {:#?}", attr);
// }
2021-06-30 02:44:21 +00:00
// The error handling here is not very descriptive and needs to be replaced with a zero-cost error system
let val: String = target
.get_attribute(&format!("dioxus-event-{}", typ))
2021-07-05 05:11:49 +00:00
.context(format!("wrong format - received {:#?}", typ))?;
2021-06-30 02:44:21 +00:00
2021-06-30 18:08:12 +00:00
let mut fields = val.splitn(3, ".");
2021-06-30 02:44:21 +00:00
let gi_id = fields
.next()
.and_then(|f| f.parse::<u64>().ok())
2021-07-05 05:11:49 +00:00
.context("failed to parse gi id")?;
2021-06-30 02:44:21 +00:00
let el_id = fields
.next()
.and_then(|f| f.parse::<usize>().ok())
2021-07-05 05:11:49 +00:00
.context("failed to parse el id")?;
2021-06-30 02:44:21 +00:00
let real_id = fields
.next()
.and_then(|f| f.parse::<u64>().ok().map(RealDomNode::new))
2021-07-05 05:11:49 +00:00
.context("failed to parse real id")?;
2021-06-30 02:44:21 +00:00
// Call the trigger
2021-06-30 18:08:12 +00:00
log::debug!("decoded gi_id: {}, li_idx: {}", gi_id, el_id);
let triggered_scope: ScopeIdx = KeyData::from_ffi(gi_id).into();
2021-06-30 02:44:21 +00:00
Ok(EventTrigger::new(
virtual_event_from_websys_event(event),
triggered_scope,
2021-07-09 16:47:41 +00:00
Some(real_id),
2021-07-05 05:11:49 +00:00
dioxus_core::events::EventPriority::High,
2021-06-30 02:44:21 +00:00
))
}
2021-06-30 18:08:12 +00:00
struct ListenerMap {}
impl ListenerMap {
fn get(&self, event: &'static str) -> bool {
false
}
}
2021-07-05 05:11:49 +00:00
struct JsStringCache {}
impl JsStringCache {}