mirror of
https://github.com/AvengeMedia/DankMaterialShell.git
synced 2025-12-06 05:25:41 -05:00
239 lines
7.9 KiB
QML
239 lines
7.9 KiB
QML
import QtQuick
|
|
import QtQuick.Effects
|
|
import Quickshell
|
|
import Quickshell.Wayland
|
|
import qs.Common
|
|
import qs.Widgets
|
|
import qs.Services
|
|
|
|
Variants {
|
|
model: {
|
|
if (SessionData.isGreeterMode) {
|
|
return Quickshell.screens;
|
|
}
|
|
return SettingsData.getFilteredScreens("wallpaper");
|
|
}
|
|
|
|
PanelWindow {
|
|
id: blurWallpaperWindow
|
|
|
|
required property var modelData
|
|
|
|
screen: modelData
|
|
|
|
WlrLayershell.layer: WlrLayer.Background
|
|
WlrLayershell.namespace: "dms:blurwallpaper"
|
|
WlrLayershell.exclusionMode: ExclusionMode.Ignore
|
|
|
|
anchors.top: true
|
|
anchors.bottom: true
|
|
anchors.left: true
|
|
anchors.right: true
|
|
|
|
color: "transparent"
|
|
|
|
mask: Region {
|
|
item: Item {}
|
|
}
|
|
|
|
Item {
|
|
id: root
|
|
anchors.fill: parent
|
|
|
|
property string source: SessionData.getMonitorWallpaper(modelData.name) || ""
|
|
property bool isColorSource: source.startsWith("#")
|
|
|
|
Connections {
|
|
target: SessionData
|
|
function onIsLightModeChanged() {
|
|
if (SessionData.perModeWallpaper) {
|
|
var newSource = SessionData.getMonitorWallpaper(modelData.name) || "";
|
|
if (newSource !== root.source) {
|
|
root.source = newSource;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function getFillMode(modeName) {
|
|
switch (modeName) {
|
|
case "Stretch":
|
|
return Image.Stretch;
|
|
case "Fit":
|
|
case "PreserveAspectFit":
|
|
return Image.PreserveAspectFit;
|
|
case "Fill":
|
|
case "PreserveAspectCrop":
|
|
return Image.PreserveAspectCrop;
|
|
case "Tile":
|
|
return Image.Tile;
|
|
case "TileVertically":
|
|
return Image.TileVertically;
|
|
case "TileHorizontally":
|
|
return Image.TileHorizontally;
|
|
case "Pad":
|
|
return Image.Pad;
|
|
default:
|
|
return Image.PreserveAspectCrop;
|
|
}
|
|
}
|
|
|
|
Component.onCompleted: {
|
|
if (source) {
|
|
const formattedSource = source.startsWith("file://") ? source : "file://" + source;
|
|
setWallpaperImmediate(formattedSource);
|
|
}
|
|
isInitialized = true;
|
|
}
|
|
|
|
property bool isInitialized: false
|
|
property real transitionProgress: 0
|
|
readonly property bool transitioning: transitionAnimation.running
|
|
|
|
onSourceChanged: {
|
|
const isColor = source.startsWith("#");
|
|
|
|
if (!source) {
|
|
setWallpaperImmediate("");
|
|
} else if (isColor) {
|
|
setWallpaperImmediate("");
|
|
} else {
|
|
if (!isInitialized || !currentWallpaper.source) {
|
|
setWallpaperImmediate(source.startsWith("file://") ? source : "file://" + source);
|
|
isInitialized = true;
|
|
} else if (CompositorService.isNiri && SessionData.isSwitchingMode) {
|
|
setWallpaperImmediate(source.startsWith("file://") ? source : "file://" + source);
|
|
} else {
|
|
changeWallpaper(source.startsWith("file://") ? source : "file://" + source);
|
|
}
|
|
}
|
|
}
|
|
|
|
function setWallpaperImmediate(newSource) {
|
|
transitionAnimation.stop();
|
|
root.transitionProgress = 0.0;
|
|
currentWallpaper.source = newSource;
|
|
nextWallpaper.source = "";
|
|
currentWallpaper.opacity = 1;
|
|
nextWallpaper.opacity = 0;
|
|
}
|
|
|
|
function changeWallpaper(newPath) {
|
|
if (newPath === currentWallpaper.source)
|
|
return;
|
|
if (!newPath || newPath.startsWith("#"))
|
|
return;
|
|
if (root.transitioning) {
|
|
transitionAnimation.stop();
|
|
root.transitionProgress = 0;
|
|
currentWallpaper.source = nextWallpaper.source;
|
|
nextWallpaper.source = "";
|
|
}
|
|
|
|
if (!currentWallpaper.source) {
|
|
setWallpaperImmediate(newPath);
|
|
return;
|
|
}
|
|
|
|
nextWallpaper.source = newPath;
|
|
|
|
if (nextWallpaper.status === Image.Ready) {
|
|
transitionAnimation.start();
|
|
}
|
|
}
|
|
|
|
Loader {
|
|
anchors.fill: parent
|
|
active: !root.source || root.isColorSource
|
|
asynchronous: true
|
|
|
|
sourceComponent: DankBackdrop {
|
|
screenName: modelData.name
|
|
}
|
|
}
|
|
|
|
Image {
|
|
id: currentWallpaper
|
|
anchors.fill: parent
|
|
visible: false
|
|
opacity: 1
|
|
asynchronous: true
|
|
smooth: true
|
|
cache: true
|
|
sourceSize: Qt.size(modelData.width, modelData.height)
|
|
fillMode: root.getFillMode(SessionData.isGreeterMode ? GreetdSettings.wallpaperFillMode : SettingsData.wallpaperFillMode)
|
|
}
|
|
|
|
Image {
|
|
id: nextWallpaper
|
|
anchors.fill: parent
|
|
visible: false
|
|
opacity: 0
|
|
asynchronous: true
|
|
smooth: true
|
|
cache: true
|
|
sourceSize: Qt.size(modelData.width, modelData.height)
|
|
fillMode: root.getFillMode(SessionData.isGreeterMode ? GreetdSettings.wallpaperFillMode : SettingsData.wallpaperFillMode)
|
|
|
|
onStatusChanged: {
|
|
if (status !== Image.Ready)
|
|
return;
|
|
if (!root.transitioning) {
|
|
transitionAnimation.start();
|
|
}
|
|
}
|
|
}
|
|
|
|
Item {
|
|
id: blurredLayer
|
|
anchors.fill: parent
|
|
|
|
MultiEffect {
|
|
anchors.fill: parent
|
|
source: currentWallpaper
|
|
visible: currentWallpaper.source !== ""
|
|
blurEnabled: true
|
|
blur: 0.8
|
|
blurMax: 75
|
|
opacity: 1 - root.transitionProgress
|
|
autoPaddingEnabled: false
|
|
}
|
|
|
|
MultiEffect {
|
|
anchors.fill: parent
|
|
source: nextWallpaper
|
|
visible: nextWallpaper.source !== ""
|
|
blurEnabled: true
|
|
blur: 0.8
|
|
blurMax: 75
|
|
opacity: root.transitionProgress
|
|
autoPaddingEnabled: false
|
|
}
|
|
}
|
|
|
|
NumberAnimation {
|
|
id: transitionAnimation
|
|
target: root
|
|
property: "transitionProgress"
|
|
from: 0.0
|
|
to: 1.0
|
|
duration: 1000
|
|
easing.type: Easing.InOutCubic
|
|
onFinished: {
|
|
const tempSource = nextWallpaper.source;
|
|
if (tempSource && nextWallpaper.status === Image.Ready && !tempSource.toString().startsWith("#")) {
|
|
currentWallpaper.source = tempSource;
|
|
}
|
|
root.transitionProgress = 0.0;
|
|
currentWallpaper.opacity = 1;
|
|
nextWallpaper.opacity = 0;
|
|
|
|
Qt.callLater(() => {
|
|
nextWallpaper.source = "";
|
|
});
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|