// // Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2024 // // Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) // #pragma once #include "td/telegram/DialogFilterId.h" #include "td/telegram/FolderId.h" #include "td/telegram/td_api.h" #include "td/utils/common.h" #include "td/utils/HashTableUtils.h" #include "td/utils/StringBuilder.h" #include <limits> #include <type_traits> namespace td { class DialogListId { int64 id = 0; static constexpr int64 FILTER_ID_SHIFT = static_cast<int64>(1) << 32; public: DialogListId() = default; explicit DialogListId(int64 dialog_list_id) : id(dialog_list_id) { if (is_folder() && get_folder_id() != FolderId::archive()) { id = FolderId::main().get(); } else if (is_filter()) { CHECK(get_filter_id().is_valid()); } } template <class T, typename = std::enable_if_t<std::is_convertible<T, int32>::value>> DialogListId(T dialog_list_id) = delete; explicit DialogListId(DialogFilterId dialog_filter_id) : id(dialog_filter_id.get() + FILTER_ID_SHIFT) { } explicit DialogListId(FolderId folder_id) : id(folder_id.get()) { } explicit DialogListId(const td_api::object_ptr<td_api::ChatList> &chat_list) { if (chat_list == nullptr) { CHECK(id == FolderId::main().get()); return; } switch (chat_list->get_id()) { case td_api::chatListArchive::ID: id = FolderId::archive().get(); break; case td_api::chatListMain::ID: CHECK(id == FolderId::main().get()); break; case td_api::chatListFolder::ID: { DialogFilterId filter_id(static_cast<const td_api::chatListFolder *>(chat_list.get())->chat_folder_id_); if (filter_id.is_valid()) { *this = DialogListId(filter_id); } break; } default: UNREACHABLE(); break; } } td_api::object_ptr<td_api::ChatList> get_chat_list_object() const { if (is_folder()) { auto folder_id = get_folder_id(); if (folder_id == FolderId::archive()) { return td_api::make_object<td_api::chatListArchive>(); } return td_api::make_object<td_api::chatListMain>(); } if (is_filter()) { return td_api::make_object<td_api::chatListFolder>(get_filter_id().get()); } UNREACHABLE(); return nullptr; } int64 get() const { return id; } bool operator==(const DialogListId &other) const { return id == other.id; } bool operator!=(const DialogListId &other) const { return id != other.id; } bool is_folder() const { return std::numeric_limits<int32>::min() <= id && id <= std::numeric_limits<int32>::max(); } bool is_filter() const { return std::numeric_limits<int32>::min() + FILTER_ID_SHIFT <= id && id <= std::numeric_limits<int32>::max() + FILTER_ID_SHIFT; } FolderId get_folder_id() const { CHECK(is_folder()); return FolderId(static_cast<int32>(id)); } DialogFilterId get_filter_id() const { CHECK(is_filter()); return DialogFilterId(static_cast<int32>(id - FILTER_ID_SHIFT)); } }; struct DialogListIdHash { uint32 operator()(DialogListId dialog_list_id) const { return Hash<int64>()(dialog_list_id.get()); } }; inline StringBuilder &operator<<(StringBuilder &string_builder, DialogListId dialog_list_id) { if (dialog_list_id.is_folder()) { auto folder_id = dialog_list_id.get_folder_id(); if (folder_id == FolderId::archive()) { return string_builder << "Archive chat list"; } if (folder_id == FolderId::main()) { return string_builder << "Main chat list"; } return string_builder << "chat list " << folder_id; } if (dialog_list_id.is_filter()) { return string_builder << "chat list " << dialog_list_id.get_filter_id(); } return string_builder << "unknown chat list " << dialog_list_id.get(); } } // namespace td