286 lines
8.5 KiB
QML
286 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 2.0 as PlasmaCore
|
||
|
import org.kde.plasma.extras 2.0 as PlasmaExtras
|
||
|
import org.kde.plasma.components 2.0 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: PlasmaCore.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: -PlasmaCore.Units.largeSpacing
|
||
|
leftMargin: -PlasmaCore.Units.largeSpacing
|
||
|
}
|
||
|
|
||
|
width: parent.width
|
||
|
|
||
|
property Item activatedItem: null
|
||
|
property var newModel: null
|
||
|
|
||
|
Behavior on opacity { NumberAnimation { duration: PlasmaCore.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: PlasmaCore.Units.longDuration }
|
||
|
ScriptAction { script: applicationsView.moveRight() }
|
||
|
}
|
||
|
},
|
||
|
Transition {
|
||
|
to: "OutgoingRight"
|
||
|
SequentialAnimation {
|
||
|
NumberAnimation { properties: "x,opacity"; easing.type: Easing.InQuad; duration: PlasmaCore.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: PlasmaCore.Units.shortDuration } }
|
||
|
}
|
||
|
|
||
|
Component.onCompleted: {
|
||
|
rootModel.cleared.connect(refreshed);
|
||
|
}
|
||
|
|
||
|
} // appViewContainer
|