import QtQuick import QtQuick.Controls import qs.Common import qs.Services import qs.Widgets Item { id: themeColorsTab property var cachedFontFamilies: [] property var cachedMonoFamilies: [] property bool fontsEnumerated: false function enumerateFonts() { var fonts = ["Default"]; var availableFonts = Qt.fontFamilies(); var rootFamilies = []; var seenFamilies = new Set(); for (var i = 0; i < availableFonts.length; i++) { var fontName = availableFonts[i]; if (fontName.startsWith(".")) continue; if (fontName === SettingsData.defaultFontFamily) continue; var rootName = fontName.replace(/ (Thin|Extra Light|Light|Regular|Medium|Semi Bold|Demi Bold|Bold|Extra Bold|Black|Heavy)$/i, "").replace(/ (Italic|Oblique|Condensed|Extended|Narrow|Wide)$/i, "").replace(/ (UI|Display|Text|Mono|Sans|Serif)$/i, function(match, suffix) { return match; }).trim(); if (!seenFamilies.has(rootName) && rootName !== "") { seenFamilies.add(rootName); rootFamilies.push(rootName); } } cachedFontFamilies = fonts.concat(rootFamilies.sort()); var monoFonts = ["Default"]; var monoFamilies = []; var seenMonoFamilies = new Set(); for (var j = 0; j < availableFonts.length; j++) { var fontName2 = availableFonts[j]; if (fontName2.startsWith(".")) continue; if (fontName2 === SettingsData.defaultMonoFontFamily) continue; var lowerName = fontName2.toLowerCase(); if (lowerName.includes("mono") || lowerName.includes("code") || lowerName.includes("console") || lowerName.includes("terminal") || lowerName.includes("courier") || lowerName.includes("dejavu sans mono") || lowerName.includes("jetbrains") || lowerName.includes("fira") || lowerName.includes("hack") || lowerName.includes("source code") || lowerName.includes("ubuntu mono") || lowerName.includes("cascadia")) { var rootName2 = fontName2.replace(/ (Thin|Extra Light|Light|Regular|Medium|Semi Bold|Demi Bold|Bold|Extra Bold|Black|Heavy)$/i, "").replace(/ (Italic|Oblique|Condensed|Extended|Narrow|Wide)$/i, "").trim(); if (!seenMonoFamilies.has(rootName2) && rootName2 !== "") { seenMonoFamilies.add(rootName2); monoFamilies.push(rootName2); } } } cachedMonoFamilies = monoFonts.concat(monoFamilies.sort()); } Component.onCompleted: { if (!fontsEnumerated) { enumerateFonts(); fontsEnumerated = true; } } DankFlickable { anchors.fill: parent anchors.topMargin: Theme.spacingL clip: true contentHeight: mainColumn.height contentWidth: width Column { id: mainColumn width: parent.width spacing: Theme.spacingXL // Theme Color StyledRect { width: parent.width height: themeSection.implicitHeight + Theme.spacingL * 2 radius: Theme.cornerRadius color: Qt.rgba(Theme.surfaceVariant.r, Theme.surfaceVariant.g, Theme.surfaceVariant.b, 0.3) border.color: Qt.rgba(Theme.outline.r, Theme.outline.g, Theme.outline.b, 0.2) border.width: 1 Column { id: themeSection anchors.fill: parent anchors.margins: Theme.spacingL spacing: Theme.spacingM Row { width: parent.width spacing: Theme.spacingM DankIcon { name: "palette" size: Theme.iconSize color: Theme.primary anchors.verticalCenter: parent.verticalCenter } StyledText { text: "Theme Color" font.pixelSize: Theme.fontSizeLarge font.weight: Font.Medium color: Theme.surfaceText anchors.verticalCenter: parent.verticalCenter } } Column { width: parent.width spacing: Theme.spacingS StyledText { text: "Current Theme: " + (Theme.isDynamicTheme ? "Auto" : (Theme.currentThemeIndex < Theme.themes.length ? Theme.themes[Theme.currentThemeIndex].name : "Blue")) font.pixelSize: Theme.fontSizeMedium color: Theme.surfaceText font.weight: Font.Medium anchors.horizontalCenter: parent.horizontalCenter } StyledText { text: { if (Theme.isDynamicTheme) return "Wallpaper-based dynamic colors"; var descriptions = ["Material blue inspired by modern interfaces", "Deep blue inspired by material 3", "Rich purple tones for BB elegance", "Natural green for productivity", "Energetic orange for creativity", "Bold red for impact", "Cool cyan for tranquility", "Vibrant pink for expression", "Warm amber for comfort", "Soft coral for gentle warmth"]; return descriptions[Theme.currentThemeIndex] || "Select a theme"; } font.pixelSize: Theme.fontSizeSmall color: Theme.surfaceVariantText anchors.horizontalCenter: parent.horizontalCenter wrapMode: Text.WordWrap width: Math.min(parent.width, 400) horizontalAlignment: Text.AlignHCenter } } Column { spacing: Theme.spacingS anchors.horizontalCenter: parent.horizontalCenter Row { spacing: Theme.spacingM anchors.horizontalCenter: parent.horizontalCenter Repeater { model: 5 Rectangle { width: 32 height: 32 radius: 16 color: Theme.themes[index].primary border.color: Theme.outline border.width: (Theme.currentThemeIndex === index && !Theme.isDynamicTheme) ? 2 : 1 scale: (Theme.currentThemeIndex === index && !Theme.isDynamicTheme) ? 1.1 : 1 Rectangle { width: nameText.contentWidth + Theme.spacingS * 2 height: nameText.contentHeight + Theme.spacingXS * 2 color: Theme.surfaceContainer border.color: Theme.outline border.width: 1 radius: Theme.cornerRadius anchors.bottom: parent.top anchors.bottomMargin: Theme.spacingXS anchors.horizontalCenter: parent.horizontalCenter visible: mouseArea.containsMouse StyledText { id: nameText text: Theme.themes[index].name font.pixelSize: Theme.fontSizeSmall color: Theme.surfaceText anchors.centerIn: parent } } MouseArea { id: mouseArea anchors.fill: parent hoverEnabled: true cursorShape: Qt.PointingHandCursor onClicked: { Theme.switchTheme(index, false); } } Behavior on scale { NumberAnimation { duration: Theme.shortDuration easing.type: Theme.emphasizedEasing } } Behavior on border.width { NumberAnimation { duration: Theme.shortDuration easing.type: Theme.emphasizedEasing } } } } } Row { spacing: Theme.spacingM anchors.horizontalCenter: parent.horizontalCenter Repeater { model: 5 Rectangle { property int themeIndex: index + 5 width: 32 height: 32 radius: 16 color: themeIndex < Theme.themes.length ? Theme.themes[themeIndex].primary : "transparent" border.color: Theme.outline border.width: Theme.currentThemeIndex === themeIndex ? 2 : 1 visible: themeIndex < Theme.themes.length scale: Theme.currentThemeIndex === themeIndex ? 1.1 : 1 Rectangle { width: nameText2.contentWidth + Theme.spacingS * 2 height: nameText2.contentHeight + Theme.spacingXS * 2 color: Theme.surfaceContainer border.color: Theme.outline border.width: 1 radius: Theme.cornerRadius anchors.bottom: parent.top anchors.bottomMargin: Theme.spacingXS anchors.horizontalCenter: parent.horizontalCenter visible: mouseArea2.containsMouse && themeIndex < Theme.themes.length StyledText { id: nameText2 text: themeIndex < Theme.themes.length ? Theme.themes[themeIndex].name : "" font.pixelSize: Theme.fontSizeSmall color: Theme.surfaceText anchors.centerIn: parent } } MouseArea { id: mouseArea2 anchors.fill: parent hoverEnabled: true cursorShape: Qt.PointingHandCursor onClicked: { if (themeIndex < Theme.themes.length) Theme.switchTheme(themeIndex); } } Behavior on scale { NumberAnimation { duration: Theme.shortDuration easing.type: Theme.emphasizedEasing } } Behavior on border.width { NumberAnimation { duration: Theme.shortDuration easing.type: Theme.emphasizedEasing } } } } } Item { width: 1 height: Theme.spacingM } Rectangle { width: 120 height: 40 radius: 20 anchors.horizontalCenter: parent.horizontalCenter color: { if (ToastService.wallpaperErrorStatus === "error" || ToastService.wallpaperErrorStatus === "matugen_missing") return Qt.rgba(Theme.error.r, Theme.error.g, Theme.error.b, 0.12); else return Qt.rgba(Theme.surfaceVariant.r, Theme.surfaceVariant.g, Theme.surfaceVariant.b, 0.3); } border.color: { if (ToastService.wallpaperErrorStatus === "error" || ToastService.wallpaperErrorStatus === "matugen_missing") return Qt.rgba(Theme.error.r, Theme.error.g, Theme.error.b, 0.5); else if (Theme.isDynamicTheme) return Theme.primary; else return Theme.outline; } border.width: Theme.isDynamicTheme ? 2 : 1 scale: Theme.isDynamicTheme ? 1.1 : (autoMouseArea.containsMouse ? 1.02 : 1) Row { anchors.centerIn: parent spacing: Theme.spacingS DankIcon { name: { if (ToastService.wallpaperErrorStatus === "error" || ToastService.wallpaperErrorStatus === "matugen_missing") return "error"; else return "palette"; } size: 16 color: { if (ToastService.wallpaperErrorStatus === "error" || ToastService.wallpaperErrorStatus === "matugen_missing") return Theme.error; else return Theme.surfaceText; } anchors.verticalCenter: parent.verticalCenter } StyledText { text: { if (ToastService.wallpaperErrorStatus === "error") return "Error"; else if (ToastService.wallpaperErrorStatus === "matugen_missing") return "No matugen"; else return "Auto"; } font.pixelSize: Theme.fontSizeMedium color: { if (ToastService.wallpaperErrorStatus === "error" || ToastService.wallpaperErrorStatus === "matugen_missing") return Theme.error; else return Theme.surfaceText; } font.weight: Font.Medium anchors.verticalCenter: parent.verticalCenter } } MouseArea { id: autoMouseArea anchors.fill: parent hoverEnabled: true cursorShape: Qt.PointingHandCursor onClicked: { if (ToastService.wallpaperErrorStatus === "matugen_missing") ToastService.showError("matugen not found - install matugen package for dynamic theming"); else if (ToastService.wallpaperErrorStatus === "error") ToastService.showError("Wallpaper processing failed - check wallpaper path"); else Theme.switchTheme(10, true); } } Rectangle { width: autoTooltipText.contentWidth + Theme.spacingM * 2 height: autoTooltipText.contentHeight + Theme.spacingS * 2 color: Theme.surfaceContainer border.color: Theme.outline border.width: 1 radius: Theme.cornerRadius anchors.bottom: parent.top anchors.bottomMargin: Theme.spacingS anchors.horizontalCenter: parent.horizontalCenter visible: autoMouseArea.containsMouse && (!Theme.isDynamicTheme || ToastService.wallpaperErrorStatus === "error" || ToastService.wallpaperErrorStatus === "matugen_missing") StyledText { id: autoTooltipText text: { if (ToastService.wallpaperErrorStatus === "matugen_missing") return "Install matugen package for dynamic themes"; else return "Dynamic wallpaper-based colors"; } font.pixelSize: Theme.fontSizeSmall color: (ToastService.wallpaperErrorStatus === "error" || ToastService.wallpaperErrorStatus === "matugen_missing") ? Theme.error : Theme.surfaceText anchors.centerIn: parent wrapMode: Text.WordWrap width: Math.min(implicitWidth, 250) horizontalAlignment: Text.AlignHCenter } } Behavior on scale { NumberAnimation { duration: Theme.shortDuration easing.type: Theme.emphasizedEasing } } Behavior on color { ColorAnimation { duration: Theme.mediumDuration easing.type: Theme.standardEasing } } Behavior on border.color { ColorAnimation { duration: Theme.mediumDuration easing.type: Theme.standardEasing } } } } } } // Transparency Settings StyledRect { width: parent.width height: transparencySection.implicitHeight + Theme.spacingL * 2 radius: Theme.cornerRadius color: Qt.rgba(Theme.surfaceVariant.r, Theme.surfaceVariant.g, Theme.surfaceVariant.b, 0.3) border.color: Qt.rgba(Theme.outline.r, Theme.outline.g, Theme.outline.b, 0.2) border.width: 1 Column { id: transparencySection anchors.fill: parent anchors.margins: Theme.spacingL spacing: Theme.spacingM Row { width: parent.width spacing: Theme.spacingM DankIcon { name: "opacity" size: Theme.iconSize color: Theme.primary anchors.verticalCenter: parent.verticalCenter } StyledText { text: "Transparency Settings" font.pixelSize: Theme.fontSizeLarge font.weight: Font.Medium color: Theme.surfaceText anchors.verticalCenter: parent.verticalCenter } } Column { width: parent.width spacing: Theme.spacingS StyledText { text: "Top Bar Transparency" font.pixelSize: Theme.fontSizeSmall color: Theme.surfaceText font.weight: Font.Medium } DankSlider { width: parent.width height: 24 value: Math.round(SettingsData.topBarTransparency * 100) minimum: 0 maximum: 100 unit: "" showValue: true onSliderValueChanged: (newValue) => { SettingsData.setTopBarTransparency(newValue / 100); } } } Column { width: parent.width spacing: Theme.spacingS StyledText { text: "Top Bar Widget Transparency" font.pixelSize: Theme.fontSizeSmall color: Theme.surfaceText font.weight: Font.Medium } DankSlider { width: parent.width height: 24 value: Math.round(SettingsData.topBarWidgetTransparency * 100) minimum: 0 maximum: 100 unit: "" showValue: true onSliderValueChanged: (newValue) => { SettingsData.setTopBarWidgetTransparency(newValue / 100); } } } Column { width: parent.width spacing: Theme.spacingS StyledText { text: "Popup Transparency" font.pixelSize: Theme.fontSizeSmall color: Theme.surfaceText font.weight: Font.Medium } DankSlider { width: parent.width height: 24 value: Math.round(SettingsData.popupTransparency * 100) minimum: 0 maximum: 100 unit: "" showValue: true onSliderValueChanged: (newValue) => { SettingsData.setPopupTransparency(newValue / 100); } } } } } // Display Settings StyledRect { width: parent.width height: displaySection.implicitHeight + Theme.spacingL * 2 radius: Theme.cornerRadius color: Qt.rgba(Theme.surfaceVariant.r, Theme.surfaceVariant.g, Theme.surfaceVariant.b, 0.3) border.color: Qt.rgba(Theme.outline.r, Theme.outline.g, Theme.outline.b, 0.2) border.width: 1 Column { id: displaySection anchors.fill: parent anchors.margins: Theme.spacingL spacing: Theme.spacingM Row { width: parent.width spacing: Theme.spacingM DankIcon { name: "monitor" size: Theme.iconSize color: Theme.primary anchors.verticalCenter: parent.verticalCenter } StyledText { text: "Display Settings" font.pixelSize: Theme.fontSizeLarge font.weight: Font.Medium color: Theme.surfaceText anchors.verticalCenter: parent.verticalCenter } } DankToggle { id: nightModeToggle width: parent.width text: "Night Mode" description: "Apply warm color temperature to reduce eye strain" checked: BrightnessService.nightModeActive onToggled: (checked) => { if (checked !== BrightnessService.nightModeActive) { if (checked) BrightnessService.enableNightMode(); else BrightnessService.disableNightMode(); } } Connections { function onNightModeActiveChanged() { nightModeToggle.checked = BrightnessService.nightModeActive; } target: BrightnessService } } DankDropdown { width: parent.width text: "Night Mode Temperature" description: BrightnessService.nightModeActive ? "Disable night mode to adjust" : "Set temperature for night mode" enabled: !BrightnessService.nightModeActive opacity: !BrightnessService.nightModeActive ? 1 : 0.6 currentValue: SessionData.nightModeTemperature + "K" options: { var temps = []; for (var i = 2500; i <= 6000; i += 500) { temps.push(i + "K"); } return temps; } onValueChanged: (value) => { var temp = parseInt(value.replace("K", "")); SessionData.setNightModeTemperature(temp); } } DankToggle { width: parent.width text: "Light Mode" description: "Use light theme instead of dark theme" checked: SessionData.isLightMode onToggled: (checked) => { SessionData.setLightMode(checked); Theme.isLightMode = checked; PortalService.setLightMode(checked); } } } } // Font Settings StyledRect { width: parent.width height: fontSection.implicitHeight + Theme.spacingL * 2 radius: Theme.cornerRadius color: Qt.rgba(Theme.surfaceVariant.r, Theme.surfaceVariant.g, Theme.surfaceVariant.b, 0.3) border.color: Qt.rgba(Theme.outline.r, Theme.outline.g, Theme.outline.b, 0.2) border.width: 1 Column { id: fontSection anchors.fill: parent anchors.margins: Theme.spacingL spacing: Theme.spacingM Row { width: parent.width spacing: Theme.spacingM DankIcon { name: "font_download" size: Theme.iconSize color: Theme.primary anchors.verticalCenter: parent.verticalCenter } StyledText { text: "Font Settings" font.pixelSize: Theme.fontSizeLarge font.weight: Font.Medium color: Theme.surfaceText anchors.verticalCenter: parent.verticalCenter } } DankDropdown { width: parent.width text: "Font Family" description: "Select system font family" currentValue: { if (SettingsData.fontFamily === SettingsData.defaultFontFamily) return "Default"; else return SettingsData.fontFamily || "Default"; } enableFuzzySearch: true popupWidthOffset: 100 maxPopupHeight: 400 options: cachedFontFamilies onValueChanged: (value) => { if (value.startsWith("Default")) SettingsData.setFontFamily(SettingsData.defaultFontFamily); else SettingsData.setFontFamily(value); } } DankDropdown { width: parent.width text: "Font Weight" description: "Select font weight" currentValue: { switch (SettingsData.fontWeight) { case Font.Thin: return "Thin"; case Font.ExtraLight: return "Extra Light"; case Font.Light: return "Light"; case Font.Normal: return "Regular"; case Font.Medium: return "Medium"; case Font.DemiBold: return "Demi Bold"; case Font.Bold: return "Bold"; case Font.ExtraBold: return "Extra Bold"; case Font.Black: return "Black"; default: return "Regular"; } } options: ["Thin", "Extra Light", "Light", "Regular", "Medium", "Demi Bold", "Bold", "Extra Bold", "Black"] onValueChanged: (value) => { var weight; switch (value) { case "Thin": weight = Font.Thin; break; case "Extra Light": weight = Font.ExtraLight; break; case "Light": weight = Font.Light; break; case "Regular": weight = Font.Normal; break; case "Medium": weight = Font.Medium; break; case "Demi Bold": weight = Font.DemiBold; break; case "Bold": weight = Font.Bold; break; case "Extra Bold": weight = Font.ExtraBold; break; case "Black": weight = Font.Black; break; default: weight = Font.Normal; break; } SettingsData.setFontWeight(weight); } } DankDropdown { width: parent.width text: "Monospace Font" description: "Select monospace font for process list and technical displays" currentValue: { if (SettingsData.monoFontFamily === SettingsData.defaultMonoFontFamily) return "Default"; return SettingsData.monoFontFamily || "Default"; } enableFuzzySearch: true popupWidthOffset: 100 maxPopupHeight: 400 options: cachedMonoFamilies onValueChanged: (value) => { if (value === "Default") SettingsData.setMonoFontFamily(SettingsData.defaultMonoFontFamily); else SettingsData.setMonoFontFamily(value); } } } } } } }