use dioxus_core::ScopeState; use dioxus_interpreter_js::{COMMON_JS, INTERPRETER_JS}; use slab::Slab; use std::{ borrow::Cow, future::Future, ops::{Deref, DerefMut}, path::{Path, PathBuf}, pin::Pin, rc::Rc, sync::Arc, }; use tokio::{ runtime::Handle, sync::{OnceCell, RwLock}, }; use wry::{ http::{status::StatusCode, Request, Response}, Result, }; use crate::{use_window, DesktopContext}; fn module_loader(root_name: &str) -> String { let js = INTERPRETER_JS.replace( "/*POST_HANDLE_EDITS*/", r#"// Prevent file inputs from opening the file dialog on click let inputs = document.querySelectorAll("input"); for (let input of inputs) { if (!input.getAttribute("data-dioxus-file-listener")) { // prevent file inputs from opening the file dialog on click const type = input.getAttribute("type"); if (type === "file") { input.setAttribute("data-dioxus-file-listener", true); input.addEventListener("click", (event) => { let target = event.target; let target_id = find_real_id(target); if (target_id !== null) { const send = (event_name) => { const message = serializeIpcMessage("file_diolog", { accept: target.getAttribute("accept"), directory: target.getAttribute("webkitdirectory") === "true", multiple: target.hasAttribute("multiple"), target: parseInt(target_id), bubbles: event_bubbles(event_name), event: event_name }); window.ipc.postMessage(message); }; send("change&input"); } event.preventDefault(); }); } } }"#, ); format!( r#" "# ) } /// An arbitrary asset is an HTTP response containing a binary body. pub type AssetResponse = Response>; /// A future that returns an [`AssetResponse`]. This future may be spawned in a new thread, /// so it must be [`Send`], [`Sync`], and `'static`. pub trait AssetFuture: Future> + Send + Sync + 'static {} impl> + Send + Sync + 'static> AssetFuture for T {} /// A request for an asset. This is a wrapper around [`Request>`] that provides methods specific to asset requests. pub struct AssetRequest { path: PathBuf, request: Request>, } impl AssetRequest { /// Get the path the asset request is for pub fn path(&self) -> &Path { &self.path } } impl From>> for AssetRequest { fn from(request: Request>) -> Self { let decoded = urlencoding::decode(request.uri().path().trim_start_matches('/')) .expect("expected URL to be UTF-8 encoded"); let path = PathBuf::from(&*decoded); Self { request, path } } } impl Deref for AssetRequest { type Target = Request>; fn deref(&self) -> &Self::Target { &self.request } } impl DerefMut for AssetRequest { fn deref_mut(&mut self) -> &mut Self::Target { &mut self.request } } /// A handler that takes an [`AssetRequest`] and returns a future that either loads the asset, or returns `None`. /// This handler is stashed indefinitely in a context object, so it must be `'static`. pub trait AssetHandler: Send + Sync + 'static { /// Handle an asset request, returning a future that either loads the asset, or returns `None` fn handle_request(&self, request: &AssetRequest) -> F; } impl F + Send + Sync + 'static> AssetHandler for T { fn handle_request(&self, request: &AssetRequest) -> F { self(request) } } type AssetHandlerRegistryInner = Slab Pin> + Send + Sync + 'static>>; #[derive(Clone)] pub struct AssetHandlerRegistry(Arc>); impl AssetHandlerRegistry { pub fn new() -> Self { AssetHandlerRegistry(Arc::new(RwLock::new(Slab::new()))) } pub async fn register_handler(&self, f: impl AssetHandler) -> usize { let mut registry = self.0.write().await; registry.insert(Box::new(move |req| Box::pin(f.handle_request(req)))) } pub async fn remove_handler(&self, id: usize) -> Option<()> { let mut registry = self.0.write().await; registry.try_remove(id).map(|_| ()) } pub async fn try_handlers(&self, req: &AssetRequest) -> Option { let registry = self.0.read().await; for (_, handler) in registry.iter() { if let Some(response) = handler(req).await { return Some(response); } } None } } /// A handle to a registered asset handler. pub struct AssetHandlerHandle { desktop: DesktopContext, handler_id: Rc>, } impl AssetHandlerHandle { /// Returns the ID for this handle. /// /// Because registering an ID is asynchronous, this may return `None` if the /// registration has not completed yet. pub fn handler_id(&self) -> Option { self.handler_id.get().copied() } } impl Drop for AssetHandlerHandle { fn drop(&mut self) { let cell = Rc::clone(&self.handler_id); let desktop = Rc::clone(&self.desktop); tokio::task::block_in_place(move || { Handle::current().block_on(async move { if let Some(id) = cell.get() { desktop.asset_handlers.remove_handler(*id).await; } }) }); } } /// Provide a callback to handle asset loading yourself. /// /// The callback takes a path as requested by the web view, and it should return `Some(response)` /// if you want to load the asset, and `None` if you want to fallback on the default behavior. pub fn use_asset_handler( cx: &ScopeState, handler: impl AssetHandler, ) -> &AssetHandlerHandle { let desktop = Rc::clone(use_window(cx)); cx.use_hook(|| { let handler_id = Rc::new(OnceCell::new()); let handler_id_ref = Rc::clone(&handler_id); let desktop_ref = Rc::clone(&desktop); cx.push_future(async move { let id = desktop.asset_handlers.register_handler(handler).await; handler_id.set(id).unwrap(); }); AssetHandlerHandle { desktop: desktop_ref, handler_id: handler_id_ref, } }) } pub(super) async fn desktop_handler( request: Request>, custom_head: Option, custom_index: Option, root_name: &str, asset_handlers: &AssetHandlerRegistry, ) -> Result { let request = AssetRequest::from(request); // If the request is for the root, we'll serve the index.html file. if request.uri().path() == "/" { // If a custom index is provided, just defer to that, expecting the user to know what they're doing. // we'll look for the closing tag and insert our little module loader there. let body = match custom_index { Some(custom_index) => custom_index .replace("", &format!("{}", module_loader(root_name))) .into_bytes(), None => { // Otherwise, we'll serve the default index.html and apply a custom head if that's specified. let mut template = include_str!("./index.html").to_string(); if let Some(custom_head) = custom_head { template = template.replace("", &custom_head); } template .replace("", &module_loader(root_name)) .into_bytes() } }; return Response::builder() .header("Content-Type", "text/html") .body(Cow::from(body)) .map_err(From::from); } else if request.uri().path() == "/common.js" { return Response::builder() .header("Content-Type", "text/javascript") .body(Cow::from(COMMON_JS.as_bytes())) .map_err(From::from); } // If the user provided a custom asset handler, then call it and return the response // if the request was handled. if let Some(response) = asset_handlers.try_handlers(&request).await { return Ok(response); } // Else, try to serve a file from the filesystem. // If the path is relative, we'll try to serve it from the assets directory. let mut asset = get_asset_root() .unwrap_or_else(|| Path::new(".").to_path_buf()) .join(&request.path); if !asset.exists() { asset = PathBuf::from("/").join(request.path); } if asset.exists() { return Response::builder() .header("Content-Type", get_mime_from_path(&asset)?) .body(Cow::from(std::fs::read(asset)?)) .map_err(From::from); } Response::builder() .status(StatusCode::NOT_FOUND) .body(Cow::from(String::from("Not Found").into_bytes())) .map_err(From::from) } #[allow(unreachable_code)] fn get_asset_root() -> Option { /* We're matching exactly how cargo-bundle works. - [x] macOS - [ ] Windows - [ ] Linux (rpm) - [ ] Linux (deb) - [ ] iOS - [ ] Android */ if std::env::var_os("CARGO").is_some() { return None; } // TODO: support for other platforms #[cfg(target_os = "macos")] { let bundle = core_foundation::bundle::CFBundle::main_bundle(); let bundle_path = bundle.path()?; let resources_path = bundle.resources_path()?; let absolute_resources_root = bundle_path.join(resources_path); let canonical_resources_root = dunce::canonicalize(absolute_resources_root).ok()?; return Some(canonical_resources_root); } None } /// Get the mime type from a path-like string fn get_mime_from_path(trimmed: &Path) -> Result<&'static str> { if trimmed.extension().is_some_and(|ext| ext == "svg") { return Ok("image/svg+xml"); } let res = match infer::get_from_path(trimmed)?.map(|f| f.mime_type()) { Some(f) => { if f == "text/plain" { get_mime_by_ext(trimmed) } else { f } } None => get_mime_by_ext(trimmed), }; Ok(res) } /// Get the mime type from a URI using its extension fn get_mime_by_ext(trimmed: &Path) -> &'static str { match trimmed.extension().and_then(|e| e.to_str()) { Some("bin") => "application/octet-stream", Some("css") => "text/css", Some("csv") => "text/csv", Some("html") => "text/html", Some("ico") => "image/vnd.microsoft.icon", Some("js") => "text/javascript", Some("json") => "application/json", Some("jsonld") => "application/ld+json", Some("mjs") => "text/javascript", Some("rtf") => "application/rtf", Some("svg") => "image/svg+xml", Some("mp4") => "video/mp4", // Assume HTML when a TLD is found for eg. `dioxus:://dioxuslabs.app` | `dioxus://hello.com` Some(_) => "text/html", // https://developer.mozilla.org/en-US/docs/Web/HTTP/Basics_of_HTTP/MIME_types/Common_types // using octet stream according to this: None => "application/octet-stream", } }