2011-11-03 14:46:05 +01:00
|
|
|
/****************************************************************************
|
|
|
|
**
|
|
|
|
** Copyright (C) 2011 Marco Martin <mart@kde.org>
|
|
|
|
**
|
|
|
|
** Copyright (C) 2011 Nokia Corporation and/or its subsidiary(-ies).
|
|
|
|
** All rights reserved.
|
|
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
|
|
|
**
|
|
|
|
** This file is part of the Qt Components project.
|
|
|
|
**
|
|
|
|
** $QT_BEGIN_LICENSE:BSD$
|
|
|
|
** You may use this file under the terms of the BSD license as follows:
|
|
|
|
**
|
|
|
|
** "Redistribution and use in source and binary forms, with or without
|
|
|
|
** modification, are permitted provided that the following conditions are
|
|
|
|
** met:
|
|
|
|
** * Redistributions of source code must retain the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer.
|
|
|
|
** * Redistributions in binary form must reproduce the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer in
|
|
|
|
** the documentation and/or other materials provided with the
|
|
|
|
** distribution.
|
|
|
|
** * Neither the name of Nokia Corporation and its Subsidiary(-ies) nor
|
|
|
|
** the names of its contributors may be used to endorse or promote
|
|
|
|
** products derived from this software without specific prior written
|
|
|
|
** permission.
|
|
|
|
**
|
|
|
|
** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
|
|
** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
|
|
** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
|
** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE."
|
|
|
|
** $QT_END_LICENSE$
|
|
|
|
**
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
import QtQuick 1.0
|
|
|
|
import org.kde.plasma.core 0.1 as PlasmaCore
|
2011-12-22 15:16:01 +02:00
|
|
|
import "private/AppManager.js" as Utils
|
2011-11-03 14:46:05 +01:00
|
|
|
import "." 0.1
|
2012-11-30 00:00:37 +01:00
|
|
|
import "private" as Private
|
2011-11-03 14:46:05 +01:00
|
|
|
|
2012-12-17 18:28:08 +01:00
|
|
|
/**
|
|
|
|
* Top-level window for short-term tasks and brief interaction with the user
|
|
|
|
*
|
|
|
|
* A dialog floats on the top layer of the application view, usually
|
|
|
|
* overlapping the area reserved for the application content. Normally, a
|
|
|
|
* dialog provides information and gives warnings to the user, or asks the user
|
|
|
|
* to answer a question or select an option.
|
|
|
|
*/
|
2011-11-03 14:46:05 +01:00
|
|
|
Item {
|
|
|
|
id: root
|
|
|
|
|
2012-12-17 18:28:08 +01:00
|
|
|
/**
|
|
|
|
* type:list<Item> A list of items in the dialog's title area. You can use
|
|
|
|
* a Text component but also any number of components that are based on
|
|
|
|
* Item. For example, you can use Text and Image components.
|
|
|
|
*/
|
2011-11-03 14:46:05 +01:00
|
|
|
property alias title: titleBar.children
|
2012-12-17 18:28:08 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* type:list<Item> A list of items in the dialog's content area. You can
|
|
|
|
* use any component that is based on Item. For example, you can use
|
|
|
|
* ListView, so that the user can select from a list of names.
|
|
|
|
*/
|
2011-11-03 14:46:05 +01:00
|
|
|
property alias content: contentItem.children
|
2012-12-17 18:28:08 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* type:list<Item> A list of items in the dialog's button area. For
|
|
|
|
* example, you can use Row or Button components but you can also use any
|
|
|
|
* number of components that are based on Item component.
|
|
|
|
*/
|
2011-11-03 14:46:05 +01:00
|
|
|
property alias buttons: buttonItem.children
|
2012-12-17 18:28:08 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The item that the dialog refers to. The dialog will usually be
|
|
|
|
* positioned relative to VisualParent
|
|
|
|
*/
|
2012-11-30 00:00:37 +01:00
|
|
|
property Item visualParent
|
2012-12-17 18:28:08 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates the dialog's phase in its life cycle. The values are as follows:
|
|
|
|
*
|
|
|
|
* - DialogStatus.Opening - the dialog is opening
|
|
|
|
* - DialogStatus.Open - the dialog is open and visible to the user
|
|
|
|
* - DialogStatus.Closing - the dialog is closing
|
|
|
|
* - DialogStatus.Closed - the dialog is closed and not visible to the user
|
|
|
|
*
|
|
|
|
* The dialog's initial status is DialogStatus.Closed.
|
|
|
|
*/
|
2011-11-03 14:46:05 +01:00
|
|
|
property int status: DialogStatus.Closed
|
|
|
|
|
|
|
|
|
|
|
|
property alias privateTitleHeight: titleBar.height
|
|
|
|
property alias privateButtonsHeight: buttonItem.height
|
|
|
|
|
2012-12-17 18:28:08 +01:00
|
|
|
/**
|
|
|
|
* This signal is emitted when the user accepts the dialog's request or the
|
|
|
|
* accept() method is called.
|
|
|
|
*
|
|
|
|
* @see rejected()
|
|
|
|
*/
|
2011-11-03 14:46:05 +01:00
|
|
|
signal accepted
|
2012-12-17 18:28:08 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* This signal is emitted when the user rejects the dialog's request or the
|
|
|
|
* reject() method is called.
|
|
|
|
*
|
|
|
|
* @see accepted()
|
|
|
|
*/
|
|
|
|
|
2011-11-03 14:46:05 +01:00
|
|
|
signal rejected
|
2012-12-17 18:28:08 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* This signal is emitted when the user taps in the area that is inside the
|
|
|
|
* dialog's visual parent area but outside the dialog's area. Normally the
|
|
|
|
* visual parent is the root object. In that case this signal is emitted if
|
|
|
|
* the user taps anywhere outside the dialog's area.
|
|
|
|
*
|
|
|
|
* @see visualParent
|
|
|
|
*/
|
2011-11-03 14:46:05 +01:00
|
|
|
signal clickedOutside
|
|
|
|
|
2012-12-17 18:28:08 +01:00
|
|
|
/**
|
|
|
|
* Shows the dialog to the user.
|
|
|
|
*/
|
2012-11-30 00:00:37 +01:00
|
|
|
function open() {
|
|
|
|
dialogLayout.parent = internalLoader.item.mainItem
|
2011-11-03 14:46:05 +01:00
|
|
|
|
2012-11-30 00:00:37 +01:00
|
|
|
if (internalLoader.dialog) {
|
|
|
|
var pos = internalLoader.dialog.popupPosition(root.visualParent, Qt.AlignCenter)
|
|
|
|
internalLoader.dialog.x = pos.x
|
|
|
|
internalLoader.dialog.y = pos.y
|
|
|
|
|
|
|
|
internalLoader.dialog.visible = true
|
|
|
|
internalLoader.dialog.activateWindow()
|
|
|
|
} else {
|
|
|
|
internalLoader.inlineDialog.open()
|
|
|
|
}
|
2011-11-03 14:46:05 +01:00
|
|
|
}
|
|
|
|
|
2012-12-17 18:28:08 +01:00
|
|
|
/**
|
|
|
|
* Accepts the dialog's request without any user interaction. The method
|
|
|
|
* emits the accepted() signal and closes the internalLoader.dialog.
|
|
|
|
*
|
|
|
|
* @see reject()
|
|
|
|
*/
|
2012-11-30 00:00:37 +01:00
|
|
|
function accept() {
|
2011-11-03 14:46:05 +01:00
|
|
|
if (status == DialogStatus.Open) {
|
2012-11-30 00:00:37 +01:00
|
|
|
if (internalLoader.dialog) {
|
|
|
|
internalLoader.dialog.visible = false
|
|
|
|
} else {
|
|
|
|
internalLoader.inlineDialog.close()
|
|
|
|
}
|
2011-11-03 14:46:05 +01:00
|
|
|
accepted()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-17 18:28:08 +01:00
|
|
|
/**
|
|
|
|
* Rejects the dialog's request without any user interaction. The method
|
|
|
|
* emits the rejected() signal and closes the internalLoader.dialog.
|
|
|
|
*
|
|
|
|
* @see accept()
|
|
|
|
*/
|
2011-11-03 14:46:05 +01:00
|
|
|
function reject() {
|
|
|
|
if (status == DialogStatus.Open) {
|
2012-11-30 00:00:37 +01:00
|
|
|
if (internalLoader.dialog) {
|
|
|
|
internalLoader.dialog.visible = false
|
|
|
|
} else {
|
|
|
|
internalLoader.inlineDialog.close()
|
|
|
|
}
|
2011-11-03 14:46:05 +01:00
|
|
|
rejected()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-12-17 18:28:08 +01:00
|
|
|
/**
|
|
|
|
* Closes the dialog without any user interaction.
|
|
|
|
*/
|
2011-11-03 14:46:05 +01:00
|
|
|
function close() {
|
2012-11-30 00:00:37 +01:00
|
|
|
if (internalLoader.dialog) {
|
|
|
|
internalLoader.dialog.visible = false
|
|
|
|
} else {
|
|
|
|
internalLoader.inlineDialog.close()
|
|
|
|
}
|
2011-11-03 14:46:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
visible: false
|
|
|
|
|
2012-11-30 00:00:37 +01:00
|
|
|
Loader {
|
|
|
|
id: internalLoader
|
|
|
|
//the root item of the scene. Determines if there is enough room for an inline dialog
|
|
|
|
property Item rootItem
|
2011-11-03 14:46:05 +01:00
|
|
|
|
2012-11-30 00:00:37 +01:00
|
|
|
//this is when the dialog is a separate window
|
|
|
|
property Item dialog: sourceComponent == dialogComponent ? item : null
|
|
|
|
//this is when the dialog is inline
|
|
|
|
property Item inlineDialog: sourceComponent == inlineDialogComponent ? item : null
|
2011-11-03 14:46:05 +01:00
|
|
|
|
2012-11-30 00:00:37 +01:00
|
|
|
property bool loadCompleted: false
|
2011-11-03 14:46:05 +01:00
|
|
|
|
2012-11-30 00:00:37 +01:00
|
|
|
Component.onCompleted: {
|
|
|
|
rootItem = Utils.rootObject()
|
|
|
|
loadCompleted = true
|
|
|
|
}
|
|
|
|
|
|
|
|
sourceComponent: {
|
|
|
|
if (loadCompleted) {
|
|
|
|
if (rootItem == null || dialogLayout.width > rootItem.width || dialogLayout.height > rootItem.height) {
|
|
|
|
dialogComponent
|
|
|
|
} else {
|
|
|
|
inlineDialogComponent
|
|
|
|
}
|
2011-11-03 14:46:05 +01:00
|
|
|
}
|
|
|
|
}
|
2012-11-30 00:00:37 +01:00
|
|
|
}
|
2011-11-03 14:46:05 +01:00
|
|
|
|
2012-11-30 00:00:37 +01:00
|
|
|
Component {
|
|
|
|
id: dialogComponent
|
|
|
|
PlasmaCore.Dialog {
|
|
|
|
windowFlags: Qt.Popup
|
|
|
|
|
|
|
|
//state: "Hidden"
|
|
|
|
visible: false
|
|
|
|
onVisibleChanged: {
|
|
|
|
if (visible) {
|
|
|
|
root.status = DialogStatus.Open
|
|
|
|
} else {
|
|
|
|
root.status = DialogStatus.Closed
|
|
|
|
}
|
|
|
|
}
|
2011-11-03 14:46:05 +01:00
|
|
|
|
2012-11-30 00:00:37 +01:00
|
|
|
mainItem: Item {
|
|
|
|
id: dialogMainItem
|
|
|
|
width: dialogLayout.width
|
|
|
|
height: dialogLayout.height
|
|
|
|
}
|
|
|
|
|
|
|
|
Component.onCompleted: dialogLayout.parent = dialogMainItem
|
|
|
|
Component.onDestruction: dialogLayout.parent = root
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Component {
|
|
|
|
id: inlineDialogComponent
|
|
|
|
Private.InlineDialog {
|
|
|
|
id: inlineDialog
|
|
|
|
visualParent: root.visualParent
|
|
|
|
property Item mainItem: inlineDialogMainItem
|
|
|
|
onStatusChanged: root.status = status
|
2011-11-03 14:46:05 +01:00
|
|
|
|
|
|
|
Item {
|
2012-11-30 00:00:37 +01:00
|
|
|
id: inlineDialogMainItem
|
|
|
|
width: dialogLayout.width
|
|
|
|
height: dialogLayout.height
|
|
|
|
}
|
2011-11-03 14:46:05 +01:00
|
|
|
|
2012-11-30 00:00:37 +01:00
|
|
|
Component.onCompleted: {
|
|
|
|
dialogLayout.parent = inlineDialogMainItem
|
2011-11-03 14:46:05 +01:00
|
|
|
}
|
2012-11-30 00:00:37 +01:00
|
|
|
Component.onDestruction: dialogLayout.parent = root
|
|
|
|
}
|
|
|
|
}
|
2011-11-03 14:46:05 +01:00
|
|
|
|
2012-11-30 00:00:37 +01:00
|
|
|
Item {
|
|
|
|
id: dialogLayout
|
2012-11-30 12:01:25 +01:00
|
|
|
width: Math.max(buttonItem.childrenRect.width, Math.min(contentItem.childrenRect.width, theme.defaultFont.mSize.width * 30))
|
|
|
|
height: titleBar.height + contentItem.childrenRect.height + buttonItem.childrenRect.height + 10
|
2011-11-03 14:46:05 +01:00
|
|
|
|
2012-11-30 00:00:37 +01:00
|
|
|
parent: internalLoader.dialog ? internalLoader.dialog : internalLoader.inlineDialog
|
|
|
|
// Consume all key events that are not processed by children
|
|
|
|
Keys.onPressed: event.accepted = true
|
|
|
|
Keys.onReleased: event.accepted = true
|
2011-11-25 19:59:00 +01:00
|
|
|
|
2012-11-30 00:00:37 +01:00
|
|
|
Item {
|
|
|
|
id: titleBar
|
|
|
|
|
2012-11-30 12:01:25 +01:00
|
|
|
height: children.length > 0 && children[0].visible ? childrenRect.height : 0
|
2012-11-30 00:00:37 +01:00
|
|
|
anchors {
|
|
|
|
top: parent.top
|
|
|
|
left: parent.left
|
|
|
|
right: parent.right
|
2011-11-03 14:46:05 +01:00
|
|
|
}
|
2012-11-30 00:00:37 +01:00
|
|
|
}
|
2011-11-03 14:46:05 +01:00
|
|
|
|
2012-11-30 00:00:37 +01:00
|
|
|
Item {
|
|
|
|
id: contentItem
|
|
|
|
|
|
|
|
clip: true
|
|
|
|
anchors {
|
|
|
|
top: titleBar.bottom
|
|
|
|
left: parent.left
|
|
|
|
right: parent.right
|
|
|
|
bottom: buttonItem.top
|
2012-11-30 12:01:25 +01:00
|
|
|
bottomMargin: 6
|
2011-11-03 14:46:05 +01:00
|
|
|
}
|
|
|
|
}
|
2011-11-25 19:59:00 +01:00
|
|
|
|
2012-11-30 00:00:37 +01:00
|
|
|
Item {
|
|
|
|
id: buttonItem
|
|
|
|
|
|
|
|
height: childrenRect.height
|
|
|
|
anchors {
|
|
|
|
left: parent.left
|
|
|
|
right: parent.right
|
|
|
|
bottom: parent.bottom
|
|
|
|
bottomMargin: 4
|
|
|
|
}
|
2011-11-03 14:46:05 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|