controller.rs 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. use crate::desktop_context::{DesktopContext, UserWindowEvent};
  2. use dioxus_core::*;
  3. use std::{
  4. collections::HashMap,
  5. sync::Arc,
  6. sync::{atomic::AtomicBool, Mutex},
  7. };
  8. use wry::{
  9. self,
  10. application::{event_loop::ControlFlow, event_loop::EventLoopProxy, window::WindowId},
  11. webview::WebView,
  12. };
  13. pub(super) struct DesktopController {
  14. pub(super) webviews: HashMap<WindowId, WebView>,
  15. pub(super) sender: futures_channel::mpsc::UnboundedSender<SchedulerMsg>,
  16. pub(super) pending_edits: Arc<Mutex<Vec<String>>>,
  17. pub(super) quit_app_on_close: bool,
  18. pub(super) is_ready: Arc<AtomicBool>,
  19. }
  20. impl DesktopController {
  21. // Launch the virtualdom on its own thread managed by tokio
  22. // returns the desktop state
  23. pub(super) fn new_on_tokio<P: Send + 'static>(
  24. root: Component<P>,
  25. props: P,
  26. proxy: EventLoopProxy<UserWindowEvent>,
  27. ) -> Self {
  28. let edit_queue = Arc::new(Mutex::new(Vec::new()));
  29. let (sender, receiver) = futures_channel::mpsc::unbounded::<SchedulerMsg>();
  30. let pending_edits = edit_queue.clone();
  31. let return_sender = sender.clone();
  32. let desktop_context_proxy = proxy.clone();
  33. std::thread::spawn(move || {
  34. // We create the runtime as multithreaded, so you can still "spawn" onto multiple threads
  35. let runtime = tokio::runtime::Builder::new_multi_thread()
  36. .enable_all()
  37. .build()
  38. .unwrap();
  39. runtime.block_on(async move {
  40. let mut dom =
  41. VirtualDom::new_with_props_and_scheduler(root, props, (sender, receiver));
  42. let window_context = DesktopContext::new(desktop_context_proxy);
  43. dom.base_scope().provide_context(window_context);
  44. todo!()
  45. // // allow other proccesses to send the new rsx text to the @dioxusin ipc channel and recieve erros on the @dioxusout channel
  46. // #[cfg(any(feature = "hot-reload", debug_assertions))]
  47. // crate::hot_reload::init(&dom);
  48. // let edits = dom.rebuild();
  49. // edit_queue
  50. // .lock()
  51. // .unwrap()
  52. // .push(serde_json::to_string(&edits.edits).unwrap());
  53. // // Make sure the window is ready for any new updates
  54. // proxy.send_event(UserWindowEvent::Update).unwrap();
  55. // loop {
  56. // dom.wait_for_work().await;
  57. // let muts = dom.work_with_deadline(|| false);
  58. // for edit in muts {
  59. // edit_queue
  60. // .lock()
  61. // .unwrap()
  62. // .push(serde_json::to_string(&edit.edits).unwrap());
  63. // }
  64. // let _ = proxy.send_event(UserWindowEvent::Update);
  65. // }
  66. })
  67. });
  68. Self {
  69. pending_edits,
  70. sender: return_sender,
  71. webviews: HashMap::new(),
  72. is_ready: Arc::new(AtomicBool::new(false)),
  73. quit_app_on_close: true,
  74. }
  75. }
  76. pub(super) fn close_window(&mut self, window_id: WindowId, control_flow: &mut ControlFlow) {
  77. self.webviews.remove(&window_id);
  78. if self.webviews.is_empty() && self.quit_app_on_close {
  79. *control_flow = ControlFlow::Exit;
  80. }
  81. }
  82. pub(super) fn try_load_ready_webviews(&mut self) {
  83. if self.is_ready.load(std::sync::atomic::Ordering::Relaxed) {
  84. let mut queue = self.pending_edits.lock().unwrap();
  85. let (_id, view) = self.webviews.iter_mut().next().unwrap();
  86. for edit in queue.drain(..) {
  87. view.evaluate_script(&format!("window.interpreter.handleEdits({})", edit))
  88. .unwrap();
  89. }
  90. }
  91. }
  92. }