feat: apply overlay to headerbar when it loses focus
This commit is contained in:
parent
9fb3d874e1
commit
8157ed5c63
5 changed files with 245 additions and 15 deletions
|
|
@ -468,6 +468,7 @@ impl Application for Window {
|
||||||
};
|
};
|
||||||
|
|
||||||
let mut header = header_bar()
|
let mut header = header_bar()
|
||||||
|
.window_id(window::Id::MAIN)
|
||||||
.title("COSMIC Design System - Iced")
|
.title("COSMIC Design System - Iced")
|
||||||
.on_close(Message::Close)
|
.on_close(Message::Close)
|
||||||
.on_drag(Message::Drag)
|
.on_drag(Message::Drag)
|
||||||
|
|
|
||||||
|
|
@ -5,7 +5,7 @@ use cosmic::{
|
||||||
iced::{self, event, window},
|
iced::{self, event, window},
|
||||||
iced_core::{id, Alignment, Length, Point},
|
iced_core::{id, Alignment, Length, Point},
|
||||||
iced_widget::{column, container, scrollable, text, text_input},
|
iced_widget::{column, container, scrollable, text, text_input},
|
||||||
widget::{button, cosmic_container},
|
widget::{button, header_bar},
|
||||||
ApplicationExt, Command,
|
ApplicationExt, Command,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -97,6 +97,7 @@ impl cosmic::Application for MultiWindow {
|
||||||
let (id, spawn_window) = window::spawn(window::Settings {
|
let (id, spawn_window) = window::spawn(window::Settings {
|
||||||
position: Default::default(),
|
position: Default::default(),
|
||||||
exit_on_close_request: count % 2 == 0,
|
exit_on_close_request: count % 2 == 0,
|
||||||
|
decorations: false,
|
||||||
..Default::default()
|
..Default::default()
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
@ -140,13 +141,18 @@ impl cosmic::Application for MultiWindow {
|
||||||
.align_items(Alignment::Center),
|
.align_items(Alignment::Center),
|
||||||
);
|
);
|
||||||
|
|
||||||
container(container(content).width(200).center_x())
|
let window_content = container(container(content).width(200).center_x())
|
||||||
.style(cosmic::style::Container::Background)
|
.style(cosmic::style::Container::Background)
|
||||||
.width(Length::Fill)
|
.width(Length::Fill)
|
||||||
.height(Length::Fill)
|
.height(Length::Fill)
|
||||||
.center_x()
|
.center_x()
|
||||||
.center_y()
|
.center_y();
|
||||||
.into()
|
|
||||||
|
if id == window::Id::MAIN {
|
||||||
|
window_content.into()
|
||||||
|
} else {
|
||||||
|
column![header_bar().window_id(id), window_content].into()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn view(&self) -> cosmic::prelude::Element<Self::Message> {
|
fn view(&self) -> cosmic::prelude::Element<Self::Message> {
|
||||||
|
|
|
||||||
|
|
@ -647,6 +647,7 @@ impl<App: Application> ApplicationExt for App {
|
||||||
.push_maybe(if core.window.show_headerbar {
|
.push_maybe(if core.window.show_headerbar {
|
||||||
Some({
|
Some({
|
||||||
let mut header = crate::widget::header_bar()
|
let mut header = crate::widget::header_bar()
|
||||||
|
.window_id(window::Id::MAIN)
|
||||||
.title(&core.window.header_title)
|
.title(&core.window.header_title)
|
||||||
.on_drag(Message::Cosmic(cosmic::Message::Drag))
|
.on_drag(Message::Cosmic(cosmic::Message::Drag))
|
||||||
.on_close(Message::Cosmic(cosmic::Message::Close));
|
.on_close(Message::Cosmic(cosmic::Message::Close));
|
||||||
|
|
|
||||||
|
|
@ -385,18 +385,12 @@ impl container::StyleSheet for Theme {
|
||||||
}
|
}
|
||||||
Container::HeaderBar => {
|
Container::HeaderBar => {
|
||||||
let palette = self.cosmic();
|
let palette = self.cosmic();
|
||||||
let mut header_top = palette.background.base;
|
let header_top = palette.background.base;
|
||||||
let header_bottom = palette.background.base;
|
|
||||||
header_top.alpha = 0.8;
|
|
||||||
|
|
||||||
container::Appearance {
|
container::Appearance {
|
||||||
icon_color: Some(Color::from(palette.accent.base)),
|
icon_color: Some(Color::from(palette.accent.base)),
|
||||||
text_color: Some(Color::from(palette.background.on)),
|
text_color: Some(Color::from(palette.background.on)),
|
||||||
background: Some(iced::Background::Gradient(iced_core::Gradient::Linear(
|
background: Some(iced::Background::Color(header_top.into())),
|
||||||
Linear::new(Radians(PI))
|
|
||||||
.add_stop(0.0, header_top.into())
|
|
||||||
.add_stop(1.0, header_bottom.into()),
|
|
||||||
))),
|
|
||||||
border_radius: BorderRadius::from([
|
border_radius: BorderRadius::from([
|
||||||
palette.corner_radii.radius_xs[0],
|
palette.corner_radii.radius_xs[0],
|
||||||
palette.corner_radii.radius_xs[3],
|
palette.corner_radii.radius_xs[3],
|
||||||
|
|
|
||||||
|
|
@ -4,8 +4,9 @@
|
||||||
use crate::{ext::CollectionWidget, widget, Element};
|
use crate::{ext::CollectionWidget, widget, Element};
|
||||||
use apply::Apply;
|
use apply::Apply;
|
||||||
use derive_setters::Setters;
|
use derive_setters::Setters;
|
||||||
use iced::Length;
|
use iced::{window, Length};
|
||||||
use std::borrow::Cow;
|
use iced_core::{renderer::Quad, widget::tree, Background, Color, Renderer, Widget};
|
||||||
|
use std::{borrow::Cow, process::Child};
|
||||||
|
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub fn header_bar<'a, Message>() -> HeaderBar<'a, Message> {
|
pub fn header_bar<'a, Message>() -> HeaderBar<'a, Message> {
|
||||||
|
|
@ -18,6 +19,7 @@ pub fn header_bar<'a, Message>() -> HeaderBar<'a, Message> {
|
||||||
start: Vec::new(),
|
start: Vec::new(),
|
||||||
center: Vec::new(),
|
center: Vec::new(),
|
||||||
end: Vec::new(),
|
end: Vec::new(),
|
||||||
|
window_id: None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -43,6 +45,10 @@ pub struct HeaderBar<'a, Message> {
|
||||||
#[setters(strip_option)]
|
#[setters(strip_option)]
|
||||||
on_minimize: Option<Message>,
|
on_minimize: Option<Message>,
|
||||||
|
|
||||||
|
/// The window id for the headerbar.
|
||||||
|
#[setters(strip_option)]
|
||||||
|
window_id: Option<iced::window::Id>,
|
||||||
|
|
||||||
/// Elements packed at the start of the headerbar.
|
/// Elements packed at the start of the headerbar.
|
||||||
#[setters(skip)]
|
#[setters(skip)]
|
||||||
start: Vec<Element<'a, Message>>,
|
start: Vec<Element<'a, Message>>,
|
||||||
|
|
@ -84,6 +90,194 @@ impl<'a, Message: Clone + 'static> HeaderBar<'a, Message> {
|
||||||
self.end.push(widget.into());
|
self.end.push(widget.into());
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Build the widget
|
||||||
|
#[must_use]
|
||||||
|
pub fn build(self) -> HeaderBarWidget<'a, Message> {
|
||||||
|
HeaderBarWidget {
|
||||||
|
window_id: self.window_id,
|
||||||
|
header_bar_inner: self.into_element(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct HeaderBarWidget<'a, Message> {
|
||||||
|
header_bar_inner: Element<'a, Message>,
|
||||||
|
window_id: Option<iced::window::Id>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, Message: Clone + 'static> Widget<Message, crate::Renderer>
|
||||||
|
for HeaderBarWidget<'a, Message>
|
||||||
|
{
|
||||||
|
fn children(&self) -> Vec<tree::Tree> {
|
||||||
|
vec![tree::Tree::new(&self.header_bar_inner)]
|
||||||
|
}
|
||||||
|
|
||||||
|
fn width(&self) -> Length {
|
||||||
|
self.header_bar_inner.width()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn height(&self) -> Length {
|
||||||
|
self.header_bar_inner.height()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn tag(&self) -> tree::Tag {
|
||||||
|
tree::Tag::of::<MyState>()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn diff(&mut self, tree: &mut tree::Tree) {
|
||||||
|
tree.diff_children(&mut [&mut self.header_bar_inner]);
|
||||||
|
let prev = tree.state.downcast_mut::<MyState>();
|
||||||
|
if prev.window_id != self.window_id {
|
||||||
|
*prev = MyState::new(self.window_id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn state(&self) -> tree::State {
|
||||||
|
let state = MyState::new(self.window_id);
|
||||||
|
tree::State::new(state)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn layout(
|
||||||
|
&self,
|
||||||
|
tree: &mut tree::Tree,
|
||||||
|
renderer: &crate::Renderer,
|
||||||
|
limits: &iced_core::layout::Limits,
|
||||||
|
) -> iced_core::layout::Node {
|
||||||
|
let child_tree = &mut tree.children[0];
|
||||||
|
let child = self
|
||||||
|
.header_bar_inner
|
||||||
|
.as_widget()
|
||||||
|
.layout(child_tree, renderer, limits);
|
||||||
|
iced_core::layout::Node::with_children(child.size(), vec![child])
|
||||||
|
}
|
||||||
|
|
||||||
|
fn draw(
|
||||||
|
&self,
|
||||||
|
tree: &tree::Tree,
|
||||||
|
renderer: &mut crate::Renderer,
|
||||||
|
theme: &<crate::Renderer as iced_core::Renderer>::Theme,
|
||||||
|
style: &iced_core::renderer::Style,
|
||||||
|
layout: iced_core::Layout<'_>,
|
||||||
|
cursor: iced_core::mouse::Cursor,
|
||||||
|
viewport: &iced_core::Rectangle,
|
||||||
|
) {
|
||||||
|
let layout_children = layout.children().next().unwrap();
|
||||||
|
let state_children = &tree.children[0];
|
||||||
|
self.header_bar_inner.as_widget().draw(
|
||||||
|
state_children,
|
||||||
|
renderer,
|
||||||
|
theme,
|
||||||
|
style,
|
||||||
|
layout_children,
|
||||||
|
cursor,
|
||||||
|
viewport,
|
||||||
|
);
|
||||||
|
|
||||||
|
let state = tree.state.downcast_ref::<MyState>();
|
||||||
|
if !state.window_has_focus {
|
||||||
|
let header_bar_appearance =
|
||||||
|
<crate::Theme as crate::iced_style::container::StyleSheet>::appearance(
|
||||||
|
theme,
|
||||||
|
&crate::theme::Container::HeaderBar,
|
||||||
|
);
|
||||||
|
let cosmic = theme.cosmic();
|
||||||
|
let mut neutral_0 = cosmic.palette.neutral_0;
|
||||||
|
neutral_0.alpha = 0.3;
|
||||||
|
|
||||||
|
// draw overlay rectangle
|
||||||
|
renderer.fill_quad(
|
||||||
|
Quad {
|
||||||
|
bounds: layout.bounds(),
|
||||||
|
border_radius: header_bar_appearance.border_radius,
|
||||||
|
border_width: 0.0,
|
||||||
|
border_color: Color::TRANSPARENT,
|
||||||
|
},
|
||||||
|
Background::Color(neutral_0.into()),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn on_event(
|
||||||
|
&mut self,
|
||||||
|
state: &mut tree::Tree,
|
||||||
|
event: iced_core::Event,
|
||||||
|
layout: iced_core::Layout<'_>,
|
||||||
|
cursor: iced_core::mouse::Cursor,
|
||||||
|
renderer: &crate::Renderer,
|
||||||
|
clipboard: &mut dyn iced_core::Clipboard,
|
||||||
|
shell: &mut iced_core::Shell<'_, Message>,
|
||||||
|
viewport: &iced_core::Rectangle,
|
||||||
|
) -> iced_core::event::Status {
|
||||||
|
if let iced_core::Event::Window(id, iced_core::window::Event::Focused) = event {
|
||||||
|
let state = state.state.downcast_mut::<MyState>();
|
||||||
|
state.focus_window(id);
|
||||||
|
} else if let iced_core::Event::Window(id, iced_core::window::Event::Unfocused) = event {
|
||||||
|
let state = state.state.downcast_mut::<MyState>();
|
||||||
|
state.unfocus_window(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
let child_state = &mut state.children[0];
|
||||||
|
let child_layout = layout.children().next().unwrap();
|
||||||
|
self.header_bar_inner.as_widget_mut().on_event(
|
||||||
|
child_state,
|
||||||
|
event,
|
||||||
|
child_layout,
|
||||||
|
cursor,
|
||||||
|
renderer,
|
||||||
|
clipboard,
|
||||||
|
shell,
|
||||||
|
viewport,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn mouse_interaction(
|
||||||
|
&self,
|
||||||
|
state: &tree::Tree,
|
||||||
|
layout: iced_core::Layout<'_>,
|
||||||
|
cursor: iced_core::mouse::Cursor,
|
||||||
|
viewport: &iced_core::Rectangle,
|
||||||
|
renderer: &crate::Renderer,
|
||||||
|
) -> iced_core::mouse::Interaction {
|
||||||
|
let child_tree = &state.children[0];
|
||||||
|
let child_layout = layout.children().next().unwrap();
|
||||||
|
self.header_bar_inner.as_widget().mouse_interaction(
|
||||||
|
child_tree,
|
||||||
|
child_layout,
|
||||||
|
cursor,
|
||||||
|
viewport,
|
||||||
|
renderer,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn operate(
|
||||||
|
&self,
|
||||||
|
state: &mut tree::Tree,
|
||||||
|
layout: iced_core::Layout<'_>,
|
||||||
|
renderer: &crate::Renderer,
|
||||||
|
operation: &mut dyn iced_core::widget::Operation<
|
||||||
|
iced_core::widget::OperationOutputWrapper<Message>,
|
||||||
|
>,
|
||||||
|
) {
|
||||||
|
let child_tree = &mut state.children[0];
|
||||||
|
let child_layout = layout.children().next().unwrap();
|
||||||
|
self.header_bar_inner
|
||||||
|
.as_widget()
|
||||||
|
.operate(child_tree, child_layout, renderer, operation);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn overlay<'b>(
|
||||||
|
&'b mut self,
|
||||||
|
state: &'b mut tree::Tree,
|
||||||
|
layout: iced_core::Layout<'_>,
|
||||||
|
renderer: &crate::Renderer,
|
||||||
|
) -> Option<iced_core::overlay::Element<'b, Message, crate::Renderer>> {
|
||||||
|
let child_tree = &mut state.children[0];
|
||||||
|
let child_layout = layout.children().next().unwrap();
|
||||||
|
self.header_bar_inner
|
||||||
|
.as_widget_mut()
|
||||||
|
.overlay(child_tree, child_layout, renderer)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a, Message: Clone + 'static> HeaderBar<'a, Message> {
|
impl<'a, Message: Clone + 'static> HeaderBar<'a, Message> {
|
||||||
|
|
@ -207,6 +401,40 @@ impl<'a, Message: Clone + 'static> HeaderBar<'a, Message> {
|
||||||
|
|
||||||
impl<'a, Message: Clone + 'static> From<HeaderBar<'a, Message>> for Element<'a, Message> {
|
impl<'a, Message: Clone + 'static> From<HeaderBar<'a, Message>> for Element<'a, Message> {
|
||||||
fn from(headerbar: HeaderBar<'a, Message>) -> Self {
|
fn from(headerbar: HeaderBar<'a, Message>) -> Self {
|
||||||
headerbar.into_element()
|
Element::new(headerbar.build())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, Message: Clone + 'static> From<HeaderBarWidget<'a, Message>> for Element<'a, Message> {
|
||||||
|
fn from(headerbar: HeaderBarWidget<'a, Message>) -> Self {
|
||||||
|
Element::new(headerbar)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct MyState {
|
||||||
|
pub window_id: Option<window::Id>,
|
||||||
|
pub window_has_focus: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl MyState {
|
||||||
|
pub fn new(id: Option<window::Id>) -> Self {
|
||||||
|
Self {
|
||||||
|
window_id: id,
|
||||||
|
window_has_focus: id.is_none(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn focus_window(&mut self, id: window::Id) {
|
||||||
|
if self.window_id != Some(id) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
self.window_has_focus = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn unfocus_window(&mut self, id: window::Id) {
|
||||||
|
if self.window_id != Some(id) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
self.window_has_focus = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue