Show a map for location events
This commit is contained in:
committed by
Volker Krause
parent
d14db326bb
commit
5b4ae764cf
@@ -57,6 +57,7 @@ add_library(neochat STATIC
|
|||||||
models/reactionmodel.cpp
|
models/reactionmodel.cpp
|
||||||
delegatesizehelper.cpp
|
delegatesizehelper.cpp
|
||||||
models/livelocationsmodel.cpp
|
models/livelocationsmodel.cpp
|
||||||
|
models/locationsmodel.cpp
|
||||||
)
|
)
|
||||||
|
|
||||||
ecm_qt_declare_logging_category(neochat
|
ecm_qt_declare_logging_category(neochat
|
||||||
|
|||||||
@@ -58,6 +58,7 @@
|
|||||||
#include "models/imagepacksmodel.h"
|
#include "models/imagepacksmodel.h"
|
||||||
#include "models/keywordnotificationrulemodel.h"
|
#include "models/keywordnotificationrulemodel.h"
|
||||||
#include "models/livelocationsmodel.h"
|
#include "models/livelocationsmodel.h"
|
||||||
|
#include "models/locationsmodel.h"
|
||||||
#include "models/messageeventmodel.h"
|
#include "models/messageeventmodel.h"
|
||||||
#include "models/messagefiltermodel.h"
|
#include "models/messagefiltermodel.h"
|
||||||
#include "models/publicroomlistmodel.h"
|
#include "models/publicroomlistmodel.h"
|
||||||
@@ -249,6 +250,7 @@ int main(int argc, char *argv[])
|
|||||||
qmlRegisterType<StateFilterModel>("org.kde.neochat", 1, 0, "StateFilterModel");
|
qmlRegisterType<StateFilterModel>("org.kde.neochat", 1, 0, "StateFilterModel");
|
||||||
qmlRegisterType<SearchModel>("org.kde.neochat", 1, 0, "SearchModel");
|
qmlRegisterType<SearchModel>("org.kde.neochat", 1, 0, "SearchModel");
|
||||||
qmlRegisterType<LiveLocationsModel>("org.kde.neochat", 1, 0, "LiveLocationsModel");
|
qmlRegisterType<LiveLocationsModel>("org.kde.neochat", 1, 0, "LiveLocationsModel");
|
||||||
|
qmlRegisterType<LocationsModel>("org.kde.neochat", 1, 0, "LocationsModel");
|
||||||
#ifdef QUOTIENT_07
|
#ifdef QUOTIENT_07
|
||||||
qmlRegisterType<PollHandler>("org.kde.neochat", 1, 0, "PollHandler");
|
qmlRegisterType<PollHandler>("org.kde.neochat", 1, 0, "PollHandler");
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
102
src/models/locationsmodel.cpp
Normal file
102
src/models/locationsmodel.cpp
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2023 Tobias Fella <tobias.fella@kde.org>
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#include "locationsmodel.h"
|
||||||
|
|
||||||
|
using namespace Quotient;
|
||||||
|
|
||||||
|
LocationsModel::LocationsModel(QObject *parent)
|
||||||
|
: QAbstractListModel(parent)
|
||||||
|
{
|
||||||
|
connect(this, &LocationsModel::roomChanged, this, [=]() {
|
||||||
|
for (const auto &event : m_room->messageEvents()) {
|
||||||
|
if (!is<RoomMessageEvent>(*event)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (event->contentJson()["msgtype"] == "m.location") {
|
||||||
|
const auto &e = *event;
|
||||||
|
addLocation(eventCast<const RoomMessageEvent>(&e));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
connect(m_room, &NeoChatRoom::aboutToAddHistoricalMessages, this, [=](const auto &events) {
|
||||||
|
for (const auto &event : events) {
|
||||||
|
if (!is<RoomMessageEvent>(*event)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (event->contentJson()["msgtype"] == "m.location") {
|
||||||
|
const auto &e = *event;
|
||||||
|
addLocation(eventCast<const RoomMessageEvent>(&e));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
connect(m_room, &NeoChatRoom::aboutToAddNewMessages, this, [=](const auto &events) {
|
||||||
|
for (const auto &event : events) {
|
||||||
|
if (!is<RoomMessageEvent>(*event)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (event->contentJson()["msgtype"] == "m.location") {
|
||||||
|
const auto &e = *event;
|
||||||
|
addLocation(eventCast<const RoomMessageEvent>(&e));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
void LocationsModel::addLocation(const RoomMessageEvent *event)
|
||||||
|
{
|
||||||
|
const auto uri = event->contentJson()["org.matrix.msc3488.location"]["uri"].toString();
|
||||||
|
const auto parts = uri.mid(4).split(QLatin1Char(','));
|
||||||
|
const auto latitude = parts[0].toFloat();
|
||||||
|
const auto longitude = parts[1].toFloat();
|
||||||
|
beginInsertRows(QModelIndex(), m_locations.size(), m_locations.size() + 1);
|
||||||
|
m_locations += LocationData{
|
||||||
|
.eventId = event->id(),
|
||||||
|
.latitude = latitude,
|
||||||
|
.longitude = longitude,
|
||||||
|
.text = event->contentJson()["body"].toString(),
|
||||||
|
.author = dynamic_cast<NeoChatUser *>(m_room->user(event->senderId())),
|
||||||
|
};
|
||||||
|
endInsertRows();
|
||||||
|
}
|
||||||
|
|
||||||
|
NeoChatRoom *LocationsModel::room() const
|
||||||
|
{
|
||||||
|
return m_room;
|
||||||
|
}
|
||||||
|
|
||||||
|
void LocationsModel::setRoom(NeoChatRoom *room)
|
||||||
|
{
|
||||||
|
if (m_room) {
|
||||||
|
disconnect(this, nullptr, m_room, nullptr);
|
||||||
|
}
|
||||||
|
m_room = room;
|
||||||
|
Q_EMIT roomChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
QHash<int, QByteArray> LocationsModel::roleNames() const
|
||||||
|
{
|
||||||
|
return {
|
||||||
|
{LongitudeRole, "longitude"},
|
||||||
|
{LatitudeRole, "latitude"},
|
||||||
|
{TextRole, "text"},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
QVariant LocationsModel::data(const QModelIndex &index, int roleName) const
|
||||||
|
{
|
||||||
|
auto row = index.row();
|
||||||
|
if (roleName == LongitudeRole) {
|
||||||
|
return m_locations[row].longitude;
|
||||||
|
} else if (roleName == LatitudeRole) {
|
||||||
|
return m_locations[row].latitude;
|
||||||
|
} else if (roleName == TextRole) {
|
||||||
|
return m_locations[row].text;
|
||||||
|
}
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
int LocationsModel::rowCount(const QModelIndex &parent) const
|
||||||
|
{
|
||||||
|
return m_locations.size();
|
||||||
|
}
|
||||||
50
src/models/locationsmodel.h
Normal file
50
src/models/locationsmodel.h
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2023 Tobias Fella <tobias.fella@kde.org>
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <QAbstractListModel>
|
||||||
|
#include <QPointer>
|
||||||
|
|
||||||
|
#include "neochatroom.h"
|
||||||
|
|
||||||
|
#include <events/roommessageevent.h>
|
||||||
|
|
||||||
|
class LocationsModel : public QAbstractListModel
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
enum Roles {
|
||||||
|
TextRole = Qt::DisplayRole,
|
||||||
|
LongitudeRole,
|
||||||
|
LatitudeRole,
|
||||||
|
};
|
||||||
|
Q_ENUM(Roles)
|
||||||
|
Q_PROPERTY(NeoChatRoom *room READ room WRITE setRoom NOTIFY roomChanged)
|
||||||
|
|
||||||
|
explicit LocationsModel(QObject *parent = nullptr);
|
||||||
|
|
||||||
|
[[nodiscard]] NeoChatRoom *room() const;
|
||||||
|
void setRoom(NeoChatRoom *room);
|
||||||
|
|
||||||
|
[[nodiscard]] QHash<int, QByteArray> roleNames() const override;
|
||||||
|
[[nodiscard]] QVariant data(const QModelIndex &index, int roleName) const override;
|
||||||
|
[[nodiscard]] int rowCount(const QModelIndex &parent) const override;
|
||||||
|
|
||||||
|
Q_SIGNALS:
|
||||||
|
void roomChanged();
|
||||||
|
|
||||||
|
private:
|
||||||
|
QPointer<NeoChatRoom> m_room;
|
||||||
|
|
||||||
|
struct LocationData {
|
||||||
|
QString eventId;
|
||||||
|
float latitude;
|
||||||
|
float longitude;
|
||||||
|
QString text;
|
||||||
|
NeoChatUser *author;
|
||||||
|
};
|
||||||
|
QList<LocationData> m_locations;
|
||||||
|
void addLocation(const Quotient::RoomMessageEvent *event);
|
||||||
|
};
|
||||||
68
src/qml/Component/FullScreenMap.qml
Normal file
68
src/qml/Component/FullScreenMap.qml
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2021 Tobias Fella <fella@posteo.de>
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
import QtQuick 2.15
|
||||||
|
import QtQuick.Controls 2.15
|
||||||
|
import QtLocation 5.15
|
||||||
|
import QtPositioning 5.15
|
||||||
|
|
||||||
|
import org.kde.kirigami 2.15 as Kirigami
|
||||||
|
|
||||||
|
ApplicationWindow {
|
||||||
|
id: root
|
||||||
|
|
||||||
|
required property var content
|
||||||
|
|
||||||
|
flags: Qt.FramelessWindowHint | Qt.WA_TranslucentBackground
|
||||||
|
visibility: Qt.WindowFullScreen
|
||||||
|
|
||||||
|
title: i18n("View Location")
|
||||||
|
|
||||||
|
Shortcut {
|
||||||
|
sequence: "Escape"
|
||||||
|
onActivated: root.destroy()
|
||||||
|
}
|
||||||
|
|
||||||
|
color: Kirigami.Theme.backgroundColor
|
||||||
|
|
||||||
|
background: AbstractButton {
|
||||||
|
onClicked: root.destroy()
|
||||||
|
}
|
||||||
|
|
||||||
|
Map {
|
||||||
|
id: map
|
||||||
|
anchors.fill: parent
|
||||||
|
property string latlong: root.content.geo_uri.split(':')[1]
|
||||||
|
property string latitude: latlong.split(',')[0]
|
||||||
|
property string longitude: latlong.split(',')[1]
|
||||||
|
center: QtPositioning.coordinate(latitude, longitude)
|
||||||
|
zoomLevel: 15
|
||||||
|
plugin: OsmLocationPlugin.plugin
|
||||||
|
MapCircle {
|
||||||
|
radius: 1500 / map.zoomLevel
|
||||||
|
color: Kirigami.Theme.highlightColor
|
||||||
|
border.color: Kirigami.Theme.linkColor
|
||||||
|
border.width: Kirigami.Units.devicePixelRatio * 2
|
||||||
|
smooth: true
|
||||||
|
opacity: 0.25
|
||||||
|
center: QtPositioning.coordinate(map.latitude, map.longitude)
|
||||||
|
}
|
||||||
|
onCopyrightLinkActivated: {
|
||||||
|
Qt.openUrlExternally(link)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Button {
|
||||||
|
anchors.top: parent.top
|
||||||
|
anchors.right: parent.right
|
||||||
|
|
||||||
|
text: i18n("Close")
|
||||||
|
icon.name: "dialog-close"
|
||||||
|
display: AbstractButton.IconOnly
|
||||||
|
|
||||||
|
width: Kirigami.Units.gridUnit * 2
|
||||||
|
height: Kirigami.Units.gridUnit * 2
|
||||||
|
|
||||||
|
onClicked: root.destroy()
|
||||||
|
}
|
||||||
|
}
|
||||||
49
src/qml/Component/LocationPage.qml
Normal file
49
src/qml/Component/LocationPage.qml
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2023 Tobias Fella <tobias.fella@kde.org>
|
||||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||||
|
|
||||||
|
import QtQuick 2.15
|
||||||
|
import QtQuick.Controls 2.15 as QQC2
|
||||||
|
import QtLocation 5.15
|
||||||
|
import QtPositioning 5.15
|
||||||
|
|
||||||
|
import org.kde.kirigami 2.20 as Kirigami
|
||||||
|
import org.kde.neochat 1.0
|
||||||
|
|
||||||
|
Kirigami.Page {
|
||||||
|
id: locationsPage
|
||||||
|
|
||||||
|
required property var room
|
||||||
|
|
||||||
|
title: i18nc("Locations on a map", "Locations")
|
||||||
|
|
||||||
|
padding: 0
|
||||||
|
|
||||||
|
Map {
|
||||||
|
id: map
|
||||||
|
anchors.fill: parent
|
||||||
|
plugin: OsmLocationPlugin.plugin
|
||||||
|
|
||||||
|
MapItemView {
|
||||||
|
model: LocationsModel {
|
||||||
|
room: locationsPage.room
|
||||||
|
}
|
||||||
|
delegate: MapQuickItem {
|
||||||
|
id: point
|
||||||
|
|
||||||
|
required property var longitude
|
||||||
|
required property var latitude
|
||||||
|
required property string text
|
||||||
|
anchorPoint.x: icon.width / 2
|
||||||
|
anchorPoint.y: icon.height / 2
|
||||||
|
coordinate: QtPositioning.coordinate(point.latitude, point.longitude)
|
||||||
|
autoFadeIn: false
|
||||||
|
sourceItem: Kirigami.Icon {
|
||||||
|
id: icon
|
||||||
|
width: height
|
||||||
|
height: Kirigami.Units.iconSizes.medium
|
||||||
|
source: "flag-blue"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -36,6 +36,7 @@ TimelineContainer {
|
|||||||
* a user's location.
|
* a user's location.
|
||||||
*/
|
*/
|
||||||
required property string asset
|
required property string asset
|
||||||
|
required property var content
|
||||||
|
|
||||||
ColumnLayout {
|
ColumnLayout {
|
||||||
Layout.maximumWidth: root.contentMaxWidth
|
Layout.maximumWidth: root.contentMaxWidth
|
||||||
@@ -92,6 +93,10 @@ TimelineContainer {
|
|||||||
|
|
||||||
TapHandler {
|
TapHandler {
|
||||||
acceptedButtons: Qt.LeftButton
|
acceptedButtons: Qt.LeftButton
|
||||||
|
onTapped: {
|
||||||
|
let map = fullScreenMap.createObject(parent, {content: root.content});
|
||||||
|
map.open()
|
||||||
|
}
|
||||||
onLongPressed: openMessageContext("")
|
onLongPressed: openMessageContext("")
|
||||||
}
|
}
|
||||||
TapHandler {
|
TapHandler {
|
||||||
@@ -99,5 +104,9 @@ TimelineContainer {
|
|||||||
onTapped: openMessageContext("")
|
onTapped: openMessageContext("")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Component {
|
||||||
|
id: fullScreenMap
|
||||||
|
FullScreenMap { }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -143,6 +143,18 @@ Kirigami.OverlayDrawer {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Kirigami.BasicListItem {
|
||||||
|
id: locationsButton
|
||||||
|
|
||||||
|
icon: "map-flat"
|
||||||
|
text: i18n("Show locations for this room")
|
||||||
|
|
||||||
|
onClicked: pageStack.pushDialogLayer("qrc:/LocationsPage.qml", {
|
||||||
|
room: room
|
||||||
|
}, {
|
||||||
|
title: i18nc("Locations on a map", "Locations")
|
||||||
|
})
|
||||||
|
}
|
||||||
Kirigami.BasicListItem {
|
Kirigami.BasicListItem {
|
||||||
id: favouriteButton
|
id: favouriteButton
|
||||||
|
|
||||||
|
|||||||
@@ -128,5 +128,7 @@
|
|||||||
<file alias="SpaceDrawer.qml">qml/Page/RoomList/SpaceDrawer.qml</file>
|
<file alias="SpaceDrawer.qml">qml/Page/RoomList/SpaceDrawer.qml</file>
|
||||||
<file alias="OsmLocationPlugin.qml">qml/Component/Timeline/OsmLocationPlugin.qml</file>
|
<file alias="OsmLocationPlugin.qml">qml/Component/Timeline/OsmLocationPlugin.qml</file>
|
||||||
<file alias="LiveLocationDelegate.qml">qml/Component/Timeline/LiveLocationDelegate.qml</file>
|
<file alias="LiveLocationDelegate.qml">qml/Component/Timeline/LiveLocationDelegate.qml</file>
|
||||||
|
<file alias="FullScreenMap.qml">qml/Component/FullScreenMap.qml</file>
|
||||||
|
<file alias="LocationsPage.qml">qml/Component/LocationPage.qml</file>
|
||||||
</qresource>
|
</qresource>
|
||||||
</RCC>
|
</RCC>
|
||||||
|
|||||||
Reference in New Issue
Block a user