mirror of
https://github.com/DioxusLabs/dioxus
synced 2025-01-19 16:24:01 +00:00
445 lines
17 KiB
Rust
445 lines
17 KiB
Rust
//! Implementation of a renderer for Dioxus on the web.
|
|
//!
|
|
//! Oustanding todos:
|
|
//! - Removing event listeners (delegation)
|
|
//! - 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, FileEngine, 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, sync::Arc};
|
|
use wasm_bindgen::{closure::Closure, prelude::wasm_bindgen, JsCast, JsValue};
|
|
use web_sys::{Document, Element, Event};
|
|
|
|
use crate::{file_engine::WebFileEngine, 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 => 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 {
|
|
if let Some(prevent_requests) = target
|
|
.get_attribute("dioxus-prevent-default")
|
|
.as_deref()
|
|
.map(|f| f.split_whitespace())
|
|
{
|
|
if prevent_requests
|
|
.map(|f| f.trim_start_matches("on"))
|
|
.any(|f| f == name)
|
|
{
|
|
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 {
|
|
let node = get_node(id.0 as u32);
|
|
if let Some(element) = node.dyn_ref::<Element>() {
|
|
log::info!("mounted event fired: {}", id.0);
|
|
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(()),
|
|
"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);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
let files = target
|
|
.dyn_ref()
|
|
.and_then(|input: &web_sys::HtmlInputElement| {
|
|
input.files().and_then(|files| {
|
|
WebFileEngine::new(files).map(|f| Arc::new(f) as Arc<dyn 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,
|
|
}
|
|
}
|
|
}
|