mirror of
https://github.com/noctalia-dev/noctalia-shell.git
synced 2026-05-11 17:08:27 +08:00
618 lines
22 KiB
QML
618 lines
22 KiB
QML
import QtQuick
|
|
import QtQuick.Controls
|
|
import QtQuick.Layouts
|
|
import Quickshell
|
|
import qs.Commons
|
|
import qs.Services
|
|
import qs.Widgets
|
|
import qs.Modules.MainScreen
|
|
|
|
SmartPanel {
|
|
id: root
|
|
|
|
// Inputs
|
|
property QsMenuHandle menu
|
|
property var trayItem: null
|
|
property string widgetSection: ""
|
|
property int widgetIndex: -1
|
|
|
|
// Internal
|
|
readonly property int menuWidth: 280
|
|
preferredWidth: menuWidth
|
|
// Height is content-driven via panelContent
|
|
|
|
// Open positioned relative to button
|
|
function openAt(buttonItem) {
|
|
open(buttonItem)
|
|
}
|
|
|
|
panelContent: Item {
|
|
id: content
|
|
|
|
// Track currently open submenu
|
|
property var activeSubMenu: null
|
|
property var activeSubMenuEntry: null
|
|
// Only show submenu in place (replace main menu)
|
|
property bool inPlaceSubmenu: true
|
|
|
|
// Let Panel size to our content
|
|
readonly property real contentPreferredWidth: root.menuWidth
|
|
readonly property real contentPreferredHeight: {
|
|
// If showing submenu in-place, size to the submenu's flickable content height
|
|
if (activeSubMenu && inPlaceSubmenu) {
|
|
const subFlickable = inPlaceSubMenuLoader.item?.children[1] // Flickable is the second child (after MouseArea)
|
|
const subHeight = subFlickable?.contentHeight || 0
|
|
return Math.min(root.screen ? root.screen.height * 0.9 : Screen.height * 0.9, subHeight + (Style.marginS * 2))
|
|
}
|
|
|
|
const mainHeight = mainFlickable.contentHeight
|
|
return Math.min(root.screen ? root.screen.height * 0.9 : Screen.height * 0.9, mainHeight + (Style.marginS * 2))
|
|
}
|
|
|
|
// Expose mask region for click-through
|
|
property alias maskRegion: background
|
|
|
|
Rectangle {
|
|
id: background
|
|
x: 0
|
|
y: 0
|
|
width: content.contentPreferredWidth
|
|
height: content.contentPreferredHeight
|
|
color: Color.mSurface
|
|
radius: Style.radiusM
|
|
}
|
|
|
|
QsMenuOpener {
|
|
id: opener
|
|
menu: root.menu
|
|
}
|
|
|
|
Component {
|
|
id: subMenuComponent
|
|
|
|
Item {
|
|
id: subMenuContainer
|
|
|
|
// MouseArea to track hover for submenu (covers entire submenu area)
|
|
MouseArea {
|
|
id: subMenuMouseArea
|
|
anchors.fill: parent
|
|
hoverEnabled: true
|
|
enabled: content.activeSubMenu !== null && !content.inPlaceSubmenu
|
|
visible: !content.inPlaceSubmenu
|
|
z: 1
|
|
acceptedButtons: Qt.NoButton // Don't intercept clicks, just track hover
|
|
onExited: {
|
|
if (content.inPlaceSubmenu)
|
|
return
|
|
Qt.callLater(() => {
|
|
if (!subMenuMouseArea.containsMouse && content.activeSubMenuEntry) {
|
|
// Find the mouseArea in the entry (it's in the second Rectangle's MouseArea)
|
|
let entryMouseArea = null
|
|
for (var i = 0; i < content.activeSubMenuEntry.children.length; i++) {
|
|
const child = content.activeSubMenuEntry.children[i]
|
|
if (child && child.children && child.children.length > 0) {
|
|
for (var j = 0; j < child.children.length; j++) {
|
|
const grandchild = child.children[j]
|
|
if (grandchild && grandchild.hoverEnabled !== undefined) {
|
|
entryMouseArea = grandchild
|
|
break
|
|
}
|
|
}
|
|
}
|
|
if (entryMouseArea)
|
|
break
|
|
}
|
|
if (!entryMouseArea || !entryMouseArea.containsMouse) {
|
|
content.activeSubMenu = null
|
|
content.activeSubMenuEntry = null
|
|
}
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
Flickable {
|
|
id: subMenuFlickable
|
|
anchors.fill: parent
|
|
contentHeight: subMenuColumnLayout.implicitHeight
|
|
interactive: true
|
|
z: 0
|
|
|
|
ColumnLayout {
|
|
id: subMenuColumnLayout
|
|
width: subMenuFlickable.width
|
|
spacing: 0
|
|
|
|
QsMenuOpener {
|
|
id: subMenuOpener
|
|
menu: content.activeSubMenu
|
|
}
|
|
|
|
// Back button (only shown when submenu is in-place)
|
|
Rectangle {
|
|
id: backEntry
|
|
visible: content.inPlaceSubmenu
|
|
Layout.preferredWidth: parent.width
|
|
Layout.preferredHeight: visible ? 28 : 0
|
|
color: Color.transparent
|
|
|
|
Rectangle {
|
|
anchors.top: parent.top
|
|
anchors.bottom: parent.bottom
|
|
anchors.topMargin: 0
|
|
anchors.bottomMargin: 0
|
|
anchors.horizontalCenter: parent.horizontalCenter
|
|
width: parent.width - (Style.marginM * 2)
|
|
color: backMouseArea.containsMouse ? Color.mHover : Color.transparent
|
|
radius: Style.radiusS
|
|
|
|
RowLayout {
|
|
anchors.fill: parent
|
|
anchors.leftMargin: Style.marginM
|
|
anchors.rightMargin: Style.marginM
|
|
spacing: Style.marginS
|
|
|
|
NIcon {
|
|
icon: "arrow-left"
|
|
pointSize: Style.fontSizeS
|
|
applyUiScale: false
|
|
verticalAlignment: Text.AlignVCenter
|
|
color: backMouseArea.containsMouse ? Color.mOnHover : Color.mOnSurface
|
|
}
|
|
|
|
NText {
|
|
Layout.fillWidth: true
|
|
color: backMouseArea.containsMouse ? Color.mOnHover : Color.mOnSurface
|
|
text: I18n.tr("settings.bar.tray.back")
|
|
pointSize: Style.fontSizeS
|
|
verticalAlignment: Text.AlignVCenter
|
|
elide: Text.ElideRight
|
|
}
|
|
}
|
|
|
|
MouseArea {
|
|
id: backMouseArea
|
|
anchors.fill: parent
|
|
hoverEnabled: true
|
|
|
|
onClicked: {
|
|
content.activeSubMenu = null
|
|
content.activeSubMenuEntry = null
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Rectangle {
|
|
visible: content.inPlaceSubmenu
|
|
Layout.preferredWidth: parent.width
|
|
Layout.preferredHeight: visible ? 8 : 0
|
|
color: Color.transparent
|
|
|
|
NDivider {
|
|
anchors.centerIn: parent
|
|
width: parent.width - (Style.marginM * 2)
|
|
visible: parent.visible
|
|
}
|
|
}
|
|
|
|
Repeater {
|
|
model: subMenuOpener.children ? [...subMenuOpener.children.values] : []
|
|
|
|
delegate: Rectangle {
|
|
id: subEntry
|
|
required property var modelData
|
|
|
|
Layout.preferredWidth: parent.width
|
|
Layout.preferredHeight: {
|
|
if (modelData?.isSeparator) {
|
|
return 8
|
|
} else {
|
|
return 28
|
|
}
|
|
}
|
|
|
|
color: Color.transparent
|
|
|
|
NDivider {
|
|
anchors.centerIn: parent
|
|
width: parent.width - (Style.marginM * 2)
|
|
visible: modelData?.isSeparator ?? false
|
|
}
|
|
|
|
Rectangle {
|
|
anchors.top: parent.top
|
|
anchors.bottom: parent.bottom
|
|
anchors.topMargin: 0
|
|
anchors.bottomMargin: 0
|
|
anchors.horizontalCenter: parent.horizontalCenter
|
|
width: parent.width - (Style.marginM * 2)
|
|
color: subMouseArea.containsMouse ? Color.mHover : Color.transparent
|
|
radius: Style.radiusS
|
|
visible: !(modelData?.isSeparator ?? false)
|
|
|
|
RowLayout {
|
|
anchors.fill: parent
|
|
anchors.leftMargin: Style.marginM
|
|
anchors.rightMargin: Style.marginM
|
|
spacing: Style.marginS
|
|
|
|
NText {
|
|
Layout.fillWidth: true
|
|
color: (modelData?.enabled ?? true) ? (subMouseArea.containsMouse ? Color.mOnHover : Color.mOnSurface) : Color.mOnSurfaceVariant
|
|
text: modelData?.text !== "" ? modelData?.text.replace(/[\n\r]+/g, ' ') : "..."
|
|
pointSize: Style.fontSizeS
|
|
verticalAlignment: Text.AlignVCenter
|
|
elide: Text.ElideRight
|
|
}
|
|
|
|
Image {
|
|
Layout.preferredWidth: Style.marginL
|
|
Layout.preferredHeight: Style.marginL
|
|
source: modelData?.icon ?? ""
|
|
visible: (modelData?.icon ?? "") !== ""
|
|
fillMode: Image.PreserveAspectFit
|
|
}
|
|
|
|
NIcon {
|
|
icon: modelData?.hasChildren ? "menu" : ""
|
|
pointSize: Style.fontSizeS
|
|
applyUiScale: false
|
|
verticalAlignment: Text.AlignVCenter
|
|
visible: modelData?.hasChildren ?? false
|
|
Layout.rightMargin: Style.marginL
|
|
color: (subMouseArea.containsMouse ? Color.mOnHover : Color.mOnSurface)
|
|
}
|
|
}
|
|
|
|
MouseArea {
|
|
id: subMouseArea
|
|
anchors.fill: parent
|
|
hoverEnabled: true
|
|
enabled: (modelData?.enabled ?? true) && !(modelData?.isSeparator ?? false)
|
|
|
|
onClicked: {
|
|
if (!modelData || modelData.isSeparator)
|
|
return
|
|
|
|
if (modelData.hasChildren) {
|
|
// Toggle nested submenu on click
|
|
if (content.activeSubMenu === modelData) {
|
|
// If already open, close it on second click
|
|
content.activeSubMenu = null
|
|
content.activeSubMenuEntry = null
|
|
return
|
|
}
|
|
|
|
// Open nested submenu in place
|
|
content.activeSubMenu = modelData
|
|
content.activeSubMenuEntry = null // No entry for nested submenus
|
|
content.inPlaceSubmenu = true
|
|
} else {
|
|
modelData.triggered()
|
|
root.close()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Main menu and in-place submenu
|
|
RowLayout {
|
|
id: rowLayout
|
|
anchors.fill: background
|
|
anchors.margins: Style.marginS
|
|
spacing: 0
|
|
|
|
// Submenu replacing main menu (in-place)
|
|
Loader {
|
|
id: inPlaceSubMenuLoader
|
|
Layout.preferredWidth: (content.activeSubMenu && content.inPlaceSubmenu) ? root.menuWidth : 0
|
|
Layout.fillHeight: true
|
|
visible: content.activeSubMenu !== null && content.inPlaceSubmenu
|
|
sourceComponent: subMenuComponent
|
|
}
|
|
|
|
// Main menu
|
|
Flickable {
|
|
id: mainFlickable
|
|
Layout.preferredWidth: root.menuWidth
|
|
Layout.fillHeight: true
|
|
contentHeight: mainColumnLayout.implicitHeight
|
|
interactive: true
|
|
visible: !(content.activeSubMenu !== null && content.inPlaceSubmenu)
|
|
|
|
ColumnLayout {
|
|
id: mainColumnLayout
|
|
width: mainFlickable.width
|
|
spacing: 0
|
|
|
|
Repeater {
|
|
model: opener.children ? [...opener.children.values] : []
|
|
|
|
delegate: Rectangle {
|
|
id: entry
|
|
required property var modelData
|
|
|
|
Layout.preferredWidth: parent.width
|
|
Layout.preferredHeight: {
|
|
if (modelData?.isSeparator) {
|
|
return 8
|
|
} else {
|
|
return 28
|
|
}
|
|
}
|
|
|
|
color: Color.transparent
|
|
|
|
NDivider {
|
|
anchors.centerIn: parent
|
|
width: parent.width - Style.marginL
|
|
visible: modelData?.isSeparator ?? false
|
|
}
|
|
|
|
Rectangle {
|
|
anchors.top: parent.top
|
|
anchors.bottom: parent.bottom
|
|
anchors.topMargin: 0
|
|
anchors.bottomMargin: 0
|
|
anchors.horizontalCenter: parent.horizontalCenter
|
|
width: parent.width - (Style.marginM * 2)
|
|
color: mouseArea.containsMouse ? Color.mHover : Color.transparent
|
|
radius: Style.radiusS
|
|
visible: !(modelData?.isSeparator ?? false)
|
|
|
|
RowLayout {
|
|
anchors.fill: parent
|
|
anchors.leftMargin: Style.marginM
|
|
anchors.rightMargin: Style.marginM
|
|
spacing: Style.marginS
|
|
|
|
NText {
|
|
id: text
|
|
Layout.fillWidth: true
|
|
color: (modelData?.enabled ?? true) ? (mouseArea.containsMouse ? Color.mOnHover : Color.mOnSurface) : Color.mOnSurfaceVariant
|
|
text: modelData?.text !== "" ? modelData?.text.replace(/[\n\r]+/g, ' ') : "..."
|
|
pointSize: Style.fontSizeS
|
|
verticalAlignment: Text.AlignVCenter
|
|
elide: Text.ElideRight
|
|
}
|
|
|
|
Image {
|
|
Layout.preferredWidth: Style.marginL
|
|
Layout.preferredHeight: Style.marginL
|
|
source: modelData?.icon ?? ""
|
|
visible: (modelData?.icon ?? "") !== ""
|
|
fillMode: Image.PreserveAspectFit
|
|
}
|
|
|
|
NIcon {
|
|
icon: modelData?.hasChildren ? "menu" : ""
|
|
pointSize: Style.fontSizeS
|
|
applyUiScale: false
|
|
verticalAlignment: Text.AlignVCenter
|
|
visible: modelData?.hasChildren ?? false
|
|
Layout.rightMargin: Style.marginS
|
|
color: (mouseArea.containsMouse ? Color.mOnHover : Color.mOnSurface)
|
|
}
|
|
}
|
|
|
|
MouseArea {
|
|
id: mouseArea
|
|
anchors.fill: parent
|
|
hoverEnabled: true
|
|
enabled: (modelData?.enabled ?? true) && !(modelData?.isSeparator ?? false)
|
|
|
|
onClicked: {
|
|
if (!modelData || modelData.isSeparator)
|
|
return
|
|
|
|
if (modelData.hasChildren) {
|
|
// Toggle submenu on click
|
|
if (content.activeSubMenuEntry === entry && content.inPlaceSubmenu) {
|
|
// If already open in-place, close it on second click
|
|
content.activeSubMenu = null
|
|
content.activeSubMenuEntry = null
|
|
return
|
|
}
|
|
|
|
// Close any other open submenu
|
|
if (content.activeSubMenuEntry && content.activeSubMenuEntry !== entry) {
|
|
content.activeSubMenu = null
|
|
content.activeSubMenuEntry = null
|
|
}
|
|
|
|
// Open submenu in place (replace main menu)
|
|
content.activeSubMenu = modelData
|
|
content.activeSubMenuEntry = entry
|
|
content.inPlaceSubmenu = true
|
|
} else {
|
|
modelData.triggered()
|
|
root.close()
|
|
}
|
|
}
|
|
|
|
onExited: {
|
|
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Rectangle {
|
|
visible: root.trayItem !== null && root.widgetSection !== "" && root.widgetIndex >= 0
|
|
Layout.preferredWidth: parent.width
|
|
Layout.preferredHeight: visible ? 8 : 0
|
|
color: Color.transparent
|
|
|
|
NDivider {
|
|
anchors.centerIn: parent
|
|
width: parent.width - Style.marginL
|
|
visible: parent.visible
|
|
}
|
|
}
|
|
|
|
Rectangle {
|
|
id: addToFavoriteEntry
|
|
visible: root.trayItem !== null && root.widgetSection !== "" && root.widgetIndex >= 0
|
|
Layout.preferredWidth: parent.width
|
|
Layout.preferredHeight: visible ? 28 : 0
|
|
color: Color.transparent
|
|
|
|
readonly property bool isFavorite: {
|
|
if (!root.trayItem || root.widgetSection === "" || root.widgetIndex < 0)
|
|
return false
|
|
const itemName = root.trayItem.tooltipTitle || root.trayItem.name || root.trayItem.id || ""
|
|
if (!itemName)
|
|
return false
|
|
var widgets = Settings.data.bar.widgets[root.widgetSection]
|
|
if (!widgets || root.widgetIndex >= widgets.length)
|
|
return false
|
|
var widgetSettings = widgets[root.widgetIndex]
|
|
if (!widgetSettings || widgetSettings.id !== "Tray")
|
|
return false
|
|
var favorites = widgetSettings.favorites || []
|
|
for (var i = 0; i < favorites.length; i++) {
|
|
if (favorites[i] === itemName)
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
Rectangle {
|
|
anchors.top: parent.top
|
|
anchors.bottom: parent.bottom
|
|
anchors.topMargin: 0
|
|
anchors.bottomMargin: 0
|
|
anchors.horizontalCenter: parent.horizontalCenter
|
|
width: parent.width - (Style.marginM * 2)
|
|
color: addToFavoriteMouseArea.containsMouse ? Qt.alpha(Color.mPrimary, 0.2) : Qt.alpha(Color.mPrimary, 0.08)
|
|
radius: Style.radiusS
|
|
border.color: Qt.alpha(Color.mPrimary, addToFavoriteMouseArea.containsMouse ? 0.4 : 0.2)
|
|
border.width: Style.borderS
|
|
|
|
RowLayout {
|
|
anchors.fill: parent
|
|
anchors.leftMargin: Style.marginM
|
|
anchors.rightMargin: Style.marginM
|
|
spacing: Style.marginS
|
|
|
|
NIcon {
|
|
icon: addToFavoriteEntry.isFavorite ? "star" : "star-off"
|
|
pointSize: Style.fontSizeS
|
|
applyUiScale: false
|
|
verticalAlignment: Text.AlignVCenter
|
|
color: Color.mPrimary
|
|
}
|
|
|
|
NText {
|
|
Layout.fillWidth: true
|
|
color: Color.mPrimary
|
|
text: addToFavoriteEntry.isFavorite ? I18n.tr("settings.bar.tray.unpin-application") : I18n.tr("settings.bar.tray.pin-application")
|
|
pointSize: Style.fontSizeS
|
|
font.weight: Font.Medium
|
|
verticalAlignment: Text.AlignVCenter
|
|
elide: Text.ElideRight
|
|
}
|
|
}
|
|
|
|
MouseArea {
|
|
id: addToFavoriteMouseArea
|
|
anchors.fill: parent
|
|
hoverEnabled: true
|
|
|
|
onClicked: {
|
|
if (addToFavoriteEntry.isFavorite) {
|
|
root.removeFromFavorites()
|
|
} else {
|
|
root.addToFavorites()
|
|
}
|
|
root.close()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Keys.onEscapePressed: root.close()
|
|
}
|
|
|
|
function addToFavorites() {
|
|
if (!trayItem || widgetSection === "" || widgetIndex < 0) {
|
|
Logger.w("TrayMenu", "Cannot add as favorite: missing tray item or widget info")
|
|
return
|
|
}
|
|
const itemName = trayItem.tooltipTitle || trayItem.name || trayItem.id || ""
|
|
if (!itemName) {
|
|
Logger.w("TrayMenu", "Cannot add as favorite: tray item has no name")
|
|
return
|
|
}
|
|
var widgets = Settings.data.bar.widgets[widgetSection]
|
|
if (!widgets || widgetIndex >= widgets.length) {
|
|
Logger.w("TrayMenu", "Cannot add as favorite: invalid widget index")
|
|
return
|
|
}
|
|
var widgetSettings = widgets[widgetIndex]
|
|
if (!widgetSettings || widgetSettings.id !== "Tray") {
|
|
Logger.w("TrayMenu", "Cannot add as favorite: widget is not a Tray widget")
|
|
return
|
|
}
|
|
var favorites = widgetSettings.favorites || []
|
|
var newFavorites = favorites.slice()
|
|
newFavorites.push(itemName)
|
|
var newSettings = Object.assign({}, widgetSettings)
|
|
newSettings.favorites = newFavorites
|
|
widgets[widgetIndex] = newSettings
|
|
Settings.data.bar.widgets[widgetSection] = widgets
|
|
Settings.saveImmediate()
|
|
if (root.screen) {
|
|
const panel = PanelService.getPanel("trayDropdownPanel", root.screen)
|
|
if (panel)
|
|
panel.close()
|
|
}
|
|
}
|
|
|
|
function removeFromFavorites() {
|
|
if (!trayItem || widgetSection === "" || widgetIndex < 0) {
|
|
Logger.w("TrayMenu", "Cannot remove from favorites: missing tray item or widget info")
|
|
return
|
|
}
|
|
const itemName = trayItem.tooltipTitle || trayItem.name || trayItem.id || ""
|
|
if (!itemName) {
|
|
Logger.w("TrayMenu", "Cannot remove from favorites: tray item has no name")
|
|
return
|
|
}
|
|
var widgets = Settings.data.bar.widgets[widgetSection]
|
|
if (!widgets || widgetIndex >= widgets.length) {
|
|
Logger.w("TrayMenu", "Cannot remove from favorites: invalid widget index")
|
|
return
|
|
}
|
|
var widgetSettings = widgets[widgetIndex]
|
|
if (!widgetSettings || widgetSettings.id !== "Tray") {
|
|
Logger.w("TrayMenu", "Cannot remove from favorites: widget is not a Tray widget")
|
|
return
|
|
}
|
|
var favorites = widgetSettings.favorites || []
|
|
var newFavorites = []
|
|
for (var i = 0; i < favorites.length; i++) {
|
|
if (favorites[i] !== itemName) {
|
|
newFavorites.push(favorites[i])
|
|
}
|
|
}
|
|
var newSettings = Object.assign({}, widgetSettings)
|
|
newSettings.favorites = newFavorites
|
|
widgets[widgetIndex] = newSettings
|
|
Settings.data.bar.widgets[widgetSection] = widgets
|
|
Settings.saveImmediate()
|
|
}
|
|
}
|