You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
latte-dock/containment/package/contents/ui/applet/ItemWrapper.qml

699 lines
27 KiB
QML

/*
* Copyright 2016 Smith AR <audoban@openmailbox.org>
* Michail Vourlakos <mvourlakos@gmail.com>
*
* This file is part of Latte-Dock
*
* Latte-Dock 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.
*
* Latte-Dock 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, see <http://www.gnu.org/licenses/>.
*/
import QtQuick 2.1
import QtQuick.Layouts 1.1
import QtGraphicalEffects 1.0
import org.kde.plasma.plasmoid 2.0
import org.kde.plasma.core 2.0 as PlasmaCore
import org.kde.plasma.components 2.0 as PlasmaComponents
import org.kde.latte.core 0.2 as LatteCore
import "../../code/MathTools.js" as MathTools
Item{
id: wrapper
width: root.isHorizontal ? length : thickness
height: root.isHorizontal ? thickness : length
readonly property int length: {
if (appletItem.isInternalViewSplitter && !root.inConfigureAppletsMode) {
return 0;
}
if (isSeparator && root.parabolicEffectEnabled) {
return -1;
}
if (appletItem.isAutoFillApplet) {
return appletItem.layouter.maxMetricsInHigherPriority ?
appletItem.maxAutoFillLength : Math.max(appletItem.minAutoFillLength, appletPreferredLength);
}
if (appletItem.latteApplet) {
return root.isHorizontal ? latteApplet.tasksWidth : latteApplet.tasksHeight;
}
return root.inConfigureAppletsMode ? Math.max(Math.min(appletItem.metrics.iconSize, root.minAppletLengthInConfigure), scaledLength) : scaledLength;
}
readonly property int thickness: {
if (appletItem.isInternalViewSplitter && !root.inConfigureAppletsMode) {
return 0;
}
if (appletItem.latteApplet) {
return root.isHorizontal ? latteApplet.tasksHeight : latteApplet.tasksWidth;
}
return scaledThickness + appletItem.metrics.margin.screenEdge
}
opacity: appletColorizer.mustBeShown && graphicsSystem.isAccelerated ? 0 : 1
property bool disableLengthScale: false
property bool disableThicknessScale: false
property bool editMode: root.inConfigureAppletsMode
property bool edgeLengthMarginsDisabled: isSeparator || !communicator.requires.lengthMarginsEnabled || !canBeHovered
property int appletWidth: applet ? applet.width : -1
property int appletHeight: applet ? applet.height : -1
property int appletMinimumWidth: applet && applet.Layout ? applet.Layout.minimumWidth : -1
property int appletMinimumHeight: applet && applet.Layout ? applet.Layout.minimumHeight : -1
property int appletPreferredWidth: applet && applet.Layout ? applet.Layout.preferredWidth : -1
property int appletPreferredHeight: applet && applet.Layout ? applet.Layout.preferredHeight : -1
property int appletMaximumWidth: applet && applet.Layout ? applet.Layout.maximumWidth : -1
property int appletMaximumHeight: applet && applet.Layout ? applet.Layout.maximumHeight : -1
readonly property int appletLength: root.isHorizontal ? appletWidth : appletHeight
readonly property int appletThickness: root.isHorizontal ? appletHeight : appletWidth
readonly property int appletMinimumLength : root.isHorizontal ? appletMinimumWidth : appletMinimumHeight
readonly property int appletMinimumThickness: root.isHorizontal ? appletMinimumHeight : appletMinimumWidth
readonly property int appletPreferredLength: root.isHorizontal ? appletPreferredWidth : appletPreferredHeight
readonly property int appletPreferredThickness: root.isHorizontal ? appletPreferredHeight : appletPreferredWidth
readonly property int appletMaximumLength: root.isHorizontal ? appletMaximumWidth : appletMaximumHeight
readonly property int appletMaximumThickness: root.isHorizontal ? appletMaximumHeight : appletMaximumWidth
property int iconSize: appletItem.metrics.iconSize
property int marginsThickness: appletItem.metrics.totals.thicknessEdges
property int marginsLength: root.inFullJustify && atScreenEdge && !parabolicEffectMarginsEnabled ? edgeLengthMargins : localLengthMargins //Fitt's Law
property int localLengthMargins: isSeparator || !communicator.requires.lengthMarginsEnabled || isInternalViewSplitter ? 0 : appletItem.lengthAppletFullMargins
property int edgeLengthMargins: edgeLengthMarginsDisabled ? 0 : appletItem.lengthAppletPadding * 2
property real scaledLength: zoomScaleLength * (layoutLength + marginsLength)
property real scaledThickness: zoomScaleThickness * (layoutThickness + marginsThickness)
property real zoomScaleLength: disableLengthScale ? 1 : zoomScale
property real zoomScaleThickness: disableThicknessScale ? 1 : zoomScale
property int layoutLength: 0
property int layoutThickness: 0
property real center:root.isHorizontal ?
(width + hiddenSpacerLeft.separatorSpace + hiddenSpacerRight.separatorSpace) / 2 :
(height + hiddenSpacerLeft.separatorSpace + hiddenSpacerRight.separatorSpace) / 2
property real zoomScale: 1
property int index: appletItem.index
property Item wrapperContainer: _wrapperContainer
property Item clickedEffect: _clickedEffect
property Item containerForOverlayIcon: _containerForOverlayIcon
property Item overlayIconLoader: _overlayIconLoader
Behavior on opacity {
NumberAnimation {
duration: 0.8 * appletItem.animations.duration.proposed
easing.type: Easing.OutCubic
}
}
// property int pHeight: applet ? applet.Layout.preferredHeight : -10
/*function debugLayouts(){
if(applet){
console.log("---------- "+ applet.pluginName +" ----------");
console.log("MinW "+applet.Layout.minimumWidth);
console.log("PW "+applet.Layout.preferredWidth);
console.log("MaxW "+applet.Layout.maximumWidth);
console.log("FillW "+applet.Layout.fillWidth);
console.log("-----");
console.log("MinH "+applet.Layout.minimumHeight);
console.log("PH "+applet.Layout.preferredHeight);
console.log("MaxH "+applet.Layout.maximumHeight);
console.log("FillH "+applet.Layout.fillHeight);
console.log("-----");
console.log("Real Applet Width: "+applet.width);
console.log("Real Applet Height: "+applet.height);
console.log("-----");
console.log("Real Wrapper Width: "+wrapper.width);
console.log("Real Wrapper Height: "+wrapper.height);
console.log("-----");
console.log("Can be hovered: " + canBeHovered);
console.log("Icon size: " + appletItem.metrics.iconSize);
console.log("Thick Margins: " + appletItem.metrics.totals.thicknessEdges);
console.log("Intern. Margins: " + (appletItem.metrics.padding.length * 2));
console.log("Intern. Margins: " + (appletItem.metrics.margin.length * 2));
console.log("Max hovered criteria: " + (appletItem.metrics.iconSize + metrics.totals.thicknessEdges));
console.log("-----");
console.log("LayoutW: " + layoutWidth);
console.log("LayoutH: " + layoutHeight);
}
}
onLayoutWidthChanged: {
debugLayouts();
}
onLayoutHeightChanged: {
debugLayouts();
}*/
onAppletLengthChanged: {
if(zoomScale === 1) {
checkCanBeHovered();
}
}
onAppletThicknessChanged: {
if(zoomScale === 1) {
checkCanBeHovered();
}
}
onAppletMinimumLengthChanged: {
if(zoomScale === 1) {
checkCanBeHovered();
}
updateAutoFillLength();
}
onAppletMinimumThicknessChanged: {
if(zoomScale === 1) {
checkCanBeHovered();
}
}
onAppletPreferredLengthChanged: updateAutoFillLength();
onAppletMaximumLengthChanged: updateAutoFillLength();
onZoomScaleChanged: {
if ((zoomScale === appletItem.parabolic.factor.zoom) && !appletItem.parabolic.directRenderingEnabled) {
appletItem.parabolic.setDirectRenderingEnabled(true);
}
if ((zoomScale > 1) && !appletItem.isZoomed) {
appletItem.isZoomed = true;
appletItem.animations.needBothAxis.addEvent(appletItem);
} else if (zoomScale == 1) {
appletItem.isZoomed = false;
appletItem.animations.needBothAxis.removeEvent(appletItem);
}
}
Binding {
target: wrapper
property: "layoutThickness"
when: !isLattePlasmoid && (wrapper.zoomScale === 1)
value: {
if (appletItem.isInternalViewSplitter){
return !root.inConfigureAppletsMode ? 0 : appletItem.metrics.iconSize;
}
return appletItem.metrics.iconSize;
}
}
Binding {
target: wrapper
property: "layoutLength"
when: !isLattePlasmoid && !appletItem.isAutoFillApplet && (wrapper.zoomScale === 1)
value: {
if (appletItem.isInternalViewSplitter){
return !root.inConfigureAppletsMode ? 0 : Math.min(appletItem.metrics.iconSize, root.maxJustifySplitterSize);
} else if (applet && ( appletMaximumLength < appletItem.metrics.iconSize
|| appletPreferredLength > appletItem.metrics.iconSize
|| appletItem.originalAppletBehavior)
&& !communicator.overlayLatteIconIsActive) {
//this way improves performance, probably because during animation the preferred sizes update a lot
if (appletMaximumLength>0 && appletMaximumLength < appletItem.metrics.iconSize){
return appletMaximumLength;
} else if (appletMinimumLength > appletItem.metrics.iconSize){
return appletMinimumLength;
} else if ((appletPreferredLength > appletItem.metrics.iconSize)
|| (appletItem.originalAppletBehavior && appletPreferredLength > 0 )){
return appletPreferredLength;
}
}
return appletItem.metrics.iconSize;
}
}
Binding {
target: wrapper
property: "disableLengthScale"
when: !(appletItem.isAutoFillApplet || appletItem.isLattePlasmoid)
value: {
var blockParabolicEffectInLength = false;
if (appletItem.isInternalViewSplitter){
return false;
} else {
if(applet && (appletMinimumLength > appletItem.metrics.iconSize) && !canBeHovered && !communicator.overlayLatteIconIsActive){
return (wrapper.zoomScale === 1);
} //it is used for plasmoids that need to scale only one axis... e.g. the Weather Plasmoid
else if(applet
&& ( appletMaximumLength < appletItem.metrics.iconSize
|| appletPreferredLength > appletItem.metrics.iconSize
|| appletItem.originalAppletBehavior)
&& !communicator.overlayLatteIconIsActive) {
//this way improves performance, probably because during animation the preferred sizes update a lot
if (appletMaximumLength>0 && appletMaximumLength < appletItem.metrics.iconSize){
return false;
} else if (appletMinimumLength > appletItem.metrics.iconSize){
return (wrapper.zoomScale === 1);
} else if ((appletPreferredLength > appletItem.metrics.iconSize)
|| (appletItem.originalAppletBehavior && appletPreferredLength > 0 )){
return (wrapper.zoomScale === 1);
}
}
}
return false;
}
}
function updateAutoFillLength() {
if (appletItem.isAutoFillApplet) {
appletItem.layouter.updateSizeForAppletsInFill();
}
}
Item{
id:_wrapperContainer
width: root.isHorizontal ? _length : _thickness
height: root.isHorizontal ? _thickness : _length
opacity: appletShadow.active ? 0 : 1
property int _length: {
if (appletItem.isAutoFillApplet && (appletItem.maxAutoFillLength>-1)){
return wrapper.length;
}
if (appletItem.isInternalViewSplitter) {
return wrapper.layoutLength;
}
return wrapper.zoomScaleLength * wrapper.layoutLength;
}
property int _thickness: {
if (appletItem.isInternalViewSplitter) {
return wrapper.layoutThickness;
}
var wrapperContainerThickness = wrapper.zoomScaleThickness * (appletItem.metrics.totals.thickness);
return appletItem.screenEdgeMarginSupported ? wrapperContainerThickness + appletItem.metrics.margin.screenEdge : wrapperContainerThickness;
}
readonly property int appliedEdgeMargin: {
if (appletItem.isInternalViewSplitter) {
return appletItem.metrics.margin.screenEdge + appletItem.metrics.margin.thickness;
}
return appletItem.screenEdgeMarginSupported ? 0 : appletItem.metrics.margin.screenEdge;
}
///Secret MouseArea to be used by the folder widget
Loader{
anchors.fill: parent
active: communicator.overlayLatteIconIsActive
&& applet.pluginName === "org.kde.plasma.folder"
&& !appletItem.acceptMouseEvents
sourceComponent: MouseArea{
onClicked: latteView.extendedInterface.toggleAppletExpanded(applet.id);
}
}
Item{
id: _containerForOverlayIcon
anchors.fill: parent
}
Loader{
id: _overlayIconLoader
anchors.fill: parent
active: communicator.overlayLatteIconIsActive
property color backgroundColor: "black"
property color glowColor: "white"
sourceComponent: LatteCore.IconItem{
id: overlayIconItem
anchors.fill: parent
source: {
if (communicator.appletIconItemIsShown())
return communicator.appletIconItem.source;
else if (communicator.appletImageItemIsShown())
return communicator.appletImageItem.source;
return "";
}
providesColors: indicators.info.needsIconColors && source != ""
usesPlasmaTheme: communicator.appletIconItemIsShown() ? communicator.appletIconItem.usesPlasmaTheme : false
Binding{
target: _overlayIconLoader
property: "backgroundColor"
when: overlayIconItem.providesColors
value: overlayIconItem.backgroundColor
}
Binding{
target: _overlayIconLoader
property: "glowColor"
when: overlayIconItem.providesColors
value: overlayIconItem.glowColor
}
Loader{
anchors.centerIn: parent
active: root.debugModeOverloadedIcons
sourceComponent: Rectangle{
width: 30
height: 30
color: "green"
opacity: 0.65
}
}
}
}
//! WrapperContainer States
states:[
State{
name: "bottom"
when: plasmoid.location === PlasmaCore.Types.BottomEdge
AnchorChanges{
target: _wrapperContainer;
anchors.horizontalCenter: parent.horizontalCenter; anchors.verticalCenter: undefined;
anchors.right: undefined; anchors.left: undefined; anchors.top: undefined; anchors.bottom: parent.bottom;
}
PropertyChanges{
target: _wrapperContainer;
anchors.leftMargin: 0; anchors.rightMargin: 0; anchors.topMargin:0; anchors.bottomMargin: _wrapperContainer.appliedEdgeMargin;
anchors.horizontalCenterOffset: 0; anchors.verticalCenterOffset: 0;
}
},
State{
name: "top"
when: plasmoid.location === PlasmaCore.Types.TopEdge
AnchorChanges{
target:_wrapperContainer;
anchors.horizontalCenter: parent.horizontalCenter; anchors.verticalCenter: undefined;
anchors.right: undefined; anchors.left: undefined; anchors.top: parent.top; anchors.bottom: undefined;
}
PropertyChanges{
target: _wrapperContainer;
anchors.leftMargin: 0; anchors.rightMargin: 0; anchors.topMargin: _wrapperContainer.appliedEdgeMargin; anchors.bottomMargin: 0;
anchors.horizontalCenterOffset: 0; anchors.verticalCenterOffset: 0;
}
},
State{
name: "left"
when: plasmoid.location === PlasmaCore.Types.LeftEdge
AnchorChanges{
target: _wrapperContainer;
anchors.horizontalCenter: undefined; anchors.verticalCenter: parent.verticalCenter;
anchors.right: undefined; anchors.left: parent.left; anchors.top: undefined; anchors.bottom: undefined;
}
PropertyChanges{
target: _wrapperContainer;
anchors.leftMargin: _wrapperContainer.appliedEdgeMargin; anchors.rightMargin: 0; anchors.topMargin:0; anchors.bottomMargin: 0;
anchors.horizontalCenterOffset: 0; anchors.verticalCenterOffset: 0;
}
},
State{
name: "right"
when: plasmoid.location === PlasmaCore.Types.RightEdge
AnchorChanges{
target: _wrapperContainer;
anchors.horizontalCenter: undefined; anchors.verticalCenter: parent.verticalCenter;
anchors.right: parent.right; anchors.left: undefined; anchors.top: undefined; anchors.bottom: undefined;
}
PropertyChanges{
target: _wrapperContainer;
anchors.leftMargin: 0; anchors.rightMargin: _wrapperContainer.appliedEdgeMargin; anchors.topMargin:0; anchors.bottomMargin: 0;
anchors.horizontalCenterOffset: 0; anchors.verticalCenterOffset: 0;
}
}
]
}
Loader{
anchors.fill: _wrapperContainer
active: appletItem.isInternalViewSplitter && root.inConfigureAppletsMode
sourceComponent: Item {
anchors.fill: parent
PlasmaCore.SvgItem{
id:splitterImage
anchors.centerIn: parent
width: Math.min(root.maxJustifySplitterSize, appletItem.metrics.iconSize)
height: width
rotation: root.isVertical ? 90 : 0
svg: PlasmaCore.Svg{
imagePath: root.universalSettings.splitterIconPath()
}
layer.enabled: graphicsSystem.isAccelerated
layer.effect: DropShadow {
radius: root.appShadowSize
fast: true
samples: 2 * radius
color: root.appShadowColor
verticalOffset: 2
}
}
}
}
///Shadow in applets
Loader{
id: appletShadow
anchors.fill: appletItem.appletWrapper
active: appletItem.applet
&& graphicsSystem.isAccelerated
&& !appletColorizer.mustBeShown
&& (root.enableShadows && applet.pluginName !== root.plasmoidName)
onActiveChanged: {
if (active && !isSeparator && graphicsSystem.isAccelerated) {
wrapperContainer.opacity = 0;
} else {
wrapperContainer.opacity = 1;
}
}
opacity: isSeparator ? 0.4 : 1
sourceComponent: DropShadow{
anchors.fill: parent
color: root.appShadowColor //"#ff080808"
fast: true
samples: 2 * radius
source: communicator.overlayLatteIconIsActive ? _wrapperContainer : appletItem.applet
radius: shadowSize
verticalOffset: root.forceTransparentPanel || root.forcePanelForBusyBackground ? 0 : 2
property int shadowSize : root.appShadowSize
}
}
BrightnessContrast{
id:hoveredImage
anchors.fill: _wrapperContainer
source: _wrapperContainer
enabled: appletItem.isSquare && !originalAppletBehavior && !indicators.info.providesHoveredAnimation && opacity != 0 ? true : false
opacity: appletMouseArea.containsMouse ? 1 : 0
brightness: 0.25
contrast: 0.15
visible: !indicators.info.providesHoveredAnimation
Behavior on opacity {
NumberAnimation { duration: appletItem.animations.speedFactor.current*appletItem.animations.duration.large }
}
}
BrightnessContrast {
id: _clickedEffect
anchors.fill: _wrapperContainer
source: _wrapperContainer
visible: clickedAnimation.running && !indicators.info.providesClickedAnimation
}
/* onHeightChanged: {
if ((index == 1)|| (index==3)){
console.log("H: "+index+" ("+zoomScale+"). "+currentLayout.children[1].height+" - "+currentLayout.children[3].height+" - "+(currentLayout.children[1].height+currentLayout.children[3].height));
}
}
onZoomScaleChanged:{
if ((index == 1)|| (index==3)){
console.log(index+" ("+zoomScale+"). "+currentLayout.children[1].height+" - "+currentLayout.children[3].height+" - "+(currentLayout.children[1].height+currentLayout.children[3].height));
}
}*/
Loader{
anchors.fill: parent
active: root.debugMode
sourceComponent: Rectangle{
anchors.fill: parent
color: "transparent"
//! red visualizer, in debug mode for the applets that use fillWidth or fillHeight
//! green, for the rest
border.color: (appletItem.isAutoFillApplet && (appletItem.maxAutoFillLength>-1) && root.isHorizontal) ? "red" : "green"
border.width: 1
}
}
Behavior on zoomScale {
id: animatedScaleBehavior
enabled: !appletItem.parabolic.directRenderingEnabled || restoreAnimation.running
NumberAnimation {
duration: 3 * appletItem.animationTime
easing.type: Easing.OutCubic
}
}
Behavior on zoomScale {
enabled: !animatedScaleBehavior.enabled
NumberAnimation { duration: 0 }
}
function calculateScales( currentMousePosition ){
if (parabolic.factor.zoom===1) {
return;
}
//use the new parabolic effect manager in order to handle all parabolic effect messages
var scales = parabolic.applyParabolicEffect(index, currentMousePosition, center);
//Left hiddenSpacer
if(appletItem.firstAppletInContainer){
hiddenSpacerLeft.nScale = scales.leftScale - 1;
}
//Right hiddenSpacer ///there is one more item in the currentLayout ????
if(appletItem.lastAppletInContainer){
hiddenSpacerRight.nScale = scales.rightScale - 1;
}
zoomScale = parabolic.factor.zoom;
} //scale
function signalUpdateScale(nIndex, nScale, step){
if(appletItem && !appletItem.containsMouse && (appletItem.index === nIndex)){
if ( ((canBeHovered && !appletItem.originalAppletBehavior) || appletItem.latteApplet)
&& (applet && applet.status !== PlasmaCore.Types.HiddenStatus)
){
if(!appletItem.latteApplet){
if(nScale >= 0)
zoomScale = nScale + step;
else
zoomScale = zoomScale + step;
}
}
}
}
function sltUpdateLowerItemScale(delegateIndex, newScale, step) {
if (delegateIndex === appletItem.index) {
if (communicator.parabolicEffectIsSupported) {
communicator.bridge.parabolic.client.hostRequestUpdateLowerItemScale(newScale, step);
return;
}
if (!appletItem.isSeparator && !appletItem.isHidden) {
//! when accepted
signalUpdateScale(delegateIndex, newScale, step);
if (newScale > 1) { // clear lower items
parabolic.sglUpdateLowerItemScale(delegateIndex-1, 1, 0);
}
} else {
parabolic.sglUpdateLowerItemScale(delegateIndex-1, newScale, step);
}
} else if ((newScale === 1) && (appletItem.index < delegateIndex)) {
//! apply zoom clearing
if (communicator.parabolicEffectIsSupported) {
communicator.bridge.parabolic.client.hostRequestUpdateLowerItemScale(1, step);
} else {
signalUpdateScale(appletItem.index, 1, 0);
}
}
}
function sltUpdateHigherItemScale(delegateIndex, newScale, step) {
if (delegateIndex === appletItem.index) {
if (communicator.parabolicEffectIsSupported) {
communicator.bridge.parabolic.client.hostRequestUpdateHigherItemScale(newScale, step);
return;
}
if (!appletItem.isSeparator && !appletItem.isHidden) {
//! when accepted
signalUpdateScale(delegateIndex, newScale, step);
if (newScale > 1) { // clear higher items
parabolic.sglUpdateHigherItemScale(delegateIndex+1, 1, 0);
}
} else {
parabolic.sglUpdateHigherItemScale(delegateIndex+1, newScale, step);
}
} else if ((newScale === 1) && (appletItem.index > delegateIndex)) {
//! apply zoom clearing
if (communicator.parabolicEffectIsSupported) {
communicator.bridge.parabolic.client.hostRequestUpdateHigherItemScale(1, step);
} else {
signalUpdateScale(appletItem.index, 1, 0);
}
}
}
Component.onCompleted: {
parabolic.sglUpdateLowerItemScale.connect(sltUpdateLowerItemScale);
parabolic.sglUpdateHigherItemScale.connect(sltUpdateHigherItemScale);
}
Component.onDestruction: {
parabolic.sglUpdateLowerItemScale.disconnect(sltUpdateLowerItemScale);
parabolic.sglUpdateHigherItemScale.disconnect(sltUpdateHigherItemScale);
}
}// Main task area // id:wrapper