nheko/src/timeline/TimelineViewManager.h

234 lines
7.9 KiB
C
Raw Normal View History

2021-03-05 00:35:15 +01:00
// SPDX-FileCopyrightText: 2021 Nheko Contributors
//
// SPDX-License-Identifier: GPL-3.0-or-later
#pragma once
2017-04-06 01:06:42 +02:00
#include <QHash>
2019-11-09 03:06:10 +01:00
#include <QQuickView>
#include <QQuickWidget>
#include <QSharedPointer>
2019-11-09 03:06:10 +01:00
#include <QWidget>
2017-04-06 01:06:42 +02:00
2019-12-05 15:31:53 +01:00
#include <mtx/common.hpp>
2020-10-27 17:45:28 +01:00
#include <mtx/responses/messages.hpp>
#include <mtx/responses/sync.hpp>
#include <type_traits>
2019-11-09 03:06:10 +01:00
#include "Cache.h"
#include "CallManager.h"
2021-04-11 16:31:49 +02:00
#include "EventAccessors.h"
2019-11-09 03:06:10 +01:00
#include "Logging.h"
#include "TimelineModel.h"
#include "Utils.h"
2020-09-22 18:07:36 +02:00
#include "WebRTCSession.h"
#include "emoji/EmojiModel.h"
#include "emoji/Provider.h"
2019-11-09 03:06:10 +01:00
class MxcImageProvider;
class BlurhashProvider;
2019-11-09 03:06:10 +01:00
class ColorImageProvider;
2020-01-27 15:59:25 +01:00
class UserSettings;
class ChatPage;
2020-10-27 17:45:28 +01:00
class DeviceVerificationFlow;
2017-04-06 01:06:42 +02:00
struct nonesuch
{
~nonesuch() = delete;
nonesuch(nonesuch const &) = delete;
void operator=(nonesuch const &) = delete;
};
namespace detail {
template<class Default, class AlwaysVoid, template<class...> class Op, class... Args>
struct detector
{
using value_t = std::false_type;
using type = Default;
};
template<class Default, template<class...> class Op, class... Args>
struct detector<Default, std::void_t<Op<Args...>>, Op, Args...>
{
using value_t = std::true_type;
using type = Op<Args...>;
};
} // namespace detail
template<template<class...> class Op, class... Args>
using is_detected = typename detail::detector<nonesuch, void, Op, Args...>::value_t;
2019-11-09 03:06:10 +01:00
class TimelineViewManager : public QObject
2017-04-06 01:06:42 +02:00
{
2017-08-26 10:33:26 +02:00
Q_OBJECT
2017-04-06 01:06:42 +02:00
2019-11-09 03:06:10 +01:00
Q_PROPERTY(
TimelineModel *timeline MEMBER timeline_ READ activeTimeline NOTIFY activeTimelineChanged)
Q_PROPERTY(
bool isInitialSync MEMBER isInitialSync_ READ isInitialSync NOTIFY initialSyncChanged)
Q_PROPERTY(
bool isNarrowView MEMBER isNarrowView_ READ isNarrowView NOTIFY narrowViewChanged)
Q_PROPERTY(
bool isWindowFocused MEMBER isWindowFocused_ READ isWindowFocused NOTIFY focusChanged)
2019-11-09 03:06:10 +01:00
public:
2020-10-17 00:57:29 +02:00
TimelineViewManager(CallManager *callManager, ChatPage *parent = nullptr);
2019-11-09 03:06:10 +01:00
QWidget *getWidget() const { return container; }
void sync(const mtx::responses::Rooms &rooms);
2019-11-09 03:06:10 +01:00
void addRoom(const QString &room_id);
2017-10-07 19:50:32 +02:00
2020-12-16 22:10:09 +01:00
void clearAll()
{
timeline_ = nullptr;
emit activeTimelineChanged(nullptr);
models.clear();
}
2019-11-09 03:06:10 +01:00
Q_INVOKABLE TimelineModel *activeTimeline() const { return timeline_; }
Q_INVOKABLE bool isInitialSync() const { return isInitialSync_; }
bool isNarrowView() const { return isNarrowView_; }
bool isWindowFocused() const { return isWindowFocused_; }
Q_INVOKABLE void openImageOverlay(QString mxcUrl, QString eventId);
Q_INVOKABLE QColor userColor(QString id, QColor background);
2020-09-03 19:51:50 +02:00
Q_INVOKABLE QString escapeEmoji(QString str) const;
2017-04-11 21:48:02 +02:00
2020-06-24 16:24:22 +02:00
Q_INVOKABLE QString userPresence(QString id) const;
Q_INVOKABLE QString userStatus(QString id) const;
Q_INVOKABLE void openLink(QString link) const;
Q_INVOKABLE void focusMessageInput();
Q_INVOKABLE void openInviteUsersDialog();
Q_INVOKABLE void openMemberListDialog() const;
Q_INVOKABLE void openLeaveRoomDialog() const;
2020-10-05 22:58:07 +02:00
Q_INVOKABLE void removeVerificationFlow(DeviceVerificationFlow *flow);
void verifyUser(QString userid);
void verifyDevice(QString userid, QString deviceid);
signals:
void clearRoomMessageCount(QString roomid);
2019-11-09 03:06:10 +01:00
void updateRoomsLastMessage(QString roomid, const DescInfo &info);
void activeTimelineChanged(TimelineModel *timeline);
void initialSyncChanged(bool isInitialSync);
2020-01-28 05:28:11 +01:00
void replyingEventChanged(QString replyingEvent);
void replyClosed();
void newDeviceVerificationRequest(DeviceVerificationFlow *flow);
void inviteUsers(QStringList users);
void showRoomList();
void narrowViewChanged();
void focusChanged();
void focusInput();
void openImageOverlayInternalCb(QString eventId, QImage img);
2017-04-06 01:06:42 +02:00
public slots:
void updateReadReceipts(const QString &room_id, const std::vector<QString> &event_ids);
2020-10-20 19:46:37 +02:00
void receivedSessionKey(const std::string &room_id, const std::string &session_id);
2020-10-28 13:06:28 +01:00
void initWithMessages(const std::vector<QString> &roomIds);
void chatFocusChanged(bool focused)
{
isWindowFocused_ = focused;
emit focusChanged();
}
2017-08-26 10:33:26 +02:00
void setHistoryView(const QString &room_id);
void highlightRoom(const QString &room_id);
2021-02-22 20:16:40 +01:00
void focusTimeline();
2020-11-25 17:02:23 +01:00
TimelineModel *getHistoryView(const QString &room_id)
{
auto room = models.find(room_id);
if (room != models.end())
return room.value().data();
else
return nullptr;
}
2019-11-09 03:06:10 +01:00
void updateColorPalette();
void queueReply(const QString &roomid,
const QString &repliedToEvent,
const QString &replyBody);
2020-07-20 00:42:48 +02:00
void queueReactionMessage(const QString &reactedEvent, const QString &reactionKey);
2020-07-11 01:19:48 +02:00
void queueCallMessage(const QString &roomid, const mtx::events::msg::CallInvite &);
void queueCallMessage(const QString &roomid, const mtx::events::msg::CallCandidates &);
void queueCallMessage(const QString &roomid, const mtx::events::msg::CallAnswer &);
void queueCallMessage(const QString &roomid, const mtx::events::msg::CallHangUp &);
void updateEncryptedDescriptions();
void setVideoCallItem();
void enableBackButton()
{
if (isNarrowView_)
return;
isNarrowView_ = true;
emit narrowViewChanged();
}
void disableBackButton()
{
if (!isNarrowView_)
return;
isNarrowView_ = false;
emit narrowViewChanged();
}
void backToRooms() { emit showRoomList(); }
QObject *completerFor(QString completerName, QString roomId = "");
2021-04-11 16:31:49 +02:00
void forwardMessageToRoom(mtx::events::collections::TimelineEvents *e,
QString roomId,
bool sentFromEncrypted);
private slots:
void openImageOverlayInternal(QString eventId, QImage img);
private:
template<class Content>
using f_t = decltype(Content::file);
template<class Content>
using u_t = decltype(Content::url);
template<typename T>
static constexpr bool messageWithFileAndUrl(const mtx::events::Event<T> &e)
{
if constexpr (is_detected<f_t, T>::value && is_detected<u_t, T>::value) {
return true;
}
return false;
}
2017-04-06 01:06:42 +02:00
private:
2019-11-09 03:06:10 +01:00
#ifdef USE_QUICK_VIEW
QQuickView *view;
#else
QQuickWidget *view;
#endif
QWidget *container;
2019-11-09 03:06:10 +01:00
MxcImageProvider *imgProvider;
ColorImageProvider *colorImgProvider;
BlurhashProvider *blurhashProvider;
2019-11-09 03:06:10 +01:00
QHash<QString, QSharedPointer<TimelineModel>> models;
2020-07-11 01:19:48 +02:00
TimelineModel *timeline_ = nullptr;
CallManager *callManager_ = nullptr;
bool isInitialSync_ = true;
bool isNarrowView_ = false;
bool isWindowFocused_ = false;
2020-01-27 15:59:25 +01:00
QHash<QString, QColor> userColors;
QHash<QString, QSharedPointer<DeviceVerificationFlow>> dvList;
2017-04-06 01:06:42 +02:00
};
2020-07-17 22:16:30 +02:00
Q_DECLARE_METATYPE(mtx::events::msg::KeyVerificationAccept)
Q_DECLARE_METATYPE(mtx::events::msg::KeyVerificationCancel)
Q_DECLARE_METATYPE(mtx::events::msg::KeyVerificationDone)
Q_DECLARE_METATYPE(mtx::events::msg::KeyVerificationKey)
Q_DECLARE_METATYPE(mtx::events::msg::KeyVerificationMac)
Q_DECLARE_METATYPE(mtx::events::msg::KeyVerificationReady)
Q_DECLARE_METATYPE(mtx::events::msg::KeyVerificationRequest)
Q_DECLARE_METATYPE(mtx::events::msg::KeyVerificationStart)