407 lines
12 KiB
QML
407 lines
12 KiB
QML
import QtQuick
|
|
import QtQuick.Layouts
|
|
import QtQuick.Controls
|
|
import Quickshell
|
|
import Quickshell.Io
|
|
import qs.Commons
|
|
import qs.Services
|
|
import qs.Widgets
|
|
|
|
ColumnLayout {
|
|
id: root
|
|
|
|
// Time dropdown options (00:00 .. 23:30)
|
|
ListModel {
|
|
id: timeOptions
|
|
}
|
|
Component.onCompleted: {
|
|
for (var h = 0; h < 24; h++) {
|
|
for (var m = 0; m < 60; m += 30) {
|
|
var hh = ("0" + h).slice(-2)
|
|
var mm = ("0" + m).slice(-2)
|
|
var key = hh + ":" + mm
|
|
timeOptions.append({
|
|
"key": key,
|
|
"name": key
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
// Check for wlsunset availability when enabling Night Light
|
|
Process {
|
|
id: wlsunsetCheck
|
|
command: ["which", "wlsunset"]
|
|
running: false
|
|
|
|
onExited: function (exitCode) {
|
|
if (exitCode === 0) {
|
|
Settings.data.nightLight.enabled = true
|
|
NightLightService.apply()
|
|
ToastService.showNotice("Night Light", "Enabled")
|
|
} else {
|
|
Settings.data.nightLight.enabled = false
|
|
ToastService.showWarning("Night Light", "wlsunset not installed")
|
|
}
|
|
}
|
|
|
|
stdout: StdioCollector {}
|
|
stderr: StdioCollector {}
|
|
}
|
|
|
|
spacing: Style.marginL * scaling
|
|
|
|
NHeader {
|
|
label: "Monitor-specific configuration"
|
|
description: "Configure scaling and brightness settings individually for each connected display."
|
|
}
|
|
|
|
ColumnLayout {
|
|
spacing: Style.marginL * scaling
|
|
|
|
Repeater {
|
|
model: Quickshell.screens || []
|
|
delegate: Rectangle {
|
|
Layout.fillWidth: true
|
|
implicitHeight: contentCol.implicitHeight + Style.marginXL * 2 * scaling
|
|
radius: Style.radiusM * scaling
|
|
color: Color.mSurfaceVariant
|
|
border.color: Color.mOutline
|
|
border.width: Math.max(1, Style.borderS * scaling)
|
|
|
|
property real localScaling: ScalingService.getScreenScale(modelData)
|
|
property var brightnessMonitor: BrightnessService.getMonitorForScreen(modelData)
|
|
|
|
Connections {
|
|
target: ScalingService
|
|
function onScaleChanged(screenName, scale) {
|
|
if (screenName === modelData.name) {
|
|
localScaling = scale
|
|
}
|
|
}
|
|
}
|
|
|
|
ColumnLayout {
|
|
id: contentCol
|
|
anchors.fill: parent
|
|
anchors.margins: Style.marginL * scaling
|
|
spacing: Style.marginXXS * scaling
|
|
|
|
NText {
|
|
text: (`${modelData.name}: ${modelData.model}` || "Unknown")
|
|
font.pointSize: Style.fontSizeL * scaling
|
|
font.weight: Style.fontWeightBold
|
|
color: Color.mPrimary
|
|
}
|
|
|
|
NText {
|
|
text: `Resolution: ${modelData.width}x${modelData.height} at (${modelData.x}, ${modelData.y})`
|
|
font.pointSize: Style.fontSizeXS * scaling
|
|
color: Color.mOnSurfaceVariant
|
|
wrapMode: Text.WordWrap
|
|
Layout.fillWidth: true
|
|
}
|
|
|
|
// Scale
|
|
ColumnLayout {
|
|
spacing: Style.marginS * scaling
|
|
Layout.fillWidth: true
|
|
|
|
RowLayout {
|
|
spacing: Style.marginM * scaling
|
|
Layout.fillWidth: true
|
|
|
|
NText {
|
|
text: "Scale"
|
|
Layout.preferredWidth: 80 * scaling
|
|
}
|
|
|
|
NValueSlider {
|
|
id: scaleSlider
|
|
from: 0.7
|
|
to: 1.8
|
|
stepSize: 0.01
|
|
value: localScaling
|
|
onPressedChanged: (pressed, value) => ScalingService.setScreenScale(modelData, value)
|
|
text: `${Math.round(localScaling * 100)}%`
|
|
Layout.fillWidth: true
|
|
}
|
|
|
|
// Reset button container
|
|
Item {
|
|
Layout.preferredWidth: 40 * scaling
|
|
Layout.preferredHeight: 30 * scaling
|
|
|
|
NIconButton {
|
|
icon: "refresh"
|
|
sizeRatio: 0.8
|
|
tooltipText: "Reset scaling"
|
|
onClicked: ScalingService.setScreenScale(modelData, 1.0)
|
|
anchors.right: parent.right
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Brightness
|
|
ColumnLayout {
|
|
spacing: Style.marginL * scaling
|
|
Layout.fillWidth: true
|
|
visible: brightnessMonitor !== undefined && brightnessMonitor !== null
|
|
|
|
RowLayout {
|
|
Layout.fillWidth: true
|
|
spacing: Style.marginM * scaling
|
|
|
|
NText {
|
|
text: "Brightness"
|
|
Layout.preferredWidth: 80 * scaling
|
|
}
|
|
|
|
NValueSlider {
|
|
Layout.fillWidth: true
|
|
from: 0
|
|
to: 1
|
|
value: brightnessMonitor ? brightnessMonitor.brightness : 0.5
|
|
stepSize: 0.01
|
|
onPressedChanged: (pressed, value) => brightnessMonitor.setBrightness(value)
|
|
text: brightnessMonitor ? Math.round(brightnessMonitor.brightness * 100) + "%" : "N/A"
|
|
}
|
|
|
|
// Empty container to match scale row layout
|
|
Item {
|
|
Layout.preferredWidth: 40 * scaling
|
|
Layout.preferredHeight: 30 * scaling
|
|
|
|
// Method text positioned in the button area
|
|
NText {
|
|
text: brightnessMonitor ? brightnessMonitor.method : ""
|
|
font.pointSize: Style.fontSizeXS * scaling
|
|
color: Color.mOnSurfaceVariant
|
|
anchors.right: parent.right
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
horizontalAlignment: Text.AlignRight
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
NDivider {
|
|
Layout.fillWidth: true
|
|
Layout.topMargin: Style.marginXL * scaling
|
|
Layout.bottomMargin: Style.marginXL * scaling
|
|
}
|
|
|
|
// Brightness Section
|
|
ColumnLayout {
|
|
spacing: Style.marginS * scaling
|
|
Layout.fillWidth: true
|
|
|
|
NHeader {
|
|
label: "Brightness"
|
|
description: "Adjust brightness related settings."
|
|
}
|
|
|
|
// Brightness Step Section
|
|
ColumnLayout {
|
|
spacing: Style.marginS * scaling
|
|
Layout.fillWidth: true
|
|
|
|
NSpinBox {
|
|
Layout.fillWidth: true
|
|
label: "Brightness Step Size"
|
|
description: "Adjust the step size for brightness changes (scroll wheel and keyboard shortcuts)."
|
|
minimum: 1
|
|
maximum: 50
|
|
value: Settings.data.brightness.brightnessStep
|
|
stepSize: 1
|
|
suffix: "%"
|
|
onValueChanged: {
|
|
Settings.data.brightness.brightnessStep = value
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
NDivider {
|
|
Layout.fillWidth: true
|
|
Layout.topMargin: Style.marginXL * scaling
|
|
Layout.bottomMargin: Style.marginXL * scaling
|
|
}
|
|
|
|
// Night Light Section
|
|
ColumnLayout {
|
|
spacing: Style.marginXS * scaling
|
|
Layout.fillWidth: true
|
|
|
|
NHeader {
|
|
label: "Night Light"
|
|
description: "Reduce blue light emission to help you sleep better and reduce eye strain."
|
|
}
|
|
}
|
|
|
|
NToggle {
|
|
label: "Enable Night Light"
|
|
description: "Apply a warm color filter to reduce blue light emission."
|
|
checked: Settings.data.nightLight.enabled
|
|
onToggled: checked => {
|
|
if (checked) {
|
|
// Verify wlsunset exists before enabling
|
|
wlsunsetCheck.running = true
|
|
} else {
|
|
Settings.data.nightLight.enabled = false
|
|
Settings.data.nightLight.forced = false
|
|
NightLightService.apply()
|
|
ToastService.showNotice("Night Light", "Disabled")
|
|
}
|
|
}
|
|
}
|
|
|
|
// Temperature
|
|
ColumnLayout {
|
|
spacing: Style.marginXS * scaling
|
|
Layout.alignment: Qt.AlignVCenter
|
|
|
|
NLabel {
|
|
label: "Color temperature"
|
|
description: "Choose two temperatures in Kelvin."
|
|
}
|
|
|
|
RowLayout {
|
|
visible: Settings.data.nightLight.enabled
|
|
spacing: Style.marginM * scaling
|
|
Layout.fillWidth: false
|
|
Layout.fillHeight: true
|
|
Layout.alignment: Qt.AlignVCenter
|
|
|
|
NText {
|
|
text: "Night"
|
|
font.pointSize: Style.fontSizeM * scaling
|
|
color: Color.mOnSurfaceVariant
|
|
Layout.alignment: Qt.AlignVCenter
|
|
}
|
|
|
|
NTextInput {
|
|
text: Settings.data.nightLight.nightTemp
|
|
inputMethodHints: Qt.ImhDigitsOnly
|
|
Layout.alignment: Qt.AlignVCenter
|
|
onEditingFinished: {
|
|
var nightTemp = parseInt(text)
|
|
var dayTemp = parseInt(Settings.data.nightLight.dayTemp)
|
|
if (!isNaN(nightTemp) && !isNaN(dayTemp)) {
|
|
// Clamp value between [1000 .. (dayTemp-500)]
|
|
var clampedValue = Math.min(dayTemp - 500, Math.max(1000, nightTemp))
|
|
text = Settings.data.nightLight.nightTemp = clampedValue.toString()
|
|
}
|
|
}
|
|
}
|
|
|
|
Item {}
|
|
|
|
NText {
|
|
text: "Day"
|
|
font.pointSize: Style.fontSizeM * scaling
|
|
color: Color.mOnSurfaceVariant
|
|
Layout.alignment: Qt.AlignVCenter
|
|
}
|
|
NTextInput {
|
|
text: Settings.data.nightLight.dayTemp
|
|
inputMethodHints: Qt.ImhDigitsOnly
|
|
Layout.alignment: Qt.AlignVCenter
|
|
onEditingFinished: {
|
|
var dayTemp = parseInt(text)
|
|
var nightTemp = parseInt(Settings.data.nightLight.nightTemp)
|
|
if (!isNaN(nightTemp) && !isNaN(dayTemp)) {
|
|
// Clamp value between [(nightTemp+500) .. 6500]
|
|
var clampedValue = Math.max(nightTemp + 500, Math.min(6500, dayTemp))
|
|
text = Settings.data.nightLight.dayTemp = clampedValue.toString()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
NToggle {
|
|
label: "Automatic Scheduling"
|
|
description: `Based on the sunset and sunrise time in <i>${LocationService.stableName}</i> - recommended.`
|
|
checked: Settings.data.nightLight.autoSchedule
|
|
onToggled: checked => Settings.data.nightLight.autoSchedule = checked
|
|
visible: Settings.data.nightLight.enabled
|
|
}
|
|
|
|
// Schedule settings
|
|
ColumnLayout {
|
|
spacing: Style.marginXS * scaling
|
|
visible: Settings.data.nightLight.enabled && !Settings.data.nightLight.autoSchedule && !Settings.data.nightLight.forced
|
|
|
|
RowLayout {
|
|
Layout.fillWidth: false
|
|
spacing: Style.marginM * scaling
|
|
|
|
NLabel {
|
|
label: "Manual Scheduling"
|
|
}
|
|
|
|
Item {// add a little more spacing
|
|
}
|
|
|
|
NText {
|
|
text: "Sunrise Time"
|
|
font.pointSize: Style.fontSizeM * scaling
|
|
color: Color.mOnSurfaceVariant
|
|
}
|
|
|
|
NComboBox {
|
|
model: timeOptions
|
|
currentKey: Settings.data.nightLight.manualSunrise
|
|
placeholder: "Select start time"
|
|
onSelected: key => Settings.data.nightLight.manualSunrise = key
|
|
minimumWidth: 120 * scaling
|
|
}
|
|
|
|
Item {// add a little more spacing
|
|
}
|
|
|
|
NText {
|
|
text: "Sunset Time"
|
|
font.pointSize: Style.fontSizeM * scaling
|
|
color: Color.mOnSurfaceVariant
|
|
}
|
|
NComboBox {
|
|
model: timeOptions
|
|
currentKey: Settings.data.nightLight.manualSunset
|
|
placeholder: "Select stop time"
|
|
onSelected: key => Settings.data.nightLight.manualSunset = key
|
|
minimumWidth: 120 * scaling
|
|
}
|
|
}
|
|
}
|
|
|
|
// Force activation toggle
|
|
NToggle {
|
|
label: "Force activation"
|
|
description: "Immediately apply night temperature without scheduling or fade."
|
|
checked: Settings.data.nightLight.forced
|
|
onToggled: checked => {
|
|
Settings.data.nightLight.forced = checked
|
|
if (checked && !Settings.data.nightLight.enabled) {
|
|
// Ensure enabled when forcing
|
|
wlsunsetCheck.running = true
|
|
} else {
|
|
NightLightService.apply()
|
|
}
|
|
}
|
|
visible: Settings.data.nightLight.enabled
|
|
}
|
|
|
|
NDivider {
|
|
Layout.fillWidth: true
|
|
Layout.topMargin: Style.marginXL * scaling
|
|
Layout.bottomMargin: Style.marginXL * scaling
|
|
}
|
|
}
|