Compare commits
50 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 1d0ee72498 | |||
| d0a3a0a269 | |||
| 9a026b2484 | |||
| 0305f5a2be | |||
| 6e43bac52b | |||
| d098375da6 | |||
| 239de807d4 | |||
| d4a53b06e0 | |||
| 6d0813089a | |||
| 4005e197eb | |||
| ef1bcf6c73 | |||
| ba67e56fda | |||
| f22c08991c | |||
| 8323bc31a0 | |||
| fa87789fcd | |||
| 6209264744 | |||
| 8e14993633 | |||
| 36fb925495 | |||
| 1a72757e41 | |||
| 90e0987f22 | |||
| afa3b0e3c4 | |||
| 41c9d9e9b4 | |||
| d92e5b4cd7 | |||
| 3963a48a9d | |||
| bd576e17dc | |||
| 54e8a265d0 | |||
| ae2a349247 | |||
| 93ffe6ca2d | |||
| e33901b23c | |||
|
7d4f563b43
|
|||
| 74ce5bb868 | |||
|
439aa9ed1e
|
|||
|
697de725fb
|
|||
| f8a10698ea | |||
|
e5936aa730
|
|||
| a3f38e6414 | |||
| a2505ee875 | |||
| f475e43c54 | |||
| c33d6ae2dd | |||
| ca19a60e5c | |||
| 6aedf6f8b7 | |||
|
233ea3efb9
|
|||
|
d19eead1f5
|
|||
| 4ea74ed516 | |||
| f00af9d70f | |||
| de11767d3b | |||
|
d0b2a5fc1d
|
|||
|
32acfa6b9f
|
|||
|
17fcf1a02c
|
|||
|
1c11549811
|
@@ -13,3 +13,5 @@ uv.lock
|
|||||||
.qtcreator/
|
.qtcreator/
|
||||||
dist/
|
dist/
|
||||||
**/target/
|
**/target/
|
||||||
|
**/test-plugins/
|
||||||
|
**/Charts/
|
||||||
|
|||||||
@@ -59,6 +59,8 @@ JsonObject {
|
|||||||
}
|
}
|
||||||
property int rounding: 8
|
property int rounding: 8
|
||||||
property int smoothing: 32
|
property int smoothing: 32
|
||||||
|
property Tray tray: Tray {
|
||||||
|
}
|
||||||
|
|
||||||
component Popouts: JsonObject {
|
component Popouts: JsonObject {
|
||||||
property bool activeWindow: true
|
property bool activeWindow: true
|
||||||
@@ -69,4 +71,7 @@ JsonObject {
|
|||||||
property bool tray: true
|
property bool tray: true
|
||||||
property bool upower: true
|
property bool upower: true
|
||||||
}
|
}
|
||||||
|
component Tray: JsonObject {
|
||||||
|
property int trayIconSize: 24
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+11
-4
@@ -100,6 +100,9 @@ Singleton {
|
|||||||
border: barConfig.border,
|
border: barConfig.border,
|
||||||
smoothing: barConfig.smoothing,
|
smoothing: barConfig.smoothing,
|
||||||
height: barConfig.height,
|
height: barConfig.height,
|
||||||
|
tray: {
|
||||||
|
trayIconSize: barConfig.tray.trayIconSize
|
||||||
|
},
|
||||||
popouts: {
|
popouts: {
|
||||||
tray: barConfig.popouts.tray,
|
tray: barConfig.popouts.tray,
|
||||||
audio: barConfig.popouts.audio,
|
audio: barConfig.popouts.audio,
|
||||||
@@ -214,6 +217,10 @@ Singleton {
|
|||||||
},
|
},
|
||||||
idle: {
|
idle: {
|
||||||
timeouts: general.idle.timeouts
|
timeouts: general.idle.timeouts
|
||||||
|
},
|
||||||
|
battery: {
|
||||||
|
popupThresholds: general.battery.popupThresholds,
|
||||||
|
critPerc: general.battery.critPerc
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -293,10 +300,10 @@ Singleton {
|
|||||||
return {
|
return {
|
||||||
enable_pp: screenshot.enable_pp,
|
enable_pp: screenshot.enable_pp,
|
||||||
mode: screenshot.mode,
|
mode: screenshot.mode,
|
||||||
corner_radius: screenshot.corner_radius,
|
radius: screenshot.radius,
|
||||||
drop_shadow: screenshot.drop_shadow,
|
shadow: screenshot.shadow,
|
||||||
rounded_corners: screenshot.rounded_corners,
|
rounding: screenshot.rounding,
|
||||||
shadow_blur_radius: screenshot.shadow_blur_radius,
|
shadow_blur: screenshot.shadow_blur,
|
||||||
shadow_color: screenshot.shadow_color,
|
shadow_color: screenshot.shadow_color,
|
||||||
shadow_offset_x: screenshot.shadow_offset_x,
|
shadow_offset_x: screenshot.shadow_offset_x,
|
||||||
shadow_offset_y: screenshot.shadow_offset_y
|
shadow_offset_y: screenshot.shadow_offset_y
|
||||||
|
|||||||
@@ -4,6 +4,8 @@ import Quickshell
|
|||||||
JsonObject {
|
JsonObject {
|
||||||
property Apps apps: Apps {
|
property Apps apps: Apps {
|
||||||
}
|
}
|
||||||
|
property Battery battery: Battery {
|
||||||
|
}
|
||||||
property Color color: Color {
|
property Color color: Color {
|
||||||
}
|
}
|
||||||
property string dateFormat: "ddd d MMM - hh:mm:ss"
|
property string dateFormat: "ddd d MMM - hh:mm:ss"
|
||||||
@@ -19,6 +21,17 @@ JsonObject {
|
|||||||
property list<string> playback: ["mpv"]
|
property list<string> playback: ["mpv"]
|
||||||
property list<string> terminal: ["kitty"]
|
property list<string> terminal: ["kitty"]
|
||||||
}
|
}
|
||||||
|
component Battery: JsonObject {
|
||||||
|
property int critPerc: 5
|
||||||
|
property list<var> popupThresholds: [
|
||||||
|
{
|
||||||
|
perc: 20,
|
||||||
|
name: qsTr("Low battery"),
|
||||||
|
message: qsTr("Battery is low"),
|
||||||
|
icon: "battery_android_frame_2"
|
||||||
|
},
|
||||||
|
]
|
||||||
|
}
|
||||||
component Color: JsonObject {
|
component Color: JsonObject {
|
||||||
property int hyprsunsetTemp: 5000
|
property int hyprsunsetTemp: 5000
|
||||||
property string mode: "dark"
|
property string mode: "dark"
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
import Quickshell.Io
|
import Quickshell.Io
|
||||||
|
|
||||||
JsonObject {
|
JsonObject {
|
||||||
property real corner_radius: 12.0
|
|
||||||
property bool drop_shadow: true
|
|
||||||
property bool enable_pp: true
|
property bool enable_pp: true
|
||||||
property string mode: "manual"
|
property string mode: "manual"
|
||||||
property bool rounded_corners: false
|
property real radius: 12.0
|
||||||
property real shadow_blur_radius: 22.0
|
property bool rounding: false
|
||||||
|
property bool shadow: true
|
||||||
|
property real shadow_blur: 22.0
|
||||||
property list<int> shadow_color: [0, 0, 0, 160]
|
property list<int> shadow_color: [0, 0, 0, 160]
|
||||||
property real shadow_offset_x: 5.0
|
property real shadow_offset_x: 5.0
|
||||||
property real shadow_offset_y: 5.0
|
property real shadow_offset_y: 5.0
|
||||||
|
|||||||
@@ -0,0 +1,47 @@
|
|||||||
|
import Quickshell
|
||||||
|
import Quickshell.Services.UPower
|
||||||
|
import QtQuick
|
||||||
|
import ZShell
|
||||||
|
import qs.Config
|
||||||
|
import qs.Components.Toast
|
||||||
|
|
||||||
|
Scope {
|
||||||
|
id: root
|
||||||
|
|
||||||
|
readonly property real currentPerc: UPower.displayDevice.percentage
|
||||||
|
readonly property list<var> popupThresholds: [...Config.general.battery.popupThresholds].sort((a, b) => b.perc - a.perc)
|
||||||
|
|
||||||
|
Connections {
|
||||||
|
function onOnBatteryChanged(): void {
|
||||||
|
if (UPower.onBattery) {
|
||||||
|
if (Config.utilities.toasts.chargingChanged)
|
||||||
|
Toaster.toast(qsTr("Charger unplugged"), qsTr("Battery is discharging"), "power_off");
|
||||||
|
} else {
|
||||||
|
if (Config.utilities.toasts.chargingChanged)
|
||||||
|
Toaster.toast(qsTr("Charger plugged in"), qsTr("Battery is charging"), "power");
|
||||||
|
for (const level of root.popupThresholds)
|
||||||
|
level.warned = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
target: UPower
|
||||||
|
}
|
||||||
|
|
||||||
|
Connections {
|
||||||
|
function onPercentageChanged(): void {
|
||||||
|
if (!UPower.onBattery)
|
||||||
|
return;
|
||||||
|
|
||||||
|
const p = UPower.displayDevice.percentage * 100;
|
||||||
|
for (const perc of root.popupThresholds) {
|
||||||
|
if (p <= perc.perc && !perc.warned) {
|
||||||
|
perc.warned = true;
|
||||||
|
console.log(perc.warned + "\n" + [...Config.general.battery.popupThresholds][0].warned);
|
||||||
|
Toaster.toast(perc.title ?? qsTr("Battery warning"), perc.message ?? qsTr("Battery perc is low"), perc.icon ?? "battery_android_alert", perc.critical ? Toast.Error : Toast.Warning);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
target: UPower.displayDevice
|
||||||
|
}
|
||||||
|
}
|
||||||
+32
-57
@@ -179,6 +179,8 @@ Singleton {
|
|||||||
property string appIcon
|
property string appIcon
|
||||||
property string appName
|
property string appName
|
||||||
property string body
|
property string body
|
||||||
|
property string cachedImageSource: ""
|
||||||
|
property bool cachingImage: false
|
||||||
property bool closed
|
property bool closed
|
||||||
readonly property Connections conn: Connections {
|
readonly property Connections conn: Connections {
|
||||||
function onActionsChanged(): void {
|
function onActionsChanged(): void {
|
||||||
@@ -214,9 +216,9 @@ Singleton {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function onImageChanged(): void {
|
function onImageChanged(): void {
|
||||||
notif.image = notif.notification.image;
|
notif.imageSource = notif.notification.image || "";
|
||||||
if (notif.notification?.image)
|
notif.image = notif.imageSource;
|
||||||
notif.dummyImageLoader.active = true;
|
notif.cacheImageIfNeeded();
|
||||||
}
|
}
|
||||||
|
|
||||||
function onResidentChanged(): void {
|
function onResidentChanged(): void {
|
||||||
@@ -233,60 +235,12 @@ Singleton {
|
|||||||
|
|
||||||
target: notif.notification
|
target: notif.notification
|
||||||
}
|
}
|
||||||
readonly property LazyLoader dummyImageLoader: LazyLoader {
|
|
||||||
active: false
|
|
||||||
|
|
||||||
PanelWindow {
|
|
||||||
color: "transparent"
|
|
||||||
implicitHeight: Config.notifs.sizes.image
|
|
||||||
implicitWidth: Config.notifs.sizes.image
|
|
||||||
|
|
||||||
mask: Region {
|
|
||||||
}
|
|
||||||
|
|
||||||
Image {
|
|
||||||
function tryCache(): void {
|
|
||||||
if (status !== Image.Ready || width != Config.notifs.sizes.image || height != Config.notifs.sizes.image)
|
|
||||||
return;
|
|
||||||
|
|
||||||
const cacheKey = notif.appName + notif.summary + notif.id;
|
|
||||||
let h1 = 0xdeadbeef, h2 = 0x41c6ce57, ch;
|
|
||||||
for (let i = 0; i < cacheKey.length; i++) {
|
|
||||||
ch = cacheKey.charCodeAt(i);
|
|
||||||
h1 = Math.imul(h1 ^ ch, 2654435761);
|
|
||||||
h2 = Math.imul(h2 ^ ch, 1597334677);
|
|
||||||
}
|
|
||||||
h1 = Math.imul(h1 ^ (h1 >>> 16), 2246822507);
|
|
||||||
h1 ^= Math.imul(h2 ^ (h2 >>> 13), 3266489909);
|
|
||||||
h2 = Math.imul(h2 ^ (h2 >>> 16), 2246822507);
|
|
||||||
h2 ^= Math.imul(h1 ^ (h1 >>> 13), 3266489909);
|
|
||||||
const hash = (h2 >>> 0).toString(16).padStart(8, 0) + (h1 >>> 0).toString(16).padStart(8, 0);
|
|
||||||
|
|
||||||
const cache = `${Paths.notifimagecache}/${hash}.png`;
|
|
||||||
ZShellIo.saveItem(this, Qt.resolvedUrl(cache), () => {
|
|
||||||
notif.image = cache;
|
|
||||||
notif.dummyImageLoader.active = false;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
anchors.fill: parent
|
|
||||||
asynchronous: true
|
|
||||||
cache: false
|
|
||||||
fillMode: Image.PreserveAspectCrop
|
|
||||||
opacity: 0
|
|
||||||
source: Qt.resolvedUrl(notif.image)
|
|
||||||
|
|
||||||
onHeightChanged: tryCache()
|
|
||||||
onStatusChanged: tryCache()
|
|
||||||
onWidthChanged: tryCache()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
property real expireTimeout: 5
|
property real expireTimeout: 5
|
||||||
property bool hasActionIcons
|
property bool hasActionIcons
|
||||||
property string id
|
|
||||||
property string image
|
property string image
|
||||||
|
property string imageSource
|
||||||
property var locks: new Set()
|
property var locks: new Set()
|
||||||
|
property string notifId
|
||||||
property Notification notification
|
property Notification notification
|
||||||
property bool popup
|
property bool popup
|
||||||
property bool resident
|
property bool resident
|
||||||
@@ -329,6 +283,26 @@ Singleton {
|
|||||||
}
|
}
|
||||||
property int urgency: NotificationUrgency.Normal
|
property int urgency: NotificationUrgency.Normal
|
||||||
|
|
||||||
|
function cacheImageIfNeeded(): void {
|
||||||
|
const source = imageSource;
|
||||||
|
|
||||||
|
if (!source || cachingImage)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (cachedImageSource === source)
|
||||||
|
return;
|
||||||
|
|
||||||
|
cachingImage = true;
|
||||||
|
|
||||||
|
ZShellIo.cacheImage(Qt.resolvedUrl(source), Paths.notifimagecache, (path, url) => {
|
||||||
|
cachedImageSource = source;
|
||||||
|
image = path;
|
||||||
|
cachingImage = false;
|
||||||
|
}, () => {
|
||||||
|
cachingImage = false;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
function close(): void {
|
function close(): void {
|
||||||
closed = true;
|
closed = true;
|
||||||
if (locks.size === 0 && root.list.includes(this)) {
|
if (locks.size === 0 && root.list.includes(this)) {
|
||||||
@@ -352,14 +326,13 @@ Singleton {
|
|||||||
if (!notification)
|
if (!notification)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
id = notification.id;
|
notifId = notification.id;
|
||||||
summary = notification.summary;
|
summary = notification.summary;
|
||||||
body = notification.body;
|
body = notification.body;
|
||||||
appIcon = notification.appIcon;
|
appIcon = notification.appIcon;
|
||||||
appName = notification.appName;
|
appName = notification.appName;
|
||||||
image = notification.image;
|
imageSource = notification.image || "";
|
||||||
if (notification?.image)
|
image = imageSource;
|
||||||
dummyImageLoader.active = true;
|
|
||||||
expireTimeout = notification.expireTimeout;
|
expireTimeout = notification.expireTimeout;
|
||||||
urgency = notification.urgency;
|
urgency = notification.urgency;
|
||||||
resident = notification.resident;
|
resident = notification.resident;
|
||||||
@@ -369,6 +342,8 @@ Singleton {
|
|||||||
text: a.text,
|
text: a.text,
|
||||||
invoke: () => a.invoke()
|
invoke: () => a.invoke()
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
cacheImageIfNeeded();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,107 +0,0 @@
|
|||||||
import Quickshell
|
|
||||||
import QtQuick
|
|
||||||
import QtQuick.Shapes
|
|
||||||
import qs.Components
|
|
||||||
import qs.Config
|
|
||||||
import qs.Modules as Modules
|
|
||||||
import qs.Modules.Notifications as Notifications
|
|
||||||
import qs.Modules.Notifications.Sidebar as Sidebar
|
|
||||||
import qs.Modules.Notifications.Sidebar.Utils as Utils
|
|
||||||
import qs.Modules.Dashboard as Dashboard
|
|
||||||
import qs.Modules.Osd as Osd
|
|
||||||
import qs.Modules.Launcher as Launcher
|
|
||||||
import qs.Modules.Resources as Resources
|
|
||||||
import qs.Modules.Drawing as Drawing
|
|
||||||
import qs.Modules.Settings as Settings
|
|
||||||
import qs.Modules.Dock as Dock
|
|
||||||
|
|
||||||
Shape {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
required property Item bar
|
|
||||||
required property Panels panels
|
|
||||||
required property PersistentProperties visibilities
|
|
||||||
|
|
||||||
anchors.fill: parent
|
|
||||||
anchors.margins: Config.barConfig.border
|
|
||||||
anchors.topMargin: bar.implicitHeight
|
|
||||||
asynchronous: true
|
|
||||||
preferredRendererType: Shape.CurveRenderer
|
|
||||||
|
|
||||||
Drawing.Background {
|
|
||||||
startX: 0
|
|
||||||
startY: wrapper.y - rounding
|
|
||||||
wrapper: root.panels.drawing
|
|
||||||
}
|
|
||||||
|
|
||||||
Resources.Background {
|
|
||||||
startX: 0 - rounding
|
|
||||||
startY: 0
|
|
||||||
wrapper: root.panels.resources
|
|
||||||
}
|
|
||||||
|
|
||||||
Osd.Background {
|
|
||||||
startX: root.width - root.panels.sidebar.width
|
|
||||||
startY: (root.height - wrapper.height) / 2 - rounding
|
|
||||||
wrapper: root.panels.osd
|
|
||||||
}
|
|
||||||
|
|
||||||
Modules.Background {
|
|
||||||
invertBottomRounding: wrapper.x <= 0
|
|
||||||
rounding: root.panels.popouts.currentName.startsWith("updates") || root.panels.popouts.currentName.startsWith("audio") ? Appearance.rounding.normal : Appearance.rounding.smallest
|
|
||||||
startX: wrapper.x - rounding
|
|
||||||
startY: wrapper.y
|
|
||||||
wrapper: root.panels.popouts
|
|
||||||
}
|
|
||||||
|
|
||||||
Notifications.Background {
|
|
||||||
sidebar: sidebar
|
|
||||||
startX: root.width
|
|
||||||
startY: 0
|
|
||||||
wrapper: root.panels.notifications
|
|
||||||
}
|
|
||||||
|
|
||||||
Launcher.Background {
|
|
||||||
startX: (root.width - wrapper.width) / 2 - rounding
|
|
||||||
startY: root.height
|
|
||||||
wrapper: root.panels.launcher
|
|
||||||
}
|
|
||||||
|
|
||||||
Dashboard.Background {
|
|
||||||
startX: root.width - root.panels.dashboard.width - rounding
|
|
||||||
startY: 0
|
|
||||||
wrapper: root.panels.dashboard
|
|
||||||
}
|
|
||||||
|
|
||||||
Utils.Background {
|
|
||||||
sidebar: sidebar
|
|
||||||
startX: root.width
|
|
||||||
startY: root.height
|
|
||||||
wrapper: root.panels.utilities
|
|
||||||
}
|
|
||||||
|
|
||||||
Sidebar.Background {
|
|
||||||
id: sidebar
|
|
||||||
|
|
||||||
panels: root.panels
|
|
||||||
startX: root.width
|
|
||||||
startY: root.panels.notifications.height
|
|
||||||
wrapper: root.panels.sidebar
|
|
||||||
}
|
|
||||||
|
|
||||||
Settings.Background {
|
|
||||||
id: settings
|
|
||||||
|
|
||||||
startX: (root.width - wrapper.width) / 2 - rounding
|
|
||||||
startY: 0
|
|
||||||
wrapper: root.panels.settings
|
|
||||||
}
|
|
||||||
|
|
||||||
Dock.Background {
|
|
||||||
id: dock
|
|
||||||
|
|
||||||
startX: (root.width - wrapper.width) / 2 - rounding
|
|
||||||
startY: root.height
|
|
||||||
wrapper: root.panels.dock
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -23,6 +23,7 @@ Canvas {
|
|||||||
ctx.save();
|
ctx.save();
|
||||||
ctx.lineWidth = root.penWidth;
|
ctx.lineWidth = root.penWidth;
|
||||||
ctx.strokeStyle = root.penColor;
|
ctx.strokeStyle = root.penColor;
|
||||||
|
ctx.lineJoin = "round";
|
||||||
ctx.lineCap = "round";
|
ctx.lineCap = "round";
|
||||||
ctx.beginPath();
|
ctx.beginPath();
|
||||||
ctx.moveTo(points[0].x, points[0].y);
|
ctx.moveTo(points[0].x, points[0].y);
|
||||||
|
|||||||
@@ -22,20 +22,20 @@ CustomMouseArea {
|
|||||||
}
|
}
|
||||||
|
|
||||||
acceptedButtons: Qt.LeftButton | Qt.RightButton
|
acceptedButtons: Qt.LeftButton | Qt.RightButton
|
||||||
anchors.fill: root.visibilities.isDrawing ? parent : undefined
|
enabled: z > 0
|
||||||
hoverEnabled: true
|
hoverEnabled: true
|
||||||
visible: root.visibilities.isDrawing
|
visible: root.visibilities.isDrawing
|
||||||
|
|
||||||
onPositionChanged: event => {
|
onPositionChanged: event => {
|
||||||
const x = event.x;
|
const x = event.x;
|
||||||
const y = event.y;
|
const y = event.y;
|
||||||
|
|
||||||
if (root.visibilities.isDrawing && (event.buttons & Qt.LeftButton)) {
|
if (root.visibilities.isDrawing && (event.buttons & Qt.LeftButton)) {
|
||||||
root.drawing.points.push(Qt.point(x, y));
|
root.drawing.points.push(Qt.point(x, y));
|
||||||
root.drawing.requestPaint();
|
root.drawing.requestPaint();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (root.inLeftPanel(root.popout, x, y)) {
|
if (!(event.buttons & Qt.LeftButton) && root.inLeftPanel(root.popout, x, y)) {
|
||||||
root.z = -2;
|
root.z = -2;
|
||||||
root.panels.drawing.expanded = true;
|
root.panels.drawing.expanded = true;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -78,7 +78,7 @@ CustomMouseArea {
|
|||||||
const dragY = y - dragStart.y;
|
const dragY = y - dragStart.y;
|
||||||
|
|
||||||
if (root.visibilities.isDrawing && !root.inLeftPanel(root.panels.drawing, x, y)) {
|
if (root.visibilities.isDrawing && !root.inLeftPanel(root.panels.drawing, x, y)) {
|
||||||
// root.input.z = 2;
|
root.input.z = 2;
|
||||||
root.panels.drawing.expanded = false;
|
root.panels.drawing.expanded = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -132,6 +132,8 @@ CustomMouseArea {
|
|||||||
if (!inDashboardArea) {
|
if (!inDashboardArea) {
|
||||||
root.dashboardShortcutActive = true;
|
root.dashboardShortcutActive = true;
|
||||||
}
|
}
|
||||||
|
if (root.panels.launcher.x + root.panels.launcher.width > root.panels.dashboardWrapper.x)
|
||||||
|
root.visibilities.launcher = false;
|
||||||
|
|
||||||
root.visibilities.settings = false;
|
root.visibilities.settings = false;
|
||||||
root.visibilities.sidebar = false;
|
root.visibilities.sidebar = false;
|
||||||
@@ -161,6 +163,11 @@ CustomMouseArea {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (root.visibilities.launcher) {
|
if (root.visibilities.launcher) {
|
||||||
|
if (root.panels.dashboardWrapper.x < root.panels.launcher.x + root.panels.launcher.width) {
|
||||||
|
console.log("true");
|
||||||
|
root.visibilities.dashboard = false;
|
||||||
|
}
|
||||||
|
|
||||||
root.visibilities.dock = false;
|
root.visibilities.dock = false;
|
||||||
root.visibilities.settings = false;
|
root.visibilities.settings = false;
|
||||||
}
|
}
|
||||||
|
|||||||
+26
-13
@@ -35,7 +35,7 @@ Variants {
|
|||||||
property var root: Quickshell.shellDir
|
property var root: Quickshell.shellDir
|
||||||
|
|
||||||
WlrLayershell.exclusionMode: ExclusionMode.Ignore
|
WlrLayershell.exclusionMode: ExclusionMode.Ignore
|
||||||
WlrLayershell.keyboardFocus: visibilities.dock || visibilities.launcher || visibilities.sidebar || visibilities.dashboard || visibilities.settings || visibilities.resources ? WlrKeyboardFocus.OnDemand : WlrKeyboardFocus.None
|
// WlrLayershell.keyboardFocus: visibilities.dock || visibilities.launcher || visibilities.sidebar || visibilities.dashboard || visibilities.settings || visibilities.resources ? WlrKeyboardFocus.OnDemand : WlrKeyboardFocus.None
|
||||||
color: "transparent"
|
color: "transparent"
|
||||||
contentItem.focus: true
|
contentItem.focus: true
|
||||||
mask: visibilities.isDrawing ? null : region
|
mask: visibilities.isDrawing ? null : region
|
||||||
@@ -229,6 +229,7 @@ Variants {
|
|||||||
id: notifsBg
|
id: notifsBg
|
||||||
|
|
||||||
panel: panels.notifications
|
panel: panels.notifications
|
||||||
|
radius: Appearance.rounding.normal
|
||||||
}
|
}
|
||||||
|
|
||||||
PanelBg {
|
PanelBg {
|
||||||
@@ -304,22 +305,34 @@ Variants {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Drawing {
|
Loader {
|
||||||
id: drawing
|
id: drawingLoader
|
||||||
|
|
||||||
|
active: visibilities.isDrawing
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
z: 2
|
z: 2
|
||||||
|
|
||||||
|
sourceComponent: Drawing {
|
||||||
|
id: drawing
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
DrawingInput {
|
Loader {
|
||||||
id: input
|
id: inputLoader
|
||||||
|
|
||||||
bar: bar
|
active: visibilities.isDrawing
|
||||||
drawing: drawing
|
anchors.fill: parent
|
||||||
panels: panels
|
|
||||||
popout: panels.drawing
|
|
||||||
visibilities: visibilities
|
|
||||||
z: 2
|
z: 2
|
||||||
|
|
||||||
|
sourceComponent: DrawingInput {
|
||||||
|
id: input
|
||||||
|
|
||||||
|
bar: bar
|
||||||
|
drawing: drawingLoader.item
|
||||||
|
panels: panels
|
||||||
|
popout: panels.drawing
|
||||||
|
visibilities: visibilities
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Interactions {
|
Interactions {
|
||||||
@@ -327,8 +340,8 @@ Variants {
|
|||||||
|
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
bar: bar
|
bar: bar
|
||||||
drawing: drawing
|
drawing: drawingLoader.item
|
||||||
input: input
|
input: inputLoader.item
|
||||||
panels: panels
|
panels: panels
|
||||||
popouts: panels.popouts
|
popouts: panels.popouts
|
||||||
screen: scope.modelData
|
screen: scope.modelData
|
||||||
@@ -339,7 +352,7 @@ Variants {
|
|||||||
id: panels
|
id: panels
|
||||||
|
|
||||||
bar: bar
|
bar: bar
|
||||||
drawingItem: drawing
|
drawingItem: drawingLoader.item
|
||||||
screen: scope.modelData
|
screen: scope.modelData
|
||||||
visibilities: visibilities
|
visibilities: visibilities
|
||||||
|
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ Singleton {
|
|||||||
PersistentProperties {
|
PersistentProperties {
|
||||||
id: props
|
id: props
|
||||||
|
|
||||||
property bool enabled: Hypr.options["animations:enabled"] === 0
|
property bool enabled: Hypr.options.animations.enabled === 0
|
||||||
|
|
||||||
reloadableId: "gamemode"
|
reloadableId: "gamemode"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -158,6 +158,5 @@ Singleton {
|
|||||||
|
|
||||||
HyprExtras {
|
HyprExtras {
|
||||||
id: extras
|
id: extras
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+8
-3
@@ -30,12 +30,17 @@ MouseArea {
|
|||||||
property real ey: screen.height
|
property real ey: screen.height
|
||||||
required property LazyLoader loader
|
required property LazyLoader loader
|
||||||
property bool onClient
|
property bool onClient
|
||||||
property real realBorderWidth: onClient ? (Hypr.options["general:border_size"] ?? 1) : 2
|
property real realBorderWidth: onClient ? (Hypr.options.general.border_size ?? 1) : 2
|
||||||
property real realRounding: onClient ? (Hypr.options["decoration:rounding"] ?? 0) : 0
|
property real realRounding: onClient ? (Hypr.options.decoration.rounding ?? 0) : 0
|
||||||
property real rsx: Math.min(sx, ex)
|
property real rsx: Math.min(sx, ex)
|
||||||
property real rsy: Math.min(sy, ey)
|
property real rsy: Math.min(sy, ey)
|
||||||
|
readonly property real scaleRatio: Hypr.monitorFor(screen).scale
|
||||||
required property ShellScreen screen
|
required property ShellScreen screen
|
||||||
property real sh: Math.abs(sy - ey)
|
property real sh: Math.abs(sy - ey)
|
||||||
|
readonly property color shadowColor: Hypr.options.decoration.shadow.color
|
||||||
|
readonly property var shadowOffset: Hypr.options.decoration.shadow.offset
|
||||||
|
readonly property int shadowRange: Hypr.options.decoration.shadow.range
|
||||||
|
readonly property int shadowRenderPower: Hypr.options.decoration.shadow.render_power
|
||||||
property real ssx
|
property real ssx
|
||||||
property real ssy
|
property real ssy
|
||||||
property real sw: Math.abs(sx - ex)
|
property real sw: Math.abs(sx - ex)
|
||||||
@@ -66,7 +71,7 @@ MouseArea {
|
|||||||
|
|
||||||
function save(): void {
|
function save(): void {
|
||||||
const tmpfile = Qt.resolvedUrl(`/tmp/zshell-picker-${Quickshell.processId}-${Date.now()}.png`);
|
const tmpfile = Qt.resolvedUrl(`/tmp/zshell-picker-${Quickshell.processId}-${Date.now()}.png`);
|
||||||
const cmd = Config.screenshot.enable_pp ? ["zshell-img-tools", "--image"] : ["swappy", "-f"];
|
const cmd = Config.screenshot.enable_pp ? ["zshell-img-tools", "--scale", root.scaleRatio, "--shadow-blur-radius", root.shadowRange, "--image"] : ["swappy", "-f"];
|
||||||
ZShellIo.saveItem(screencopy, tmpfile, Qt.rect(Math.ceil(rsx), Math.ceil(rsy), Math.floor(sw), Math.floor(sh)), path => Quickshell.execDetached([...cmd, path]));
|
ZShellIo.saveItem(screencopy, tmpfile, Qt.rect(Math.ceil(rsx), Math.ceil(rsy), Math.floor(sw), Math.floor(sh)), path => Quickshell.execDetached([...cmd, path]));
|
||||||
closeAnim.start();
|
closeAnim.start();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ import Quickshell
|
|||||||
Singleton {
|
Singleton {
|
||||||
property var extraOpts: ({})
|
property var extraOpts: ({})
|
||||||
readonly property list<var> fuzzyPrepped: useFuzzy ? list.map(e => {
|
readonly property list<var> fuzzyPrepped: useFuzzy ? list.map(e => {
|
||||||
console.log(useFuzzy);
|
|
||||||
const obj = {
|
const obj = {
|
||||||
_item: e
|
_item: e
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -17,6 +17,7 @@ Singleton {
|
|||||||
property var disks: []
|
property var disks: []
|
||||||
property real gpuMemTotal: 0
|
property real gpuMemTotal: 0
|
||||||
property real gpuMemUsed
|
property real gpuMemUsed
|
||||||
|
property string gpuName
|
||||||
property real gpuPerc
|
property real gpuPerc
|
||||||
property real gpuTemp
|
property real gpuTemp
|
||||||
readonly property string gpuType: Config.services.gpuType.toUpperCase() || autoGpuType
|
readonly property string gpuType: Config.services.gpuType.toUpperCase() || autoGpuType
|
||||||
|
|||||||
@@ -53,14 +53,12 @@ Searcher {
|
|||||||
};
|
};
|
||||||
|
|
||||||
root.crops = updated;
|
root.crops = updated;
|
||||||
monitorCrops.writeAdapter();
|
|
||||||
monitorCrops.reload();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function setWallpaper(path: string): void {
|
function setWallpaper(path: string): void {
|
||||||
actualCurrent = path;
|
actualCurrent = path;
|
||||||
WallpaperPath.currentWallpaperPath = path;
|
WallpaperPath.currentWallpaperPath = path;
|
||||||
Quickshell.screens.forEach(n => setCrop(n.name, Qt.rect(0, 0, 0, 0), Qt.rect(0, 0, 0, 0), 1.0));
|
Quickshell.screens.forEach(n => setCrop(n.name, Qt.rect(0, 0, 1, 1), Qt.rect(0, 0, 0, 0), 1.0));
|
||||||
Quickshell.execDetached(["zshell-cli", "wallpaper", "lockscreen", "--input-image", `${root.actualCurrent}`, "--output-path", `${Paths.state}/lockscreen_bg.png`, "--blur-amount", `${Config.lock.blurAmount}`]);
|
Quickshell.execDetached(["zshell-cli", "wallpaper", "lockscreen", "--input-image", `${root.actualCurrent}`, "--output-path", `${Paths.state}/lockscreen_bg.png`, "--blur-amount", `${Config.lock.blurAmount}`]);
|
||||||
if (Config.general.color.schemeGeneration)
|
if (Config.general.color.schemeGeneration)
|
||||||
Quickshell.execDetached(["zshell-cli", "scheme", "generate", "--image-path", `${root.actualCurrent}`, "--scheme", `${Config.colors.schemeType}`, "--mode", `${Config.general.color.mode}`]);
|
Quickshell.execDetached(["zshell-cli", "scheme", "generate", "--image-path", `${root.actualCurrent}`, "--scheme", `${Config.colors.schemeType}`, "--mode", `${Config.general.color.mode}`]);
|
||||||
|
|||||||
@@ -1,68 +0,0 @@
|
|||||||
import QtQuick
|
|
||||||
import QtQuick.Shapes
|
|
||||||
import qs.Components
|
|
||||||
import qs.Config
|
|
||||||
|
|
||||||
ShapePath {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
readonly property bool flatten: wrapper.height < rounding * 2
|
|
||||||
property real ibr: invertBottomRounding ? -1 : 1
|
|
||||||
required property bool invertBottomRounding
|
|
||||||
property real rounding: Appearance.rounding.smallest
|
|
||||||
readonly property real roundingY: flatten ? wrapper.height / 2 : rounding
|
|
||||||
required property Wrapper wrapper
|
|
||||||
|
|
||||||
fillColor: DynamicColors.palette.m3surface
|
|
||||||
strokeWidth: -1
|
|
||||||
|
|
||||||
Behavior on fillColor {
|
|
||||||
CAnim {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: 0
|
|
||||||
relativeY: root.wrapper.height - root.roundingY - root.roundingY * root.ibr
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
direction: root.invertBottomRounding ? PathArc.Clockwise : PathArc.Counterclockwise
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: root.roundingY * root.ibr
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: root.wrapper.width - root.rounding * 2
|
|
||||||
relativeY: 0
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
direction: PathArc.Counterclockwise
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: -root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: 0
|
|
||||||
relativeY: -(root.wrapper.height - root.roundingY * 2)
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: -root.roundingY
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,65 +0,0 @@
|
|||||||
import qs.Components
|
|
||||||
import qs.Config
|
|
||||||
import QtQuick
|
|
||||||
import QtQuick.Shapes
|
|
||||||
|
|
||||||
ShapePath {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
readonly property bool flatten: wrapper.height < rounding * 2
|
|
||||||
readonly property real rounding: Appearance.rounding.normal
|
|
||||||
readonly property real roundingY: flatten ? wrapper.height / 2 : rounding
|
|
||||||
required property Wrapper wrapper
|
|
||||||
|
|
||||||
fillColor: DynamicColors.palette.m3surface
|
|
||||||
strokeWidth: -1
|
|
||||||
|
|
||||||
Behavior on fillColor {
|
|
||||||
CAnim {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: 0
|
|
||||||
relativeY: root.wrapper.height - root.roundingY * 2
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
direction: PathArc.Counterclockwise
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: root.wrapper.width - root.rounding * 2
|
|
||||||
relativeY: 0
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: 0
|
|
||||||
relativeY: -(root.wrapper.height)
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: -root.roundingY
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -20,7 +20,7 @@ Item {
|
|||||||
required property PersistentProperties visibilities
|
required property PersistentProperties visibilities
|
||||||
|
|
||||||
implicitHeight: content.implicitHeight
|
implicitHeight: content.implicitHeight
|
||||||
implicitWidth: content.implicitWidth || 854 // Hard coded fallback for first open
|
implicitWidth: content.implicitWidth || 854
|
||||||
opacity: 1 - offsetScale
|
opacity: 1 - offsetScale
|
||||||
visible: offsetScale < 1
|
visible: offsetScale < 1
|
||||||
|
|
||||||
|
|||||||
@@ -1,66 +0,0 @@
|
|||||||
import QtQuick
|
|
||||||
import QtQuick.Shapes
|
|
||||||
import qs.Components
|
|
||||||
import qs.Config
|
|
||||||
|
|
||||||
ShapePath {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
readonly property bool flatten: wrapper.height < rounding * 2
|
|
||||||
readonly property real rounding: Appearance.rounding.normal
|
|
||||||
readonly property real roundingY: flatten ? wrapper.height / 2 : rounding
|
|
||||||
required property Wrapper wrapper
|
|
||||||
|
|
||||||
fillColor: DynamicColors.palette.m3surface
|
|
||||||
strokeWidth: -1
|
|
||||||
|
|
||||||
Behavior on fillColor {
|
|
||||||
CAnim {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
direction: PathArc.Counterclockwise
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: -root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: 0
|
|
||||||
relativeY: -(root.wrapper.height - root.roundingY * 2)
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: -root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: root.wrapper.width - root.rounding * 2
|
|
||||||
relativeY: 0
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: 0
|
|
||||||
relativeY: root.wrapper.height - root.roundingY * 2
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
direction: PathArc.Counterclockwise
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: root.roundingY
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -9,18 +9,17 @@ Item {
|
|||||||
id: root
|
id: root
|
||||||
|
|
||||||
property int contentHeight
|
property int contentHeight
|
||||||
|
property real offsetScale: shouldBeActive ? 0 : 1
|
||||||
required property var panels
|
required property var panels
|
||||||
required property ShellScreen screen
|
required property ShellScreen screen
|
||||||
|
readonly property bool shouldBeActive: visibilities.dock && Config.dock.enable
|
||||||
required property PersistentProperties visibilities
|
required property PersistentProperties visibilities
|
||||||
|
|
||||||
readonly property bool shouldBeActive: visibilities.dock
|
|
||||||
property real offsetScale: shouldBeActive ? 0 : 1
|
|
||||||
|
|
||||||
visible: offsetScale < 1
|
|
||||||
anchors.bottomMargin: (-implicitHeight - 5) * offsetScale
|
anchors.bottomMargin: (-implicitHeight - 5) * offsetScale
|
||||||
implicitHeight: content.implicitHeight
|
implicitHeight: content.implicitHeight
|
||||||
implicitWidth: content.implicitWidth || 400
|
implicitWidth: content.implicitWidth || 400
|
||||||
opacity: 1 - offsetScale
|
opacity: 1 - offsetScale
|
||||||
|
visible: offsetScale < 1
|
||||||
|
|
||||||
Behavior on offsetScale {
|
Behavior on offsetScale {
|
||||||
Anim {
|
Anim {
|
||||||
@@ -32,10 +31,10 @@ Item {
|
|||||||
Loader {
|
Loader {
|
||||||
id: content
|
id: content
|
||||||
|
|
||||||
|
active: root.shouldBeActive || root.visible
|
||||||
anchors.left: parent.left
|
anchors.left: parent.left
|
||||||
anchors.top: parent.top
|
anchors.top: parent.top
|
||||||
|
asynchronous: true
|
||||||
active: root.shouldBeActive || root.visible
|
|
||||||
|
|
||||||
sourceComponent: Content {
|
sourceComponent: Content {
|
||||||
panels: root.panels
|
panels: root.panels
|
||||||
|
|||||||
@@ -1,66 +0,0 @@
|
|||||||
import QtQuick
|
|
||||||
import QtQuick.Shapes
|
|
||||||
import qs.Components
|
|
||||||
import qs.Config
|
|
||||||
|
|
||||||
ShapePath {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
readonly property bool flatten: wrapper.width < rounding * 2
|
|
||||||
readonly property real rounding: Appearance.rounding.normal
|
|
||||||
readonly property real roundingX: flatten ? wrapper.width / 2 : rounding
|
|
||||||
required property Wrapper wrapper
|
|
||||||
|
|
||||||
fillColor: DynamicColors.palette.m3surface
|
|
||||||
strokeWidth: -1
|
|
||||||
|
|
||||||
Behavior on fillColor {
|
|
||||||
CAnim {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
direction: PathArc.Counterclockwise
|
|
||||||
radiusX: Math.min(root.rounding, root.wrapper.width)
|
|
||||||
radiusY: root.rounding
|
|
||||||
relativeX: root.roundingX
|
|
||||||
relativeY: root.rounding
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: root.wrapper.width - root.roundingX * 2
|
|
||||||
relativeY: 0
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: Math.min(root.rounding, root.wrapper.width)
|
|
||||||
radiusY: root.rounding
|
|
||||||
relativeX: root.roundingX
|
|
||||||
relativeY: root.rounding
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: 0
|
|
||||||
relativeY: root.wrapper.height - root.rounding * 2
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: Math.min(root.rounding, root.wrapper.width)
|
|
||||||
radiusY: root.rounding
|
|
||||||
relativeX: -root.roundingX
|
|
||||||
relativeY: root.rounding
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: -(root.wrapper.width - root.roundingX * 2)
|
|
||||||
relativeY: 0
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
direction: PathArc.Counterclockwise
|
|
||||||
radiusX: Math.min(root.rounding, root.wrapper.width)
|
|
||||||
radiusY: root.rounding
|
|
||||||
relativeX: -root.roundingX
|
|
||||||
relativeY: root.rounding
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -44,10 +44,10 @@ Item {
|
|||||||
Loader {
|
Loader {
|
||||||
id: icon
|
id: icon
|
||||||
|
|
||||||
active: Qt.binding(() => root.shouldBeActive || root.visible)
|
active: root.shouldBeActive || root.visible
|
||||||
anchors.right: parent.right
|
anchors.right: parent.right
|
||||||
anchors.verticalCenter: parent.verticalCenter
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
height: content.contentItem.height
|
asynchronous: true
|
||||||
opacity: root.expanded ? 0 : 1
|
opacity: root.expanded ? 0 : 1
|
||||||
|
|
||||||
Behavior on opacity {
|
Behavior on opacity {
|
||||||
@@ -63,8 +63,10 @@ Item {
|
|||||||
Loader {
|
Loader {
|
||||||
id: content
|
id: content
|
||||||
|
|
||||||
|
active: root.shouldBeActive || root.visible
|
||||||
anchors.right: parent.right
|
anchors.right: parent.right
|
||||||
anchors.verticalCenter: parent.verticalCenter
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
|
asynchronous: true
|
||||||
opacity: root.expanded ? 1 : 0
|
opacity: root.expanded ? 1 : 0
|
||||||
|
|
||||||
Behavior on opacity {
|
Behavior on opacity {
|
||||||
@@ -75,7 +77,5 @@ Item {
|
|||||||
drawing: root.drawing
|
drawing: root.drawing
|
||||||
visibilities: root.visibilities
|
visibilities: root.visibilities
|
||||||
}
|
}
|
||||||
|
|
||||||
Component.onCompleted: active = Qt.binding(() => root.shouldBeActive || root.visible)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,66 +0,0 @@
|
|||||||
import QtQuick
|
|
||||||
import QtQuick.Shapes
|
|
||||||
import qs.Components
|
|
||||||
import qs.Config
|
|
||||||
|
|
||||||
ShapePath {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
readonly property bool flatten: wrapper.height < rounding * 2
|
|
||||||
readonly property real rounding: Appearance.rounding.smallest + 5
|
|
||||||
readonly property real roundingY: flatten ? wrapper.height / 2 : rounding
|
|
||||||
required property Wrapper wrapper
|
|
||||||
|
|
||||||
fillColor: DynamicColors.palette.m3surface
|
|
||||||
strokeWidth: -1
|
|
||||||
|
|
||||||
Behavior on fillColor {
|
|
||||||
CAnim {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
direction: PathArc.Counterclockwise
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: -root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: 0
|
|
||||||
relativeY: -(root.wrapper.height - root.roundingY * 2)
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: -root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: root.wrapper.width - root.rounding * 2
|
|
||||||
relativeY: 0
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: 0
|
|
||||||
relativeY: root.wrapper.height - root.roundingY * 2
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
direction: PathArc.Counterclockwise
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: root.roundingY
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -4,6 +4,7 @@ import Quickshell
|
|||||||
import QtQuick
|
import QtQuick
|
||||||
import qs.Components
|
import qs.Components
|
||||||
import qs.Config
|
import qs.Config
|
||||||
|
import qs.Modules.Launcher.Services
|
||||||
|
|
||||||
Item {
|
Item {
|
||||||
id: root
|
id: root
|
||||||
@@ -11,34 +12,24 @@ Item {
|
|||||||
property int contentHeight
|
property int contentHeight
|
||||||
readonly property real maxHeight: {
|
readonly property real maxHeight: {
|
||||||
let max = screen.height - Appearance.spacing.large * 2;
|
let max = screen.height - Appearance.spacing.large * 2;
|
||||||
if (visibilities.resources)
|
if (visibilities.resources && panels.resourcesWrapper.x + panels.resourcesWrapper.width > root.x)
|
||||||
max -= panels.resources.nonAnimHeight;
|
max -= panels.resources.nonAnimHeight;
|
||||||
if (visibilities.dashboard && panels.dashboard.x < root.x + root.implicitWidth)
|
if (panels.popouts.hasCurrent)
|
||||||
max -= panels.dashboard.nonAnimHeight;
|
if (panels.popouts.current.x + panels.popouts.current.width > root.x && panels.popouts.current.x < root.x + root.width)
|
||||||
if (panels.popouts.currentName.startsWith("updates"))
|
max -= panels.popouts.nonAnimHeight;
|
||||||
max -= panels.popouts.nonAnimHeight;
|
|
||||||
return max;
|
return max;
|
||||||
}
|
}
|
||||||
|
property real offsetScale: shouldBeActive ? 0 : 1
|
||||||
required property var panels
|
required property var panels
|
||||||
required property ShellScreen screen
|
required property ShellScreen screen
|
||||||
required property PersistentProperties visibilities
|
|
||||||
readonly property bool shouldBeActive: visibilities.launcher
|
readonly property bool shouldBeActive: visibilities.launcher
|
||||||
property real offsetScale: shouldBeActive ? 0 : 1
|
required property PersistentProperties visibilities
|
||||||
|
|
||||||
onShouldBeActiveChanged: {
|
|
||||||
if (shouldBeActive) {
|
|
||||||
implicitHeight = Qt.binding(() => content.implicitHeight);
|
|
||||||
timer.stop();
|
|
||||||
} else {
|
|
||||||
implicitHeight = implicitHeight;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
visible: offsetScale < 1
|
|
||||||
anchors.bottomMargin: (-implicitHeight - 5) * offsetScale
|
anchors.bottomMargin: (-implicitHeight - 5) * offsetScale
|
||||||
implicitHeight: content.implicitHeight
|
implicitHeight: content.implicitHeight
|
||||||
implicitWidth: content.implicitWidth || 400
|
implicitWidth: content.implicitWidth || 400
|
||||||
opacity: 1 - offsetScale
|
opacity: 1 - offsetScale
|
||||||
|
visible: offsetScale < 1
|
||||||
|
|
||||||
Behavior on offsetScale {
|
Behavior on offsetScale {
|
||||||
Anim {
|
Anim {
|
||||||
@@ -47,61 +38,26 @@ Item {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
onMaxHeightChanged: timer.start()
|
Component.onCompleted: Qt.callLater(() => Apps)
|
||||||
|
onShouldBeActiveChanged: {
|
||||||
Connections {
|
if (shouldBeActive)
|
||||||
function onEnabledChanged(): void {
|
implicitHeight = Qt.binding(() => content.implicitHeight);
|
||||||
timer.start();
|
else
|
||||||
}
|
implicitHeight = implicitHeight;
|
||||||
|
|
||||||
function onMaxShownChanged(): void {
|
|
||||||
timer.start();
|
|
||||||
}
|
|
||||||
|
|
||||||
target: Config.launcher
|
|
||||||
}
|
|
||||||
|
|
||||||
Connections {
|
|
||||||
function onValuesChanged(): void {
|
|
||||||
if (DesktopEntries.applications.values.length < Config.launcher.maxAppsShown)
|
|
||||||
timer.start();
|
|
||||||
}
|
|
||||||
|
|
||||||
target: DesktopEntries.applications
|
|
||||||
}
|
|
||||||
|
|
||||||
Timer {
|
|
||||||
id: timer
|
|
||||||
|
|
||||||
interval: Appearance.anim.durations.small
|
|
||||||
|
|
||||||
onRunningChanged: {
|
|
||||||
if (running && !root.shouldBeActive) {
|
|
||||||
content.visible = false;
|
|
||||||
content.active = true;
|
|
||||||
} else {
|
|
||||||
root.contentHeight = Math.min(root.maxHeight, content.implicitHeight);
|
|
||||||
content.active = Qt.binding(() => root.shouldBeActive || root.visible);
|
|
||||||
content.visible = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Loader {
|
Loader {
|
||||||
id: content
|
id: content
|
||||||
|
|
||||||
active: false
|
active: root.shouldBeActive || root.visible
|
||||||
anchors.horizontalCenter: parent.horizontalCenter
|
anchors.horizontalCenter: parent.horizontalCenter
|
||||||
anchors.top: parent.top
|
anchors.top: parent.top
|
||||||
|
asynchronous: true
|
||||||
|
|
||||||
sourceComponent: Content {
|
sourceComponent: Content {
|
||||||
maxHeight: root.maxHeight
|
maxHeight: root.maxHeight
|
||||||
panels: root.panels
|
panels: root.panels
|
||||||
visibilities: root.visibilities
|
visibilities: root.visibilities
|
||||||
|
|
||||||
Component.onCompleted: root.contentHeight = implicitHeight
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Component.onCompleted: timer.start()
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,59 +0,0 @@
|
|||||||
import qs.Components
|
|
||||||
import qs.Config
|
|
||||||
import QtQuick
|
|
||||||
import QtQuick.Shapes
|
|
||||||
|
|
||||||
ShapePath {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
readonly property bool flatten: wrapper.height < rounding * 2
|
|
||||||
readonly property real rounding: 8
|
|
||||||
readonly property real roundingY: flatten ? wrapper.height / 2 : rounding
|
|
||||||
required property var sidebar
|
|
||||||
required property Wrapper wrapper
|
|
||||||
|
|
||||||
fillColor: DynamicColors.palette.m3surface
|
|
||||||
strokeWidth: -1
|
|
||||||
|
|
||||||
Behavior on fillColor {
|
|
||||||
CAnim {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: -(root.wrapper.width + root.rounding)
|
|
||||||
relativeY: 0
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: 0
|
|
||||||
relativeY: root.wrapper.height - root.roundingY * 2
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
direction: PathArc.Counterclockwise
|
|
||||||
radiusX: root.sidebar.notifsRoundingX
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.sidebar.notifsRoundingX
|
|
||||||
relativeY: root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: root.wrapper.height > 0 ? root.wrapper.width - root.rounding - root.sidebar.notifsRoundingX : root.wrapper.width
|
|
||||||
relativeY: 0
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: root.rounding
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: root.rounding
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -8,7 +8,7 @@ import QtQuick
|
|||||||
Item {
|
Item {
|
||||||
id: root
|
id: root
|
||||||
|
|
||||||
readonly property int padding: 6
|
readonly property int padding: Appearance.padding.smaller
|
||||||
required property Item panels
|
required property Item panels
|
||||||
required property PersistentProperties visibilities
|
required property PersistentProperties visibilities
|
||||||
|
|
||||||
@@ -54,7 +54,7 @@ Item {
|
|||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
anchors.margins: root.padding
|
anchors.margins: root.padding
|
||||||
color: "transparent"
|
color: "transparent"
|
||||||
radius: Appearance.rounding.smallest / 2
|
radius: Appearance.rounding.normal - root.padding
|
||||||
|
|
||||||
CustomListView {
|
CustomListView {
|
||||||
id: list
|
id: list
|
||||||
@@ -72,7 +72,7 @@ Item {
|
|||||||
required property NotifServer.Notif modelData
|
required property NotifServer.Notif modelData
|
||||||
readonly property alias nonAnimHeight: notif.nonAnimHeight
|
readonly property alias nonAnimHeight: notif.nonAnimHeight
|
||||||
|
|
||||||
implicitHeight: notif.implicitHeight + (idx === 0 ? 0 : 8)
|
implicitHeight: notif.implicitHeight + (idx === 0 ? 0 : Appearance.spacing.small)
|
||||||
implicitWidth: notif.implicitWidth
|
implicitWidth: notif.implicitWidth
|
||||||
|
|
||||||
ListView.onRemove: removeAnim.start()
|
ListView.onRemove: removeAnim.start()
|
||||||
@@ -151,48 +151,6 @@ Item {
|
|||||||
property: "y"
|
property: "y"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ExtraIndicator {
|
|
||||||
anchors.top: parent.top
|
|
||||||
extra: {
|
|
||||||
const count = list.count;
|
|
||||||
if (count === 0)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
const scrollY = list.contentY;
|
|
||||||
|
|
||||||
let height = 0;
|
|
||||||
for (let i = 0; i < count; i++) {
|
|
||||||
height += (list.itemAtIndex(i)?.nonAnimHeight ?? 0) + 8;
|
|
||||||
|
|
||||||
if (height - 8 >= scrollY)
|
|
||||||
return i;
|
|
||||||
}
|
|
||||||
|
|
||||||
return count;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ExtraIndicator {
|
|
||||||
anchors.bottom: parent.bottom
|
|
||||||
extra: {
|
|
||||||
const count = list.count;
|
|
||||||
if (count === 0)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
const scrollY = list.contentHeight - (list.contentY + list.height);
|
|
||||||
|
|
||||||
let height = 0;
|
|
||||||
for (let i = count - 1; i >= 0; i--) {
|
|
||||||
height += (list.itemAtIndex(i)?.nonAnimHeight ?? 0) + 8;
|
|
||||||
|
|
||||||
if (height - 8 >= scrollY)
|
|
||||||
return count - i - 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,54 +0,0 @@
|
|||||||
import QtQuick
|
|
||||||
import QtQuick.Shapes
|
|
||||||
import qs.Components
|
|
||||||
import qs.Config
|
|
||||||
|
|
||||||
ShapePath {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
readonly property bool flatten: wrapper.width < rounding * 2
|
|
||||||
readonly property real notifsRoundingX: panels.notifications.height > 0 && notifsWidthDiff < rounding * 2 ? notifsWidthDiff / 2 : rounding
|
|
||||||
readonly property real notifsWidthDiff: panels.notifications.width - wrapper.width
|
|
||||||
required property var panels
|
|
||||||
readonly property real rounding: Config.barConfig.rounding
|
|
||||||
readonly property real utilsRoundingX: utilsWidthDiff < rounding * 2 ? utilsWidthDiff / 2 : rounding
|
|
||||||
readonly property real utilsWidthDiff: panels.utilities.width - wrapper.width
|
|
||||||
required property Wrapper wrapper
|
|
||||||
|
|
||||||
fillColor: DynamicColors.palette.m3surface
|
|
||||||
strokeWidth: -1
|
|
||||||
|
|
||||||
Behavior on fillColor {
|
|
||||||
CAnim {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: -root.wrapper.width - root.notifsRoundingX
|
|
||||||
relativeY: 0
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.notifsRoundingX
|
|
||||||
radiusY: root.rounding
|
|
||||||
relativeX: root.notifsRoundingX
|
|
||||||
relativeY: root.rounding
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: 0
|
|
||||||
relativeY: root.wrapper.height - root.rounding * 2
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.utilsRoundingX
|
|
||||||
radiusY: root.rounding
|
|
||||||
relativeX: -root.utilsRoundingX
|
|
||||||
relativeY: root.rounding
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: root.wrapper.width + root.utilsRoundingX
|
|
||||||
relativeY: 0
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,66 +0,0 @@
|
|||||||
import QtQuick
|
|
||||||
import QtQuick.Shapes
|
|
||||||
import qs.Components
|
|
||||||
import qs.Config
|
|
||||||
|
|
||||||
ShapePath {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
readonly property bool flatten: wrapper.width < rounding * 2
|
|
||||||
readonly property real rounding: 10
|
|
||||||
readonly property real roundingX: flatten ? wrapper.width / 2 : rounding
|
|
||||||
required property Wrapper wrapper
|
|
||||||
|
|
||||||
fillColor: DynamicColors.palette.m3surface
|
|
||||||
strokeWidth: -1
|
|
||||||
|
|
||||||
Behavior on fillColor {
|
|
||||||
CAnim {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: Math.min(root.rounding, root.wrapper.width)
|
|
||||||
radiusY: root.rounding
|
|
||||||
relativeX: -root.roundingX
|
|
||||||
relativeY: root.rounding
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: -(root.wrapper.width - root.roundingX * 3)
|
|
||||||
relativeY: 0
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
direction: PathArc.Counterclockwise
|
|
||||||
radiusX: Math.min(root.rounding * 2, root.wrapper.width)
|
|
||||||
radiusY: root.rounding * 2
|
|
||||||
relativeX: -root.roundingX * 2
|
|
||||||
relativeY: root.rounding * 2
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: 0
|
|
||||||
relativeY: root.wrapper.height - root.rounding * 4
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
direction: PathArc.Counterclockwise
|
|
||||||
radiusX: Math.min(root.rounding * 2, root.wrapper.width)
|
|
||||||
radiusY: root.rounding * 2
|
|
||||||
relativeX: root.roundingX * 2
|
|
||||||
relativeY: root.rounding * 2
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: root.wrapper.width - root.roundingX * 3
|
|
||||||
relativeY: 0
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: Math.min(root.rounding, root.wrapper.width)
|
|
||||||
radiusY: root.rounding
|
|
||||||
relativeX: root.roundingX
|
|
||||||
relativeY: root.rounding
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,65 +0,0 @@
|
|||||||
import qs.Components
|
|
||||||
import qs.Config
|
|
||||||
import QtQuick
|
|
||||||
import QtQuick.Shapes
|
|
||||||
|
|
||||||
ShapePath {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
readonly property bool flatten: wrapper.height < rounding * 2
|
|
||||||
readonly property real rounding: Appearance.rounding.normal
|
|
||||||
readonly property real roundingY: flatten ? wrapper.height / 2 : rounding
|
|
||||||
required property Wrapper wrapper
|
|
||||||
|
|
||||||
fillColor: DynamicColors.palette.m3surface
|
|
||||||
strokeWidth: -1
|
|
||||||
|
|
||||||
Behavior on fillColor {
|
|
||||||
CAnim {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: 0
|
|
||||||
relativeY: root.wrapper.height
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: -root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: root.wrapper.width - root.rounding * 2
|
|
||||||
relativeY: 0
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
direction: PathArc.Counterclockwise
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: -root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: 0
|
|
||||||
relativeY: -(root.wrapper.height - root.roundingY * 2)
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: -root.roundingY
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -8,7 +8,7 @@ import qs.Config
|
|||||||
Item {
|
Item {
|
||||||
id: root
|
id: root
|
||||||
|
|
||||||
readonly property real nonAnimHeight: state === "visible" ? (content.item?.nonAnimHeight ?? 0) : 0
|
readonly property real nonAnimHeight: content.item?.nonAnimHeight ?? 0
|
||||||
property real offsetScale: shouldBeActive ? 0 : 1
|
property real offsetScale: shouldBeActive ? 0 : 1
|
||||||
readonly property bool shouldBeActive: root.visibilities.resources
|
readonly property bool shouldBeActive: root.visibilities.resources
|
||||||
required property PersistentProperties visibilities
|
required property PersistentProperties visibilities
|
||||||
@@ -31,8 +31,7 @@ Item {
|
|||||||
id: content
|
id: content
|
||||||
|
|
||||||
active: root.shouldBeActive || root.visible
|
active: root.shouldBeActive || root.visible
|
||||||
anchors.bottom: parent.bottom
|
anchors.centerIn: parent
|
||||||
anchors.horizontalCenter: parent.horizontalCenter
|
|
||||||
|
|
||||||
sourceComponent: Content {
|
sourceComponent: Content {
|
||||||
padding: Appearance.padding.normal
|
padding: Appearance.padding.normal
|
||||||
|
|||||||
@@ -1,66 +0,0 @@
|
|||||||
import QtQuick
|
|
||||||
import QtQuick.Shapes
|
|
||||||
import qs.Components
|
|
||||||
import qs.Config
|
|
||||||
|
|
||||||
ShapePath {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
readonly property bool flatten: wrapper.height < rounding * 2
|
|
||||||
readonly property real rounding: Appearance.rounding.large
|
|
||||||
readonly property real roundingY: flatten ? wrapper.height / 2 : rounding
|
|
||||||
required property Wrapper wrapper
|
|
||||||
|
|
||||||
fillColor: DynamicColors.palette.m3surface
|
|
||||||
strokeWidth: -1
|
|
||||||
|
|
||||||
Behavior on fillColor {
|
|
||||||
CAnim {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.roundingY, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: 0
|
|
||||||
relativeY: root.wrapper.height - root.roundingY * 2
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
direction: PathArc.Counterclockwise
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: root.wrapper.width - root.rounding * 2
|
|
||||||
relativeY: 0
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
direction: PathArc.Counterclockwise
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: -root.roundingY
|
|
||||||
}
|
|
||||||
|
|
||||||
PathLine {
|
|
||||||
relativeX: 0
|
|
||||||
relativeY: -(root.wrapper.height - root.roundingY * 2)
|
|
||||||
}
|
|
||||||
|
|
||||||
PathArc {
|
|
||||||
radiusX: root.rounding
|
|
||||||
radiusY: Math.min(root.rounding, root.wrapper.height)
|
|
||||||
relativeX: root.rounding
|
|
||||||
relativeY: -root.roundingY
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -56,6 +56,21 @@ SettingsPage {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SettingsSection {
|
||||||
|
sectionId: "Tray"
|
||||||
|
|
||||||
|
SettingsHeader {
|
||||||
|
name: "System tray"
|
||||||
|
}
|
||||||
|
|
||||||
|
SettingSpinBox {
|
||||||
|
min: 16
|
||||||
|
name: "Tray icon size"
|
||||||
|
object: Config.barConfig.tray
|
||||||
|
setting: "trayIconSize"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
SettingsSection {
|
SettingsSection {
|
||||||
sectionId: "Popouts"
|
sectionId: "Popouts"
|
||||||
|
|
||||||
|
|||||||
@@ -43,79 +43,92 @@ SettingsPage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Separator {
|
Separator {
|
||||||
visible: Config.screenshot.mode === "manual"
|
shouldBeActive: Config.screenshot.mode === "manual"
|
||||||
}
|
}
|
||||||
|
|
||||||
SettingSpinBox {
|
SettingSpinBox {
|
||||||
min: 0
|
min: 0
|
||||||
name: "Corner radius"
|
name: "Corner radius"
|
||||||
object: Config.screenshot
|
object: Config.screenshot
|
||||||
setting: "corner_radius"
|
setting: "radius"
|
||||||
|
shouldBeActive: Config.screenshot.mode === "manual"
|
||||||
step: 1
|
step: 1
|
||||||
visible: Config.screenshot.mode === "manual"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Separator {
|
Separator {
|
||||||
visible: Config.screenshot.mode === "manual"
|
shouldBeActive: Config.screenshot.mode === "manual"
|
||||||
}
|
}
|
||||||
|
|
||||||
SettingSwitch {
|
SettingSwitch {
|
||||||
name: "Enable drop shadow"
|
name: "Enable shadow"
|
||||||
object: Config.screenshot
|
object: Config.screenshot
|
||||||
setting: "drop_shadow"
|
setting: "shadow"
|
||||||
visible: Config.screenshot.mode === "manual"
|
shouldBeActive: Config.screenshot.mode === "manual"
|
||||||
}
|
}
|
||||||
|
|
||||||
Separator {
|
Separator {
|
||||||
visible: Config.screenshot.mode === "manual"
|
shouldBeActive: Config.screenshot.mode === "manual"
|
||||||
}
|
}
|
||||||
|
|
||||||
SettingSwitch {
|
SettingSwitch {
|
||||||
name: "Enable rounded corners"
|
name: "Enable rounded corners"
|
||||||
object: Config.screenshot
|
object: Config.screenshot
|
||||||
setting: "rounded_corners"
|
setting: "rounding"
|
||||||
visible: Config.screenshot.mode === "manual"
|
shouldBeActive: Config.screenshot.mode === "manual"
|
||||||
}
|
}
|
||||||
|
|
||||||
Separator {
|
Separator {
|
||||||
visible: Config.screenshot.mode === "manual"
|
shouldBeActive: Config.screenshot.mode === "manual"
|
||||||
}
|
}
|
||||||
|
|
||||||
SettingSpinBox {
|
SettingSpinBox {
|
||||||
min: 0
|
min: 0
|
||||||
name: "Shadow blur radius"
|
name: "Shadow blur amount"
|
||||||
object: Config.screenshot
|
object: Config.screenshot
|
||||||
setting: "shadow_blur_radius"
|
setting: "shadow_blur"
|
||||||
|
shouldBeActive: Config.screenshot.mode === "manual"
|
||||||
step: 1
|
step: 1
|
||||||
visible: Config.screenshot.mode === "manual"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Separator {
|
Separator {
|
||||||
visible: Config.screenshot.mode === "manual"
|
shouldBeActive: Config.screenshot.mode === "manual"
|
||||||
}
|
}
|
||||||
|
|
||||||
SettingSwitch {
|
// SettingSwitch {
|
||||||
name: "Shadow color broken atm"
|
// name: "Shadow color broken atm"
|
||||||
object: Config.Screenshot
|
// object: Config.Screenshot
|
||||||
setting: "shadow_color"
|
// setting: "shadow_color"
|
||||||
visible: Config.screenshot.mode === "manual"
|
// shouldBeActive: Config.screenshot.mode === "manual"
|
||||||
}
|
// }
|
||||||
|
//
|
||||||
|
// Separator {
|
||||||
|
// shouldBeActive: Config.screenshot.mode === "manual"
|
||||||
|
// }
|
||||||
|
|
||||||
Separator {
|
// SettingSpinBox {
|
||||||
visible: Config.screenshot.mode === "manual"
|
// min: 1
|
||||||
}
|
// name: "Shadow passes"
|
||||||
|
// object: Config.screenshot
|
||||||
|
// setting: "shadow_blur_passes"
|
||||||
|
// shouldBeActive: Config.screenshot.mode === "manual"
|
||||||
|
// step: 1
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// Separator {
|
||||||
|
// shouldBeActive: Config.screenshot.mode === "manual"
|
||||||
|
// }
|
||||||
|
|
||||||
SettingSpinBox {
|
SettingSpinBox {
|
||||||
min: 0
|
min: 0
|
||||||
name: "Shadow offset X"
|
name: "Shadow offset X"
|
||||||
object: Config.screenshot
|
object: Config.screenshot
|
||||||
setting: "shadow_offset_x"
|
setting: "shadow_offset_x"
|
||||||
|
shouldBeActive: Config.screenshot.mode === "manual"
|
||||||
step: 1
|
step: 1
|
||||||
visible: Config.screenshot.mode === "manual"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Separator {
|
Separator {
|
||||||
visible: Config.screenshot.mode === "manual"
|
shouldBeActive: Config.screenshot.mode === "manual"
|
||||||
}
|
}
|
||||||
|
|
||||||
SettingSpinBox {
|
SettingSpinBox {
|
||||||
@@ -123,8 +136,8 @@ SettingsPage {
|
|||||||
name: "Shadow offset Y"
|
name: "Shadow offset Y"
|
||||||
object: Config.screenshot
|
object: Config.screenshot
|
||||||
setting: "shadow_offset_y"
|
setting: "shadow_offset_y"
|
||||||
|
shouldBeActive: Config.screenshot.mode === "manual"
|
||||||
step: 1
|
step: 1
|
||||||
visible: Config.screenshot.mode === "manual"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -134,7 +134,7 @@ Item {
|
|||||||
anchors.right: parent.right
|
anchors.right: parent.right
|
||||||
anchors.top: searchBar.bottom
|
anchors.top: searchBar.bottom
|
||||||
anchors.topMargin: Appearance.spacing.smaller
|
anchors.topMargin: Appearance.spacing.smaller
|
||||||
color: DynamicColors.tPalette.m3surfaceContainerLowest
|
color: DynamicColors.tPalette.m3surfaceContainer
|
||||||
radius: Appearance.rounding.normal
|
radius: Appearance.rounding.normal
|
||||||
|
|
||||||
StackView {
|
StackView {
|
||||||
|
|||||||
@@ -80,12 +80,26 @@ Item {
|
|||||||
required property ShellScreen modelData
|
required property ShellScreen modelData
|
||||||
|
|
||||||
function applyCrop(): void {
|
function applyCrop(): void {
|
||||||
const croprect = cropRect.mapToItem(scaledImg, 0, 0, cropRect.width, cropRect.height);
|
if (!cropRectLoader.item) return;
|
||||||
const upscaledRect = Qt.rect((croprect.x - cropRect.imageX) / scaledImg.paintedWidth, (croprect.y - cropRect.imageY) / scaledImg.paintedHeight, croprect.width / scaledImg.paintedWidth, croprect.height / scaledImg.paintedHeight);
|
const cropRect = cropRectLoader.item;
|
||||||
Wallpapers.setCrop(delegate.modelData.name, upscaledRect, croprect, cropRect.zoom);
|
|
||||||
|
// We need to calculate the exact percentage coordinates that map perfectly
|
||||||
|
// to our C++ backend, regardless of current display scaling
|
||||||
|
const cropXPercent = (cropRect.x - cropRect.imageX) / scaledImg.paintedWidth;
|
||||||
|
const cropYPercent = (cropRect.y - cropRect.imageY) / scaledImg.paintedHeight;
|
||||||
|
const cropWidthPercent = cropRect.width / scaledImg.paintedWidth;
|
||||||
|
const cropHeightPercent = cropRect.height / scaledImg.paintedHeight;
|
||||||
|
|
||||||
|
const finalRect = Qt.rect(cropXPercent, cropYPercent, cropWidthPercent, cropHeightPercent);
|
||||||
|
|
||||||
|
// We just pass the percentages directly to the backend
|
||||||
|
Wallpapers.setCrop(delegate.modelData.name, finalRect, finalRect, cropRect.zoom);
|
||||||
}
|
}
|
||||||
|
|
||||||
function zoomClipRect(zoom: real): void {
|
function zoomClipRect(zoom: real): void {
|
||||||
|
if (!cropRectLoader.item) return;
|
||||||
|
const cropRect = cropRectLoader.item;
|
||||||
|
|
||||||
let oldCenterX = cropRect.x + cropRect.width * 0.5;
|
let oldCenterX = cropRect.x + cropRect.width * 0.5;
|
||||||
let oldCenterY = cropRect.y + cropRect.height * 0.5;
|
let oldCenterY = cropRect.y + cropRect.height * 0.5;
|
||||||
|
|
||||||
@@ -128,7 +142,7 @@ Item {
|
|||||||
Layout.preferredHeight: 10
|
Layout.preferredHeight: 10
|
||||||
from: 1.0
|
from: 1.0
|
||||||
to: 5.0
|
to: 5.0
|
||||||
value: cropRect.zoom
|
value: cropRectLoader.item ? cropRectLoader.item.zoom : 1.0
|
||||||
|
|
||||||
onMoved: {
|
onMoved: {
|
||||||
delegate.zoomClipRect(value);
|
delegate.zoomClipRect(value);
|
||||||
@@ -156,15 +170,20 @@ Item {
|
|||||||
sourceSize.width: parent.width
|
sourceSize.width: parent.width
|
||||||
|
|
||||||
onPaintedWidthChanged: {
|
onPaintedWidthChanged: {
|
||||||
if (paintedWidth > 0) {
|
if (paintedWidth > 0 && cropRectLoader.item) {
|
||||||
scaledImg.displayData = Wallpapers.getCrop(delegate.modelData.name);
|
cropRectLoader.item.restoreFromData();
|
||||||
cropRect.zoom = Wallpapers.getCrop(delegate.modelData.name).zoom;
|
}
|
||||||
cropRect.restoreFromData();
|
}
|
||||||
|
onSourceChanged: {
|
||||||
|
if (cropRectLoader.item) {
|
||||||
|
cropRectLoader.item.restoreFromData();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
onStatusChanged: {
|
||||||
|
if (scaledImg.status == Image.Ready && cropRectLoader.item) {
|
||||||
|
cropRectLoader.item.restoreFromData();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
onSourceChanged: cropRect.clampToBounds()
|
|
||||||
onStatusChanged: if (scaledImg.status == Image.Ready)
|
|
||||||
cropRect.clampToBounds()
|
|
||||||
|
|
||||||
CustomText {
|
CustomText {
|
||||||
id: monitorId
|
id: monitorId
|
||||||
@@ -177,72 +196,85 @@ Item {
|
|||||||
text: delegate.modelData.name
|
text: delegate.modelData.name
|
||||||
}
|
}
|
||||||
|
|
||||||
CustomRect {
|
Loader {
|
||||||
id: cropRect
|
id: cropRectLoader
|
||||||
|
active: scaledImg.paintedWidth > 0 && scaledImg.status == Image.Ready
|
||||||
|
|
||||||
|
sourceComponent: Component {
|
||||||
|
CustomRect {
|
||||||
|
id: cropRect
|
||||||
|
|
||||||
property real aspectRatio: delegate.modelData.width / delegate.modelData.height
|
property real aspectRatio: delegate.modelData.width / delegate.modelData.height
|
||||||
readonly property real baseHeight: baseWidth / aspectRatio
|
readonly property real baseHeight: baseWidth / aspectRatio
|
||||||
readonly property real baseWidth: {
|
readonly property real baseWidth: {
|
||||||
let fittedHeight = scaledImg.paintedHeight;
|
let fittedHeight = scaledImg.paintedHeight;
|
||||||
let fittedWidth = fittedHeight * aspectRatio;
|
let fittedWidth = fittedHeight * aspectRatio;
|
||||||
|
|
||||||
if (fittedWidth > scaledImg.paintedWidth) {
|
if (fittedWidth > scaledImg.paintedWidth) {
|
||||||
fittedWidth = scaledImg.paintedWidth;
|
fittedWidth = scaledImg.paintedWidth;
|
||||||
fittedHeight = fittedWidth / aspectRatio;
|
fittedHeight = fittedWidth / aspectRatio;
|
||||||
}
|
}
|
||||||
|
|
||||||
return fittedWidth;
|
return fittedWidth;
|
||||||
}
|
}
|
||||||
readonly property real imageX: (scaledImg.width - scaledImg.paintedWidth) / 2
|
readonly property real imageX: (scaledImg.width - scaledImg.paintedWidth) / 2
|
||||||
readonly property real imageY: (scaledImg.height - scaledImg.paintedHeight) / 2
|
readonly property real imageY: (scaledImg.height - scaledImg.paintedHeight) / 2
|
||||||
property real imgAspectRatio: scaledImg.paintedWidth / scaledImg.paintedHeight
|
property real imgAspectRatio: scaledImg.paintedWidth / scaledImg.paintedHeight
|
||||||
property real zoom: scaledImg.displayData.zoom
|
property real zoom: 1.0
|
||||||
|
|
||||||
function centerInImage() {
|
function centerInImage() {
|
||||||
x = imageX + (scaledImg.paintedWidth - width) / 2;
|
x = imageX + (scaledImg.paintedWidth - width) / 2;
|
||||||
y = imageY + (scaledImg.paintedHeight - height) / 2;
|
y = imageY + (scaledImg.paintedHeight - height) / 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
function clampToBounds() {
|
function clampToBounds() {
|
||||||
x = Math.max(imageX, Math.min(x, imageX + scaledImg.paintedWidth - width));
|
x = Math.max(imageX, Math.min(x, imageX + scaledImg.paintedWidth - width));
|
||||||
|
|
||||||
y = Math.max(imageY, Math.min(y, imageY + scaledImg.paintedHeight - height));
|
y = Math.max(imageY, Math.min(y, imageY + scaledImg.paintedHeight - height));
|
||||||
}
|
}
|
||||||
|
|
||||||
function restoreFromData() {
|
function restoreFromData() {
|
||||||
let data = scaledImg.displayData;
|
let data = Wallpapers.getCrop(delegate.modelData.name);
|
||||||
|
|
||||||
if (data && data.scaledX !== 0 || data.scaledY !== 0 || data.scaledWidth !== 0 || data.scaledHeight !== 0) {
|
if (data && (Math.abs(data.x) > 0.001 || Math.abs(data.y) > 0.001 || Math.abs(data.width - 1.0) > 0.001 || Math.abs(data.height - 1.0) > 0.001)) {
|
||||||
x = data.scaledX;
|
zoom = data.zoom > 0 ? data.zoom : 1.0;
|
||||||
y = data.scaledY;
|
x = imageX + (data.x * scaledImg.paintedWidth);
|
||||||
|
y = imageY + (data.y * scaledImg.paintedHeight);
|
||||||
|
|
||||||
|
clampToBounds();
|
||||||
|
} else {
|
||||||
|
zoom = 1.0;
|
||||||
|
centerInImage();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
clampToBounds();
|
border.color: DynamicColors.palette.m3primary
|
||||||
} else {
|
border.width: 2
|
||||||
zoom = 1.0;
|
height: baseHeight / zoom
|
||||||
centerInImage();
|
opacity: 1
|
||||||
|
width: baseWidth / zoom
|
||||||
|
|
||||||
|
Behavior on opacity {
|
||||||
|
Anim {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Component.onCompleted: {
|
||||||
|
restoreFromData();
|
||||||
|
}
|
||||||
|
onHeightChanged: clampToBounds()
|
||||||
|
onWidthChanged: clampToBounds()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
border.color: DynamicColors.palette.m3primary
|
|
||||||
border.width: 2
|
|
||||||
height: baseHeight / zoom
|
|
||||||
opacity: 1
|
|
||||||
width: baseWidth / zoom
|
|
||||||
|
|
||||||
Behavior on opacity {
|
|
||||||
Anim {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Component.onCompleted: clampToBounds()
|
|
||||||
onHeightChanged: clampToBounds()
|
|
||||||
onWidthChanged: clampToBounds()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MouseArea {
|
MouseArea {
|
||||||
id: mouse
|
id: mouse
|
||||||
|
|
||||||
function updateCrop(mouseX, mouseY) {
|
function updateCrop(mouseX, mouseY) {
|
||||||
|
if (!cropRectLoader.item) return;
|
||||||
|
const cropRect = cropRectLoader.item;
|
||||||
|
|
||||||
let nx = mouseX - cropRect.width * 0.5;
|
let nx = mouseX - cropRect.width * 0.5;
|
||||||
let ny = mouseY - cropRect.height * 0.5;
|
let ny = mouseY - cropRect.height * 0.5;
|
||||||
|
|
||||||
|
|||||||
@@ -32,10 +32,9 @@ Item {
|
|||||||
Loader {
|
Loader {
|
||||||
id: content
|
id: content
|
||||||
|
|
||||||
active: true
|
active: root.shouldBeActive || root.visible
|
||||||
anchors.bottom: parent.bottom
|
anchors.bottom: parent.bottom
|
||||||
anchors.horizontalCenter: parent.horizontalCenter
|
anchors.horizontalCenter: parent.horizontalCenter
|
||||||
visible: true
|
|
||||||
|
|
||||||
sourceComponent: Content {
|
sourceComponent: Content {
|
||||||
screen: root.screen
|
screen: root.screen
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ import QtQuick
|
|||||||
import QtQuick.VectorImage
|
import QtQuick.VectorImage
|
||||||
import Quickshell
|
import Quickshell
|
||||||
import Quickshell.Services.SystemTray
|
import Quickshell.Services.SystemTray
|
||||||
|
import qs.Helpers
|
||||||
import qs.Modules
|
import qs.Modules
|
||||||
import qs.Components
|
import qs.Components
|
||||||
import qs.Config
|
import qs.Config
|
||||||
@@ -11,12 +12,36 @@ Item {
|
|||||||
id: root
|
id: root
|
||||||
|
|
||||||
property bool current: popouts.currentName.startsWith(`traymenu${ind}`) && popouts.hasCurrent
|
property bool current: popouts.currentName.startsWith(`traymenu${ind}`) && popouts.hasCurrent
|
||||||
property bool hasLoaded: false
|
readonly property real dpr: Hypr.monitorFor(loader.screen).scale
|
||||||
required property int ind
|
required property int ind
|
||||||
required property SystemTrayItem item
|
required property SystemTrayItem item
|
||||||
required property RowLayout loader
|
required property RowLayout loader
|
||||||
required property Wrapper popouts
|
required property Wrapper popouts
|
||||||
|
|
||||||
|
function resolveIcon(app: string, icon: string): string {
|
||||||
|
if (app === "chrome_status_icon_1") {
|
||||||
|
return Quickshell.iconPath("discord-tray");
|
||||||
|
} else if (app === "AyuGramDesktop") {
|
||||||
|
if (icon === Quickshell.iconPath("com.ayugram.desktop-attention-symbolic"))
|
||||||
|
return Quickshell.iconPath("telegram-attention-panel");
|
||||||
|
else if (icon === Quickshell.iconPath("com.ayugram.desktop-mute-symbolic"))
|
||||||
|
return Quickshell.iconPath("telegram-mute-panel");
|
||||||
|
else if (icon === Quickshell.iconPath("com.ayugram.desktop-symbolic"))
|
||||||
|
return Quickshell.iconPath("telegram-panel");
|
||||||
|
} else if (app === "TelegramDesktop") {
|
||||||
|
if (icon === Quickshell.iconPath("org.telegram.desktop-symbolic"))
|
||||||
|
return Quickshell.iconPath("telegram-panel");
|
||||||
|
else if (icon === Quickshell.iconPath("org.telegram.desktop-attention-symbolic"))
|
||||||
|
return Quickshell.iconPath("telegram-attention-panel");
|
||||||
|
else if (icon === Quickshell.iconPath("org.telegram.desktop-mute-symbolic"))
|
||||||
|
return Quickshell.iconPath("telegram-mute-panel");
|
||||||
|
} else if (app === "steam") {
|
||||||
|
return Quickshell.iconPath("steam_tray_mono");
|
||||||
|
}
|
||||||
|
|
||||||
|
return root.item.icon;
|
||||||
|
}
|
||||||
|
|
||||||
CustomRect {
|
CustomRect {
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
anchors.margins: 3
|
anchors.margins: 3
|
||||||
@@ -30,7 +55,8 @@ Item {
|
|||||||
onClicked: {
|
onClicked: {
|
||||||
if (mouse.button === Qt.LeftButton) {
|
if (mouse.button === Qt.LeftButton) {
|
||||||
root.item.activate();
|
root.item.activate();
|
||||||
} else if (mouse.button === Qt.RightButton) {
|
console.log(icon.source + "\n" + root.item.id);
|
||||||
|
} else if (mouse.button === Qt.RightButton && Config.barConfig.popouts.tray) {
|
||||||
root.popouts.currentName = `traymenu${root.ind}`;
|
root.popouts.currentName = `traymenu${root.ind}`;
|
||||||
root.popouts.currentCenter = Qt.binding(() => root.mapToItem(root.loader, root.implicitWidth / 2, 0).x);
|
root.popouts.currentCenter = Qt.binding(() => root.mapToItem(root.loader, root.implicitWidth / 2, 0).x);
|
||||||
root.popouts.hasCurrent = true;
|
root.popouts.hasCurrent = true;
|
||||||
@@ -48,9 +74,11 @@ Item {
|
|||||||
id: icon
|
id: icon
|
||||||
|
|
||||||
anchors.centerIn: parent
|
anchors.centerIn: parent
|
||||||
|
antialiasing: true
|
||||||
color: root.current ? DynamicColors.palette.m3onPrimary : DynamicColors.palette.m3onSurface
|
color: root.current ? DynamicColors.palette.m3onPrimary : DynamicColors.palette.m3onSurface
|
||||||
implicitSize: 22
|
implicitSize: Config.barConfig.tray.trayIconSize * root.dpr
|
||||||
layer.enabled: Config.general.color.smart || Config.general.color.scheduleDark
|
layer.enabled: Config.general.color.smart || Config.general.color.scheduleDark
|
||||||
source: root.item.icon
|
scale: 1 / root.dpr
|
||||||
|
source: root.resolveIcon(root.item.id, root.item.icon)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,12 +23,12 @@ RowLayout {
|
|||||||
let modRowPos = sysTrayMod.mapToItem(sysModRow, modPos.x, modPos.y);
|
let modRowPos = sysTrayMod.mapToItem(sysModRow, modPos.x, modPos.y);
|
||||||
let child = sysModRow.childAt(modRowPos.x, modRowPos.y);
|
let child = sysModRow.childAt(modRowPos.x, modRowPos.y);
|
||||||
if (child) {
|
if (child) {
|
||||||
if (child.objectName === "audioWidget")
|
if (child.objectName === "audioWidget" && Config.barConfig.popouts.audio)
|
||||||
return {
|
return {
|
||||||
id: "audio",
|
id: "audio",
|
||||||
item: child
|
item: child
|
||||||
};
|
};
|
||||||
if (child.objectName === "upowerWidget")
|
if (child.objectName === "upowerWidget" && Config.barConfig.popouts.upower)
|
||||||
return {
|
return {
|
||||||
id: "upower",
|
id: "upower",
|
||||||
item: child
|
item: child
|
||||||
|
|||||||
+122
-111
@@ -11,150 +11,161 @@ import qs.Helpers
|
|||||||
CustomClippingRect {
|
CustomClippingRect {
|
||||||
id: root
|
id: root
|
||||||
|
|
||||||
|
readonly property bool hasUpdates: Object.keys(Updates.updates)?.length > 0
|
||||||
readonly property int itemHeight: 50 + Appearance.padding.smaller * 2
|
readonly property int itemHeight: 50 + Appearance.padding.smaller * 2
|
||||||
required property var wrapper
|
required property var wrapper
|
||||||
|
|
||||||
color: DynamicColors.tPalette.m3surfaceContainer
|
color: DynamicColors.tPalette.m3surfaceContainer
|
||||||
implicitHeight: updatesList.visible ? updatesList.implicitHeight + Appearance.padding.small * 2 : noUpdates.height
|
implicitHeight: hasUpdates ? updatesListLoader.item?.implicitHeight + Appearance.padding.small * 2 : noUpdatesLoader.item.height
|
||||||
implicitWidth: updatesList.visible ? updatesList.contentWidth + Appearance.padding.small * 2 : noUpdates.width
|
implicitWidth: hasUpdates ? updatesListLoader.item?.contentWidth + Appearance.padding.small * 2 : noUpdatesLoader.item.width
|
||||||
radius: Appearance.rounding.small
|
radius: Appearance.rounding.small
|
||||||
|
|
||||||
Item {
|
Loader {
|
||||||
id: noUpdates
|
id: noUpdatesLoader
|
||||||
|
|
||||||
|
active: !root.hasUpdates
|
||||||
anchors.centerIn: parent
|
anchors.centerIn: parent
|
||||||
height: 200
|
|
||||||
visible: script.values.length === 0
|
|
||||||
width: 600
|
|
||||||
|
|
||||||
MaterialIcon {
|
sourceComponent: Item {
|
||||||
id: noUpdatesIcon
|
id: noUpdates
|
||||||
|
|
||||||
anchors.horizontalCenter: parent.horizontalCenter
|
height: 200
|
||||||
anchors.top: parent.top
|
width: 300
|
||||||
color: DynamicColors.tPalette.m3onSurfaceVariant
|
|
||||||
font.pointSize: Appearance.font.size.extraLarge * 3
|
|
||||||
horizontalAlignment: Text.AlignHCenter
|
|
||||||
text: "check"
|
|
||||||
}
|
|
||||||
|
|
||||||
CustomText {
|
MaterialIcon {
|
||||||
anchors.horizontalCenter: parent.horizontalCenter
|
id: noUpdatesIcon
|
||||||
anchors.top: noUpdatesIcon.bottom
|
|
||||||
color: DynamicColors.tPalette.m3onSurfaceVariant
|
anchors.horizontalCenter: parent.horizontalCenter
|
||||||
horizontalAlignment: Text.AlignHCenter
|
anchors.top: parent.top
|
||||||
text: qsTr("No updates available")
|
color: DynamicColors.tPalette.m3onSurfaceVariant
|
||||||
verticalAlignment: Text.AlignVCenter
|
font.pointSize: Appearance.font.size.extraLarge * 3
|
||||||
|
horizontalAlignment: Text.AlignHCenter
|
||||||
|
text: "check"
|
||||||
|
}
|
||||||
|
|
||||||
|
CustomText {
|
||||||
|
anchors.horizontalCenter: parent.horizontalCenter
|
||||||
|
anchors.top: noUpdatesIcon.bottom
|
||||||
|
color: DynamicColors.tPalette.m3onSurfaceVariant
|
||||||
|
horizontalAlignment: Text.AlignHCenter
|
||||||
|
text: qsTr("No updates available")
|
||||||
|
verticalAlignment: Text.AlignVCenter
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
CustomListView {
|
Loader {
|
||||||
id: updatesList
|
id: updatesListLoader
|
||||||
|
|
||||||
|
active: root.hasUpdates
|
||||||
anchors.centerIn: parent
|
anchors.centerIn: parent
|
||||||
contentHeight: childrenRect.height
|
|
||||||
contentWidth: 600
|
|
||||||
displayMarginBeginning: root.itemHeight
|
|
||||||
displayMarginEnd: root.itemHeight
|
|
||||||
implicitHeight: Math.min(contentHeight, (root.itemHeight + spacing) * 5 - spacing)
|
|
||||||
implicitWidth: contentWidth
|
|
||||||
spacing: Appearance.spacing.normal
|
|
||||||
visible: script.values.length > 0
|
|
||||||
|
|
||||||
delegate: CustomRect {
|
sourceComponent: CustomListView {
|
||||||
id: update
|
id: updatesList
|
||||||
|
|
||||||
required property var modelData
|
contentHeight: childrenRect.height
|
||||||
readonly property list<string> sections: modelData.update.split(" ")
|
contentWidth: 600
|
||||||
|
displayMarginBeginning: root.itemHeight
|
||||||
|
displayMarginEnd: root.itemHeight
|
||||||
|
implicitHeight: Math.min(contentHeight, (root.itemHeight + spacing) * 5 - spacing)
|
||||||
|
implicitWidth: contentWidth
|
||||||
|
spacing: Appearance.spacing.normal
|
||||||
|
|
||||||
// anchors.left: parent.left
|
delegate: CustomRect {
|
||||||
// anchors.right: parent.right
|
id: update
|
||||||
color: DynamicColors.tPalette.m3surfaceContainer
|
|
||||||
implicitHeight: root.itemHeight
|
|
||||||
implicitWidth: 600
|
|
||||||
radius: Appearance.rounding.small - Appearance.padding.small
|
|
||||||
|
|
||||||
RowLayout {
|
required property var modelData
|
||||||
anchors.fill: parent
|
readonly property list<string> sections: modelData.update.split(" ")
|
||||||
anchors.leftMargin: Appearance.padding.smaller
|
|
||||||
anchors.rightMargin: Appearance.padding.smaller
|
|
||||||
|
|
||||||
MaterialIcon {
|
// anchors.left: parent.left
|
||||||
font.pointSize: Appearance.font.size.large * 2
|
// anchors.right: parent.right
|
||||||
text: "package_2"
|
color: DynamicColors.tPalette.m3surfaceContainer
|
||||||
}
|
implicitHeight: root.itemHeight
|
||||||
|
implicitWidth: 600
|
||||||
ColumnLayout {
|
radius: Appearance.rounding.small - Appearance.padding.small
|
||||||
Layout.fillWidth: true
|
|
||||||
|
|
||||||
CustomText {
|
|
||||||
Layout.fillWidth: true
|
|
||||||
Layout.preferredHeight: 25
|
|
||||||
elide: Text.ElideRight
|
|
||||||
font.pointSize: Appearance.font.size.large
|
|
||||||
text: update.sections[0]
|
|
||||||
}
|
|
||||||
|
|
||||||
CustomText {
|
|
||||||
Layout.fillWidth: true
|
|
||||||
color: DynamicColors.palette.m3onSurfaceVariant
|
|
||||||
text: Updates.formatUpdateTime(update.modelData.timestamp)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
RowLayout {
|
RowLayout {
|
||||||
Layout.fillHeight: true
|
anchors.fill: parent
|
||||||
Layout.preferredWidth: 300
|
anchors.leftMargin: Appearance.padding.smaller
|
||||||
|
anchors.rightMargin: Appearance.padding.smaller
|
||||||
MarqueeText {
|
|
||||||
id: versionFrom
|
|
||||||
|
|
||||||
Layout.fillHeight: true
|
|
||||||
Layout.preferredWidth: 125
|
|
||||||
animate: true
|
|
||||||
color: DynamicColors.palette.m3tertiary
|
|
||||||
font.pointSize: Appearance.font.size.large
|
|
||||||
horizontalAlignment: Text.AlignHCenter
|
|
||||||
marqueeEnabled: true
|
|
||||||
pauseMs: 4000
|
|
||||||
text: update.sections[1]
|
|
||||||
width: 125
|
|
||||||
}
|
|
||||||
|
|
||||||
MaterialIcon {
|
MaterialIcon {
|
||||||
Layout.fillHeight: true
|
font.pointSize: Appearance.font.size.large * 2
|
||||||
color: DynamicColors.palette.m3secondary
|
text: "package_2"
|
||||||
font.pointSize: Appearance.font.size.extraLarge
|
|
||||||
horizontalAlignment: Text.AlignHCenter
|
|
||||||
text: "arrow_right_alt"
|
|
||||||
verticalAlignment: Text.AlignVCenter
|
|
||||||
}
|
}
|
||||||
|
|
||||||
MarqueeText {
|
ColumnLayout {
|
||||||
id: versionTo
|
Layout.fillWidth: true
|
||||||
|
|
||||||
|
CustomText {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
Layout.preferredHeight: 25
|
||||||
|
elide: Text.ElideRight
|
||||||
|
font.pointSize: Appearance.font.size.large
|
||||||
|
text: update.sections[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
CustomText {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
color: DynamicColors.palette.m3onSurfaceVariant
|
||||||
|
text: Updates.formatUpdateTime(update.modelData.timestamp)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
RowLayout {
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
Layout.preferredWidth: 120
|
Layout.preferredWidth: 300
|
||||||
animate: true
|
|
||||||
color: DynamicColors.palette.m3primary
|
MarqueeText {
|
||||||
font.pointSize: Appearance.font.size.large
|
id: versionFrom
|
||||||
horizontalAlignment: Text.AlignHCenter
|
|
||||||
marqueeEnabled: true
|
Layout.fillHeight: true
|
||||||
pauseMs: 4000
|
Layout.preferredWidth: 125
|
||||||
text: update.sections[3]
|
animate: true
|
||||||
width: 125
|
color: DynamicColors.palette.m3tertiary
|
||||||
|
font.pointSize: Appearance.font.size.large
|
||||||
|
horizontalAlignment: Text.AlignHCenter
|
||||||
|
marqueeEnabled: true
|
||||||
|
pauseMs: 4000
|
||||||
|
text: update.sections[1]
|
||||||
|
width: 125
|
||||||
|
}
|
||||||
|
|
||||||
|
MaterialIcon {
|
||||||
|
Layout.fillHeight: true
|
||||||
|
color: DynamicColors.palette.m3secondary
|
||||||
|
font.pointSize: Appearance.font.size.extraLarge
|
||||||
|
horizontalAlignment: Text.AlignHCenter
|
||||||
|
text: "arrow_right_alt"
|
||||||
|
verticalAlignment: Text.AlignVCenter
|
||||||
|
}
|
||||||
|
|
||||||
|
MarqueeText {
|
||||||
|
id: versionTo
|
||||||
|
|
||||||
|
Layout.fillHeight: true
|
||||||
|
Layout.preferredWidth: 120
|
||||||
|
animate: true
|
||||||
|
color: DynamicColors.palette.m3primary
|
||||||
|
font.pointSize: Appearance.font.size.large
|
||||||
|
horizontalAlignment: Text.AlignHCenter
|
||||||
|
marqueeEnabled: true
|
||||||
|
pauseMs: 4000
|
||||||
|
text: update.sections[3]
|
||||||
|
width: 125
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
model: ScriptModel {
|
||||||
model: ScriptModel {
|
id: script
|
||||||
id: script
|
|
||||||
|
|
||||||
objectProp: "update"
|
objectProp: "update"
|
||||||
values: Object.entries(Updates.updates).sort((a, b) => b[1] - a[1]).map(([update, timestamp]) => ({
|
values: Object.entries(Updates.updates).sort((a, b) => b[1] - a[1]).map(([update, timestamp]) => ({
|
||||||
update,
|
update,
|
||||||
timestamp
|
timestamp
|
||||||
}))
|
}))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,80 +0,0 @@
|
|||||||
pragma ComponentBehavior: Bound
|
|
||||||
|
|
||||||
import QtQuick
|
|
||||||
import qs.Components
|
|
||||||
import qs.Helpers
|
|
||||||
import qs.Config
|
|
||||||
|
|
||||||
Item {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
property Image current: one
|
|
||||||
property string source: Wallpapers.current
|
|
||||||
|
|
||||||
anchors.fill: parent
|
|
||||||
|
|
||||||
Component.onCompleted: {
|
|
||||||
if (source)
|
|
||||||
Qt.callLater(() => one.update());
|
|
||||||
}
|
|
||||||
onSourceChanged: {
|
|
||||||
if (!source) {
|
|
||||||
current = null;
|
|
||||||
} else if (current === one) {
|
|
||||||
two.update();
|
|
||||||
} else {
|
|
||||||
one.update();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Img {
|
|
||||||
id: one
|
|
||||||
}
|
|
||||||
|
|
||||||
Img {
|
|
||||||
id: two
|
|
||||||
}
|
|
||||||
|
|
||||||
component Img: Image {
|
|
||||||
id: img
|
|
||||||
|
|
||||||
function update(): void {
|
|
||||||
if (source === root.source) {
|
|
||||||
root.current = this;
|
|
||||||
} else {
|
|
||||||
source = root.source;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
anchors.fill: parent
|
|
||||||
asynchronous: true
|
|
||||||
fillMode: Image.PreserveAspectCrop
|
|
||||||
opacity: 0
|
|
||||||
retainWhileLoading: true
|
|
||||||
scale: Wallpapers.showPreview ? 1 : 0.8
|
|
||||||
sourceClipRect: Qt.rect(Config.background.sourceClipX, Config.background.sourceClipY, Config.background.sourceClipW, Config.background.sourceClipH)
|
|
||||||
|
|
||||||
states: State {
|
|
||||||
name: "visible"
|
|
||||||
when: root.current === img
|
|
||||||
|
|
||||||
PropertyChanges {
|
|
||||||
img.opacity: 1
|
|
||||||
img.scale: 1
|
|
||||||
}
|
|
||||||
}
|
|
||||||
transitions: Transition {
|
|
||||||
Anim {
|
|
||||||
duration: Config.background.wallFadeDuration
|
|
||||||
properties: "opacity,scale"
|
|
||||||
target: img
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
onStatusChanged: {
|
|
||||||
if (status === Image.Ready) {
|
|
||||||
root.current = this;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -6,6 +6,7 @@ import QtQuick
|
|||||||
import qs.Components
|
import qs.Components
|
||||||
import qs.Helpers
|
import qs.Helpers
|
||||||
import qs.Config
|
import qs.Config
|
||||||
|
import ZShell.Internal
|
||||||
|
|
||||||
Item {
|
Item {
|
||||||
id: root
|
id: root
|
||||||
@@ -15,58 +16,64 @@ Item {
|
|||||||
|
|
||||||
function refreshData(): void {
|
function refreshData(): void {
|
||||||
Hyprland.refreshMonitors();
|
Hyprland.refreshMonitors();
|
||||||
const scale = Hyprland.monitorFor(root.screen).scale;
|
let scale = Hyprland.monitorFor(root.screen).scale;
|
||||||
if (scale > 0 && img.resScale !== scale) {
|
if (scale <= 0)
|
||||||
img.resScale = scale;
|
scale = 1.0; // Fallback to avoid zeroes on initialization
|
||||||
img.sourceSize.width = root.screen.width * scale;
|
|
||||||
|
if (root.screen.width > 0 && root.screen.height > 0) {
|
||||||
|
img.screenResolution = Qt.size(root.screen.width * scale, root.screen.height * scale);
|
||||||
}
|
}
|
||||||
|
|
||||||
const displayData = Wallpapers.getCrop(root.screen.name);
|
const displayData = Wallpapers.getCrop(root.screen.name);
|
||||||
const displayRect = Qt.rect(img.sourceSize.width * displayData.x, img.implicitHeight * displayData.y, img.sourceSize.width * displayData.width, img.implicitHeight * displayData.height);
|
|
||||||
img.anchors.fill = null;
|
if (displayData) {
|
||||||
img.zoom = displayData.zoom;
|
img.cropX = displayData.x !== undefined ? displayData.x : 0.0;
|
||||||
img.x = -(displayRect.x * displayData.zoom / img.resScale);
|
img.cropY = displayData.y !== undefined ? displayData.y : 0.0;
|
||||||
img.y = -(displayRect.y * displayData.zoom / img.resScale);
|
img.cropWidth = (displayData.width !== undefined && displayData.width > 0) ? displayData.width : 1.0;
|
||||||
|
img.cropHeight = (displayData.height !== undefined && displayData.height > 0) ? displayData.height : 1.0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
|
|
||||||
Image {
|
Component.onCompleted: root.refreshData()
|
||||||
|
|
||||||
|
Connections {
|
||||||
|
function onHeightChanged() {
|
||||||
|
root.refreshData();
|
||||||
|
}
|
||||||
|
|
||||||
|
function onWidthChanged() {
|
||||||
|
root.refreshData();
|
||||||
|
}
|
||||||
|
|
||||||
|
target: root.screen
|
||||||
|
}
|
||||||
|
|
||||||
|
WallpaperImage {
|
||||||
id: img
|
id: img
|
||||||
|
|
||||||
property int displayH
|
anchors.fill: parent
|
||||||
property int displayW
|
|
||||||
property real resScale
|
|
||||||
property real zoom: 1.0
|
|
||||||
|
|
||||||
asynchronous: true
|
|
||||||
fillMode: Image.PreserveAspectCrop
|
|
||||||
height: implicitHeight * zoom / resScale
|
|
||||||
opacity: 1
|
|
||||||
retainWhileLoading: true
|
|
||||||
source: root.source
|
source: root.source
|
||||||
sourceSize.width: root.screen.width * resScale
|
|
||||||
width: implicitWidth * zoom / resScale
|
|
||||||
|
|
||||||
Behavior on height {
|
Behavior on cropHeight {
|
||||||
Anim {
|
Anim {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Behavior on width {
|
Behavior on cropWidth {
|
||||||
Anim {
|
Anim {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Behavior on x {
|
Behavior on cropX {
|
||||||
Anim {
|
Anim {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Behavior on y {
|
Behavior on cropY {
|
||||||
Anim {
|
Anim {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Behavior on zoom {
|
||||||
onStatusChanged: {
|
Anim {
|
||||||
if (img.status == Image.Ready) {
|
|
||||||
root.refreshData();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ import qs.Modules.DesktopIcons
|
|||||||
|
|
||||||
Loader {
|
Loader {
|
||||||
active: Config.background.enabled
|
active: Config.background.enabled
|
||||||
asynchronous: true
|
asynchronous: false
|
||||||
|
|
||||||
sourceComponent: Variants {
|
sourceComponent: Variants {
|
||||||
model: Quickshell.screens
|
model: Quickshell.screens
|
||||||
|
|||||||
+4
-49
@@ -15,9 +15,8 @@ Item {
|
|||||||
property real currentCenter
|
property real currentCenter
|
||||||
property alias currentName: popoutState.currentName
|
property alias currentName: popoutState.currentName
|
||||||
property string detachedMode
|
property string detachedMode
|
||||||
readonly property bool isDetached: detachedMode.length > 0
|
|
||||||
property alias hasCurrent: popoutState.hasCurrent
|
property alias hasCurrent: popoutState.hasCurrent
|
||||||
readonly property real nonAnimHeight: children.find(c => c.shouldBeActive)?.implicitHeight ?? content.implicitHeight
|
readonly property real nonAnimHeight: content.implicitHeight || 150
|
||||||
readonly property real nonAnimWidth: children.find(c => c.shouldBeActive)?.implicitWidth ?? content.implicitWidth
|
readonly property real nonAnimWidth: children.find(c => c.shouldBeActive)?.implicitWidth ?? content.implicitWidth
|
||||||
required property real offsetScale
|
required property real offsetScale
|
||||||
property string queuedMode
|
property string queuedMode
|
||||||
@@ -28,29 +27,13 @@ Item {
|
|||||||
detachedMode = "";
|
detachedMode = "";
|
||||||
}
|
}
|
||||||
|
|
||||||
function detach(mode: string): void {
|
|
||||||
setAnims(true);
|
|
||||||
if (mode === "winfo") {
|
|
||||||
detachedMode = mode;
|
|
||||||
} else {
|
|
||||||
queuedMode = mode;
|
|
||||||
detachedMode = "any";
|
|
||||||
}
|
|
||||||
setAnims(false);
|
|
||||||
focus = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
function setAnims(detach: bool): void {
|
|
||||||
const type = `expressive${detach ? "Slow" : "Default"}Spatial`;
|
|
||||||
animLength = Appearance.anim.durations[type];
|
|
||||||
animCurve = Appearance.anim.curves[type];
|
|
||||||
}
|
|
||||||
|
|
||||||
focus: hasCurrent
|
focus: hasCurrent
|
||||||
implicitHeight: nonAnimHeight
|
implicitHeight: nonAnimHeight
|
||||||
implicitWidth: nonAnimWidth
|
implicitWidth: nonAnimWidth
|
||||||
|
|
||||||
Behavior on implicitHeight {
|
Behavior on implicitHeight {
|
||||||
|
enabled: root.offsetScale < 1
|
||||||
|
|
||||||
Anim {
|
Anim {
|
||||||
duration: root.animLength
|
duration: root.animLength
|
||||||
easing.bezierCurve: root.animCurve
|
easing.bezierCurve: root.animCurve
|
||||||
@@ -72,42 +55,14 @@ Item {
|
|||||||
Comp {
|
Comp {
|
||||||
id: content
|
id: content
|
||||||
|
|
||||||
// anchors.horizontalCenter: parent.horizontalCenter
|
|
||||||
// anchors.top: parent.top
|
|
||||||
anchors.centerIn: parent
|
anchors.centerIn: parent
|
||||||
shouldBeActive: root.hasCurrent && !root.detachedMode
|
shouldBeActive: root.hasCurrent
|
||||||
|
|
||||||
sourceComponent: Content {
|
sourceComponent: Content {
|
||||||
popouts: popoutState
|
popouts: popoutState
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Comp {
|
|
||||||
// id: winfo
|
|
||||||
//
|
|
||||||
// anchors.centerIn: parent
|
|
||||||
// shouldBeActive: root.detachedMode === "winfo"
|
|
||||||
//
|
|
||||||
// sourceComponent: WindowInfo {
|
|
||||||
// client: Hypr.activeToplevel
|
|
||||||
// screen: root.screen
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// Comp {
|
|
||||||
// id: controlCenter
|
|
||||||
//
|
|
||||||
// anchors.centerIn: parent
|
|
||||||
// shouldBeActive: root.detachedMode === "any"
|
|
||||||
//
|
|
||||||
// sourceComponent: ControlCenter {
|
|
||||||
// active: root.queuedMode
|
|
||||||
// screen: root.screen
|
|
||||||
//
|
|
||||||
// onClose: root.close()
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
|
|
||||||
component Comp: Loader {
|
component Comp: Loader {
|
||||||
id: comp
|
id: comp
|
||||||
|
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ qml_module(ZShell-internal
|
|||||||
circularbuffer.hpp circularbuffer.cpp
|
circularbuffer.hpp circularbuffer.cpp
|
||||||
sparklineitem.hpp sparklineitem.cpp
|
sparklineitem.hpp sparklineitem.cpp
|
||||||
arcgauge.hpp arcgauge.cpp
|
arcgauge.hpp arcgauge.cpp
|
||||||
|
wallpaperimage.hpp wallpaperimage.cpp
|
||||||
LIBRARIES
|
LIBRARIES
|
||||||
Qt::Gui
|
Qt::Gui
|
||||||
Qt::Quick
|
Qt::Quick
|
||||||
|
|||||||
@@ -4,7 +4,7 @@
|
|||||||
#include <qpainter.h>
|
#include <qpainter.h>
|
||||||
#include <qpen.h>
|
#include <qpen.h>
|
||||||
|
|
||||||
namespace caelestia::internal {
|
namespace ZShell::internal {
|
||||||
|
|
||||||
ArcGauge::ArcGauge(QQuickItem* parent)
|
ArcGauge::ArcGauge(QQuickItem* parent)
|
||||||
: QQuickPaintedItem(parent) {
|
: QQuickPaintedItem(parent) {
|
||||||
@@ -116,4 +116,4 @@ void ArcGauge::setLineWidth(qreal width) {
|
|||||||
update();
|
update();
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace caelestia::internal
|
} // namespace ZShell::internal
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
#include <qqmlintegration.h>
|
#include <qqmlintegration.h>
|
||||||
#include <qquickpainteditem.h>
|
#include <qquickpainteditem.h>
|
||||||
|
|
||||||
namespace caelestia::internal {
|
namespace ZShell::internal {
|
||||||
|
|
||||||
class ArcGauge : public QQuickPaintedItem {
|
class ArcGauge : public QQuickPaintedItem {
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
@@ -58,4 +58,4 @@ qreal m_sweepAngle = 1.5 * M_PI;
|
|||||||
qreal m_lineWidth = 10.0;
|
qreal m_lineWidth = 10.0;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace caelestia::internal
|
} // namespace ZShell::internal
|
||||||
|
|||||||
@@ -1,11 +1,18 @@
|
|||||||
#include "hyprextras.hpp"
|
#include "hyprextras.hpp"
|
||||||
#include "hyprdevices.hpp"
|
#include "hyprdevices.hpp"
|
||||||
|
|
||||||
|
#include <functional>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
#include <qdir.h>
|
#include <qdir.h>
|
||||||
|
#include <qcolor.h>
|
||||||
#include <qjsonarray.h>
|
#include <qjsonarray.h>
|
||||||
|
#include <qjsondocument.h>
|
||||||
|
#include <qjsonobject.h>
|
||||||
#include <qlocalsocket.h>
|
#include <qlocalsocket.h>
|
||||||
#include <qloggingcategory.h>
|
#include <qloggingcategory.h>
|
||||||
#include <qmetatype.h>
|
#include <qmetatype.h>
|
||||||
|
#include <qobject.h>
|
||||||
#include <qregularexpression.h>
|
#include <qregularexpression.h>
|
||||||
#include <qvariant.h>
|
#include <qvariant.h>
|
||||||
|
|
||||||
@@ -163,6 +170,86 @@ static QString buildHlConfigCall(const QString& key, const QVariant& value) {
|
|||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static QColor colorFromInt(quint32 value) {
|
||||||
|
const int a = (value >> 24) & 0xFF;
|
||||||
|
const int r = (value >> 16) & 0xFF;
|
||||||
|
const int g = (value >> 8) & 0xFF;
|
||||||
|
const int b = value & 0xFF;
|
||||||
|
|
||||||
|
return QColor(r, g, b, a);
|
||||||
|
}
|
||||||
|
|
||||||
|
static QVariant parseGetOptionValue(const QJsonObject& obj) {
|
||||||
|
if (obj.contains(QStringLiteral("bool"))) {
|
||||||
|
return obj.value(QStringLiteral("bool")).toBool();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (obj.contains(QStringLiteral("int"))) {
|
||||||
|
const auto value = obj.value(QStringLiteral("int")).toInt();
|
||||||
|
|
||||||
|
const auto option = obj.value(QStringLiteral("option")).toString();
|
||||||
|
|
||||||
|
if (option.contains(QStringLiteral("color")) || option.contains(QStringLiteral("col."))) {
|
||||||
|
return colorFromInt(static_cast<quint32>(value));
|
||||||
|
}
|
||||||
|
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (obj.contains(QStringLiteral("float"))) {
|
||||||
|
return obj.value(QStringLiteral("float")).toDouble();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (obj.contains(QStringLiteral("str"))) {
|
||||||
|
return obj.value(QStringLiteral("str")).toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (obj.contains(QStringLiteral("current"))) {
|
||||||
|
return obj.value(QStringLiteral("current")).toVariant();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (obj.contains(QStringLiteral("value"))) {
|
||||||
|
return obj.value(QStringLiteral("value")).toVariant();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (obj.contains(QStringLiteral("vec2"))) {
|
||||||
|
return obj.value(QStringLiteral("vec2")).toVariant();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (obj.contains(QStringLiteral("data"))) {
|
||||||
|
const auto data = obj.value(QStringLiteral("data"));
|
||||||
|
if (data.isObject()) {
|
||||||
|
const auto d = data.toObject();
|
||||||
|
if (d.contains(QStringLiteral("current"))) {
|
||||||
|
return d.value(QStringLiteral("current")).toVariant();
|
||||||
|
}
|
||||||
|
if (d.contains(QStringLiteral("value"))) {
|
||||||
|
return d.value(QStringLiteral("value")).toVariant();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return data.toVariant();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
static void insertNestedValue(QVariantMap& root, const QStringList& path, const QVariant& value) {
|
||||||
|
if (path.isEmpty()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (path.size() == 1) {
|
||||||
|
root.insert(path.first(), value);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const QString head = path.first();
|
||||||
|
QVariantMap child = root.value(head).toMap();
|
||||||
|
insertNestedValue(child, path.mid(1), value);
|
||||||
|
root.insert(head, child);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
HyprExtras::HyprExtras(QObject* parent)
|
HyprExtras::HyprExtras(QObject* parent)
|
||||||
@@ -203,7 +290,7 @@ HyprExtras::HyprExtras(QObject* parent)
|
|||||||
m_socket->connectToServer(m_eventSocket, QLocalSocket::ReadOnly);
|
m_socket->connectToServer(m_eventSocket, QLocalSocket::ReadOnly);
|
||||||
}
|
}
|
||||||
|
|
||||||
QVariantHash HyprExtras::options() const {
|
QVariantMap HyprExtras::options() const {
|
||||||
return m_options;
|
return m_options;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -269,30 +356,64 @@ void HyprExtras::refreshOptions() {
|
|||||||
m_optionsRefresh->close();
|
m_optionsRefresh->close();
|
||||||
}
|
}
|
||||||
|
|
||||||
m_optionsRefresh = makeRequestJson(QStringLiteral("descriptions"), [this](bool success, const QJsonDocument& response) {
|
++m_optionsRefreshGeneration;
|
||||||
m_optionsRefresh.reset();
|
const quint64 generation = m_optionsRefreshGeneration;
|
||||||
if (!success) {
|
|
||||||
|
static const QStringList optionKeys = {
|
||||||
|
QStringLiteral("general:border_size"),
|
||||||
|
QStringLiteral("decoration:rounding"),
|
||||||
|
QStringLiteral("animations:enabled"),
|
||||||
|
QStringLiteral("decoration:shadow:enabled"),
|
||||||
|
QStringLiteral("decoration:shadow:offset"),
|
||||||
|
QStringLiteral("decoration:shadow:color"),
|
||||||
|
QStringLiteral("decoration:shadow:range"),
|
||||||
|
QStringLiteral("decoration:shadow:render_power"),
|
||||||
|
};
|
||||||
|
|
||||||
|
auto nextOptions = std::make_shared<QVariantMap>();
|
||||||
|
|
||||||
|
auto step = std::make_shared<std::function<void(int)> >();
|
||||||
|
*step = [this, generation, nextOptions, step](int index) {
|
||||||
|
if (generation != m_optionsRefreshGeneration) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const auto options = response.array();
|
if (index >= optionKeys.size()) {
|
||||||
bool dirty = false;
|
if (m_options != *nextOptions) {
|
||||||
|
m_options = *nextOptions;
|
||||||
for (const auto& o : std::as_const(options)) {
|
emit optionsChanged();
|
||||||
const auto obj = o.toObject();
|
|
||||||
const auto key = obj.value(QStringLiteral("value")).toString();
|
|
||||||
const auto value = obj.value(QStringLiteral("data")).toObject().value(QStringLiteral("current")).toVariant();
|
|
||||||
|
|
||||||
if (m_options.value(key) != value) {
|
|
||||||
dirty = true;
|
|
||||||
m_options.insert(key, value);
|
|
||||||
}
|
}
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dirty) {
|
const QString key = optionKeys.at(index);
|
||||||
emit optionsChanged();
|
|
||||||
}
|
m_optionsRefresh = makeRequestJson(
|
||||||
});
|
QStringLiteral("getoption ") + key,
|
||||||
|
[this, generation, nextOptions, step, index, key](bool success, const QJsonDocument& response)
|
||||||
|
{
|
||||||
|
m_optionsRefresh.reset();
|
||||||
|
|
||||||
|
if (generation != m_optionsRefreshGeneration) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (success && response.isObject()) {
|
||||||
|
const QVariant value = parseGetOptionValue(response.object());
|
||||||
|
if (value.isValid()) {
|
||||||
|
insertNestedValue(*nextOptions, key.split(QLatin1Char(':'), Qt::SkipEmptyParts), value);
|
||||||
|
} else {
|
||||||
|
qCWarning(lcHypr) << "refreshOptions: getoption returned no usable value for" << key;
|
||||||
|
}
|
||||||
|
} else if (!success) {
|
||||||
|
qCWarning(lcHypr) << "refreshOptions: getoption request error for" << key;
|
||||||
|
}
|
||||||
|
|
||||||
|
(*step)(index + 1);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
(*step)(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void HyprExtras::refreshDevices() {
|
void HyprExtras::refreshDevices() {
|
||||||
|
|||||||
@@ -1,9 +1,13 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <functional>
|
||||||
|
|
||||||
|
#include <qjsondocument.h>
|
||||||
#include <qlocalsocket.h>
|
#include <qlocalsocket.h>
|
||||||
#include <qobject.h>
|
#include <qobject.h>
|
||||||
#include <qqmlintegration.h>
|
#include <qqmlintegration.h>
|
||||||
#include <qsharedpointer.h>
|
#include <qsharedpointer.h>
|
||||||
|
#include <qstringlist.h>
|
||||||
#include <qvariant.h>
|
#include <qvariant.h>
|
||||||
|
|
||||||
namespace ZShell::internal::hypr {
|
namespace ZShell::internal::hypr {
|
||||||
@@ -15,13 +19,13 @@ Q_OBJECT
|
|||||||
QML_ELEMENT
|
QML_ELEMENT
|
||||||
Q_MOC_INCLUDE("hyprdevices.hpp")
|
Q_MOC_INCLUDE("hyprdevices.hpp")
|
||||||
|
|
||||||
Q_PROPERTY(QVariantHash options READ options NOTIFY optionsChanged)
|
Q_PROPERTY(QVariantMap options READ options NOTIFY optionsChanged)
|
||||||
Q_PROPERTY(ZShell::internal::hypr::HyprDevices* devices READ devices CONSTANT)
|
Q_PROPERTY(ZShell::internal::hypr::HyprDevices* devices READ devices CONSTANT)
|
||||||
|
|
||||||
public:
|
public:
|
||||||
explicit HyprExtras(QObject* parent = nullptr);
|
explicit HyprExtras(QObject* parent = nullptr);
|
||||||
|
|
||||||
[[nodiscard]] QVariantHash options() const;
|
[[nodiscard]] QVariantMap options() const;
|
||||||
[[nodiscard]] HyprDevices* devices() const;
|
[[nodiscard]] HyprDevices* devices() const;
|
||||||
|
|
||||||
Q_INVOKABLE void message(const QString& message);
|
Q_INVOKABLE void message(const QString& message);
|
||||||
@@ -42,11 +46,12 @@ QString m_eventSocket;
|
|||||||
QLocalSocket* m_socket;
|
QLocalSocket* m_socket;
|
||||||
bool m_socketValid;
|
bool m_socketValid;
|
||||||
|
|
||||||
QVariantHash m_options;
|
QVariantMap m_options;
|
||||||
HyprDevices* const m_devices;
|
HyprDevices* const m_devices;
|
||||||
|
|
||||||
SocketPtr m_optionsRefresh;
|
SocketPtr m_optionsRefresh;
|
||||||
SocketPtr m_devicesRefresh;
|
SocketPtr m_devicesRefresh;
|
||||||
|
quint64 m_optionsRefreshGeneration = 0;
|
||||||
|
|
||||||
void socketError(QLocalSocket::LocalSocketError error) const;
|
void socketError(QLocalSocket::LocalSocketError error) const;
|
||||||
void socketStateChanged(QLocalSocket::LocalSocketState state);
|
void socketStateChanged(QLocalSocket::LocalSocketState state);
|
||||||
|
|||||||
@@ -0,0 +1,199 @@
|
|||||||
|
#include "wallpaperimage.hpp"
|
||||||
|
|
||||||
|
#include <QStandardPaths>
|
||||||
|
#include <QCryptographicHash>
|
||||||
|
#include <QDir>
|
||||||
|
#include <QFileInfo>
|
||||||
|
#include <QtConcurrent>
|
||||||
|
#include <QSGImageNode>
|
||||||
|
#include <QQuickWindow>
|
||||||
|
|
||||||
|
namespace ZShell::internal {
|
||||||
|
|
||||||
|
WallpaperImage::WallpaperImage(QQuickItem *parent)
|
||||||
|
: QQuickItem(parent)
|
||||||
|
{
|
||||||
|
setFlag(ItemHasContents, true);
|
||||||
|
connect(&m_imageWatcher, &QFutureWatcher<QImage>::finished, this, &WallpaperImage::handleImageLoaded);
|
||||||
|
}
|
||||||
|
|
||||||
|
WallpaperImage::~WallpaperImage() {
|
||||||
|
if (m_texture) delete m_texture;
|
||||||
|
}
|
||||||
|
|
||||||
|
void WallpaperImage::setSource(const QUrl &source) {
|
||||||
|
if (m_source == source) return;
|
||||||
|
m_source = source;
|
||||||
|
emit sourceChanged();
|
||||||
|
loadImage();
|
||||||
|
}
|
||||||
|
|
||||||
|
void WallpaperImage::setScreenResolution(const QSize &screenResolution) {
|
||||||
|
if (m_screenResolution == screenResolution) return;
|
||||||
|
m_screenResolution = screenResolution;
|
||||||
|
emit screenResolutionChanged();
|
||||||
|
loadImage();
|
||||||
|
}
|
||||||
|
|
||||||
|
void WallpaperImage::setZoom(qreal zoom) {
|
||||||
|
if (qFuzzyCompare(m_zoom, zoom)) return;
|
||||||
|
m_zoom = zoom;
|
||||||
|
emit zoomChanged();
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void WallpaperImage::setCropX(qreal x) {
|
||||||
|
if (qFuzzyCompare(m_cropX, x)) return;
|
||||||
|
m_cropX = x;
|
||||||
|
emit cropXChanged();
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void WallpaperImage::setCropY(qreal y) {
|
||||||
|
if (qFuzzyCompare(m_cropY, y)) return;
|
||||||
|
m_cropY = y;
|
||||||
|
emit cropYChanged();
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void WallpaperImage::setCropWidth(qreal w) {
|
||||||
|
if (w <= 0.0) w = 1.0;
|
||||||
|
if (qFuzzyCompare(m_cropWidth, w)) return;
|
||||||
|
m_cropWidth = w;
|
||||||
|
emit cropWidthChanged();
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void WallpaperImage::setCropHeight(qreal h) {
|
||||||
|
if (h <= 0.0) h = 1.0;
|
||||||
|
if (qFuzzyCompare(m_cropHeight, h)) return;
|
||||||
|
m_cropHeight = h;
|
||||||
|
emit cropHeightChanged();
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
|
||||||
|
QString WallpaperImage::getCacheFilePath() const {
|
||||||
|
if (m_source.isEmpty() || m_screenResolution.isEmpty()) return QString();
|
||||||
|
|
||||||
|
QString cachePath = QStandardPaths::writableLocation(QStandardPaths::GenericCacheLocation) + "/zshell/imagecache";
|
||||||
|
QDir().mkpath(cachePath);
|
||||||
|
|
||||||
|
// Hash the source URL + resolution
|
||||||
|
QString id = m_source.toString() + "_" + QString::number(m_screenResolution.width()) + "x" + QString::number(m_screenResolution.height());
|
||||||
|
QByteArray hash = QCryptographicHash::hash(id.toUtf8(), QCryptographicHash::Md5).toHex();
|
||||||
|
|
||||||
|
return cachePath + "/" + hash + ".png";
|
||||||
|
}
|
||||||
|
|
||||||
|
void WallpaperImage::loadImage() {
|
||||||
|
if (m_source.isEmpty()) return;
|
||||||
|
|
||||||
|
QString cacheFile = getCacheFilePath();
|
||||||
|
QString sourceFile = m_source.isLocalFile() ? m_source.toLocalFile() : m_source.toString();
|
||||||
|
|
||||||
|
// Qt resource path correction if passed as a standard URL string
|
||||||
|
if (sourceFile.startsWith("qrc:/")) {
|
||||||
|
sourceFile = sourceFile.mid(3); // Converts "qrc:/" to ":/"
|
||||||
|
}
|
||||||
|
|
||||||
|
QSize targetRes = m_screenResolution;
|
||||||
|
|
||||||
|
// Run off the main thread to avoid blocking the UI
|
||||||
|
QFuture<QImage> future = QtConcurrent::run([sourceFile, cacheFile, targetRes]() -> QImage {
|
||||||
|
if (!targetRes.isEmpty() && !cacheFile.isEmpty() && QFileInfo::exists(cacheFile)) {
|
||||||
|
QImage cached(cacheFile);
|
||||||
|
if (!cached.isNull()) return cached;
|
||||||
|
}
|
||||||
|
|
||||||
|
QImage original(sourceFile);
|
||||||
|
if (original.isNull()) return QImage();
|
||||||
|
|
||||||
|
if (targetRes.isEmpty()) {
|
||||||
|
// Screen resolution not set yet by QML, return the unscaled original for now to prevent a black screen
|
||||||
|
return original;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if original is strictly larger than screen resolution
|
||||||
|
if (original.width() > targetRes.width() || original.height() > targetRes.height()) {
|
||||||
|
QImage scaled = original.scaled(targetRes, Qt::KeepAspectRatioByExpanding, Qt::SmoothTransformation);
|
||||||
|
if (!cacheFile.isEmpty()) scaled.save(cacheFile, "PNG");
|
||||||
|
return scaled;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Otherwise just cache and return the original
|
||||||
|
if (!cacheFile.isEmpty()) original.save(cacheFile, "PNG");
|
||||||
|
return original;
|
||||||
|
});
|
||||||
|
|
||||||
|
m_imageWatcher.setFuture(future);
|
||||||
|
}
|
||||||
|
|
||||||
|
void WallpaperImage::handleImageLoaded() {
|
||||||
|
m_image = m_imageWatcher.result();
|
||||||
|
m_textureDirty = true;
|
||||||
|
update(); // Request redraw
|
||||||
|
}
|
||||||
|
|
||||||
|
QSGNode *WallpaperImage::updatePaintNode(QSGNode *oldNode, UpdatePaintNodeData *) {
|
||||||
|
auto *node = static_cast<QSGImageNode *>(oldNode);
|
||||||
|
|
||||||
|
if (m_image.isNull()) {
|
||||||
|
delete node;
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!node) {
|
||||||
|
node = window()->createImageNode();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (m_textureDirty) {
|
||||||
|
if (m_texture) delete m_texture;
|
||||||
|
m_texture = window()->createTextureFromImage(m_image, QQuickWindow::TextureHasAlphaChannel);
|
||||||
|
m_textureDirty = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (m_texture) {
|
||||||
|
node->setTexture(m_texture);
|
||||||
|
node->setRect(boundingRect());
|
||||||
|
node->setFiltering(QSGTexture::Linear);
|
||||||
|
|
||||||
|
qreal cW = m_cropWidth / m_zoom;
|
||||||
|
qreal cH = m_cropHeight / m_zoom;
|
||||||
|
|
||||||
|
QRectF reqRect(
|
||||||
|
m_cropX * m_texture->textureSize().width(),
|
||||||
|
m_cropY * m_texture->textureSize().height(),
|
||||||
|
cW * m_texture->textureSize().width(),
|
||||||
|
cH * m_texture->textureSize().height()
|
||||||
|
);
|
||||||
|
|
||||||
|
QRectF bounds = boundingRect();
|
||||||
|
if (bounds.isEmpty() || reqRect.isEmpty()) return node;
|
||||||
|
|
||||||
|
qreal targetRatio = bounds.width() / bounds.height();
|
||||||
|
qreal reqRatio = reqRect.width() / reqRect.height();
|
||||||
|
|
||||||
|
QRectF sourceRect = reqRect;
|
||||||
|
|
||||||
|
// Force 'PreserveAspectCrop' behavior on the requested region
|
||||||
|
if (reqRatio > targetRatio) {
|
||||||
|
// Requested region is too wide, center-crop the sides
|
||||||
|
qreal newWidth = reqRect.height() * targetRatio;
|
||||||
|
qreal xOffset = (reqRect.width() - newWidth) / 2.0;
|
||||||
|
sourceRect.setX(reqRect.x() + xOffset);
|
||||||
|
sourceRect.setWidth(newWidth);
|
||||||
|
} else if (reqRatio < targetRatio) {
|
||||||
|
// Requested region is too tall, center-crop the top/bottom
|
||||||
|
qreal newHeight = reqRect.width() / targetRatio;
|
||||||
|
qreal yOffset = (reqRect.height() - newHeight) / 2.0;
|
||||||
|
sourceRect.setY(reqRect.y() + yOffset);
|
||||||
|
sourceRect.setHeight(newHeight);
|
||||||
|
}
|
||||||
|
|
||||||
|
node->setSourceRect(sourceRect);
|
||||||
|
}
|
||||||
|
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace ZShell::internal
|
||||||
@@ -0,0 +1,95 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <QQuickItem>
|
||||||
|
#include <QImage>
|
||||||
|
#include <QUrl>
|
||||||
|
#include <QSGTexture>
|
||||||
|
#include <QFutureWatcher>
|
||||||
|
#include <QtQml/qqml.h>
|
||||||
|
|
||||||
|
namespace ZShell::internal {
|
||||||
|
|
||||||
|
class WallpaperImage : public QQuickItem {
|
||||||
|
Q_OBJECT
|
||||||
|
QML_NAMED_ELEMENT(WallpaperImage)
|
||||||
|
Q_PROPERTY(QUrl source READ source WRITE setSource NOTIFY sourceChanged)
|
||||||
|
Q_PROPERTY(QSize screenResolution READ screenResolution WRITE setScreenResolution NOTIFY screenResolutionChanged)
|
||||||
|
Q_PROPERTY(qreal zoom READ zoom WRITE setZoom NOTIFY zoomChanged)
|
||||||
|
|
||||||
|
Q_PROPERTY(qreal cropX READ cropX WRITE setCropX NOTIFY cropXChanged)
|
||||||
|
Q_PROPERTY(qreal cropY READ cropY WRITE setCropY NOTIFY cropYChanged)
|
||||||
|
Q_PROPERTY(qreal cropWidth READ cropWidth WRITE setCropWidth NOTIFY cropWidthChanged)
|
||||||
|
Q_PROPERTY(qreal cropHeight READ cropHeight WRITE setCropHeight NOTIFY cropHeightChanged)
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit WallpaperImage(QQuickItem *parent = nullptr);
|
||||||
|
~WallpaperImage() override;
|
||||||
|
|
||||||
|
QUrl source() const {
|
||||||
|
return m_source;
|
||||||
|
}
|
||||||
|
void setSource(const QUrl &source);
|
||||||
|
|
||||||
|
QSize screenResolution() const {
|
||||||
|
return m_screenResolution;
|
||||||
|
}
|
||||||
|
void setScreenResolution(const QSize &screenResolution);
|
||||||
|
|
||||||
|
qreal zoom() const {
|
||||||
|
return m_zoom;
|
||||||
|
}
|
||||||
|
void setZoom(qreal zoom);
|
||||||
|
|
||||||
|
qreal cropX() const {
|
||||||
|
return m_cropX;
|
||||||
|
}
|
||||||
|
void setCropX(qreal x);
|
||||||
|
|
||||||
|
qreal cropY() const {
|
||||||
|
return m_cropY;
|
||||||
|
}
|
||||||
|
void setCropY(qreal y);
|
||||||
|
|
||||||
|
qreal cropWidth() const {
|
||||||
|
return m_cropWidth;
|
||||||
|
}
|
||||||
|
void setCropWidth(qreal w);
|
||||||
|
|
||||||
|
qreal cropHeight() const {
|
||||||
|
return m_cropHeight;
|
||||||
|
}
|
||||||
|
void setCropHeight(qreal h);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
QSGNode *updatePaintNode(QSGNode *oldNode, UpdatePaintNodeData *updatePaintNodeData) override;
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void sourceChanged();
|
||||||
|
void screenResolutionChanged();
|
||||||
|
void zoomChanged();
|
||||||
|
void cropXChanged();
|
||||||
|
void cropYChanged();
|
||||||
|
void cropWidthChanged();
|
||||||
|
void cropHeightChanged();
|
||||||
|
|
||||||
|
private:
|
||||||
|
void loadImage();
|
||||||
|
void handleImageLoaded();
|
||||||
|
QString getCacheFilePath() const;
|
||||||
|
|
||||||
|
QUrl m_source;
|
||||||
|
QSize m_screenResolution;
|
||||||
|
qreal m_zoom = 1.0;
|
||||||
|
|
||||||
|
qreal m_cropX = 0.0;
|
||||||
|
qreal m_cropY = 0.0;
|
||||||
|
qreal m_cropWidth = 1.0;
|
||||||
|
qreal m_cropHeight = 1.0;
|
||||||
|
|
||||||
|
QImage m_image;
|
||||||
|
QSGTexture *m_texture = nullptr;
|
||||||
|
bool m_textureDirty = false;
|
||||||
|
QFutureWatcher<QImage> m_imageWatcher;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace ZShell::internal
|
||||||
+307
-83
@@ -1,131 +1,355 @@
|
|||||||
#include "writefile.hpp"
|
#include "writefile.hpp"
|
||||||
|
|
||||||
#include <QtConcurrent/qtconcurrentrun.h>
|
#include <QtConcurrent/qtconcurrentrun.h>
|
||||||
|
#include <QtCore/QCryptographicHash>
|
||||||
|
#include <QtCore/QSaveFile>
|
||||||
|
#include <QtGui/QImageReader>
|
||||||
|
#include <QtQuick/qquickimageprovider.h>
|
||||||
#include <QtQuick/qquickitemgrabresult.h>
|
#include <QtQuick/qquickitemgrabresult.h>
|
||||||
#include <QtQuick/qquickwindow.h>
|
#include <QtQuick/qquickwindow.h>
|
||||||
#include <qdir.h>
|
|
||||||
#include <qfileinfo.h>
|
#include <QDir>
|
||||||
#include <qfuturewatcher.h>
|
#include <QFile>
|
||||||
#include <qqmlengine.h>
|
#include <QFileInfo>
|
||||||
|
#include <QFutureWatcher>
|
||||||
|
#include <QImage>
|
||||||
|
#include <QJSValue>
|
||||||
|
#include <QQmlEngine>
|
||||||
|
#include <QSize>
|
||||||
|
#include <QVariant>
|
||||||
|
|
||||||
namespace ZShell {
|
namespace ZShell {
|
||||||
|
|
||||||
|
// ============================================================
|
||||||
|
// saveItem
|
||||||
|
// ============================================================
|
||||||
|
|
||||||
void ZShellIo::saveItem(QQuickItem* target, const QUrl& path) {
|
void ZShellIo::saveItem(QQuickItem* target, const QUrl& path) {
|
||||||
this->saveItem(target, path, QRect(), QJSValue(), QJSValue());
|
this->saveItem(target, path, QRect(), QJSValue(), QJSValue());
|
||||||
}
|
}
|
||||||
|
|
||||||
void ZShellIo::saveItem(QQuickItem* target, const QUrl& path, const QRect& rect) {
|
void ZShellIo::saveItem(QQuickItem* target, const QUrl& path, const QRect& rect) {
|
||||||
this->saveItem(target, path, rect, QJSValue(), QJSValue());
|
this->saveItem(target, path, rect, QJSValue(), QJSValue());
|
||||||
}
|
}
|
||||||
|
|
||||||
void ZShellIo::saveItem(QQuickItem* target, const QUrl& path, QJSValue onSaved) {
|
void ZShellIo::saveItem(QQuickItem* target, const QUrl& path, QJSValue onSaved) {
|
||||||
this->saveItem(target, path, QRect(), onSaved, QJSValue());
|
this->saveItem(target, path, QRect(), onSaved, QJSValue());
|
||||||
}
|
}
|
||||||
|
|
||||||
void ZShellIo::saveItem(QQuickItem* target, const QUrl& path, QJSValue onSaved, QJSValue onFailed) {
|
void ZShellIo::saveItem(QQuickItem* target, const QUrl& path, QJSValue onSaved, QJSValue onFailed) {
|
||||||
this->saveItem(target, path, QRect(), onSaved, onFailed);
|
this->saveItem(target, path, QRect(), onSaved, onFailed);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ZShellIo::saveItem(QQuickItem* target, const QUrl& path, const QRect& rect, QJSValue onSaved) {
|
void ZShellIo::saveItem(QQuickItem* target, const QUrl& path, const QRect& rect, QJSValue onSaved) {
|
||||||
this->saveItem(target, path, rect, onSaved, QJSValue());
|
this->saveItem(target, path, rect, onSaved, QJSValue());
|
||||||
}
|
}
|
||||||
|
|
||||||
void ZShellIo::saveItem(QQuickItem* target, const QUrl& path, const QRect& rect, QJSValue onSaved, QJSValue onFailed) {
|
void ZShellIo::saveItem(
|
||||||
if (!target) {
|
QQuickItem* target,
|
||||||
qWarning() << "ZShellIo::saveItem: a target is required";
|
const QUrl& path,
|
||||||
return;
|
const QRect& rect,
|
||||||
}
|
QJSValue onSaved,
|
||||||
|
QJSValue onFailed
|
||||||
|
) {
|
||||||
|
if (!target) {
|
||||||
|
qWarning() << "ZShellIo::saveItem: a target is required";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (!path.isLocalFile()) {
|
if (!path.isLocalFile()) {
|
||||||
qWarning() << "ZShellIo::saveItem:" << path << "is not a local file";
|
qWarning() << "ZShellIo::saveItem:" << path << "is not a local file";
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!target->window()) {
|
if (!target->window()) {
|
||||||
qWarning() << "ZShellIo::saveItem: unable to save target" << target << "without a window";
|
qWarning() << "ZShellIo::saveItem: unable to save target"
|
||||||
return;
|
<< target
|
||||||
}
|
<< "without a window";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
auto scaledRect = rect;
|
auto scaledRect = rect;
|
||||||
const qreal scale = target->window()->devicePixelRatio();
|
|
||||||
if (rect.isValid() && !qFuzzyCompare(scale + 1.0, 2.0)) {
|
|
||||||
scaledRect =
|
|
||||||
QRectF(rect.left() * scale, rect.top() * scale, rect.width() * scale, rect.height() * scale).toRect();
|
|
||||||
}
|
|
||||||
|
|
||||||
const QSharedPointer<const QQuickItemGrabResult> grabResult = target->grabToImage();
|
const qreal scale = target->window()->devicePixelRatio();
|
||||||
|
|
||||||
QObject::connect(grabResult.data(), &QQuickItemGrabResult::ready, this,
|
if (rect.isValid() && !qFuzzyCompare(scale + 1.0, 2.0)) {
|
||||||
[grabResult, scaledRect, path, onSaved, onFailed, this]() {
|
scaledRect = QRectF(
|
||||||
const auto future = QtConcurrent::run([=]() {
|
rect.left() * scale,
|
||||||
QImage image = grabResult->image();
|
rect.top() * scale,
|
||||||
|
rect.width() * scale,
|
||||||
|
rect.height() * scale
|
||||||
|
).toRect();
|
||||||
|
}
|
||||||
|
|
||||||
if (scaledRect.isValid()) {
|
const QSharedPointer<const QQuickItemGrabResult> grabResult =
|
||||||
image = image.copy(scaledRect);
|
target->grabToImage();
|
||||||
}
|
|
||||||
|
|
||||||
const QString file = path.toLocalFile();
|
QObject::connect(
|
||||||
const QString parent = QFileInfo(file).absolutePath();
|
grabResult.data(),
|
||||||
return QDir().mkpath(parent) && image.save(file);
|
&QQuickItemGrabResult::ready,
|
||||||
});
|
this,
|
||||||
|
[grabResult, scaledRect, path, onSaved, onFailed, this]() {
|
||||||
|
const auto future = QtConcurrent::run([grabResult, scaledRect, path]() {
|
||||||
|
QImage image = grabResult->image();
|
||||||
|
|
||||||
auto* watcher = new QFutureWatcher<bool>(this);
|
if (scaledRect.isValid()) {
|
||||||
auto* engine = qmlEngine(this);
|
image = image.copy(scaledRect);
|
||||||
|
}
|
||||||
|
|
||||||
QObject::connect(watcher, &QFutureWatcher<bool>::finished, this, [=]() {
|
const QString file = path.toLocalFile();
|
||||||
if (watcher->result()) {
|
const QString parent = QFileInfo(file).absolutePath();
|
||||||
if (onSaved.isCallable()) {
|
|
||||||
onSaved.call(
|
QDir().mkpath(parent);
|
||||||
{ QJSValue(path.toLocalFile()), engine->toScriptValue(QVariant::fromValue(path)) });
|
|
||||||
}
|
QSaveFile out(file);
|
||||||
} else {
|
if (!out.open(QIODevice::WriteOnly)) {
|
||||||
qWarning() << "ZShellIo::saveItem: failed to save" << path;
|
return false;
|
||||||
if (onFailed.isCallable()) {
|
}
|
||||||
onFailed.call({ engine->toScriptValue(QVariant::fromValue(path)) });
|
|
||||||
}
|
if (!image.save(&out, "PNG")) {
|
||||||
}
|
return false;
|
||||||
watcher->deleteLater();
|
}
|
||||||
});
|
|
||||||
watcher->setFuture(future);
|
return out.commit();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
auto* watcher = new QFutureWatcher<bool>(this);
|
||||||
|
auto* engine = qmlEngine(this);
|
||||||
|
|
||||||
|
QObject::connect(watcher, &QFutureWatcher<bool>::finished, this, [=]() {
|
||||||
|
if (watcher->result()) {
|
||||||
|
if (onSaved.isCallable() && engine) {
|
||||||
|
onSaved.call({
|
||||||
|
engine->toScriptValue(path.toLocalFile()),
|
||||||
|
engine->toScriptValue(path)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
qWarning() << "ZShellIo::saveItem: failed to save" << path;
|
||||||
|
if (onFailed.isCallable() && engine) {
|
||||||
|
onFailed.call({
|
||||||
|
engine->toScriptValue(path)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
watcher->deleteLater();
|
||||||
|
});
|
||||||
|
|
||||||
|
watcher->setFuture(future);
|
||||||
|
}
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ============================================================
|
||||||
|
// cacheImage
|
||||||
|
// ============================================================
|
||||||
|
|
||||||
|
void ZShellIo::cacheImage(const QUrl& source, const QString& cacheDir) {
|
||||||
|
this->cacheImage(source, cacheDir, QJSValue(), QJSValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
void ZShellIo::cacheImage(const QUrl& source, const QString& cacheDir, QJSValue onSaved) {
|
||||||
|
this->cacheImage(source, cacheDir, onSaved, QJSValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
void ZShellIo::cacheImage(
|
||||||
|
const QUrl& source,
|
||||||
|
const QString& cacheDir,
|
||||||
|
QJSValue onSaved,
|
||||||
|
QJSValue onFailed
|
||||||
|
) {
|
||||||
|
if (cacheDir.isEmpty()) {
|
||||||
|
qWarning() << "ZShellIo::cacheImage: cacheDir is empty";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
QImage image;
|
||||||
|
if (!loadSourceImage(source, image)) {
|
||||||
|
qWarning() << "ZShellIo::cacheImage: failed to load source image" << source;
|
||||||
|
auto* engine = qmlEngine(this);
|
||||||
|
if (onFailed.isCallable() && engine) {
|
||||||
|
onFailed.call({
|
||||||
|
engine->toScriptValue(source),
|
||||||
|
engine->toScriptValue(cacheDir)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto future = QtConcurrent::run([image, cacheDir]() -> QString {
|
||||||
|
if (image.isNull()) {
|
||||||
|
return QString();
|
||||||
|
}
|
||||||
|
|
||||||
|
const QImage normalized = image.convertToFormat(QImage::Format_RGBA8888);
|
||||||
|
|
||||||
|
const QByteArray bytes(
|
||||||
|
reinterpret_cast<const char*>(normalized.constBits()),
|
||||||
|
qsizetype(normalized.sizeInBytes())
|
||||||
|
);
|
||||||
|
|
||||||
|
const QByteArray digest =
|
||||||
|
QCryptographicHash::hash(bytes, QCryptographicHash::Sha256).toHex();
|
||||||
|
|
||||||
|
QDir dir(cacheDir);
|
||||||
|
if (!dir.exists() && !QDir().mkpath(cacheDir)) {
|
||||||
|
return QString();
|
||||||
|
}
|
||||||
|
|
||||||
|
const QString finalPath = dir.filePath(QString::fromLatin1(digest) + ".png");
|
||||||
|
|
||||||
|
if (QFile::exists(finalPath)) {
|
||||||
|
return finalPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
QSaveFile out(finalPath);
|
||||||
|
if (!out.open(QIODevice::WriteOnly)) {
|
||||||
|
return QString();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!normalized.save(&out, "PNG")) {
|
||||||
|
return QString();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!out.commit()) {
|
||||||
|
return QString();
|
||||||
|
}
|
||||||
|
|
||||||
|
return finalPath;
|
||||||
|
});
|
||||||
|
|
||||||
|
auto* watcher = new QFutureWatcher<QString>(this);
|
||||||
|
auto* engine = qmlEngine(this);
|
||||||
|
|
||||||
|
QObject::connect(watcher, &QFutureWatcher<QString>::finished, this, [=]() {
|
||||||
|
const QString finalPath = watcher->result();
|
||||||
|
|
||||||
|
if (!finalPath.isEmpty()) {
|
||||||
|
if (onSaved.isCallable() && engine) {
|
||||||
|
onSaved.call({
|
||||||
|
engine->toScriptValue(finalPath),
|
||||||
|
engine->toScriptValue(QUrl::fromLocalFile(finalPath))
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
qWarning() << "ZShellIo::cacheImage: failed to cache" << source;
|
||||||
|
if (onFailed.isCallable() && engine) {
|
||||||
|
onFailed.call({
|
||||||
|
engine->toScriptValue(source),
|
||||||
|
engine->toScriptValue(cacheDir)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
watcher->deleteLater();
|
||||||
|
});
|
||||||
|
|
||||||
|
watcher->setFuture(future);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ============================================================
|
||||||
|
// loadSourceImage
|
||||||
|
// ============================================================
|
||||||
|
|
||||||
|
bool ZShellIo::loadSourceImage(const QUrl& source, QImage& image) const {
|
||||||
|
image = QImage();
|
||||||
|
|
||||||
|
if (source.isLocalFile()) {
|
||||||
|
QImageReader reader(source.toLocalFile());
|
||||||
|
reader.setAutoTransform(true);
|
||||||
|
image = reader.read();
|
||||||
|
return !image.isNull();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (source.scheme() == "image") {
|
||||||
|
auto* engine = qmlEngine(const_cast<ZShellIo*>(this));
|
||||||
|
if (!engine) {
|
||||||
|
qWarning() << "ZShellIo::loadSourceImage: no QQmlEngine";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const QString providerId = source.host();
|
||||||
|
|
||||||
|
const QString imageId =
|
||||||
|
source.path().startsWith('/')
|
||||||
|
? source.path().mid(1)
|
||||||
|
: source.path();
|
||||||
|
|
||||||
|
auto* providerBase = engine->imageProvider(providerId);
|
||||||
|
if (!providerBase) {
|
||||||
|
qWarning() << "ZShellIo::loadSourceImage: provider not found"
|
||||||
|
<< providerId;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto* provider = dynamic_cast<QQuickImageProvider*>(providerBase);
|
||||||
|
if (!provider) {
|
||||||
|
qWarning() << "ZShellIo::loadSourceImage: provider is not a QQuickImageProvider"
|
||||||
|
<< providerId;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
QSize size;
|
||||||
|
|
||||||
|
switch (provider->imageType()) {
|
||||||
|
case QQuickImageProvider::Image:
|
||||||
|
image = provider->requestImage(imageId, &size, QSize());
|
||||||
|
break;
|
||||||
|
|
||||||
|
case QQuickImageProvider::Pixmap:
|
||||||
|
image = provider->requestPixmap(imageId, &size, QSize()).toImage();
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
qWarning() << "ZShellIo::loadSourceImage: unsupported provider type"
|
||||||
|
<< providerId;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return !image.isNull();
|
||||||
|
}
|
||||||
|
|
||||||
|
qWarning() << "ZShellIo::loadSourceImage: unsupported source" << source;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// ============================================================
|
||||||
|
// File ops
|
||||||
|
// ============================================================
|
||||||
|
|
||||||
bool ZShellIo::copyFile(const QUrl& source, const QUrl& target, bool overwrite) const {
|
bool ZShellIo::copyFile(const QUrl& source, const QUrl& target, bool overwrite) const {
|
||||||
if (!source.isLocalFile()) {
|
if (!source.isLocalFile()) {
|
||||||
qWarning() << "ZShellIo::copyFile: source" << source << "is not a local file";
|
qWarning() << "ZShellIo::copyFile: source" << source << "is not a local file";
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (!target.isLocalFile()) {
|
if (!target.isLocalFile()) {
|
||||||
qWarning() << "ZShellIo::copyFile: target" << target << "is not a local file";
|
qWarning() << "ZShellIo::copyFile: target" << target << "is not a local file";
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (overwrite) {
|
if (overwrite) {
|
||||||
if (!QFile::remove(target.toLocalFile())) {
|
QFile::remove(target.toLocalFile());
|
||||||
qWarning() << "ZShellIo::copyFile: overwrite was specified but failed to remove" << target.toLocalFile();
|
}
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return QFile::copy(source.toLocalFile(), target.toLocalFile());
|
return QFile::copy(source.toLocalFile(), target.toLocalFile());
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ZShellIo::deleteFile(const QUrl& path) const {
|
bool ZShellIo::deleteFile(const QUrl& path) const {
|
||||||
if (!path.isLocalFile()) {
|
if (!path.isLocalFile()) {
|
||||||
qWarning() << "ZShellIo::deleteFile: path" << path << "is not a local file";
|
qWarning() << "ZShellIo::deleteFile: path" << path << "is not a local file";
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return QFile::remove(path.toLocalFile());
|
return QFile::remove(path.toLocalFile());
|
||||||
}
|
}
|
||||||
|
|
||||||
QString ZShellIo::toLocalFile(const QUrl& url) const {
|
QString ZShellIo::toLocalFile(const QUrl& url) const {
|
||||||
if (!url.isLocalFile()) {
|
if (!url.isLocalFile()) {
|
||||||
qWarning() << "ZShellIo::toLocalFile: given url is not a local file" << url;
|
qWarning() << "ZShellIo::toLocalFile: given url is not a local file" << url;
|
||||||
return QString();
|
return QString();
|
||||||
}
|
}
|
||||||
|
|
||||||
return url.toLocalFile();
|
return url.toLocalFile();
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace ZShell
|
} // namespace ZShell
|
||||||
|
|||||||
@@ -1,31 +1,40 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <QtQuick/qquickitem.h>
|
#include <QtQuick/qquickitem.h>
|
||||||
#include <qobject.h>
|
#include <QImage>
|
||||||
|
#include <QJSValue>
|
||||||
|
#include <QObject>
|
||||||
#include <qqmlintegration.h>
|
#include <qqmlintegration.h>
|
||||||
|
#include <QUrl>
|
||||||
|
|
||||||
namespace ZShell {
|
namespace ZShell {
|
||||||
|
|
||||||
class ZShellIo : public QObject {
|
class ZShellIo : public QObject {
|
||||||
|
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
QML_ELEMENT
|
QML_ELEMENT
|
||||||
QML_SINGLETON
|
QML_SINGLETON
|
||||||
|
|
||||||
public:
|
public:
|
||||||
// clang-format off
|
// clang-format off
|
||||||
Q_INVOKABLE void saveItem(QQuickItem* target, const QUrl& path);
|
Q_INVOKABLE void saveItem(QQuickItem* target, const QUrl& path);
|
||||||
Q_INVOKABLE void saveItem(QQuickItem* target, const QUrl& path, const QRect& rect);
|
Q_INVOKABLE void saveItem(QQuickItem* target, const QUrl& path, const QRect& rect);
|
||||||
Q_INVOKABLE void saveItem(QQuickItem* target, const QUrl& path, QJSValue onSaved);
|
Q_INVOKABLE void saveItem(QQuickItem* target, const QUrl& path, QJSValue onSaved);
|
||||||
Q_INVOKABLE void saveItem(QQuickItem* target, const QUrl& path, QJSValue onSaved, QJSValue onFailed);
|
Q_INVOKABLE void saveItem(QQuickItem* target, const QUrl& path, QJSValue onSaved, QJSValue onFailed);
|
||||||
Q_INVOKABLE void saveItem(QQuickItem* target, const QUrl& path, const QRect& rect, QJSValue onSaved);
|
Q_INVOKABLE void saveItem(QQuickItem* target, const QUrl& path, const QRect& rect, QJSValue onSaved);
|
||||||
Q_INVOKABLE void saveItem(QQuickItem* target, const QUrl& path, const QRect& rect, QJSValue onSaved, QJSValue onFailed);
|
Q_INVOKABLE void saveItem(QQuickItem* target, const QUrl& path, const QRect& rect, QJSValue onSaved, QJSValue onFailed);
|
||||||
// clang-format on
|
|
||||||
|
|
||||||
Q_INVOKABLE bool copyFile(const QUrl& source, const QUrl& target, bool overwrite = true) const;
|
Q_INVOKABLE void cacheImage(const QUrl& source, const QString& cacheDir);
|
||||||
Q_INVOKABLE bool deleteFile(const QUrl& path) const;
|
Q_INVOKABLE void cacheImage(const QUrl& source, const QString& cacheDir, QJSValue onSaved);
|
||||||
Q_INVOKABLE QString toLocalFile(const QUrl& url) const;
|
Q_INVOKABLE void cacheImage(const QUrl& source, const QString& cacheDir, QJSValue onSaved, QJSValue onFailed);
|
||||||
|
// clang-format on
|
||||||
|
|
||||||
|
Q_INVOKABLE bool copyFile(const QUrl& source, const QUrl& target, bool overwrite = true) const;
|
||||||
|
Q_INVOKABLE bool deleteFile(const QUrl& path) const;
|
||||||
|
Q_INVOKABLE QString toLocalFile(const QUrl& url) const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
bool loadSourceImage(const QUrl& source, QImage& image) const;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
} // namespace ZShell
|
} // namespace ZShell
|
||||||
|
|||||||
@@ -1,16 +1,52 @@
|
|||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
import sys
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
import typer
|
import typer
|
||||||
|
from typer._completion_shared import install, _get_shell_name
|
||||||
from zshell.subcommands import shell, scheme, screenshot, wallpaper, record
|
from zshell.subcommands import shell, scheme, screenshot, wallpaper, record
|
||||||
|
|
||||||
app = typer.Typer()
|
app = typer.Typer(name="zshell-cli", add_completion=False)
|
||||||
|
|
||||||
app.add_typer(shell.app, name="shell")
|
app.add_typer(shell.app, name="shell")
|
||||||
app.add_typer(scheme.app, name="scheme")
|
app.add_typer(scheme.app, name="scheme")
|
||||||
app.add_typer(screenshot.app, name="screenshot")
|
app.add_typer(screenshot.app, name="screenshot")
|
||||||
app.add_typer(wallpaper.app, name="wallpaper")
|
app.add_typer(wallpaper.app, name="wallpaper")
|
||||||
app.add_typer(record.app, name="record")
|
app.add_typer(record.app, name="record")
|
||||||
# app.add_typer(preset.app, name="preset")
|
|
||||||
|
|
||||||
|
def _completion_installed() -> bool:
|
||||||
|
shell = _get_shell_name()
|
||||||
|
match shell:
|
||||||
|
case "zsh":
|
||||||
|
return (Path.home() / ".zfunc" / "_zshell-cli").exists()
|
||||||
|
case "bash":
|
||||||
|
return (Path.home() / ".bash_completions" / "zshell-cli.sh").exists()
|
||||||
|
case "fish":
|
||||||
|
return (Path.home() / ".config" / "fish" / "completions" / "zshell-cli.fish").exists()
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
def _install_completion() -> None:
|
||||||
|
if _completion_installed():
|
||||||
|
print("zshell-cli: Shell completion already installed.")
|
||||||
|
raise typer.Exit()
|
||||||
|
shell = _get_shell_name()
|
||||||
|
if shell is None:
|
||||||
|
print("zshell-cli: Unable to detect shell type.", file=sys.stderr)
|
||||||
|
raise typer.Exit(code=1)
|
||||||
|
try:
|
||||||
|
_, path = install(prog_name="zshell-cli")
|
||||||
|
print(f"zshell-cli: Shell completion installed ({shell}: {path})")
|
||||||
|
print("zshell-cli: Restart your shell or source the file to enable tab-completion.")
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
def main() -> None:
|
def main() -> None:
|
||||||
|
if "--install-autocomplete" in sys.argv:
|
||||||
|
_install_completion()
|
||||||
|
return
|
||||||
|
if sys.stdout.isatty() and not _completion_installed():
|
||||||
|
print("zshell-cli: Tip: run with --install-autocomplete for tab completion.", file=sys.stderr)
|
||||||
app()
|
app()
|
||||||
|
|||||||
@@ -18,8 +18,7 @@ TEMP_RECORDING = STATE_DIR / "recording.mp4"
|
|||||||
REPLAY_RECORDING = STATE_DIR / "replay.mp4"
|
REPLAY_RECORDING = STATE_DIR / "replay.mp4"
|
||||||
NOTIF_ID_FILE = STATE_DIR / "notifid.txt"
|
NOTIF_ID_FILE = STATE_DIR / "notifid.txt"
|
||||||
|
|
||||||
RECORDINGS_DIR = os.getenv("ZSHELL_RECORDINGS_DIR",
|
RECORDINGS_DIR = os.getenv("ZSHELL_RECORDINGS_DIR", str(Path(HOME) / "Videos/Recordings"))
|
||||||
str(Path(HOME) / "Videos/Recordings"))
|
|
||||||
|
|
||||||
|
|
||||||
def _read_extra_args() -> list[str]:
|
def _read_extra_args() -> list[str]:
|
||||||
@@ -36,7 +35,7 @@ def _is_recording() -> bool:
|
|||||||
return subprocess.run(["pidof", RECORDER], stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL).returncode == 0
|
return subprocess.run(["pidof", RECORDER], stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL).returncode == 0
|
||||||
|
|
||||||
|
|
||||||
def _notify(summary: str, body: str = "", actions: list = None, timeout: int = 5000) -> Optional[int]:
|
def _notify(summary: str, body: str = "", actions: list | None = None, timeout: int = 5000) -> Optional[int]:
|
||||||
args = ["notify-send", summary, body, "-t", str(timeout), "-p"]
|
args = ["notify-send", summary, body, "-t", str(timeout), "-p"]
|
||||||
if actions:
|
if actions:
|
||||||
for action in actions:
|
for action in actions:
|
||||||
@@ -49,14 +48,12 @@ def _notify(summary: str, body: str = "", actions: list = None, timeout: int = 5
|
|||||||
|
|
||||||
|
|
||||||
def _close_notification(notif_id: int):
|
def _close_notification(notif_id: int):
|
||||||
subprocess.run(["notify-send", "--close", str(notif_id)],
|
subprocess.run(["notify-send", "--close", str(notif_id)], stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
|
||||||
stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
|
|
||||||
|
|
||||||
|
|
||||||
def _get_monitors() -> list[dict]:
|
def _get_monitors() -> list[dict]:
|
||||||
try:
|
try:
|
||||||
res = subprocess.run(["hyprctl", "monitors", "-j"],
|
res = subprocess.run(["hyprctl", "monitors", "-j"], capture_output=True, text=True)
|
||||||
capture_output=True, text=True)
|
|
||||||
return json.loads(res.stdout)
|
return json.loads(res.stdout)
|
||||||
except Exception:
|
except Exception:
|
||||||
return []
|
return []
|
||||||
@@ -92,6 +89,7 @@ def _slurp_region() -> Optional[str]:
|
|||||||
|
|
||||||
def _parse_geometry(geometry: str) -> Optional[tuple[int, int, int, int]]:
|
def _parse_geometry(geometry: str) -> Optional[tuple[int, int, int, int]]:
|
||||||
import re
|
import re
|
||||||
|
|
||||||
match = re.match(r"(\d+)x(\d+)\+(\d+)\+(\d+)", geometry)
|
match = re.match(r"(\d+)x(\d+)\+(\d+)\+(\d+)", geometry)
|
||||||
if match:
|
if match:
|
||||||
return int(match.group(3)), int(match.group(4)), int(match.group(1)), int(match.group(2))
|
return int(match.group(3)), int(match.group(4)), int(match.group(1)), int(match.group(2))
|
||||||
@@ -139,8 +137,7 @@ def start_recording(region: Optional[str], sound: bool):
|
|||||||
cmd.extend(extra_args)
|
cmd.extend(extra_args)
|
||||||
cmd.extend(["-o", str(TEMP_RECORDING)])
|
cmd.extend(["-o", str(TEMP_RECORDING)])
|
||||||
|
|
||||||
subprocess.Popen(cmd, start_new_session=True,
|
subprocess.Popen(cmd, start_new_session=True, stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
|
||||||
stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
|
|
||||||
|
|
||||||
notif_id = _notify("Recording started", f"Saving to {TEMP_RECORDING}")
|
notif_id = _notify("Recording started", f"Saving to {TEMP_RECORDING}")
|
||||||
if notif_id is not None:
|
if notif_id is not None:
|
||||||
@@ -148,14 +145,12 @@ def start_recording(region: Optional[str], sound: bool):
|
|||||||
|
|
||||||
time.sleep(1)
|
time.sleep(1)
|
||||||
if not _is_recording():
|
if not _is_recording():
|
||||||
_notify("Recording failed",
|
_notify("Recording failed", "Check gpu-screen-recorder output.", timeout=5000)
|
||||||
"Check gpu-screen-recorder output.", timeout=5000)
|
|
||||||
raise typer.Exit(code=1)
|
raise typer.Exit(code=1)
|
||||||
|
|
||||||
|
|
||||||
def stop_recording(clipboard: bool):
|
def stop_recording(clipboard: bool):
|
||||||
subprocess.run(["pkill", "-f", RECORDER],
|
subprocess.run(["pkill", "-f", RECORDER], stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
|
||||||
stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
|
|
||||||
|
|
||||||
for _ in range(50):
|
for _ in range(50):
|
||||||
if not _is_recording():
|
if not _is_recording():
|
||||||
@@ -178,30 +173,31 @@ def stop_recording(clipboard: bool):
|
|||||||
NOTIF_ID_FILE.unlink()
|
NOTIF_ID_FILE.unlink()
|
||||||
|
|
||||||
if clipboard:
|
if clipboard:
|
||||||
subprocess.run(["wl-copy", "--type", "text/uri-list", f"file://{final_path}"],
|
subprocess.run(
|
||||||
stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
|
["wl-copy", "--type", "text/uri-list", f"file://{final_path}"],
|
||||||
|
stdout=subprocess.DEVNULL,
|
||||||
|
stderr=subprocess.DEVNULL,
|
||||||
|
)
|
||||||
|
|
||||||
_notify("Recording stopped", f"Saved to {final_path}", timeout=5000)
|
_notify("Recording stopped", f"Saved to {final_path}", timeout=5000)
|
||||||
|
|
||||||
|
|
||||||
def toggle_pause():
|
def toggle_pause():
|
||||||
subprocess.run(["pkill", "-USR2", "-f", RECORDER],
|
subprocess.run(["pkill", "-USR2", "-f", RECORDER], stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
|
||||||
stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL)
|
|
||||||
typer.echo("Toggled pause.")
|
typer.echo("Toggled pause.")
|
||||||
|
|
||||||
|
|
||||||
@app.command()
|
@app.command()
|
||||||
def record(
|
def record(
|
||||||
region: Optional[str] = typer.Option(
|
region: Optional[str] = typer.Option(
|
||||||
None, "--region", "-r",
|
None,
|
||||||
|
"--region",
|
||||||
|
"-r",
|
||||||
help="Record a region. Use 'slurp' (or omit value) to select interactively, or give 'WxH+X+Y'.",
|
help="Record a region. Use 'slurp' (or omit value) to select interactively, or give 'WxH+X+Y'.",
|
||||||
),
|
),
|
||||||
sound: bool = typer.Option(
|
sound: bool = typer.Option(False, "--sound", "-s", help="Record audio from default output."),
|
||||||
False, "--sound", "-s", help="Record audio from default output."),
|
pause: bool = typer.Option(False, "--pause", "-p", help="Toggle pause/resume."),
|
||||||
pause: bool = typer.Option(
|
clipboard: bool = typer.Option(False, "--clipboard", "-c", help="Copy the final recording path to clipboard."),
|
||||||
False, "--pause", "-p", help="Toggle pause/resume."),
|
|
||||||
clipboard: bool = typer.Option(
|
|
||||||
False, "--clipboard", "-c", help="Copy the final recording path to clipboard."),
|
|
||||||
):
|
):
|
||||||
"""Start or stop a screen recording with gpu-screen-recorder."""
|
"""Start or stop a screen recording with gpu-screen-recorder."""
|
||||||
if pause:
|
if pause:
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import typer
|
|||||||
import json
|
import json
|
||||||
import shutil
|
import shutil
|
||||||
import os
|
import os
|
||||||
|
import sys
|
||||||
import re
|
import re
|
||||||
import subprocess
|
import subprocess
|
||||||
|
|
||||||
@@ -15,11 +16,61 @@ from materialyoucolor.score.score import Score
|
|||||||
from materialyoucolor.dynamiccolor.material_dynamic_colors import MaterialDynamicColors
|
from materialyoucolor.dynamiccolor.material_dynamic_colors import MaterialDynamicColors
|
||||||
from materialyoucolor.hct.hct import Hct
|
from materialyoucolor.hct.hct import Hct
|
||||||
from materialyoucolor.utils.color_utils import argb_from_rgb
|
from materialyoucolor.utils.color_utils import argb_from_rgb
|
||||||
from materialyoucolor.utils.math_utils import difference_degrees, rotation_direction, sanitize_degrees_double
|
from materialyoucolor.utils.math_utils import (
|
||||||
|
difference_degrees,
|
||||||
|
rotation_direction,
|
||||||
|
sanitize_degrees_double,
|
||||||
|
)
|
||||||
|
|
||||||
app = typer.Typer()
|
app = typer.Typer()
|
||||||
|
|
||||||
|
|
||||||
|
def _complete_scheme_name(incomplete):
|
||||||
|
schemes = [
|
||||||
|
"fruit-salad",
|
||||||
|
"expressive",
|
||||||
|
"monochrome",
|
||||||
|
"rainbow",
|
||||||
|
"tonal-spot",
|
||||||
|
"neutral",
|
||||||
|
"fidelity",
|
||||||
|
"content",
|
||||||
|
"vibrant",
|
||||||
|
]
|
||||||
|
return [s for s in schemes if incomplete in s]
|
||||||
|
|
||||||
|
|
||||||
|
def _complete_preset(incomplete):
|
||||||
|
results = []
|
||||||
|
for sid, meta in list_schemes().items():
|
||||||
|
for v in meta.variants:
|
||||||
|
preset = f"{sid}:{v.id}"
|
||||||
|
if incomplete in preset:
|
||||||
|
results.append((preset, f"{meta.name} - {v.name}"))
|
||||||
|
return results
|
||||||
|
|
||||||
|
|
||||||
|
def _complete_mode(incomplete):
|
||||||
|
return [m for m in ("dark", "light") if incomplete in m]
|
||||||
|
|
||||||
|
|
||||||
|
def _complete_accent(ctx, incomplete):
|
||||||
|
preset_val = ctx.params.get("preset")
|
||||||
|
if preset_val:
|
||||||
|
try:
|
||||||
|
p_scheme, p_variant = resolve_preset(preset_val)
|
||||||
|
for v in list_schemes()[p_scheme].variants:
|
||||||
|
if v.id == p_variant:
|
||||||
|
return [a for a in v.accents if incomplete in a]
|
||||||
|
except (ValueError, KeyError):
|
||||||
|
pass
|
||||||
|
all_accents = set()
|
||||||
|
for meta in list_schemes().values():
|
||||||
|
for v in meta.variants:
|
||||||
|
all_accents.update(v.accents)
|
||||||
|
return [a for a in sorted(all_accents) if incomplete in a]
|
||||||
|
|
||||||
|
|
||||||
@app.command()
|
@app.command()
|
||||||
def list_presets(
|
def list_presets(
|
||||||
json_format: bool = typer.Option(False, "--json", help="Output in JSON format"),
|
json_format: bool = typer.Option(False, "--json", help="Output in JSON format"),
|
||||||
@@ -30,7 +81,7 @@ def list_presets(
|
|||||||
for sid, meta in sorted(schemes.items()):
|
for sid, meta in sorted(schemes.items()):
|
||||||
variants = {}
|
variants = {}
|
||||||
for v in meta.variants:
|
for v in meta.variants:
|
||||||
entry = {"modes": sorted(v.modes)}
|
entry: dict[str, Any] = {"modes": sorted(v.modes)}
|
||||||
if v.accents:
|
if v.accents:
|
||||||
entry["accents"] = sorted(v.accents)
|
entry["accents"] = sorted(v.accents)
|
||||||
entry["default_accent"] = sorted(v.accents)[0]
|
entry["default_accent"] = sorted(v.accents)[0]
|
||||||
@@ -55,14 +106,35 @@ def list_presets(
|
|||||||
|
|
||||||
@app.command()
|
@app.command()
|
||||||
def generate(
|
def generate(
|
||||||
image_path: Optional[Path] = typer.Option(None, help="Path to source image. Required for image mode."),
|
image_path: Optional[Path] = typer.Option(
|
||||||
scheme: Optional[str] = typer.Option(
|
None, help="Path to source image. Required for image mode."
|
||||||
None, help="Color scheme algorithm to use for image mode. Ignored in preset mode."
|
),
|
||||||
|
scheme: Optional[str] = typer.Option(
|
||||||
|
None,
|
||||||
|
help="Color scheme algorithm to use for image mode. Ignored in preset mode.",
|
||||||
|
autocompletion=_complete_scheme_name,
|
||||||
|
),
|
||||||
|
preset: Optional[str] = typer.Option(
|
||||||
|
None,
|
||||||
|
help="Name of a premade scheme in this format: <scheme>:<variant>",
|
||||||
|
autocompletion=_complete_preset,
|
||||||
|
),
|
||||||
|
mode: Optional[str] = typer.Option(
|
||||||
|
None,
|
||||||
|
help="Mode of the preset scheme (dark or light).",
|
||||||
|
autocompletion=_complete_mode,
|
||||||
|
),
|
||||||
|
accent: Optional[str] = typer.Option(
|
||||||
|
None,
|
||||||
|
help="Accent for schemes that support it (e.g. mauve).",
|
||||||
|
autocompletion=_complete_accent,
|
||||||
),
|
),
|
||||||
preset: Optional[str] = typer.Option(None, help="Name of a premade scheme in this format: <scheme>:<variant>"),
|
|
||||||
mode: Optional[str] = typer.Option(None, help="Mode of the preset scheme (dark or light)."),
|
|
||||||
accent: Optional[str] = typer.Option(None, help="Accent for schemes that support it (e.g. mauve)."),
|
|
||||||
):
|
):
|
||||||
|
if not any([image_path, scheme, preset, mode, accent]):
|
||||||
|
print(
|
||||||
|
"Hint: use --preset <scheme>:<variant> or --image-path <path>",
|
||||||
|
file=sys.stderr,
|
||||||
|
)
|
||||||
|
|
||||||
HOME = str(os.getenv("HOME"))
|
HOME = str(os.getenv("HOME"))
|
||||||
OUTPUT = Path(HOME + "/.local/state/zshell/scheme.json")
|
OUTPUT = Path(HOME + "/.local/state/zshell/scheme.json")
|
||||||
@@ -200,11 +272,15 @@ def generate(
|
|||||||
def harmonize(from_hct: Hct, to_hct: Hct, tone_boost: float) -> Hct:
|
def harmonize(from_hct: Hct, to_hct: Hct, tone_boost: float) -> Hct:
|
||||||
diff = difference_degrees(from_hct.hue, to_hct.hue)
|
diff = difference_degrees(from_hct.hue, to_hct.hue)
|
||||||
rotation = min(diff * 0.8, 100)
|
rotation = min(diff * 0.8, 100)
|
||||||
output_hue = sanitize_degrees_double(from_hct.hue + rotation * rotation_direction(from_hct.hue, to_hct.hue))
|
output_hue = sanitize_degrees_double(
|
||||||
|
from_hct.hue + rotation * rotation_direction(from_hct.hue, to_hct.hue)
|
||||||
|
)
|
||||||
tone = max(0.0, min(100.0, from_hct.tone * (1 + tone_boost)))
|
tone = max(0.0, min(100.0, from_hct.tone * (1 + tone_boost)))
|
||||||
return Hct.from_hct(output_hue, from_hct.chroma, tone)
|
return Hct.from_hct(output_hue, from_hct.chroma, tone)
|
||||||
|
|
||||||
def terminal_palette(colors: dict[str, str], mode: str, variant: str) -> dict[str, str]:
|
def terminal_palette(
|
||||||
|
colors: dict[str, str], mode: str, variant: str
|
||||||
|
) -> dict[str, str]:
|
||||||
light = mode.lower() == "light"
|
light = mode.lower() == "light"
|
||||||
|
|
||||||
key_hex = (
|
key_hex = (
|
||||||
@@ -236,7 +312,7 @@ def generate(
|
|||||||
|
|
||||||
image = Image.open(image_path)
|
image = Image.open(image_path)
|
||||||
image = image.convert("RGB")
|
image = image.convert("RGB")
|
||||||
image.thumbnail(size, Image.NEAREST)
|
image.thumbnail(size, Image.Resampling.NEAREST)
|
||||||
|
|
||||||
thumbnail_file.parent.mkdir(parents=True, exist_ok=True)
|
thumbnail_file.parent.mkdir(parents=True, exist_ok=True)
|
||||||
image.save(thumbnail_path, "JPEG")
|
image.save(thumbnail_path, "JPEG")
|
||||||
@@ -268,8 +344,15 @@ def generate(
|
|||||||
is_dark = ""
|
is_dark = ""
|
||||||
|
|
||||||
with Image.open(image_path) as img:
|
with Image.open(image_path) as img:
|
||||||
img.thumbnail((1, 1), Image.LANCZOS)
|
img.thumbnail((1, 1), Image.Resampling.LANCZOS)
|
||||||
hct = Hct.from_int(argb_from_rgb(*img.getpixel((0, 0))))
|
px = img.getpixel((0, 0))
|
||||||
|
if isinstance(px, (int, float)):
|
||||||
|
r = g = b = int(px)
|
||||||
|
elif px is not None:
|
||||||
|
r, g, b = int(px[0]), int(px[1]), int(px[2])
|
||||||
|
else:
|
||||||
|
r = g = b = 0
|
||||||
|
hct = Hct.from_int(argb_from_rgb(r, g, b))
|
||||||
is_dark = "light" if hct.tone > 50 else "dark"
|
is_dark = "light" if hct.tone > 50 else "dark"
|
||||||
|
|
||||||
return is_dark
|
return is_dark
|
||||||
@@ -431,6 +514,8 @@ def generate(
|
|||||||
|
|
||||||
raw = tpl_path.read_text(encoding="utf-8")
|
raw = tpl_path.read_text(encoding="utf-8")
|
||||||
out_path, body = split_directive_and_body(raw)
|
out_path, body = split_directive_and_body(raw)
|
||||||
|
if out_path is None:
|
||||||
|
continue
|
||||||
|
|
||||||
out_path.parent.mkdir(parents=True, exist_ok=True)
|
out_path.parent.mkdir(parents=True, exist_ok=True)
|
||||||
|
|
||||||
@@ -484,23 +569,30 @@ def generate(
|
|||||||
with CONFIG.open() as f:
|
with CONFIG.open() as f:
|
||||||
config = json.load(f)
|
config = json.load(f)
|
||||||
|
|
||||||
scheme = scheme or config["colors"]["schemeType"]
|
scheme_type = config["colors"].get("schemeType", "fruit-salad")
|
||||||
|
scheme = scheme or scheme_type
|
||||||
|
assert isinstance(scheme, str)
|
||||||
config_mode = config["general"]["color"]["mode"]
|
config_mode = config["general"]["color"]["mode"]
|
||||||
smart = bool(config["general"]["color"].get("smart", False))
|
smart = bool(config["general"]["color"].get("smart", False))
|
||||||
scheme_class = get_scheme_class(scheme)
|
scheme_class = get_scheme_class(scheme)
|
||||||
|
|
||||||
|
p_variant = "default"
|
||||||
if preset:
|
if preset:
|
||||||
p_scheme, p_variant = resolve_preset(preset)
|
p_scheme, p_variant = resolve_preset(preset)
|
||||||
schemes = list_schemes()
|
schemes = list_schemes()
|
||||||
if accent and p_scheme in schemes:
|
if accent and p_scheme in schemes:
|
||||||
meta = schemes[p_scheme]
|
meta = schemes[p_scheme]
|
||||||
var_accents = next((v.accents for v in meta.variants if v.id == p_variant), ())
|
var_accents = next(
|
||||||
|
(v.accents for v in meta.variants if v.id == p_variant), ()
|
||||||
|
)
|
||||||
if accent not in var_accents:
|
if accent not in var_accents:
|
||||||
available = ", ".join(var_accents) if var_accents else "none"
|
available = ", ".join(var_accents) if var_accents else "none"
|
||||||
raise typer.BadParameter(
|
raise typer.BadParameter(
|
||||||
f"Accent '{accent}' not available for '{p_scheme}:{p_variant}'. Available accents: {available}"
|
f"Accent '{accent}' not available for '{p_scheme}:{p_variant}'. Available accents: {available}"
|
||||||
)
|
)
|
||||||
palette_obj = get_palette(p_scheme, p_variant, mode or config_mode, accent=accent)
|
palette_obj = get_palette(
|
||||||
|
p_scheme, p_variant, mode or config_mode, accent=accent
|
||||||
|
)
|
||||||
colors = palette_obj.colors
|
colors = palette_obj.colors
|
||||||
effective_mode = palette_obj.mode
|
effective_mode = palette_obj.mode
|
||||||
name = palette_obj.scheme
|
name = palette_obj.scheme
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ import subprocess
|
|||||||
import sys
|
import sys
|
||||||
import time
|
import time
|
||||||
|
|
||||||
import click
|
|
||||||
import typer
|
import typer
|
||||||
|
|
||||||
args = ["qs", "-c", "zshell"]
|
args = ["qs", "-c", "zshell"]
|
||||||
@@ -14,7 +13,8 @@ app = typer.Typer()
|
|||||||
def kill():
|
def kill():
|
||||||
result = subprocess.run(args + ["kill"], capture_output=True)
|
result = subprocess.run(args + ["kill"], capture_output=True)
|
||||||
if result.returncode != 0:
|
if result.returncode != 0:
|
||||||
raise click.ClickException("No running instance to kill.")
|
sys.stderr.write("No running instance to kill.\n")
|
||||||
|
sys.exit(1)
|
||||||
sys.stderr.write(result.stderr.decode())
|
sys.stderr.write(result.stderr.decode())
|
||||||
|
|
||||||
|
|
||||||
@@ -23,10 +23,12 @@ def start_instance(no_daemon: bool = False) -> None:
|
|||||||
stdout = result.stdout.decode().strip()
|
stdout = result.stdout.decode().strip()
|
||||||
if stdout:
|
if stdout:
|
||||||
if "already running" in stdout.lower():
|
if "already running" in stdout.lower():
|
||||||
raise click.ClickException(stdout)
|
sys.stderr.write(stdout + "\n")
|
||||||
|
sys.exit(1)
|
||||||
if result.returncode != 0:
|
if result.returncode != 0:
|
||||||
stderr = result.stderr.decode().strip()
|
stderr = result.stderr.decode().strip()
|
||||||
raise click.ClickException(stderr)
|
sys.stderr.write(stderr + "\n")
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
|
|
||||||
@app.command()
|
@app.command()
|
||||||
@@ -50,7 +52,9 @@ def restart(no_daemon: bool = False):
|
|||||||
def show():
|
def show():
|
||||||
result = subprocess.run(args + ["ipc"] + ["show"], capture_output=True)
|
result = subprocess.run(args + ["ipc"] + ["show"], capture_output=True)
|
||||||
if result.returncode != 0:
|
if result.returncode != 0:
|
||||||
raise click.ClickException(result.stderr.decode().strip())
|
sys.stderr.write(result.stderr.decode())
|
||||||
|
sys.exit(1)
|
||||||
|
sys.stdout.write(result.stdout.decode())
|
||||||
sys.stderr.write(result.stderr.decode())
|
sys.stderr.write(result.stderr.decode())
|
||||||
|
|
||||||
|
|
||||||
@@ -58,7 +62,8 @@ def show():
|
|||||||
def log():
|
def log():
|
||||||
result = subprocess.run(args + ["log"], capture_output=True)
|
result = subprocess.run(args + ["log"], capture_output=True)
|
||||||
if result.returncode != 0:
|
if result.returncode != 0:
|
||||||
raise click.ClickException(result.stderr.decode().strip())
|
sys.stderr.write(result.stderr.decode())
|
||||||
|
sys.exit(1)
|
||||||
sys.stdout.write(result.stdout.decode())
|
sys.stdout.write(result.stdout.decode())
|
||||||
sys.stderr.write(result.stderr.decode())
|
sys.stderr.write(result.stderr.decode())
|
||||||
|
|
||||||
@@ -67,7 +72,8 @@ def log():
|
|||||||
def lock():
|
def lock():
|
||||||
result = subprocess.run(args + ["ipc"] + ["call"] + ["lock"] + ["lock"], capture_output=True)
|
result = subprocess.run(args + ["ipc"] + ["call"] + ["lock"] + ["lock"], capture_output=True)
|
||||||
if result.returncode != 0:
|
if result.returncode != 0:
|
||||||
raise click.ClickException(result.stderr.decode().strip())
|
sys.stderr.write(result.stderr.decode())
|
||||||
|
sys.exit(1)
|
||||||
sys.stderr.write(result.stderr.decode())
|
sys.stderr.write(result.stderr.decode())
|
||||||
|
|
||||||
|
|
||||||
@@ -75,5 +81,6 @@ def lock():
|
|||||||
def call(target: str, method: str, method_args: list[str] = typer.Argument(None)):
|
def call(target: str, method: str, method_args: list[str] = typer.Argument(None)):
|
||||||
result = subprocess.run(args + ["ipc"] + ["call"] + [target] + [method] + (method_args or []), capture_output=True)
|
result = subprocess.run(args + ["ipc"] + ["call"] + [target] + [method] + (method_args or []), capture_output=True)
|
||||||
if result.returncode != 0:
|
if result.returncode != 0:
|
||||||
raise click.ClickException(result.stderr.decode().strip())
|
sys.stderr.write(result.stderr.decode())
|
||||||
|
sys.exit(1)
|
||||||
sys.stderr.write(result.stderr.decode())
|
sys.stderr.write(result.stderr.decode())
|
||||||
|
|||||||
@@ -34,9 +34,9 @@ def lockscreen(
|
|||||||
return
|
return
|
||||||
|
|
||||||
if size[0] < 3840 or size[1] < 2160:
|
if size[0] < 3840 or size[1] < 2160:
|
||||||
img = img.resize((size[0] // 2, size[1] // 2), Image.NEAREST)
|
img = img.resize((size[0] // 2, size[1] // 2), Image.Resampling.NEAREST)
|
||||||
else:
|
else:
|
||||||
img = img.resize((size[0] // 4, size[1] // 4), Image.NEAREST)
|
img = img.resize((size[0] // 4, size[1] // 4), Image.Resampling.NEAREST)
|
||||||
|
|
||||||
img = img.filter(ImageFilter.GaussianBlur(blur_amount))
|
img = img.filter(ImageFilter.GaussianBlur(blur_amount))
|
||||||
|
|
||||||
|
|||||||
@@ -62,8 +62,9 @@ class TestStart:
|
|||||||
class TestShow:
|
class TestShow:
|
||||||
@patch("zshell.subcommands.shell.subprocess.run")
|
@patch("zshell.subcommands.shell.subprocess.run")
|
||||||
def test_show_runs_ipc_show(self, mock_run):
|
def test_show_runs_ipc_show(self, mock_run):
|
||||||
mock_run.return_value = CompletedProcess([], 0, b"", b"target visibilities\n")
|
mock_run.return_value = CompletedProcess([], 0, b"target visibilities\n", b"")
|
||||||
invoke("show")
|
result = invoke("show")
|
||||||
|
assert "target visibilities" in result.output
|
||||||
mock_run.assert_called_once_with(["qs", "-c", "zshell", "ipc", "show"], capture_output=True)
|
mock_run.assert_called_once_with(["qs", "-c", "zshell", "ipc", "show"], capture_output=True)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -180,6 +180,14 @@ export const settingsIndex = [
|
|||||||
section: "Bar",
|
section: "Bar",
|
||||||
keywords: ["smoothing", "rounding"],
|
keywords: ["smoothing", "rounding"],
|
||||||
},
|
},
|
||||||
|
// System tray section
|
||||||
|
{
|
||||||
|
name: "Tray icon size",
|
||||||
|
category: "bar",
|
||||||
|
categoryName: "Bar",
|
||||||
|
section: "Tray",
|
||||||
|
keywords: ["tray", "icon", "size"],
|
||||||
|
},
|
||||||
// Popouts section
|
// Popouts section
|
||||||
{
|
{
|
||||||
name: "Tray",
|
name: "Tray",
|
||||||
@@ -1003,7 +1011,7 @@ export const settingsIndex = [
|
|||||||
keywords: ["corner", "radius"],
|
keywords: ["corner", "radius"],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Enable drop shadow",
|
name: "Enable shadow",
|
||||||
category: "screenshot",
|
category: "screenshot",
|
||||||
categoryName: "Screenshot",
|
categoryName: "Screenshot",
|
||||||
section: "Screenshot",
|
section: "Screenshot",
|
||||||
@@ -1017,19 +1025,19 @@ export const settingsIndex = [
|
|||||||
keywords: ["rounded", "corners"],
|
keywords: ["rounded", "corners"],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Shadow blur radius",
|
name: "Shadow blur amount",
|
||||||
category: "screenshot",
|
category: "screenshot",
|
||||||
categoryName: "Screenshot",
|
categoryName: "Screenshot",
|
||||||
section: "Screenshot",
|
section: "Screenshot",
|
||||||
keywords: ["blur", "shadow", "radius"],
|
keywords: ["blur", "shadow", "radius"],
|
||||||
},
|
},
|
||||||
{
|
// {
|
||||||
name: "Shadow color",
|
// name: "Shadow color",
|
||||||
category: "screenshot",
|
// category: "screenshot",
|
||||||
categoryName: "Screenshot",
|
// categoryName: "Screenshot",
|
||||||
section: "Screenshot",
|
// section: "Screenshot",
|
||||||
keywords: ["color", "shadow"],
|
// keywords: ["color", "shadow"],
|
||||||
},
|
// },
|
||||||
{
|
{
|
||||||
name: "Shadow offset X",
|
name: "Shadow offset X",
|
||||||
category: "screenshot",
|
category: "screenshot",
|
||||||
|
|||||||
@@ -1,19 +1,25 @@
|
|||||||
//@ pragma UseQApplication
|
//@ pragma UseQApplication
|
||||||
//@ pragma Env QSG_RENDER_LOOP=threaded
|
//@ pragma Env QSG_RENDER_LOOP=threaded
|
||||||
//@ pragma Env QSG_RHI_BACKEND=vulkan
|
// @ pragma Env QSG_RHI_BACKEND=vulkan
|
||||||
//@ pragma Env QSG_NO_VSYNC=1
|
//@ pragma Env QSG_NO_VSYNC=1
|
||||||
//@ pragma Env QS_NO_RELOAD_POPUP=1
|
//@ pragma Env QS_NO_RELOAD_POPUP=1
|
||||||
//@ pragma Env QT_SCALE_FACTOR_ROUNDING_POLICY=Round
|
//@ pragma Env QT_SCALE_FACTOR_ROUNDING_POLICY=Round
|
||||||
//@ pragma DropExpensiveFonts
|
//@ pragma DropExpensiveFonts
|
||||||
import Quickshell
|
import Quickshell
|
||||||
|
import Quickshell.Services.UPower
|
||||||
import qs.Modules
|
import qs.Modules
|
||||||
import qs.Modules.Wallpaper
|
import qs.Modules.Wallpaper
|
||||||
import qs.Modules.Lock
|
import qs.Modules.Lock
|
||||||
import qs.Drawers
|
import qs.Drawers
|
||||||
import qs.Helpers
|
import qs.Helpers
|
||||||
import qs.Modules.Polkit
|
import qs.Modules.Polkit
|
||||||
|
import qs.Daemons
|
||||||
|
|
||||||
ShellRoot {
|
ShellRoot {
|
||||||
|
id: root
|
||||||
|
|
||||||
|
readonly property bool laptop: UPower.displayDevice.isLaptopBattery
|
||||||
|
|
||||||
settings.watchFiles: true
|
settings.watchFiles: true
|
||||||
|
|
||||||
Windows {
|
Windows {
|
||||||
@@ -38,4 +44,11 @@ ShellRoot {
|
|||||||
|
|
||||||
Polkit {
|
Polkit {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
LazyLoader {
|
||||||
|
activeAsync: root.laptop
|
||||||
|
|
||||||
|
component: Battery {
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Generated
+3
-807
@@ -8,47 +8,12 @@ version = "2.0.1"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "320119579fcad9c21884f5c4861d16174d0e06250625266f50fe6898340abefa"
|
checksum = "320119579fcad9c21884f5c4861d16174d0e06250625266f50fe6898340abefa"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "aligned"
|
|
||||||
version = "0.4.3"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "ee4508988c62edf04abd8d92897fca0c2995d907ce1dfeaf369dac3716a40685"
|
|
||||||
dependencies = [
|
|
||||||
"as-slice",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "aligned-vec"
|
|
||||||
version = "0.6.4"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "dc890384c8602f339876ded803c97ad529f3842aba97f6392b3dba0dd171769b"
|
|
||||||
dependencies = [
|
|
||||||
"equator",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "anyhow"
|
name = "anyhow"
|
||||||
version = "1.0.102"
|
version = "1.0.102"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "7f202df86484c868dbad7eaa557ef785d5c66295e41b460ef922eca0723b842c"
|
checksum = "7f202df86484c868dbad7eaa557ef785d5c66295e41b460ef922eca0723b842c"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "arbitrary"
|
|
||||||
version = "1.4.2"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "c3d036a3c4ab069c7b410a2ce876bd74808d2d0888a82667669f8e783a898bf1"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "arg_enum_proc_macro"
|
|
||||||
version = "0.3.4"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "0ae92a5119aa49cdbcf6b9f893fe4e1d98b04ccbf82ee0584ad948a44a734dea"
|
|
||||||
dependencies = [
|
|
||||||
"proc-macro2",
|
|
||||||
"quote",
|
|
||||||
"syn",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "arrayref"
|
name = "arrayref"
|
||||||
version = "0.3.9"
|
version = "0.3.9"
|
||||||
@@ -61,103 +26,18 @@ version = "0.7.6"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "7c02d123df017efcdfbd739ef81735b36c5ba83ec3c59c80a9d7ecc718f92e50"
|
checksum = "7c02d123df017efcdfbd739ef81735b36c5ba83ec3c59c80a9d7ecc718f92e50"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "as-slice"
|
|
||||||
version = "0.2.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "516b6b4f0e40d50dcda9365d53964ec74560ad4284da2e7fc97122cd83174516"
|
|
||||||
dependencies = [
|
|
||||||
"stable_deref_trait",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "autocfg"
|
name = "autocfg"
|
||||||
version = "1.5.0"
|
version = "1.5.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "c08606f8c3cbf4ce6ec8e28fb0014a2c086708fe954eaa885384a6165172e7e8"
|
checksum = "c08606f8c3cbf4ce6ec8e28fb0014a2c086708fe954eaa885384a6165172e7e8"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "av-scenechange"
|
|
||||||
version = "0.14.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "0f321d77c20e19b92c39e7471cf986812cbb46659d2af674adc4331ef3f18394"
|
|
||||||
dependencies = [
|
|
||||||
"aligned",
|
|
||||||
"anyhow",
|
|
||||||
"arg_enum_proc_macro",
|
|
||||||
"arrayvec",
|
|
||||||
"log",
|
|
||||||
"num-rational",
|
|
||||||
"num-traits",
|
|
||||||
"pastey",
|
|
||||||
"rayon",
|
|
||||||
"thiserror",
|
|
||||||
"v_frame",
|
|
||||||
"y4m",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "av1-grain"
|
|
||||||
version = "0.2.5"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "8cfddb07216410377231960af4fcab838eaa12e013417781b78bd95ee22077f8"
|
|
||||||
dependencies = [
|
|
||||||
"anyhow",
|
|
||||||
"arrayvec",
|
|
||||||
"log",
|
|
||||||
"nom",
|
|
||||||
"num-rational",
|
|
||||||
"v_frame",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "avif-serialize"
|
|
||||||
version = "0.8.8"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "375082f007bd67184fb9c0374614b29f9aaa604ec301635f72338bb65386a53d"
|
|
||||||
dependencies = [
|
|
||||||
"arrayvec",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "bit_field"
|
|
||||||
version = "0.10.3"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "1e4b40c7323adcfc0a41c4b88143ed58346ff65a288fc144329c5c45e05d70c6"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "bitflags"
|
|
||||||
version = "1.3.2"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a"
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "bitflags"
|
name = "bitflags"
|
||||||
version = "2.11.1"
|
version = "2.11.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "c4512299f36f043ab09a583e57bceb5a5aab7a73db1805848e8fef3c9e8c78b3"
|
checksum = "c4512299f36f043ab09a583e57bceb5a5aab7a73db1805848e8fef3c9e8c78b3"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "bitstream-io"
|
|
||||||
version = "4.10.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "7eff00be299a18769011411c9def0d827e8f2d7bf0c3dbf53633147a8867fd1f"
|
|
||||||
dependencies = [
|
|
||||||
"no_std_io2",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "built"
|
|
||||||
version = "0.8.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "f4ad8f11f288f48ca24471bbd51ac257aaeaaa07adae295591266b792902ae64"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "bumpalo"
|
|
||||||
version = "3.20.2"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "5d20789868f4b01b2f2caec9f5c4e0213b41e3e5702a50157d699ae31ced2fcb"
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "bytemuck"
|
name = "bytemuck"
|
||||||
version = "1.25.0"
|
version = "1.25.0"
|
||||||
@@ -170,30 +50,12 @@ version = "0.1.0"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "8f1fe948ff07f4bd06c30984e69f5b4899c516a3ef74f34df92a2df2ab535495"
|
checksum = "8f1fe948ff07f4bd06c30984e69f5b4899c516a3ef74f34df92a2df2ab535495"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "cc"
|
|
||||||
version = "1.2.61"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "d16d90359e986641506914ba71350897565610e87ce0ad9e6f28569db3dd5c6d"
|
|
||||||
dependencies = [
|
|
||||||
"find-msvc-tools",
|
|
||||||
"jobserver",
|
|
||||||
"libc",
|
|
||||||
"shlex",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "cfg-if"
|
name = "cfg-if"
|
||||||
version = "1.0.4"
|
version = "1.0.4"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "9330f8b2ff13f34540b44e946ef35111825727b38d33286ef986142615121801"
|
checksum = "9330f8b2ff13f34540b44e946ef35111825727b38d33286ef986142615121801"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "color_quant"
|
|
||||||
version = "1.1.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "3d7b894f5411737b7867f4827955924d7c254fc9f4d91a6aad6b097804b1018b"
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "crc32fast"
|
name = "crc32fast"
|
||||||
version = "1.5.0"
|
version = "1.5.0"
|
||||||
@@ -203,84 +65,6 @@ dependencies = [
|
|||||||
"cfg-if",
|
"cfg-if",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "crossbeam-deque"
|
|
||||||
version = "0.8.6"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "9dd111b7b7f7d55b72c0a6ae361660ee5853c9af73f70c3c2ef6858b950e2e51"
|
|
||||||
dependencies = [
|
|
||||||
"crossbeam-epoch",
|
|
||||||
"crossbeam-utils",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "crossbeam-epoch"
|
|
||||||
version = "0.9.18"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "5b82ac4a3c2ca9c3460964f020e1402edd5753411d7737aa39c3714ad1b5420e"
|
|
||||||
dependencies = [
|
|
||||||
"crossbeam-utils",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "crossbeam-utils"
|
|
||||||
version = "0.8.21"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "d0a5c400df2834b80a4c3327b3aad3a4c4cd4de0629063962b03235697506a28"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "crunchy"
|
|
||||||
version = "0.2.4"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "460fbee9c2c2f33933d720630a6a0bac33ba7053db5344fac858d4b8952d77d5"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "either"
|
|
||||||
version = "1.15.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "48c757948c5ede0e46177b7add2e67155f70e33c07fea8284df6576da70b3719"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "equator"
|
|
||||||
version = "0.4.2"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "4711b213838dfee0117e3be6ac926007d7f433d7bbe33595975d4190cb07e6fc"
|
|
||||||
dependencies = [
|
|
||||||
"equator-macro",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "equator-macro"
|
|
||||||
version = "0.4.2"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "44f23cf4b44bfce11a86ace86f8a73ffdec849c9fd00a386a53d278bd9e81fb3"
|
|
||||||
dependencies = [
|
|
||||||
"proc-macro2",
|
|
||||||
"quote",
|
|
||||||
"syn",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "exr"
|
|
||||||
version = "1.74.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "4300e043a56aa2cb633c01af81ca8f699a321879a7854d3896a0ba89056363be"
|
|
||||||
dependencies = [
|
|
||||||
"bit_field",
|
|
||||||
"half",
|
|
||||||
"lebe",
|
|
||||||
"miniz_oxide",
|
|
||||||
"rayon-core",
|
|
||||||
"smallvec",
|
|
||||||
"zune-inflate",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "fax"
|
|
||||||
version = "0.2.7"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "caf1079563223d5d59d83c85886a56e586cfd5c1a26292e971a0fa266531ac5a"
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "fdeflate"
|
name = "fdeflate"
|
||||||
version = "0.3.7"
|
version = "0.3.7"
|
||||||
@@ -290,12 +74,6 @@ dependencies = [
|
|||||||
"simd-adler32",
|
"simd-adler32",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "find-msvc-tools"
|
|
||||||
version = "0.1.9"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "5baebc0774151f905a1a2cc41989300b1e6fbb29aff0ceffa1064fdd3088d582"
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "flate2"
|
name = "flate2"
|
||||||
version = "1.1.9"
|
version = "1.1.9"
|
||||||
@@ -306,39 +84,6 @@ dependencies = [
|
|||||||
"miniz_oxide",
|
"miniz_oxide",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "getrandom"
|
|
||||||
version = "0.3.4"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "899def5c37c4fd7b2664648c28120ecec138e4d395b459e5ca34f9cce2dd77fd"
|
|
||||||
dependencies = [
|
|
||||||
"cfg-if",
|
|
||||||
"libc",
|
|
||||||
"r-efi",
|
|
||||||
"wasip2",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "gif"
|
|
||||||
version = "0.14.2"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "ee8cfcc411d9adbbaba82fb72661cc1bcca13e8bba98b364e62b2dba8f960159"
|
|
||||||
dependencies = [
|
|
||||||
"color_quant",
|
|
||||||
"weezl",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "half"
|
|
||||||
version = "2.7.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "6ea2d84b969582b4b1864a92dc5d27cd2b77b622a8d79306834f1be5ba20d84b"
|
|
||||||
dependencies = [
|
|
||||||
"cfg-if",
|
|
||||||
"crunchy",
|
|
||||||
"zerocopy",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "image"
|
name = "image"
|
||||||
version = "0.25.10"
|
version = "0.25.10"
|
||||||
@@ -347,56 +92,9 @@ checksum = "85ab80394333c02fe689eaf900ab500fbd0c2213da414687ebf995a65d5a6104"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"bytemuck",
|
"bytemuck",
|
||||||
"byteorder-lite",
|
"byteorder-lite",
|
||||||
"color_quant",
|
|
||||||
"exr",
|
|
||||||
"gif",
|
|
||||||
"image-webp",
|
|
||||||
"moxcms",
|
"moxcms",
|
||||||
"num-traits",
|
"num-traits",
|
||||||
"png 0.18.1",
|
"png",
|
||||||
"qoi",
|
|
||||||
"ravif",
|
|
||||||
"rayon",
|
|
||||||
"rgb",
|
|
||||||
"tiff",
|
|
||||||
"zune-core",
|
|
||||||
"zune-jpeg",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "image-webp"
|
|
||||||
version = "0.2.4"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "525e9ff3e1a4be2fbea1fdf0e98686a6d98b4d8f937e1bf7402245af1909e8c3"
|
|
||||||
dependencies = [
|
|
||||||
"byteorder-lite",
|
|
||||||
"quick-error",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "imgref"
|
|
||||||
version = "1.12.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "40fac9d56ed6437b198fddba683305e8e2d651aa42647f00f5ae542e7f5c94a2"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "interpolate_name"
|
|
||||||
version = "0.2.4"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "c34819042dc3d3971c46c2190835914dfbe0c3c13f61449b2997f4e9722dfa60"
|
|
||||||
dependencies = [
|
|
||||||
"proc-macro2",
|
|
||||||
"quote",
|
|
||||||
"syn",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "itertools"
|
|
||||||
version = "0.14.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "2b192c782037fadd9cfa75548310488aabdbf3d2da73885b31bd0abd03351285"
|
|
||||||
dependencies = [
|
|
||||||
"either",
|
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@@ -405,63 +103,12 @@ version = "1.0.18"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "8f42a60cbdf9a97f5d2305f08a87dc4e09308d1276d28c869c684d7777685682"
|
checksum = "8f42a60cbdf9a97f5d2305f08a87dc4e09308d1276d28c869c684d7777685682"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "jobserver"
|
|
||||||
version = "0.1.34"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "9afb3de4395d6b3e67a780b6de64b51c978ecf11cb9a462c66be7d4ca9039d33"
|
|
||||||
dependencies = [
|
|
||||||
"getrandom",
|
|
||||||
"libc",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "lebe"
|
|
||||||
version = "0.5.3"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "7a79a3332a6609480d7d0c9eab957bca6b455b91bb84e66d19f5ff66294b85b8"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "libc"
|
|
||||||
version = "0.2.186"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "68ab91017fe16c622486840e4c83c9a37afeff978bd239b5293d61ece587de66"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "libfuzzer-sys"
|
|
||||||
version = "0.4.12"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "f12a681b7dd8ce12bff52488013ba614b869148d54dd79836ab85aafdd53f08d"
|
|
||||||
dependencies = [
|
|
||||||
"arbitrary",
|
|
||||||
"cc",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "log"
|
name = "log"
|
||||||
version = "0.4.29"
|
version = "0.4.29"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "5e5032e24019045c762d3c0f28f5b6b8bbf38563a65908389bf7978758920897"
|
checksum = "5e5032e24019045c762d3c0f28f5b6b8bbf38563a65908389bf7978758920897"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "loop9"
|
|
||||||
version = "0.1.5"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "0fae87c125b03c1d2c0150c90365d7d6bcc53fb73a9acaef207d2d065860f062"
|
|
||||||
dependencies = [
|
|
||||||
"imgref",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "maybe-rayon"
|
|
||||||
version = "0.1.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "8ea1f30cedd69f0a2954655f7188c6a834246d2bcf1e315e2ac40c4b24dc9519"
|
|
||||||
dependencies = [
|
|
||||||
"cfg-if",
|
|
||||||
"rayon",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "memchr"
|
name = "memchr"
|
||||||
version = "2.8.0"
|
version = "2.8.0"
|
||||||
@@ -488,77 +135,6 @@ dependencies = [
|
|||||||
"pxfm",
|
"pxfm",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "new_debug_unreachable"
|
|
||||||
version = "1.0.6"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "650eef8c711430f1a879fdd01d4745a7deea475becfb90269c06775983bbf086"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "no_std_io2"
|
|
||||||
version = "0.9.3"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "b51ed7824b6e07d354605f4abb3d9d300350701299da96642ee084f5ce631550"
|
|
||||||
dependencies = [
|
|
||||||
"memchr",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "nom"
|
|
||||||
version = "8.0.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "df9761775871bdef83bee530e60050f7e54b1105350d6884eb0fb4f46c2f9405"
|
|
||||||
dependencies = [
|
|
||||||
"memchr",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "noop_proc_macro"
|
|
||||||
version = "0.3.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "0676bb32a98c1a483ce53e500a81ad9c3d5b3f7c920c28c24e9cb0980d0b5bc8"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "num-bigint"
|
|
||||||
version = "0.4.6"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "a5e44f723f1133c9deac646763579fdb3ac745e418f2a7af9cd0c431da1f20b9"
|
|
||||||
dependencies = [
|
|
||||||
"num-integer",
|
|
||||||
"num-traits",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "num-derive"
|
|
||||||
version = "0.4.2"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "ed3955f1a9c7c0c15e092f9c887db08b1fc683305fdf6eb6684f22555355e202"
|
|
||||||
dependencies = [
|
|
||||||
"proc-macro2",
|
|
||||||
"quote",
|
|
||||||
"syn",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "num-integer"
|
|
||||||
version = "0.1.46"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "7969661fd2958a5cb096e56c8e1ad0444ac2bbcd0061bd28660485a44879858f"
|
|
||||||
dependencies = [
|
|
||||||
"num-traits",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "num-rational"
|
|
||||||
version = "0.4.2"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "f83d14da390562dca69fc84082e73e548e1ad308d24accdedd2720017cb37824"
|
|
||||||
dependencies = [
|
|
||||||
"num-bigint",
|
|
||||||
"num-integer",
|
|
||||||
"num-traits",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "num-traits"
|
name = "num-traits"
|
||||||
version = "0.2.19"
|
version = "0.2.19"
|
||||||
@@ -568,59 +144,19 @@ dependencies = [
|
|||||||
"autocfg",
|
"autocfg",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "once_cell"
|
|
||||||
version = "1.21.4"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "9f7c3e4beb33f85d45ae3e3a1792185706c8e16d043238c593331cc7cd313b50"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "paste"
|
|
||||||
version = "1.0.15"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "57c0d7b74b563b49d38dae00a0c37d4d6de9b432382b2892f0574ddcae73fd0a"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "pastey"
|
|
||||||
version = "0.1.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "35fb2e5f958ec131621fdd531e9fc186ed768cbe395337403ae56c17a74c68ec"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "png"
|
|
||||||
version = "0.17.16"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "82151a2fc869e011c153adc57cf2789ccb8d9906ce52c0b39a6b5697749d7526"
|
|
||||||
dependencies = [
|
|
||||||
"bitflags 1.3.2",
|
|
||||||
"crc32fast",
|
|
||||||
"fdeflate",
|
|
||||||
"flate2",
|
|
||||||
"miniz_oxide",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "png"
|
name = "png"
|
||||||
version = "0.18.1"
|
version = "0.18.1"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "60769b8b31b2a9f263dae2776c37b1b28ae246943cf719eb6946a1db05128a61"
|
checksum = "60769b8b31b2a9f263dae2776c37b1b28ae246943cf719eb6946a1db05128a61"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"bitflags 2.11.1",
|
"bitflags",
|
||||||
"crc32fast",
|
"crc32fast",
|
||||||
"fdeflate",
|
"fdeflate",
|
||||||
"flate2",
|
"flate2",
|
||||||
"miniz_oxide",
|
"miniz_oxide",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "ppv-lite86"
|
|
||||||
version = "0.2.21"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "85eae3c4ed2f50dcfe72643da4befc30deadb458a9b590d720cde2f2b1e97da9"
|
|
||||||
dependencies = [
|
|
||||||
"zerocopy",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "proc-macro2"
|
name = "proc-macro2"
|
||||||
version = "1.0.106"
|
version = "1.0.106"
|
||||||
@@ -630,46 +166,12 @@ dependencies = [
|
|||||||
"unicode-ident",
|
"unicode-ident",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "profiling"
|
|
||||||
version = "1.0.17"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "3eb8486b569e12e2c32ad3e204dbaba5e4b5b216e9367044f25f1dba42341773"
|
|
||||||
dependencies = [
|
|
||||||
"profiling-procmacros",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "profiling-procmacros"
|
|
||||||
version = "1.0.17"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "52717f9a02b6965224f95ca2a81e2e0c5c43baacd28ca057577988930b6c3d5b"
|
|
||||||
dependencies = [
|
|
||||||
"quote",
|
|
||||||
"syn",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "pxfm"
|
name = "pxfm"
|
||||||
version = "0.1.29"
|
version = "0.1.29"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "e0c5ccf5294c6ccd63a74f1565028353830a9c2f5eb0c682c355c471726a6e3f"
|
checksum = "e0c5ccf5294c6ccd63a74f1565028353830a9c2f5eb0c682c355c471726a6e3f"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "qoi"
|
|
||||||
version = "0.4.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "7f6d64c71eb498fe9eae14ce4ec935c555749aef511cca85b5568910d6e48001"
|
|
||||||
dependencies = [
|
|
||||||
"bytemuck",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "quick-error"
|
|
||||||
version = "2.0.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "a993555f31e5a609f617c12db6250dedcac1b0a85076912c436e6fc9b2c8e6a3"
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "quote"
|
name = "quote"
|
||||||
version = "1.0.45"
|
version = "1.0.45"
|
||||||
@@ -679,123 +181,6 @@ dependencies = [
|
|||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "r-efi"
|
|
||||||
version = "5.3.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "69cdb34c158ceb288df11e18b4bd39de994f6657d83847bdffdbd7f346754b0f"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "rand"
|
|
||||||
version = "0.9.4"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "44c5af06bb1b7d3216d91932aed5265164bf384dc89cd6ba05cf59a35f5f76ea"
|
|
||||||
dependencies = [
|
|
||||||
"rand_chacha",
|
|
||||||
"rand_core",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "rand_chacha"
|
|
||||||
version = "0.9.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "d3022b5f1df60f26e1ffddd6c66e8aa15de382ae63b3a0c1bfc0e4d3e3f325cb"
|
|
||||||
dependencies = [
|
|
||||||
"ppv-lite86",
|
|
||||||
"rand_core",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "rand_core"
|
|
||||||
version = "0.9.5"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "76afc826de14238e6e8c374ddcc1fa19e374fd8dd986b0d2af0d02377261d83c"
|
|
||||||
dependencies = [
|
|
||||||
"getrandom",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "rav1e"
|
|
||||||
version = "0.8.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "43b6dd56e85d9483277cde964fd1bdb0428de4fec5ebba7540995639a21cb32b"
|
|
||||||
dependencies = [
|
|
||||||
"aligned-vec",
|
|
||||||
"arbitrary",
|
|
||||||
"arg_enum_proc_macro",
|
|
||||||
"arrayvec",
|
|
||||||
"av-scenechange",
|
|
||||||
"av1-grain",
|
|
||||||
"bitstream-io",
|
|
||||||
"built",
|
|
||||||
"cfg-if",
|
|
||||||
"interpolate_name",
|
|
||||||
"itertools",
|
|
||||||
"libc",
|
|
||||||
"libfuzzer-sys",
|
|
||||||
"log",
|
|
||||||
"maybe-rayon",
|
|
||||||
"new_debug_unreachable",
|
|
||||||
"noop_proc_macro",
|
|
||||||
"num-derive",
|
|
||||||
"num-traits",
|
|
||||||
"paste",
|
|
||||||
"profiling",
|
|
||||||
"rand",
|
|
||||||
"rand_chacha",
|
|
||||||
"simd_helpers",
|
|
||||||
"thiserror",
|
|
||||||
"v_frame",
|
|
||||||
"wasm-bindgen",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "ravif"
|
|
||||||
version = "0.13.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "e52310197d971b0f5be7fe6b57530dcd27beb35c1b013f29d66c1ad73fbbcc45"
|
|
||||||
dependencies = [
|
|
||||||
"avif-serialize",
|
|
||||||
"imgref",
|
|
||||||
"loop9",
|
|
||||||
"quick-error",
|
|
||||||
"rav1e",
|
|
||||||
"rayon",
|
|
||||||
"rgb",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "rayon"
|
|
||||||
version = "1.12.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "fb39b166781f92d482534ef4b4b1b2568f42613b53e5b6c160e24cfbfa30926d"
|
|
||||||
dependencies = [
|
|
||||||
"either",
|
|
||||||
"rayon-core",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "rayon-core"
|
|
||||||
version = "1.13.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "22e18b0f0062d30d4230b2e85ff77fdfe4326feb054b9783a3460d8435c8ab91"
|
|
||||||
dependencies = [
|
|
||||||
"crossbeam-deque",
|
|
||||||
"crossbeam-utils",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "rgb"
|
|
||||||
version = "0.8.53"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "47b34b781b31e5d73e9fbc8689c70551fd1ade9a19e3e28cfec8580a79290cc4"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "rustversion"
|
|
||||||
version = "1.0.22"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "b39cdef0fa800fc44525c84ccb54a029961a8215f9619753635a9c0d2538d46d"
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "serde"
|
name = "serde"
|
||||||
version = "1.0.228"
|
version = "1.0.228"
|
||||||
@@ -839,39 +224,12 @@ dependencies = [
|
|||||||
"zmij",
|
"zmij",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "shlex"
|
|
||||||
version = "1.3.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64"
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "simd-adler32"
|
name = "simd-adler32"
|
||||||
version = "0.3.9"
|
version = "0.3.9"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "703d5c7ef118737c72f1af64ad2f6f8c5e1921f818cdcb97b8fe6fc69bf66214"
|
checksum = "703d5c7ef118737c72f1af64ad2f6f8c5e1921f818cdcb97b8fe6fc69bf66214"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "simd_helpers"
|
|
||||||
version = "0.1.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "95890f873bec569a0362c235787f3aca6e1e887302ba4840839bcc6459c42da6"
|
|
||||||
dependencies = [
|
|
||||||
"quote",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "smallvec"
|
|
||||||
version = "1.15.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "67b1b7a3b5fe4f1376887184045fcf45c69e92af734b7aaddc05fb777b6fbd03"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "stable_deref_trait"
|
|
||||||
version = "1.2.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "6ce2be8dc25455e1f91df71bfa12ad37d7af1092ae736f3a6cd0e37bc7810596"
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "strict-num"
|
name = "strict-num"
|
||||||
version = "0.1.1"
|
version = "0.1.1"
|
||||||
@@ -889,40 +247,6 @@ dependencies = [
|
|||||||
"unicode-ident",
|
"unicode-ident",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "thiserror"
|
|
||||||
version = "2.0.18"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "4288b5bcbc7920c07a1149a35cf9590a2aa808e0bc1eafaade0b80947865fbc4"
|
|
||||||
dependencies = [
|
|
||||||
"thiserror-impl",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "thiserror-impl"
|
|
||||||
version = "2.0.18"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "ebc4ee7f67670e9b64d05fa4253e753e016c6c95ff35b89b7941d6b856dec1d5"
|
|
||||||
dependencies = [
|
|
||||||
"proc-macro2",
|
|
||||||
"quote",
|
|
||||||
"syn",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "tiff"
|
|
||||||
version = "0.11.3"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "b63feaf3343d35b6ca4d50483f94843803b0f51634937cc2ec519fc32232bc52"
|
|
||||||
dependencies = [
|
|
||||||
"fax",
|
|
||||||
"flate2",
|
|
||||||
"half",
|
|
||||||
"quick-error",
|
|
||||||
"weezl",
|
|
||||||
"zune-jpeg",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "tiny-skia"
|
name = "tiny-skia"
|
||||||
version = "0.11.4"
|
version = "0.11.4"
|
||||||
@@ -934,7 +258,6 @@ dependencies = [
|
|||||||
"bytemuck",
|
"bytemuck",
|
||||||
"cfg-if",
|
"cfg-if",
|
||||||
"log",
|
"log",
|
||||||
"png 0.17.16",
|
|
||||||
"tiny-skia-path",
|
"tiny-skia-path",
|
||||||
]
|
]
|
||||||
|
|
||||||
@@ -955,109 +278,6 @@ version = "1.0.24"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "e6e4313cd5fcd3dad5cafa179702e2b244f760991f45397d14d4ebf38247da75"
|
checksum = "e6e4313cd5fcd3dad5cafa179702e2b244f760991f45397d14d4ebf38247da75"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "v_frame"
|
|
||||||
version = "0.3.9"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "666b7727c8875d6ab5db9533418d7c764233ac9c0cff1d469aec8fa127597be2"
|
|
||||||
dependencies = [
|
|
||||||
"aligned-vec",
|
|
||||||
"num-traits",
|
|
||||||
"wasm-bindgen",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "wasip2"
|
|
||||||
version = "1.0.3+wasi-0.2.9"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "20064672db26d7cdc89c7798c48a0fdfac8213434a1186e5ef29fd560ae223d6"
|
|
||||||
dependencies = [
|
|
||||||
"wit-bindgen",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "wasm-bindgen"
|
|
||||||
version = "0.2.120"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "df52b6d9b87e0c74c9edfa1eb2d9bf85e5d63515474513aa50fa181b3c4f5db1"
|
|
||||||
dependencies = [
|
|
||||||
"cfg-if",
|
|
||||||
"once_cell",
|
|
||||||
"rustversion",
|
|
||||||
"wasm-bindgen-macro",
|
|
||||||
"wasm-bindgen-shared",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "wasm-bindgen-macro"
|
|
||||||
version = "0.2.120"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "78b1041f495fb322e64aca85f5756b2172e35cd459376e67f2a6c9dffcedb103"
|
|
||||||
dependencies = [
|
|
||||||
"quote",
|
|
||||||
"wasm-bindgen-macro-support",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "wasm-bindgen-macro-support"
|
|
||||||
version = "0.2.120"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "9dcd0ff20416988a18ac686d4d4d0f6aae9ebf08a389ff5d29012b05af2a1b41"
|
|
||||||
dependencies = [
|
|
||||||
"bumpalo",
|
|
||||||
"proc-macro2",
|
|
||||||
"quote",
|
|
||||||
"syn",
|
|
||||||
"wasm-bindgen-shared",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "wasm-bindgen-shared"
|
|
||||||
version = "0.2.120"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "49757b3c82ebf16c57d69365a142940b384176c24df52a087fb748e2085359ea"
|
|
||||||
dependencies = [
|
|
||||||
"unicode-ident",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "weezl"
|
|
||||||
version = "0.1.12"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "a28ac98ddc8b9274cb41bb4d9d4d5c425b6020c50c46f25559911905610b4a88"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "wit-bindgen"
|
|
||||||
version = "0.57.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "1ebf944e87a7c253233ad6766e082e3cd714b5d03812acc24c318f549614536e"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "y4m"
|
|
||||||
version = "0.8.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "7a5a4b21e1a62b67a2970e6831bc091d7b87e119e7f9791aef9702e3bef04448"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "zerocopy"
|
|
||||||
version = "0.8.48"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "eed437bf9d6692032087e337407a86f04cd8d6a16a37199ed57949d415bd68e9"
|
|
||||||
dependencies = [
|
|
||||||
"zerocopy-derive",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "zerocopy-derive"
|
|
||||||
version = "0.8.48"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "70e3cd084b1788766f53af483dd21f93881ff30d7320490ec3ef7526d203bad4"
|
|
||||||
dependencies = [
|
|
||||||
"proc-macro2",
|
|
||||||
"quote",
|
|
||||||
"syn",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "zmij"
|
name = "zmij"
|
||||||
version = "1.0.21"
|
version = "1.0.21"
|
||||||
@@ -1066,7 +286,7 @@ checksum = "b8848ee67ecc8aedbaf3e4122217aff892639231befc6a1b58d29fff4c2cabaa"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "zshell-img-tools"
|
name = "zshell-img-tools"
|
||||||
version = "0.1.0"
|
version = "0.2.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"image",
|
"image",
|
||||||
@@ -1074,27 +294,3 @@ dependencies = [
|
|||||||
"serde_json",
|
"serde_json",
|
||||||
"tiny-skia",
|
"tiny-skia",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "zune-core"
|
|
||||||
version = "0.5.1"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "cb8a0807f7c01457d0379ba880ba6322660448ddebc890ce29bb64da71fb40f9"
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "zune-inflate"
|
|
||||||
version = "0.2.54"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "73ab332fe2f6680068f3582b16a24f90ad7096d5d39b974d1c0aff0125116f02"
|
|
||||||
dependencies = [
|
|
||||||
"simd-adler32",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "zune-jpeg"
|
|
||||||
version = "0.5.15"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "27bc9d5b815bc103f142aa054f561d9187d191692ec7c2d1e2b4737f8dbd7296"
|
|
||||||
dependencies = [
|
|
||||||
"zune-core",
|
|
||||||
]
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "zshell-img-tools"
|
name = "zshell-img-tools"
|
||||||
version = "0.1.0"
|
version = "0.2.0"
|
||||||
edition = "2024"
|
edition = "2024"
|
||||||
|
|
||||||
[[bin]]
|
[[bin]]
|
||||||
@@ -8,10 +8,10 @@ name = "zshell-img-tools"
|
|||||||
path = "src/main.rs"
|
path = "src/main.rs"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
image = { version = "0.25", features = ["png"] }
|
image = { version = "0.25", default-features = false, features = ["png"] }
|
||||||
tiny-skia = "0.11"
|
tiny-skia = { version = "0.11", default-features = false, features = ["std", "simd"] }
|
||||||
serde = { version = "1", features = ["derive"] }
|
serde = { version = "1", features = ["derive"] }
|
||||||
anyhow = "1"
|
anyhow = "1.0"
|
||||||
serde_json = "1.0.149"
|
serde_json = "1.0.149"
|
||||||
|
|
||||||
[profile.release]
|
[profile.release]
|
||||||
|
|||||||
@@ -1,6 +0,0 @@
|
|||||||
# What_That_Claude_DO?
|
|
||||||
|
|
||||||
What That Claude Do? (WTCD)
|
|
||||||
A repository of random things I ask Claude to do for me.
|
|
||||||
|
|
||||||
In this case it is creating a screenshot tool
|
|
||||||
@@ -10,14 +10,13 @@ pub struct Config {
|
|||||||
|
|
||||||
#[derive(Debug, Clone, Serialize, Deserialize)]
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
||||||
pub struct EffectsConfig {
|
pub struct EffectsConfig {
|
||||||
pub mode: String,
|
pub radius: f32,
|
||||||
pub rounded_corners: bool,
|
pub shadow: bool,
|
||||||
pub corner_radius: f32,
|
pub rounding: bool,
|
||||||
pub drop_shadow: bool,
|
pub shadow_blur: f32,
|
||||||
pub shadow_blur_radius: f32,
|
pub shadow_color: [u8; 4],
|
||||||
pub shadow_offset_x: f32,
|
pub shadow_offset_x: f32,
|
||||||
pub shadow_offset_y: f32,
|
pub shadow_offset_y: f32,
|
||||||
pub shadow_color: [u8; 4],
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Config {
|
impl Config {
|
||||||
|
|||||||
@@ -5,15 +5,15 @@ use tiny_skia::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
pub fn apply_effects(img: RgbaImage, cfg: &EffectsConfig) -> RgbaImage {
|
pub fn apply_effects(img: RgbaImage, cfg: &EffectsConfig) -> RgbaImage {
|
||||||
let img = if cfg.rounded_corners {
|
let img = if cfg.rounding {
|
||||||
apply_rounded_corners(img, cfg.corner_radius)
|
apply_rounding(img, cfg.radius)
|
||||||
} else {
|
} else {
|
||||||
img
|
img
|
||||||
};
|
};
|
||||||
if cfg.drop_shadow {
|
if cfg.shadow {
|
||||||
apply_drop_shadow(
|
apply_shadow(
|
||||||
img,
|
img,
|
||||||
cfg.shadow_blur_radius,
|
cfg.shadow_blur,
|
||||||
cfg.shadow_offset_x,
|
cfg.shadow_offset_x,
|
||||||
cfg.shadow_offset_y,
|
cfg.shadow_offset_y,
|
||||||
cfg.shadow_color,
|
cfg.shadow_color,
|
||||||
@@ -23,7 +23,7 @@ pub fn apply_effects(img: RgbaImage, cfg: &EffectsConfig) -> RgbaImage {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn apply_rounded_corners(img: RgbaImage, radius: f32) -> RgbaImage {
|
pub fn apply_rounding(img: RgbaImage, radius: f32) -> RgbaImage {
|
||||||
let (w, h) = img.dimensions();
|
let (w, h) = img.dimensions();
|
||||||
let mut mask = Pixmap::new(w, h).expect("mask pixmap");
|
let mut mask = Pixmap::new(w, h).expect("mask pixmap");
|
||||||
let path = rounded_rect_path(0.0, 0.0, w as f32, h as f32, radius);
|
let path = rounded_rect_path(0.0, 0.0, w as f32, h as f32, radius);
|
||||||
@@ -47,16 +47,21 @@ pub fn apply_rounded_corners(img: RgbaImage, radius: f32) -> RgbaImage {
|
|||||||
pixmap_to_rgba_image(pixmap)
|
pixmap_to_rgba_image(pixmap)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn apply_drop_shadow(
|
pub fn apply_shadow(
|
||||||
img: RgbaImage,
|
img: RgbaImage,
|
||||||
blur_radius: f32,
|
blur: f32,
|
||||||
offset_x: f32,
|
offset_x: f32,
|
||||||
offset_y: f32,
|
offset_y: f32,
|
||||||
|
// blur_passes: u32,
|
||||||
shadow_color: [u8; 4],
|
shadow_color: [u8; 4],
|
||||||
) -> RgbaImage {
|
) -> RgbaImage {
|
||||||
let (iw, ih) = img.dimensions();
|
let (iw, ih) = img.dimensions();
|
||||||
let br = blur_radius.ceil() as u32;
|
let br = blur.ceil() as u32;
|
||||||
let spread = br * 2;
|
let bp = 1;
|
||||||
|
// Original idea
|
||||||
|
// let spread = br * bp;
|
||||||
|
// Claude is hallucinating but let's try it **Worked btw**
|
||||||
|
let spread = (br as f32 * (bp as f32).sqrt() * 2.0).ceil() as u32;
|
||||||
|
|
||||||
let extra_left = spread + (-offset_x).max(0.0).ceil() as u32;
|
let extra_left = spread + (-offset_x).max(0.0).ceil() as u32;
|
||||||
let extra_top = spread + (-offset_y).max(0.0).ceil() as u32;
|
let extra_top = spread + (-offset_y).max(0.0).ceil() as u32;
|
||||||
@@ -86,8 +91,11 @@ pub fn apply_drop_shadow(
|
|||||||
|
|
||||||
tint_pixmap_as_shadow(&mut shadow_pixmap, shadow_color);
|
tint_pixmap_as_shadow(&mut shadow_pixmap, shadow_color);
|
||||||
|
|
||||||
|
// Shadow
|
||||||
let shadow_img = pixmap_to_rgba_image(shadow_pixmap);
|
let shadow_img = pixmap_to_rgba_image(shadow_pixmap);
|
||||||
let blurred = box_blur_rgba(&shadow_img, br);
|
// Shadow blur
|
||||||
|
let blurred = box_blur_rgba(&shadow_img, br, bp);
|
||||||
|
// Shadow pos
|
||||||
let blurred_pixmap = rgba_image_to_pixmap(&blurred);
|
let blurred_pixmap = rgba_image_to_pixmap(&blurred);
|
||||||
|
|
||||||
let mut canvas = Pixmap::new(canvas_w, canvas_h).expect("canvas pixmap");
|
let mut canvas = Pixmap::new(canvas_w, canvas_h).expect("canvas pixmap");
|
||||||
@@ -136,6 +144,7 @@ fn rounded_rect_path(x: f32, y: f32, w: f32, h: f32, r: f32) -> Path {
|
|||||||
pb.finish().expect("rounded rect path")
|
pb.finish().expect("rounded rect path")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Shadow pos
|
||||||
fn rgba_image_to_pixmap(img: &RgbaImage) -> Pixmap {
|
fn rgba_image_to_pixmap(img: &RgbaImage) -> Pixmap {
|
||||||
let (w, h) = img.dimensions();
|
let (w, h) = img.dimensions();
|
||||||
let mut pixmap = Pixmap::new(w, h).expect("pixmap alloc");
|
let mut pixmap = Pixmap::new(w, h).expect("pixmap alloc");
|
||||||
@@ -154,6 +163,7 @@ fn rgba_image_to_pixmap(img: &RgbaImage) -> Pixmap {
|
|||||||
pixmap
|
pixmap
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Shadow
|
||||||
fn pixmap_to_rgba_image(pixmap: Pixmap) -> RgbaImage {
|
fn pixmap_to_rgba_image(pixmap: Pixmap) -> RgbaImage {
|
||||||
let (w, h) = (pixmap.width(), pixmap.height());
|
let (w, h) = (pixmap.width(), pixmap.height());
|
||||||
let mut out = RgbaImage::new(w, h);
|
let mut out = RgbaImage::new(w, h);
|
||||||
@@ -176,31 +186,16 @@ fn pixmap_to_rgba_image(pixmap: Pixmap) -> RgbaImage {
|
|||||||
out
|
out
|
||||||
}
|
}
|
||||||
|
|
||||||
fn tint_pixmap_as_shadow(pixmap: &mut Pixmap, color: [u8; 4]) {
|
// Shadow blur
|
||||||
let [sr, sg, sb, _] = color;
|
fn box_blur_rgba(img: &RgbaImage, radius: u32, bp: u32) -> RgbaImage {
|
||||||
for px in pixmap.pixels_mut() {
|
|
||||||
let a = px.alpha();
|
|
||||||
if a > 0 {
|
|
||||||
let af = a as f32 / 255.0;
|
|
||||||
*px = tiny_skia::PremultipliedColorU8::from_rgba(
|
|
||||||
(sr as f32 * af) as u8,
|
|
||||||
(sg as f32 * af) as u8,
|
|
||||||
(sb as f32 * af) as u8,
|
|
||||||
a,
|
|
||||||
)
|
|
||||||
.unwrap_or(tiny_skia::PremultipliedColorU8::TRANSPARENT);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn box_blur_rgba(img: &RgbaImage, radius: u32) -> RgbaImage {
|
|
||||||
if radius == 0 {
|
if radius == 0 {
|
||||||
return img.clone();
|
return img.clone();
|
||||||
}
|
}
|
||||||
let mut buf = sliding_horizontal(img, radius);
|
let mut buf = img.clone();
|
||||||
buf = sliding_vertical(&buf, radius);
|
for _ in 0..bp {
|
||||||
buf = sliding_horizontal(&buf, radius);
|
buf = sliding_horizontal(&buf, radius);
|
||||||
buf = sliding_vertical(&buf, radius);
|
buf = sliding_vertical(&buf, radius);
|
||||||
|
}
|
||||||
buf
|
buf
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -250,6 +245,23 @@ fn sliding_horizontal(img: &RgbaImage, radius: u32) -> RgbaImage {
|
|||||||
out
|
out
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn tint_pixmap_as_shadow(pixmap: &mut Pixmap, color: [u8; 4]) {
|
||||||
|
let [sr, sg, sb, _] = color;
|
||||||
|
for px in pixmap.pixels_mut() {
|
||||||
|
let a = px.alpha();
|
||||||
|
if a > 0 {
|
||||||
|
let af = a as f32 / 255.0;
|
||||||
|
*px = tiny_skia::PremultipliedColorU8::from_rgba(
|
||||||
|
(sr as f32 * af) as u8,
|
||||||
|
(sg as f32 * af) as u8,
|
||||||
|
(sb as f32 * af) as u8,
|
||||||
|
a,
|
||||||
|
)
|
||||||
|
.unwrap_or(tiny_skia::PremultipliedColorU8::TRANSPARENT);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn sliding_vertical(img: &RgbaImage, radius: u32) -> RgbaImage {
|
fn sliding_vertical(img: &RgbaImage, radius: u32) -> RgbaImage {
|
||||||
let (w, h) = img.dimensions();
|
let (w, h) = img.dimensions();
|
||||||
let r = radius as i32;
|
let r = radius as i32;
|
||||||
|
|||||||
+147
-107
@@ -1,21 +1,19 @@
|
|||||||
mod config;
|
mod config;
|
||||||
mod effects;
|
mod effects;
|
||||||
|
|
||||||
use anyhow::{Context, Result, bail};
|
use anyhow::{bail, Context, Result};
|
||||||
use std::io::Write as _;
|
use std::io::Write as _;
|
||||||
use std::process::{Command, Stdio};
|
use std::process::{Command, Stdio};
|
||||||
|
|
||||||
/// CLI overrides that map 1:1 to `EffectsConfig` fields.
|
|
||||||
/// All fields are `Option<T>` so we can tell "not supplied" from any concrete value.
|
|
||||||
#[derive(Default)]
|
#[derive(Default)]
|
||||||
struct CliOverrides {
|
struct CliOverrides {
|
||||||
rounded_corners: Option<bool>,
|
rounding: Option<bool>,
|
||||||
corner_radius: Option<f32>,
|
radius: Option<f32>,
|
||||||
drop_shadow: Option<bool>,
|
shadow: Option<bool>,
|
||||||
shadow_blur_radius: Option<f32>,
|
shadow_blur: Option<f32>,
|
||||||
shadow_offset_x: Option<f32>,
|
shadow_offset_x: Option<f32>,
|
||||||
shadow_offset_y: Option<f32>,
|
shadow_offset_y: Option<f32>,
|
||||||
/// Accepted as four comma-separated u8 values, e.g. `255,0,0,200`
|
// Accepted as four comma-separated u8 values, e.g. `255,0,0,200`
|
||||||
shadow_color: Option<[u8; 4]>,
|
shadow_color: Option<[u8; 4]>,
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -30,137 +28,145 @@ fn parse_bool(s: &str) -> Result<bool> {
|
|||||||
fn parse_shadow_color(s: &str) -> Result<[u8; 4]> {
|
fn parse_shadow_color(s: &str) -> Result<[u8; 4]> {
|
||||||
let parts: Vec<&str> = s.split(',').collect();
|
let parts: Vec<&str> = s.split(',').collect();
|
||||||
if parts.len() != 4 {
|
if parts.len() != 4 {
|
||||||
bail!("--shadow_color expects four comma-separated u8 values, e.g. 255,0,0,200");
|
bail!("--shadow-color expects four comma-separated u8 values, e.g. 255,0,0,200");
|
||||||
}
|
}
|
||||||
let r = parts[0]
|
let r = parts[0]
|
||||||
.trim()
|
.trim()
|
||||||
.parse::<u8>()
|
.parse::<u8>()
|
||||||
.context("shadow_color red channel")?;
|
.context("shadow-color red channel")?;
|
||||||
let g = parts[1]
|
let g = parts[1]
|
||||||
.trim()
|
.trim()
|
||||||
.parse::<u8>()
|
.parse::<u8>()
|
||||||
.context("shadow_color green channel")?;
|
.context("shadow-color green channel")?;
|
||||||
let b = parts[2]
|
let b = parts[2]
|
||||||
.trim()
|
.trim()
|
||||||
.parse::<u8>()
|
.parse::<u8>()
|
||||||
.context("shadow_color blue channel")?;
|
.context("shadow-color blue channel")?;
|
||||||
let a = parts[3]
|
let a = parts[3]
|
||||||
.trim()
|
.trim()
|
||||||
.parse::<u8>()
|
.parse::<u8>()
|
||||||
.context("shadow_color alpha channel")?;
|
.context("shadow-color alpha channel")?;
|
||||||
Ok([r, g, b, a])
|
Ok([r, g, b, a])
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() -> Result<()> {
|
fn extract_image_path() -> Option<String> {
|
||||||
|
let args: Vec<String> = std::env::args().skip(1).collect();
|
||||||
|
args.windows(2)
|
||||||
|
.find(|w| w[0] == "--image")
|
||||||
|
.map(|w| w[1].clone())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
if let Some(path) = extract_image_path() && let Err(e) = run() {
|
||||||
|
eprintln!("Error: {}", e);
|
||||||
|
push_image(&path).ok();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn run() -> Result<()> {
|
||||||
let args: Vec<String> = std::env::args().skip(1).collect();
|
let args: Vec<String> = std::env::args().skip(1).collect();
|
||||||
|
|
||||||
let mut image_path: Option<String> = None;
|
let mut image_path: Option<String> = None;
|
||||||
let mut overrides = CliOverrides::default();
|
let mut overrides = CliOverrides::default();
|
||||||
|
let mut scale: Option<f32> = None;
|
||||||
|
|
||||||
let mut i = 0;
|
let mut i = 0;
|
||||||
while i < args.len() {
|
while i < args.len() {
|
||||||
match args[i].as_str() {
|
match args[i].as_str() {
|
||||||
"--image" => {
|
"--image" => {
|
||||||
i += 1;
|
image_path = Some(next_arg(&args, &mut i, "--image")?);
|
||||||
image_path = Some(
|
|
||||||
args.get(i)
|
|
||||||
.cloned()
|
|
||||||
.context("Expected a path after --image")?,
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
"--rounded_corners" => {
|
"--rounding" => {
|
||||||
i += 1;
|
let val = next_arg(&args, &mut i, "--rounding")?;
|
||||||
let val = args
|
overrides.rounding = Some(parse_bool(&val)?);
|
||||||
.get(i)
|
|
||||||
.context("Expected true/false after --rounded_corners")?;
|
|
||||||
overrides.rounded_corners = Some(parse_bool(val)?);
|
|
||||||
}
|
}
|
||||||
"--corner_radius" => {
|
"--radius" => {
|
||||||
i += 1;
|
let val = next_arg(&args, &mut i, "--radius")?;
|
||||||
let val = args
|
overrides.radius = Some(val.parse::<f32>().context("--radius must be a number")?);
|
||||||
.get(i)
|
}
|
||||||
.context("Expected a number after --corner_radius")?;
|
"--shadow" => {
|
||||||
overrides.corner_radius = Some(
|
let val = next_arg(&args, &mut i, "--shadow")?;
|
||||||
|
overrides.shadow = Some(parse_bool(&val)?);
|
||||||
|
}
|
||||||
|
"--shadow-blur" => {
|
||||||
|
let val = next_arg(&args, &mut i, "--shadow-blur")?;
|
||||||
|
overrides.shadow_blur = Some(
|
||||||
val.parse::<f32>()
|
val.parse::<f32>()
|
||||||
.context("--corner_radius must be a number")?,
|
.context("--shadow-blur must be a number")?,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
"--drop_shadow" => {
|
"--shadow-offset-x" => {
|
||||||
i += 1;
|
let val = next_arg(&args, &mut i, "--shadow-offset-x")?;
|
||||||
let val = args
|
|
||||||
.get(i)
|
|
||||||
.context("Expected true/false after --drop_shadow")?;
|
|
||||||
overrides.drop_shadow = Some(parse_bool(val)?);
|
|
||||||
}
|
|
||||||
"--shadow_blur_radius" => {
|
|
||||||
i += 1;
|
|
||||||
let val = args
|
|
||||||
.get(i)
|
|
||||||
.context("Expected a number after --shadow_blur_radius")?;
|
|
||||||
overrides.shadow_blur_radius = Some(
|
|
||||||
val.parse::<f32>()
|
|
||||||
.context("--shadow_blur_radius must be a number")?,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
"--shadow_offset_x" => {
|
|
||||||
i += 1;
|
|
||||||
let val = args
|
|
||||||
.get(i)
|
|
||||||
.context("Expected a number after --shadow_offset_x")?;
|
|
||||||
overrides.shadow_offset_x = Some(
|
overrides.shadow_offset_x = Some(
|
||||||
val.parse::<f32>()
|
val.parse::<f32>()
|
||||||
.context("--shadow_offset_x must be a number")?,
|
.context("--shadow-offset-x must be a number")?,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
"--shadow_offset_y" => {
|
"--shadow-offset-y" => {
|
||||||
i += 1;
|
let val = next_arg(&args, &mut i, "--shadow-offset-y")?;
|
||||||
let val = args
|
|
||||||
.get(i)
|
|
||||||
.context("Expected a number after --shadow_offset_y")?;
|
|
||||||
overrides.shadow_offset_y = Some(
|
overrides.shadow_offset_y = Some(
|
||||||
val.parse::<f32>()
|
val.parse::<f32>()
|
||||||
.context("--shadow_offset_y must be a number")?,
|
.context("--shadow-offset-y must be a number")?,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
"--shadow_color" => {
|
"--shadow-color" => {
|
||||||
i += 1;
|
let val = next_arg(&args, &mut i, "--shadow-color")?;
|
||||||
let val = args
|
overrides.shadow_color = Some(parse_shadow_color(&val)?);
|
||||||
.get(i)
|
|
||||||
.context("Expected r,g,b,a after --shadow_color")?;
|
|
||||||
overrides.shadow_color = Some(parse_shadow_color(val)?);
|
|
||||||
}
|
}
|
||||||
unknown => bail!("Unknown argument: {unknown}"),
|
"--scale" => {
|
||||||
|
let val = next_arg(&args, &mut i, "--scale")?;
|
||||||
|
scale = Some(val.parse::<f32>().context("--scale must be a number")?);
|
||||||
|
}
|
||||||
|
unknown => bail!("Unknown argument: {}", unknown),
|
||||||
}
|
}
|
||||||
|
|
||||||
i += 1;
|
i += 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
let image_path = image_path.context("Missing --image <path>")?;
|
let image_path = image_path.context("Missing --image <path>")?;
|
||||||
|
|
||||||
let config = config::Config::load().context("Failed to load config")?;
|
// Check if any arguments were provided
|
||||||
|
let cli_args_provided = overrides.rounding.is_some()
|
||||||
|
|| overrides.radius.is_some()
|
||||||
|
|| overrides.shadow.is_some()
|
||||||
|
|| overrides.shadow_blur.is_some()
|
||||||
|
|| overrides.shadow_offset_x.is_some()
|
||||||
|
|| overrides.shadow_offset_y.is_some()
|
||||||
|
|| overrides.shadow_color.is_some();
|
||||||
|
let mut effects = if cli_args_provided {
|
||||||
|
// If all args provided
|
||||||
|
let rounding = overrides.rounding.context("Missing --rounding")?;
|
||||||
|
let radius = overrides.radius.context("Missing --radius")?;
|
||||||
|
let shadow = overrides.shadow.context("Missing --shadow")?;
|
||||||
|
let shadow_blur = overrides.shadow_blur.context("Missing --shadow-blur")?;
|
||||||
|
let shadow_offset_x = overrides
|
||||||
|
.shadow_offset_x
|
||||||
|
.context("Missing --shadow-offset-x")?;
|
||||||
|
let shadow_offset_y = overrides
|
||||||
|
.shadow_offset_y
|
||||||
|
.context("Missing --shadow-offset-y")?;
|
||||||
|
let shadow_color = overrides.shadow_color.context("Missing --shadow-color")?;
|
||||||
|
config::EffectsConfig {
|
||||||
|
rounding,
|
||||||
|
radius,
|
||||||
|
shadow,
|
||||||
|
shadow_blur,
|
||||||
|
shadow_offset_x,
|
||||||
|
shadow_offset_y,
|
||||||
|
shadow_color,
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// If not all args were provided use config file
|
||||||
|
let config = config::Config::load()?;
|
||||||
|
config.screenshot
|
||||||
|
};
|
||||||
|
|
||||||
let mut effects = config.screenshot;
|
// if scale is set do
|
||||||
if effects.mode == "auto" {
|
if let Some(scale) = scale.filter(|&s| s != 1.0) {
|
||||||
if let Some(v) = overrides.rounded_corners {
|
effects.radius *= scale;
|
||||||
effects.rounded_corners = v;
|
effects.shadow_blur *= scale;
|
||||||
}
|
effects.shadow_offset_x *= scale;
|
||||||
if let Some(v) = overrides.corner_radius {
|
effects.shadow_offset_y *= scale;
|
||||||
effects.corner_radius = v;
|
|
||||||
}
|
|
||||||
if let Some(v) = overrides.drop_shadow {
|
|
||||||
effects.drop_shadow = v;
|
|
||||||
}
|
|
||||||
if let Some(v) = overrides.shadow_blur_radius {
|
|
||||||
effects.shadow_blur_radius = v;
|
|
||||||
}
|
|
||||||
if let Some(v) = overrides.shadow_offset_x {
|
|
||||||
effects.shadow_offset_x = v;
|
|
||||||
}
|
|
||||||
if let Some(v) = overrides.shadow_offset_y {
|
|
||||||
effects.shadow_offset_y = v;
|
|
||||||
}
|
|
||||||
if let Some(v) = overrides.shadow_color {
|
|
||||||
effects.shadow_color = v;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Err(e) = process_image(&image_path, &effects) {
|
if let Err(e) = process_image(&image_path, &effects) {
|
||||||
@@ -170,6 +176,49 @@ fn main() -> Result<()> {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn next_arg(args: &[String], i: &mut usize, flag: &str) -> Result<String> {
|
||||||
|
*i += 1;
|
||||||
|
|
||||||
|
let val = args
|
||||||
|
.get(*i)
|
||||||
|
.context(format!("Expected value after {}", flag))?;
|
||||||
|
|
||||||
|
if val.starts_with('-') {
|
||||||
|
bail!("Expected value after {}, found flag {}", flag, val);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(val.clone())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn push_image(path: &str) -> Result<()> {
|
||||||
|
let img = image::open(path)
|
||||||
|
.with_context(|| format!("Failed to open image '{path}'"))?
|
||||||
|
.into_rgba8();
|
||||||
|
|
||||||
|
let mut png_bytes: Vec<u8> = Vec::new();
|
||||||
|
image::DynamicImage::ImageRgba8(img)
|
||||||
|
.write_to(
|
||||||
|
&mut std::io::Cursor::new(&mut png_bytes),
|
||||||
|
image::ImageFormat::Png,
|
||||||
|
)
|
||||||
|
.context("Failed to encode processed image as PNG")?;
|
||||||
|
|
||||||
|
let mut child = Command::new("swappy")
|
||||||
|
.args(["-f", "-"])
|
||||||
|
.stdin(Stdio::piped())
|
||||||
|
.spawn()
|
||||||
|
.context("Failed to spawn swappy. Is it installed and in PATH?")?;
|
||||||
|
|
||||||
|
// Writes the PNG bytes to swappy's stdin and then closes
|
||||||
|
if let Some(mut stdin) = child.stdin.take() {
|
||||||
|
stdin
|
||||||
|
.write_all(&png_bytes)
|
||||||
|
.context("Failed to write image data to swappy")?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
fn process_image(path: &str, effects: &config::EffectsConfig) -> Result<()> {
|
fn process_image(path: &str, effects: &config::EffectsConfig) -> Result<()> {
|
||||||
let img = image::open(path)
|
let img = image::open(path)
|
||||||
.with_context(|| format!("Failed to open image '{path}'"))?
|
.with_context(|| format!("Failed to open image '{path}'"))?
|
||||||
@@ -191,20 +240,11 @@ fn process_image(path: &str, effects: &config::EffectsConfig) -> Result<()> {
|
|||||||
.spawn()
|
.spawn()
|
||||||
.context("Failed to spawn swappy. Is it installed and in PATH?")?;
|
.context("Failed to spawn swappy. Is it installed and in PATH?")?;
|
||||||
|
|
||||||
child
|
// Writes the PNG bytes to swappy's stdin and then closes
|
||||||
.stdin
|
if let Some(mut stdin) = child.stdin.take() {
|
||||||
.take()
|
stdin
|
||||||
.context("Failed to get swappy stdin")?
|
.write_all(&png_bytes)
|
||||||
.write_all(&png_bytes)
|
.context("Failed to write image data to swappy")?;
|
||||||
.context("Failed to write image data to swappy")?;
|
|
||||||
|
|
||||||
let status = child.wait().context("Failed to wait for swappy")?;
|
|
||||||
|
|
||||||
if !status.success() {
|
|
||||||
eprintln!(
|
|
||||||
"swappy exited with non-zero status for '{}': {}",
|
|
||||||
path, status
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
|
|||||||
Reference in New Issue
Block a user