forked from Shinonome/dots-hyprland
513 lines
22 KiB
QML
513 lines
22 KiB
QML
pragma ComponentBehavior: Bound
|
|
import qs.modules.common
|
|
import qs.modules.common.functions
|
|
import qs.modules.common.widgets
|
|
import qs.services
|
|
import QtQuick
|
|
import QtQuick.Controls
|
|
import Qt.labs.synchronizer
|
|
import Quickshell
|
|
import Quickshell.Io
|
|
import Quickshell.Wayland
|
|
import Quickshell.Hyprland
|
|
|
|
PanelWindow {
|
|
id: root
|
|
visible: false
|
|
color: "transparent"
|
|
WlrLayershell.namespace: "quickshell:regionSelector"
|
|
WlrLayershell.layer: WlrLayer.Overlay
|
|
WlrLayershell.keyboardFocus: WlrKeyboardFocus.OnDemand
|
|
exclusionMode: ExclusionMode.Ignore
|
|
anchors {
|
|
left: true
|
|
right: true
|
|
top: true
|
|
bottom: true
|
|
}
|
|
|
|
// TODO: Ask: sidebar AI; Ocr: tesseract
|
|
enum SnipAction { Copy, Edit, Search, CharRecognition, Record, RecordWithSound }
|
|
enum SelectionMode { RectCorners, Circle }
|
|
property var action: RegionSelection.SnipAction.Copy
|
|
property var selectionMode: RegionSelection.SelectionMode.RectCorners
|
|
signal dismiss()
|
|
|
|
property string screenshotDir: Directories.screenshotTemp
|
|
property string imageSearchEngineBaseUrl: Config.options.search.imageSearch.imageSearchEngineBaseUrl
|
|
property string fileUploadApiEndpoint: "https://uguu.se/upload"
|
|
property color overlayColor: "#88111111"
|
|
property color brightText: Appearance.m3colors.darkmode ? Appearance.colors.colOnLayer0 : Appearance.colors.colLayer0
|
|
property color brightSecondary: Appearance.m3colors.darkmode ? Appearance.colors.colSecondary : Appearance.colors.colOnSecondary
|
|
property color brightTertiary: Appearance.m3colors.darkmode ? Appearance.colors.colTertiary : Qt.lighter(Appearance.colors.colPrimary)
|
|
property color selectionBorderColor: ColorUtils.mix(brightText, brightSecondary, 0.5)
|
|
property color selectionFillColor: "#33ffffff"
|
|
property color windowBorderColor: brightSecondary
|
|
property color windowFillColor: ColorUtils.transparentize(windowBorderColor, 0.85)
|
|
property color imageBorderColor: brightTertiary
|
|
property color imageFillColor: ColorUtils.transparentize(imageBorderColor, 0.85)
|
|
property color onBorderColor: "#ff000000"
|
|
readonly property var windows: [...HyprlandData.windowList].sort((a, b) => {
|
|
// Sort floating=true windows before others
|
|
if (a.floating === b.floating) return 0;
|
|
return a.floating ? -1 : 1;
|
|
})
|
|
readonly property var layers: HyprlandData.layers
|
|
readonly property real falsePositivePreventionRatio: 0.5
|
|
|
|
readonly property HyprlandMonitor hyprlandMonitor: Hyprland.monitorFor(screen)
|
|
readonly property real monitorScale: hyprlandMonitor.scale
|
|
readonly property real monitorOffsetX: hyprlandMonitor.x
|
|
readonly property real monitorOffsetY: hyprlandMonitor.y
|
|
property int activeWorkspaceId: hyprlandMonitor.activeWorkspace?.id ?? 0
|
|
property string screenshotPath: `${root.screenshotDir}/image-${screen.name}`
|
|
property real dragStartX: 0
|
|
property real dragStartY: 0
|
|
property real draggingX: 0
|
|
property real draggingY: 0
|
|
property real dragDiffX: 0
|
|
property real dragDiffY: 0
|
|
property bool draggedAway: (dragDiffX !== 0 || dragDiffY !== 0)
|
|
property bool dragging: false
|
|
property list<point> points: []
|
|
property var mouseButton: null
|
|
property var imageRegions: []
|
|
readonly property list<var> windowRegions: RegionFunctions.filterWindowRegionsByLayers(
|
|
root.windows.filter(w => w.workspace.id === root.activeWorkspaceId),
|
|
root.layerRegions
|
|
).map(window => {
|
|
return {
|
|
at: [window.at[0] - root.monitorOffsetX, window.at[1] - root.monitorOffsetY],
|
|
size: [window.size[0], window.size[1]],
|
|
class: window.class,
|
|
title: window.title,
|
|
}
|
|
})
|
|
readonly property list<var> layerRegions: {
|
|
const layersOfThisMonitor = root.layers[root.hyprlandMonitor.name]
|
|
const topLayers = layersOfThisMonitor?.levels["2"]
|
|
if (!topLayers) return [];
|
|
const nonBarTopLayers = topLayers
|
|
.filter(layer => !(layer.namespace.includes(":bar") || layer.namespace.includes(":verticalBar") || layer.namespace.includes(":dock")))
|
|
.map(layer => {
|
|
return {
|
|
at: [layer.x, layer.y],
|
|
size: [layer.w, layer.h],
|
|
namespace: layer.namespace,
|
|
}
|
|
})
|
|
const offsetAdjustedLayers = nonBarTopLayers.map(layer => {
|
|
return {
|
|
at: [layer.at[0] - root.monitorOffsetX, layer.at[1] - root.monitorOffsetY],
|
|
size: layer.size,
|
|
namespace: layer.namespace,
|
|
}
|
|
});
|
|
return offsetAdjustedLayers;
|
|
}
|
|
|
|
property bool isCircleSelection: (root.selectionMode === RegionSelection.SelectionMode.Circle)
|
|
property bool enableWindowRegions: Config.options.regionSelector.targetRegions.windows && !isCircleSelection
|
|
property bool enableLayerRegions: Config.options.regionSelector.targetRegions.layers && !isCircleSelection
|
|
property bool enableContentRegions: Config.options.regionSelector.targetRegions.content
|
|
property real targetRegionOpacity: Config.options.regionSelector.targetRegions.opacity
|
|
property bool contentRegionOpacity: Config.options.regionSelector.targetRegions.contentRegionOpacity
|
|
|
|
property real targetedRegionX: -1
|
|
property real targetedRegionY: -1
|
|
property real targetedRegionWidth: 0
|
|
property real targetedRegionHeight: 0
|
|
function targetedRegionValid() {
|
|
return (root.targetedRegionX >= 0 && root.targetedRegionY >= 0)
|
|
}
|
|
function setRegionToTargeted() {
|
|
root.regionX = root.targetedRegionX;
|
|
root.regionY = root.targetedRegionY;
|
|
root.regionWidth = root.targetedRegionWidth;
|
|
root.regionHeight = root.targetedRegionHeight;
|
|
}
|
|
|
|
function updateTargetedRegion(x, y) {
|
|
// Image regions
|
|
const clickedRegion = root.imageRegions.find(region => {
|
|
return region.at[0] <= x && x <= region.at[0] + region.size[0] && region.at[1] <= y && y <= region.at[1] + region.size[1];
|
|
});
|
|
if (clickedRegion) {
|
|
root.targetedRegionX = clickedRegion.at[0];
|
|
root.targetedRegionY = clickedRegion.at[1];
|
|
root.targetedRegionWidth = clickedRegion.size[0];
|
|
root.targetedRegionHeight = clickedRegion.size[1];
|
|
return;
|
|
}
|
|
|
|
// Layer regions
|
|
const clickedLayer = root.layerRegions.find(region => {
|
|
return region.at[0] <= x && x <= region.at[0] + region.size[0] && region.at[1] <= y && y <= region.at[1] + region.size[1];
|
|
});
|
|
if (clickedLayer) {
|
|
root.targetedRegionX = clickedLayer.at[0];
|
|
root.targetedRegionY = clickedLayer.at[1];
|
|
root.targetedRegionWidth = clickedLayer.size[0];
|
|
root.targetedRegionHeight = clickedLayer.size[1];
|
|
return;
|
|
}
|
|
|
|
// Window regions
|
|
const clickedWindow = root.windowRegions.find(region => {
|
|
return region.at[0] <= x && x <= region.at[0] + region.size[0] && region.at[1] <= y && y <= region.at[1] + region.size[1];
|
|
});
|
|
if (clickedWindow) {
|
|
root.targetedRegionX = clickedWindow.at[0];
|
|
root.targetedRegionY = clickedWindow.at[1];
|
|
root.targetedRegionWidth = clickedWindow.size[0];
|
|
root.targetedRegionHeight = clickedWindow.size[1];
|
|
return;
|
|
}
|
|
|
|
root.targetedRegionX = -1;
|
|
root.targetedRegionY = -1;
|
|
root.targetedRegionWidth = 0;
|
|
root.targetedRegionHeight = 0;
|
|
}
|
|
|
|
property real regionWidth: Math.abs(draggingX - dragStartX)
|
|
property real regionHeight: Math.abs(draggingY - dragStartY)
|
|
property real regionX: Math.min(dragStartX, draggingX)
|
|
property real regionY: Math.min(dragStartY, draggingY)
|
|
|
|
Process {
|
|
id: screenshotProc
|
|
running: true
|
|
command: ["bash", "-c", `mkdir -p '${StringUtils.shellSingleQuoteEscape(root.screenshotDir)}' && grim -o '${StringUtils.shellSingleQuoteEscape(root.screen.name)}' '${StringUtils.shellSingleQuoteEscape(root.screenshotPath)}'`]
|
|
onExited: (exitCode, exitStatus) => {
|
|
if (root.enableContentRegions) imageDetectionProcess.running = true;
|
|
root.preparationDone = !checkRecordingProc.running;
|
|
}
|
|
}
|
|
property bool isRecording: root.action === RegionSelection.SnipAction.Record || root.action === RegionSelection.SnipAction.RecordWithSound
|
|
property bool recordingShouldStop: false
|
|
Process {
|
|
id: checkRecordingProc
|
|
running: isRecording
|
|
command: ["pidof", "wf-recorder"]
|
|
onExited: (exitCode, exitStatus) => {
|
|
root.preparationDone = !screenshotProc.running
|
|
root.recordingShouldStop = (exitCode === 0);
|
|
}
|
|
}
|
|
property bool preparationDone: false
|
|
onPreparationDoneChanged: {
|
|
if (!preparationDone) return;
|
|
if (root.isRecording && root.recordingShouldStop) {
|
|
Quickshell.execDetached([Directories.recordScriptPath]);
|
|
root.dismiss();
|
|
return;
|
|
}
|
|
root.visible = true;
|
|
}
|
|
|
|
Process {
|
|
id: imageDetectionProcess
|
|
command: ["bash", "-c", `${Directories.scriptPath}/images/find-regions-venv.sh `
|
|
+ `--hyprctl `
|
|
+ `--image '${StringUtils.shellSingleQuoteEscape(root.screenshotPath)}' `
|
|
+ `--max-width ${Math.round(root.screen.width * root.falsePositivePreventionRatio)} `
|
|
+ `--max-height ${Math.round(root.screen.height * root.falsePositivePreventionRatio)} `]
|
|
stdout: StdioCollector {
|
|
id: imageDimensionCollector
|
|
onStreamFinished: {
|
|
imageRegions = RegionFunctions.filterImageRegions(
|
|
JSON.parse(imageDimensionCollector.text),
|
|
root.windowRegions
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
function snip() {
|
|
// Validity check
|
|
if (root.regionWidth <= 0 || root.regionHeight <= 0) {
|
|
console.warn("[Region Selector] Invalid region size, skipping snip.");
|
|
root.dismiss();
|
|
}
|
|
|
|
// Clamp region to screen bounds
|
|
root.regionX = Math.max(0, Math.min(root.regionX, root.screen.width - root.regionWidth));
|
|
root.regionY = Math.max(0, Math.min(root.regionY, root.screen.height - root.regionHeight));
|
|
root.regionWidth = Math.max(0, Math.min(root.regionWidth, root.screen.width - root.regionX));
|
|
root.regionHeight = Math.max(0, Math.min(root.regionHeight, root.screen.height - root.regionY));
|
|
|
|
// Adjust action
|
|
if (root.action === RegionSelection.SnipAction.Copy || root.action === RegionSelection.SnipAction.Edit) {
|
|
root.action = root.mouseButton === Qt.RightButton ? RegionSelection.SnipAction.Edit : RegionSelection.SnipAction.Copy;
|
|
}
|
|
|
|
// Set command for action
|
|
const rx = Math.round(root.regionX * root.monitorScale);
|
|
const ry = Math.round(root.regionY * root.monitorScale);
|
|
const rw = Math.round(root.regionWidth * root.monitorScale);
|
|
const rh = Math.round(root.regionHeight * root.monitorScale);
|
|
const cropBase = `magick ${StringUtils.shellSingleQuoteEscape(root.screenshotPath)} `
|
|
+ `-crop ${rw}x${rh}+${rx}+${ry}`
|
|
const cropToStdout = `${cropBase} -`
|
|
const cropInPlace = `${cropBase} '${StringUtils.shellSingleQuoteEscape(root.screenshotPath)}'`
|
|
const cleanup = `rm '${StringUtils.shellSingleQuoteEscape(root.screenshotPath)}'`
|
|
const slurpRegion = `${rx},${ry} ${rw}x${rh}`
|
|
const uploadAndGetUrl = (filePath) => {
|
|
return `curl -sF files[]=@'${StringUtils.shellSingleQuoteEscape(filePath)}' ${root.fileUploadApiEndpoint} | jq -r '.files[0].url'`
|
|
}
|
|
switch (root.action) {
|
|
case RegionSelection.SnipAction.Copy:
|
|
snipProc.command = ["bash", "-c", `${cropToStdout} | wl-copy && ${cleanup}`]
|
|
break;
|
|
case RegionSelection.SnipAction.Edit:
|
|
snipProc.command = ["bash", "-c", `${cropToStdout} | swappy -f - && ${cleanup}`]
|
|
break;
|
|
case RegionSelection.SnipAction.Search:
|
|
snipProc.command = ["bash", "-c", `${cropInPlace} && xdg-open "${root.imageSearchEngineBaseUrl}$(${uploadAndGetUrl(root.screenshotPath)})" && ${cleanup}`]
|
|
break;
|
|
case RegionSelection.SnipAction.CharRecognition:
|
|
snipProc.command = ["bash", "-c", `${cropInPlace} && tesseract '${StringUtils.shellSingleQuoteEscape(root.screenshotPath)}' stdout -l $(tesseract --list-langs | awk 'NR>1{print $1}' | tr '\\n' '+' | sed 's/\\+$/\\n/') | wl-copy && ${cleanup}`]
|
|
break;
|
|
case RegionSelection.SnipAction.Record:
|
|
snipProc.command = ["bash", "-c", `${Directories.recordScriptPath} --region '${slurpRegion}'`]
|
|
break;
|
|
case RegionSelection.SnipAction.RecordWithSound:
|
|
snipProc.command = ["bash", "-c", `${Directories.recordScriptPath} --region '${slurpRegion}' --sound`]
|
|
break;
|
|
default:
|
|
console.warn("[Region Selector] Unknown snip action, skipping snip.");
|
|
root.dismiss();
|
|
return;
|
|
}
|
|
|
|
// Image post-processing
|
|
snipProc.startDetached();
|
|
root.dismiss();
|
|
}
|
|
|
|
Process {
|
|
id: snipProc
|
|
}
|
|
|
|
ScreencopyView {
|
|
anchors.fill: parent
|
|
live: false
|
|
captureSource: root.screen
|
|
|
|
focus: root.visible
|
|
Keys.onPressed: (event) => { // Esc to close
|
|
if (event.key === Qt.Key_Escape) {
|
|
root.dismiss();
|
|
}
|
|
}
|
|
|
|
MouseArea {
|
|
id: mouseArea
|
|
anchors.fill: parent
|
|
cursorShape: Qt.CrossCursor
|
|
acceptedButtons: Qt.LeftButton | Qt.RightButton
|
|
hoverEnabled: true
|
|
|
|
// Controls
|
|
onPressed: (mouse) => {
|
|
root.dragStartX = mouse.x;
|
|
root.dragStartY = mouse.y;
|
|
root.draggingX = mouse.x;
|
|
root.draggingY = mouse.y;
|
|
root.dragging = true;
|
|
root.mouseButton = mouse.button;
|
|
}
|
|
onReleased: (mouse) => {
|
|
// Detect if it was a click -> Try to select targeted region
|
|
if (root.draggingX === root.dragStartX && root.draggingY === root.dragStartY) {
|
|
if (root.targetedRegionValid()) {
|
|
root.setRegionToTargeted();
|
|
}
|
|
}
|
|
// Circle dragging?
|
|
else if (root.selectionMode === RegionSelection.SelectionMode.Circle) {
|
|
const padding = Config.options.regionSelector.circle.padding + Config.options.regionSelector.circle.strokeWidth / 2;
|
|
const dragPoints = (root.points.length > 0) ? root.points : [{ x: mouseArea.mouseX, y: mouseArea.mouseY }];
|
|
const maxX = Math.max(...dragPoints.map(p => p.x));
|
|
const minX = Math.min(...dragPoints.map(p => p.x));
|
|
const maxY = Math.max(...dragPoints.map(p => p.y));
|
|
const minY = Math.min(...dragPoints.map(p => p.y));
|
|
root.regionX = minX - padding;
|
|
root.regionY = minY - padding;
|
|
root.regionWidth = maxX - minX + padding * 2;
|
|
root.regionHeight = maxY - minY + padding * 2;
|
|
}
|
|
root.snip();
|
|
}
|
|
onPositionChanged: (mouse) => {
|
|
root.updateTargetedRegion(mouse.x, mouse.y);
|
|
if (!root.dragging) return;
|
|
root.draggingX = mouse.x;
|
|
root.draggingY = mouse.y;
|
|
root.dragDiffX = mouse.x - root.dragStartX;
|
|
root.dragDiffY = mouse.y - root.dragStartY;
|
|
root.points.push({ x: mouse.x, y: mouse.y });
|
|
}
|
|
|
|
Loader {
|
|
z: 2
|
|
anchors.fill: parent
|
|
active: root.selectionMode === RegionSelection.SelectionMode.RectCorners
|
|
sourceComponent: RectCornersSelectionDetails {
|
|
regionX: root.regionX
|
|
regionY: root.regionY
|
|
regionWidth: root.regionWidth
|
|
regionHeight: root.regionHeight
|
|
mouseX: mouseArea.mouseX
|
|
mouseY: mouseArea.mouseY
|
|
color: root.selectionBorderColor
|
|
overlayColor: root.overlayColor
|
|
}
|
|
}
|
|
|
|
Loader {
|
|
z: 2
|
|
anchors.fill: parent
|
|
active: root.selectionMode === RegionSelection.SelectionMode.Circle
|
|
sourceComponent: CircleSelectionDetails {
|
|
color: root.selectionBorderColor
|
|
overlayColor: root.overlayColor
|
|
points: root.points
|
|
}
|
|
}
|
|
|
|
// Window regions
|
|
Repeater {
|
|
model: ScriptModel {
|
|
values: root.enableWindowRegions ? root.windowRegions : []
|
|
}
|
|
delegate: TargetRegion {
|
|
z: 2
|
|
required property var modelData
|
|
clientDimensions: modelData
|
|
showIcon: true
|
|
targeted: !root.draggedAway &&
|
|
(root.targetedRegionX === modelData.at[0]
|
|
&& root.targetedRegionY === modelData.at[1]
|
|
&& root.targetedRegionWidth === modelData.size[0]
|
|
&& root.targetedRegionHeight === modelData.size[1])
|
|
|
|
opacity: root.draggedAway ? 0 : root.targetRegionOpacity
|
|
borderColor: root.windowBorderColor
|
|
fillColor: targeted ? root.windowFillColor : "transparent"
|
|
text: `${modelData.class}`
|
|
radius: Appearance.rounding.windowRounding
|
|
}
|
|
}
|
|
|
|
// Layer regions
|
|
Repeater {
|
|
model: ScriptModel {
|
|
values: root.enableLayerRegions ? root.layerRegions : []
|
|
}
|
|
delegate: TargetRegion {
|
|
z: 3
|
|
required property var modelData
|
|
clientDimensions: modelData
|
|
targeted: !root.draggedAway &&
|
|
(root.targetedRegionX === modelData.at[0]
|
|
&& root.targetedRegionY === modelData.at[1]
|
|
&& root.targetedRegionWidth === modelData.size[0]
|
|
&& root.targetedRegionHeight === modelData.size[1])
|
|
|
|
opacity: root.draggedAway ? 0 : root.targetRegionOpacity
|
|
borderColor: root.windowBorderColor
|
|
fillColor: targeted ? root.windowFillColor : "transparent"
|
|
text: `${modelData.namespace}`
|
|
radius: Appearance.rounding.windowRounding
|
|
}
|
|
}
|
|
|
|
// Content regions
|
|
Repeater {
|
|
model: ScriptModel {
|
|
values: root.enableContentRegions ? root.imageRegions : []
|
|
}
|
|
delegate: TargetRegion {
|
|
z: 4
|
|
required property var modelData
|
|
clientDimensions: modelData
|
|
targeted: !root.draggedAway &&
|
|
(root.targetedRegionX === modelData.at[0]
|
|
&& root.targetedRegionY === modelData.at[1]
|
|
&& root.targetedRegionWidth === modelData.size[0]
|
|
&& root.targetedRegionHeight === modelData.size[1])
|
|
|
|
opacity: root.draggedAway ? 0 : root.contentRegionOpacity
|
|
borderColor: root.imageBorderColor
|
|
fillColor: targeted ? root.imageFillColor : "transparent"
|
|
text: Translation.tr("Content region")
|
|
}
|
|
}
|
|
|
|
// Controls
|
|
Row {
|
|
id: regionSelectionControls
|
|
z: 9999
|
|
anchors {
|
|
horizontalCenter: parent.horizontalCenter
|
|
bottom: parent.bottom
|
|
bottomMargin: -height
|
|
}
|
|
opacity: 0
|
|
Connections {
|
|
target: root
|
|
function onVisibleChanged() {
|
|
if (!visible) return;
|
|
regionSelectionControls.anchors.bottomMargin = 8;
|
|
regionSelectionControls.opacity = 1;
|
|
}
|
|
}
|
|
Behavior on opacity {
|
|
animation: Appearance.animation.elementMoveFast.numberAnimation.createObject(this)
|
|
}
|
|
Behavior on anchors.bottomMargin {
|
|
animation: Appearance.animation.elementMove.numberAnimation.createObject(this)
|
|
}
|
|
spacing: 6
|
|
|
|
OptionsToolbar {
|
|
Synchronizer on action {
|
|
property alias source: root.action
|
|
}
|
|
Synchronizer on selectionMode {
|
|
property alias source: root.selectionMode
|
|
}
|
|
onDismiss: root.dismiss();
|
|
}
|
|
Item {
|
|
anchors {
|
|
verticalCenter: parent.verticalCenter
|
|
}
|
|
implicitWidth: closeFab.implicitWidth
|
|
implicitHeight: closeFab.implicitHeight
|
|
StyledRectangularShadow {
|
|
target: closeFab
|
|
radius: closeFab.buttonRadius
|
|
}
|
|
FloatingActionButton {
|
|
id: closeFab
|
|
baseSize: 48
|
|
iconText: "close"
|
|
onClicked: root.dismiss();
|
|
StyledToolTip {
|
|
text: Translation.tr("Close")
|
|
}
|
|
colBackground: Appearance.colors.colTertiaryContainer
|
|
colBackgroundHover: Appearance.colors.colTertiaryContainerHover
|
|
colRipple: Appearance.colors.colTertiaryContainerActive
|
|
colOnBackground: Appearance.colors.colOnTertiaryContainer
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
}
|