2022-10-09 02:35:03 -07:00
|
|
|
use apply::Apply;
|
2022-10-11 15:27:39 +02:00
|
|
|
use derive_setters::*;
|
2022-10-28 21:59:41 -07:00
|
|
|
use iced::{self, alignment::Vertical, widget, Length};
|
2022-10-09 02:35:03 -07:00
|
|
|
use iced_lazy::Component;
|
2022-10-28 21:59:41 -07:00
|
|
|
use crate::{theme, Element, Renderer};
|
2022-10-09 02:35:03 -07:00
|
|
|
|
2022-10-11 15:27:39 +02:00
|
|
|
#[derive(Setters)]
|
2022-10-09 16:39:10 +02:00
|
|
|
pub struct HeaderBar<Message> {
|
2022-10-09 02:35:03 -07:00
|
|
|
title: String,
|
|
|
|
|
nav_title: String,
|
|
|
|
|
sidebar_active: bool,
|
|
|
|
|
show_minimize: bool,
|
|
|
|
|
show_maximize: bool,
|
2022-10-11 15:27:39 +02:00
|
|
|
#[setters(strip_option)]
|
|
|
|
|
on_close: Option<Message>,
|
|
|
|
|
#[setters(strip_option)]
|
|
|
|
|
on_drag: Option<Message>,
|
|
|
|
|
#[setters(strip_option)]
|
|
|
|
|
on_maximize: Option<Message>,
|
|
|
|
|
#[setters(strip_option)]
|
|
|
|
|
on_minimize: Option<Message>,
|
|
|
|
|
#[setters(strip_option)]
|
|
|
|
|
on_sidebar_toggle: Option<Message>,
|
2022-10-09 02:35:03 -07:00
|
|
|
}
|
|
|
|
|
|
2022-10-11 15:27:39 +02:00
|
|
|
pub fn header_bar<Message>() -> HeaderBar<Message> {
|
|
|
|
|
HeaderBar {
|
|
|
|
|
title: String::default(),
|
|
|
|
|
nav_title: String::default(),
|
|
|
|
|
sidebar_active: false,
|
|
|
|
|
show_minimize: false,
|
|
|
|
|
show_maximize: false,
|
|
|
|
|
on_sidebar_toggle: None,
|
|
|
|
|
on_close: None,
|
|
|
|
|
on_drag: None,
|
|
|
|
|
on_maximize: None,
|
2022-10-09 11:25:46 -07:00
|
|
|
on_minimize: None,
|
2022-10-11 15:27:39 +02:00
|
|
|
}
|
2022-10-09 02:35:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
|
pub enum HeaderEvent {
|
|
|
|
|
Close,
|
|
|
|
|
ToggleSidebar,
|
|
|
|
|
Drag,
|
2022-10-09 16:39:10 +02:00
|
|
|
Minimize,
|
|
|
|
|
Maximize,
|
2022-10-09 02:35:03 -07:00
|
|
|
}
|
|
|
|
|
|
2022-10-09 16:39:10 +02:00
|
|
|
impl<Message: Clone> Component<Message, Renderer> for HeaderBar<Message> {
|
2022-10-09 02:35:03 -07:00
|
|
|
type State = ();
|
|
|
|
|
|
|
|
|
|
type Event = HeaderEvent;
|
|
|
|
|
|
2022-10-09 16:39:10 +02:00
|
|
|
fn update(&mut self, _state: &mut Self::State, event: Self::Event) -> Option<Message> {
|
2022-10-09 02:35:03 -07:00
|
|
|
match event {
|
2022-10-09 11:25:46 -07:00
|
|
|
HeaderEvent::Close => self.on_close.clone(),
|
|
|
|
|
|
|
|
|
|
HeaderEvent::ToggleSidebar => self.on_sidebar_toggle.clone(),
|
|
|
|
|
|
|
|
|
|
HeaderEvent::Drag => self.on_drag.clone(),
|
|
|
|
|
|
|
|
|
|
HeaderEvent::Maximize => self.on_maximize.clone(),
|
|
|
|
|
|
|
|
|
|
HeaderEvent::Minimize => self.on_minimize.clone(),
|
2022-10-09 02:35:03 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-10-09 16:39:10 +02:00
|
|
|
fn view(&self, _state: &Self::State) -> Element<Self::Event> {
|
2022-10-09 02:35:03 -07:00
|
|
|
let nav_button = {
|
|
|
|
|
let text = widget::text(&self.nav_title)
|
|
|
|
|
.vertical_alignment(Vertical::Center)
|
|
|
|
|
.width(Length::Shrink)
|
|
|
|
|
.height(Length::Fill);
|
|
|
|
|
|
|
|
|
|
let icon = super::icon(
|
|
|
|
|
if self.sidebar_active {
|
|
|
|
|
"go-previous-symbolic"
|
|
|
|
|
} else {
|
|
|
|
|
"go-next-symbolic"
|
|
|
|
|
},
|
|
|
|
|
24,
|
|
|
|
|
)
|
|
|
|
|
.width(Length::Units(24))
|
|
|
|
|
.height(Length::Fill);
|
|
|
|
|
|
|
|
|
|
widget::row!(text, icon)
|
|
|
|
|
.padding(4)
|
|
|
|
|
.spacing(4)
|
|
|
|
|
.apply(widget::button)
|
|
|
|
|
.style(theme::Button::Primary)
|
|
|
|
|
.on_press(HeaderEvent::ToggleSidebar)
|
|
|
|
|
.apply(widget::container)
|
|
|
|
|
.center_y()
|
|
|
|
|
.height(Length::Fill)
|
|
|
|
|
.into()
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let content = widget::container(widget::text(&self.title))
|
|
|
|
|
.center_x()
|
|
|
|
|
.center_y()
|
|
|
|
|
.width(Length::Fill)
|
|
|
|
|
.height(Length::Fill)
|
|
|
|
|
.into();
|
2022-10-09 16:39:10 +02:00
|
|
|
|
2022-10-09 02:35:03 -07:00
|
|
|
let window_controls = {
|
2022-10-28 21:59:41 -07:00
|
|
|
let mut widgets: Vec<Element<HeaderEvent>> = Vec::with_capacity(3);
|
2022-10-09 02:35:03 -07:00
|
|
|
|
|
|
|
|
let icon = |name, size, on_press| {
|
|
|
|
|
super::icon(name, size)
|
|
|
|
|
.apply(widget::button)
|
|
|
|
|
.style(theme::Button::Primary)
|
|
|
|
|
.on_press(on_press)
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if self.show_minimize {
|
2022-10-09 16:39:10 +02:00
|
|
|
widgets.push(icon("window-minimize-symbolic", 16, HeaderEvent::Minimize).into());
|
2022-10-09 02:35:03 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if self.show_maximize {
|
2022-10-09 16:39:10 +02:00
|
|
|
widgets.push(icon("window-maximize-symbolic", 16, HeaderEvent::Maximize).into());
|
2022-10-09 02:35:03 -07:00
|
|
|
}
|
|
|
|
|
|
2022-10-09 16:39:10 +02:00
|
|
|
widgets.push(icon("window-close-symbolic", 16, HeaderEvent::Close).into());
|
2022-10-09 02:35:03 -07:00
|
|
|
|
|
|
|
|
widget::row(widgets)
|
|
|
|
|
.spacing(8)
|
|
|
|
|
.apply(widget::container)
|
|
|
|
|
.height(Length::Fill)
|
|
|
|
|
.center_y()
|
|
|
|
|
.into()
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
widget::row(vec![nav_button, content, window_controls])
|
|
|
|
|
.height(Length::Units(50))
|
|
|
|
|
.padding(10)
|
|
|
|
|
.apply(widget::event_container)
|
|
|
|
|
.center_y()
|
|
|
|
|
.on_press(HeaderEvent::Drag)
|
2022-10-09 16:39:10 +02:00
|
|
|
.on_release(HeaderEvent::Maximize)
|
2022-10-09 02:35:03 -07:00
|
|
|
.into()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-10-09 16:39:10 +02:00
|
|
|
impl<'a, Message: Clone + 'a> From<HeaderBar<Message>> for Element<'a, Message> {
|
2022-10-09 02:35:03 -07:00
|
|
|
fn from(header_bar: HeaderBar<Message>) -> Self {
|
|
|
|
|
iced_lazy::component(header_bar)
|
|
|
|
|
}
|
2022-10-09 16:39:10 +02:00
|
|
|
}
|