diff options
| author | 2020-11-10 22:57:19 +0100 | |
|---|---|---|
| committer | 2020-11-10 22:57:19 +0100 | |
| commit | 2f5a3dacd933a52931a1bb169138d52402413956 (patch) | |
| tree | 65995b4304543ea6a6400af3c8e11eae630dd22b /native/src | |
| parent | d0402d072d3f4e128c55fc0a6184c5f7c712bb20 (diff) | |
| parent | 86fa12229e7117306b8297a09aa22c99802600c8 (diff) | |
| download | iced-2f5a3dacd933a52931a1bb169138d52402413956.tar.gz iced-2f5a3dacd933a52931a1bb169138d52402413956.tar.bz2 iced-2f5a3dacd933a52931a1bb169138d52402413956.zip | |
Merge pull request #608 from hecrj/remove-pane-grid-focus
Improve flexibility of `PaneGrid`
Diffstat (limited to '')
| -rw-r--r-- | native/src/subscription.rs | 22 | ||||
| -rw-r--r-- | native/src/subscription/events.rs | 19 | ||||
| -rw-r--r-- | native/src/widget/pane_grid.rs | 154 | ||||
| -rw-r--r-- | native/src/widget/pane_grid/content.rs | 4 | ||||
| -rw-r--r-- | native/src/widget/pane_grid/state.rs | 129 | 
5 files changed, 77 insertions, 251 deletions
| diff --git a/native/src/subscription.rs b/native/src/subscription.rs index 0d002c6c..18750abf 100644 --- a/native/src/subscription.rs +++ b/native/src/subscription.rs @@ -43,5 +43,25 @@ use events::Events;  /// [`Subscription`]: type.Subscription.html  /// [`Event`]: ../enum.Event.html  pub fn events() -> Subscription<Event> { -    Subscription::from_recipe(Events) +    Subscription::from_recipe(Events { f: Some }) +} + +/// Returns a [`Subscription`] that filters all the runtime events with the +/// provided function, producing messages accordingly. +/// +/// This subscription will call the provided function for every [`Event`] +/// handled by the runtime. If the function: +/// +/// - Returns `None`, the [`Event`] will be discarded. +/// - Returns `Some` message, the `Message` will be produced. +/// +/// [`Subscription`]: type.Subscription.html +/// [`Event`]: ../enum.Event.html +pub fn events_with<Message>( +    f: fn(Event) -> Option<Message>, +) -> Subscription<Message> +where +    Message: 'static + Send, +{ +    Subscription::from_recipe(Events { f })  } diff --git a/native/src/subscription/events.rs b/native/src/subscription/events.rs index ceae467d..a1ae6051 100644 --- a/native/src/subscription/events.rs +++ b/native/src/subscription/events.rs @@ -2,17 +2,26 @@ use crate::{      subscription::{EventStream, Recipe},      Event, Hasher,  }; +use iced_futures::futures::future; +use iced_futures::futures::StreamExt;  use iced_futures::BoxStream; -pub struct Events; +pub struct Events<Message> { +    pub(super) f: fn(Event) -> Option<Message>, +} -impl Recipe<Hasher, Event> for Events { -    type Output = Event; +impl<Message> Recipe<Hasher, Event> for Events<Message> +where +    Message: 'static + Send, +{ +    type Output = Message;      fn hash(&self, state: &mut Hasher) {          use std::hash::Hash; -        std::any::TypeId::of::<Self>().hash(state); +        struct Marker; +        std::any::TypeId::of::<Marker>().hash(state); +        self.f.hash(state);      }      fn stream( @@ -20,5 +29,7 @@ impl Recipe<Hasher, Event> for Events {          event_stream: EventStream,      ) -> BoxStream<Self::Output> {          event_stream +            .filter_map(move |event| future::ready((self.f)(event))) +            .boxed()      }  } diff --git a/native/src/widget/pane_grid.rs b/native/src/widget/pane_grid.rs index 276bfae3..9d36bae6 100644 --- a/native/src/widget/pane_grid.rs +++ b/native/src/widget/pane_grid.rs @@ -29,8 +29,8 @@ pub use state::{Focus, State};  pub use title_bar::TitleBar;  use crate::{ -    container, keyboard, layout, mouse, overlay, row, text, Clipboard, Element, -    Event, Hasher, Layout, Length, Point, Rectangle, Size, Vector, Widget, +    container, layout, mouse, overlay, row, text, Clipboard, Element, Event, +    Hasher, Layout, Length, Point, Rectangle, Size, Vector, Widget,  };  /// A collection of panes distributed using either vertical or horizontal splits @@ -73,7 +73,7 @@ use crate::{  /// let (mut state, _) = pane_grid::State::new(PaneState::SomePane);  ///  /// let pane_grid = -///     PaneGrid::new(&mut state, |pane, state, focus| { +///     PaneGrid::new(&mut state, |pane, state| {  ///         pane_grid::Content::new(match state {  ///             PaneState::SomePane => Text::new("This is some pane"),  ///             PaneState::AnotherKindOfPane => Text::new("This is another kind of pane"), @@ -92,10 +92,9 @@ pub struct PaneGrid<'a, Message, Renderer: self::Renderer> {      width: Length,      height: Length,      spacing: u16, -    modifier_keys: keyboard::ModifiersState, +    on_click: Option<Box<dyn Fn(Pane) -> Message + 'a>>,      on_drag: Option<Box<dyn Fn(DragEvent) -> Message + 'a>>,      on_resize: Option<(u16, Box<dyn Fn(ResizeEvent) -> Message + 'a>)>, -    on_key_press: Option<Box<dyn Fn(KeyPressEvent) -> Option<Message> + 'a>>,  }  impl<'a, Message, Renderer> PaneGrid<'a, Message, Renderer> @@ -112,31 +111,13 @@ where      /// [`Pane`]: struct.Pane.html      pub fn new<T>(          state: &'a mut State<T>, -        view: impl Fn( -            Pane, -            &'a mut T, -            Option<Focus>, -        ) -> Content<'a, Message, Renderer>, +        view: impl Fn(Pane, &'a mut T) -> Content<'a, Message, Renderer>,      ) -> Self {          let elements = { -            let action = state.internal.action(); -            let current_focus = action.focus(); -              state                  .panes                  .iter_mut() -                .map(move |(pane, pane_state)| { -                    let focus = match current_focus { -                        Some((focused_pane, focus)) -                            if *pane == focused_pane => -                        { -                            Some(focus) -                        } -                        _ => None, -                    }; - -                    (*pane, view(*pane, pane_state, focus)) -                }) +                .map(|(pane, pane_state)| (*pane, view(*pane, pane_state)))                  .collect()          }; @@ -146,13 +127,9 @@ where              width: Length::Fill,              height: Length::Fill,              spacing: 0, -            modifier_keys: keyboard::ModifiersState { -                control: true, -                ..Default::default() -            }, +            on_click: None,              on_drag: None,              on_resize: None, -            on_key_press: None,          }      } @@ -180,18 +157,16 @@ where          self      } -    /// Sets the modifier keys of the [`PaneGrid`]. -    /// -    /// The modifier keys will need to be pressed to trigger key events. -    /// -    /// The default modifier key is `Ctrl`. +    /// Sets the message that will be produced when a [`Pane`] of the +    /// [`PaneGrid`] is clicked.      /// +    /// [`Pane`]: struct.Pane.html      /// [`PaneGrid`]: struct.PaneGrid.html -    pub fn modifier_keys( -        mut self, -        modifier_keys: keyboard::ModifiersState, -    ) -> Self { -        self.modifier_keys = modifier_keys; +    pub fn on_click<F>(mut self, f: F) -> Self +    where +        F: 'a + Fn(Pane) -> Message, +    { +        self.on_click = Some(Box::new(f));          self      } @@ -225,31 +200,6 @@ where          self.on_resize = Some((leeway, Box::new(f)));          self      } - -    /// Captures hotkey interactions with the [`PaneGrid`], using the provided -    /// function to produce messages. -    /// -    /// The function will be called when: -    ///   - a [`Pane`] is focused -    ///   - a key is pressed -    ///   - all the modifier keys are pressed -    /// -    /// If the function returns `None`, the key press event will be discarded -    /// without producing any message. -    /// -    /// This method is particularly useful to implement hotkey interactions. -    /// For instance, you can use it to enable splitting, swapping, or resizing -    /// panes by pressing combinations of keys. -    /// -    /// [`PaneGrid`]: struct.PaneGrid.html -    /// [`Pane`]: struct.Pane.html -    pub fn on_key_press<F>(mut self, f: F) -> Self -    where -        F: 'a + Fn(KeyPressEvent) -> Option<Message>, -    { -        self.on_key_press = Some(Box::new(f)); -        self -    }  }  impl<'a, Message, Renderer> PaneGrid<'a, Message, Renderer> @@ -268,24 +218,20 @@ where              );          if let Some(((pane, content), layout)) = clicked_region.next() { -            match &self.on_drag { -                Some(on_drag) => { -                    if content.can_be_picked_at(layout, cursor_position) { -                        let pane_position = layout.position(); +            if let Some(on_click) = &self.on_click { +                messages.push(on_click(*pane)); +            } -                        let origin = cursor_position -                            - Vector::new(pane_position.x, pane_position.y); +            if let Some(on_drag) = &self.on_drag { +                if content.can_be_picked_at(layout, cursor_position) { +                    let pane_position = layout.position(); -                        self.state.pick_pane(pane, origin); +                    let origin = cursor_position +                        - Vector::new(pane_position.x, pane_position.y); -                        messages -                            .push(on_drag(DragEvent::Picked { pane: *pane })); -                    } else { -                        self.state.focus(pane); -                    } -                } -                None => { -                    self.state.focus(pane); +                    self.state.pick_pane(pane, origin); + +                    messages.push(on_drag(DragEvent::Picked { pane: *pane }));                  }              }          } @@ -390,18 +336,6 @@ pub struct ResizeEvent {      pub ratio: f32,  } -/// An event produced during a key press interaction of a [`PaneGrid`]. -/// -/// [`PaneGrid`]: struct.PaneGrid.html -#[derive(Debug, Clone, Copy)] -pub struct KeyPressEvent { -    /// The key that was pressed. -    pub key_code: keyboard::KeyCode, - -    /// The state of the modifier keys when the key was pressed. -    pub modifiers: keyboard::ModifiersState, -} -  impl<'a, Message, Renderer> Widget<Message, Renderer>      for PaneGrid<'a, Message, Renderer>  where @@ -495,17 +429,10 @@ where                                  );                              }                          } -                    } else { -                        // TODO: Encode cursor availability in the type system -                        if cursor_position.x > 0.0 && cursor_position.y > 0.0 { -                            self.state.unfocus(); -                        }                      }                  }                  mouse::Event::ButtonReleased(mouse::Button::Left) => {                      if let Some((pane, _)) = self.state.picked_pane() { -                        self.state.focus(&pane); -                          if let Some(on_drag) = &self.on_drag {                              let mut dropped_region = self                                  .elements @@ -527,8 +454,10 @@ where                              messages.push(on_drag(event));                          } + +                        self.state.idle();                      } else if self.state.picked_split().is_some() { -                        self.state.drop_split(); +                        self.state.idle();                      }                  }                  mouse::Event::CursorMoved { .. } => { @@ -536,31 +465,6 @@ where                  }                  _ => {}              }, -            Event::Keyboard(keyboard_event) => { -                match keyboard_event { -                    keyboard::Event::KeyPressed { -                        modifiers, -                        key_code, -                    } => { -                        if let Some(on_key_press) = &self.on_key_press { -                            // TODO: Discard when event is captured -                            if let Some(_) = self.state.active_pane() { -                                if modifiers.matches(self.modifier_keys) { -                                    if let Some(message) = -                                        on_key_press(KeyPressEvent { -                                            key_code, -                                            modifiers, -                                        }) -                                    { -                                        messages.push(message); -                                    } -                                } -                            } -                        } -                    } -                    _ => {} -                } -            }              _ => {}          } diff --git a/native/src/widget/pane_grid/content.rs b/native/src/widget/pane_grid/content.rs index 1d339b75..5bfbb9ed 100644 --- a/native/src/widget/pane_grid/content.rs +++ b/native/src/widget/pane_grid/content.rs @@ -41,9 +41,9 @@ where          self      } -    /// Sets the style of the [`TitleBar`]. +    /// Sets the style of the [`Content`].      /// -    /// [`TitleBar`]: struct.TitleBar.html +    /// [`Content`]: struct.Content.html      pub fn style(mut self, style: impl Into<Renderer::Style>) -> Self {          self.style = style.into();          self diff --git a/native/src/widget/pane_grid/state.rs b/native/src/widget/pane_grid/state.rs index fb59c846..be36b070 100644 --- a/native/src/widget/pane_grid/state.rs +++ b/native/src/widget/pane_grid/state.rs @@ -72,7 +72,7 @@ impl<T> State<T> {              internal: Internal {                  layout,                  last_id, -                action: Action::Idle { focus: None }, +                action: Action::Idle,              },          }      } @@ -122,45 +122,10 @@ impl<T> State<T> {          &self.internal.layout      } -    /// Returns the focused [`Pane`] of the [`State`], if there is one. -    /// -    /// [`Pane`]: struct.Pane.html -    /// [`State`]: struct.State.html -    pub fn focused(&self) -> Option<Pane> { -        self.internal.focused_pane() -    } - -    /// Returns the active [`Pane`] of the [`State`], if there is one. -    /// -    /// A [`Pane`] is active if it is focused and is __not__ being dragged. -    /// -    /// [`Pane`]: struct.Pane.html -    /// [`State`]: struct.State.html -    pub fn active(&self) -> Option<Pane> { -        self.internal.active_pane() -    } -      /// Returns the adjacent [`Pane`] of another [`Pane`] in the given      /// direction, if there is one.      /// -    /// ## Example -    /// You can combine this with [`State::active`] to find the pane that is -    /// adjacent to the current active one, and then swap them. For instance: -    /// -    /// ``` -    /// # use iced_native::pane_grid; -    /// # -    /// # let (mut state, _) = pane_grid::State::new(()); -    /// # -    /// if let Some(active) = state.active() { -    ///     if let Some(adjacent) = state.adjacent(&active, pane_grid::Direction::Right) { -    ///         state.swap(&active, &adjacent); -    ///     } -    /// } -    /// ``` -    ///      /// [`Pane`]: struct.Pane.html -    /// [`State::active`]: struct.State.html#method.active      pub fn adjacent(&self, pane: &Pane, direction: Direction) -> Option<Pane> {          let regions = self              .internal @@ -194,20 +159,6 @@ impl<T> State<T> {          Some(*pane)      } -    /// Focuses the given [`Pane`]. -    /// -    /// [`Pane`]: struct.Pane.html -    pub fn focus(&mut self, pane: &Pane) { -        self.internal.focus(pane); -    } - -    /// Unfocuses the current focused [`Pane`]. -    /// -    /// [`Pane`]: struct.Pane.html -    pub fn unfocus(&mut self) { -        self.internal.unfocus(); -    } -      /// Splits the given [`Pane`] into two in the given [`Axis`] and      /// initializing the new [`Pane`] with the provided internal state.      /// @@ -236,7 +187,6 @@ impl<T> State<T> {          node.split(new_split, axis, new_pane);          let _ = self.panes.insert(new_pane, state); -        self.focus(&new_pane);          Some((new_pane, new_split))      } @@ -277,13 +227,13 @@ impl<T> State<T> {          let _ = self.internal.layout.resize(split, ratio);      } -    /// Closes the given [`Pane`] and returns its internal state, if it exists. +    /// Closes the given [`Pane`] and returns its internal state and its closest +    /// sibling, if it exists.      ///      /// [`Pane`]: struct.Pane.html -    pub fn close(&mut self, pane: &Pane) -> Option<T> { +    pub fn close(&mut self, pane: &Pane) -> Option<(T, Pane)> {          if let Some(sibling) = self.internal.layout.remove(pane) { -            self.focus(&sibling); -            self.panes.remove(pane) +            self.panes.remove(pane).map(|state| (state, sibling))          } else {              None          } @@ -329,52 +279,12 @@ pub struct Internal {  #[derive(Debug, Clone, Copy, PartialEq)]  pub enum Action { -    Idle { -        focus: Option<Pane>, -    }, -    Dragging { -        pane: Pane, -        origin: Point, -        focus: Option<Pane>, -    }, -    Resizing { -        split: Split, -        axis: Axis, -        focus: Option<Pane>, -    }, -} - -impl Action { -    pub fn focus(&self) -> Option<(Pane, Focus)> { -        match self { -            Action::Idle { focus } | Action::Resizing { focus, .. } => { -                focus.map(|pane| (pane, Focus::Idle)) -            } -            Action::Dragging { pane, .. } => Some((*pane, Focus::Dragging)), -        } -    } +    Idle, +    Dragging { pane: Pane, origin: Point }, +    Resizing { split: Split, axis: Axis },  }  impl Internal { -    pub fn action(&self) -> Action { -        self.action -    } - -    pub fn focused_pane(&self) -> Option<Pane> { -        match self.action { -            Action::Idle { focus } => focus, -            Action::Dragging { focus, .. } => focus, -            Action::Resizing { focus, .. } => focus, -        } -    } - -    pub fn active_pane(&self) -> Option<Pane> { -        match self.action { -            Action::Idle { focus } => focus, -            _ => None, -        } -    } -      pub fn picked_pane(&self) -> Option<(Pane, Point)> {          match self.action {              Action::Dragging { pane, origin, .. } => Some((pane, origin)), @@ -405,17 +315,10 @@ impl Internal {          self.layout.split_regions(spacing, size)      } -    pub fn focus(&mut self, pane: &Pane) { -        self.action = Action::Idle { focus: Some(*pane) }; -    } -      pub fn pick_pane(&mut self, pane: &Pane, origin: Point) { -        let focus = self.focused_pane(); -          self.action = Action::Dragging {              pane: *pane,              origin, -            focus,          };      } @@ -426,26 +329,14 @@ impl Internal {              return;          } -        let focus = self.action.focus().map(|(pane, _)| pane); -          self.action = Action::Resizing {              split: *split,              axis, -            focus,          };      } -    pub fn drop_split(&mut self) { -        match self.action { -            Action::Resizing { focus, .. } => { -                self.action = Action::Idle { focus }; -            } -            _ => {} -        } -    } - -    pub fn unfocus(&mut self) { -        self.action = Action::Idle { focus: None }; +    pub fn idle(&mut self) { +        self.action = Action::Idle;      }      pub fn hash_layout(&self, hasher: &mut Hasher) { | 
