123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471 |
- //! Implementation of a renderer for Dioxus on the web.
- //!
- //! Oustanding todos:
- //! - Passive event listeners
- //! - no-op event listener patch for safari
- //! - tests to ensure dyn_into works for various event types.
- //! - Partial delegation?
- use dioxus_core::{
- BorrowedAttributeValue, ElementId, Mutation, Template, TemplateAttribute, TemplateNode,
- };
- use dioxus_html::{event_bubbles, CompositionData, FormData, MountedData};
- use dioxus_interpreter_js::{get_node, minimal_bindings, save_template, Channel};
- use futures_channel::mpsc;
- use js_sys::Array;
- use rustc_hash::FxHashMap;
- use std::{any::Any, rc::Rc};
- use wasm_bindgen::{closure::Closure, prelude::wasm_bindgen, JsCast, JsValue};
- use web_sys::{Document, Element, Event};
- use crate::Config;
- pub struct WebsysDom {
- document: Document,
- #[allow(dead_code)]
- pub(crate) root: Element,
- templates: FxHashMap<String, u32>,
- max_template_id: u32,
- pub(crate) interpreter: Channel,
- event_channel: mpsc::UnboundedSender<UiEvent>,
- }
- pub struct UiEvent {
- pub name: String,
- pub bubbles: bool,
- pub element: ElementId,
- pub data: Rc<dyn Any>,
- }
- impl WebsysDom {
- pub fn new(cfg: Config, event_channel: mpsc::UnboundedSender<UiEvent>) -> Self {
- // eventually, we just want to let the interpreter do all the work of decoding events into our event type
- // a match here in order to avoid some error during runtime browser test
- let document = load_document();
- let root = match document.get_element_by_id(&cfg.rootname) {
- Some(root) => root,
- None => {
- web_sys::console::error_1(
- &format!(
- "element '#{}' not found. mounting to the body.",
- cfg.rootname
- )
- .into(),
- );
- document.create_element("body").ok().unwrap()
- }
- };
- let interpreter = Channel::default();
- let handler: Closure<dyn FnMut(&Event)> = Closure::wrap(Box::new({
- let event_channel = event_channel.clone();
- move |event: &web_sys::Event| {
- let name = event.type_();
- let element = walk_event_for_id(event);
- let bubbles = dioxus_html::event_bubbles(name.as_str());
- if let Some((element, target)) = element {
- let prevent_event;
- if let Some(prevent_requests) = target
- .get_attribute("dioxus-prevent-default")
- .as_deref()
- .map(|f| f.split_whitespace())
- {
- prevent_event = prevent_requests
- .map(|f| f.trim_start_matches("on"))
- .any(|f| f == name);
- } else {
- prevent_event = false;
- }
- // Prevent forms from submitting and redirecting
- if name == "submit" {
- // On forms the default behavior is not to submit, if prevent default is set then we submit the form
- if !prevent_event {
- event.prevent_default();
- }
- } else if prevent_event {
- event.prevent_default();
- }
- let data = virtual_event_from_websys_event(event.clone(), target);
- let _ = event_channel.unbounded_send(UiEvent {
- name,
- bubbles,
- element,
- data,
- });
- }
- }
- }));
- dioxus_interpreter_js::initilize(
- root.clone().unchecked_into(),
- handler.as_ref().unchecked_ref(),
- );
- handler.forget();
- Self {
- document,
- root,
- interpreter,
- templates: FxHashMap::default(),
- max_template_id: 0,
- event_channel,
- }
- }
- pub fn mount(&mut self) {
- self.interpreter.mount_to_root();
- }
- pub fn load_templates(&mut self, templates: &[Template]) {
- for template in templates {
- let mut roots = vec![];
- for root in template.roots {
- roots.push(self.create_template_node(root))
- }
- self.templates
- .insert(template.name.to_owned(), self.max_template_id);
- save_template(roots, self.max_template_id);
- self.max_template_id += 1
- }
- }
- fn create_template_node(&self, v: &TemplateNode) -> web_sys::Node {
- use TemplateNode::*;
- match v {
- Element {
- tag,
- namespace,
- attrs,
- children,
- ..
- } => {
- let el = match namespace {
- Some(ns) => self.document.create_element_ns(Some(ns), tag).unwrap(),
- None => self.document.create_element(tag).unwrap(),
- };
- for attr in *attrs {
- if let TemplateAttribute::Static {
- name,
- value,
- namespace,
- } = attr
- {
- minimal_bindings::setAttributeInner(
- el.clone().into(),
- name,
- JsValue::from_str(value),
- *namespace,
- );
- }
- }
- for child in *children {
- let _ = el.append_child(&self.create_template_node(child));
- }
- el.dyn_into().unwrap()
- }
- Text { text } => self.document.create_text_node(text).dyn_into().unwrap(),
- DynamicText { .. } => self.document.create_text_node("p").dyn_into().unwrap(),
- Dynamic { .. } => {
- let el = self.document.create_element("pre").unwrap();
- let _ = el.toggle_attribute("hidden");
- el.dyn_into().unwrap()
- }
- }
- }
- pub fn apply_edits(&mut self, mut edits: Vec<Mutation>) {
- use Mutation::*;
- let i = &mut self.interpreter;
- // we need to apply the mount events last, so we collect them here
- let mut to_mount = Vec::new();
- for edit in &edits {
- match edit {
- AppendChildren { id, m } => i.append_children(id.0 as u32, *m as u32),
- AssignId { path, id } => {
- i.assign_id(path.as_ptr() as u32, path.len() as u8, id.0 as u32)
- }
- CreatePlaceholder { id } => i.create_placeholder(id.0 as u32),
- CreateTextNode { value, id } => i.create_text_node(value, id.0 as u32),
- HydrateText { path, value, id } => {
- i.hydrate_text(path.as_ptr() as u32, path.len() as u8, value, id.0 as u32)
- }
- LoadTemplate { name, index, id } => {
- if let Some(tmpl_id) = self.templates.get(*name) {
- i.load_template(*tmpl_id, *index as u32, id.0 as u32)
- }
- }
- ReplaceWith { id, m } => i.replace_with(id.0 as u32, *m as u32),
- ReplacePlaceholder { path, m } => {
- i.replace_placeholder(path.as_ptr() as u32, path.len() as u8, *m as u32)
- }
- InsertAfter { id, m } => i.insert_after(id.0 as u32, *m as u32),
- InsertBefore { id, m } => i.insert_before(id.0 as u32, *m as u32),
- SetAttribute {
- name,
- value,
- id,
- ns,
- } => match value {
- BorrowedAttributeValue::Text(txt) => {
- i.set_attribute(id.0 as u32, name, txt, ns.unwrap_or_default())
- }
- BorrowedAttributeValue::Float(f) => {
- i.set_attribute(id.0 as u32, name, &f.to_string(), ns.unwrap_or_default())
- }
- BorrowedAttributeValue::Int(n) => {
- i.set_attribute(id.0 as u32, name, &n.to_string(), ns.unwrap_or_default())
- }
- BorrowedAttributeValue::Bool(b) => i.set_attribute(
- id.0 as u32,
- name,
- if *b { "true" } else { "false" },
- ns.unwrap_or_default(),
- ),
- BorrowedAttributeValue::None => {
- i.remove_attribute(id.0 as u32, name, ns.unwrap_or_default())
- }
- _ => unreachable!(),
- },
- SetText { value, id } => i.set_text(id.0 as u32, value),
- NewEventListener { name, id, .. } => {
- match *name {
- // mounted events are fired immediately after the element is mounted.
- "mounted" => {
- to_mount.push(*id);
- }
- _ => {
- i.new_event_listener(name, id.0 as u32, event_bubbles(name) as u8);
- }
- }
- }
- RemoveEventListener { name, id } => match *name {
- "mounted" => {}
- _ => {
- i.remove_event_listener(name, id.0 as u32, event_bubbles(name) as u8);
- }
- },
- Remove { id } => i.remove(id.0 as u32),
- PushRoot { id } => i.push_root(id.0 as u32),
- }
- }
- edits.clear();
- i.flush();
- for id in to_mount {
- self.send_mount_event(id);
- }
- }
- pub(crate) fn send_mount_event(&self, id: ElementId) {
- let node = get_node(id.0 as u32);
- if let Some(element) = node.dyn_ref::<Element>() {
- let data: MountedData = element.into();
- let data = Rc::new(data);
- let _ = self.event_channel.unbounded_send(UiEvent {
- name: "mounted".to_string(),
- bubbles: false,
- element: id,
- data,
- });
- }
- }
- }
- // todo: some of these events are being casted to the wrong event type.
- // We need tests that simulate clicks/etc and make sure every event type works.
- pub fn virtual_event_from_websys_event(event: web_sys::Event, target: Element) -> Rc<dyn Any> {
- use dioxus_html::events::*;
- match event.type_().as_str() {
- "copy" | "cut" | "paste" => Rc::new(ClipboardData {}),
- "compositionend" | "compositionstart" | "compositionupdate" => {
- make_composition_event(&event)
- }
- "keydown" | "keypress" | "keyup" => Rc::new(KeyboardData::from(event)),
- "focus" | "blur" | "focusout" | "focusin" => Rc::new(FocusData {}),
- "change" | "input" | "invalid" | "reset" | "submit" => read_input_to_data(target),
- "click" | "contextmenu" | "dblclick" | "doubleclick" | "mousedown" | "mouseenter"
- | "mouseleave" | "mousemove" | "mouseout" | "mouseover" | "mouseup" => {
- Rc::new(MouseData::from(event))
- }
- "drag" | "dragend" | "dragenter" | "dragexit" | "dragleave" | "dragover" | "dragstart"
- | "drop" => {
- let mouse = MouseData::from(event);
- Rc::new(DragData { mouse })
- }
- "pointerdown" | "pointermove" | "pointerup" | "pointercancel" | "gotpointercapture"
- | "lostpointercapture" | "pointerenter" | "pointerleave" | "pointerover" | "pointerout" => {
- Rc::new(PointerData::from(event))
- }
- "select" => Rc::new(SelectionData {}),
- "touchcancel" | "touchend" | "touchmove" | "touchstart" => Rc::new(TouchData::from(event)),
- "scroll" => Rc::new(ScrollData {}),
- "wheel" => Rc::new(WheelData::from(event)),
- "animationstart" | "animationend" | "animationiteration" => {
- Rc::new(AnimationData::from(event))
- }
- "transitionend" => Rc::new(TransitionData::from(event)),
- "abort" | "canplay" | "canplaythrough" | "durationchange" | "emptied" | "encrypted"
- | "ended" | "loadeddata" | "loadedmetadata" | "loadstart" | "pause" | "play"
- | "playing" | "progress" | "ratechange" | "seeked" | "seeking" | "stalled" | "suspend"
- | "timeupdate" | "volumechange" | "waiting" => Rc::new(MediaData {}),
- "error" => Rc::new(ImageData { load_error: true }),
- "load" => Rc::new(ImageData { load_error: false }),
- "toggle" => Rc::new(ToggleData {}),
- _ => Rc::new(()),
- }
- }
- fn make_composition_event(event: &Event) -> Rc<CompositionData> {
- let evt: &web_sys::CompositionEvent = event.dyn_ref().unwrap();
- Rc::new(CompositionData {
- data: evt.data().unwrap_or_default(),
- })
- }
- pub(crate) fn load_document() -> Document {
- web_sys::window()
- .expect("should have access to the Window")
- .document()
- .expect("should have access to the Document")
- }
- fn read_input_to_data(target: Element) -> Rc<FormData> {
- // todo: these handlers might get really slow if the input box gets large and allocation pressure is heavy
- // don't have a good solution with the serialized event problem
- let value: String = target
- .dyn_ref()
- .map(|input: &web_sys::HtmlInputElement| {
- // todo: special case more input types
- match input.type_().as_str() {
- "checkbox" => {
- match input.checked() {
- true => "true".to_string(),
- false => "false".to_string(),
- }
- },
- _ => {
- input.value()
- }
- }
- })
- .or_else(|| {
- target
- .dyn_ref()
- .map(|input: &web_sys::HtmlTextAreaElement| input.value())
- })
- // select elements are NOT input events - because - why woudn't they be??
- .or_else(|| {
- target
- .dyn_ref()
- .map(|input: &web_sys::HtmlSelectElement| input.value())
- })
- .or_else(|| {
- target
- .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 mut values = std::collections::HashMap::new();
- // try to fill in form values
- if let Some(form) = target.dyn_ref::<web_sys::HtmlFormElement>() {
- let form_data = get_form_data(form);
- for value in form_data.entries().into_iter().flatten() {
- if let Ok(array) = value.dyn_into::<Array>() {
- if let Some(name) = array.get(0).as_string() {
- if let Ok(item_values) = array.get(1).dyn_into::<Array>() {
- let item_values =
- item_values.iter().filter_map(|v| v.as_string()).collect();
- values.insert(name, item_values);
- }
- }
- }
- }
- }
- #[cfg(not(feature = "file_engine"))]
- let files = None;
- #[cfg(feature = "file_engine")]
- let files = target
- .dyn_ref()
- .and_then(|input: &web_sys::HtmlInputElement| {
- input.files().and_then(|files| {
- #[allow(clippy::arc_with_non_send_sync)]
- crate::file_engine::WebFileEngine::new(files)
- .map(|f| std::sync::Arc::new(f) as std::sync::Arc<dyn dioxus_html::FileEngine>)
- })
- });
- Rc::new(FormData {
- value,
- values,
- files,
- })
- }
- // web-sys does not expose the keys api for form data, so we need to manually bind to it
- #[wasm_bindgen(inline_js = r#"
- export function get_form_data(form) {
- let values = new Map();
- const formData = new FormData(form);
- for (let name of formData.keys()) {
- values.set(name, formData.getAll(name));
- }
- return values;
- }
- "#)]
- extern "C" {
- fn get_form_data(form: &web_sys::HtmlFormElement) -> js_sys::Map;
- }
- fn walk_event_for_id(event: &web_sys::Event) -> Option<(ElementId, web_sys::Element)> {
- let target = event
- .target()
- .expect("missing target")
- .dyn_into::<web_sys::Node>()
- .expect("not a valid node");
- let mut current_target_element = target.dyn_ref::<web_sys::Element>().cloned();
- loop {
- match (
- current_target_element
- .as_ref()
- .and_then(|el| el.get_attribute("data-dioxus-id").map(|f| f.parse())),
- current_target_element,
- ) {
- // This node is an element, and has a dioxus id, so we can stop walking
- (Some(Ok(id)), Some(target)) => return Some((ElementId(id), target)),
- // Walk the tree upwards until we actually find an event target
- (None, target_element) => {
- let parent = match target_element.as_ref() {
- Some(el) => el.parent_element(),
- // if this is the first node and not an element, we need to get the parent from the target node
- None => target.parent_element(),
- };
- match parent {
- Some(parent) => current_target_element = Some(parent),
- _ => return None,
- }
- }
- // This node is an element with an invalid dioxus id, give up
- _ => return None,
- }
- }
- }
|