123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378 |
- //! Dioxus Desktop Renderer
- //!
- //! Render the Dioxus VirtualDom using the platform's native WebView implementation.
- //!
- //! # Desktop
- //!
- //! One of Dioxus' killer features is the ability to quickly build a native desktop app that looks and feels the same across platforms. Apps built with Dioxus are typically <5mb in size and use existing system resources, so they won't hog extreme amounts of RAM or memory.
- //!
- //! Dioxus Desktop is built off Tauri. Right now there aren't any Dioxus abstractions over keyboard shortcuts, menubar, handling, etc, so you'll want to leverage Tauri - mostly [Wry](http://github.com/tauri-apps/wry/) and [Tao](http://github.com/tauri-apps/tao)) directly. The next major release of Dioxus-Desktop will include components and hooks for notifications, global shortcuts, menubar, etc.
- //!
- //!
- //! ## Getting Set up
- //!
- //! Getting Set up with Dioxus-Desktop is quite easy. Make sure you have Rust and Cargo installed, and then create a new project:
- //!
- //! ```shell
- //! $ cargo new --bin demo
- //! $ cd app
- //! ```
- //!
- //! Add Dioxus with the `desktop` feature:
- //!
- //! ```shell
- //! $ cargo add dioxus --features desktop
- //! ```
- //!
- //! Edit your `main.rs`:
- //!
- //! ```rust
- //! // main.rs
- //! use dioxus::prelude::*;
- //!
- //! fn main() {
- //! dioxus::desktop::launch(app);
- //! }
- //!
- //! fn app(cx: Scope) -> Element {
- //! cx.render(rsx!{
- //! div {
- //! "hello world!"
- //! }
- //! })
- //! }
- //! ```
- //!
- //!
- //! To configure the webview, menubar, and other important desktop-specific features, checkout out some of the launch configuration in the [API reference](https://docs.rs/dioxus-desktop/).
- //!
- //! ## Future Steps
- //!
- //! Make sure to read the [Dioxus Guide](https://dioxuslabs.com/guide) if you already haven't!
- pub mod cfg;
- pub mod escape;
- pub mod events;
- use cfg::DesktopConfig;
- use dioxus_core::*;
- use std::{
- collections::{HashMap, VecDeque},
- sync::atomic::AtomicBool,
- sync::{Arc, RwLock},
- };
- use tao::{
- accelerator::{Accelerator, SysMods},
- event::{Event, StartCause, WindowEvent},
- event_loop::{ControlFlow, EventLoop, EventLoopWindowTarget},
- keyboard::{KeyCode, ModifiersState},
- menu::{MenuBar, MenuItem},
- window::{Window, WindowId},
- };
- pub use wry;
- pub use wry::application as tao;
- use wry::{
- application::event_loop::EventLoopProxy,
- webview::RpcRequest,
- webview::{WebView, WebViewBuilder},
- };
- /// Launch the WebView and run the event loop.
- ///
- /// This function will start a multithreaded Tokio runtime as well the WebView event loop.
- ///
- /// ```rust
- /// use dioxus::prelude::*;
- ///
- /// fn main() {
- /// dioxus::desktop::launch(app);
- /// }
- ///
- /// fn app(cx: Scope) -> Element {
- /// cx.render(rsx!{
- /// h1 {"hello world!"}
- /// })
- /// }
- /// ```
- pub fn launch(root: Component) {
- launch_with_props(root, (), |c| c)
- }
- /// Launch the WebView and run the event loop, with configuration.
- ///
- /// This function will start a multithreaded Tokio runtime as well the WebView event loop.
- ///
- /// You can configure the WebView window with a configuration closure
- ///
- /// ```rust
- /// use dioxus::prelude::*;
- ///
- /// fn main() {
- /// dioxus::desktop::launch_cfg(app, |c| c.with_window(|w| w.with_title("My App")));
- /// }
- ///
- /// fn app(cx: Scope) -> Element {
- /// cx.render(rsx!{
- /// h1 {"hello world!"}
- /// })
- /// }
- /// ```
- pub fn launch_cfg(
- root: Component,
- config_builder: impl for<'a, 'b> FnOnce(&'b mut DesktopConfig<'a>) -> &'b mut DesktopConfig<'a>,
- ) {
- launch_with_props(root, (), config_builder)
- }
- /// Launch the WebView and run the event loop, with configuration and root props.
- ///
- /// This function will start a multithreaded Tokio runtime as well the WebView event loop.
- ///
- /// You can configure the WebView window with a configuration closure
- ///
- /// ```rust
- /// use dioxus::prelude::*;
- ///
- /// fn main() {
- /// dioxus::desktop::launch_cfg(app, AppProps { name: "asd" }, |c| c);
- /// }
- ///
- /// struct AppProps {
- /// name: &'static str
- /// }
- ///
- /// fn app(cx: Scope<AppProps>) -> Element {
- /// cx.render(rsx!{
- /// h1 {"hello {cx.props.name}!"}
- /// })
- /// }
- /// ```
- pub fn launch_with_props<P: 'static + Send>(
- root: Component<P>,
- props: P,
- builder: impl for<'a, 'b> FnOnce(&'b mut DesktopConfig<'a>) -> &'b mut DesktopConfig<'a>,
- ) {
- let mut desktop_cfg = DesktopConfig::new();
- builder(&mut desktop_cfg);
- let event_loop = EventLoop::with_user_event();
- let mut desktop = DesktopController::new_on_tokio(root, props, event_loop.create_proxy());
- let quit_hotkey = Accelerator::new(SysMods::Cmd, KeyCode::KeyQ);
- let modifiers = ModifiersState::default();
- event_loop.run(move |window_event, event_loop, control_flow| {
- *control_flow = ControlFlow::Wait;
- match window_event {
- Event::NewEvents(StartCause::Init) => desktop.new_window(&desktop_cfg, event_loop),
- Event::WindowEvent {
- event, window_id, ..
- } => {
- match event {
- WindowEvent::CloseRequested => *control_flow = ControlFlow::Exit,
- WindowEvent::Destroyed { .. } => desktop.close_window(window_id, control_flow),
- WindowEvent::KeyboardInput { event, .. } => {
- if quit_hotkey.matches(&modifiers, &event.physical_key) {
- desktop.close_window(window_id, control_flow);
- }
- }
- WindowEvent::Resized(_) | WindowEvent::Moved(_) => {
- if let Some(view) = desktop.webviews.get_mut(&window_id) {
- let _ = view.resize();
- }
- }
- // TODO: we want to shuttle all of these events into the user's app or provide some handler
- _ => {}
- }
- }
- Event::UserEvent(_evt) => {
- desktop.try_load_ready_webviews();
- }
- Event::MainEventsCleared => {
- desktop.try_load_ready_webviews();
- }
- Event::Resumed => {}
- Event::Suspended => {}
- Event::LoopDestroyed => {}
- Event::RedrawRequested(_id) => {}
- _ => {}
- }
- })
- }
- pub enum UserWindowEvent {
- Start,
- Update,
- }
- pub struct DesktopController {
- pub webviews: HashMap<WindowId, WebView>,
- pub sender: futures_channel::mpsc::UnboundedSender<SchedulerMsg>,
- pub pending_edits: Arc<RwLock<VecDeque<String>>>,
- pub quit_app_on_close: bool,
- pub is_ready: Arc<AtomicBool>,
- }
- impl DesktopController {
- // Launch the virtualdom on its own thread managed by tokio
- // returns the desktop state
- pub fn new_on_tokio<P: Send + 'static>(
- root: Component<P>,
- props: P,
- evt: EventLoopProxy<UserWindowEvent>,
- ) -> Self {
- let edit_queue = Arc::new(RwLock::new(VecDeque::new()));
- let pending_edits = edit_queue.clone();
- let (sender, receiver) = futures_channel::mpsc::unbounded::<SchedulerMsg>();
- let return_sender = sender.clone();
- std::thread::spawn(move || {
- // We create the runtim as multithreaded, so you can still "spawn" onto multiple threads
- let runtime = tokio::runtime::Builder::new_multi_thread()
- .enable_all()
- .build()
- .unwrap();
- runtime.block_on(async move {
- let mut dom =
- VirtualDom::new_with_props_and_scheduler(root, props, (sender, receiver));
- let edits = dom.rebuild();
- edit_queue
- .write()
- .unwrap()
- .push_front(serde_json::to_string(&edits.edits).unwrap());
- loop {
- dom.wait_for_work().await;
- let mut muts = dom.work_with_deadline(|| false);
- while let Some(edit) = muts.pop() {
- edit_queue
- .write()
- .unwrap()
- .push_front(serde_json::to_string(&edit.edits).unwrap());
- }
- let _ = evt.send_event(UserWindowEvent::Update);
- }
- })
- });
- Self {
- pending_edits,
- sender: return_sender,
- webviews: HashMap::new(),
- is_ready: Arc::new(AtomicBool::new(false)),
- quit_app_on_close: true,
- }
- }
- pub fn new_window(
- &mut self,
- cfg: &DesktopConfig,
- event_loop: &EventLoopWindowTarget<UserWindowEvent>,
- ) {
- let builder = cfg.window.clone().with_menu({
- // create main menubar menu
- let mut menu_bar_menu = MenuBar::new();
- // create `first_menu`
- let mut first_menu = MenuBar::new();
- first_menu.add_native_item(MenuItem::About("App".to_string()));
- first_menu.add_native_item(MenuItem::Services);
- first_menu.add_native_item(MenuItem::Separator);
- first_menu.add_native_item(MenuItem::Hide);
- first_menu.add_native_item(MenuItem::HideOthers);
- first_menu.add_native_item(MenuItem::ShowAll);
- first_menu.add_native_item(MenuItem::Quit);
- first_menu.add_native_item(MenuItem::CloseWindow);
- // create second menu
- let mut second_menu = MenuBar::new();
- // second_menu.add_submenu("Sub menu", true, my_sub_menu);
- second_menu.add_native_item(MenuItem::Copy);
- second_menu.add_native_item(MenuItem::Paste);
- second_menu.add_native_item(MenuItem::SelectAll);
- menu_bar_menu.add_submenu("First menu", true, first_menu);
- menu_bar_menu.add_submenu("Second menu", true, second_menu);
- menu_bar_menu
- });
- let window = builder.build(event_loop).unwrap();
- let window_id = window.id();
- let (is_ready, sender) = (self.is_ready.clone(), self.sender.clone());
- let webview = WebViewBuilder::new(window)
- .unwrap()
- .with_url("wry://index.html")
- .unwrap()
- .with_rpc_handler(move |_window: &Window, req: RpcRequest| {
- match req.method.as_str() {
- "user_event" => {
- let event = events::trigger_from_serialized(req.params.unwrap());
- log::debug!("User event: {:?}", event);
- sender.unbounded_send(SchedulerMsg::Event(event)).unwrap();
- }
- "initialize" => {
- is_ready.store(true, std::sync::atomic::Ordering::Relaxed);
- }
- _ => {}
- }
- // response always driven through eval.
- // unfortunately, it seems to be pretty slow, so we might want to look into an RPC form
- None
- })
- // Any content that that uses the `wry://` scheme will be shuttled through this handler as a "special case"
- // For now, we only serve two pieces of content which get included as bytes into the final binary.
- .with_custom_protocol("wry".into(), move |request| {
- let path = request.uri().replace("wry://", "");
- let (data, meta) = match path.as_str() {
- "index.html" | "index.html/" | "/index.html" => {
- (include_bytes!("./index.html").to_vec(), "text/html")
- }
- "index.html/index.js" => {
- (include_bytes!("./index.js").to_vec(), "text/javascript")
- }
- _ => (include_bytes!("./index.html").to_vec(), "text/html"),
- };
- wry::http::ResponseBuilder::new().mimetype(meta).body(data)
- })
- .build()
- .unwrap();
- self.webviews.insert(window_id, webview);
- }
- pub fn close_window(&mut self, window_id: WindowId, control_flow: &mut ControlFlow) {
- self.webviews.remove(&window_id);
- if self.webviews.is_empty() && self.quit_app_on_close {
- *control_flow = ControlFlow::Exit;
- }
- }
- pub fn try_load_ready_webviews(&mut self) {
- if self.is_ready.load(std::sync::atomic::Ordering::Relaxed) {
- let mut queue = self.pending_edits.write().unwrap();
- let (_id, view) = self.webviews.iter_mut().next().unwrap();
- while let Some(edit) = queue.pop_back() {
- view.evaluate_script(&format!("window.interpreter.handleEdits({})", edit))
- .unwrap();
- }
- }
- }
- }
|