/* This file is part of Telegram Desktop, the official desktop application for the Telegram messaging service. For license and copyright information please follow this link: https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL */ #pragma once #include "data/data_message_reaction_id.h" #include "base/timer.h" #include "base/type_traits.h" class History; namespace Data { class Session; class ForumTopic; class Thread; struct ItemNotification; enum class ItemNotificationType; } // namespace Data namespace Ui { struct PeerUserpicView; } // namespace Ui namespace Main { class Session; } // namespace Main namespace Platform { namespace Notifications { class Manager; } // namespace Notifications } // namespace Platform namespace Media::Audio { class Track; } // namespace Media::Audio namespace Window { class SessionController; } // namespace Window namespace Window::Notifications { enum class ManagerType { Dummy, Default, Native, }; enum class ChangeType { SoundEnabled, FlashBounceEnabled, IncludeMuted, CountMessages, DesktopEnabled, ViewParams, MaxCount, Corner, DemoIsShown, DemoIsHidden, }; } // namespace Window::Notifications namespace base { template <> struct custom_is_fast_copy_type : std::true_type { }; } // namespace base namespace Window::Notifications { class Manager; class System final { public: System(); ~System(); [[nodiscard]] Main::Session *findSession(uint64 sessionId) const; void createManager(); void setManager(std::unique_ptr manager); [[nodiscard]] Manager &manager() const; void checkDelayed(); void schedule(Data::ItemNotification notification); void clearFromTopic(not_null topic); void clearFromHistory(not_null history); void clearIncomingFromTopic(not_null topic); void clearIncomingFromHistory(not_null history); void clearFromSession(not_null session); void clearFromItem(not_null item); void clearAll(); void clearAllFast(); void updateAll(); [[nodiscard]] rpl::producer settingsChanged() const; void notifySettingsChanged(ChangeType type); void playSound(not_null session, DocumentId id); [[nodiscard]] rpl::lifetime &lifetime() { return _lifetime; } private: struct Waiter; struct SkipState { enum Value { Unknown, Skip, DontSkip }; Value value = Value::Unknown; bool silent = false; }; struct NotificationInHistoryKey { NotificationInHistoryKey(Data::ItemNotification notification); NotificationInHistoryKey( MsgId messageId, Data::ItemNotificationType type); MsgId messageId = 0; Data::ItemNotificationType type = Data::ItemNotificationType(); friend inline auto operator<=>( NotificationInHistoryKey a, NotificationInHistoryKey b) = default; }; struct Timing { crl::time delay = 0; crl::time when = 0; }; struct ReactionNotificationId { FullMsgId itemId; uint64 sessionId = 0; friend inline bool operator<( ReactionNotificationId a, ReactionNotificationId b) { return std::pair(a.itemId, a.sessionId) < std::pair(b.itemId, b.sessionId); } }; void clearForThreadIf(Fn)> predicate); [[nodiscard]] SkipState skipNotification( Data::ItemNotification notification) const; [[nodiscard]] SkipState computeSkipState( Data::ItemNotification notification) const; [[nodiscard]] Timing countTiming( not_null thread, crl::time minimalDelay) const; [[nodiscard]] bool skipReactionNotification( not_null item) const; void showNext(); void showGrouped(); void ensureSoundCreated(); [[nodiscard]] not_null lookupSound( not_null owner, DocumentId id); void registerThread(not_null thread); base::flat_map< not_null, base::flat_map> _whenMaps; base::flat_map, Waiter> _waiters; base::flat_map, Waiter> _settingWaiters; base::Timer _waitTimer; base::Timer _waitForAllGroupedTimer; base::flat_map< not_null, base::flat_map> _whenAlerts; mutable base::flat_map< ReactionNotificationId, crl::time> _sentReactionNotifications; std::unique_ptr _manager; rpl::event_stream _settingsChanged; std::unique_ptr _soundTrack; base::flat_map< DocumentId, std::unique_ptr> _customSoundTracks; base::flat_map< not_null, rpl::lifetime> _watchedTopics; int _lastForwardedCount = 0; uint64 _lastHistorySessionId = 0; FullMsgId _lastHistoryItemId; rpl::lifetime _lifetime; }; class Manager { public: struct ContextId { uint64 sessionId = 0; PeerId peerId = 0; MsgId topicRootId = 0; friend inline auto operator<=>( const ContextId&, const ContextId&) = default; [[nodiscard]] auto toTuple() const { return std::make_tuple( sessionId, peerId.value, topicRootId.bare); } template [[nodiscard]] static auto FromTuple(const T &tuple) { return ContextId{ std::get<0>(tuple), PeerIdHelper(std::get<1>(tuple)), std::get<2>(tuple), }; } }; struct NotificationId { ContextId contextId; MsgId msgId = 0; friend inline auto operator<=>( const NotificationId&, const NotificationId&) = default; [[nodiscard]] auto toTuple() const { return std::make_tuple( contextId.toTuple(), msgId.bare); } template [[nodiscard]] static auto FromTuple(const T &tuple) { return NotificationId{ ContextId::FromTuple(std::get<0>(tuple)), std::get<1>(tuple), }; } }; struct NotificationFields { not_null item; int forwardedCount = 0; PeerData *reactionFrom = nullptr; Data::ReactionId reactionId; }; explicit Manager(not_null system) : _system(system) { } void showNotification(NotificationFields fields) { doShowNotification(std::move(fields)); } void updateAll() { doUpdateAll(); } void clearAll() { doClearAll(); } void clearAllFast() { doClearAllFast(); } void clearFromItem(not_null item) { doClearFromItem(item); } void clearFromTopic(not_null topic) { doClearFromTopic(topic); } void clearFromHistory(not_null history) { doClearFromHistory(history); } void clearFromSession(not_null session) { doClearFromSession(session); } void notificationActivated( NotificationId id, const TextWithTags &draft = {}); void notificationReplied(NotificationId id, const TextWithTags &reply); struct DisplayOptions { bool hideNameAndPhoto = false; bool hideMessageText = false; bool hideMarkAsRead = false; bool hideReplyButton = false; }; [[nodiscard]] DisplayOptions getNotificationOptions( HistoryItem *item, Data::ItemNotificationType type) const; [[nodiscard]] static TextWithEntities ComposeReactionEmoji( not_null session, const Data::ReactionId &reaction); [[nodiscard]] static TextWithEntities ComposeReactionNotification( not_null item, const Data::ReactionId &reaction, bool hideContent); [[nodiscard]] TextWithEntities addTargetAccountName( TextWithEntities title, not_null session); [[nodiscard]] QString addTargetAccountName( const QString &title, not_null session); [[nodiscard]] virtual ManagerType type() const = 0; [[nodiscard]] bool skipAudio() const { return doSkipAudio(); } [[nodiscard]] bool skipToast() const { return doSkipToast(); } [[nodiscard]] bool skipFlashBounce() const { return doSkipFlashBounce(); } virtual ~Manager() = default; protected: [[nodiscard]] not_null system() const { return _system; } virtual void doUpdateAll() = 0; virtual void doShowNotification(NotificationFields &&fields) = 0; virtual void doClearAll() = 0; virtual void doClearAllFast() = 0; virtual void doClearFromItem(not_null item) = 0; virtual void doClearFromTopic(not_null topic) = 0; virtual void doClearFromHistory(not_null history) = 0; virtual void doClearFromSession(not_null session) = 0; virtual bool doSkipAudio() const = 0; virtual bool doSkipToast() const = 0; virtual bool doSkipFlashBounce() const = 0; [[nodiscard]] virtual bool forceHideDetails() const { return false; } virtual void onBeforeNotificationActivated(NotificationId id) { } virtual void onAfterNotificationActivated( NotificationId id, not_null window) { } [[nodiscard]] virtual QString accountNameSeparator(); private: void openNotificationMessage( not_null history, MsgId messageId); const not_null _system; }; class NativeManager : public Manager { public: [[nodiscard]] ManagerType type() const override { return ManagerType::Native; } protected: using Manager::Manager; void doUpdateAll() override { doClearAllFast(); } void doClearAll() override { doClearAllFast(); } void doShowNotification(NotificationFields &&fields) override; bool forceHideDetails() const override; virtual void doShowNativeNotification( not_null peer, MsgId topicRootId, Ui::PeerUserpicView &userpicView, MsgId msgId, const QString &title, const QString &subtitle, const QString &msg, DisplayOptions options) = 0; }; class DummyManager : public NativeManager { public: using NativeManager::NativeManager; [[nodiscard]] ManagerType type() const override { return ManagerType::Dummy; } protected: void doShowNativeNotification( not_null peer, MsgId topicRootId, Ui::PeerUserpicView &userpicView, MsgId msgId, const QString &title, const QString &subtitle, const QString &msg, DisplayOptions options) override { } void doClearAllFast() override { } void doClearFromItem(not_null item) override { } void doClearFromTopic(not_null topic) override { } void doClearFromHistory(not_null history) override { } void doClearFromSession(not_null session) override { } bool doSkipAudio() const override { return false; } bool doSkipToast() const override { return false; } bool doSkipFlashBounce() const override { return false; } }; [[nodiscard]] QString WrapFromScheduled(const QString &text); } // namespace Window::Notifications