Devtools: Split state keys into a separate list
This commit is contained in:
@@ -163,6 +163,8 @@ add_library(neochat STATIC
|
|||||||
models/sortfilterroomtreemodel.h
|
models/sortfilterroomtreemodel.h
|
||||||
mediamanager.cpp
|
mediamanager.cpp
|
||||||
mediamanager.h
|
mediamanager.h
|
||||||
|
models/statekeysmodel.cpp
|
||||||
|
models/statekeysmodel.h
|
||||||
)
|
)
|
||||||
|
|
||||||
qt_add_qml_module(neochat URI org.kde.neochat NO_PLUGIN
|
qt_add_qml_module(neochat URI org.kde.neochat NO_PLUGIN
|
||||||
@@ -327,6 +329,7 @@ qt_add_qml_module(neochat URI org.kde.neochat NO_PLUGIN
|
|||||||
qml/FeatureFlagPage.qml
|
qml/FeatureFlagPage.qml
|
||||||
qml/IgnoredUsersDialog.qml
|
qml/IgnoredUsersDialog.qml
|
||||||
qml/AccountData.qml
|
qml/AccountData.qml
|
||||||
|
qml/StateKeys.qml
|
||||||
RESOURCES
|
RESOURCES
|
||||||
qml/confetti.png
|
qml/confetti.png
|
||||||
qml/glowdot.png
|
qml/glowdot.png
|
||||||
|
|||||||
85
src/models/statekeysmodel.cpp
Normal file
85
src/models/statekeysmodel.cpp
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2024 Tobias Fella <tobias.fella@kde.org>
|
||||||
|
// SPDX-License-Identifier: LGPL-2.0-or-later
|
||||||
|
|
||||||
|
#include "statekeysmodel.h"
|
||||||
|
|
||||||
|
StateKeysModel::StateKeysModel(QObject *parent)
|
||||||
|
: QAbstractListModel(parent)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
QHash<int, QByteArray> StateKeysModel::roleNames() const
|
||||||
|
{
|
||||||
|
return {
|
||||||
|
{StateKeyRole, "stateKey"},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
QVariant StateKeysModel::data(const QModelIndex &index, int role) const
|
||||||
|
{
|
||||||
|
Q_ASSERT(checkIndex(index, QAbstractItemModel::CheckIndexOption::IndexIsValid));
|
||||||
|
const auto row = index.row();
|
||||||
|
switch (role) {
|
||||||
|
case StateKeyRole:
|
||||||
|
return m_stateKeys[row]->stateKey();
|
||||||
|
}
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
int StateKeysModel::rowCount(const QModelIndex &parent) const
|
||||||
|
{
|
||||||
|
Q_UNUSED(parent);
|
||||||
|
return m_stateKeys.count();
|
||||||
|
}
|
||||||
|
|
||||||
|
NeoChatRoom *StateKeysModel::room() const
|
||||||
|
{
|
||||||
|
return m_room;
|
||||||
|
}
|
||||||
|
|
||||||
|
void StateKeysModel::loadState()
|
||||||
|
{
|
||||||
|
if (!m_room || m_eventType.isEmpty()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
beginResetModel();
|
||||||
|
m_stateKeys = m_room->currentState().eventsOfType(m_eventType);
|
||||||
|
endResetModel();
|
||||||
|
}
|
||||||
|
|
||||||
|
void StateKeysModel::setRoom(NeoChatRoom *room)
|
||||||
|
{
|
||||||
|
if (m_room) {
|
||||||
|
disconnect(m_room, nullptr, this, nullptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
m_room = room;
|
||||||
|
Q_EMIT roomChanged();
|
||||||
|
loadState();
|
||||||
|
|
||||||
|
connect(room, &NeoChatRoom::changed, this, [this] {
|
||||||
|
loadState();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
QString StateKeysModel::eventType() const
|
||||||
|
{
|
||||||
|
return m_eventType;
|
||||||
|
}
|
||||||
|
|
||||||
|
void StateKeysModel::setEventType(const QString &eventType)
|
||||||
|
{
|
||||||
|
m_eventType = eventType;
|
||||||
|
Q_EMIT eventTypeChanged();
|
||||||
|
loadState();
|
||||||
|
}
|
||||||
|
|
||||||
|
QByteArray StateKeysModel::stateEventJson(const QModelIndex &index)
|
||||||
|
{
|
||||||
|
const auto row = index.row();
|
||||||
|
const auto event = m_stateKeys[row];
|
||||||
|
const auto json = event->fullJson();
|
||||||
|
return QJsonDocument(json).toJson();
|
||||||
|
}
|
||||||
|
|
||||||
|
#include "moc_statekeysmodel.cpp"
|
||||||
83
src/models/statekeysmodel.h
Normal file
83
src/models/statekeysmodel.h
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2024 Tobias Fella <tobias.fella@kde.org>
|
||||||
|
// SPDX-License-Identifier: LGPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <QAbstractListModel>
|
||||||
|
#include <QQmlEngine>
|
||||||
|
|
||||||
|
#include "neochatroom.h"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @class StateKeysModel
|
||||||
|
*
|
||||||
|
* This class defines the model for visualising the state keys for a certain type in a room.
|
||||||
|
*/
|
||||||
|
class StateKeysModel : public QAbstractListModel
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
QML_ELEMENT
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief The current room that the model is getting its state events from.
|
||||||
|
*/
|
||||||
|
Q_PROPERTY(NeoChatRoom *room READ room WRITE setRoom NOTIFY roomChanged REQUIRED)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief The event type to list the stateKeys for
|
||||||
|
*/
|
||||||
|
Q_PROPERTY(QString eventType READ eventType WRITE setEventType NOTIFY eventTypeChanged REQUIRED)
|
||||||
|
|
||||||
|
public:
|
||||||
|
/**
|
||||||
|
* @brief Defines the model roles.
|
||||||
|
*/
|
||||||
|
enum Roles {
|
||||||
|
StateKeyRole, /**< The state key of the state event. */
|
||||||
|
};
|
||||||
|
Q_ENUM(Roles)
|
||||||
|
|
||||||
|
explicit StateKeysModel(QObject *parent = nullptr);
|
||||||
|
|
||||||
|
NeoChatRoom *room() const;
|
||||||
|
void setRoom(NeoChatRoom *room);
|
||||||
|
|
||||||
|
QString eventType() const;
|
||||||
|
void setEventType(const QString &eventType);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Get the given role value at the given index.
|
||||||
|
*
|
||||||
|
* @sa QAbstractItemModel::data
|
||||||
|
*/
|
||||||
|
QVariant data(const QModelIndex &index, int role) const override;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Number of rows in the model.
|
||||||
|
*
|
||||||
|
* @sa QAbstractItemModel::rowCount
|
||||||
|
*/
|
||||||
|
int rowCount(const QModelIndex &parent) const override;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Returns a mapping from Role enum values to role names.
|
||||||
|
*
|
||||||
|
* @sa Roles, QAbstractItemModel::roleNames()
|
||||||
|
*/
|
||||||
|
QHash<int, QByteArray> roleNames() const override;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Get the full JSON for an event.
|
||||||
|
*/
|
||||||
|
Q_INVOKABLE QByteArray stateEventJson(const QModelIndex &index);
|
||||||
|
|
||||||
|
Q_SIGNALS:
|
||||||
|
void roomChanged();
|
||||||
|
void eventTypeChanged();
|
||||||
|
|
||||||
|
private:
|
||||||
|
NeoChatRoom *m_room = nullptr;
|
||||||
|
QString m_eventType;
|
||||||
|
QVector<const Quotient::StateEvent *> m_stateKeys;
|
||||||
|
void loadState();
|
||||||
|
};
|
||||||
@@ -10,16 +10,16 @@ StateModel::StateModel(QObject *parent)
|
|||||||
|
|
||||||
QHash<int, QByteArray> StateModel::roleNames() const
|
QHash<int, QByteArray> StateModel::roleNames() const
|
||||||
{
|
{
|
||||||
return {{TypeRole, "type"}, {StateKeyRole, "stateKey"}};
|
return {{TypeRole, "type"}, {EventCountRole, "eventCount"}};
|
||||||
}
|
}
|
||||||
QVariant StateModel::data(const QModelIndex &index, int role) const
|
QVariant StateModel::data(const QModelIndex &index, int role) const
|
||||||
{
|
{
|
||||||
auto row = index.row();
|
auto row = index.row();
|
||||||
switch (role) {
|
switch (role) {
|
||||||
case TypeRole:
|
case TypeRole:
|
||||||
return m_stateEvents[row].first;
|
return m_stateEvents.keys()[row];
|
||||||
case StateKeyRole:
|
case EventCountRole:
|
||||||
return m_stateEvents[row].second;
|
return m_stateEvents.values()[row].count();
|
||||||
}
|
}
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
@@ -27,7 +27,7 @@ QVariant StateModel::data(const QModelIndex &index, int role) const
|
|||||||
int StateModel::rowCount(const QModelIndex &parent) const
|
int StateModel::rowCount(const QModelIndex &parent) const
|
||||||
{
|
{
|
||||||
Q_UNUSED(parent);
|
Q_UNUSED(parent);
|
||||||
return m_room->currentState().events().size();
|
return m_stateEvents.count();
|
||||||
}
|
}
|
||||||
|
|
||||||
NeoChatRoom *StateModel::room() const
|
NeoChatRoom *StateModel::room() const
|
||||||
@@ -35,26 +35,39 @@ NeoChatRoom *StateModel::room() const
|
|||||||
return m_room;
|
return m_room;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void StateModel::loadState()
|
||||||
|
{
|
||||||
|
beginResetModel();
|
||||||
|
m_stateEvents.clear();
|
||||||
|
const auto keys = m_room->currentState().events().keys();
|
||||||
|
for (const auto &[type, stateKey] : keys) {
|
||||||
|
if (!m_stateEvents.contains(type)) {
|
||||||
|
m_stateEvents[type] = {};
|
||||||
|
}
|
||||||
|
m_stateEvents[type] += stateKey;
|
||||||
|
}
|
||||||
|
endResetModel();
|
||||||
|
}
|
||||||
|
|
||||||
void StateModel::setRoom(NeoChatRoom *room)
|
void StateModel::setRoom(NeoChatRoom *room)
|
||||||
{
|
{
|
||||||
m_room = room;
|
m_room = room;
|
||||||
Q_EMIT roomChanged();
|
Q_EMIT roomChanged();
|
||||||
beginResetModel();
|
loadState();
|
||||||
m_stateEvents.clear();
|
|
||||||
m_stateEvents = m_room->currentState().events().keys();
|
|
||||||
endResetModel();
|
|
||||||
connect(room, &NeoChatRoom::changed, this, [this] {
|
connect(room, &NeoChatRoom::changed, this, [this] {
|
||||||
beginResetModel();
|
loadState();
|
||||||
m_stateEvents.clear();
|
|
||||||
m_stateEvents = m_room->currentState().events().keys();
|
|
||||||
endResetModel();
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
QByteArray StateModel::stateEventJson(const QModelIndex &index)
|
QByteArray StateModel::stateEventJson(const QModelIndex &index)
|
||||||
{
|
{
|
||||||
auto row = index.row();
|
auto row = index.row();
|
||||||
return QJsonDocument(m_room->currentState().events()[m_stateEvents[row]]->fullJson()).toJson();
|
const auto type = m_stateEvents.keys()[row];
|
||||||
|
const auto stateKey = m_stateEvents.values()[row][0];
|
||||||
|
const auto event = m_room->currentState().get(type, stateKey);
|
||||||
|
|
||||||
|
return QJsonDocument(event->fullJson()).toJson();
|
||||||
}
|
}
|
||||||
|
|
||||||
#include "moc_statemodel.cpp"
|
#include "moc_statemodel.cpp"
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ public:
|
|||||||
*/
|
*/
|
||||||
enum Roles {
|
enum Roles {
|
||||||
TypeRole = 0, /**< The type of the state event. */
|
TypeRole = 0, /**< The type of the state event. */
|
||||||
StateKeyRole, /**< The state key of the state event. */
|
EventCountRole, /**< Number of events of this type. */
|
||||||
};
|
};
|
||||||
Q_ENUM(Roles)
|
Q_ENUM(Roles)
|
||||||
|
|
||||||
@@ -58,11 +58,9 @@ public:
|
|||||||
* @sa Roles, QAbstractItemModel::roleNames()
|
* @sa Roles, QAbstractItemModel::roleNames()
|
||||||
*/
|
*/
|
||||||
QHash<int, QByteArray> roleNames() const override;
|
QHash<int, QByteArray> roleNames() const override;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Get the full JSON for an event.
|
* @brief Get the full JSON for an event.
|
||||||
*
|
|
||||||
* This is used to avoid having the model hold all the JSON data. The JSON for
|
|
||||||
* a single item is only ever shown, no need to access simultaneously.
|
|
||||||
*/
|
*/
|
||||||
Q_INVOKABLE QByteArray stateEventJson(const QModelIndex &index);
|
Q_INVOKABLE QByteArray stateEventJson(const QModelIndex &index);
|
||||||
|
|
||||||
@@ -73,10 +71,8 @@ private:
|
|||||||
NeoChatRoom *m_room = nullptr;
|
NeoChatRoom *m_room = nullptr;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief The room state events in a QList.
|
* @brief A map from state event type to number of events of that type
|
||||||
*
|
|
||||||
* This is done for performance, accessing all the data from the parent QHash
|
|
||||||
* was slower.
|
|
||||||
*/
|
*/
|
||||||
QList<std::pair<QString, QString>> m_stateEvents;
|
QMap<QString, QList<QString>> m_stateEvents;
|
||||||
|
void loadState();
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -36,38 +36,20 @@ ColumnLayout {
|
|||||||
FormCard.FormTextDelegate {
|
FormCard.FormTextDelegate {
|
||||||
text: i18n("Room Id: %1", root.room.id)
|
text: i18n("Room Id: %1", root.room.id)
|
||||||
}
|
}
|
||||||
FormCard.FormCheckDelegate {
|
|
||||||
text: i18n("Show m.room.member events")
|
|
||||||
checked: true
|
|
||||||
onToggled: {
|
|
||||||
if (checked) {
|
|
||||||
stateEventFilterModel.removeStateEventTypeFiltered("m.room.member");
|
|
||||||
} else {
|
|
||||||
stateEventFilterModel.addStateEventTypeFiltered("m.room.member");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
FormCard.FormCheckDelegate {
|
|
||||||
id: roomAccountDataVisibleCheck
|
|
||||||
text: i18n("Show room account data")
|
|
||||||
checked: false
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
FormCard.FormHeader {
|
FormCard.FormHeader {
|
||||||
visible: roomAccountDataVisibleCheck.checked
|
|
||||||
title: i18n("Room Account Data")
|
title: i18n("Room Account Data")
|
||||||
}
|
}
|
||||||
FormCard.FormCard {
|
FormCard.FormCard {
|
||||||
visible: roomAccountDataVisibleCheck.checked
|
|
||||||
Repeater {
|
Repeater {
|
||||||
model: root.room.accountDataEventTypes
|
model: root.room.accountDataEventTypes
|
||||||
delegate: FormCard.FormButtonDelegate {
|
delegate: FormCard.FormButtonDelegate {
|
||||||
text: modelData
|
text: modelData
|
||||||
onClicked: applicationWindow().pageStack.pushDialogLayer(Qt.createComponent('org.kde.neochat', 'MessageSourceSheet.qml'), {
|
onClicked: applicationWindow().pageStack.pushDialogLayer(Qt.createComponent('org.kde.neochat', 'MessageSourceSheet.qml'), {
|
||||||
"sourceText": root.room.roomAcountDataJson(text)
|
sourceText: root.room.roomAcountDataJson(text)
|
||||||
}, {
|
}, {
|
||||||
"title": i18n("Event Source"),
|
title: i18n("Event Source"),
|
||||||
"width": Kirigami.Units.gridUnit * 25
|
width: Kirigami.Units.gridUnit * 25
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -78,24 +60,36 @@ ColumnLayout {
|
|||||||
}
|
}
|
||||||
FormCard.FormCard {
|
FormCard.FormCard {
|
||||||
Repeater {
|
Repeater {
|
||||||
model: StateFilterModel {
|
model: StateModel {
|
||||||
id: stateEventFilterModel
|
id: stateModel
|
||||||
sourceModel: StateModel {
|
room: root.room
|
||||||
id: stateModel
|
|
||||||
room: root.room
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
delegate: FormCard.FormButtonDelegate {
|
delegate: FormCard.FormButtonDelegate {
|
||||||
text: model.type
|
text: model.type
|
||||||
description: model.stateKey
|
description: i18ncp("'Event' being some JSON data, not something physically happening.", "%1 event of this type", "%1 events of this type", model.eventCount)
|
||||||
onClicked: applicationWindow().pageStack.pushDialogLayer(Qt.createComponent('org.kde.neochat', 'MessageSourceSheet.qml'), {
|
onClicked: {
|
||||||
sourceText: stateModel.stateEventJson(stateEventFilterModel.mapToSource(stateEventFilterModel.index(model.index, 0)))
|
if (model.eventCount === 1) {
|
||||||
}, {
|
onClicked: applicationWindow().pageStack.pushDialogLayer(Qt.createComponent('org.kde.neochat', 'MessageSourceSheet.qml'), {
|
||||||
title: i18n("Event Source"),
|
sourceText: stateModel.stateEventJson(stateModel.index(model.index, 0))
|
||||||
width: Kirigami.Units.gridUnit * 25
|
}, {
|
||||||
})
|
title: i18n("Event Source"),
|
||||||
|
width: Kirigami.Units.gridUnit * 25
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
pageStack.pushDialogLayer(stateKeysComponent, {
|
||||||
|
room: root.room,
|
||||||
|
eventType: model.type
|
||||||
|
}, {
|
||||||
|
title: i18nc("'Event' being some JSON data, not something physically happening.", "Event Information")
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Component {
|
||||||
|
id: stateKeysComponent
|
||||||
|
StateKeys {}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
43
src/qml/StateKeys.qml
Normal file
43
src/qml/StateKeys.qml
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2024 Tobias Fella <tobias.fella@kde.org>
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL
|
||||||
|
|
||||||
|
import QtQuick
|
||||||
|
import QtQuick.Layouts
|
||||||
|
|
||||||
|
import org.kde.kirigami as Kirigami
|
||||||
|
import org.kde.kirigamiaddons.formcard as FormCard
|
||||||
|
import org.kde.kitemmodels
|
||||||
|
|
||||||
|
import org.kde.neochat
|
||||||
|
|
||||||
|
FormCard.FormCardPage {
|
||||||
|
id: root
|
||||||
|
|
||||||
|
required property NeoChatRoom room
|
||||||
|
required property string eventType
|
||||||
|
|
||||||
|
title: root.eventType
|
||||||
|
|
||||||
|
FormCard.FormHeader {
|
||||||
|
title: i18nc("The name of one instance of a state of configuration. Unless you really know what you're doing, best leave this untranslated.", "State Keys")
|
||||||
|
}
|
||||||
|
FormCard.FormCard {
|
||||||
|
Repeater {
|
||||||
|
model: StateKeysModel {
|
||||||
|
id: stateKeysModel
|
||||||
|
room: root.room
|
||||||
|
eventType: root.eventType
|
||||||
|
}
|
||||||
|
|
||||||
|
delegate: FormCard.FormButtonDelegate {
|
||||||
|
text: model.stateKey
|
||||||
|
onClicked: applicationWindow().pageStack.pushDialogLayer('qrc:/org/kde/neochat/qml/MessageSourceSheet.qml', {
|
||||||
|
sourceText: stateKeysModel.stateEventJson(stateKeysModel.index(model.index, 0))
|
||||||
|
}, {
|
||||||
|
title: i18nc("@title:window", "Event Source"),
|
||||||
|
width: Kirigami.Units.gridUnit * 25
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user