// Copyright © SixtyFPS GmbH // SPDX-License-Identifier: MIT import { StyleMetrics } from "std-widgets.slint"; component SideBarItem inherits Rectangle { in property selected; in property has-focus; in-out property text<=> label.text; callback clicked<=>touch.clicked; min-height: self.visible ? l.preferred-height : 0px; states [ pressed when touch.pressed: { state.opacity: 0.8; } hover when touch.has-hover: { state.opacity: 0.6; } selected when root.selected: { state.opacity: 1; } focused when root.has-focus: { state.opacity: 0.8; } ]state := Rectangle { opacity: 0; border-width: 2px; border-radius: 10px; border-color: StyleMetrics.default-text-color; background: StyleMetrics.window-background; animate opacity { duration: 150ms; } animate border-width { duration: 150ms; } } l := HorizontalLayout { y: (parent.height - self.height) / 2; padding: StyleMetrics.layout-padding; spacing: 0px; label := Text { color: StyleMetrics.default-text-color; vertical-alignment: center; } } touch := TouchArea { width: 100%; height: 100%; } } export component SideBar inherits Rectangle { in property <[string]> model: []; in property <[bool]> available: []; in property title<=> label.text; out property current-item: 0; out property current-focused: fs.has-focus ? fs.focused-tab : -1; // The currently focused tab width: 180px; forward-focus: fs; accessible-role: tab; accessible-delegate-focus: root.current-focused >= 0 ? root.current-focused : root.current-item; Rectangle { background: StyleMetrics.window-background.darker(0.2); fs := FocusScope { key-pressed(event) => { if (event.text == "\n") { root.current-item = root.current-focused; return accept; } if (event.text == Key.UpArrow) { self.focused-tab = Math.max(self.focused-tab - 1, 0); return accept; } if (event.text == Key.DownArrow) { self.focused-tab = Math.min(self.focused-tab + 1, root.model.length - 1); return accept; } return reject; } key-released(event) => { if (event.text == " ") { root.current-item = root.current-focused; return accept; } return reject; } property focused-tab: 0; x: 0; width: 0; // Do not react on clicks } } VerticalLayout { padding: StyleMetrics.layout-padding; spacing: StyleMetrics.layout-spacing; alignment: start; label := Text { font-size: 16px; horizontal-alignment: center; } navigation := VerticalLayout { spacing: 10px; alignment: start; vertical-stretch: 0; for item[index] in root.model: SideBarItem { visible: root.available[index]; clicked => { root.current-item = index; } has-focus: index == root.current-focused; text: item; selected: index == root.current-item; } } VerticalLayout { bottom := VerticalLayout { padding-left: StyleMetrics.layout-padding; padding-right: StyleMetrics.layout-padding; @children } } } }