matrixion/include/ChatPage.h

194 lines
6.2 KiB
C
Raw Permalink 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
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 <QTimer>
#include <QWidget>
#include <mtx.hpp>
2017-10-28 15:46:39 +03:00
class Cache;
class MatrixClient;
class OverlayModal;
class QuickSwitcher;
class RoomList;
class RoomSettings;
class RoomState;
class SideBarActions;
class Splitter;
class TextInputWidget;
class TimelineViewManager;
class TopRoomBar;
class TypingDisplay;
class UserInfoWidget;
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;
2017-04-06 02:06:42 +03:00
class ChatPage : public QWidget
{
Q_OBJECT
2017-04-06 02:06:42 +03:00
public:
ChatPage(QSharedPointer<MatrixClient> client, QWidget *parent = 0);
~ChatPage();
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();
2017-04-06 02:06:42 +03:00
signals:
void contentLoaded();
void close();
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();
private slots:
void showUnreadMessageNotification(int count);
void updateTopBarAvatar(const QString &roomid, const QPixmap &img);
void updateOwnProfileInfo(const QUrl &avatar_url, const QString &display_name);
void setOwnAvatar(const QPixmap &img);
void initialSyncCompleted(const mtx::responses::Sync &response);
void syncCompleted(const mtx::responses::Sync &response);
void syncFailed(const QString &msg);
void changeTopRoomInfo(const QString &room_id);
void logout();
void addRoom(const QString &room_id);
void removeRoom(const QString &room_id);
2017-12-19 23:36:12 +03:00
void removeInvite(const QString &room_id);
2017-04-06 02:06:42 +03:00
private:
using UserID = QString;
using RoomStates = QMap<UserID, RoomState>;
using Membership = mtx::events::StateEvent<mtx::events::state::Member>;
using Memberships = std::map<std::string, Membership>;
using JoinedRooms = std::map<std::string, mtx::responses::JoinedRoom>;
using LeftRooms = std::map<std::string, mtx::responses::LeftRoom>;
void removeLeftRooms(const LeftRooms &rooms);
void updateJoinedRooms(const JoinedRooms &rooms);
RoomStates generateMembershipDifference(const JoinedRooms &rooms,
const RoomStates &states) const;
void updateTypingUsers(const QString &roomid, const std::vector<std::string> &user_ids);
using MemberEvent = mtx::events::StateEvent<mtx::events::state::Member>;
void updateUserDisplayName(const MemberEvent &event);
void updateUserAvatarUrl(const MemberEvent &event);
void loadStateFromCache();
2017-10-20 22:32:48 +03:00
void deleteConfigs();
void resetUI();
template<class Collection>
Memberships getMemberships(const std::vector<Collection> &events) const;
template<class Collection>
void updateUserMetadata(const std::vector<Collection> &collection);
QHBoxLayout *topLayout_;
Splitter *splitter;
2017-05-19 19:55:38 +03:00
2017-11-09 00:09:15 +03:00
QFrame *sideBar_;
QVBoxLayout *sideBarLayout_;
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
RoomList *room_list_;
TimelineViewManager *view_manager_;
2017-10-15 22:08:51 +03:00
SideBarActions *sidebarActions_;
2017-04-06 02:06:42 +03:00
TopRoomBar *top_bar_;
TextInputWidget *text_input_;
2017-10-04 11:33:34 +03:00
TypingDisplay *typingDisplay_;
2017-04-06 02:06:42 +03:00
// Safety net if consensus is not possible or too slow.
QTimer *showContentTimer_;
2017-10-07 20:50:32 +03:00
QTimer *consensusTimer_;
2017-04-06 02:06:42 +03:00
QString current_room_;
QMap<QString, QPixmap> room_avatars_;
2017-04-06 02:06:42 +03:00
UserInfoWidget *user_info_widget_;
2017-04-06 02:06:42 +03:00
QMap<QString, RoomState> state_manager_;
QMap<QString, QSharedPointer<RoomSettings>> settingsManager_;
2017-10-04 11:33:34 +03:00
// Keeps track of the users currently typing on each room.
QMap<QString, QList<QString>> typingUsers_;
QTimer *typingRefresher_;
2017-10-04 11:33:34 +03:00
2017-10-07 20:09:34 +03:00
QSharedPointer<QuickSwitcher> quickSwitcher_;
QSharedPointer<OverlayModal> quickSwitcherModal_;
2017-08-15 21:06:27 +03:00
// Matrix Client API provider.
QSharedPointer<MatrixClient> client_;
// LMDB wrapper.
QSharedPointer<Cache> cache_;
2017-10-20 22:32:48 +03:00
// If the number of failures exceeds a certain threshold we
// return to the login page.
int initialSyncFailures = 0;
2017-04-06 02:06:42 +03:00
};
template<class Collection>
void
ChatPage::updateUserMetadata(const std::vector<Collection> &collection)
{
using Member = mtx::events::StateEvent<mtx::events::state::Member>;
for (auto &event : collection) {
if (mpark::holds_alternative<Member>(event)) {
auto member = mpark::get<Member>(event);
updateUserAvatarUrl(member);
updateUserDisplayName(member);
}
}
}
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>;
for (auto &event : collection) {
if (mpark::holds_alternative<Member>(event)) {
auto member = mpark::get<Member>(event);
memberships.emplace(member.state_key, member);
}
}
return memberships;
}