Implement loupe example
This commit is contained in:
parent
f4d6648601
commit
7cba9a3e7f
2 changed files with 195 additions and 0 deletions
10
examples/loupe/Cargo.toml
Normal file
10
examples/loupe/Cargo.toml
Normal file
|
|
@ -0,0 +1,10 @@
|
|||
[package]
|
||||
name = "loupe"
|
||||
version = "0.1.0"
|
||||
authors = ["Héctor Ramón Jiménez <hector0193@gmail.com>"]
|
||||
edition = "2021"
|
||||
publish = false
|
||||
|
||||
[dependencies]
|
||||
iced.workspace = true
|
||||
iced.features = ["advanced", "debug"]
|
||||
185
examples/loupe/src/main.rs
Normal file
185
examples/loupe/src/main.rs
Normal file
|
|
@ -0,0 +1,185 @@
|
|||
use iced::widget::{button, column, container, text};
|
||||
use iced::{Alignment, Element, Length, Sandbox, Settings};
|
||||
|
||||
use loupe::loupe;
|
||||
|
||||
pub fn main() -> iced::Result {
|
||||
Counter::run(Settings::default())
|
||||
}
|
||||
|
||||
struct Counter {
|
||||
value: i32,
|
||||
}
|
||||
|
||||
#[derive(Debug, Clone, Copy)]
|
||||
enum Message {
|
||||
IncrementPressed,
|
||||
DecrementPressed,
|
||||
}
|
||||
|
||||
impl Sandbox for Counter {
|
||||
type Message = Message;
|
||||
|
||||
fn new() -> Self {
|
||||
Self { value: 0 }
|
||||
}
|
||||
|
||||
fn title(&self) -> String {
|
||||
String::from("Counter - Iced")
|
||||
}
|
||||
|
||||
fn update(&mut self, message: Message) {
|
||||
match message {
|
||||
Message::IncrementPressed => {
|
||||
self.value += 1;
|
||||
}
|
||||
Message::DecrementPressed => {
|
||||
self.value -= 1;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn view(&self) -> Element<Message> {
|
||||
container(loupe(
|
||||
3.0,
|
||||
column![
|
||||
button("Increment").on_press(Message::IncrementPressed),
|
||||
text(self.value).size(50),
|
||||
button("Decrement").on_press(Message::DecrementPressed)
|
||||
]
|
||||
.padding(20)
|
||||
.align_items(Alignment::Center),
|
||||
))
|
||||
.width(Length::Fill)
|
||||
.height(Length::Fill)
|
||||
.center_x()
|
||||
.center_y()
|
||||
.into()
|
||||
}
|
||||
}
|
||||
|
||||
mod loupe {
|
||||
use iced::advanced::layout::{self, Layout};
|
||||
use iced::advanced::renderer;
|
||||
use iced::advanced::widget::{self, Widget};
|
||||
use iced::advanced::Renderer as _;
|
||||
use iced::mouse;
|
||||
use iced::{
|
||||
Color, Element, Length, Rectangle, Renderer, Size, Theme,
|
||||
Transformation,
|
||||
};
|
||||
|
||||
pub fn loupe<'a, Message>(
|
||||
zoom: f32,
|
||||
content: impl Into<Element<'a, Message>>,
|
||||
) -> Loupe<'a, Message>
|
||||
where
|
||||
Message: 'static,
|
||||
{
|
||||
Loupe {
|
||||
zoom,
|
||||
content: content.into().explain(Color::BLACK),
|
||||
}
|
||||
}
|
||||
|
||||
pub struct Loupe<'a, Message> {
|
||||
zoom: f32,
|
||||
content: Element<'a, Message>,
|
||||
}
|
||||
|
||||
impl<'a, Message> Widget<Message, Theme, Renderer> for Loupe<'a, Message> {
|
||||
fn tag(&self) -> widget::tree::Tag {
|
||||
self.content.as_widget().tag()
|
||||
}
|
||||
|
||||
fn state(&self) -> widget::tree::State {
|
||||
self.content.as_widget().state()
|
||||
}
|
||||
|
||||
fn children(&self) -> Vec<widget::Tree> {
|
||||
self.content.as_widget().children()
|
||||
}
|
||||
|
||||
fn diff(&self, tree: &mut widget::Tree) {
|
||||
self.content.as_widget().diff(tree);
|
||||
}
|
||||
|
||||
fn size(&self) -> Size<Length> {
|
||||
self.content.as_widget().size()
|
||||
}
|
||||
|
||||
fn layout(
|
||||
&self,
|
||||
tree: &mut widget::Tree,
|
||||
renderer: &Renderer,
|
||||
limits: &layout::Limits,
|
||||
) -> layout::Node {
|
||||
self.content.as_widget().layout(tree, renderer, limits)
|
||||
}
|
||||
|
||||
fn draw(
|
||||
&self,
|
||||
tree: &widget::Tree,
|
||||
renderer: &mut Renderer,
|
||||
theme: &Theme,
|
||||
style: &renderer::Style,
|
||||
layout: Layout<'_>,
|
||||
cursor: mouse::Cursor,
|
||||
viewport: &Rectangle,
|
||||
) {
|
||||
let bounds = layout.bounds();
|
||||
|
||||
if let Some(position) = cursor.position_in(bounds) {
|
||||
renderer.with_layer(bounds, |renderer| {
|
||||
renderer.with_transformation(
|
||||
Transformation::translate(
|
||||
bounds.x + position.x * (1.0 - self.zoom),
|
||||
bounds.y + position.y * (1.0 - self.zoom),
|
||||
) * Transformation::scale(self.zoom)
|
||||
* Transformation::translate(-bounds.x, -bounds.y),
|
||||
|renderer| {
|
||||
self.content.as_widget().draw(
|
||||
tree,
|
||||
renderer,
|
||||
theme,
|
||||
style,
|
||||
layout,
|
||||
mouse::Cursor::Unavailable,
|
||||
viewport,
|
||||
);
|
||||
},
|
||||
);
|
||||
});
|
||||
} else {
|
||||
self.content.as_widget().draw(
|
||||
tree, renderer, theme, style, layout, cursor, viewport,
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
fn mouse_interaction(
|
||||
&self,
|
||||
_state: &widget::Tree,
|
||||
layout: Layout<'_>,
|
||||
cursor: mouse::Cursor,
|
||||
_viewport: &Rectangle,
|
||||
_renderer: &Renderer,
|
||||
) -> mouse::Interaction {
|
||||
if cursor.is_over(layout.bounds()) {
|
||||
mouse::Interaction::ZoomIn
|
||||
} else {
|
||||
mouse::Interaction::Idle
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl<'a, Message> From<Loupe<'a, Message>>
|
||||
for Element<'a, Message, Theme, Renderer>
|
||||
where
|
||||
Message: 'a,
|
||||
{
|
||||
fn from(loupe: Loupe<'a, Message>) -> Self {
|
||||
Self::new(loupe)
|
||||
}
|
||||
}
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue