2019-10-25 14:44:55 +03:00
|
|
|
/****************************************************************************
|
|
|
|
**
|
|
|
|
** Copyright (C) 2019 The Qt Company Ltd.
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
|
|
|
**
|
|
|
|
** This file is part of Qt Creator.
|
|
|
|
**
|
|
|
|
** Commercial License Usage
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
|
|
**
|
|
|
|
** GNU General Public License Usage
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
|
|
|
**
|
|
|
|
****************************************************************************/
|
|
|
|
|
2021-06-28 14:04:43 +03:00
|
|
|
import QtQuick 6.0
|
|
|
|
import QtQuick3D 6.0
|
2019-10-25 14:44:55 +03:00
|
|
|
|
2020-02-25 14:37:45 +02:00
|
|
|
Item {
|
2019-10-25 14:44:55 +03:00
|
|
|
id: iconGizmo
|
|
|
|
|
2020-01-28 10:40:44 +02:00
|
|
|
property Node activeScene: null
|
|
|
|
property Node scene: null
|
2019-10-25 14:44:55 +03:00
|
|
|
property View3D view3D
|
|
|
|
property bool highlightOnHover: true
|
|
|
|
property Node targetNode: null
|
2019-12-10 13:14:30 +02:00
|
|
|
property var selectedNodes: []
|
2019-11-26 17:56:00 +02:00
|
|
|
readonly property bool selected: {
|
|
|
|
for (var i = 0; i < selectedNodes.length; ++i) {
|
|
|
|
if (selectedNodes[i] === targetNode)
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2020-06-01 15:38:19 +03:00
|
|
|
property bool hasMouse: false
|
2020-10-09 11:14:14 +02:00
|
|
|
property bool hidden: false
|
|
|
|
property bool locked: false
|
2022-01-21 17:15:01 +02:00
|
|
|
property bool globalShow: true
|
|
|
|
property bool canBeVisible: activeScene === scene && !hidden && (targetNode ? targetNode.visible : false)
|
2019-10-25 14:44:55 +03:00
|
|
|
|
|
|
|
property alias iconSource: iconImage.source
|
|
|
|
|
2019-11-26 17:56:00 +02:00
|
|
|
signal clicked(Node node, bool multi)
|
2019-10-25 14:44:55 +03:00
|
|
|
|
2020-06-01 15:38:19 +03:00
|
|
|
onSelectedChanged: {
|
|
|
|
if (selected)
|
|
|
|
hasMouse = false;
|
|
|
|
}
|
|
|
|
|
2022-01-21 17:15:01 +02:00
|
|
|
visible: canBeVisible && globalShow
|
2019-10-25 14:44:55 +03:00
|
|
|
|
|
|
|
Overlay2D {
|
2019-11-18 15:02:07 +02:00
|
|
|
id: iconOverlay
|
2020-02-25 14:37:45 +02:00
|
|
|
targetNode: iconGizmo.targetNode
|
2019-10-25 14:44:55 +03:00
|
|
|
targetView: view3D
|
|
|
|
visible: iconGizmo.visible && !isBehindCamera
|
|
|
|
|
|
|
|
Rectangle {
|
2019-11-18 15:02:07 +02:00
|
|
|
id: iconRect
|
2020-06-01 15:38:19 +03:00
|
|
|
|
2019-11-13 13:42:43 +02:00
|
|
|
width: iconImage.width
|
|
|
|
height: iconImage.height
|
2019-10-25 14:44:55 +03:00
|
|
|
x: -width / 2
|
2019-11-18 13:10:54 +02:00
|
|
|
y: -height / 2
|
2019-10-25 14:44:55 +03:00
|
|
|
color: "transparent"
|
|
|
|
border.color: "#7777ff"
|
2020-10-09 11:14:14 +02:00
|
|
|
border.width: !iconGizmo.locked && iconGizmo.highlightOnHover && iconGizmo.hasMouse ? 2 : 0
|
2019-10-25 14:44:55 +03:00
|
|
|
radius: 5
|
2019-11-26 17:56:00 +02:00
|
|
|
opacity: iconGizmo.selected ? 0.2 : 1
|
2019-10-25 14:44:55 +03:00
|
|
|
Image {
|
|
|
|
id: iconImage
|
2019-11-13 13:42:43 +02:00
|
|
|
fillMode: Image.Pad
|
2019-10-25 14:44:55 +03:00
|
|
|
MouseArea {
|
|
|
|
id: iconMouseArea
|
|
|
|
anchors.fill: parent
|
2021-06-08 16:13:18 +03:00
|
|
|
onPressed: (mouse)=> {
|
2020-01-09 17:32:41 +02:00
|
|
|
// Ignore singleselection mouse presses when we have single object selected
|
|
|
|
// so that the icon gizmo doesn't hijack mouse clicks meant for other gizmos
|
|
|
|
if (iconGizmo.selected && !(mouse.modifiers & Qt.ControlModifier)
|
|
|
|
&& selectedNodes.length === 1) {
|
2019-11-26 17:56:00 +02:00
|
|
|
mouse.accepted = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-08 16:13:18 +03:00
|
|
|
onClicked: (mouse)=> {
|
|
|
|
iconGizmo.clicked(iconGizmo.targetNode,
|
|
|
|
mouse.modifiers & Qt.ControlModifier);
|
|
|
|
}
|
2019-11-26 17:56:00 +02:00
|
|
|
hoverEnabled: iconGizmo.highlightOnHover && !iconGizmo.selected
|
|
|
|
acceptedButtons: Qt.LeftButton
|
2020-06-01 15:38:19 +03:00
|
|
|
|
|
|
|
// onPositionChanged, onContainsMouseAreaChanged, and hasMouse are used instead
|
|
|
|
// of just using containsMouse directly, because containsMouse
|
|
|
|
// cannot be relied upon to update correctly in some situations.
|
|
|
|
// This is likely because the overlapping 3D mouse areas of the gizmos get
|
|
|
|
// the mouse events instead of this area, so mouse leaving the area
|
|
|
|
// doesn't always update containsMouse property.
|
|
|
|
onPositionChanged: {
|
|
|
|
if (!iconGizmo.selected)
|
|
|
|
iconGizmo.hasMouse = containsMouse;
|
|
|
|
}
|
|
|
|
|
|
|
|
onContainsMouseChanged: {
|
|
|
|
if (!iconGizmo.selected)
|
|
|
|
iconGizmo.hasMouse = containsMouse;
|
|
|
|
else
|
|
|
|
iconGizmo.hasMouse = false;
|
|
|
|
}
|
2019-10-25 14:44:55 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|