| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624 |
- /*
- 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 "data/data_peer_values.h"
- #include "lang/lang_keys.h"
- #include "data/data_channel.h"
- #include "data/data_chat.h"
- #include "data/data_user.h"
- #include "data/data_changes.h"
- #include "data/data_forum_topic.h"
- #include "data/data_session.h"
- #include "data/data_message_reactions.h"
- #include "main/main_session.h"
- #include "main/main_app_config.h"
- #include "ui/image/image_prepare.h"
- #include "base/unixtime.h"
- namespace Data {
- namespace {
- constexpr auto kMinOnlineChangeTimeout = crl::time(1000);
- constexpr auto kMaxOnlineChangeTimeout = 86400 * crl::time(1000);
- constexpr auto kSecondsInDay = 86400;
- int OnlinePhraseChangeInSeconds(LastseenStatus status, TimeId now) {
- const auto till = status.onlineTill();
- if (till > now) {
- return till - now;
- } else if (status.isHidden()) {
- return std::numeric_limits<int>::max();
- }
- const auto minutes = (now - till) / 60;
- if (minutes < 60) {
- return (minutes + 1) * 60 - (now - till);
- }
- const auto hours = (now - till) / 3600;
- if (hours < 12) {
- return (hours + 1) * 3600 - (now - till);
- }
- const auto nowFull = base::unixtime::parse(now);
- const auto tomorrow = nowFull.date().addDays(1).startOfDay();
- return std::max(static_cast<TimeId>(nowFull.secsTo(tomorrow)), 0);
- }
- std::optional<QString> OnlineTextSpecial(not_null<UserData*> user) {
- if (user->isNotificationsUser()) {
- return tr::lng_status_service_notifications(tr::now);
- } else if (user->isSupport()) {
- return tr::lng_status_support(tr::now);
- } else if (user->isBot()) {
- if (const auto count = user->botInfo->activeUsers) {
- return tr::lng_bot_status_users(
- tr::now,
- lt_count_decimal,
- count);
- }
- return tr::lng_status_bot(tr::now);
- } else if (user->isServiceUser()) {
- return tr::lng_status_support(tr::now);
- }
- return std::nullopt;
- }
- std::optional<QString> OnlineTextCommon(LastseenStatus status, TimeId now) {
- if (status.isOnline(now)) {
- return tr::lng_status_online(tr::now);
- } else if (status.isLongAgo()) {
- return tr::lng_status_offline(tr::now);
- } else if (status.isRecently()) {
- return tr::lng_status_recently(tr::now);
- } else if (status.isWithinWeek()) {
- return tr::lng_status_last_week(tr::now);
- } else if (status.isWithinMonth()) {
- return tr::lng_status_last_month(tr::now);
- } else if (status.isHidden()) {
- return tr::lng_status_recently(tr::now);
- }
- return std::nullopt;
- }
- [[nodiscard]] int UniqueReactionsLimit(not_null<Main::AppConfig*> config) {
- return config->get<int>("reactions_uniq_max", 11);
- }
- } // namespace
- inline auto AdminRightsValue(not_null<ChannelData*> channel) {
- return channel->adminRightsValue();
- }
- inline auto AdminRightsValue(
- not_null<ChannelData*> channel,
- ChatAdminRights mask) {
- return FlagsValueWithMask(AdminRightsValue(channel), mask);
- }
- inline auto AdminRightValue(
- not_null<ChannelData*> channel,
- ChatAdminRight flag) {
- return SingleFlagValue(AdminRightsValue(channel), flag);
- }
- inline auto AdminRightsValue(not_null<ChatData*> chat) {
- return chat->adminRightsValue();
- }
- inline auto AdminRightsValue(
- not_null<ChatData*> chat,
- ChatAdminRights mask) {
- return FlagsValueWithMask(AdminRightsValue(chat), mask);
- }
- inline auto AdminRightValue(
- not_null<ChatData*> chat,
- ChatAdminRight flag) {
- return SingleFlagValue(AdminRightsValue(chat), flag);
- }
- inline auto RestrictionsValue(not_null<ChannelData*> channel) {
- return channel->restrictionsValue();
- }
- inline auto RestrictionsValue(
- not_null<ChannelData*> channel,
- ChatRestrictions mask) {
- return FlagsValueWithMask(RestrictionsValue(channel), mask);
- }
- inline auto RestrictionValue(
- not_null<ChannelData*> channel,
- ChatRestriction flag) {
- return SingleFlagValue(RestrictionsValue(channel), flag);
- }
- inline auto DefaultRestrictionsValue(not_null<ChannelData*> channel) {
- return channel->defaultRestrictionsValue();
- }
- inline auto DefaultRestrictionsValue(
- not_null<ChannelData*> channel,
- ChatRestrictions mask) {
- return FlagsValueWithMask(DefaultRestrictionsValue(channel), mask);
- }
- inline auto DefaultRestrictionValue(
- not_null<ChannelData*> channel,
- ChatRestriction flag) {
- return SingleFlagValue(DefaultRestrictionsValue(channel), flag);
- }
- inline auto DefaultRestrictionsValue(not_null<ChatData*> chat) {
- return chat->defaultRestrictionsValue();
- }
- inline auto DefaultRestrictionsValue(
- not_null<ChatData*> chat,
- ChatRestrictions mask) {
- return FlagsValueWithMask(DefaultRestrictionsValue(chat), mask);
- }
- inline auto DefaultRestrictionValue(
- not_null<ChatData*> chat,
- ChatRestriction flag) {
- return SingleFlagValue(DefaultRestrictionsValue(chat), flag);
- }
- // Duplicated in CanSendAnyOf().
- [[nodiscard]] rpl::producer<bool> CanSendAnyOfValue(
- not_null<Thread*> thread,
- ChatRestrictions rights,
- bool forbidInForums) {
- if (const auto topic = thread->asTopic()) {
- using Flag = ChannelDataFlag;
- const auto mask = Flag()
- | Flag::Left
- | Flag::JoinToWrite
- | Flag::HasLink
- | Flag::Forbidden
- | Flag::Creator;
- const auto channel = topic->channel();
- return rpl::combine(
- PeerFlagsValue(channel.get(), mask),
- RestrictionsValue(channel, rights),
- DefaultRestrictionsValue(channel, rights),
- AdminRightsValue(channel, ChatAdminRight::ManageTopics),
- topic->session().changes().topicFlagsValue(
- topic,
- TopicUpdate::Flag::Closed),
- [=](
- ChannelDataFlags flags,
- ChatRestrictions sendRestriction,
- ChatRestrictions defaultSendRestriction,
- auto,
- auto) {
- const auto notAmInFlags = Flag::Left | Flag::Forbidden;
- const auto allowed = !(flags & notAmInFlags)
- || ((flags & Flag::HasLink)
- && !(flags & Flag::JoinToWrite));
- return allowed
- && ((flags & Flag::Creator)
- || (!sendRestriction && !defaultSendRestriction))
- && (!topic->closed() || topic->canToggleClosed());
- });
- }
- return CanSendAnyOfValue(thread->peer(), rights, forbidInForums);
- }
- // Duplicated in CanSendAnyOf().
- [[nodiscard]] rpl::producer<bool> CanSendAnyOfValue(
- not_null<PeerData*> peer,
- ChatRestrictions rights,
- bool forbidInForums) {
- if (const auto user = peer->asUser()) {
- if (user->isRepliesChat() || user->isVerifyCodes()) {
- return rpl::single(false);
- }
- using namespace rpl::mappers;
- const auto other = rights & ~(ChatRestriction::SendVoiceMessages
- | ChatRestriction::SendVideoMessages);
- auto allowedAny = PeerFlagsValue(
- user,
- (UserDataFlag::Deleted | UserDataFlag::RequiresPremiumToWrite)
- ) | rpl::map([=](UserDataFlags flags) {
- return (flags & UserDataFlag::Deleted)
- ? rpl::single(false)
- : !(flags & UserDataFlag::RequiresPremiumToWrite)
- ? rpl::single(true)
- : AmPremiumValue(&user->session());
- }) | rpl::flatten_latest();
- if (other) {
- return allowedAny;
- }
- const auto mask = UserDataFlag::VoiceMessagesForbidden;
- return rpl::combine(
- std::move(allowedAny),
- PeerFlagValue(user, mask),
- _1 && !_2);
- } else if (const auto chat = peer->asChat()) {
- const auto mask = ChatDataFlag()
- | ChatDataFlag::Deactivated
- | ChatDataFlag::Forbidden
- | ChatDataFlag::Left
- | ChatDataFlag::Creator;
- return rpl::combine(
- PeerFlagsValue(chat, mask),
- AdminRightsValue(chat),
- DefaultRestrictionsValue(chat, rights),
- [rights](
- ChatDataFlags flags,
- Data::Flags<ChatAdminRights>::Change adminRights,
- ChatRestrictions defaultSendRestrictions) {
- const auto amOutFlags = ChatDataFlag()
- | ChatDataFlag::Deactivated
- | ChatDataFlag::Forbidden
- | ChatDataFlag::Left;
- return !(flags & amOutFlags)
- && ((flags & ChatDataFlag::Creator)
- || (adminRights.value != ChatAdminRights(0))
- || (rights & ~defaultSendRestrictions));
- });
- } else if (const auto channel = peer->asChannel()) {
- using Flag = ChannelDataFlag;
- const auto mask = Flag()
- | Flag::Left
- | Flag::Forum
- | Flag::JoinToWrite
- | Flag::HasLink
- | Flag::Forbidden
- | Flag::Creator
- | Flag::Broadcast;
- return rpl::combine(
- PeerFlagsValue(channel, mask),
- AdminRightValue(
- channel,
- ChatAdminRight::PostMessages),
- channel->unrestrictedByBoostsValue(),
- RestrictionsValue(channel, rights),
- DefaultRestrictionsValue(channel, rights),
- [=](
- ChannelDataFlags flags,
- bool postMessagesRight,
- bool unrestrictedByBoosts,
- ChatRestrictions sendRestriction,
- ChatRestrictions defaultSendRestriction) {
- const auto notAmInFlags = Flag::Left | Flag::Forbidden;
- const auto forumRestriction = forbidInForums
- && (flags & Flag::Forum);
- const auto allowed = !(flags & notAmInFlags)
- || ((flags & Flag::HasLink)
- && !(flags & Flag::JoinToWrite));
- const auto restricted = sendRestriction
- | (defaultSendRestriction && !unrestrictedByBoosts);
- return allowed
- && !forumRestriction
- && (postMessagesRight
- || (flags & Flag::Creator)
- || (!(flags & Flag::Broadcast)
- && (rights & ~restricted)));
- });
- }
- Unexpected("Peer type in Data::CanSendAnyOfValue.");
- }
- // This is duplicated in PeerData::canPinMessages().
- rpl::producer<bool> CanPinMessagesValue(not_null<PeerData*> peer) {
- using namespace rpl::mappers;
- if (const auto user = peer->asUser()) {
- return PeerFlagsValue(
- user,
- UserDataFlag::CanPinMessages
- ) | rpl::map(_1 != UserDataFlag(0));
- } else if (const auto chat = peer->asChat()) {
- const auto mask = 0
- | ChatDataFlag::Deactivated
- | ChatDataFlag::Forbidden
- | ChatDataFlag::Left
- | ChatDataFlag::Creator;
- return rpl::combine(
- PeerFlagsValue(chat, mask),
- AdminRightValue(chat, ChatAdminRight::PinMessages),
- DefaultRestrictionValue(chat, ChatRestriction::PinMessages),
- [](
- ChatDataFlags flags,
- bool adminRightAllows,
- bool defaultRestriction) {
- const auto amOutFlags = 0
- | ChatDataFlag::Deactivated
- | ChatDataFlag::Forbidden
- | ChatDataFlag::Left;
- return !(flags & amOutFlags)
- && ((flags & ChatDataFlag::Creator)
- || adminRightAllows
- || !defaultRestriction);
- });
- } else if (const auto megagroup = peer->asMegagroup()) {
- if (megagroup->amCreator()) {
- return rpl::single(true);
- }
- return rpl::combine(
- AdminRightValue(megagroup, ChatAdminRight::PinMessages),
- DefaultRestrictionValue(megagroup, ChatRestriction::PinMessages),
- PeerFlagsValue(
- megagroup,
- ChannelDataFlag::Username | ChannelDataFlag::Location),
- megagroup->restrictionsValue()
- ) | rpl::map([=](
- bool adminRightAllows,
- bool defaultRestriction,
- ChannelDataFlags usernameOrLocation,
- Data::Flags<ChatRestrictions>::Change restrictions) {
- return adminRightAllows
- || (!usernameOrLocation
- && !defaultRestriction
- && !(restrictions.value & ChatRestriction::PinMessages));
- });
- } else if (const auto channel = peer->asChannel()) {
- if (channel->amCreator()) {
- return rpl::single(true);
- }
- return AdminRightValue(channel, ChatAdminRight::EditMessages);
- }
- Unexpected("Peer type in CanPinMessagesValue.");
- }
- rpl::producer<bool> CanManageGroupCallValue(not_null<PeerData*> peer) {
- const auto flag = ChatAdminRight::ManageCall;
- if (const auto chat = peer->asChat()) {
- return chat->amCreator()
- ? (rpl::single(true) | rpl::type_erased())
- : AdminRightValue(chat, flag);
- } else if (const auto channel = peer->asChannel()) {
- return channel->amCreator()
- ? (rpl::single(true) | rpl::type_erased())
- : AdminRightValue(channel, flag);
- }
- return rpl::single(false);
- }
- rpl::producer<bool> PeerPremiumValue(not_null<PeerData*> peer) {
- const auto user = peer->asUser();
- if (!user) {
- return rpl::single(false);
- }
- return user->flagsValue(
- ) | rpl::filter([=](UserData::Flags::Change change) {
- return (change.diff & UserDataFlag::Premium);
- }) | rpl::map([=] {
- return user->isPremium();
- });
- }
- rpl::producer<bool> AmPremiumValue(not_null<Main::Session*> session) {
- return PeerPremiumValue(session->user());
- }
- TimeId SortByOnlineValue(not_null<UserData*> user, TimeId now) {
- if (user->isServiceUser() || user->isBot()) {
- return -1;
- }
- const auto lastseen = user->lastseen();
- if (const auto till = lastseen.onlineTill()) {
- return till;
- } else if (lastseen.isRecently()) {
- return now - 3 * kSecondsInDay;
- } else if (lastseen.isWithinWeek()) {
- return now - 7 * kSecondsInDay;
- } else if (lastseen.isWithinMonth()) {
- return now - 30 * kSecondsInDay;
- } else {
- return 0;
- }
- }
- crl::time OnlineChangeTimeout(Data::LastseenStatus status, TimeId now) {
- const auto result = OnlinePhraseChangeInSeconds(status, now);
- Assert(result >= 0);
- return std::clamp(
- result * crl::time(1000),
- kMinOnlineChangeTimeout,
- kMaxOnlineChangeTimeout);
- }
- crl::time OnlineChangeTimeout(not_null<UserData*> user, TimeId now) {
- if (user->isServiceUser() || user->isBot()) {
- return kMaxOnlineChangeTimeout;
- }
- return OnlineChangeTimeout(user->lastseen(), now);
- }
- QString OnlineText(Data::LastseenStatus status, TimeId now) {
- if (const auto common = OnlineTextCommon(status, now)) {
- return *common;
- }
- const auto till = status.onlineTill();
- Assert(till > 0);
- const auto minutes = (now - till) / 60;
- if (!minutes) {
- return tr::lng_status_lastseen_now(tr::now);
- } else if (minutes < 60) {
- return tr::lng_status_lastseen_minutes(tr::now, lt_count, minutes);
- }
- const auto hours = (now - till) / 3600;
- if (hours < 12) {
- return tr::lng_status_lastseen_hours(tr::now, lt_count, hours);
- }
- const auto onlineFull = base::unixtime::parse(till);
- const auto nowFull = base::unixtime::parse(now);
- const auto locale = QLocale();
- if (onlineFull.date() == nowFull.date()) {
- const auto onlineTime = locale.toString(onlineFull.time(), QLocale::ShortFormat);
- return tr::lng_status_lastseen_today(tr::now, lt_time, onlineTime);
- } else if (onlineFull.date().addDays(1) == nowFull.date()) {
- const auto onlineTime = locale.toString(onlineFull.time(), QLocale::ShortFormat);
- return tr::lng_status_lastseen_yesterday(tr::now, lt_time, onlineTime);
- }
- const auto date = locale.toString(onlineFull.date(), QLocale::ShortFormat);
- return tr::lng_status_lastseen_date(tr::now, lt_date, date);
- }
- QString OnlineText(not_null<UserData*> user, TimeId now) {
- if (const auto special = OnlineTextSpecial(user)) {
- return *special;
- }
- return OnlineText(user->lastseen(), now);
- }
- QString OnlineTextFull(not_null<UserData*> user, TimeId now) {
- if (const auto special = OnlineTextSpecial(user)) {
- return *special;
- } else if (const auto common = OnlineTextCommon(user->lastseen(), now)) {
- return *common;
- }
- const auto till = user->lastseen().onlineTill();
- const auto onlineFull = base::unixtime::parse(till);
- const auto nowFull = base::unixtime::parse(now);
- const auto locale = QLocale();
- if (onlineFull.date() == nowFull.date()) {
- const auto onlineTime = locale.toString(onlineFull.time(), QLocale::ShortFormat);
- return tr::lng_status_lastseen_today(tr::now, lt_time, onlineTime);
- } else if (onlineFull.date().addDays(1) == nowFull.date()) {
- const auto onlineTime = locale.toString(onlineFull.time(), QLocale::ShortFormat);
- return tr::lng_status_lastseen_yesterday(tr::now, lt_time, onlineTime);
- }
- const auto date = locale.toString(onlineFull.date(), QLocale::ShortFormat);
- const auto time = locale.toString(onlineFull.time(), QLocale::ShortFormat);
- return tr::lng_status_lastseen_date_time(tr::now, lt_date, date, lt_time, time);
- }
- bool OnlineTextActive(not_null<UserData*> user, TimeId now) {
- return !user->isServiceUser()
- && !user->isBot()
- && user->lastseen().isOnline(now);
- }
- bool IsUserOnline(not_null<UserData*> user, TimeId now) {
- if (!now) {
- now = base::unixtime::now();
- }
- return OnlineTextActive(user, now);
- }
- bool ChannelHasActiveCall(not_null<ChannelData*> channel) {
- return (channel->flags() & ChannelDataFlag::CallNotEmpty);
- }
- bool ChannelHasSubscriptionUntilDate(ChannelData *channel) {
- return channel && channel->subscriptionUntilDate() > 0;
- }
- rpl::producer<QImage> PeerUserpicImageValue(
- not_null<PeerData*> peer,
- int size,
- std::optional<int> radius) {
- return [=](auto consumer) {
- auto result = rpl::lifetime();
- struct State {
- Ui::PeerUserpicView view;
- rpl::lifetime waiting;
- InMemoryKey key = {};
- bool empty = true;
- Fn<void()> push;
- };
- const auto state = result.make_state<State>();
- state->push = [=] {
- const auto key = peer->userpicUniqueKey(state->view);
- const auto loading = Ui::PeerUserpicLoading(state->view);
- if (loading && !state->waiting) {
- peer->session().downloaderTaskFinished(
- ) | rpl::start_with_next(state->push, state->waiting);
- } else if (!loading && state->waiting) {
- state->waiting.destroy();
- }
- if (!state->empty && (loading || key == state->key)) {
- return;
- }
- state->key = key;
- state->empty = false;
- consumer.put_next(
- PeerData::GenerateUserpicImage(
- peer,
- state->view,
- size,
- radius));
- };
- peer->session().changes().peerFlagsValue(
- peer,
- PeerUpdate::Flag::Photo
- ) | rpl::start_with_next(state->push, result);
- return result;
- };
- }
- const AllowedReactions &PeerAllowedReactions(not_null<PeerData*> peer) {
- if (const auto chat = peer->asChat()) {
- return chat->allowedReactions();
- } else if (const auto channel = peer->asChannel()) {
- return channel->allowedReactions();
- } else {
- static const auto result = AllowedReactions{
- .type = AllowedReactionsType::All,
- };
- return result;
- }
- }
- rpl::producer<AllowedReactions> PeerAllowedReactionsValue(
- not_null<PeerData*> peer) {
- return peer->session().changes().peerFlagsValue(
- peer,
- Data::PeerUpdate::Flag::Reactions
- ) | rpl::map([=]{
- return PeerAllowedReactions(peer);
- });
- }
- int UniqueReactionsLimit(not_null<PeerData*> peer) {
- if (const auto channel = peer->asChannel()) {
- if (const auto limit = channel->allowedReactions().maxCount) {
- return limit;
- }
- } else if (const auto chat = peer->asChat()) {
- if (const auto limit = chat->allowedReactions().maxCount) {
- return limit;
- }
- }
- return UniqueReactionsLimit(&peer->session().appConfig());
- }
- rpl::producer<int> UniqueReactionsLimitValue(
- not_null<PeerData*> peer) {
- auto configValue = peer->session().appConfig().value(
- ) | rpl::map([config = &peer->session().appConfig()] {
- return UniqueReactionsLimit(config);
- }) | rpl::distinct_until_changed();
- if (const auto channel = peer->asChannel()) {
- return rpl::combine(
- PeerAllowedReactionsValue(peer),
- std::move(configValue)
- ) | rpl::map([=](const auto &allowedReactions, int limit) {
- return allowedReactions.maxCount
- ? allowedReactions.maxCount
- : limit;
- });
- } else if (const auto chat = peer->asChat()) {
- return rpl::combine(
- PeerAllowedReactionsValue(peer),
- std::move(configValue)
- ) | rpl::map([=](const auto &allowedReactions, int limit) {
- return allowedReactions.maxCount
- ? allowedReactions.maxCount
- : limit;
- });
- }
- return configValue;
- }
- } // namespace Data
|