175 lines
3.8 KiB
QML
175 lines
3.8 KiB
QML
import QtQuick
|
|
import qs.Services
|
|
|
|
Window {
|
|
id: root
|
|
|
|
readonly property real scaling: Scaling.scale(screen)
|
|
property bool isVisible: false
|
|
property string text: "Placeholder"
|
|
property Item target: null
|
|
property int delay: Style.tooltipDelay
|
|
property bool positionAbove: false
|
|
|
|
flags: Qt.ToolTip | Qt.FramelessWindowHint | Qt.WindowStaysOnTopHint
|
|
color: "transparent"
|
|
visible: false
|
|
|
|
onIsVisibleChanged: {
|
|
if (isVisible) {
|
|
if (delay > 0) {
|
|
timerShow.running = true
|
|
} else {
|
|
_showNow()
|
|
}
|
|
} else {
|
|
_hideNow()
|
|
}
|
|
}
|
|
|
|
function show() {
|
|
isVisible = true
|
|
}
|
|
function hide() {
|
|
isVisible = false
|
|
timerShow.running = false
|
|
}
|
|
|
|
function _showNow() {
|
|
// Compute new size everytime we show the tooltip
|
|
width = Math.max(50 * scaling, tooltipText.implicitWidth + Style.marginLarge * 2 * scaling)
|
|
height = Math.max(50 * scaling, tooltipText.implicitHeight + Style.marginMedium * 2 * scaling)
|
|
|
|
if (!target) {
|
|
return
|
|
}
|
|
|
|
if (positionAbove) {
|
|
// Position tooltip above the target
|
|
var pos = target.mapToGlobal(0, 0)
|
|
x = pos.x - width / 2 + target.width / 2
|
|
y = pos.y - height - 12 // 12 px margin above
|
|
} else {
|
|
// Position tooltip below the target
|
|
var pos = target.mapToGlobal(0, target.height)
|
|
x = pos.x - width / 2 + target.width / 2
|
|
y = pos.y + 12 // 12 px margin below
|
|
}
|
|
|
|
// Start with animation values
|
|
tooltipRect.scaleValue = 0.8
|
|
tooltipRect.opacityValue = 0.0
|
|
visible = true
|
|
|
|
// Use a timer to trigger the animation after the component is visible
|
|
showTimer.start()
|
|
}
|
|
|
|
function _hideNow() {
|
|
// Start hide animation
|
|
tooltipRect.scaleValue = 0.8
|
|
tooltipRect.opacityValue = 0.0
|
|
|
|
// Hide after animation completes
|
|
hideTimer.start()
|
|
}
|
|
|
|
Connections {
|
|
target: root.target
|
|
function onXChanged() {
|
|
if (root.visible) {
|
|
root._showNow()
|
|
}
|
|
}
|
|
function onYChanged() {
|
|
if (root.visible) {
|
|
root._showNow()
|
|
}
|
|
}
|
|
function onWidthChanged() {
|
|
if (root.visible) {
|
|
root._showNow()
|
|
}
|
|
}
|
|
function onHeightChanged() {
|
|
if (root.visible) {
|
|
root._showNow()
|
|
}
|
|
}
|
|
}
|
|
|
|
Timer {
|
|
id: timerShow
|
|
interval: delay
|
|
running: false
|
|
repeat: false
|
|
onTriggered: {
|
|
_showNow()
|
|
running = false
|
|
}
|
|
}
|
|
|
|
// Timer to hide tooltip after animation
|
|
Timer {
|
|
id: hideTimer
|
|
interval: Style.animationNormal
|
|
repeat: false
|
|
onTriggered: {
|
|
visible = false
|
|
}
|
|
}
|
|
|
|
// Timer to trigger show animation
|
|
Timer {
|
|
id: showTimer
|
|
interval: 10 // Very short delay to ensure component is visible
|
|
repeat: false
|
|
onTriggered: {
|
|
// Animate to final values
|
|
tooltipRect.scaleValue = 1.0
|
|
tooltipRect.opacityValue = 1.0
|
|
}
|
|
}
|
|
|
|
Rectangle {
|
|
id: tooltipRect
|
|
anchors.fill: parent
|
|
radius: Style.radiusMedium * scaling
|
|
color: Colors.colorSurface
|
|
border.color: Colors.colorOutline
|
|
border.width: Math.max(1, Style.borderThin * scaling)
|
|
z: 1
|
|
|
|
// Animation properties
|
|
property real scaleValue: 1.0
|
|
property real opacityValue: 1.0
|
|
|
|
scale: scaleValue
|
|
opacity: opacityValue
|
|
|
|
// Animation behaviors
|
|
Behavior on scale {
|
|
NumberAnimation {
|
|
duration: Style.animationNormal
|
|
easing.type: Easing.OutExpo
|
|
}
|
|
}
|
|
|
|
Behavior on opacity {
|
|
NumberAnimation {
|
|
duration: Style.animationNormal
|
|
easing.type: Easing.OutQuad
|
|
}
|
|
}
|
|
|
|
NText {
|
|
id: tooltipText
|
|
anchors.centerIn: parent
|
|
text: root.text
|
|
font.pointSize: Style.fontSizeMedium * scaling
|
|
horizontalAlignment: Text.AlignHCenter
|
|
verticalAlignment: Text.AlignVCenter
|
|
wrapMode: Text.Wrap
|
|
}
|
|
}
|
|
}
|