287 lines
8.5 KiB
QML
287 lines
8.5 KiB
QML
/*
|
|
Copyright (C) 2011 Martin Gräßlin <mgraesslin@kde.org>
|
|
Copyright (C) 2012 Gregor Taetzner <gregor@freenet.de>
|
|
Copyright 2014 Sebastian Kügler <sebas@kde.org>
|
|
Copyright (C) 2015-2018 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 org.kde.plasma.core as PlasmaCore
|
|
import org.kde.kirigami as Kirigami
|
|
import org.kde.plasma.extras as PlasmaExtras
|
|
import org.kde.plasma.components as PlasmaComponents
|
|
|
|
Item {
|
|
id: appViewContainer
|
|
|
|
anchors.fill: parent
|
|
|
|
objectName: "ApplicationsView"
|
|
|
|
property ListView listView: applicationsView.listView
|
|
|
|
function decrementCurrentIndex() {
|
|
applicationsView.decrementCurrentIndex();
|
|
}
|
|
|
|
function incrementCurrentIndex() {
|
|
applicationsView.incrementCurrentIndex();
|
|
}
|
|
|
|
function activateCurrentIndex(start) {
|
|
if (!applicationsView.currentItem.modelChildren) {
|
|
if (!start) {
|
|
return;
|
|
}
|
|
}
|
|
applicationsView.state = "OutgoingLeft";
|
|
}
|
|
|
|
function openContextMenu() {
|
|
applicationsView.currentItem.openActionMenu();
|
|
}
|
|
|
|
function deactivateCurrentIndex() {
|
|
if (crumbModel.count > 0) { // this is not the case when switching from the "Applications" to the "Favorites" tab using the "Left" key
|
|
breadcrumbsElement.children[crumbModel.count-1].clickCrumb();
|
|
applicationsView.state = "OutgoingRight";
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
function reset() {
|
|
applicationsView.model = rootModel;
|
|
applicationsView.clearBreadcrumbs();
|
|
}
|
|
|
|
function refreshed() {
|
|
reset();
|
|
updatedLabelTimer.running = true;
|
|
}
|
|
|
|
Connections {
|
|
target: plasmoid
|
|
function onExpandedChanged() {
|
|
if (!plasmoid.expanded) {
|
|
reset();
|
|
}
|
|
}
|
|
}
|
|
|
|
Item {
|
|
id: crumbContainer
|
|
|
|
anchors {
|
|
top: parent.top
|
|
left: parent.left
|
|
right: parent.right
|
|
}
|
|
height: childrenRect.height
|
|
|
|
Behavior on opacity { NumberAnimation { duration: Kirigami.Units.longDuration } }
|
|
|
|
Flickable {
|
|
id: breadcrumbFlickable
|
|
anchors {
|
|
top: parent.top
|
|
left: parent.left
|
|
right: parent.right
|
|
}
|
|
height: breadcrumbsElement.height
|
|
boundsBehavior: Flickable.StopAtBounds
|
|
|
|
contentWidth: breadcrumbsElement.width
|
|
pixelAligned: true
|
|
//contentX: contentWidth - width
|
|
|
|
// HACK: Align the content to right for RTL locales
|
|
leftMargin: LayoutMirroring.enabled ? Math.max(0, width - contentWidth) : 0
|
|
|
|
PlasmaComponents.ButtonRow {
|
|
id: breadcrumbsElement
|
|
|
|
exclusive: false
|
|
|
|
Breadcrumb {
|
|
id: rootBreadcrumb
|
|
root: true
|
|
text: i18n("All Applications")
|
|
depth: 0
|
|
}
|
|
Repeater {
|
|
model: ListModel {
|
|
id: crumbModel
|
|
// Array of the models
|
|
property var models: []
|
|
}
|
|
|
|
Breadcrumb {
|
|
root: false
|
|
text: model.text
|
|
}
|
|
}
|
|
onWidthChanged: {
|
|
if (LayoutMirroring.enabled) {
|
|
breadcrumbFlickable.contentX = -Math.max(0, breadcrumbsElement.width - breadcrumbFlickable.width)
|
|
} else {
|
|
breadcrumbFlickable.contentX = Math.max(0, breadcrumbsElement.width - breadcrumbFlickable.width)
|
|
}
|
|
}
|
|
}
|
|
} // Flickable
|
|
} // crumbContainer
|
|
|
|
KickoffListView {
|
|
id: applicationsView
|
|
|
|
anchors {
|
|
top: crumbContainer.bottom
|
|
bottom: parent.bottom
|
|
rightMargin: -Kirigami.Units.gridUnit
|
|
leftMargin: -Kirigami.Units.gridUnit
|
|
}
|
|
|
|
width: parent.width
|
|
|
|
property Item activatedItem: null
|
|
property var newModel: null
|
|
|
|
Behavior on opacity { NumberAnimation { duration: Kirigami.Units.longDuration } }
|
|
|
|
focus: true
|
|
|
|
appView: true
|
|
|
|
model: rootModel
|
|
|
|
function moveLeft() {
|
|
state = "";
|
|
// newModelIndex set by clicked breadcrumb
|
|
var oldModel = applicationsView.model;
|
|
applicationsView.model = applicationsView.newModel;
|
|
|
|
var oldModelIndex = model.rowForModel(oldModel);
|
|
listView.currentIndex = oldModelIndex;
|
|
listView.positionViewAtIndex(oldModelIndex, ListView.Center);
|
|
}
|
|
|
|
function moveRight() {
|
|
state = "";
|
|
activatedItem.activate()
|
|
applicationsView.listView.positionViewAtBeginning()
|
|
}
|
|
|
|
function clearBreadcrumbs() {
|
|
crumbModel.clear();
|
|
crumbModel.models = [];
|
|
}
|
|
|
|
onReset: appViewContainer.reset()
|
|
|
|
onAddBreadcrumb: {
|
|
crumbModel.append({"text": title, "depth": crumbModel.count+1})
|
|
crumbModel.models.push(model);
|
|
}
|
|
|
|
states: [
|
|
State {
|
|
name: "OutgoingLeft"
|
|
PropertyChanges {
|
|
target: applicationsView
|
|
x: -parent.width
|
|
opacity: 0.0
|
|
}
|
|
},
|
|
State {
|
|
name: "OutgoingRight"
|
|
PropertyChanges {
|
|
target: applicationsView
|
|
x: parent.width
|
|
opacity: 0.0
|
|
}
|
|
}
|
|
]
|
|
|
|
transitions: [
|
|
Transition {
|
|
to: "OutgoingLeft"
|
|
SequentialAnimation {
|
|
// We need to cache the currentItem since the selection can move during animation,
|
|
// and we want the item that has been clicked on, not the one that is under the
|
|
// mouse once the animation is done
|
|
ScriptAction { script: applicationsView.activatedItem = applicationsView.currentItem }
|
|
NumberAnimation { properties: "x,opacity"; easing.type: Easing.InQuad; duration: Kirigami.Units.longDuration }
|
|
ScriptAction { script: applicationsView.moveRight() }
|
|
}
|
|
},
|
|
Transition {
|
|
to: "OutgoingRight"
|
|
SequentialAnimation {
|
|
NumberAnimation { properties: "x,opacity"; easing.type: Easing.InQuad; duration: Kirigami.Units.longDuration }
|
|
ScriptAction { script: applicationsView.moveLeft() }
|
|
}
|
|
}
|
|
]
|
|
}
|
|
|
|
MouseArea {
|
|
anchors.fill: parent
|
|
|
|
acceptedButtons: Qt.BackButton
|
|
|
|
onClicked: {
|
|
deactivateCurrentIndex()
|
|
}
|
|
}
|
|
|
|
Timer {
|
|
id: updatedLabelTimer
|
|
interval: 1500
|
|
running: false
|
|
repeat: true
|
|
|
|
onRunningChanged: {
|
|
if (running) {
|
|
updatedLabel.opacity = 1;
|
|
crumbContainer.opacity = 0.3;
|
|
applicationsView.scrollArea.opacity = 0.3;
|
|
}
|
|
}
|
|
onTriggered: {
|
|
updatedLabel.opacity = 0;
|
|
crumbContainer.opacity = 1;
|
|
applicationsView.scrollArea.opacity = 1;
|
|
running = false;
|
|
}
|
|
}
|
|
|
|
PlasmaComponents.Label {
|
|
id: updatedLabel
|
|
text: i18n("Applications updated.")
|
|
opacity: 0
|
|
visible: opacity != 0
|
|
anchors.centerIn: parent
|
|
|
|
Behavior on opacity { NumberAnimation { duration: Kirigami.Units.shortDuration } }
|
|
}
|
|
|
|
Component.onCompleted: {
|
|
rootModel.cleared.connect(refreshed);
|
|
}
|
|
|
|
} // appViewContainer
|