mirror of
https://github.com/HifiExperiments/overte.git
synced 2025-05-07 10:08:35 +02:00
234 lines
6.9 KiB
QML
234 lines
6.9 KiB
QML
import QtQuick 2.5
|
|
import QtQuick.Controls 1.4
|
|
import QtQuick.Dialogs 1.2 as OriginalDialogs;
|
|
|
|
import "../dialogs"
|
|
import "../menus"
|
|
|
|
// This is our primary 'desktop' object to which all VR dialogs and
|
|
// windows will be childed.
|
|
FocusScope {
|
|
id: desktop
|
|
anchors.fill: parent;
|
|
objectName: "desktop"
|
|
|
|
// Allows QML/JS to find the desktop through the parent chain
|
|
property bool desktopRoot: true
|
|
|
|
// The VR version of the primary menu
|
|
property var rootMenu: Menu {
|
|
id: rootMenu; objectName: "rootMenu"
|
|
Component.onCompleted: {
|
|
console.log("ROOT_MENU " + rootMenu);
|
|
}
|
|
}
|
|
|
|
QtObject {
|
|
id: d
|
|
readonly property int zBasisNormal: 0
|
|
readonly property int zBasisAlwaysOnTop: 4096
|
|
readonly property int zBasisModal: 8192
|
|
readonly property var messageDialogBuilder: Component { MessageDialog { } }
|
|
readonly property var nativeMessageDialogBuilder: Component { OriginalDialogs.MessageDialog { } }
|
|
|
|
function findChild(item, name) {
|
|
for (var i = 0; i < item.children.length; ++i) {
|
|
if (item.children[i].objectName === name) {
|
|
return item.children[i];
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
function findParentMatching(item, predicate) {
|
|
while (item) {
|
|
if (predicate(item)) {
|
|
break;
|
|
}
|
|
item = item.parent;
|
|
}
|
|
return item;
|
|
}
|
|
|
|
function isDesktop(item) {
|
|
return item.desktopRoot;
|
|
}
|
|
|
|
function isTopLevelWindow(item) {
|
|
return item.topLevelWindow;
|
|
}
|
|
|
|
function isAlwaysOnTopWindow(window) {
|
|
return window.alwaysOnTop;
|
|
}
|
|
|
|
function isModalWindow(window) {
|
|
return window.modality !== Qt.NonModal;
|
|
}
|
|
|
|
function getTopLevelWindows(predicate) {
|
|
var currentWindows = [];
|
|
if (!desktop) {
|
|
console.log("Could not find desktop for " + item)
|
|
return currentWindows;
|
|
}
|
|
|
|
for (var i = 0; i < desktop.children.length; ++i) {
|
|
var child = desktop.children[i];
|
|
if (isTopLevelWindow(child) && (!predicate || predicate(child))) {
|
|
currentWindows.push(child)
|
|
}
|
|
}
|
|
return currentWindows;
|
|
}
|
|
|
|
|
|
function getDesktopWindow(item) {
|
|
return findParentMatching(item, isTopLevelWindow)
|
|
}
|
|
|
|
function fixupZOrder(windows, basis, topWindow) {
|
|
windows.sort(function(a, b){
|
|
return a.z - b.z;
|
|
});
|
|
|
|
if ((topWindow.z >= basis) && (windows[windows.length - 1] === topWindow)) {
|
|
return;
|
|
}
|
|
|
|
var lastZ = -1;
|
|
var lastTargetZ = basis - 1;
|
|
for (var i = 0; i < windows.length; ++i) {
|
|
var window = windows[i];
|
|
if (!window.visible) {
|
|
continue
|
|
}
|
|
|
|
if (topWindow && (topWindow === window)) {
|
|
continue
|
|
}
|
|
|
|
if (window.z > lastZ) {
|
|
lastZ = window.z;
|
|
++lastTargetZ;
|
|
}
|
|
if (DebugQML) {
|
|
console.log("Assigning z order " + lastTargetZ + " to " + window)
|
|
}
|
|
|
|
window.z = lastTargetZ;
|
|
}
|
|
if (topWindow) {
|
|
++lastTargetZ;
|
|
if (DebugQML) {
|
|
console.log("Assigning z order " + lastTargetZ + " to " + topWindow)
|
|
}
|
|
topWindow.z = lastTargetZ;
|
|
}
|
|
|
|
return lastTargetZ;
|
|
}
|
|
|
|
function raiseWindow(item) {
|
|
var targetWindow = getDesktopWindow(item);
|
|
if (!targetWindow) {
|
|
console.warn("Could not find top level window for " + item);
|
|
return;
|
|
}
|
|
|
|
if (!desktop) {
|
|
console.warn("Could not find desktop for window " + targetWindow);
|
|
return;
|
|
}
|
|
|
|
var predicate;
|
|
var zBasis;
|
|
if (isModalWindow(targetWindow)) {
|
|
predicate = isModalWindow;
|
|
zBasis = zBasisModal
|
|
} else if (isAlwaysOnTopWindow(targetWindow)) {
|
|
predicate = function(window) {
|
|
return (isAlwaysOnTopWindow(window) && !isModalWindow(window));
|
|
}
|
|
zBasis = zBasisAlwaysOnTop
|
|
} else {
|
|
predicate = function(window) {
|
|
return (!isAlwaysOnTopWindow(window) && !isModalWindow(window));
|
|
}
|
|
zBasis = zBasisNormal
|
|
}
|
|
|
|
var windows = getTopLevelWindows(predicate);
|
|
fixupZOrder(windows, zBasis, targetWindow);
|
|
}
|
|
}
|
|
|
|
MenuMouseHandler { id: menuPopperUpper }
|
|
|
|
function raise(item) {
|
|
d.raiseWindow(item);
|
|
}
|
|
|
|
function messageBox(properties) {
|
|
// Debugging: native message dialog for comparison
|
|
// d.nativeMessageDialogBuilder.createObject(desktop, properties);
|
|
return d.messageDialogBuilder.createObject(desktop, properties);
|
|
}
|
|
|
|
function popupMenu(point) {
|
|
menuPopperUpper.popup(desktop, rootMenu.items, point);
|
|
}
|
|
|
|
function toggleMenu(point) {
|
|
menuPopperUpper.toggle(desktop, rootMenu.items, point);
|
|
}
|
|
|
|
Keys.onEscapePressed: {
|
|
if (menuPopperUpper.closeLastMenu()) {
|
|
event.accepted = true;
|
|
return;
|
|
}
|
|
event.accepted = false;
|
|
}
|
|
|
|
Keys.onLeftPressed: {
|
|
if (menuPopperUpper.closeLastMenu()) {
|
|
event.accepted = true;
|
|
return;
|
|
}
|
|
event.accepted = false;
|
|
}
|
|
|
|
|
|
function unfocusWindows() {
|
|
var windows = d.getTopLevelWindows();
|
|
for (var i = 0; i < windows.length; ++i) {
|
|
windows[i].focus = false;
|
|
}
|
|
desktop.focus = true;
|
|
}
|
|
|
|
// Debugging help for figuring out focus issues
|
|
property var offscreenWindow;
|
|
onOffscreenWindowChanged: offscreenWindow.activeFocusItemChanged.connect(onWindowFocusChanged);
|
|
function onWindowFocusChanged() {
|
|
console.log("Focus item is " + offscreenWindow.activeFocusItem);
|
|
var focusedItem = offscreenWindow.activeFocusItem ;
|
|
if (DebugQML && focusedItem) {
|
|
var rect = desktop.mapToItem(null, focusedItem.x, focusedItem.y, focusedItem.width, focusedItem.height);
|
|
focusDebugger.visible = true
|
|
focusDebugger.x = rect.x;
|
|
focusDebugger.y = rect.y;
|
|
focusDebugger.width = rect.width
|
|
focusDebugger.height = rect.height
|
|
}
|
|
}
|
|
Rectangle {
|
|
id: focusDebugger;
|
|
z: 9999; visible: false; color: "red"
|
|
ColorAnimation on color { from: "#7fffff00"; to: "#7f0000ff"; duration: 1000; loops: 9999 }
|
|
}
|
|
}
|
|
|
|
|
|
|