matrixion/src/ChatPage.h

297 lines
10 KiB
C
Raw Normal View History

2017-04-06 02:06:42 +03:00
/*
* nheko Copyright (C) 2017 Konstantinos Sideris <siderisk@auth.gr>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#pragma once
2017-04-06 02:06:42 +03:00
#include <atomic>
2019-12-14 19:08:36 +03:00
#include <optional>
#include <stack>
2019-12-14 19:08:36 +03:00
#include <variant>
2019-12-05 17:31:53 +03:00
#include <mtx/common.hpp>
#include <mtx/requests.hpp>
#include <mtx/responses.hpp>
#include <mtxclient/http/errors.hpp>
2017-11-09 00:09:15 +03:00
#include <QFrame>
2017-10-28 15:46:39 +03:00
#include <QHBoxLayout>
#include <QMap>
2017-04-06 02:06:42 +03:00
#include <QPixmap>
#include <QPoint>
2017-04-06 02:06:42 +03:00
#include <QTimer>
#include <QWidget>
#include "CacheCryptoStructs.h"
#include "CacheStructs.h"
2020-07-11 02:19:48 +03:00
#include "CallManager.h"
2018-01-09 16:07:32 +03:00
#include "CommunitiesList.h"
#include "Utils.h"
#include "notifications/Manager.h"
#include "popups/UserMentions.h"
2017-10-28 15:46:39 +03:00
class OverlayModal;
class QuickSwitcher;
class RoomList;
class SideBarActions;
class Splitter;
class TextInputWidget;
class TimelineViewManager;
class UserInfoWidget;
class UserSettings;
class NotificationsManager;
2020-07-29 00:55:47 +03:00
class TimelineModel;
2017-04-06 02:06:42 +03:00
constexpr int CONSENSUS_TIMEOUT = 1000;
constexpr int SHOW_CONTENT_TIMEOUT = 3000;
constexpr int TYPING_REFRESH_TIMEOUT = 10000;
namespace mtx::http {
using RequestErr = const std::optional<mtx::http::ClientError> &;
}
2017-04-06 02:06:42 +03:00
class ChatPage : public QWidget
{
Q_OBJECT
2017-04-06 02:06:42 +03:00
public:
2020-02-04 06:58:43 +03:00
ChatPage(QSharedPointer<UserSettings> userSettings, QWidget *parent = nullptr);
2017-04-06 02:06:42 +03:00
// Initialize all the components of the UI.
void bootstrap(QString userid, QString homeserver, QString token);
2017-10-20 21:39:05 +03:00
void showQuickSwitcher();
QString currentRoom() const { return current_room_; }
2018-01-03 19:05:49 +03:00
static ChatPage *instance() { return instance_; }
QSharedPointer<UserSettings> userSettings() { return userSettings_; }
2018-05-02 15:30:08 +03:00
void deleteConfigs();
2017-04-06 02:06:42 +03:00
2020-05-18 04:30:04 +03:00
CommunitiesList *communitiesList() { return communitiesList_; }
//! Calculate the width of the message timeline.
uint64_t timelineWidth();
//! Hide the room & group list (if it was visible).
void hideSideBars();
//! Show the room/group list (if it was visible).
void showSideBars();
void initiateLogout();
void query_keys(const std::string &req,
std::function<void(const UserKeyCache &, mtx::http::RequestErr)> cb);
2020-04-13 17:22:30 +03:00
void focusMessageInput();
QString status() const;
void setStatus(const QString &status);
2020-06-08 21:26:37 +03:00
mtx::presence::PresenceState currentPresence() const;
public slots:
void leaveRoom(const QString &room_id);
void createRoom(const mtx::requests::CreateRoom &req);
void inviteUser(QString userid, QString reason);
void kickUser(QString userid, QString reason);
void banUser(QString userid, QString reason);
void unbanUser(QString userid, QString reason);
signals:
void connectionLost();
void connectionRestored();
void notificationsRetrieved(const mtx::responses::Notifications &);
void highlightedNotifsRetrieved(const mtx::responses::Notifications &,
const QPoint widgetPos);
void uploadFailed(const QString &msg);
2019-12-05 17:31:53 +03:00
void mediaUploaded(const QString &roomid,
const QString &filename,
2019-12-14 19:08:36 +03:00
const std::optional<mtx::crypto::EncryptedFile> &file,
const QString &url,
2019-12-05 17:31:53 +03:00
const QString &mimeClass,
const QString &mime,
qint64 dsize,
2020-01-12 18:39:01 +03:00
const QSize &dimensions,
2020-04-13 17:22:30 +03:00
const QString &blurhash);
void contentLoaded();
2018-04-24 23:57:49 +03:00
void closing();
void changeWindowTitle(const QString &msg);
void unreadMessages(int count);
2017-10-08 22:38:38 +03:00
void showNotification(const QString &msg);
2017-10-20 22:32:48 +03:00
void showLoginPage(const QString &msg);
2017-11-02 01:41:13 +03:00
void showUserSettingsPage();
void showOverlayProgressBar();
2018-04-21 16:34:50 +03:00
void ownProfileOk();
void setUserDisplayName(const QString &name);
void setUserAvatar(const QString &avatar);
void loggedOut();
void trySyncCb();
void tryDelayedSyncCb();
void tryInitialSyncCb();
void newSyncResponse(mtx::responses::Sync res);
void leftRoom(const QString &room_id);
2018-04-21 16:34:50 +03:00
void initializeRoomList(QMap<QString, RoomInfo>);
void initializeViews(const mtx::responses::Rooms &rooms);
void initializeEmptyViews(const std::map<QString, mtx::responses::Timeline> &msgs);
void initializeMentions(const QMap<QString, mtx::responses::Notifications> &notifs);
2018-04-21 16:34:50 +03:00
void syncUI(const mtx::responses::Rooms &rooms);
void syncRoomlist(const std::map<QString, RoomInfo> &updates);
void syncTags(const std::map<QString, RoomInfo> &updates);
void dropToLoginPageCb(const QString &msg);
void notifyMessage(const QString &roomid,
const QString &eventid,
const QString &roomname,
const QString &sender,
const QString &message,
const QImage &icon);
2018-07-14 12:08:16 +03:00
void updateGroupsInfo(const mtx::responses::JoinedGroups &groups);
void retrievedPresence(const QString &statusMsg, mtx::presence::PresenceState state);
void themeChanged();
void decryptSidebarChanged();
2018-07-14 12:08:16 +03:00
//! Signals for device verificaiton
void receivedDeviceVerificationAccept(
2020-07-17 23:16:30 +03:00
const mtx::events::msg::KeyVerificationAccept &message);
void receivedDeviceVerificationRequest(
2020-07-17 23:16:30 +03:00
const mtx::events::msg::KeyVerificationRequest &message,
std::string sender);
void receivedRoomDeviceVerificationRequest(
2020-07-29 00:55:47 +03:00
const mtx::events::RoomEvent<mtx::events::msg::KeyVerificationRequest> &message,
TimelineModel *model);
void receivedDeviceVerificationCancel(
2020-07-17 23:16:30 +03:00
const mtx::events::msg::KeyVerificationCancel &message);
void receivedDeviceVerificationKey(const mtx::events::msg::KeyVerificationKey &message);
void receivedDeviceVerificationMac(const mtx::events::msg::KeyVerificationMac &message);
void receivedDeviceVerificationStart(const mtx::events::msg::KeyVerificationStart &message,
2020-07-17 23:16:30 +03:00
std::string sender);
void receivedDeviceVerificationReady(const mtx::events::msg::KeyVerificationReady &message);
void receivedDeviceVerificationDone(const mtx::events::msg::KeyVerificationDone &message);
private slots:
void showUnreadMessageNotification(int count);
void logout();
void removeRoom(const QString &room_id);
void dropToLoginPage(const QString &msg);
void joinRoom(const QString &room);
void sendTypingNotifications();
void handleSyncResponse(mtx::responses::Sync res);
2017-04-06 02:06:42 +03:00
private:
2018-01-03 19:05:49 +03:00
static ChatPage *instance_;
//! Handler callback for initial sync. It doesn't run on the main thread so all
//! communication with the GUI should be done through signals.
void initialSyncHandler(const mtx::responses::Sync &res, mtx::http::RequestErr err);
void startInitialSync();
void tryInitialSync();
void trySync();
void ensureOneTimeKeyCount(const std::map<std::string, uint16_t> &counts);
void getProfileInfo();
//! Check if the given room is currently open.
bool isRoomActive(const QString &room_id)
{
return isActiveWindow() && currentRoom() == room_id;
}
using UserID = QString;
using Membership = mtx::events::StateEvent<mtx::events::state::Member>;
using Memberships = std::map<std::string, Membership>;
2018-04-21 16:34:50 +03:00
using LeftRooms = std::map<std::string, mtx::responses::LeftRoom>;
void removeLeftRooms(const LeftRooms &rooms);
void loadStateFromCache();
2017-10-20 22:32:48 +03:00
void resetUI();
//! Decides whether or not to hide the group's sidebar.
void setGroupViewState(bool isEnabled);
template<class Collection>
Memberships getMemberships(const std::vector<Collection> &events) const;
//! Update the room with the new notification count.
void updateRoomNotificationCount(const QString &room_id,
uint16_t notification_count,
uint16_t highlight_count);
//! Send desktop notification for the received messages.
2020-06-10 12:27:21 +03:00
void sendNotifications(const mtx::responses::Notifications &);
2019-08-20 01:11:38 +03:00
void showNotificationsDialog(const QPoint &point);
2020-07-11 02:19:48 +03:00
template<typename T>
void connectCallMessage();
QHBoxLayout *topLayout_;
Splitter *splitter;
2017-05-19 19:55:38 +03:00
2018-01-09 16:07:32 +03:00
QWidget *sideBar_;
QVBoxLayout *sideBarLayout_;
2018-01-09 16:07:32 +03:00
QWidget *sideBarTopWidget_;
QVBoxLayout *sideBarTopWidgetLayout_;
2017-05-19 19:55:38 +03:00
2017-11-09 00:09:15 +03:00
QFrame *content_;
QVBoxLayout *contentLayout_;
2017-04-06 02:06:42 +03:00
2018-01-09 16:07:32 +03:00
CommunitiesList *communitiesList_;
RoomList *room_list_;
2018-01-09 16:07:32 +03:00
TimelineViewManager *view_manager_;
2017-10-15 22:08:51 +03:00
SideBarActions *sidebarActions_;
2017-04-06 02:06:42 +03:00
TextInputWidget *text_input_;
2017-04-06 02:06:42 +03:00
QTimer connectivityTimer_;
std::atomic_bool isConnected_;
2017-04-06 02:06:42 +03:00
QString current_room_;
2018-01-09 16:07:32 +03:00
QString current_community_;
UserInfoWidget *user_info_widget_;
2017-04-06 02:06:42 +03:00
popups::UserMentions *user_mentions_popup_;
QTimer *typingRefresher_;
2017-10-04 11:33:34 +03:00
// Global user settings.
QSharedPointer<UserSettings> userSettings_;
NotificationsManager notificationsManager;
2020-07-11 02:19:48 +03:00
CallManager callManager_;
2017-04-06 02:06:42 +03:00
};
template<class Collection>
std::map<std::string, mtx::events::StateEvent<mtx::events::state::Member>>
ChatPage::getMemberships(const std::vector<Collection> &collection) const
{
std::map<std::string, mtx::events::StateEvent<mtx::events::state::Member>> memberships;
using Member = mtx::events::StateEvent<mtx::events::state::Member>;
2018-02-28 13:12:07 +03:00
for (const auto &event : collection) {
2019-12-14 19:08:36 +03:00
if (auto member = std::get_if<Member>(event)) {
memberships.emplace(member->state_key, *member);
}
}
return memberships;
}