123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604 |
- use std::{collections::HashMap, rc::Rc, sync::Arc};
- use dioxus_core::{
- events::{EventTrigger, VirtualEvent},
- prelude::ScopeIdx,
- virtual_dom::RealDomNode,
- };
- use fxhash::FxHashMap;
- use nohash_hasher::IntMap;
- use wasm_bindgen::{closure::Closure, JsCast};
- use web_sys::{
- window, Document, Element, Event, HtmlElement, HtmlInputElement, HtmlOptionElement, Node,
- };
- pub struct WebsysDom {
- pub stack: Stack,
- nodes: IntMap<u32, Node>,
- document: Document,
- root: Element,
- event_receiver: async_channel::Receiver<EventTrigger>,
- trigger: Arc<dyn Fn(EventTrigger)>,
- // every callback gets a monotomically increasing callback ID
- callback_id: usize,
- // map of listener types to number of those listeners
- listeners: FxHashMap<String, (usize, Closure<dyn FnMut(&Event)>)>,
- // Map of callback_id to component index and listener id
- callback_map: FxHashMap<usize, (usize, usize)>,
- // 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
- //
- // `ptns` = Percy text node separator
- // TODO
- last_node_was_text: bool,
- node_counter: Counter,
- }
- 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");
- let (sender, mut receiver) = async_channel::unbounded::<EventTrigger>();
- let sender_callback = Arc::new(move |ev| {
- let mut c = sender.clone();
- wasm_bindgen_futures::spawn_local(async move {
- c.send(ev).await.unwrap();
- });
- });
- let mut nodes =
- HashMap::with_capacity_and_hasher(1000, nohash_hasher::BuildNoHashHasher::default());
- nodes.insert(0_u32, root.clone().dyn_into::<Node>().unwrap());
- Self {
- stack: Stack::with_capacity(10),
- nodes,
- callback_id: 0,
- listeners: FxHashMap::default(),
- callback_map: FxHashMap::default(),
- document,
- event_receiver: receiver,
- trigger: sender_callback,
- root,
- last_node_was_text: false,
- node_counter: Counter(0),
- }
- }
- pub async fn wait_for_event(&mut self) -> Option<EventTrigger> {
- let v = self.event_receiver.recv().await.unwrap();
- Some(v)
- }
- }
- struct Counter(u32);
- impl Counter {
- fn next(&mut self) -> u32 {
- self.0 += 1;
- self.0
- }
- }
- impl dioxus_core::diff::RealDom for WebsysDom {
- fn push_root(&mut self, root: dioxus_core::virtual_dom::RealDomNode) {
- log::debug!("Called `[`push_root] {:?}", root);
- let domnode = self.nodes.get(&root.0).expect("Failed to pop know root");
- self.stack.push(domnode.clone());
- }
- fn append_child(&mut self) {
- 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) {
- 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) {
- log::debug!("Called [`remove`]");
- todo!()
- }
- fn remove_all_children(&mut self) {
- log::debug!("Called [`remove_all_children`]");
- todo!()
- }
- fn create_text_node(&mut self, text: &str) -> dioxus_core::virtual_dom::RealDomNode {
- let nid = self.node_counter.next();
- let textnode = self
- .document
- .create_text_node(text)
- .dyn_into::<Node>()
- .unwrap();
- self.stack.push(textnode.clone());
- self.nodes.insert(nid, textnode);
- log::debug!("Called [`create_text_node`]: {}, {}", text, nid);
- RealDomNode::new(nid)
- }
- fn create_element(&mut self, tag: &str) -> dioxus_core::virtual_dom::RealDomNode {
- let el = self
- .document
- .create_element(tag)
- .unwrap()
- .dyn_into::<Node>()
- .unwrap();
- self.stack.push(el.clone());
- let nid = self.node_counter.next();
- self.nodes.insert(nid, el);
- log::debug!("Called [`create_element`]: {}, {:?}", tag, nid);
- RealDomNode::new(nid)
- }
- fn create_element_ns(
- &mut self,
- tag: &str,
- namespace: &str,
- ) -> dioxus_core::virtual_dom::RealDomNode {
- let el = self
- .document
- .create_element_ns(Some(namespace), tag)
- .unwrap()
- .dyn_into::<Node>()
- .unwrap();
- self.stack.push(el.clone());
- let nid = self.node_counter.next();
- self.nodes.insert(nid, el);
- log::debug!("Called [`create_element_ns`]: {:}", nid);
- RealDomNode::new(nid)
- }
- fn new_event_listener(
- &mut self,
- event: &str,
- scope: dioxus_core::prelude::ScopeIdx,
- el_id: usize,
- real_id: RealDomNode,
- ) {
- 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));
- let (gi_id, gi_gen) = (&scope).into_raw_parts();
- el.set_attribute(
- &format!("dioxus-event-{}", event),
- &format!("{}.{}.{}.{}", gi_id, gi_gen, el_id, real_id.0),
- )
- .unwrap();
- // Register the callback to decode
- if let Some(entry) = self.listeners.get_mut(event) {
- entry.0 += 1;
- } else {
- let trigger = self.trigger.clone();
- 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
- let res = || -> 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;
- let val: String = target
- .get_attribute(&format!("dioxus-event-{}", typ))
- .context("")?;
- let mut fields = val.splitn(4, ".");
- let gi_id = fields
- .next()
- .and_then(|f| f.parse::<usize>().ok())
- .context("")?;
- let gi_gen = fields
- .next()
- .and_then(|f| f.parse::<u64>().ok())
- .context("")?;
- let el_id = fields
- .next()
- .and_then(|f| f.parse::<usize>().ok())
- .context("")?;
- let real_id = fields
- .next()
- .and_then(|f| f.parse::<u32>().ok().map(RealDomNode::new))
- .context("")?;
- // Call the trigger
- log::debug!(
- "decoded gi_id: {}, gi_gen: {}, li_idx: {}",
- gi_id,
- gi_gen,
- el_id
- );
- let triggered_scope = ScopeIdx::from_raw_parts(gi_id, gi_gen);
- Ok(EventTrigger::new(
- virtual_event_from_websys_event(event),
- triggered_scope,
- real_id,
- ))
- };
- match res() {
- Ok(synthetic_event) => trigger.as_ref()(synthetic_event),
- Err(_) => log::error!("Error decoding Dioxus event attribute."),
- };
- }) 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) {
- log::debug!("Called [`remove_event_listener`]: {}", event);
- todo!()
- }
- fn set_text(&mut self, text: &str) {
- log::debug!("Called [`set_text`]: {}", text);
- self.stack.top().set_text_content(Some(text))
- }
- fn set_attribute(&mut self, name: &str, value: &str, is_namespaced: bool) {
- 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 {
- if let Some(el) = self.stack.top().dyn_ref::<Element>() {
- el.set_attribute(name, value).unwrap();
- }
- }
- }
- fn remove_attribute(&mut self, name: &str) {
- 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 {
- log::debug!("Called [`raw_node_as_any_mut`]");
- todo!()
- }
- }
- #[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);
- impl dioxus_core::events::on::MouseEvent for CustomMouseEvent {
- fn alt_key(&self) -> bool {
- self.0.alt_key()
- }
- fn button(&self) -> i16 {
- self.0.button()
- }
- fn buttons(&self) -> u16 {
- self.0.buttons()
- }
- fn client_x(&self) -> i32 {
- self.0.client_x()
- }
- fn client_y(&self) -> i32 {
- self.0.client_y()
- }
- fn ctrl_key(&self) -> bool {
- self.0.ctrl_key()
- }
- fn meta_key(&self) -> bool {
- self.0.meta_key()
- }
- fn page_x(&self) -> i32 {
- self.0.page_x()
- }
- fn page_y(&self) -> i32 {
- self.0.page_y()
- }
- fn screen_x(&self) -> i32 {
- self.0.screen_x()
- }
- fn screen_y(&self) -> i32 {
- self.0.screen_y()
- }
- fn shift_key(&self) -> bool {
- self.0.shift_key()
- }
- // 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)
- }
- }
- VirtualEvent::MouseEvent(Rc::new(CustomMouseEvent(evt)))
- // 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,
- }
- }
|