crop region now correct
This commit is contained in:
@@ -2,8 +2,8 @@ pragma ComponentBehavior: Bound
|
||||
|
||||
import QtQuick
|
||||
import QtQuick.Layouts
|
||||
import QtQuick.Controls
|
||||
import Quickshell
|
||||
import Quickshell.Hyprland
|
||||
import qs.Config
|
||||
import qs.Components
|
||||
import qs.Helpers
|
||||
@@ -13,124 +13,162 @@ RowLayout {
|
||||
|
||||
required property ShellScreen screen
|
||||
|
||||
Layout.fillWidth: true
|
||||
spacing: Appearance.spacing.normal
|
||||
|
||||
Repeater {
|
||||
model: Quickshell.screens
|
||||
|
||||
delegate: ImgCrop {
|
||||
}
|
||||
}
|
||||
|
||||
component ImgCrop: Item {
|
||||
id: cropper
|
||||
|
||||
readonly property var displayData: Wallpapers.getCrop(modelData.name)
|
||||
required property ShellScreen modelData
|
||||
|
||||
Layout.fillHeight: true
|
||||
Layout.fillWidth: true
|
||||
|
||||
Image {
|
||||
id: imageView
|
||||
|
||||
property real displayH: paintedHeight
|
||||
property real displayW: paintedWidth
|
||||
property real displayX: (width - paintedWidth) * 0.5
|
||||
property real displayY: (height - paintedHeight) * 0.5
|
||||
property real scaleX: sourceW / displayW
|
||||
property real scaleY: sourceH / displayH
|
||||
property real sourceH: cropper.modelData.height
|
||||
property real sourceW: cropper.modelData.width
|
||||
|
||||
anchors.fill: parent
|
||||
fillMode: Image.PreserveAspectFit
|
||||
smooth: true
|
||||
source: Wallpapers.current
|
||||
}
|
||||
|
||||
Item {
|
||||
id: overlay
|
||||
id: delegate
|
||||
|
||||
clip: true
|
||||
height: imageView.displayH
|
||||
width: imageView.displayW
|
||||
x: imageView.displayX
|
||||
y: imageView.displayY
|
||||
required property ShellScreen modelData
|
||||
|
||||
CustomRect {
|
||||
id: cropRect
|
||||
Layout.fillHeight: true
|
||||
Layout.fillWidth: true
|
||||
|
||||
property real aspectRatio: cropper.modelData.width / cropper.modelData.height
|
||||
readonly property rect sourceRect: Qt.rect(x * imageView.scaleX, y * imageView.scaleY, width * imageView.scaleX, height * imageView.scaleY)
|
||||
property real zoom: cropper.displayData.zoom
|
||||
Image {
|
||||
id: scaledImg
|
||||
|
||||
function clampToBounds() {
|
||||
x = Math.max(0, Math.min(x, overlay.width - width));
|
||||
property var displayData
|
||||
readonly property real imageRatio: scaledImg.sourceSize.width / scaledImg.sourceSize.height
|
||||
property real monitorScale: 1.0
|
||||
readonly property real scaleDownX: scaledImg.width / scaledImg.sourceSize.width
|
||||
readonly property real scaleDownY: scaledImg.height / scaledImg.sourceSize.height
|
||||
readonly property real scaleX: (scaledImg.sourceSize.width * monitorScale) / scaledImg.width
|
||||
readonly property real scaleY: (scaledImg.sourceSize.height * monitorScale) / scaledImg.height
|
||||
|
||||
y = Math.max(0, Math.min(y, overlay.height - height));
|
||||
anchors.left: parent.left
|
||||
anchors.right: parent.right
|
||||
anchors.verticalCenter: parent.verticalCenter
|
||||
height: width / imageRatio
|
||||
source: Wallpapers.current
|
||||
|
||||
Component.onCompleted: {
|
||||
Hyprland.refreshMonitors();
|
||||
// monitorScale = Hyprland.monitorFor(delegate.modelData).scale;
|
||||
}
|
||||
onScaleYChanged: console.log(scaleX, scaleY, "scale")
|
||||
onStatusChanged: {
|
||||
if (scaledImg.status == Image.Ready) {
|
||||
console.log(scaledImg.sourceSize.width, scaledImg.sourceSize.height, scaledImg.width, scaledImg.height, delegate.modelData.width, delegate.modelData.height);
|
||||
}
|
||||
}
|
||||
|
||||
border.color: DynamicColors.palette.m3primary
|
||||
border.width: 2
|
||||
color: DynamicColors.tPalette.m3primary
|
||||
height: width / aspectRatio
|
||||
radius: Appearance.rounding.small
|
||||
visible: imageView.status === Image.Ready
|
||||
width: Math.min(overlay.width / zoom, overlay.height * aspectRatio / zoom)
|
||||
x: cropper.displayData.x / imageView.scaleX
|
||||
y: cropper.displayData.y / imageView.scaleY
|
||||
}
|
||||
Connections {
|
||||
function onLoaded(): void {
|
||||
scaledImg.displayData = Wallpapers.getCrop(delegate.modelData.name);
|
||||
cropRect.zoom = scaledImg.displayData.zoom;
|
||||
cropRect.restoreFromData();
|
||||
}
|
||||
|
||||
MouseArea {
|
||||
function updateCrop(mouseX, mouseY) {
|
||||
let nx = mouseX - cropRect.width * 0.5;
|
||||
let ny = mouseY - cropRect.height * 0.5;
|
||||
|
||||
nx = Math.max(0, Math.min(nx, overlay.width - cropRect.width));
|
||||
|
||||
ny = Math.max(0, Math.min(ny, overlay.height - cropRect.height));
|
||||
|
||||
cropRect.x = nx;
|
||||
cropRect.y = ny;
|
||||
target: Wallpapers.monitorCrops
|
||||
}
|
||||
|
||||
anchors.fill: parent
|
||||
hoverEnabled: true
|
||||
preventStealing: true
|
||||
CustomRect {
|
||||
id: cropRect
|
||||
|
||||
onPositionChanged: mouse => {
|
||||
if (pressed)
|
||||
property real aspectRatio: delegate.modelData.width / delegate.modelData.height
|
||||
readonly property real baseHeight: baseWidth / aspectRatio
|
||||
readonly property real baseWidth: {
|
||||
let fittedHeight = scaledImg.height;
|
||||
let fittedWidth = fittedHeight * aspectRatio;
|
||||
|
||||
if (fittedWidth > scaledImg.width) {
|
||||
fittedWidth = scaledImg.width;
|
||||
fittedHeight = fittedWidth / aspectRatio;
|
||||
}
|
||||
|
||||
return fittedWidth;
|
||||
}
|
||||
readonly property real imageX: (scaledImg.width - scaledImg.width) / 2
|
||||
readonly property real imageY: (scaledImg.height - scaledImg.height) / 2
|
||||
property real zoom: scaledImg.displayData.zoom
|
||||
|
||||
function centerInImage() {
|
||||
x = imageX + (scaledImg.width - width) / 2;
|
||||
y = imageY + (scaledImg.height - height) / 2;
|
||||
}
|
||||
|
||||
function clampToBounds() {
|
||||
x = Math.max(imageX, Math.min(x, imageX + scaledImg.width - width));
|
||||
|
||||
y = Math.max(imageY, Math.min(y, imageY + scaledImg.height - height));
|
||||
}
|
||||
|
||||
function restoreFromData() {
|
||||
let data = scaledImg.displayData;
|
||||
|
||||
if (data && data.scaledX !== 0 || data.scaledY !== 0 || data.scaledWidth !== 0 || data.scaledHeight !== 0) {
|
||||
console.log("true");
|
||||
x = imageX + data.scaledX;
|
||||
y = imageY + data.scaledY;
|
||||
|
||||
clampToBounds();
|
||||
} else {
|
||||
console.log("false");
|
||||
zoom = 1.0;
|
||||
centerInImage();
|
||||
}
|
||||
}
|
||||
|
||||
border.color: "lime"
|
||||
border.width: 2
|
||||
height: baseHeight / zoom
|
||||
width: baseWidth / zoom
|
||||
|
||||
onHeightChanged: clampToBounds()
|
||||
onWidthChanged: clampToBounds()
|
||||
}
|
||||
|
||||
MouseArea {
|
||||
id: mouse
|
||||
|
||||
function updateCrop(mouseX, mouseY) {
|
||||
let nx = mouseX - cropRect.width * 0.5;
|
||||
let ny = mouseY - cropRect.height * 0.5;
|
||||
|
||||
nx = Math.max(cropRect.imageX, Math.min(nx, cropRect.imageX + scaledImg.width - cropRect.width));
|
||||
|
||||
ny = Math.max(cropRect.imageY, Math.min(ny, cropRect.imageY + scaledImg.height - cropRect.height));
|
||||
|
||||
cropRect.x = nx;
|
||||
cropRect.y = ny;
|
||||
}
|
||||
|
||||
anchors.fill: parent
|
||||
hoverEnabled: true
|
||||
preventStealing: true
|
||||
|
||||
onPositionChanged: mouse => {
|
||||
if (pressed)
|
||||
updateCrop(mouse.x, mouse.y);
|
||||
}
|
||||
onPressed: mouse => {
|
||||
updateCrop(mouse.x, mouse.y);
|
||||
}
|
||||
onPressed: mouse => {
|
||||
updateCrop(mouse.x, mouse.y);
|
||||
}
|
||||
onReleased: {
|
||||
Wallpapers.recentlyChanged = false;
|
||||
Wallpapers.setCrop(cropper.modelData.name, cropRect.sourceRect, cropRect.zoom);
|
||||
// Config.background.sourceClipX = cropRect.sourceRect.x;
|
||||
// Config.background.sourceClipY = cropRect.sourceRect.y;
|
||||
// Config.background.sourceClipW = cropRect.sourceRect.width;
|
||||
// Config.background.sourceClipH = cropRect.sourceRect.height;
|
||||
// Config.save();
|
||||
}
|
||||
onWheel: wheel => {
|
||||
let oldCenterX = cropRect.x + cropRect.width * 0.5;
|
||||
let oldCenterY = cropRect.y + cropRect.height * 0.5;
|
||||
}
|
||||
onReleased: {
|
||||
const croprect = cropRect.mapToItem(scaledImg, 0, 0, cropRect.width, cropRect.height);
|
||||
const upscaledRect = Qt.rect(croprect.x / scaledImg.width, croprect.y / scaledImg.height, croprect.width / scaledImg.width, croprect.height / scaledImg.height);
|
||||
Wallpapers.setCrop(delegate.modelData.name, upscaledRect, croprect, cropRect.zoom);
|
||||
}
|
||||
onWheel: wheel => {
|
||||
let oldCenterX = cropRect.x + cropRect.width * 0.5;
|
||||
let oldCenterY = cropRect.y + cropRect.height * 0.5;
|
||||
|
||||
if (wheel.angleDelta.y > 0)
|
||||
cropRect.zoom *= 1.1;
|
||||
else
|
||||
cropRect.zoom /= 1.1;
|
||||
if (wheel.angleDelta.y > 0)
|
||||
cropRect.zoom *= 1.1;
|
||||
else
|
||||
cropRect.zoom /= 1.1;
|
||||
|
||||
cropRect.zoom = Math.max(1.0, Math.min(cropRect.zoom, 10.0));
|
||||
Config.background.zoom = cropRect.zoom;
|
||||
cropRect.zoom = Math.max(1.0, Math.min(cropRect.zoom, 5.0));
|
||||
console.log(cropRect.zoom);
|
||||
|
||||
cropRect.x = oldCenterX - cropRect.width * 0.5;
|
||||
cropRect.y = oldCenterY - cropRect.height * 0.5;
|
||||
cropRect.x = oldCenterX - cropRect.width * 0.5;
|
||||
cropRect.y = oldCenterY - cropRect.height * 0.5;
|
||||
|
||||
cropRect.clampToBounds();
|
||||
cropRect.clampToBounds();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user