calls_group_invite_controller.cpp 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311
  1. /*
  2. This file is part of Telegram Desktop,
  3. the official desktop application for the Telegram messaging service.
  4. For license and copyright information please follow this link:
  5. https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
  6. */
  7. #include "calls/group/calls_group_invite_controller.h"
  8. #include "api/api_chat_participants.h"
  9. #include "calls/group/calls_group_call.h"
  10. #include "calls/group/calls_group_menu.h"
  11. #include "boxes/peer_lists_box.h"
  12. #include "data/data_user.h"
  13. #include "data/data_channel.h"
  14. #include "data/data_session.h"
  15. #include "data/data_group_call.h"
  16. #include "main/main_session.h"
  17. #include "ui/text/text_utilities.h"
  18. #include "ui/layers/generic_box.h"
  19. #include "ui/widgets/labels.h"
  20. #include "apiwrap.h"
  21. #include "lang/lang_keys.h"
  22. #include "styles/style_calls.h"
  23. #include "styles/style_dialogs.h" // searchedBarHeight
  24. namespace Calls::Group {
  25. namespace {
  26. [[nodiscard]] object_ptr<Ui::RpWidget> CreateSectionSubtitle(
  27. QWidget *parent,
  28. rpl::producer<QString> text) {
  29. auto result = object_ptr<Ui::FixedHeightWidget>(
  30. parent,
  31. st::searchedBarHeight);
  32. const auto raw = result.data();
  33. raw->paintRequest(
  34. ) | rpl::start_with_next([=](QRect clip) {
  35. auto p = QPainter(raw);
  36. p.fillRect(clip, st::groupCallMembersBgOver);
  37. }, raw->lifetime());
  38. const auto label = Ui::CreateChild<Ui::FlatLabel>(
  39. raw,
  40. std::move(text),
  41. st::groupCallBoxLabel);
  42. raw->widthValue(
  43. ) | rpl::start_with_next([=](int width) {
  44. const auto padding = st::groupCallInviteDividerPadding;
  45. const auto available = width - padding.left() - padding.right();
  46. label->resizeToNaturalWidth(available);
  47. label->moveToLeft(padding.left(), padding.top(), width);
  48. }, label->lifetime());
  49. return result;
  50. }
  51. } // namespace
  52. InviteController::InviteController(
  53. not_null<PeerData*> peer,
  54. base::flat_set<not_null<UserData*>> alreadyIn)
  55. : ParticipantsBoxController(CreateTag{}, nullptr, peer, Role::Members)
  56. , _peer(peer)
  57. , _alreadyIn(std::move(alreadyIn)) {
  58. SubscribeToMigration(
  59. _peer,
  60. lifetime(),
  61. [=](not_null<ChannelData*> channel) { _peer = channel; });
  62. }
  63. void InviteController::prepare() {
  64. delegate()->peerListSetHideEmpty(true);
  65. ParticipantsBoxController::prepare();
  66. delegate()->peerListSetAboveWidget(CreateSectionSubtitle(
  67. nullptr,
  68. tr::lng_group_call_invite_members()));
  69. delegate()->peerListSetAboveSearchWidget(CreateSectionSubtitle(
  70. nullptr,
  71. tr::lng_group_call_invite_members()));
  72. }
  73. void InviteController::rowClicked(not_null<PeerListRow*> row) {
  74. delegate()->peerListSetRowChecked(row, !row->checked());
  75. }
  76. base::unique_qptr<Ui::PopupMenu> InviteController::rowContextMenu(
  77. QWidget *parent,
  78. not_null<PeerListRow*> row) {
  79. return nullptr;
  80. }
  81. void InviteController::itemDeselectedHook(not_null<PeerData*> peer) {
  82. }
  83. bool InviteController::hasRowFor(not_null<PeerData*> peer) const {
  84. return (delegate()->peerListFindRow(peer->id.value) != nullptr);
  85. }
  86. bool InviteController::isAlreadyIn(not_null<UserData*> user) const {
  87. return _alreadyIn.contains(user);
  88. }
  89. std::unique_ptr<PeerListRow> InviteController::createRow(
  90. not_null<PeerData*> participant) const {
  91. const auto user = participant->asUser();
  92. if (!user
  93. || user->isSelf()
  94. || user->isBot()
  95. || user->isInaccessible()) {
  96. return nullptr;
  97. }
  98. auto result = std::make_unique<PeerListRow>(user);
  99. _rowAdded.fire_copy(user);
  100. _inGroup.emplace(user);
  101. if (isAlreadyIn(user)) {
  102. result->setDisabledState(PeerListRow::State::DisabledChecked);
  103. }
  104. return result;
  105. }
  106. auto InviteController::peersWithRows() const
  107. -> not_null<const base::flat_set<not_null<UserData*>>*> {
  108. return &_inGroup;
  109. }
  110. rpl::producer<not_null<UserData*>> InviteController::rowAdded() const {
  111. return _rowAdded.events();
  112. }
  113. InviteContactsController::InviteContactsController(
  114. not_null<PeerData*> peer,
  115. base::flat_set<not_null<UserData*>> alreadyIn,
  116. not_null<const base::flat_set<not_null<UserData*>>*> inGroup,
  117. rpl::producer<not_null<UserData*>> discoveredInGroup)
  118. : AddParticipantsBoxController(peer, std::move(alreadyIn))
  119. , _inGroup(inGroup)
  120. , _discoveredInGroup(std::move(discoveredInGroup)) {
  121. }
  122. void InviteContactsController::prepareViewHook() {
  123. AddParticipantsBoxController::prepareViewHook();
  124. delegate()->peerListSetAboveWidget(CreateSectionSubtitle(
  125. nullptr,
  126. tr::lng_contacts_header()));
  127. delegate()->peerListSetAboveSearchWidget(CreateSectionSubtitle(
  128. nullptr,
  129. tr::lng_group_call_invite_search_results()));
  130. std::move(
  131. _discoveredInGroup
  132. ) | rpl::start_with_next([=](not_null<UserData*> user) {
  133. if (auto row = delegate()->peerListFindRow(user->id.value)) {
  134. delegate()->peerListRemoveRow(row);
  135. }
  136. }, _lifetime);
  137. }
  138. std::unique_ptr<PeerListRow> InviteContactsController::createRow(
  139. not_null<UserData*> user) {
  140. return _inGroup->contains(user)
  141. ? nullptr
  142. : AddParticipantsBoxController::createRow(user);
  143. }
  144. object_ptr<Ui::BoxContent> PrepareInviteBox(
  145. not_null<GroupCall*> call,
  146. Fn<void(TextWithEntities&&)> showToast) {
  147. const auto real = call->lookupReal();
  148. if (!real) {
  149. return nullptr;
  150. }
  151. const auto peer = call->peer();
  152. auto alreadyIn = peer->owner().invitedToCallUsers(real->id());
  153. for (const auto &participant : real->participants()) {
  154. if (const auto user = participant.peer->asUser()) {
  155. alreadyIn.emplace(user);
  156. }
  157. }
  158. alreadyIn.emplace(peer->session().user());
  159. auto controller = std::make_unique<InviteController>(peer, alreadyIn);
  160. controller->setStyleOverrides(
  161. &st::groupCallInviteMembersList,
  162. &st::groupCallMultiSelect);
  163. auto contactsController = std::make_unique<InviteContactsController>(
  164. peer,
  165. std::move(alreadyIn),
  166. controller->peersWithRows(),
  167. controller->rowAdded());
  168. contactsController->setStyleOverrides(
  169. &st::groupCallInviteMembersList,
  170. &st::groupCallMultiSelect);
  171. const auto weak = base::make_weak(call);
  172. const auto invite = [=](const std::vector<not_null<UserData*>> &users) {
  173. const auto call = weak.get();
  174. if (!call) {
  175. return;
  176. }
  177. const auto result = call->inviteUsers(users);
  178. if (const auto user = std::get_if<not_null<UserData*>>(&result)) {
  179. showToast(tr::lng_group_call_invite_done_user(
  180. tr::now,
  181. lt_user,
  182. Ui::Text::Bold((*user)->firstName),
  183. Ui::Text::WithEntities));
  184. } else if (const auto count = std::get_if<int>(&result)) {
  185. if (*count > 0) {
  186. showToast(tr::lng_group_call_invite_done_many(
  187. tr::now,
  188. lt_count,
  189. *count,
  190. Ui::Text::RichLangValue));
  191. }
  192. } else {
  193. Unexpected("Result in GroupCall::inviteUsers.");
  194. }
  195. };
  196. const auto inviteWithAdd = [=](
  197. std::shared_ptr<Ui::Show> show,
  198. const std::vector<not_null<UserData*>> &users,
  199. const std::vector<not_null<UserData*>> &nonMembers,
  200. Fn<void()> finish) {
  201. peer->session().api().chatParticipants().add(
  202. show,
  203. peer,
  204. nonMembers,
  205. true,
  206. [=](bool) { invite(users); finish(); });
  207. };
  208. const auto inviteWithConfirmation = [=](
  209. not_null<PeerListsBox*> parentBox,
  210. const std::vector<not_null<UserData*>> &users,
  211. const std::vector<not_null<UserData*>> &nonMembers,
  212. Fn<void()> finish) {
  213. if (nonMembers.empty()) {
  214. invite(users);
  215. finish();
  216. return;
  217. }
  218. const auto name = peer->name();
  219. const auto text = (nonMembers.size() == 1)
  220. ? tr::lng_group_call_add_to_group_one(
  221. tr::now,
  222. lt_user,
  223. nonMembers.front()->shortName(),
  224. lt_group,
  225. name)
  226. : (nonMembers.size() < users.size())
  227. ? tr::lng_group_call_add_to_group_some(tr::now, lt_group, name)
  228. : tr::lng_group_call_add_to_group_all(tr::now, lt_group, name);
  229. const auto shared = std::make_shared<QPointer<Ui::GenericBox>>();
  230. const auto finishWithConfirm = [=] {
  231. if (*shared) {
  232. (*shared)->closeBox();
  233. }
  234. finish();
  235. };
  236. const auto done = [=] {
  237. const auto show = (*shared) ? (*shared)->uiShow() : nullptr;
  238. inviteWithAdd(show, users, nonMembers, finishWithConfirm);
  239. };
  240. auto box = ConfirmBox({
  241. .text = text,
  242. .confirmed = done,
  243. .confirmText = tr::lng_participant_invite(),
  244. });
  245. *shared = box.data();
  246. parentBox->getDelegate()->showBox(
  247. std::move(box),
  248. Ui::LayerOption::KeepOther,
  249. anim::type::normal);
  250. };
  251. auto initBox = [=, controller = controller.get()](
  252. not_null<PeerListsBox*> box) {
  253. box->setTitle(tr::lng_group_call_invite_title());
  254. box->addButton(tr::lng_group_call_invite_button(), [=] {
  255. const auto rows = box->collectSelectedRows();
  256. const auto users = ranges::views::all(
  257. rows
  258. ) | ranges::views::transform([](not_null<PeerData*> peer) {
  259. return not_null<UserData*>(peer->asUser());
  260. }) | ranges::to_vector;
  261. const auto nonMembers = ranges::views::all(
  262. users
  263. ) | ranges::views::filter([&](not_null<UserData*> user) {
  264. return !controller->hasRowFor(user);
  265. }) | ranges::to_vector;
  266. const auto finish = [box = Ui::MakeWeak(box)]() {
  267. if (box) {
  268. box->closeBox();
  269. }
  270. };
  271. inviteWithConfirmation(box, users, nonMembers, finish);
  272. });
  273. box->addButton(tr::lng_cancel(), [=] { box->closeBox(); });
  274. };
  275. auto controllers = std::vector<std::unique_ptr<PeerListController>>();
  276. controllers.push_back(std::move(controller));
  277. controllers.push_back(std::move(contactsController));
  278. return Box<PeerListsBox>(std::move(controllers), initBox);
  279. }
  280. } // namespace Calls::Group