/* 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 */ #include "info/profile/info_profile_values.h" #include #include #include #include #include "observer_peer.h" #include "messenger.h" #include "auth_session.h" #include "ui/wrap/slide_wrap.h" #include "data/data_peer_values.h" #include "data/data_shared_media.h" #include "data/data_feed.h" #include "data/data_session.h" namespace Info { namespace Profile { rpl::producer PhoneValue( not_null user) { return Notify::PeerUpdateValue( user, Notify::PeerUpdate::Flag::UserPhoneChanged ) | rpl::map([user] { return App::formatPhone(user->phone()); }) | WithEmptyEntities(); } auto PlainBioValue( not_null user) { return Notify::PeerUpdateValue( user, Notify::PeerUpdate::Flag::AboutChanged ) | rpl::map([user] { return user->about(); }); } rpl::producer BioValue( not_null user) { return PlainBioValue(user) | ToSingleLine() | WithEmptyEntities(); } auto PlainUsernameValue( not_null peer) { return Notify::PeerUpdateValue( peer, Notify::PeerUpdate::Flag::UsernameChanged ) | rpl::map([peer] { return peer->userName(); }); } rpl::producer UsernameValue( not_null user) { return PlainUsernameValue( user ) | rpl::map([](QString &&username) { return username.isEmpty() ? QString() : ('@' + username); }) | WithEmptyEntities(); } rpl::producer PlainAboutValue( not_null peer) { if (auto channel = peer->asChannel()) { return Notify::PeerUpdateValue( channel, Notify::PeerUpdate::Flag::AboutChanged ) | rpl::map([channel] { return channel->about(); }); } else if (auto user = peer->asUser()) { if (user->botInfo) { return PlainBioValue(user); } } return rpl::single(QString()); } rpl::producer AboutValue( not_null peer) { auto flags = TextParseLinks | TextParseMentions | TextParseHashtags; if (peer->isUser()) { flags |= TextParseBotCommands; } return PlainAboutValue( peer ) | WithEmptyEntities( ) | rpl::map([=](TextWithEntities &&text) { TextUtilities::ParseEntities(text, flags); return std::move(text); }); } rpl::producer LinkValue( not_null peer) { return PlainUsernameValue( peer ) | rpl::map([](QString &&username) { return username.isEmpty() ? QString() : Messenger::Instance().createInternalLinkFull(username); }); } rpl::producer NotificationsEnabledValue( not_null peer) { return Notify::PeerUpdateValue( peer, Notify::PeerUpdate::Flag::NotificationsEnabled ) | rpl::map([peer] { return !peer->isMuted(); }) | rpl::distinct_until_changed(); } rpl::producer IsContactValue( not_null user) { return Notify::PeerUpdateValue( user, Notify::PeerUpdate::Flag::UserIsContact ) | rpl::map([user] { return user->isContact(); }); } rpl::producer CanInviteBotToGroupValue( not_null user) { if (!user->botInfo) { return rpl::single(false); } return Notify::PeerUpdateValue( user, Notify::PeerUpdate::Flag::BotCanAddToGroups ) | rpl::map([user] { return !user->botInfo->cantJoinGroups; }); } rpl::producer CanShareContactValue( not_null user) { return Notify::PeerUpdateValue( user, Notify::PeerUpdate::Flag::UserCanShareContact ) | rpl::map([user] { return user->canShareThisContact(); }); } rpl::producer CanAddContactValue( not_null user) { using namespace rpl::mappers; return rpl::combine( IsContactValue(user), CanShareContactValue(user), !_1 && _2); } rpl::producer AmInChannelValue( not_null channel) { return Notify::PeerUpdateValue( channel, Notify::PeerUpdate::Flag::ChannelAmIn ) | rpl::map([channel] { return channel->amIn(); }); } rpl::producer MembersCountValue( not_null peer) { if (auto chat = peer->asChat()) { return Notify::PeerUpdateValue( peer, Notify::PeerUpdate::Flag::MembersChanged ) | rpl::map([chat] { return chat->amIn() ? std::max(chat->count, int(chat->participants.size())) : 0; }); } else if (auto channel = peer->asChannel()) { return Notify::PeerUpdateValue( channel, Notify::PeerUpdate::Flag::MembersChanged ) | rpl::map([channel] { return channel->membersCount(); }); } Unexpected("User in MembersCountViewer()."); } rpl::producer AdminsCountValue( not_null channel) { using Flag = Notify::PeerUpdate::Flag; return Notify::PeerUpdateValue( channel, Flag::AdminsChanged | Flag::ChannelRightsChanged ) | rpl::map([channel] { return channel->canViewAdmins() ? channel->adminsCount() : 0; }); } rpl::producer RestrictedCountValue( not_null channel) { using Flag = Notify::PeerUpdate::Flag; return Notify::PeerUpdateValue( channel, Flag::BannedUsersChanged | Flag::ChannelRightsChanged ) | rpl::map([channel] { return channel->canViewBanned() ? channel->restrictedCount() : 0; }); } rpl::producer KickedCountValue( not_null channel) { using Flag = Notify::PeerUpdate::Flag; return Notify::PeerUpdateValue( channel, Flag::BannedUsersChanged | Flag::ChannelRightsChanged ) | rpl::map([channel] { return channel->canViewBanned() ? channel->kickedCount() : 0; }); } rpl::producer SharedMediaCountValue( not_null peer, PeerData *migrated, Storage::SharedMediaType type) { auto aroundId = 0; auto limit = 0; auto updated = SharedMediaMergedViewer( SharedMediaMergedKey( SparseIdsMergedSlice::Key( peer->id, migrated ? migrated->id : 0, aroundId), type), limit, limit ) | rpl::map([](const SparseIdsMergedSlice &slice) { return slice.fullCount(); }) | rpl::filter_optional(); return rpl::single(0) | rpl::then(std::move(updated)); } rpl::producer CommonGroupsCountValue( not_null user) { return Notify::PeerUpdateValue( user, Notify::PeerUpdate::Flag::UserCommonChatsChanged ) | rpl::map([user] { return user->commonChatsCount(); }); } rpl::producer CanAddMemberValue( not_null peer) { if (auto chat = peer->asChat()) { return Notify::PeerUpdateValue( chat, Notify::PeerUpdate::Flag::ChatCanEdit ) | rpl::map([chat] { return chat->canEdit(); }); } else if (auto channel = peer->asChannel()) { return Notify::PeerUpdateValue( channel, Notify::PeerUpdate::Flag::ChannelRightsChanged ) | rpl::map([channel] { return channel->canAddMembers(); }); } return rpl::single(false); } rpl::producer VerifiedValue( not_null peer) { if (auto user = peer->asUser()) { return Data::PeerFlagValue(user, MTPDuser::Flag::f_verified); } else if (auto channel = peer->asChannel()) { return Data::PeerFlagValue( channel, MTPDchannel::Flag::f_verified); } return rpl::single(false); } rpl::producer FeedChannelsCountValue( not_null feed) { using Flag = Data::FeedUpdateFlag; return rpl::single( Data::FeedUpdate{ feed, Flag::Channels } ) | rpl::then( Auth().data().feedUpdated() ) | rpl::filter([=](const Data::FeedUpdate &update) { return (update.feed == feed) && (update.flag == Flag::Channels); }) | rpl::filter([=] { return feed->channelsLoaded(); }) | rpl::map([=] { return int(feed->channels().size()); }) | rpl::distinct_until_changed(); } } // namespace Profile } // namespace Info