// SPDX-FileCopyrightText: 2018-2019 Black Hat // SPDX-License-Identifier: GPL-3.0-only #pragma once #include #include #include "neochatconnection.h" #include "notificationsmanager.h" #include class TrayIcon; class QQuickTextDocument; namespace QKeychain { class ReadPasswordJob; } /** * @class Controller * * A singleton class designed to help manage the application. * * There are also a bunch of helper functions that currently don't fit anywhere * else. */ class Controller : public QObject { Q_OBJECT QML_ELEMENT QML_SINGLETON /** * @brief The current connection for the rest of NeoChat to use. */ Q_PROPERTY(NeoChatConnection *activeConnection READ activeConnection WRITE setActiveConnection NOTIFY activeConnectionChanged) /** * @brief Whether the OS NeoChat is running on supports sytem tray icons. */ Q_PROPERTY(bool supportSystemTray READ supportSystemTray CONSTANT) /** * @brief Whether NeoChat is running as a flatpak. * * This is the only way to gate NeoChat features in flatpaks in QML. */ Q_PROPERTY(bool isFlatpak READ isFlatpak CONSTANT) Q_PROPERTY(QStringList accountsLoading MEMBER m_accountsLoading NOTIFY accountsLoadingChanged) Q_PROPERTY(bool csSupported READ csSupported CONSTANT) public: static Controller &instance(); static Controller *create(QQmlEngine *engine, QJSEngine *) { engine->setObjectOwnership(&instance(), QQmlEngine::CppOwnership); return &instance(); } void setActiveConnection(NeoChatConnection *connection); [[nodiscard]] NeoChatConnection *activeConnection() const; /** * @brief Add a new connection to the account registry. */ void addConnection(NeoChatConnection *c); /** * @brief Drop a connection from the account registry. */ void dropConnection(NeoChatConnection *c); /** * @brief Save an access token to the keychain for the given account. */ void saveAccessTokenToKeyChain(const QString &userId, const QByteArray &accessToken); [[nodiscard]] bool supportSystemTray() const; bool isFlatpak() const; /** * @brief Start listening for notifications in dbus-activated mode. * These notifications will quit the application when closed. */ static void listenForNotifications(); /** * @brief Clear an existing invite notification for the given room. * * Nothing happens if the given room doesn't have an invite notification. */ Q_INVOKABLE void clearInvitationNotification(const QString &roomId); Q_INVOKABLE QString loadFileContent(const QString &path) const; Quotient::AccountRegistry &accounts(); static void setTestMode(bool testMode); Q_INVOKABLE void removeConnection(const QString &userId); bool csSupported() const; /** * @brief Revert all configuration values to their default. * * The parameters along with their defaults are specified in the config file * neochatconfig.kcfg. */ Q_INVOKABLE void revertToDefaultConfig(); Q_INVOKABLE bool isImageShown(const QString &eventId); Q_INVOKABLE void markImageShown(const QString &eventId); private: explicit Controller(QObject *parent = nullptr); QPointer m_connection; TrayIcon *m_trayIcon = nullptr; QKeychain::ReadPasswordJob *loadAccessTokenFromKeyChain(const QString &account); Quotient::AccountRegistry m_accountRegistry; QStringList m_accountsLoading; QMap> m_connectionsLoading; QString m_endpoint; QStringList m_shownImages; NotificationsManager m_notificationsManager; private Q_SLOTS: void invokeLogin(); void setQuitOnLastWindowClosed(); void updateBadgeNotificationCount(NeoChatConnection *connection, int count); Q_SIGNALS: /** * @brief Request a error message be shown to the user. */ void errorOccured(const QString &error); void connectionAdded(NeoChatConnection *connection); void connectionDropped(NeoChatConnection *connection); void activeConnectionChanged(NeoChatConnection *connection); void accountsLoadingChanged(); };