summaryrefslogtreecommitdiffstats
path: root/lazy
diff options
context:
space:
mode:
authorLibravatar Nick Senger <dev@nsenger.com>2021-12-05 23:27:08 -0800
committerLibravatar Nick Senger <dev@nsenger.com>2021-12-10 09:53:44 -0800
commita92307890fcc3a567ec3d1b1cffebb59ae98991e (patch)
tree2de3c564746861d2a8944fcb878c2ef22cada63d /lazy
parent8cbba944581a81931848352ec45646d882d9b3d0 (diff)
downloadiced-a92307890fcc3a567ec3d1b1cffebb59ae98991e.tar.gz
iced-a92307890fcc3a567ec3d1b1cffebb59ae98991e.tar.bz2
iced-a92307890fcc3a567ec3d1b1cffebb59ae98991e.zip
feat: enable overlay in component
Diffstat (limited to 'lazy')
-rw-r--r--lazy/src/component.rs392
1 files changed, 351 insertions, 41 deletions
diff --git a/lazy/src/component.rs b/lazy/src/component.rs
index 698f1796..bdd0addb 100644
--- a/lazy/src/component.rs
+++ b/lazy/src/component.rs
@@ -4,10 +4,12 @@ use iced_native::mouse;
use iced_native::overlay;
use iced_native::renderer;
use iced_native::{
- Clipboard, Element, Hasher, Length, Point, Rectangle, Shell, Widget,
+ Clipboard, Element, Hasher, Length, Point, Rectangle, Shell, Size, Widget,
};
use ouroboros::self_referencing;
+use std::cell::RefCell;
+use std::hash::Hash;
use std::marker::PhantomData;
pub fn view<'a, C, Message, Renderer>(
@@ -19,16 +21,22 @@ where
Renderer: iced_native::Renderer + 'a,
{
Element::new(Instance {
- state: Some(
+ state: RefCell::new(Some(
StateBuilder {
component: Box::new(component),
- cache_builder: |state| Cache {
- element: state.view(),
- message: PhantomData,
+ cache_builder: |state| {
+ Some(
+ CacheBuilder {
+ element: state.view(),
+ message: PhantomData,
+ overlay_builder: |_| None,
+ }
+ .build(),
+ )
},
}
.build(),
- ),
+ )),
})
}
@@ -41,7 +49,7 @@ pub trait Component<Message, Renderer> {
}
struct Instance<'a, Message, Renderer, Event> {
- state: Option<State<'a, Message, Renderer, Event>>,
+ state: RefCell<Option<State<'a, Message, Renderer, Event>>>,
}
#[self_referencing]
@@ -50,12 +58,17 @@ struct State<'a, Message: 'a, Renderer: 'a, Event: 'a> {
#[borrows(mut component)]
#[covariant]
- cache: Cache<'this, Message, Renderer, Event>,
+ cache: Option<Cache<'this, Message, Renderer, Event>>,
}
-struct Cache<'a, Message, Renderer, Event> {
+#[self_referencing]
+struct Cache<'a, Message, Renderer: 'a, Event: 'a> {
element: Element<'a, Event, Renderer>,
message: PhantomData<Message>,
+
+ #[borrows(mut element)]
+ #[covariant]
+ overlay: Option<overlay::Element<'this, Event, Renderer>>,
}
impl<'a, Message, Renderer, Event> Widget<Message, Renderer>
@@ -64,11 +77,47 @@ where
Renderer: iced_native::Renderer,
{
fn width(&self) -> Length {
- self.state.as_ref().unwrap().borrow_cache().element.width()
+ self.state
+ .borrow_mut()
+ .as_mut()
+ .unwrap()
+ .with_cache_mut(|cache| {
+ let element = cache.take().unwrap().into_heads().element;
+ let width = element.width();
+
+ *cache = Some(
+ CacheBuilder {
+ element,
+ message: PhantomData,
+ overlay_builder: |_| None,
+ }
+ .build(),
+ );
+
+ width
+ })
}
fn height(&self) -> Length {
- self.state.as_ref().unwrap().borrow_cache().element.height()
+ self.state
+ .borrow_mut()
+ .as_mut()
+ .unwrap()
+ .with_cache_mut(|cache| {
+ let element = cache.take().unwrap().into_heads().element;
+ let height = element.height();
+
+ *cache = Some(
+ CacheBuilder {
+ element,
+ message: PhantomData,
+ overlay_builder: |_| None,
+ }
+ .build(),
+ );
+
+ height
+ })
}
fn layout(
@@ -77,11 +126,24 @@ where
limits: &layout::Limits,
) -> layout::Node {
self.state
- .as_ref()
+ .borrow_mut()
+ .as_mut()
.unwrap()
- .borrow_cache()
- .element
- .layout(renderer, limits)
+ .with_cache_mut(|cache| {
+ let element = cache.take().unwrap().into_heads().element;
+ let layout = element.layout(renderer, limits);
+
+ *cache = Some(
+ CacheBuilder {
+ element,
+ message: PhantomData,
+ overlay_builder: |_| None,
+ }
+ .build(),
+ );
+
+ layout
+ })
}
fn on_event(
@@ -96,16 +158,32 @@ where
let mut local_messages = Vec::new();
let mut local_shell = Shell::new(&mut local_messages);
- let event_status =
- self.state.as_mut().unwrap().with_cache_mut(|cache| {
- cache.element.on_event(
+ let event_status = self
+ .state
+ .borrow_mut()
+ .as_mut()
+ .unwrap()
+ .with_cache_mut(|cache| {
+ let mut element = cache.take().unwrap().into_heads().element;
+ let event_status = element.on_event(
event,
layout,
cursor_position,
renderer,
clipboard,
&mut local_shell,
- )
+ );
+
+ *cache = Some(
+ CacheBuilder {
+ element,
+ message: PhantomData,
+ overlay_builder: |_| None,
+ }
+ .build(),
+ );
+
+ event_status
});
if !local_messages.is_empty() {
@@ -119,12 +197,18 @@ where
shell.publish(message);
}
- self.state = Some(
+ *self.state.borrow_mut() = Some(
StateBuilder {
component,
- cache_builder: |state| Cache {
- element: state.view(),
- message: PhantomData,
+ cache_builder: |state| {
+ Some(
+ CacheBuilder {
+ element: state.view(),
+ message: PhantomData,
+ overlay_builder: |_| None,
+ }
+ .build(),
+ )
},
}
.build(),
@@ -144,22 +228,49 @@ where
cursor_position: Point,
viewport: &Rectangle,
) {
- self.state.as_ref().unwrap().borrow_cache().element.draw(
- renderer,
- style,
- layout,
- cursor_position,
- viewport,
- )
+ self.state
+ .borrow_mut()
+ .as_mut()
+ .unwrap()
+ .with_cache_mut(|cache| {
+ let element = cache.take().unwrap().into_heads().element;
+ element.draw(
+ renderer,
+ style,
+ layout,
+ cursor_position,
+ viewport,
+ );
+
+ *cache = Some(
+ CacheBuilder {
+ element,
+ message: PhantomData,
+ overlay_builder: |_| None,
+ }
+ .build(),
+ );
+ })
}
fn hash_layout(&self, state: &mut Hasher) {
self.state
- .as_ref()
+ .borrow_mut()
+ .as_mut()
.unwrap()
- .borrow_cache()
- .element
- .hash_layout(state)
+ .with_cache_mut(|cache| {
+ let element = cache.take().unwrap().into_heads().element;
+ element.hash_layout(state);
+
+ *cache = Some(
+ CacheBuilder {
+ element,
+ message: PhantomData,
+ overlay_builder: |_| None,
+ }
+ .build(),
+ );
+ })
}
fn mouse_interaction(
@@ -169,18 +280,217 @@ where
viewport: &Rectangle,
) -> mouse::Interaction {
self.state
- .as_ref()
+ .borrow_mut()
+ .as_mut()
.unwrap()
- .borrow_cache()
- .element
- .mouse_interaction(layout, cursor_position, viewport)
+ .with_cache_mut(|cache| {
+ let element = cache.take().unwrap().into_heads().element;
+ let mouse_interaction = element.mouse_interaction(
+ layout,
+ cursor_position,
+ viewport,
+ );
+
+ *cache = Some(
+ CacheBuilder {
+ element,
+ message: PhantomData,
+ overlay_builder: |_| None,
+ }
+ .build(),
+ );
+
+ mouse_interaction
+ })
}
fn overlay(
&mut self,
- _layout: Layout<'_>,
+ layout: Layout<'_>,
) -> Option<overlay::Element<'_, Message, Renderer>> {
- // TODO: Rethink overlay composability
- None
+ let has_overlay = self
+ .state
+ .borrow_mut()
+ .as_mut()
+ .unwrap()
+ .with_cache_mut(|cache| {
+ let element = cache.take().unwrap().into_heads().element;
+
+ *cache = Some(
+ CacheBuilder {
+ element,
+ message: PhantomData,
+ overlay_builder: |element| element.overlay(layout),
+ }
+ .build(),
+ );
+
+ cache.as_ref().unwrap().borrow_overlay().is_some()
+ });
+
+ let Self { state, .. } = self;
+
+ has_overlay.then(|| {
+ overlay::Element::new(
+ layout.position(),
+ Box::new(Overlay { state }),
+ )
+ })
+ }
+}
+
+struct Overlay<'a, 'b, Message, Event, Renderer> {
+ state: &'b RefCell<Option<State<'a, Message, Renderer, Event>>>,
+}
+
+impl<'a, 'b, Message, Event, Renderer> overlay::Overlay<Message, Renderer>
+ for Overlay<'a, 'b, Message, Event, Renderer>
+where
+ Renderer: iced_native::Renderer,
+{
+ fn layout(
+ &self,
+ renderer: &Renderer,
+ bounds: Size,
+ position: Point,
+ ) -> layout::Node {
+ self.state
+ .borrow_mut()
+ .as_mut()
+ .unwrap()
+ .with_cache_mut(|cache| {
+ cache.as_mut().unwrap().with_overlay_mut(|overlay| {
+ *overlay = overlay.take().map(|x| {
+ let vector = position - x.position();
+ x.translate(vector)
+ });
+ overlay
+ .as_mut()
+ .map(|overlay| overlay.layout(renderer, bounds))
+ .unwrap_or_else(|| layout::Node::new(Size::ZERO))
+ })
+ })
+ }
+
+ fn draw(
+ &self,
+ renderer: &mut Renderer,
+ style: &renderer::Style,
+ layout: Layout<'_>,
+ cursor_position: Point,
+ ) {
+ self.state.borrow().as_ref().unwrap().with_cache(|cache| {
+ if let Some(overlay) =
+ cache.as_ref().unwrap().borrow_overlay().as_ref()
+ {
+ overlay.draw(renderer, style, layout, cursor_position);
+ }
+ })
+ }
+
+ fn mouse_interaction(
+ &self,
+ layout: Layout<'_>,
+ cursor_position: Point,
+ viewport: &Rectangle,
+ ) -> mouse::Interaction {
+ self.state.borrow().as_ref().unwrap().with_cache(|cache| {
+ cache
+ .as_ref()
+ .unwrap()
+ .borrow_overlay()
+ .as_ref()
+ .map(|overlay| {
+ overlay.mouse_interaction(layout, cursor_position, viewport)
+ })
+ .unwrap_or(mouse::Interaction::default())
+ })
+ }
+
+ fn hash_layout(&self, state: &mut Hasher, position: Point) {
+ struct Marker;
+ std::any::TypeId::of::<Marker>().hash(state);
+
+ (position.x as u32).hash(state);
+ (position.y as u32).hash(state);
+
+ self.state.borrow().as_ref().unwrap().with_cache(|cache| {
+ if let Some(overlay) =
+ cache.as_ref().unwrap().borrow_overlay().as_ref()
+ {
+ overlay.hash_layout(state);
+ }
+ })
+ }
+
+ fn on_event(
+ &mut self,
+ event: iced_native::Event,
+ layout: Layout<'_>,
+ cursor_position: Point,
+ renderer: &Renderer,
+ clipboard: &mut dyn Clipboard,
+ shell: &mut Shell<'_, Message>,
+ ) -> iced_native::event::Status {
+ let mut local_messages = Vec::new();
+ let mut local_shell = Shell::new(&mut local_messages);
+
+ let event_status = self
+ .state
+ .borrow_mut()
+ .as_mut()
+ .unwrap()
+ .with_cache_mut(|cache| {
+ cache.as_mut().unwrap().with_overlay_mut(|overlay| {
+ overlay
+ .as_mut()
+ .map(|overlay| {
+ overlay.on_event(
+ event,
+ layout,
+ cursor_position,
+ renderer,
+ clipboard,
+ &mut local_shell,
+ )
+ })
+ .unwrap_or(iced_native::event::Status::Ignored)
+ })
+ });
+
+ if !local_messages.is_empty() {
+ let mut component =
+ self.state.take().unwrap().into_heads().component;
+
+ for message in local_messages
+ .into_iter()
+ .filter_map(|message| component.update(message))
+ {
+ shell.publish(message);
+ }
+
+ *self.state.borrow_mut() = Some(
+ StateBuilder {
+ component,
+ cache_builder: |state| {
+ Some(
+ CacheBuilder {
+ element: state.view(),
+ message: PhantomData,
+ overlay_builder: |element| {
+ element.overlay(layout)
+ },
+ }
+ .build(),
+ )
+ },
+ }
+ .build(),
+ );
+
+ shell.invalidate_layout();
+ }
+
+ event_status
}
}