feat(StatusTagSelector): Introduced readonly tags and possibility of icon
- Different tag color depending if the entry `isReadonly` or not. - Tag disabled if the entry `isReadonly`. - Possibility to show icon if entry contains the proper info. - Added property `orderByReadonly` to position `isReadonly` entries in the left. Updated sandbox project examples according to new tag updates. Closes #694
This commit is contained in:
parent
18d385cf2b
commit
70834fc187
|
@ -83,6 +83,8 @@ Column {
|
||||||
icon: ""
|
icon: ""
|
||||||
isIdenticon: false
|
isIdenticon: false
|
||||||
onlineStatus: 3
|
onlineStatus: 3
|
||||||
|
isReadonly: true
|
||||||
|
tagIcon: "crown"
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
publicId: "0x1"
|
publicId: "0x1"
|
||||||
|
@ -90,6 +92,8 @@ Column {
|
||||||
icon: ""
|
icon: ""
|
||||||
isIdenticon: false
|
isIdenticon: false
|
||||||
onlineStatus: 1
|
onlineStatus: 1
|
||||||
|
isReadonly: false
|
||||||
|
tagIcon: ""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
toLabelText: qsTr("To: ")
|
toLabelText: qsTr("To: ")
|
||||||
|
|
|
@ -852,6 +852,7 @@ CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2I
|
||||||
icon: "
|
icon: "
|
||||||
nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2ImYgiNITTlTdG1nUZ5a92VITQxITFiJmIIjSE0htAYQrMHAAD//+wwFVpz+yqXAAAAAElFTkSuQmCC"
|
nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2ImYgiNITTlTdG1nUZ5a92VITQxITFiJmIIjSE0htAYQrMHAAD//+wwFVpz+yqXAAAAAElFTkSuQmCC"
|
||||||
isIdenticon: true
|
isIdenticon: true
|
||||||
|
isAdmin: false
|
||||||
ringSpecModel: [ ListElement {colorId: 13; segmentLength: 5},
|
ringSpecModel: [ ListElement {colorId: 13; segmentLength: 5},
|
||||||
ListElement {colorId: 31; segmentLength: 5},
|
ListElement {colorId: 31; segmentLength: 5},
|
||||||
ListElement {colorId: 10; segmentLength: 1},
|
ListElement {colorId: 10; segmentLength: 1},
|
||||||
|
@ -868,6 +869,7 @@ CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2I
|
||||||
isMutualContact: false
|
isMutualContact: false
|
||||||
onlineStatus: false
|
onlineStatus: false
|
||||||
icon: ""
|
icon: ""
|
||||||
|
isAdmin: false
|
||||||
isIdenticon: false
|
isIdenticon: false
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
|
@ -877,6 +879,7 @@ CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2I
|
||||||
trustIndicator: StatusContactVerificationIcons.TrustedType.None
|
trustIndicator: StatusContactVerificationIcons.TrustedType.None
|
||||||
isMutualContact: false
|
isMutualContact: false
|
||||||
onlineStatus: true
|
onlineStatus: true
|
||||||
|
isAdmin: false
|
||||||
icon: "
|
icon: "
|
||||||
ExhKZ4a9Uq3TZviZmIITSG0DRvlqcbqVbrlouZiCE0htD4h0hjCI0hNN5aNIbQGKKPxEzEEBpDaAyhMYTmDAAA//+gYCErzmCpCQAAAABJRU5ErkJggg=="
|
ExhKZ4a9Uq3TZviZmIITSG0DRvlqcbqVbrlouZiCE0htD4h0hjCI0hNN5aNIbQGKKPxEzEEBpDaAyhMYTmDAAA//+gYCErzmCpCQAAAABJRU5ErkJggg=="
|
||||||
isIdenticon: true
|
isIdenticon: true
|
||||||
|
@ -897,6 +900,7 @@ CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2I
|
||||||
onlineStatus: false
|
onlineStatus: false
|
||||||
icon: ""
|
icon: ""
|
||||||
isIdenticon: false
|
isIdenticon: false
|
||||||
|
isAdmin: false
|
||||||
ringSpecModel: [ ListElement {colorId: 0; segmentLength: 1},
|
ringSpecModel: [ ListElement {colorId: 0; segmentLength: 1},
|
||||||
ListElement {colorId: 28; segmentLength: 1},
|
ListElement {colorId: 28; segmentLength: 1},
|
||||||
ListElement {colorId: 31; segmentLength: 1},
|
ListElement {colorId: 31; segmentLength: 1},
|
||||||
|
|
|
@ -9,17 +9,21 @@ Item {
|
||||||
property ListModel asortedContacts: ListModel {
|
property ListModel asortedContacts: ListModel {
|
||||||
ListElement {
|
ListElement {
|
||||||
publicId: "0x0"
|
publicId: "0x0"
|
||||||
name: "Maria"
|
name: "emily.eth"
|
||||||
icon: ""
|
icon: ""
|
||||||
isIdenticon: false
|
isIdenticon: false
|
||||||
onlineStatus: 3
|
onlineStatus: 3
|
||||||
|
isReadonly: false
|
||||||
|
tagIcon: ""
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
publicId: "0x1"
|
publicId: "0x1"
|
||||||
name: "James"
|
name: "vitalikbuterin"
|
||||||
icon: "https://pbs.twimg.com/profile_images/1369221718338895873/T_5fny6o_400x400.jpg"
|
icon: "https://pbs.twimg.com/profile_images/1369221718338895873/T_5fny6o_400x400.jpg"
|
||||||
isIdenticon: false
|
isIdenticon: false
|
||||||
onlineStatus: 1
|
onlineStatus: 1
|
||||||
|
isReadonly: false
|
||||||
|
tagIcon: ""
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
publicId: "0x2"
|
publicId: "0x2"
|
||||||
|
@ -27,13 +31,17 @@ Item {
|
||||||
icon: ""
|
icon: ""
|
||||||
isIdenticon: false
|
isIdenticon: false
|
||||||
onlineStatus: 2
|
onlineStatus: 2
|
||||||
|
isReadonly: false
|
||||||
|
tagIcon: ""
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
publicId: "0x3"
|
publicId: "0x3"
|
||||||
name: "Tracy"
|
name: "carmen.eth"
|
||||||
icon: ""
|
icon: ""
|
||||||
isIdenticon: true
|
isIdenticon: true
|
||||||
onlineStatus: 3
|
onlineStatus: 3
|
||||||
|
isReadonly: true
|
||||||
|
tagIcon: "crown"
|
||||||
}
|
}
|
||||||
ListElement {
|
ListElement {
|
||||||
publicId: "0x4"
|
publicId: "0x4"
|
||||||
|
@ -41,6 +49,8 @@ Item {
|
||||||
icon: ""
|
icon: ""
|
||||||
isIdenticon: false
|
isIdenticon: false
|
||||||
onlineStatus: 3
|
onlineStatus: 3
|
||||||
|
isReadonly: false
|
||||||
|
tagIcon: ""
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -132,6 +132,12 @@ Item {
|
||||||
By default is set to false.
|
By default is set to false.
|
||||||
*/
|
*/
|
||||||
property bool showSortedListOnlyWhenText: false
|
property bool showSortedListOnlyWhenText: false
|
||||||
|
/*!
|
||||||
|
\qmlproperty bool StatusTagSelector::orderByReadonly
|
||||||
|
This property will decide if the displayed tag names will be ordered from left to right starting with the items that are marked as `isReadonly`.
|
||||||
|
By default is set to true (readonly names at left).
|
||||||
|
*/
|
||||||
|
property bool orderByReadonly: true
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
\qmlmethod
|
\qmlmethod
|
||||||
|
@ -146,9 +152,12 @@ Item {
|
||||||
\qmlmethod
|
\qmlmethod
|
||||||
This function is used to insert a new tag.
|
This function is used to insert a new tag.
|
||||||
*/
|
*/
|
||||||
function insertTag(name, id) {
|
function insertTag(name, id, isReadonly, tagIcon) {
|
||||||
if (!find(namesModel, function(item) { return item.publicId === id }) && namesModel.count < root.nameCountLimit) {
|
if (!find(namesModel, function(item) { return item.publicId === id }) && namesModel.count < root.nameCountLimit) {
|
||||||
namesModel.insert(namesModel.count, {"name": name, "publicId": id});
|
if(orderByReadonly && isReadonly)
|
||||||
|
namesModel.insert(0, {"name": name, "publicId": id, "isReadonly": !!isReadonly, "tagIcon": tagIcon ? tagIcon : ""});
|
||||||
|
else
|
||||||
|
namesModel.insert(namesModel.count, {"name": name, "publicId": id, "isReadonly": !!isReadonly, "tagIcon": tagIcon ? tagIcon : ""});
|
||||||
addMember(id);
|
addMember(id);
|
||||||
edit.clear();
|
edit.clear();
|
||||||
}
|
}
|
||||||
|
@ -165,11 +174,17 @@ Item {
|
||||||
var entry = inputModel.get(i);
|
var entry = inputModel.get(i);
|
||||||
if (entry.name.toLowerCase().includes(text.toLowerCase()) &&
|
if (entry.name.toLowerCase().includes(text.toLowerCase()) &&
|
||||||
!find(namesModel, function(item) { return item.name === entry.name })) {
|
!find(namesModel, function(item) { return item.name === entry.name })) {
|
||||||
sortedList.append({"publicId": entry.publicId, "name": entry.name,
|
sortedList.append({"publicId": entry.publicId,
|
||||||
"nickName": entry.nickName, "trustIndicator": entry.trustIndicator,
|
"name": entry.name,
|
||||||
"isMutualContact": entry.isMutualContact, "ringSpecModel": entry.ringSpecModel,
|
"nickName": entry.nickName,
|
||||||
"icon": entry.icon, "isIdenticon": entry.isIdenticon,
|
"trustIndicator": entry.trustIndicator,
|
||||||
"onlineStatus": entry.onlineStatus});
|
"isMutualContact": entry.isMutualContact,
|
||||||
|
"ringSpecModel": entry.ringSpecModel,
|
||||||
|
"icon": entry.icon,
|
||||||
|
"isIdenticon": entry.isIdenticon,
|
||||||
|
"onlineStatus": entry.onlineStatus,
|
||||||
|
"tagIcon": entry.tagIcon ? entry.tagIcon : "",
|
||||||
|
"isReadonly": !!entry.isReadonly});
|
||||||
userListView.model = sortedList;
|
userListView.model = sortedList;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -198,11 +213,28 @@ Item {
|
||||||
2 * bgRect.anchors.margins +
|
2 * bgRect.anchors.margins +
|
||||||
userListView.anchors.topMargin + userListView.anchors.bottomMargin +
|
userListView.anchors.topMargin + userListView.anchors.bottomMargin +
|
||||||
(userListView.model.count * 64): 0
|
(userListView.model.count * 64): 0
|
||||||
|
|
||||||
|
function orderNamesModel() {
|
||||||
|
if(root.orderByReadonly) {
|
||||||
|
for(var i = 0; i < namesModel.count; i++) {
|
||||||
|
var entry = namesModel.get(i)
|
||||||
|
if(entry.isReadonly) {
|
||||||
|
namesModel.move(i, 0, 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
implicitWidth: 448
|
implicitWidth: 448
|
||||||
implicitHeight: (tagSelectorRect.height + d.suggestionContainerHeight) > root.maxHeight ? root.maxHeight : (tagSelectorRect.height + d.suggestionContainerHeight)
|
implicitHeight: (tagSelectorRect.height + d.suggestionContainerHeight) > root.maxHeight ? root.maxHeight : (tagSelectorRect.height + d.suggestionContainerHeight)
|
||||||
|
|
||||||
|
onOrderByReadonlyChanged: { d.orderNamesModel() }
|
||||||
|
Component.onCompleted: {
|
||||||
|
// Component initialization:
|
||||||
|
d.orderNamesModel()
|
||||||
|
}
|
||||||
|
|
||||||
Rectangle {
|
Rectangle {
|
||||||
id: tagSelectorRect
|
id: tagSelectorRect
|
||||||
width: parent.width
|
width: parent.width
|
||||||
|
@ -247,29 +279,63 @@ Item {
|
||||||
onCountChanged: { scrollToEnd(); }
|
onCountChanged: { scrollToEnd(); }
|
||||||
delegate: Rectangle {
|
delegate: Rectangle {
|
||||||
id: nameDelegate
|
id: nameDelegate
|
||||||
width: (nameText.contentWidth + 34)
|
|
||||||
|
property int tagMargins: 8
|
||||||
|
property int tagIconsSize: 20
|
||||||
|
function getTagColor(isReadonly) {
|
||||||
|
if(isReadonly) {
|
||||||
|
return Theme.palette.baseColor1
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return mouseArea.containsMouse ? Theme.palette.miscColor1 : Theme.palette.primaryColor1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
width: tagRow.implicitWidth + 2 * nameDelegate.tagMargins
|
||||||
height: 30
|
height: 30
|
||||||
color: mouseArea.containsMouse ? Theme.palette.miscColor1 : Theme.palette.primaryColor1
|
color: getTagColor(model.isReadonly)
|
||||||
radius: 8
|
radius: 8
|
||||||
StatusBaseText {
|
Row {
|
||||||
id: nameText
|
id: tagRow
|
||||||
|
height: parent.height
|
||||||
anchors.left: parent.left
|
anchors.left: parent.left
|
||||||
anchors.leftMargin: 8
|
anchors.leftMargin: nameDelegate.tagMargins
|
||||||
|
anchors.rightMargin: nameDelegate.tagMargins
|
||||||
|
spacing: 2
|
||||||
|
|
||||||
|
StatusIcon {
|
||||||
|
visible: model.tagIcon
|
||||||
anchors.verticalCenter: parent.verticalCenter
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
color: Theme.palette.indirectColor1
|
color: Theme.palette.indirectColor1
|
||||||
|
width: model.tagIcon ? nameDelegate.tagIconsSize : 0
|
||||||
|
height: nameDelegate.tagIconsSize
|
||||||
|
icon: model.tagIcon
|
||||||
|
}
|
||||||
|
StatusBaseText {
|
||||||
|
id: nameText
|
||||||
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
|
color: Theme.palette.indirectColor1
|
||||||
|
font.pixelSize: 15
|
||||||
text: name
|
text: name
|
||||||
}
|
}
|
||||||
StatusIcon {
|
StatusIcon {
|
||||||
anchors.left: nameText.right
|
id: closeIcon
|
||||||
|
visible: !model.isReadonly
|
||||||
|
anchors.leftMargin: nameDelegate.tagMargins
|
||||||
anchors.verticalCenter: parent.verticalCenter
|
anchors.verticalCenter: parent.verticalCenter
|
||||||
color: Theme.palette.indirectColor1
|
color: Theme.palette.indirectColor1
|
||||||
|
width: nameDelegate.tagIconsSize
|
||||||
|
height: nameDelegate.tagIconsSize
|
||||||
icon: "close"
|
icon: "close"
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
MouseArea {
|
MouseArea {
|
||||||
id: mouseArea
|
id: mouseArea
|
||||||
|
enabled: !model.isReadonly
|
||||||
anchors.fill: parent
|
anchors.fill: parent
|
||||||
hoverEnabled: true
|
hoverEnabled: true
|
||||||
cursorShape: Qt.PointingHandCursor
|
cursorShape: enabled ? Qt.PointingHandCursor : Qt.ArrowCursor
|
||||||
onClicked: {
|
onClicked: {
|
||||||
removeMember(publicId);
|
removeMember(publicId);
|
||||||
namesModel.remove(index, 1);
|
namesModel.remove(index, 1);
|
||||||
|
@ -299,7 +365,10 @@ Item {
|
||||||
namesModel.remove((namesList.count-1), 1);
|
namesModel.remove((namesList.count-1), 1);
|
||||||
}
|
}
|
||||||
if ((event.key === Qt.Key_Return || event.key === Qt.Key_Enter) && (sortedList.count > 0)) {
|
if ((event.key === Qt.Key_Return || event.key === Qt.Key_Enter) && (sortedList.count > 0)) {
|
||||||
root.insertTag(sortedList.get(userListView.currentIndex).name, sortedList.get(userListView.currentIndex).publicId);
|
root.insertTag(sortedList.get(userListView.currentIndex).name,
|
||||||
|
sortedList.get(userListView.currentIndex).publicId,
|
||||||
|
sortedList.get(userListView.currentIndex).isReadonly,
|
||||||
|
sortedList.get(userListView.currentIndex).tagIcon);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Keys.onUpPressed: { userListView.decrementCurrentIndex(); }
|
Keys.onUpPressed: { userListView.decrementCurrentIndex(); }
|
||||||
|
@ -405,7 +474,7 @@ Item {
|
||||||
ringSettings.ringSpecModel: root.ringSpecModelGetter(publicId)
|
ringSettings.ringSpecModel: root.ringSpecModelGetter(publicId)
|
||||||
color: (sensor.containsMouse || highlighted) ? Theme.palette.baseColor2 : "transparent"
|
color: (sensor.containsMouse || highlighted) ? Theme.palette.baseColor2 : "transparent"
|
||||||
onClicked: {
|
onClicked: {
|
||||||
root.insertTag(model.name, model.publicId);
|
root.insertTag(model.name, model.publicId, model.isAdmin, model.isAdmin ? "crown" : "");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue