booru: put context menu in loader, add tooltip for tags

This commit is contained in:
end-4
2025-05-16 14:56:14 +02:00
parent 3b764ca70e
commit 9a68f80ffa
2 changed files with 98 additions and 67 deletions
@@ -92,3 +92,20 @@ function splitMarkdownBlocks(markdown) {
function escapeBackslashes(str) { function escapeBackslashes(str) {
return str.replace(/\\/g, '\\\\'); return str.replace(/\\/g, '\\\\');
} }
function wordWrap(str, maxLen) {
if (!str) return "";
let words = str.split(" ");
let lines = [];
let current = "";
for (let i = 0; i < words.length; ++i) {
if ((current + (current.length > 0 ? " " : "") + words[i]).length > maxLen) {
if (current.length > 0) lines.push(current);
current = words[i];
} else {
current += (current.length > 0 ? " " : "") + words[i];
}
}
if (current.length > 0) lines.push(current);
return lines.join("\n");
}
@@ -7,6 +7,7 @@ import Qt.labs.platform
import QtQuick import QtQuick
import QtQuick.Controls import QtQuick.Controls
import QtQuick.Layouts import QtQuick.Layouts
import Quickshell
import Quickshell.Io import Quickshell.Io
import Quickshell.Hyprland import Quickshell.Hyprland
import Qt5Compat.GraphicalEffects import Qt5Compat.GraphicalEffects
@@ -21,9 +22,9 @@ Button {
property string nsfwPath property string nsfwPath
property string fileName: decodeURIComponent((imageData.file_url).substring((imageData.file_url).lastIndexOf('/') + 1)) property string fileName: decodeURIComponent((imageData.file_url).substring((imageData.file_url).lastIndexOf('/') + 1))
property string filePath: `${root.previewDownloadPath}/${root.fileName}` property string filePath: `${root.previewDownloadPath}/${root.fileName}`
property int maxTagStringLineLength: 50
property bool showActions: false property bool showActions: false
Process { Process {
id: downloadProcess id: downloadProcess
running: false running: false
@@ -95,6 +96,10 @@ Button {
PointingHandInteraction {} PointingHandInteraction {}
StyledToolTip {
content: StringUtils.wordWrap(root.imageData.tags, root.maxTagStringLineLength)
}
background: Rectangle { background: Rectangle {
color: menuButton.down ? Appearance.transparentize(Appearance.mix(Appearance.m3colors.m3surface, Appearance.m3colors.m3onSurface, 0.6), 0.1) : color: menuButton.down ? Appearance.transparentize(Appearance.mix(Appearance.m3colors.m3surface, Appearance.m3colors.m3onSurface, 0.6), 0.1) :
menuButton.hovered ? Appearance.transparentize(Appearance.mix(Appearance.m3colors.m3surface, Appearance.m3colors.m3onSurface, 0.8), 0.2) : menuButton.hovered ? Appearance.transparentize(Appearance.mix(Appearance.m3colors.m3surface, Appearance.m3colors.m3onSurface, 0.8), 0.2) :
@@ -114,83 +119,92 @@ Button {
} }
} }
Rectangle { Loader {
id: contextMenu id: contextMenuLoader
opacity: root.showActions ? 1 : 0 active: root.showActions
visible: opacity > 0
radius: Appearance.rounding.small
color: Appearance.m3colors.m3surfaceContainer
anchors.top: menuButton.bottom anchors.top: menuButton.bottom
anchors.right: parent.right anchors.right: parent.right
anchors.margins: 8 anchors.margins: 8
implicitHeight: contextMenuColumnLayout.implicitHeight + radius * 2
implicitWidth: contextMenuColumnLayout.implicitWidth
Behavior on opacity { sourceComponent: Item {
NumberAnimation { width: contextMenu.width
duration: Appearance.animation.elementMoveFast.duration height: contextMenu.height
easing.type: Appearance.animation.elementMoveFast.type
easing.bezierCurve: Appearance.animation.elementMoveFast.bezierCurve
}
}
ColumnLayout { Rectangle {
id: contextMenuColumnLayout id: contextMenu
anchors.centerIn: parent anchors.centerIn: parent
spacing: 0 opacity: root.showActions ? 1 : 0
visible: opacity > 0
radius: Appearance.rounding.small
color: Appearance.m3colors.m3surfaceContainer
implicitHeight: contextMenuColumnLayout.implicitHeight + radius * 2
implicitWidth: contextMenuColumnLayout.implicitWidth
MenuButton { Behavior on opacity {
id: openFileLinkButton NumberAnimation {
Layout.fillWidth: true duration: Appearance.animation.elementMoveFast.duration
buttonText: qsTr("Open file link") easing.type: Appearance.animation.elementMoveFast.type
onClicked: { easing.bezierCurve: Appearance.animation.elementMoveFast.bezierCurve
root.showActions = false }
Qt.openUrlExternally(root.imageData.file_url) }
ColumnLayout {
id: contextMenuColumnLayout
anchors.centerIn: parent
spacing: 0
MenuButton {
id: openFileLinkButton
Layout.fillWidth: true
buttonText: qsTr("Open file link")
onClicked: {
root.showActions = false
Qt.openUrlExternally(root.imageData.file_url)
}
}
MenuButton {
id: sourceButton
visible: root.imageData.source && root.imageData.source.length > 0
Layout.fillWidth: true
buttonText: StringUtils.format(qsTr("Go to source ({0})"), StringUtils.getDomain(root.imageData.source))
enabled: root.imageData.source && root.imageData.source.length > 0
onClicked: {
root.showActions = false
Qt.openUrlExternally(root.imageData.source)
}
}
MenuButton {
id: downloadButton
Layout.fillWidth: true
buttonText: "Download"
onClicked: {
root.showActions = false
Hyprland.dispatch(`exec curl '${root.imageData.file_url}' -o '${root.imageData.is_nsfw ? root.nsfwPath : root.downloadPath}/${root.fileName}' && notify-send '${qsTr("Download complete")}' '${root.downloadPath}/${root.fileName}'`)
}
}
} }
} }
MenuButton {
id: sourceButton DropShadow {
visible: root.imageData.source && root.imageData.source.length > 0 opacity: root.showActions ? 1 : 0
Layout.fillWidth: true visible: opacity > 0
buttonText: StringUtils.format(qsTr("Go to source ({0})"), StringUtils.getDomain(root.imageData.source)) anchors.fill: contextMenu
enabled: root.imageData.source && root.imageData.source.length > 0 source: contextMenu
onClicked: { radius: Appearance.sizes.elevationMargin
root.showActions = false samples: radius * 2 + 1
Hyprland.dispatch("global quickshell:sidebarLeftClose") color: Appearance.colors.colShadow
Qt.openUrlExternally(root.imageData.source) verticalOffset: 2
} horizontalOffset: 0
}
MenuButton { Behavior on opacity {
id: downloadButton NumberAnimation {
Layout.fillWidth: true duration: Appearance.animation.elementMoveFast.duration
buttonText: "Download" easing.type: Appearance.animation.elementMoveFast.type
onClicked: { easing.bezierCurve: Appearance.animation.elementMoveFast.bezierCurve
root.showActions = false }
Hyprland.dispatch(`exec curl '${root.imageData.file_url}' -o '${root.imageData.is_nsfw ? root.nsfwPath : root.downloadPath}/${root.fileName}' && notify-send '${qsTr("Download complete")}' '${root.downloadPath}/${root.fileName}'`)
} }
} }
} }
} }
DropShadow {
opacity: root.showActions ? 1 : 0
visible: opacity > 0
anchors.fill: contextMenu
source: contextMenu
radius: Appearance.sizes.elevationMargin
samples: radius * 2 + 1
color: Appearance.colors.colShadow
verticalOffset: 2
horizontalOffset: 0
Behavior on opacity {
NumberAnimation {
duration: Appearance.animation.elementMoveFast.duration
easing.type: Appearance.animation.elementMoveFast.type
easing.bezierCurve: Appearance.animation.elementMoveFast.bezierCurve
}
}
}
} }
} }