1use crate::cosmic_theme::{Density, Spacing};
5use crate::{Element, theme, widget};
6use apply::Apply;
7use derive_setters::Setters;
8use iced_core::{Length, Size, Vector, Widget, layout, text, widget::tree};
9use std::borrow::Cow;
10
11#[must_use]
12pub fn header_bar<'a, Message>() -> HeaderBar<'a, Message> {
13 HeaderBar {
14 title: Cow::Borrowed(""),
15 on_close: None,
16 on_drag: None,
17 on_maximize: None,
18 on_minimize: None,
19 on_right_click: None,
20 start: Vec::new(),
21 center: Vec::new(),
22 end: Vec::new(),
23 density: None,
24 focused: false,
25 maximized: false,
26 sharp_corners: false,
27 is_ssd: false,
28 on_double_click: None,
29 transparent: false,
30 }
31}
32
33#[derive(Setters)]
34pub struct HeaderBar<'a, Message> {
35 #[setters(skip)]
37 title: Cow<'a, str>,
38
39 #[setters(strip_option)]
41 on_close: Option<Message>,
42
43 #[setters(strip_option)]
45 on_drag: Option<Message>,
46
47 #[setters(strip_option)]
49 on_maximize: Option<Message>,
50
51 #[setters(strip_option)]
53 on_minimize: Option<Message>,
54
55 #[setters(strip_option)]
58 on_double_click: Option<Message>,
59
60 #[setters(strip_option)]
62 on_right_click: Option<Message>,
63
64 #[setters(skip)]
66 start: Vec<Element<'a, Message>>,
67
68 #[setters(skip)]
70 center: Vec<Element<'a, Message>>,
71
72 #[setters(skip)]
74 end: Vec<Element<'a, Message>>,
75
76 #[setters(strip_option)]
78 density: Option<Density>,
79
80 focused: bool,
82
83 maximized: bool,
85
86 sharp_corners: bool,
88
89 is_ssd: bool,
91
92 transparent: bool,
94}
95
96impl<'a, Message: Clone + 'static> HeaderBar<'a, Message> {
97 #[must_use]
99 pub fn title(mut self, title: impl Into<Cow<'a, str>> + 'a) -> Self {
100 self.title = title.into();
101 self
102 }
103
104 #[must_use]
106 pub fn start(mut self, widget: impl Into<Element<'a, Message>> + 'a) -> Self {
107 self.start.push(widget.into());
108 self
109 }
110
111 #[must_use]
113 pub fn center(mut self, widget: impl Into<Element<'a, Message>> + 'a) -> Self {
114 self.center.push(widget.into());
115 self
116 }
117
118 #[must_use]
120 pub fn end(mut self, widget: impl Into<Element<'a, Message>> + 'a) -> Self {
121 self.end.push(widget.into());
122 self
123 }
124}
125
126pub struct HeaderBarWidget<'a, Message> {
127 start: Element<'a, Message>,
128 center: Option<Element<'a, Message>>,
129 end: Element<'a, Message>,
130}
131
132impl<'a, Message> HeaderBarWidget<'a, Message> {
133 pub fn new(
134 start: Element<'a, Message>,
135 center: Option<Element<'a, Message>>,
136 end: Element<'a, Message>,
137 ) -> Self {
138 Self { start, center, end }
139 }
140
141 fn elems(&self) -> impl Iterator<Item = &Element<'a, Message>> {
142 std::iter::once(&self.start)
143 .chain(std::iter::once(&self.end))
144 .chain(self.center.as_ref())
145 }
146
147 fn elems_mut(&mut self) -> impl Iterator<Item = &mut Element<'a, Message>> {
148 std::iter::once(&mut self.start)
149 .chain(std::iter::once(&mut self.end))
150 .chain(self.center.as_mut())
151 }
152}
153
154impl<'a, Message: Clone + 'static> Widget<Message, crate::Theme, crate::Renderer>
155 for HeaderBarWidget<'a, Message>
156{
157 fn diff(&mut self, tree: &mut tree::Tree) {
158 if let Some(center) = &mut self.center {
159 tree.diff_children(&mut [&mut self.start, &mut self.end, center]);
160 } else {
161 tree.diff_children(&mut [&mut self.start, &mut self.end]);
162 }
163 }
164
165 fn children(&self) -> Vec<tree::Tree> {
166 self.elems().map(tree::Tree::new).collect()
167 }
168
169 fn size(&self) -> Size<Length> {
170 Size {
171 width: Length::Fill,
172 height: Length::Shrink,
173 }
174 }
175
176 fn layout(
177 &mut self,
178 tree: &mut tree::Tree,
179 renderer: &crate::Renderer,
180 limits: &layout::Limits,
181 ) -> layout::Node {
182 let width = limits.max().width;
183 let height = limits.max().height;
184 let gap = 8.0;
185
186 let end_node =
187 self.end
188 .as_widget_mut()
189 .layout(&mut tree.children[1], renderer, &limits.loose());
190 let end_width = end_node.size().width;
191
192 let start_available = (width - end_width - gap).max(0.0);
193 let start_node = self.start.as_widget_mut().layout(
194 &mut tree.children[0],
195 renderer,
196 &layout::Limits::new(Size::ZERO, Size::new(start_available, height)),
197 );
198 let start_width = start_node.size().width;
199
200 let vcenter = |node: layout::Node, x: f32| -> layout::Node {
201 let dy = ((height - node.size().height) / 2.0).max(0.0);
202 node.translate(Vector::new(x, dy))
203 };
204
205 let mut child_nodes = Vec::with_capacity(3);
206 child_nodes.push(vcenter(start_node, 0.0));
207 child_nodes.push(vcenter(end_node, width - end_width));
208
209 if let Some(center) = &mut self.center {
210 let slot_start = start_width + gap;
211 let slot_end = (width - end_width - gap).max(slot_start);
212 let slot_width = slot_end - slot_start;
213 let natural_width = center
215 .as_widget_mut()
216 .layout(&mut tree.children[2], renderer, &limits.loose())
217 .size()
218 .width;
219
220 let node = center.as_widget_mut().layout(
221 &mut tree.children[2],
222 renderer,
223 &layout::Limits::new(Size::ZERO, Size::new(slot_width, height)),
224 );
225
226 let ideal_x = (width - natural_width) / 2.0;
227 let max_x = (width - end_width - gap - natural_width).max(slot_start);
228 let center_x = ideal_x.clamp(slot_start, max_x);
229
230 child_nodes.push(vcenter(node, center_x))
231 }
232
233 layout::Node::with_children(Size::new(width, height), child_nodes)
234 }
235
236 fn draw(
237 &self,
238 tree: &tree::Tree,
239 renderer: &mut crate::Renderer,
240 theme: &crate::Theme,
241 style: &iced_core::renderer::Style,
242 layout: iced_core::Layout<'_>,
243 cursor: iced_core::mouse::Cursor,
244 viewport: &iced_core::Rectangle,
245 ) {
246 self.elems()
247 .zip(&tree.children)
248 .zip(layout.children())
249 .for_each(|((e, s), l)| {
250 e.as_widget()
251 .draw(s, renderer, theme, style, l, cursor, viewport);
252 });
253 }
254
255 fn update(
256 &mut self,
257 state: &mut tree::Tree,
258 event: &iced_core::Event,
259 layout: iced_core::Layout<'_>,
260 cursor: iced_core::mouse::Cursor,
261 renderer: &crate::Renderer,
262 clipboard: &mut dyn iced_core::Clipboard,
263 shell: &mut iced_core::Shell<'_, Message>,
264 viewport: &iced_core::Rectangle,
265 ) {
266 self.elems_mut()
267 .zip(&mut state.children)
268 .zip(layout.children())
269 .for_each(|((e, s), l)| {
270 e.as_widget_mut()
271 .update(s, event, l, cursor, renderer, clipboard, shell, viewport);
272 });
273 }
274
275 fn mouse_interaction(
276 &self,
277 state: &tree::Tree,
278 layout: iced_core::Layout<'_>,
279 cursor: iced_core::mouse::Cursor,
280 viewport: &iced_core::Rectangle,
281 renderer: &crate::Renderer,
282 ) -> iced_core::mouse::Interaction {
283 self.elems()
284 .zip(&state.children)
285 .zip(layout.children())
286 .map(|((e, s), l)| {
287 e.as_widget()
288 .mouse_interaction(s, l, cursor, viewport, renderer)
289 })
290 .max()
291 .unwrap_or(iced_core::mouse::Interaction::None)
292 }
293
294 fn operate(
295 &mut self,
296 state: &mut tree::Tree,
297 layout: iced_core::Layout<'_>,
298 renderer: &crate::Renderer,
299 operation: &mut dyn iced_core::widget::Operation<()>,
300 ) {
301 self.elems_mut()
302 .zip(&mut state.children)
303 .zip(layout.children())
304 .for_each(|((e, s), l)| {
305 e.as_widget_mut().operate(s, l, renderer, operation);
306 });
307 }
308
309 fn overlay<'b>(
310 &'b mut self,
311 state: &'b mut tree::Tree,
312 layout: iced_core::Layout<'b>,
313 renderer: &crate::Renderer,
314 viewport: &iced_core::Rectangle,
315 translation: Vector,
316 ) -> Option<iced_core::overlay::Element<'b, Message, crate::Theme, crate::Renderer>> {
317 self.elems_mut()
318 .zip(&mut state.children)
319 .zip(layout.children())
320 .find_map(|((e, s), l)| {
321 e.as_widget_mut()
322 .overlay(s, l, renderer, viewport, translation)
323 })
324 }
325
326 fn drag_destinations(
327 &self,
328 state: &tree::Tree,
329 layout: iced_core::Layout<'_>,
330 renderer: &crate::Renderer,
331 dnd_rectangles: &mut iced_core::clipboard::DndDestinationRectangles,
332 ) {
333 self.elems()
334 .zip(&state.children)
335 .zip(layout.children())
336 .for_each(|((e, s), l)| {
337 e.as_widget()
338 .drag_destinations(s, l, renderer, dnd_rectangles);
339 });
340 }
341
342 #[cfg(feature = "a11y")]
343 fn a11y_nodes(
345 &self,
346 layout: iced_core::Layout<'_>,
347 state: &tree::Tree,
348 p: iced::mouse::Cursor,
349 ) -> iced_accessibility::A11yTree {
350 iced_accessibility::A11yTree::join(
351 self.elems()
352 .zip(&state.children)
353 .zip(layout.children())
354 .map(|((e, s), l)| e.as_widget().a11y_nodes(l, s, p)),
355 )
356 }
357}
358
359impl<'a, Message: Clone + 'static> From<HeaderBarWidget<'a, Message>> for Element<'a, Message> {
360 fn from(w: HeaderBarWidget<'a, Message>) -> Self {
361 Element::new(w)
362 }
363}
364
365impl<'a, Message: Clone + 'static> HeaderBar<'a, Message> {
366 pub fn view(mut self) -> Element<'a, Message> {
368 let Spacing {
369 space_xxxs,
370 space_xxs,
371 ..
372 } = theme::spacing();
373
374 let start = std::mem::take(&mut self.start);
376 let center = std::mem::take(&mut self.center);
377 let mut end = std::mem::take(&mut self.end);
378
379 end.push(self.window_controls(space_xxs));
381
382 let padding = if self.is_ssd {
383 [2, 8, 2, 8]
384 } else {
385 match (
386 self.density.unwrap_or_else(crate::config::header_size),
387 self.maximized, ) {
389 (Density::Compact, true) => [4, 8, 4, 8],
390 (Density::Compact, false) => [3, 7, 4, 7],
391 (_, true) => [8, 8, 8, 8],
392 (_, false) => [7, 7, 8, 7],
393 }
394 };
395
396 let start = widget::row::with_children(start)
397 .spacing(space_xxxs)
398 .align_y(iced::Alignment::Center)
399 .into();
400 let center = if !center.is_empty() {
401 Some(
402 widget::row::with_children(center)
403 .spacing(space_xxxs)
404 .align_y(iced::Alignment::Center)
405 .into(),
406 )
407 } else if !self.title.is_empty() {
408 Some(
409 widget::text::heading(self.title)
410 .wrapping(text::Wrapping::None)
411 .ellipsize(text::Ellipsize::End(text::EllipsizeHeightLimit::Lines(1)))
412 .into(),
413 )
414 } else {
415 None
416 };
417 let end = widget::row::with_children(end)
418 .spacing(space_xxs)
419 .align_y(iced::Alignment::Center)
420 .into();
421
422 let mut widget = HeaderBarWidget::new(start, center, end)
423 .apply(widget::container)
424 .class(theme::Container::HeaderBar {
425 focused: self.focused,
426 sharp_corners: self.sharp_corners,
427 transparent: self.transparent,
428 })
429 .height(Length::Fixed(32.0 + padding[0] as f32 + padding[2] as f32))
430 .padding(padding)
431 .apply(widget::mouse_area);
432
433 if let Some(message) = self.on_drag {
434 widget = widget.on_drag(message);
435 }
436 if let Some(message) = self.on_maximize {
437 widget = widget.on_release(message);
438 }
439 if let Some(message) = self.on_double_click {
440 widget = widget.on_double_press(message);
441 }
442 if let Some(message) = self.on_right_click {
443 widget = widget.on_right_press(message);
444 }
445
446 widget.into()
447 }
448
449 fn window_controls(&mut self, spacing: u16) -> Element<'a, Message> {
451 macro_rules! icon {
452 ($name:expr, $size:expr, $on_press:expr) => {{
453 widget::icon::from_name($name)
454 .apply(widget::button::icon)
455 .padding(8)
456 .class(theme::Button::HeaderBar)
457 .selected(self.focused)
458 .icon_size($size)
459 .on_press($on_press)
460 }};
461 }
462
463 widget::row::with_capacity(3)
464 .push_maybe(
465 self.on_minimize
466 .take()
467 .map(|m| icon!("window-minimize-symbolic", 16, m)),
468 )
469 .push_maybe(self.on_maximize.take().map(|m| {
470 if self.maximized {
471 icon!("window-restore-symbolic", 16, m)
472 } else {
473 icon!("window-maximize-symbolic", 16, m)
474 }
475 }))
476 .push_maybe(
477 self.on_close
478 .take()
479 .map(|m| icon!("window-close-symbolic", 16, m)),
480 )
481 .spacing(spacing)
482 .align_y(iced::Alignment::Center)
483 .into()
484 }
485}
486
487impl<'a, Message: Clone + 'static> From<HeaderBar<'a, Message>> for Element<'a, Message> {
488 fn from(headerbar: HeaderBar<'a, Message>) -> Self {
489 headerbar.view()
490 }
491}