2 * Copyright (C) 2014-2017 Canonical Ltd.
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation; version 3.
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
13 * You should have received a copy of the GNU General Public License
14 * along with this program. If not, see <http://www.gnu.org/licenses/>.
18import Lomiri.Components 1.3
19import QtMir.Application 0.1
20import "Spread/MathUtils.js" as MathUtils
21import Lomiri.ApplicationMenu 0.1
22import Lomiri.Indicators 0.1 as Indicators
23import "../Components/PanelState"
28 // The DecoratedWindow takes requestedWidth/requestedHeight and asks its surface to be resized to that
29 // (minus the window decoration size in case hasDecoration and showDecoration are true)
30 // The surface might not be able to resize to the requested values. It will return its actual size
31 // in implicitWidth/implicitHeight.
33 property alias application: applicationWindow.application
34 property alias surface: applicationWindow.surface
35 readonly property alias focusedSurface: applicationWindow.focusedSurface
36 readonly property alias supportsResize: applicationWindow.supportsResize
37 property alias active: decoration.active
38 readonly property alias title: applicationWindow.title
39 property alias maximizeButtonShown: decoration.maximizeButtonShown
40 property alias interactive: applicationWindow.interactive
41 readonly property alias orientationChangesEnabled: applicationWindow.orientationChangesEnabled
42 property alias windowControlButtonsVisible: decoration.windowControlButtonsVisible
43 property PanelState panelState
45 // Changing this will actually add/remove a decoration, meaning, requestedHeight will take the decoration into account.
46 property bool hasDecoration: true
47 // This will temporarily show/hide the decoration without actually changing the surface's dimensions
48 property real showDecoration: 1
49 property alias decorationHeight: decoration.height
50 property bool animateDecoration: false
51 property bool showHighlight: false
52 property int highlightSize: units.gu(1)
53 property real shadowOpacity: 0
54 property bool darkening: false
55 property bool lightMode: false
57 property real requestedWidth
58 property real requestedHeight
59 property real scaleToPreviewProgress: 0
60 property int scaleToPreviewSize: units.gu(30)
62 property alias surfaceOrientationAngle: applicationWindow.surfaceOrientationAngle
64 // Height of the decoration that's actually being displayed at this moment. Will match decorationHeight
65 // when the decoration is being fully displayed
66 readonly property real actualDecorationHeight: Math.min(d.visibleDecorationHeight, d.requestedDecorationHeight)
68 readonly property bool counterRotate: surfaceOrientationAngle != 0 && surfaceOrientationAngle != 180
70 readonly property int minimumWidth: !counterRotate ? applicationWindow.minimumWidth : applicationWindow.minimumHeight
71 readonly property int minimumHeight: actualDecorationHeight + (!counterRotate ? applicationWindow.minimumHeight : applicationWindow.minimumWidth)
72 readonly property int maximumWidth: !counterRotate ? applicationWindow.maximumWidth : applicationWindow.maximumHeight
73 readonly property int maximumHeight: (root.decorationShown && applicationWindow.maximumHeight > 0 ? decoration.height : 0)
74 + (!counterRotate ? applicationWindow.maximumHeight : applicationWindow.maximumWidth)
75 readonly property int widthIncrement: !counterRotate ? applicationWindow.widthIncrement : applicationWindow.heightIncrement
76 readonly property int heightIncrement: !counterRotate ? applicationWindow.heightIncrement : applicationWindow.widthIncrement
78 property alias overlayShown: decoration.overlayShown
79 property alias boundsItem: moveHandler.boundsItem
80 readonly property alias dragging: moveHandler.dragging
82 readonly property Item clientAreaItem: applicationWindow
84 property alias altDragEnabled: altDragHandler.enabled
86 property alias clipSurface: applicationWindow.clip
88 property Item windowMargins
91 signal maximizeClicked()
92 signal maximizeHorizontallyClicked()
93 signal maximizeVerticallyClicked()
94 signal minimizeClicked()
95 signal decorationPressed()
96 signal decorationReleased()
98 function cancelDrag() {
99 moveHandler.cancelDrag();
104 property int requestedDecorationHeight: root.hasDecoration ? decoration.height : 0
105 Behavior on requestedDecorationHeight { enabled: root.animateDecoration; LomiriNumberAnimation { } }
107 property int visibleDecorationHeight: root.hasDecoration ? root.showDecoration * decoration.height : 0
108 Behavior on visibleDecorationHeight { enabled: root.animateDecoration; LomiriNumberAnimation { } }
114 name: "normal"; when: root.scaleToPreviewProgress <= 0 && root.application.state === ApplicationInfoInterface.Running
117 implicitWidth: counterRotate ? applicationWindow.implicitHeight : applicationWindow.implicitWidth
118 implicitHeight: root.actualDecorationHeight + (counterRotate ? applicationWindow.implicitWidth: applicationWindow.implicitHeight)
122 name: "normalSuspended"; when: root.scaleToPreviewProgress <= 0 && root.application.state !== ApplicationInfoInterface.Running
126 implicitWidth: counterRotate ? applicationWindow.requestedHeight : applicationWindow.requestedWidth
127 implicitHeight: root.actualDecorationHeight + (counterRotate ? applicationWindow.requestedWidth: applicationWindow.requestedHeight)
131 name: "preview"; when: root.scaleToPreviewProgress > 0
134 implicitWidth: MathUtils.linearAnimation(0, 1, applicationWindow.requestedWidth, root.scaleToPreviewSize, root.scaleToPreviewProgress)
135 implicitHeight: MathUtils.linearAnimation(0, 1, applicationWindow.requestedHeight, root.scaleToPreviewSize, root.scaleToPreviewProgress)
138 target: applicationWindow;
139// requestedWidth: applicationWindow.oldRequestedWidth
140// requestedHeight: applicationWindow.oldRequestedHeight
141 width: MathUtils.linearAnimation(0, 1, applicationWindow.requestedWidth, applicationWindow.minSize, root.scaleToPreviewProgress)
142 height: MathUtils.linearAnimation(0, 1, applicationWindow.requestedHeight, applicationWindow.minSize, root.scaleToPreviewProgress)
143 itemScale: root.implicitWidth / width
150 id: selectionHighlight
151 objectName: "selectionHighlight"
153 anchors.margins: -root.highlightSize
155 opacity: showHighlight ? 0.55 : 0
162 left: parent.left; top: parent.top; right: parent.right
163 margins: active ? -units.gu(2) : -units.gu(1.5)
165 height: Math.min(applicationWindow.implicitHeight, applicationWindow.height) * applicationWindow.itemScale
166 + root.actualDecorationHeight * Math.min(1, root.showDecoration) + (active ? units.gu(4) : units.gu(3))
167 source: "../graphics/dropshadow2gu.sci"
168 opacity: root.shadowOpacity
172 id: applicationWindow
173 objectName: "appWindow"
174 anchors.top: parent.top
175 anchors.topMargin: root.actualDecorationHeight * Math.min(1, root.showDecoration)
176 anchors.left: parent.left
178 height: implicitHeight
179 requestedHeight: !counterRotate ? root.requestedHeight - d.requestedDecorationHeight : root.requestedWidth
180 requestedWidth: !counterRotate ? root.requestedWidth : root.requestedHeight - d.requestedDecorationHeight
181// property int oldRequestedWidth: requestedWidth
182// property int oldRequestedHeight: requestedHeight
183// onRequestedWidthChanged: oldRequestedWidth = requestedWidth
184// onRequestedHeightChanged: oldRequestedHeight = requestedHeight
187 property real itemScale: 1
188 property real minSize: Math.min(root.scaleToPreviewSize, Math.min(requestedHeight, Math.min(requestedWidth, Math.min(implicitHeight, implicitWidth))))
192 id: rotationTransform
193 readonly property int rotationAngle: applicationWindow.application &&
194 applicationWindow.application.rotatesWindowContents
195 ? ((360 - applicationWindow.surfaceOrientationAngle) % 360) : 0
197 if (rotationAngle == 90) return applicationWindow.height / 2;
198 else if (rotationAngle == 270) return applicationWindow.width / 2;
199 else if (rotationAngle == 180) return applicationWindow.width / 2;
203 if (rotationAngle == 90) return applicationWindow.height / 2;
204 else if (rotationAngle == 270) return applicationWindow.width / 2;
205 else if (rotationAngle == 180) return applicationWindow.height / 2;
211 xScale: applicationWindow.itemScale
212 yScale: applicationWindow.itemScale
219 closeButtonVisible: true
220 objectName: "appWindowDecoration"
222 anchors { left: parent.left; top: parent.top; right: parent.right }
223 height: units.gu(3) // a default value. overwritten by root.decorationHeight
225 title: applicationWindow.title
226 windowMoving: moveHandler.moving && !altDragHandler.dragging
227 panelState: root.panelState
228 lightMode: root.lightMode
230 opacity: root.hasDecoration ? Math.min(1, root.showDecoration) : 0
231 Behavior on opacity { LomiriNumberAnimation { } }
232 visible: opacity > 0 // don't eat input when decoration is fully translucent
234 onPressed: root.decorationPressed();
235 onPressedChanged: moveHandler.handlePressedChanged(pressed, pressedButtons, mouseX, mouseY)
236 onPressedChangedEx: moveHandler.handlePressedChanged(pressed, pressedButtons, mouseX, mouseY)
237 onPositionChanged: moveHandler.handlePositionChanged(mouse)
239 root.decorationReleased();
240 moveHandler.handleReleased();
243 onCloseClicked: root.closeClicked();
244 onMaximizeClicked: { root.decorationPressed(); root.maximizeClicked(); }
245 onMaximizeHorizontallyClicked: { root.decorationPressed(); root.maximizeHorizontallyClicked(); }
246 onMaximizeVerticallyClicked: { root.decorationPressed(); root.maximizeVerticallyClicked(); }
247 onMinimizeClicked: root.minimizeClicked();
252 (panelState.focusedPersistentSurfaceId === surface.persistentId && !panelState.decorationsVisible)
254 menu: sharedAppModel.model
256 Indicators.SharedLomiriMenuModel {
258 property var menus: surface ? ApplicationMenuRegistry.getMenusForSurface(surface.persistentId) : []
259 property var menuService: menus.length > 0 ? menus[0] : undefined
261 busName: menuService ? menuService.service : ""
262 menuObjectPath: menuService && menuService.menuPath ? menuService.menuPath : ""
263 actions: menuService && menuService.actionPath ? { "lomiri": menuService.actionPath } : {}
267 target: ApplicationMenuRegistry
268 onSurfaceMenuRegistered: {
269 if (surface && surfaceId === surface.persistentId) {
270 sharedAppModel.menus = Qt.binding(function() { return surface ? ApplicationMenuRegistry.getMenusForSurface(surface.persistentId) : [] });
273 onSurfaceMenuUnregistered: {
274 if (surface && surfaceId === surface.persistentId) {
275 sharedAppModel.menus = Qt.binding(function() { return surface ? ApplicationMenuRegistry.getMenusForSurface(surface.persistentId) : [] });
283 anchors.fill: applicationWindow
284 acceptedButtons: Qt.LeftButton
285 property bool dragging: false
286 cursorShape: undefined // don't interfere with the cursor shape set by the underlying MirSurfaceItem
289 if (mouse.button == Qt.LeftButton && mouse.modifiers == Qt.AltModifier) {
290 root.decorationPressed(); // to raise it
291 moveHandler.handlePressedChanged(true, Qt.LeftButton, mouse.x, mouse.y);
293 mouse.accepted = true;
295 mouse.accepted = false;
300 moveHandler.handlePositionChanged(mouse);
305 moveHandler.handlePressedChanged(false, Qt.LeftButton);
306 root.decorationReleased(); // commits the fake preview max rectangle
307 moveHandler.handleReleased();
315 objectName: "moveHandler"
317 buttonsWidth: decoration.buttonsWidth
323 opacity: root.darkening && !root.showHighlight ? 0.05 : 0
324 Behavior on opacity { LomiriNumberAnimation { duration: LomiriAnimation.SnapDuration } }