dom.rs 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. //! Implementation of a renderer for Dioxus on the web.
  2. //!
  3. //! Outstanding todos:
  4. //! - Passive event listeners
  5. //! - no-op event listener patch for safari
  6. //! - tests to ensure dyn_into works for various event types.
  7. //! - Partial delegation?
  8. use dioxus_core::ElementId;
  9. use dioxus_html::PlatformEventData;
  10. use dioxus_interpreter_js::Channel;
  11. use futures_channel::mpsc;
  12. use rustc_hash::FxHashMap;
  13. use wasm_bindgen::{closure::Closure, JsCast};
  14. use web_sys::{Document, Element, Event};
  15. use crate::{load_document, virtual_event_from_websys_event, Config, WebEventConverter};
  16. pub struct WebsysDom {
  17. pub(crate) document: Document,
  18. #[allow(dead_code)]
  19. pub(crate) root: Element,
  20. pub(crate) templates: FxHashMap<String, u16>,
  21. pub(crate) max_template_id: u16,
  22. pub(crate) interpreter: Channel,
  23. #[cfg(feature = "mounted")]
  24. pub(crate) event_channel: mpsc::UnboundedSender<UiEvent>,
  25. #[cfg(feature = "mounted")]
  26. pub(crate) queued_mounted_events: Vec<ElementId>,
  27. }
  28. pub struct UiEvent {
  29. pub name: String,
  30. pub bubbles: bool,
  31. pub element: ElementId,
  32. pub data: PlatformEventData,
  33. }
  34. //fn get_document(elem: &web_sys::Element) ->
  35. impl WebsysDom {
  36. pub fn new(cfg: Config, event_channel: mpsc::UnboundedSender<UiEvent>) -> Self {
  37. let (document, root) = match cfg.root {
  38. crate::cfg::ConfigRoot::RootName(rootname) => {
  39. // eventually, we just want to let the interpreter do all the work of decoding events into our event type
  40. // a match here in order to avoid some error during runtime browser test
  41. let document = load_document();
  42. let root = match document.get_element_by_id(&rootname) {
  43. Some(root) => root,
  44. None => {
  45. web_sys::console::error_1(
  46. &format!("element '#{}' not found. mounting to the body.", rootname)
  47. .into(),
  48. );
  49. document.create_element("body").ok().unwrap()
  50. }
  51. };
  52. (document, root)
  53. }
  54. crate::cfg::ConfigRoot::RootElement(root) => {
  55. let document = match root.owner_document() {
  56. Some(document) => document,
  57. None => load_document(),
  58. };
  59. (document, root)
  60. }
  61. };
  62. let interpreter = Channel::default();
  63. let handler: Closure<dyn FnMut(&Event)> = Closure::wrap(Box::new({
  64. let event_channel = event_channel.clone();
  65. move |event: &web_sys::Event| {
  66. let name = event.type_();
  67. let element = walk_event_for_id(event);
  68. let bubbles = dioxus_html::event_bubbles(name.as_str());
  69. if let Some((element, target)) = element {
  70. let prevent_event;
  71. if let Some(prevent_requests) = target
  72. .get_attribute("dioxus-prevent-default")
  73. .as_deref()
  74. .map(|f| f.split_whitespace())
  75. {
  76. prevent_event = prevent_requests
  77. .map(|f| f.trim_start_matches("on"))
  78. .any(|f| f == name);
  79. } else {
  80. prevent_event = false;
  81. }
  82. // Prevent forms from submitting and redirecting
  83. if name == "submit" {
  84. // On forms the default behavior is not to submit, if prevent default is set then we submit the form
  85. if !prevent_event {
  86. event.prevent_default();
  87. }
  88. } else if prevent_event {
  89. event.prevent_default();
  90. }
  91. let data = virtual_event_from_websys_event(event.clone(), target);
  92. let _ = event_channel.unbounded_send(UiEvent {
  93. name,
  94. bubbles,
  95. element,
  96. data,
  97. });
  98. }
  99. }
  100. }));
  101. dioxus_interpreter_js::initialize(
  102. root.clone().unchecked_into(),
  103. handler.as_ref().unchecked_ref(),
  104. );
  105. dioxus_html::set_event_converter(Box::new(WebEventConverter));
  106. handler.forget();
  107. Self {
  108. document,
  109. root,
  110. interpreter,
  111. templates: FxHashMap::default(),
  112. max_template_id: 0,
  113. #[cfg(feature = "mounted")]
  114. event_channel,
  115. #[cfg(feature = "mounted")]
  116. queued_mounted_events: Default::default(),
  117. }
  118. }
  119. pub fn mount(&mut self) {
  120. self.interpreter.mount_to_root();
  121. }
  122. }
  123. fn walk_event_for_id(event: &web_sys::Event) -> Option<(ElementId, web_sys::Element)> {
  124. let target = event
  125. .target()
  126. .expect("missing target")
  127. .dyn_into::<web_sys::Node>()
  128. .expect("not a valid node");
  129. let mut current_target_element = target.dyn_ref::<web_sys::Element>().cloned();
  130. loop {
  131. match (
  132. current_target_element
  133. .as_ref()
  134. .and_then(|el| el.get_attribute("data-dioxus-id").map(|f| f.parse())),
  135. current_target_element,
  136. ) {
  137. // This node is an element, and has a dioxus id, so we can stop walking
  138. (Some(Ok(id)), Some(target)) => return Some((ElementId(id), target)),
  139. // Walk the tree upwards until we actually find an event target
  140. (None, target_element) => {
  141. let parent = match target_element.as_ref() {
  142. Some(el) => el.parent_element(),
  143. // if this is the first node and not an element, we need to get the parent from the target node
  144. None => target.parent_element(),
  145. };
  146. match parent {
  147. Some(parent) => current_target_element = Some(parent),
  148. _ => return None,
  149. }
  150. }
  151. // This node is an element with an invalid dioxus id, give up
  152. _ => return None,
  153. }
  154. }
  155. }