aerothemeplasma/Plasma/Plasma_Widgets/User/io.gitgud.wackyideas.SevenStart/contents/ui/CompactRepresentation.qml

255 lines
10 KiB
QML
Raw Normal View History

2021-07-16 19:33:52 +00:00
/***************************************************************************
* Copyright (C) 2013-2014 by Eike Hein <hein@kde.org> *
* *
* This program is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. *
* *
* This program is distributed in the hope that it will be useful, *
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
* GNU General Public License for more details. *
* *
* You should have received a copy of the GNU General Public License *
* along with this program; if not, write to the *
* Free Software Foundation, Inc., *
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA . *
***************************************************************************/
import QtQuick 2.0
import QtQuick.Layouts 1.1
import org.kde.plasma.plasmoid 2.0
import org.kde.plasma.core 2.0 as PlasmaCore
import org.kde.kwindowsystem 1.0
2021-07-16 19:33:52 +00:00
Item {
id: root
readonly property var screenGeometry: plasmoid.screenGeometry
readonly property bool inPanel: (plasmoid.location == PlasmaCore.Types.TopEdge
2023-08-24 22:32:11 +00:00
|| plasmoid.location == PlasmaCore.Types.RightEdge
|| plasmoid.location == PlasmaCore.Types.BottomEdge
|| plasmoid.location == PlasmaCore.Types.LeftEdge)
2021-07-16 19:33:52 +00:00
readonly property bool vertical: (plasmoid.formFactor == PlasmaCore.Types.Vertical)
2023-08-24 22:32:11 +00:00
readonly property bool useCustomButtonImage: (plasmoid.configuration.useCustomButtonImage)
// Should the orb be rendered in its own dialog window so that it can stick out of the panel?
readonly property bool stickOutOrb: plasmoid.configuration.stickOutOrb && inPanel && !editMode
2021-07-16 19:33:52 +00:00
property QtObject dashWindow: null
property QtObject orb: null
property QtObject contextMenu: null
2023-09-27 17:50:10 +00:00
property alias orbTimer: orbTimer
2021-07-16 19:33:52 +00:00
Plasmoid.status: dashWindow && dashWindow.visible ? PlasmaCore.Types.RequiresAttentionStatus : PlasmaCore.Types.PassiveStatus
onWidthChanged: updateSizeHints()
onHeightChanged: updateSizeHints()
clip: true
2021-07-16 19:33:52 +00:00
function updateSizeHints() {
if (useCustomButtonImage) {
if (vertical) {
var scaledHeight = Math.floor(parent.width * (floatingOrbPanel.buttonIcon.implicitHeight / floatingOrbPanel.buttonIcon.implicitWidth));
2021-07-16 19:33:52 +00:00
root.Layout.minimumHeight = scaledHeight;
root.Layout.maximumHeight = scaledHeight;
root.Layout.minimumWidth = units.iconSizes.small;
root.Layout.maximumWidth = inPanel ? units.iconSizeHints.panel : -1;
} else {
var scaledWidth = Math.floor(parent.height * (floatingOrbPanel.buttonIcon.implicitWidth / floatingOrbPanel.buttonIcon.implicitHeight));
2021-07-16 19:33:52 +00:00
root.Layout.minimumWidth = scaledWidth;
root.Layout.maximumWidth = scaledWidth;
root.Layout.minimumHeight = units.iconSizes.small;
root.Layout.maximumHeight = inPanel ? units.iconSizeHints.panel : -1;
}
} else {
root.Layout.minimumWidth = units.iconSizes.small;
root.Layout.maximumWidth = inPanel ? units.iconSizeHints.panel : -1;
root.Layout.minimumHeight = units.iconSizes.small
root.Layout.maximumHeight = inPanel ? units.iconSizeHints.panel : -1;
}
if(stickOutOrb && orb) {
root.Layout.minimumWidth = orb.width + panelSvg.margins.right*(compositing ? 0 : 1);
root.Layout.maximumWidth = orb.width + panelSvg.margins.right*(compositing ? 0 : 1);
root.Layout.minimumHeight = orb.height;
root.Layout.maximumHeight = orb.height;
}
2023-09-27 17:50:10 +00:00
// This has to be done, or else the orb won't be positioned correctly. ??????????
orb.y += 5;
orb.y -= 5; // ??????????????????????????????????????
2021-07-16 19:33:52 +00:00
}
onStickOutOrbChanged: {
updateSizeHints();
positionOrb();
}
2023-09-27 17:50:10 +00:00
/*
* The following code gets the ContainmentInterface instance which is used for keeping track of edit mode's state.
* This allows us to hide the StartOrb object so the user can actually highlight and select this plasmoid during edit mode.
*/
property var containmentInterface: null
readonly property bool editMode: containmentInterface ? containmentInterface.editMode : false
onParentChanged: {
if (parent) {
for (var obj = root, depth = 0; !!obj; obj = obj.parent, depth++) {
if (obj.toString().startsWith('ContainmentInterface')) {
// desktop containment / plasmoidviewer
// Note: This doesn't always work. FolderViewDropArea may not yet have
// ContainmentInterface as a parent when this loop runs.
if (typeof obj['editMode'] === 'boolean') {
root.containmentInterface = obj
break
}
} else if (obj.toString().startsWith('DeclarativeDropArea')) {
// panel containment
if (typeof obj['Plasmoid'] !== 'undefined' && obj['Plasmoid'].toString().startsWith('ContainmentInterface')) {
if (typeof obj['Plasmoid']['editMode'] === 'boolean') {
root.containmentInterface = obj.Plasmoid
break
}
}
}
}
}
}
2021-07-16 19:33:52 +00:00
Connections {
target: units.iconSizeHints
2023-08-24 22:32:11 +00:00
function onPanelChanged() { updateSizeHints(); }
2021-07-16 19:33:52 +00:00
}
2024-01-20 02:08:06 +00:00
Connections {
target: plasmoid
function onScreenGeometryChanged() {
positionOrb();
}
function onScreenChanged() {
positionOrb();
}
}
property bool compositing: kwindowsystem.compositingActive
2021-07-16 19:33:52 +00:00
onCompositingChanged: {
updateSizeHints();
positionOrb();
2023-09-27 17:50:10 +00:00
if(compositing) {
orb.x += panelSvg.margins.left;
2023-09-27 17:50:10 +00:00
}
compositingFix.start();
}
function positionOrb() {
var pos = plasmoid.mapToGlobal(kicker.x, kicker.y); // Gets the global position of this plasmoid, in screen coordinates.
orb.width = floatingOrbPanel.buttonIcon.implicitWidth + panelSvg.margins.left;
orb.height = floatingOrbPanel.buttonIcon.implicitHeight;
orb.x = pos.x;
orb.y = pos.y + panelSvg.margins.bottom;
// Keep the orb positioned exactly on the bottom if it is rendered out of bounds (beyond the screen geometry)
if(orb.y + orb.height > plasmoid.screenGeometry.height) {
orb.y -= orb.y + orb.height - plasmoid.screenGeometry.height;
}
}
// If the url is empty (default value), then use the fallback url. Otherwise, return the url path relative to
// the location of the source code.
2023-08-24 22:32:11 +00:00
function getResolvedUrl(url, fallback) {
if(url.toString() === "") {
return Qt.resolvedUrl(fallback);
}
return url;
}
2023-09-27 17:50:10 +00:00
function showMenu() {
dashWindow.visible = !dashWindow.visible;
dashWindow.showingAllPrograms = false;
plasmoid.nativeInterface.setActiveWin(dashWindow);
dashWindow.m_searchField.focus = true;
orb.raise();
}
property bool menuShown: dashWindow.visible
2023-08-24 22:32:11 +00:00
property int opacityDuration: 250
function createContextMenu(pos) {
contextMenu = Qt.createQmlObject("ContextMenu {}", root);
contextMenu.fillActions();
contextMenu.show();
}
2023-08-24 22:32:11 +00:00
/*
* Three IconItems are used in order to achieve the same look and feel as Windows 7's
* orbs. When the menu is closed, hovering over the orb results in the hovered icon
* gradually appearing into view, and clicking on the orb causes an instant change in
* visibility, where the normal and hovered icons are invisible, and the pressed icon
* is visible.
*
* When they're bounded by the panel, these icons will by default try to fill up as
* much space as they can in the compact representation while preserving their aspect
* ratio.
2023-08-24 22:32:11 +00:00
*/
2021-07-16 19:33:52 +00:00
FloatingOrb {
id: floatingOrbPanel
visible: (!stickOutOrb)
2023-08-24 22:32:11 +00:00
anchors.fill: parent
objectName: "innerorb"
2021-07-16 19:33:52 +00:00
}
2023-09-27 17:50:10 +00:00
// Handles all mouse events for the popup orb
2021-07-16 19:33:52 +00:00
MouseArea
{
id: mouseAreaCompositingOff
2021-07-16 19:33:52 +00:00
anchors.fill: parent
hoverEnabled: true
visible: stickOutOrb
2023-09-27 17:50:10 +00:00
acceptedButtons: Qt.LeftButton
2021-07-16 19:33:52 +00:00
onClicked: {
2023-09-27 17:50:10 +00:00
showMenu();
}
}
2023-09-27 17:50:10 +00:00
// I hate this
// So the only way I could reasonably think of to make this work is running the function
// with a delay.
Timer {
id: compositingFix
interval: 150
onTriggered: {
if(!compositing) {
plasmoid.nativeInterface.setTransparentWindow();
}
2021-07-16 19:33:52 +00:00
}
}
2023-09-27 17:50:10 +00:00
// Even worse, this just makes things even more unsophisticated. If someone has a better
// way of solving this, I would love to know.
Timer {
id: orbTimer
interval: 15
onTriggered: {
plasmoid.nativeInterface.setOrb(orb);
// Currently hardcoded, will make it configurable soon, when it's been properly tested and hopefully slightly refactored.
plasmoid.nativeInterface.setMask(Qt.resolvedUrl("./orbs/mask.png"), false);
plasmoid.nativeInterface.setWinState(orb);
plasmoid.nativeInterface.setWinType(orb);
plasmoid.nativeInterface.setDashWindow(dashWindow);
updateSizeHints();
positionOrb();
}
}
2021-07-16 19:33:52 +00:00
Component.onCompleted: {
dashWindow = Qt.createQmlObject("MenuRepresentation {}", root);
orb = Qt.createQmlObject("StartOrb {}", root);
2021-07-16 19:33:52 +00:00
plasmoid.activated.connect(function() {
2023-09-27 17:50:10 +00:00
showMenu();
2021-07-16 19:33:52 +00:00
});
2023-09-27 17:50:10 +00:00
2021-07-16 19:33:52 +00:00
}
}