feat(StatusChatList): Add drag and drop support of list items

This implements drag and drop capabilities of chat items within a `StatusChatList`.
The commit introduces a `DelegateModal` to visually reorder chat items
when they're being dragged and dropped onto a `DropArea`.

To persist the new order of chat items, various signals have been introduced to chat
list related components:

```qml
StatusChatList {

    onChatItemReordered: function (id, from, to) {
        // ...
    }
}

StatusChatListAndCategories {

    onChatItemReordered: function (categoryId, chatId, from, to) {
        // ...
    }
}
```

There's no such API on the `StatusChatListCategory` type because that one already
exposes its underlying `StatusChatList` via `chatList`, which makes the signal available.

Dragging and dropping chat items is disabled by default and needs to be turned on
using the `draggableItems` property:

```qml
StatusChatList {
    draggableItems: true
    ...
}
```
This commit is contained in:
B.Melnik 2021-07-28 14:08:56 +03:00 committed by Michał Cieślak
parent 67031ad5b1
commit 85ee81cfa3
5 changed files with 254 additions and 96 deletions

View File

@ -467,6 +467,7 @@ Rectangle {
width: leftPanel.width width: leftPanel.width
height: implicitHeight > (leftPanel.height - 64) ? implicitHeight + 8 : leftPanel.height - 64 height: implicitHeight > (leftPanel.height - 64) ? implicitHeight + 8 : leftPanel.height - 64
draggableItems: true
chatList.model: models.demoCommunityChatListItems chatList.model: models.demoCommunityChatListItems
categoryList.model: models.demoCommunityCategoryItems categoryList.model: models.demoCommunityCategoryItems

View File

@ -13,6 +13,7 @@ QtObject {
unreadMessagesCount: 0 unreadMessagesCount: 0
mentionsCount: 0 mentionsCount: 0
color: "blue" color: "blue"
position: 0
} }
ListElement { ListElement {
chatId: "1" chatId: "1"
@ -22,6 +23,7 @@ QtObject {
color: "red" color: "red"
unreadMessagesCount: 1 unreadMessagesCount: 1
mentionsCount: 1 mentionsCount: 1
position: 1
} }
ListElement { ListElement {
chatId: "2" chatId: "2"
@ -32,6 +34,7 @@ QtObject {
identicon: " identicon: "
CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2ImYgiNITTlTdG1nUZ5a92VITQxITFiJmIIjSE0htAYQrMHAAD//+wwFVpz+yqXAAAAAElFTkSuQmCC" CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2ImYgiNITTlTdG1nUZ5a92VITQxITFiJmIIjSE0htAYQrMHAAD//+wwFVpz+yqXAAAAAElFTkSuQmCC"
unreadMessagesCount: 0 unreadMessagesCount: 0
position: 2
} }
ListElement { ListElement {
chatId: "3" chatId: "3"
@ -40,6 +43,7 @@ CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2I
muted: false muted: false
color: "purple" color: "purple"
unreadMessagesCount: 0 unreadMessagesCount: 0
position: 3
} }
ListElement { ListElement {
chatId: "4" chatId: "4"
@ -48,6 +52,7 @@ CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2I
muted: true muted: true
color: "Orange" color: "Orange"
unreadMessagesCount: 0 unreadMessagesCount: 0
position: 4
} }
ListElement { ListElement {
chatId: "5" chatId: "5"
@ -56,6 +61,7 @@ CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2I
muted: false muted: false
color: "green" color: "green"
unreadMessagesCount: 0 unreadMessagesCount: 0
position: 5
} }
} }
@ -68,6 +74,7 @@ CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2I
muted: false muted: false
unreadMessagesCount: 0 unreadMessagesCount: 0
color: "orange" color: "orange"
position: 0
} }
ListElement { ListElement {
chatId: "1" chatId: "1"
@ -77,6 +84,7 @@ CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2I
unreadMessagesCount: 0 unreadMessagesCount: 0
color: "orange" color: "orange"
categoryId: "public" categoryId: "public"
position: 0
} }
ListElement { ListElement {
chatId: "2" chatId: "2"
@ -86,6 +94,7 @@ CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2I
unreadMessagesCount: 0 unreadMessagesCount: 0
color: "orange" color: "orange"
categoryId: "public" categoryId: "public"
position: 1
} }
ListElement { ListElement {
chatId: "3" chatId: "3"
@ -95,6 +104,7 @@ CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2I
unreadMessagesCount: 0 unreadMessagesCount: 0
color: "orange" color: "orange"
categoryId: "dev" categoryId: "dev"
position: 0
} }
} }

View File

@ -1,4 +1,6 @@
import QtQuick 2.13 import QtQuick 2.13
import QtQml.Models 2.14
import QtQuick.Controls 2.13 as QC
import StatusQ.Core 0.1 import StatusQ.Core 0.1
import StatusQ.Core.Theme 0.1 import StatusQ.Core.Theme 0.1
@ -13,7 +15,8 @@ Column {
property string categoryId: "" property string categoryId: ""
property string selectedChatId: "" property string selectedChatId: ""
property alias chatListItems: statusChatListItems property alias chatListItems: delegateModel
property bool draggableItems: false
property Component popupMenu property Component popupMenu
@ -23,6 +26,19 @@ Column {
signal chatItemSelected(string id) signal chatItemSelected(string id)
signal chatItemUnmuted(string id) signal chatItemUnmuted(string id)
signal chatItemReordered(string id, int from, int to)
function getAbsolutePosition(node) {
var returnPos = {};
returnPos.x = 0;
returnPos.y = 0;
if (node !== undefined && node !== null) {
var parentValue = getAbsolutePosition(node.parent);
returnPos.x = parentValue.x + node.x;
returnPos.y = parentValue.y + node.y;
}
return returnPos;
}
onPopupMenuChanged: { onPopupMenuChanged: {
if (!!popupMenu) { if (!!popupMenu) {
@ -30,21 +46,77 @@ Column {
} }
} }
Repeater { DelegateModel {
id: statusChatListItems id: delegateModel
delegate: StatusChatListItem {
delegate: Item {
id: draggable
width: statusChatListItem.width
height: statusChatListItem.height
property alias chatListItem: statusChatListItem
visible: {
if (!!statusChatList.filterFn) {
return statusChatList.filterFn(model, statusChatList.categoryId)
}
return true
}
MouseArea {
id: dragSensor
anchors.fill: parent
cursorShape: active ? Qt.ClosedHandCursor : Qt.PointingHandCursor
hoverEnabled: true
pressAndHoldInterval: 150
enabled: statusChatList.draggableItems
property bool active: false
property real startY: 0
property real startX: 0
drag.target: draggedListItemLoader.item
drag.threshold: 0.1
drag.filterChildren: true
onPressed: {
startY = mouseY
startX = mouseX
}
onPressAndHold: active = true
onReleased: {
if (active) {
statusChatList.chatItemReordered(statusChatListItem.chatId, statusChatListItem.originalOrder, statusChatListItem.originalOrder)
}
active = false
}
onMouseYChanged: {
if ((Math.abs(startY - mouseY) > 1) && pressed) {
active = true
}
}
onMouseXChanged: {
if ((Math.abs(startX - mouseX) > 1) && pressed) {
active = true
}
}
StatusChatListItem {
id: statusChatListItem id: statusChatListItem
property string profileImage: "" property string profileImage: ""
opacity: dragSensor.active ? 0.0 : 1.0
Component.onCompleted: { Component.onCompleted: {
if (typeof statusChatList.profileImageFn === "function") { if (typeof statusChatList.profileImageFn === "function") {
profileImage = statusChatList.profileImageFn(model.chatId || model.id) || "" profileImage = statusChatList.profileImageFn(model.chatId || model.id) || ""
} }
} }
originalOrder: model.position
chatId: model.chatId || model.id chatId: model.chatId || model.id
categoryId: model.categoryId || ""
name: !!statusChatList.chatNameFn ? statusChatList.chatNameFn(model) : model.name name: !!statusChatList.chatNameFn ? statusChatList.chatNameFn(model) : model.name
type: model.chatType type: model.chatType
muted: !!model.muted muted: !!model.muted
@ -59,6 +131,7 @@ Column {
image.isIdenticon: !!!profileImage && !!!model.identityImage && !!model.identicon image.isIdenticon: !!!profileImage && !!!model.identityImage && !!model.identicon
image.source: profileImage || model.identityImage || model.identicon || "" image.source: profileImage || model.identityImage || model.identicon || ""
sensor.cursorShape: dragSensor.cursorShape
onClicked: { onClicked: {
if (mouse.button === Qt.RightButton && !!statusChatList.popupMenu) { if (mouse.button === Qt.RightButton && !!statusChatList.popupMenu) {
statusChatListItem.highlighted = true statusChatListItem.highlighted = true
@ -90,13 +163,70 @@ Column {
} }
} }
onUnmute: statusChatList.chatItemUnmuted(model.chatId || model.id) onUnmute: statusChatList.chatItemUnmuted(model.chatId || model.id)
visible: {
if (!!statusChatList.filterFn) {
return statusChatList.filterFn(model, statusChatList.categoryId)
}
return true
} }
} }
DropArea {
id: dropArea
width: dragSensor.active ? 0 : parent.width
height: dragSensor.active ? 0 : parent.height
keys: ["chat-item-category-" + statusChatListItem.categoryId]
onEntered: reorderDelay.start()
onDropped: statusChatList.chatItemReordered(statusChatListItem.chatId, drag.source.originalOrder, statusChatListItem.DelegateModel.itemsIndex)
Timer {
id: reorderDelay
interval: 100
repeat: false
onTriggered: {
if (dropArea.containsDrag) {
dropArea.drag.source.chatListItem.originalOrder = statusChatListItem.originalOrder
delegateModel.items.move(dropArea.drag.source.DelegateModel.itemsIndex, draggable.DelegateModel.itemsIndex)
}
}
}
}
Loader {
id: draggedListItemLoader
active: dragSensor.active
sourceComponent: StatusChatListItem {
property var globalPosition: statusChatList.getAbsolutePosition(draggable)
parent: QC.Overlay.overlay
sensor.cursorShape: dragSensor.cursorShape
Drag.active: dragSensor.active
Drag.hotSpot.x: width / 2
Drag.hotSpot.y: height / 2
Drag.keys: ["chat-item-category-" + categoryId]
Drag.source: draggable
Component.onCompleted: {
x = globalPosition.x
y = globalPosition.y
}
chatId: draggable.chatListItem.chatId
categoryId: draggable.chatListItem.categoryId
name: draggable.chatListItem.name
type: draggable.chatListItem.type
muted: draggable.chatListItem.muted
dragged: true
hasUnreadMessages: draggable.chatListItem.hasUnreadMessages
hasMention: draggable.chatListItem.hasMention
badge.value: draggable.chatListItem.badge.value
selected: draggable.chatListItem.selected
icon.color: draggable.chatListItem.icon.color
image.isIdenticon: draggable.chatListItem.image.isIdenticon
image.source: draggable.chatListItem.image.source
}
}
}
}
Repeater {
id: statusChatListItems
model: delegateModel
} }
Loader { Loader {

View File

@ -16,6 +16,7 @@ Item {
property alias chatList: statusChatList.chatListItems property alias chatList: statusChatList.chatListItems
property alias categoryList: statusChatListCategories property alias categoryList: statusChatListCategories
property alias sensor: sensor property alias sensor: sensor
property bool draggableItems: false
property Component categoryPopupMenu property Component categoryPopupMenu
property Component chatListPopupMenu property Component chatListPopupMenu
@ -23,6 +24,7 @@ Item {
signal chatItemSelected(string id) signal chatItemSelected(string id)
signal chatItemUnmuted(string id) signal chatItemUnmuted(string id)
signal chatItemReordered(string categoryId, string chatId, int from, int to)
signal categoryAddButtonClicked(string id) signal categoryAddButtonClicked(string id)
onPopupMenuChanged: { onPopupMenuChanged: {
@ -54,14 +56,17 @@ Item {
StatusChatList { StatusChatList {
id: statusChatList id: statusChatList
anchors.horizontalCenter: parent.horizontalCenter anchors.horizontalCenter: parent.horizontalCenter
visible: !!chatListItems.model && chatListItems.count > 0 visible: chatListItems.count > 0
selectedChatId: statusChatListAndCategories.selectedChatId selectedChatId: statusChatListAndCategories.selectedChatId
onChatItemSelected: statusChatListAndCategories.chatItemSelected(id) onChatItemSelected: statusChatListAndCategories.chatItemSelected(id)
onChatItemUnmuted: statusChatListAndCategories.chatItemUnmuted(id) onChatItemUnmuted: statusChatListAndCategories.chatItemUnmuted(id)
onChatItemReordered: statusChatListAndCategories.chatItemReordered(categoryId, id, from, to)
draggableItems: statusChatListAndCategories.draggableItems
filterFn: function (model) { filterFn: function (model) {
return !!!model.categoryId return !!!model.categoryId
} }
popupMenu: statusChatListAndCategories.chatListPopupMenu popupMenu: statusChatListAndCategories.chatListPopupMenu
} }
Repeater { Repeater {
@ -78,6 +83,8 @@ Item {
chatList.selectedChatId: statusChatListAndCategories.selectedChatId chatList.selectedChatId: statusChatListAndCategories.selectedChatId
chatList.onChatItemSelected: statusChatListAndCategories.chatItemSelected(id) chatList.onChatItemSelected: statusChatListAndCategories.chatItemSelected(id)
chatList.onChatItemUnmuted: statusChatListAndCategories.chatItemUnmuted(id) chatList.onChatItemUnmuted: statusChatListAndCategories.chatItemUnmuted(id)
chatList.onChatItemReordered: statusChatListAndCategories.chatItemReordered(model.categoryId, id, from, to)
chatList.draggableItems: statusChatListAndCategories.draggableItems
popupMenu: statusChatListAndCategories.categoryPopupMenu popupMenu: statusChatListAndCategories.categoryPopupMenu
chatListPopupMenu: statusChatListAndCategories.chatListPopupMenu chatListPopupMenu: statusChatListAndCategories.chatListPopupMenu

View File

@ -1,4 +1,8 @@
import QtQuick 2.13 import QtQuick 2.13
import QtQml.Models 2.13
import QtQuick.Controls 2.13 as QC
import QtGraphicalEffects 1.13
import StatusQ.Core 0.1 import StatusQ.Core 0.1
import StatusQ.Core.Theme 0.1 import StatusQ.Core.Theme 0.1
import StatusQ.Components 0.1 import StatusQ.Components 0.1
@ -7,7 +11,10 @@ import StatusQ.Controls 0.1
Rectangle { Rectangle {
id: statusChatListItem id: statusChatListItem
objectName: "chatItem"
property int originalOrder: -1
property string chatId: "" property string chatId: ""
property string categoryId: ""
property string name: "" property string name: ""
property alias badge: statusBadge property alias badge: statusBadge
property bool hasUnreadMessages: false property bool hasUnreadMessages: false
@ -20,6 +27,8 @@ Rectangle {
property int type: StatusChatListItem.Type.PublicChat property int type: StatusChatListItem.Type.PublicChat
property bool highlighted: false property bool highlighted: false
property bool selected: false property bool selected: false
property bool dragged: false
property alias sensor: sensor
signal clicked(var mouse) signal clicked(var mouse)
signal unmute() signal unmute()
@ -46,6 +55,8 @@ Rectangle {
return sensor.containsMouse || highlighted ? Theme.palette.statusChatListItem.hoverBackgroundColor : Theme.palette.baseColor4 return sensor.containsMouse || highlighted ? Theme.palette.statusChatListItem.hoverBackgroundColor : Theme.palette.baseColor4
} }
opacity: dragged ? 0.7 : 1
MouseArea { MouseArea {
id: sensor id: sensor
@ -148,10 +159,10 @@ Rectangle {
anchors.rightMargin: 6 anchors.rightMargin: 6
anchors.verticalCenter: parent.verticalCenter anchors.verticalCenter: parent.verticalCenter
text: statusChatListItem.type === StatusChatListItem.Type.PublicChat && text: (statusChatListItem.type === StatusChatListItem.Type.PublicChat &&
!statusChatListItem.name.startsWith("#") ? !statusChatListItem.name.startsWith("#") ?
"#" + statusChatListItem.name : "#" + statusChatListItem.name :
statusChatListItem.name statusChatListItem.name)
elide: Text.ElideRight elide: Text.ElideRight
color: { color: {
if (statusChatListItem.muted && !sensor.containsMouse && !statusChatListItem.highlighted) { if (statusChatListItem.muted && !sensor.containsMouse && !statusChatListItem.highlighted) {
@ -207,6 +218,5 @@ Rectangle {
border.color: color border.color: color
visible: statusBadge.value > 0 visible: statusBadge.value > 0
} }
} }
} }