controller.rs 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106
  1. use crate::desktop_context::{DesktopContext, UserWindowEvent};
  2. use dioxus_core::*;
  3. use std::{
  4. collections::{HashMap, VecDeque},
  5. sync::atomic::AtomicBool,
  6. sync::{Arc, RwLock},
  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<RwLock<VecDeque<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(RwLock::new(VecDeque::new()));
  29. let pending_edits = edit_queue.clone();
  30. let (sender, receiver) = futures_channel::mpsc::unbounded::<SchedulerMsg>();
  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. let edits = dom.rebuild();
  45. edit_queue
  46. .write()
  47. .unwrap()
  48. .push_front(serde_json::to_string(&edits.edits).unwrap());
  49. // Make sure the window is ready for any new updates
  50. proxy.send_event(UserWindowEvent::Update).unwrap();
  51. loop {
  52. dom.wait_for_work().await;
  53. let mut muts = dom.work_with_deadline(|| false);
  54. while let Some(edit) = muts.pop() {
  55. edit_queue
  56. .write()
  57. .unwrap()
  58. .push_front(serde_json::to_string(&edit.edits).unwrap());
  59. }
  60. let _ = proxy.send_event(UserWindowEvent::Update);
  61. }
  62. })
  63. });
  64. Self {
  65. pending_edits,
  66. sender: return_sender,
  67. webviews: HashMap::new(),
  68. is_ready: Arc::new(AtomicBool::new(false)),
  69. quit_app_on_close: true,
  70. }
  71. }
  72. pub(super) fn close_window(&mut self, window_id: WindowId, control_flow: &mut ControlFlow) {
  73. self.webviews.remove(&window_id);
  74. if self.webviews.is_empty() && self.quit_app_on_close {
  75. *control_flow = ControlFlow::Exit;
  76. }
  77. }
  78. pub(super) fn try_load_ready_webviews(&mut self) {
  79. if self.is_ready.load(std::sync::atomic::Ordering::Relaxed) {
  80. let mut queue = self.pending_edits.write().unwrap();
  81. let (_id, view) = self.webviews.iter_mut().next().unwrap();
  82. while let Some(edit) = queue.pop_back() {
  83. view.evaluate_script(&format!("window.interpreter.handleEdits({})", edit))
  84. .unwrap();
  85. }
  86. }
  87. }
  88. }