ui.rs 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  1. use std::collections::HashMap;
  2. use crate::{
  3. component::Component, layout::{LayoutCache, LayoutNode}, text::TextRenderer, window::{Window, WindowBuilder, WindowComponent, WindowEvent, WindowStateAccess}
  4. };
  5. #[derive(Debug)]
  6. pub enum UIControlMsg {
  7. Terminate,
  8. }
  9. pub trait UIComponent: Sized + Component<ParentMsg = UIControlMsg> {
  10. fn init(&mut self, ui_handle: UIHandle);
  11. fn poll(&mut self) -> Vec<UIControlMsg>;
  12. }
  13. pub(crate) struct UIState {
  14. pub(crate) layout_cache: std::rc::Rc<LayoutCache>,
  15. pub(crate) window_states:
  16. HashMap<winit::window::WindowId, std::rc::Weak<dyn WindowStateAccess>>,
  17. pub(crate) text_renderer: TextRenderer,
  18. }
  19. pub struct UIHandle<'l> {
  20. pub(crate) state: &'l mut UIState,
  21. pub(crate) eloop: &'l winit::event_loop::ActiveEventLoop,
  22. }
  23. impl<'l> UIHandle<'l> {
  24. pub fn window_builder<'r>(&'r mut self) -> WindowBuilder<'r, 'l> {
  25. WindowBuilder::new(self)
  26. }
  27. pub fn new_layout_node(&self) -> LayoutNode {
  28. LayoutNode::new(self.state.layout_cache.clone())
  29. }
  30. }
  31. pub struct UI<UIC: UIComponent> {
  32. state: UIState,
  33. event_loop: Option<winit::event_loop::EventLoop<()>>,
  34. ui_component: UIC,
  35. autoclose: bool,
  36. initialized: bool,
  37. }
  38. impl<UIC: UIComponent> UI<UIC> {
  39. pub fn new(uic: UIC) -> Self {
  40. let lcache = LayoutCache::new();
  41. let eloop = winit::event_loop::EventLoop::builder().build().unwrap();
  42. eloop.set_control_flow(winit::event_loop::ControlFlow::Wait);
  43. Self {
  44. state: UIState {
  45. layout_cache: lcache,
  46. window_states: Default::default(),
  47. text_renderer: TextRenderer::new(),
  48. },
  49. event_loop: Some(eloop),
  50. ui_component: uic,
  51. autoclose: true,
  52. initialized: false,
  53. }
  54. }
  55. pub fn disable_autoclose(mut self) -> Self {
  56. self.autoclose = false;
  57. self
  58. }
  59. pub fn ui_component(&self) -> &UIC {
  60. &self.ui_component
  61. }
  62. pub fn ui_component_mut(&mut self) -> &mut UIC {
  63. &mut self.ui_component
  64. }
  65. /// Public helper function, delivers a message to the UI component and then processes whatever
  66. /// control messages result.
  67. pub fn deliver_msg(&mut self, msg: UIC::Msg) {
  68. let cmsgs = self.ui_component.process(msg);
  69. self.process_control_msgs(cmsgs.into_iter());
  70. }
  71. fn process_control_msgs(&mut self, msgs: impl Iterator<Item = UIControlMsg>) {
  72. for cmsg in msgs {
  73. match cmsg {
  74. UIControlMsg::Terminate => {
  75. todo!()
  76. // self.ui_running = false;
  77. }
  78. }
  79. }
  80. }
  81. pub fn run(mut self) {
  82. let eloop = self.event_loop.take().unwrap();
  83. eloop.run_app(&mut self).unwrap();
  84. }
  85. fn pump_events(&mut self, eloop: &winit::event_loop::ActiveEventLoop) {
  86. let evts = self.ui_component.poll();
  87. for evt in evts.into_iter() {
  88. match evt {
  89. UIControlMsg::Terminate => eloop.exit(),
  90. }
  91. }
  92. }
  93. }
  94. impl<UIC: UIComponent> winit::application::ApplicationHandler<()> for UI<UIC> {
  95. fn resumed(&mut self, event_loop: &winit::event_loop::ActiveEventLoop) {
  96. if !self.initialized {
  97. self.ui_component.init(UIHandle {
  98. eloop: event_loop,
  99. state: &mut self.state,
  100. });
  101. self.initialized = true;
  102. }
  103. }
  104. fn window_event(
  105. &mut self,
  106. event_loop: &winit::event_loop::ActiveEventLoop,
  107. window_id: winit::window::WindowId,
  108. event: winit::event::WindowEvent,
  109. ) {
  110. let Some(wsa) = self
  111. .state
  112. .window_states
  113. .get(&window_id)
  114. .map(std::rc::Weak::upgrade)
  115. .flatten()
  116. else {
  117. println!("superfluous event: {event:?}");
  118. return;
  119. };
  120. match event {
  121. winit::event::WindowEvent::CloseRequested => {
  122. wsa.push_event(WindowEvent::CloseRequested);
  123. self.pump_events(event_loop);
  124. }
  125. winit::event::WindowEvent::Destroyed => {
  126. self.state.window_states.remove(&window_id);
  127. }
  128. winit::event::WindowEvent::RedrawRequested => {
  129. wsa.request_redraw();
  130. self.pump_events(event_loop);
  131. }
  132. _ => (),
  133. }
  134. }
  135. }
  136. pub struct OpinionatedUIComponent<'l, WC: WindowComponent> {
  137. window: Option<Window<WC>>,
  138. builder: Option<Box<dyn FnOnce(&UIHandle) -> WC + 'l>>,
  139. }
  140. impl<'l, WC: WindowComponent> OpinionatedUIComponent<'l, WC> {
  141. }
  142. impl<'l, WC: WindowComponent<ParentMsg = UIControlMsg>> Component for OpinionatedUIComponent<'l, WC> {
  143. type ParentMsg = UIControlMsg;
  144. type Msg = UIControlMsg;
  145. fn process(&mut self, msg: Self::Msg) -> Vec<Self::ParentMsg> {
  146. vec![msg]
  147. }
  148. }
  149. impl<'l, WC: WindowComponent<ParentMsg = UIControlMsg>> UIComponent for OpinionatedUIComponent<'l, WC> {
  150. fn init(&mut self, mut ui_handle: UIHandle) {
  151. if let Some(builder) = self.builder.take() {
  152. self.window = Some(ui_handle.window_builder().build(builder));
  153. }
  154. }
  155. fn poll(&mut self) -> Vec<UIControlMsg> {
  156. if let Some(window) = self.window.as_mut() {
  157. window.poll()
  158. } else {
  159. vec![]
  160. }
  161. }
  162. }
  163. pub fn make_opinionated_ui<'l, WC: WindowComponent<ParentMsg = UIControlMsg>>(wc: impl 'l + FnOnce(&UIHandle) -> WC) -> UI<OpinionatedUIComponent<'l, WC>> {
  164. let ouic = OpinionatedUIComponent {
  165. window: None,
  166. builder: Some(Box::new(wc)),
  167. };
  168. let ui = UI::new(ouic);
  169. ui
  170. }