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