123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232 |
- use crate::desktop_context::{EditQueue, EventData};
- use crate::protocol::{self, AssetHandlerRegistry};
- use crate::{desktop_context::UserWindowEvent, Config};
- use tao::event_loop::{EventLoopProxy, EventLoopWindowTarget};
- pub use wry;
- pub use wry::application as tao;
- use wry::application::window::Window;
- use wry::webview::{WebContext, WebView, WebViewBuilder};
- pub(crate) fn build(
- cfg: &mut Config,
- event_loop: &EventLoopWindowTarget<UserWindowEvent>,
- proxy: EventLoopProxy<UserWindowEvent>,
- ) -> (WebView, WebContext, AssetHandlerRegistry, EditQueue) {
- let builder = cfg.window.clone();
- let window = builder.with_visible(false).build(event_loop).unwrap();
- let file_handler = cfg.file_drop_handler.take();
- let custom_head = cfg.custom_head.clone();
- let index_file = cfg.custom_index.clone();
- let root_name = cfg.root_name.clone();
- let assets_head = {
- #[cfg(all(
- debug_assertions,
- any(
- target_os = "windows",
- target_os = "macos",
- target_os = "linux",
- target_os = "dragonfly",
- target_os = "freebsd",
- target_os = "netbsd",
- target_os = "openbsd"
- )
- ))]
- {
- None
- }
- #[cfg(not(all(
- debug_assertions,
- any(
- target_os = "windows",
- target_os = "macos",
- target_os = "linux",
- target_os = "dragonfly",
- target_os = "freebsd",
- target_os = "netbsd",
- target_os = "openbsd"
- )
- )))]
- {
- let head = crate::protocol::get_asset_root_or_default();
- let head = head.join("dist/__assets_head.html");
- match std::fs::read_to_string(&head) {
- Ok(s) => Some(s),
- Err(err) => {
- tracing::error!("Failed to read {head:?}: {err}");
- None
- }
- }
- }
- };
- // TODO: restore the menu bar with muda: https://github.com/tauri-apps/muda/blob/dev/examples/wry.rs
- // if cfg.enable_default_menu_bar {
- // builder = builder.with_menu(build_default_menu_bar());
- // }
- // We assume that if the icon is None in cfg, then the user just didnt set it
- if cfg.window.window.window_icon.is_none() {
- window.set_window_icon(Some(
- tao::window::Icon::from_rgba(
- include_bytes!("./assets/default_icon.bin").to_vec(),
- 460,
- 460,
- )
- .expect("image parse failed"),
- ));
- }
- let mut web_context = WebContext::new(cfg.data_dir.clone());
- let edit_queue = EditQueue::default();
- let headless = !cfg.window.window.visible;
- let asset_handlers = AssetHandlerRegistry::new();
- let asset_handlers_ref = asset_handlers.clone();
- let mut webview = WebViewBuilder::new(window)
- .unwrap()
- .with_transparent(cfg.window.window.transparent)
- .with_url("dioxus://index.html/")
- .unwrap()
- .with_ipc_handler(move |window: &Window, payload: String| {
- // defer the event to the main thread
- if let Ok(message) = serde_json::from_str(&payload) {
- _ = proxy.send_event(UserWindowEvent(EventData::Ipc(message), window.id()));
- }
- })
- .with_asynchronous_custom_protocol(String::from("dioxus"), {
- let edit_queue = edit_queue.clone();
- move |request, responder| {
- let custom_head = custom_head.clone();
- let index_file = index_file.clone();
- let assets_head = assets_head.clone();
- let root_name = root_name.clone();
- let asset_handlers_ref = asset_handlers_ref.clone();
- let edit_queue = edit_queue.clone();
- tokio::spawn(async move {
- protocol::desktop_handler(
- request,
- custom_head.clone(),
- index_file.clone(),
- assets_head.clone(),
- &root_name,
- &asset_handlers_ref,
- &edit_queue,
- headless,
- responder,
- )
- .await;
- });
- }
- })
- .with_file_drop_handler(move |window, evet| {
- file_handler
- .as_ref()
- .map(|handler| handler(window, evet))
- .unwrap_or_default()
- })
- .with_web_context(&mut web_context);
- #[cfg(windows)]
- {
- // Windows has a platform specific settings to disable the browser shortcut keys
- use wry::webview::WebViewBuilderExtWindows;
- webview = webview.with_browser_accelerator_keys(false);
- }
- if let Some(color) = cfg.background_color {
- webview = webview.with_background_color(color);
- }
- // These are commented out because wry is currently broken in wry
- // let mut web_context = WebContext::new(cfg.data_dir.clone());
- // .with_web_context(&mut web_context);
- for (name, handler) in cfg.protocols.drain(..) {
- webview = webview.with_custom_protocol(name, move |r| handler(r))
- }
- if cfg.disable_context_menu {
- // in release mode, we don't want to show the dev tool or reload menus
- webview = webview.with_initialization_script(
- r#"
- if (document.addEventListener) {
- document.addEventListener('contextmenu', function(e) {
- e.preventDefault();
- }, false);
- } else {
- document.attachEvent('oncontextmenu', function() {
- window.event.returnValue = false;
- });
- }
- "#,
- )
- } else {
- // in debug, we are okay with the reload menu showing and dev tool
- webview = webview.with_devtools(true);
- }
- (
- webview.build().unwrap(),
- web_context,
- asset_handlers,
- edit_queue,
- )
- }
- // /// Builds a standard menu bar depending on the users platform. It may be used as a starting point
- // /// to further customize the menu bar and pass it to a [`WindowBuilder`](tao::window::WindowBuilder).
- // /// > Note: The default menu bar enables macOS shortcuts like cut/copy/paste.
- // /// > The menu bar differs per platform because of constraints introduced
- // /// > by [`MenuItem`](tao::menu::MenuItem).
- // pub fn build_default_menu_bar() -> MenuBar {
- // let mut menu_bar = MenuBar::new();
- // // since it is uncommon on windows to have an "application menu"
- // // we add a "window" menu to be more consistent across platforms with the standard menu
- // let mut window_menu = MenuBar::new();
- // #[cfg(target_os = "macos")]
- // {
- // window_menu.add_native_item(MenuItem::EnterFullScreen);
- // window_menu.add_native_item(MenuItem::Zoom);
- // window_menu.add_native_item(MenuItem::Separator);
- // }
- // window_menu.add_native_item(MenuItem::Hide);
- // #[cfg(target_os = "macos")]
- // {
- // window_menu.add_native_item(MenuItem::HideOthers);
- // window_menu.add_native_item(MenuItem::ShowAll);
- // }
- // window_menu.add_native_item(MenuItem::Minimize);
- // window_menu.add_native_item(MenuItem::CloseWindow);
- // window_menu.add_native_item(MenuItem::Separator);
- // window_menu.add_native_item(MenuItem::Quit);
- // menu_bar.add_submenu("Window", true, window_menu);
- // // since tao supports none of the below items on linux we should only add them on macos/windows
- // #[cfg(not(target_os = "linux"))]
- // {
- // let mut edit_menu = MenuBar::new();
- // #[cfg(target_os = "macos")]
- // {
- // edit_menu.add_native_item(MenuItem::Undo);
- // edit_menu.add_native_item(MenuItem::Redo);
- // edit_menu.add_native_item(MenuItem::Separator);
- // }
- // edit_menu.add_native_item(MenuItem::Cut);
- // edit_menu.add_native_item(MenuItem::Copy);
- // edit_menu.add_native_item(MenuItem::Paste);
- // #[cfg(target_os = "macos")]
- // {
- // edit_menu.add_native_item(MenuItem::Separator);
- // edit_menu.add_native_item(MenuItem::SelectAll);
- // }
- // menu_bar.add_submenu("Edit", true, edit_menu);
- // }
- // menu_bar
- // }
|