176 lines
5.3 KiB
QML
176 lines
5.3 KiB
QML
pragma ComponentBehavior: Bound
|
|
|
|
import QtQuick
|
|
import QtQuick.Layouts
|
|
import qs.Components
|
|
import qs.Helpers
|
|
import qs.Config
|
|
|
|
ColumnLayout {
|
|
id: root
|
|
|
|
required property int rootHeight
|
|
|
|
anchors.left: parent.left
|
|
anchors.right: parent.right
|
|
anchors.margins: Appearance.padding.large * 2
|
|
|
|
spacing: Appearance.spacing.small
|
|
|
|
Loader {
|
|
Layout.topMargin: Appearance.padding.large * 2
|
|
Layout.bottomMargin: -Appearance.padding.large
|
|
Layout.alignment: Qt.AlignHCenter
|
|
|
|
active: root.rootHeight > 610
|
|
visible: active
|
|
|
|
sourceComponent: CustomText {
|
|
text: qsTr("Weather")
|
|
color: DynamicColors.palette.m3primary
|
|
font.pointSize: Appearance.font.size.extraLarge
|
|
font.weight: 500
|
|
}
|
|
}
|
|
|
|
RowLayout {
|
|
Layout.fillWidth: true
|
|
spacing: Appearance.spacing.large
|
|
|
|
MaterialIcon {
|
|
animate: true
|
|
text: Weather.icon
|
|
color: DynamicColors.palette.m3secondary
|
|
font.pointSize: Appearance.font.size.extraLarge * 2.5
|
|
}
|
|
|
|
ColumnLayout {
|
|
spacing: Appearance.spacing.small
|
|
|
|
CustomText {
|
|
Layout.fillWidth: true
|
|
|
|
animate: true
|
|
text: Weather.description
|
|
color: DynamicColors.palette.m3secondary
|
|
font.pointSize: Appearance.font.size.large
|
|
font.weight: 500
|
|
elide: Text.ElideRight
|
|
}
|
|
|
|
CustomText {
|
|
Layout.fillWidth: true
|
|
|
|
animate: true
|
|
text: qsTr("Humidity: %1%").arg(Weather.humidity)
|
|
color: DynamicColors.palette.m3onSurfaceVariant
|
|
font.pointSize: Appearance.font.size.normal
|
|
elide: Text.ElideRight
|
|
}
|
|
}
|
|
|
|
Loader {
|
|
Layout.rightMargin: Appearance.padding.smaller
|
|
active: root.width > 400
|
|
visible: active
|
|
|
|
sourceComponent: ColumnLayout {
|
|
spacing: Appearance.spacing.small
|
|
|
|
CustomText {
|
|
Layout.fillWidth: true
|
|
|
|
animate: true
|
|
text: Weather.temp
|
|
color: DynamicColors.palette.m3primary
|
|
horizontalAlignment: Text.AlignRight
|
|
font.pointSize: Appearance.font.size.extraLarge
|
|
font.weight: 500
|
|
elide: Text.ElideLeft
|
|
}
|
|
|
|
CustomText {
|
|
Layout.fillWidth: true
|
|
|
|
animate: true
|
|
text: qsTr("Feels like: %1").arg(Weather.feelsLike)
|
|
color: DynamicColors.palette.m3outline
|
|
horizontalAlignment: Text.AlignRight
|
|
font.pointSize: Appearance.font.size.smaller
|
|
elide: Text.ElideLeft
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Loader {
|
|
id: forecastLoader
|
|
|
|
Layout.topMargin: Appearance.spacing.smaller
|
|
Layout.bottomMargin: Appearance.padding.large * 2
|
|
Layout.fillWidth: true
|
|
|
|
active: root.rootHeight > 820
|
|
visible: active
|
|
|
|
sourceComponent: RowLayout {
|
|
spacing: Appearance.spacing.large
|
|
|
|
Repeater {
|
|
model: {
|
|
const forecast = Weather.hourlyForecast;
|
|
const count = root.width < 320 ? 3 : root.width < 400 ? 4 : 5;
|
|
if (!forecast)
|
|
return Array.from({
|
|
length: count
|
|
}, () => null);
|
|
|
|
return forecast.slice(0, count);
|
|
}
|
|
|
|
ColumnLayout {
|
|
id: forecastHour
|
|
|
|
required property var modelData
|
|
|
|
Layout.fillWidth: true
|
|
spacing: Appearance.spacing.small
|
|
|
|
CustomText {
|
|
Layout.fillWidth: true
|
|
text: {
|
|
const hour = forecastHour.modelData?.hour ?? 0;
|
|
return hour > 12 ? `${(hour - 12).toString().padStart(2, "0")} PM` : `${hour.toString().padStart(2, "0")} AM`;
|
|
}
|
|
color: DynamicColors.palette.m3outline
|
|
horizontalAlignment: Text.AlignHCenter
|
|
font.pointSize: Appearance.font.size.larger
|
|
}
|
|
|
|
MaterialIcon {
|
|
Layout.alignment: Qt.AlignHCenter
|
|
text: forecastHour.modelData?.icon ?? "cloud_alert"
|
|
font.pointSize: Appearance.font.size.extraLarge * 1.5
|
|
font.weight: 500
|
|
}
|
|
|
|
CustomText {
|
|
Layout.alignment: Qt.AlignHCenter
|
|
text: Config.services.useFahrenheit ? `${forecastHour.modelData?.tempF ?? 0}°F` : `${forecastHour.modelData?.tempC ?? 0}°C`
|
|
color: DynamicColors.palette.m3secondary
|
|
font.pointSize: Appearance.font.size.larger
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Timer {
|
|
running: true
|
|
triggeredOnStart: true
|
|
repeat: true
|
|
interval: 900000 // 15 minutes
|
|
onTriggered: Weather.reload()
|
|
}
|
|
}
|