mirror of https://github.com/procxx/kepka.git
Edit admin and restricted rights in channels.
This commit is contained in:
parent
513a9f8d45
commit
7d2d5c6100
|
@ -140,6 +140,8 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
||||||
"lng_error_pinned_max#one" = "Sorry, you can pin no more than {count} chat to the top.";
|
"lng_error_pinned_max#one" = "Sorry, you can pin no more than {count} chat to the top.";
|
||||||
"lng_error_pinned_max#other" = "Sorry, you can pin no more than {count} chats to the top.";
|
"lng_error_pinned_max#other" = "Sorry, you can pin no more than {count} chats to the top.";
|
||||||
"lng_error_public_groups_denied" = "Unfortunately, you were banned from participating in public groups.\n{more_info}";
|
"lng_error_public_groups_denied" = "Unfortunately, you were banned from participating in public groups.\n{more_info}";
|
||||||
|
"lng_error_cant_edit_admin" = "Sorry, you can't edit permissions for this admin.";
|
||||||
|
"lng_error_cant_add_member" = "Sorry, you can't add the bot to this group.";
|
||||||
|
|
||||||
"lng_edit_deleted" = "This message was deleted";
|
"lng_edit_deleted" = "This message was deleted";
|
||||||
"lng_edit_too_long" = "Your message text is too long";
|
"lng_edit_too_long" = "Your message text is too long";
|
||||||
|
@ -577,6 +579,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
||||||
"lng_profile_delete_and_exit" = "Leave";
|
"lng_profile_delete_and_exit" = "Leave";
|
||||||
"lng_profile_kick" = "Remove";
|
"lng_profile_kick" = "Remove";
|
||||||
"lng_profile_admin" = "admin";
|
"lng_profile_admin" = "admin";
|
||||||
|
"lng_profile_edit_admin" = "Edit";
|
||||||
"lng_profile_sure_kick" = "Remove {user} from the group?";
|
"lng_profile_sure_kick" = "Remove {user} from the group?";
|
||||||
"lng_profile_sure_kick_channel" = "Remove {user} from the channel?";
|
"lng_profile_sure_kick_channel" = "Remove {user} from the channel?";
|
||||||
"lng_profile_sure_kick_admin" = "Remove {user} from administrators?";
|
"lng_profile_sure_kick_admin" = "Remove {user} from administrators?";
|
||||||
|
@ -993,7 +996,8 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
||||||
"lng_context_unpin_from_top" = "Unpin from top";
|
"lng_context_unpin_from_top" = "Unpin from top";
|
||||||
|
|
||||||
"lng_context_promote_admin" = "Promote to admin";
|
"lng_context_promote_admin" = "Promote to admin";
|
||||||
"lng_context_remove_admin" = "Remove from admins";
|
"lng_context_edit_permissions" = "Edit permissions";
|
||||||
|
"lng_context_restrict_user" = "Restrict user";
|
||||||
"lng_context_remove_from_group" = "Remove from group";
|
"lng_context_remove_from_group" = "Remove from group";
|
||||||
|
|
||||||
"lng_context_copy_link" = "Copy Link";
|
"lng_context_copy_link" = "Copy Link";
|
||||||
|
@ -1246,7 +1250,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
||||||
|
|
||||||
"lng_rights_edit_admin" = "Edit administrator";
|
"lng_rights_edit_admin" = "Edit administrator";
|
||||||
"lng_rights_edit_admin_header" = "What can this admin do?";
|
"lng_rights_edit_admin_header" = "What can this admin do?";
|
||||||
"lng_rights_about_add_admins_yes" = "This admin will be able to add new admins with the same (or more limited) permissions than they have.";
|
"lng_rights_about_add_admins_yes" = "This admin will be able to add new admins with the same (or more limited) permissions.";
|
||||||
"lng_rights_about_add_admins_no" = "This admin will not be able to add new admins.";
|
"lng_rights_about_add_admins_no" = "This admin will not be able to add new admins.";
|
||||||
"lng_rights_user_restrictions" = "User restrictions";
|
"lng_rights_user_restrictions" = "User restrictions";
|
||||||
"lng_rights_user_restrictions_header" = "What can this user do?";
|
"lng_rights_user_restrictions_header" = "What can this user do?";
|
||||||
|
@ -1254,12 +1258,13 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
||||||
"lng_rights_channel_info" = "Change channel info";
|
"lng_rights_channel_info" = "Change channel info";
|
||||||
"lng_rights_channel_post" = "Post messages";
|
"lng_rights_channel_post" = "Post messages";
|
||||||
"lng_rights_channel_edit" = "Edit messages of others";
|
"lng_rights_channel_edit" = "Edit messages of others";
|
||||||
|
"lng_rights_channel_delete" = "Delete messages of others";
|
||||||
"lng_rights_group_info" = "Change group info";
|
"lng_rights_group_info" = "Change group info";
|
||||||
"lng_rights_group_ban" = "Ban users";
|
"lng_rights_group_ban" = "Ban users";
|
||||||
"lng_rights_group_invite" = "Add users";
|
"lng_rights_group_invite" = "Add users";
|
||||||
"lng_rights_group_invite_link" = "Invite users via link";
|
//"lng_rights_group_invite_link" = "Invite users via link";
|
||||||
"lng_rights_group_pin" = "Pin messages";
|
"lng_rights_group_pin" = "Pin messages";
|
||||||
"lng_rights_delete" = "Delete messages of others";
|
"lng_rights_group_delete" = "Delete messages";
|
||||||
"lng_rights_add_admins" = "Add new admins";
|
"lng_rights_add_admins" = "Add new admins";
|
||||||
"lng_rights_chat_read" = "Read messages";
|
"lng_rights_chat_read" = "Read messages";
|
||||||
"lng_rights_chat_send_text" = "Send messages";
|
"lng_rights_chat_send_text" = "Send messages";
|
||||||
|
|
|
@ -317,6 +317,7 @@ void ApiWrap::gotChatFull(PeerData *peer, const MTPmessages_ChatFull &result, mt
|
||||||
channel->setAbout(qs(f.vabout));
|
channel->setAbout(qs(f.vabout));
|
||||||
channel->setMembersCount(f.has_participants_count() ? f.vparticipants_count.v : 0);
|
channel->setMembersCount(f.has_participants_count() ? f.vparticipants_count.v : 0);
|
||||||
channel->setAdminsCount(f.has_admins_count() ? f.vadmins_count.v : 0);
|
channel->setAdminsCount(f.has_admins_count() ? f.vadmins_count.v : 0);
|
||||||
|
channel->setRestrictedCount(f.has_banned_count() ? f.vbanned_count.v : 0);
|
||||||
channel->setKickedCount(f.has_kicked_count() ? f.vkicked_count.v : 0);
|
channel->setKickedCount(f.has_kicked_count() ? f.vkicked_count.v : 0);
|
||||||
channel->setInviteLink((f.vexported_invite.type() == mtpc_chatInviteExported) ? qs(f.vexported_invite.c_chatInviteExported().vlink) : QString());
|
channel->setInviteLink((f.vexported_invite.type() == mtpc_chatInviteExported) ? qs(f.vexported_invite.c_chatInviteExported().vlink) : QString());
|
||||||
if (auto h = App::historyLoaded(channel->id)) {
|
if (auto h = App::historyLoaded(channel->id)) {
|
||||||
|
@ -335,8 +336,8 @@ void ApiWrap::gotChatFull(PeerData *peer, const MTPmessages_ChatFull &result, mt
|
||||||
}
|
}
|
||||||
channel->fullUpdated();
|
channel->fullUpdated();
|
||||||
|
|
||||||
if (canViewAdmins != channel->canViewAdmins()) Notify::peerUpdatedDelayed(channel, Notify::PeerUpdate::Flag::ChannelCanViewAdmins);
|
if (canViewAdmins != channel->canViewAdmins()
|
||||||
if (canViewMembers != channel->canViewMembers()) Notify::peerUpdatedDelayed(channel, Notify::PeerUpdate::Flag::ChannelCanViewMembers);
|
|| canViewMembers != channel->canViewMembers()) Notify::peerUpdatedDelayed(channel, Notify::PeerUpdate::Flag::ChannelRightsChanged);
|
||||||
|
|
||||||
notifySettingReceived(MTP_inputNotifyPeer(peer->input), f.vnotify_settings);
|
notifySettingReceived(MTP_inputNotifyPeer(peer->input), f.vnotify_settings);
|
||||||
}
|
}
|
||||||
|
@ -553,16 +554,26 @@ void ApiWrap::lastParticipantsDone(ChannelData *peer, const MTPchannels_ChannelP
|
||||||
auto needBotsInfos = false;
|
auto needBotsInfos = false;
|
||||||
auto botStatus = peer->mgInfo->botStatus;
|
auto botStatus = peer->mgInfo->botStatus;
|
||||||
auto keyboardBotFound = !h || !h->lastKeyboardFrom;
|
auto keyboardBotFound = !h || !h->lastKeyboardFrom;
|
||||||
auto emptyRights = MTP_channelAdminRights(MTP_flags(0));
|
auto emptyAdminRights = MTP_channelAdminRights(MTP_flags(0));
|
||||||
|
auto emptyRestrictedRights = MTP_channelBannedRights(MTP_flags(0), MTP_int(0));
|
||||||
for_const (auto &participant, v) {
|
for_const (auto &participant, v) {
|
||||||
auto userId = UserId(0);
|
auto userId = UserId(0);
|
||||||
auto rights = emptyRights;
|
auto adminCanEdit = false;
|
||||||
|
auto adminRights = emptyAdminRights;
|
||||||
|
auto restrictedRights = emptyRestrictedRights;
|
||||||
|
|
||||||
switch (participant.type()) {
|
switch (participant.type()) {
|
||||||
case mtpc_channelParticipant: userId = participant.c_channelParticipant().vuser_id.v; break;
|
case mtpc_channelParticipant: userId = participant.c_channelParticipant().vuser_id.v; break;
|
||||||
case mtpc_channelParticipantSelf: userId = participant.c_channelParticipantSelf().vuser_id.v; break;
|
case mtpc_channelParticipantSelf: userId = participant.c_channelParticipantSelf().vuser_id.v; break;
|
||||||
case mtpc_channelParticipantAdmin: userId = participant.c_channelParticipantAdmin().vuser_id.v; rights = participant.c_channelParticipantAdmin().vadmin_rights; break;
|
case mtpc_channelParticipantAdmin:
|
||||||
case mtpc_channelParticipantBanned: userId = participant.c_channelParticipantBanned().vuser_id.v; break;
|
userId = participant.c_channelParticipantAdmin().vuser_id.v;
|
||||||
|
adminCanEdit = participant.c_channelParticipantAdmin().is_can_edit();
|
||||||
|
adminRights = participant.c_channelParticipantAdmin().vadmin_rights;
|
||||||
|
break;
|
||||||
|
case mtpc_channelParticipantBanned:
|
||||||
|
userId = participant.c_channelParticipantBanned().vuser_id.v;
|
||||||
|
restrictedRights = participant.c_channelParticipantBanned().vbanned_rights;
|
||||||
|
break;
|
||||||
case mtpc_channelParticipantCreator: userId = participant.c_channelParticipantCreator().vuser_id.v; break;
|
case mtpc_channelParticipantCreator: userId = participant.c_channelParticipantCreator().vuser_id.v; break;
|
||||||
}
|
}
|
||||||
if (!userId) {
|
if (!userId) {
|
||||||
|
@ -570,6 +581,9 @@ void ApiWrap::lastParticipantsDone(ChannelData *peer, const MTPchannels_ChannelP
|
||||||
}
|
}
|
||||||
|
|
||||||
auto u = App::user(userId);
|
auto u = App::user(userId);
|
||||||
|
if (participant.type() == mtpc_channelParticipantCreator) {
|
||||||
|
peer->mgInfo->creator = u;
|
||||||
|
}
|
||||||
if (bots) {
|
if (bots) {
|
||||||
if (u->botInfo) {
|
if (u->botInfo) {
|
||||||
peer->mgInfo->bots.insert(u);
|
peer->mgInfo->bots.insert(u);
|
||||||
|
@ -584,8 +598,10 @@ void ApiWrap::lastParticipantsDone(ChannelData *peer, const MTPchannels_ChannelP
|
||||||
} else {
|
} else {
|
||||||
if (peer->mgInfo->lastParticipants.indexOf(u) < 0) {
|
if (peer->mgInfo->lastParticipants.indexOf(u) < 0) {
|
||||||
peer->mgInfo->lastParticipants.push_back(u);
|
peer->mgInfo->lastParticipants.push_back(u);
|
||||||
if (rights.c_channelAdminRights().vflags.v) {
|
if (adminRights.c_channelAdminRights().vflags.v) {
|
||||||
peer->mgInfo->lastAdmins.insert(u, rights);
|
peer->mgInfo->lastAdmins.insert(u, MegagroupInfo::Admin { adminRights, adminCanEdit });
|
||||||
|
} else if (restrictedRights.c_channelBannedRights().vflags.v != 0) {
|
||||||
|
peer->mgInfo->lastRestricted.insert(u, MegagroupInfo::Restricted { restrictedRights });
|
||||||
}
|
}
|
||||||
if (u->botInfo) {
|
if (u->botInfo) {
|
||||||
peer->mgInfo->bots.insert(u);
|
peer->mgInfo->bots.insert(u);
|
||||||
|
@ -647,6 +663,9 @@ void ApiWrap::requestSelfParticipant(ChannelData *channel) {
|
||||||
auto &d = p.vparticipant.c_channelParticipantCreator();
|
auto &d = p.vparticipant.c_channelParticipantCreator();
|
||||||
channel->inviter = _session->userId();
|
channel->inviter = _session->userId();
|
||||||
channel->inviteDate = date(MTP_int(channel->date));
|
channel->inviteDate = date(MTP_int(channel->date));
|
||||||
|
if (channel->mgInfo) {
|
||||||
|
channel->mgInfo->creator = App::self();
|
||||||
|
}
|
||||||
} break;
|
} break;
|
||||||
case mtpc_channelParticipantAdmin: {
|
case mtpc_channelParticipantAdmin: {
|
||||||
auto &d = p.vparticipant.c_channelParticipantAdmin();
|
auto &d = p.vparticipant.c_channelParticipantAdmin();
|
||||||
|
@ -671,44 +690,17 @@ void ApiWrap::kickParticipant(PeerData *peer, UserData *user) {
|
||||||
if (_kickRequests.contains(kick)) return;
|
if (_kickRequests.contains(kick)) return;
|
||||||
|
|
||||||
if (auto channel = peer->asChannel()) {
|
if (auto channel = peer->asChannel()) {
|
||||||
//auto requestId = request(MTPchannels_KickFromChannel(channel->inputChannel, user->inputUser, MTP_bool(true))).done([this, peer, user](const MTPUpdates &result) {
|
auto rights = ChannelData::KickedRestrictedRights();
|
||||||
// applyUpdates(result);
|
auto requestId = request(MTPchannels_EditBanned(channel->inputChannel, user->inputUser, rights)).done([this, channel, user, rights](const MTPUpdates &result) {
|
||||||
|
applyUpdates(result);
|
||||||
|
|
||||||
// _kickRequests.remove(KickRequest(peer, user));
|
_kickRequests.remove(KickRequest(channel, user));
|
||||||
// if (auto channel = peer->asMegagroup()) {
|
channel->applyEditBanned(user, rights);
|
||||||
// auto megagroupInfo = channel->mgInfo;
|
}).fail([this, kick](const RPCError &error) {
|
||||||
|
_kickRequests.remove(kick);
|
||||||
|
}).send();
|
||||||
|
|
||||||
// auto i = megagroupInfo->lastParticipants.indexOf(user);
|
_kickRequests.insert(kick, requestId);
|
||||||
// if (i >= 0) {
|
|
||||||
// megagroupInfo->lastParticipants.removeAt(i);
|
|
||||||
// }
|
|
||||||
|
|
||||||
// if (channel->membersCount() > 1) {
|
|
||||||
// channel->setMembersCount(channel->membersCount() - 1);
|
|
||||||
// } else {
|
|
||||||
// megagroupInfo->lastParticipantsStatus |= MegagroupInfo::LastParticipantsCountOutdated;
|
|
||||||
// megagroupInfo->lastParticipantsCount = 0;
|
|
||||||
// }
|
|
||||||
// channel->setKickedCount(channel->kickedCount() + 1);
|
|
||||||
// if (megagroupInfo->lastAdmins.contains(user)) {
|
|
||||||
// megagroupInfo->lastAdmins.remove(user);
|
|
||||||
// if (channel->adminsCount() > 1) {
|
|
||||||
// channel->setAdminsCount(channel->adminsCount() - 1);
|
|
||||||
// }
|
|
||||||
// Notify::peerUpdatedDelayed(channel, Notify::PeerUpdate::Flag::AdminsChanged);
|
|
||||||
// }
|
|
||||||
// megagroupInfo->bots.remove(user);
|
|
||||||
// if (megagroupInfo->bots.isEmpty() && megagroupInfo->botStatus > 0) {
|
|
||||||
// megagroupInfo->botStatus = -1;
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// Notify::peerUpdatedDelayed(peer, Notify::PeerUpdate::Flag::MembersChanged);
|
|
||||||
// fullPeerUpdated().notify(peer);
|
|
||||||
//}).fail([this, kick](const RPCError &error) {
|
|
||||||
// _kickRequests.remove(kick);
|
|
||||||
//}).send();
|
|
||||||
|
|
||||||
//_kickRequests.insert(kick, requestId);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -717,22 +709,22 @@ void ApiWrap::unblockParticipant(PeerData *peer, UserData *user) {
|
||||||
if (_kickRequests.contains(kick)) return;
|
if (_kickRequests.contains(kick)) return;
|
||||||
|
|
||||||
if (auto channel = peer->asChannel()) {
|
if (auto channel = peer->asChannel()) {
|
||||||
//auto requestId = request(MTPchannels_KickFromChannel(channel->inputChannel, user->inputUser, MTP_bool(false))).done([this, peer, user](const MTPUpdates &result) {
|
auto requestId = request(MTPchannels_EditBanned(channel->inputChannel, user->inputUser, MTP_channelBannedRights(MTP_flags(0), MTP_int(0)))).done([this, peer, user](const MTPUpdates &result) {
|
||||||
// applyUpdates(result);
|
applyUpdates(result);
|
||||||
|
|
||||||
// _kickRequests.remove(KickRequest(peer, user));
|
_kickRequests.remove(KickRequest(peer, user));
|
||||||
// if (auto channel = peer->asMegagroup()) {
|
if (auto channel = peer->asMegagroup()) {
|
||||||
// if (channel->kickedCount() > 0) {
|
if (channel->kickedCount() > 0) {
|
||||||
// channel->setKickedCount(channel->kickedCount() - 1);
|
channel->setKickedCount(channel->kickedCount() - 1);
|
||||||
// } else {
|
} else {
|
||||||
// channel->updateFull(true);
|
channel->updateFull(true);
|
||||||
// }
|
}
|
||||||
// }
|
}
|
||||||
//}).fail([this, kick](const RPCError &error) {
|
}).fail([this, kick](const RPCError &error) {
|
||||||
// _kickRequests.remove(kick);
|
_kickRequests.remove(kick);
|
||||||
//}).send();
|
}).send();
|
||||||
|
|
||||||
//_kickRequests.insert(kick, requestId);
|
_kickRequests.insert(kick, requestId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -681,11 +681,9 @@ namespace {
|
||||||
|
|
||||||
auto cdata = data->asChannel();
|
auto cdata = data->asChannel();
|
||||||
auto wasInChannel = cdata->amIn();
|
auto wasInChannel = cdata->amIn();
|
||||||
auto canEditInformation = cdata->canEditInformation();
|
|
||||||
auto canViewAdmins = cdata->canViewAdmins();
|
auto canViewAdmins = cdata->canViewAdmins();
|
||||||
auto canViewMembers = cdata->canViewMembers();
|
auto canViewMembers = cdata->canViewMembers();
|
||||||
auto canAddMembers = cdata->canAddMembers();
|
auto canAddMembers = cdata->canAddMembers();
|
||||||
auto wasEditor = cdata->amEditor();
|
|
||||||
|
|
||||||
if (minimal) {
|
if (minimal) {
|
||||||
auto mask = MTPDchannel::Flag::f_broadcast | MTPDchannel::Flag::f_verified | MTPDchannel::Flag::f_megagroup | MTPDchannel::Flag::f_democracy;
|
auto mask = MTPDchannel::Flag::f_broadcast | MTPDchannel::Flag::f_verified | MTPDchannel::Flag::f_megagroup | MTPDchannel::Flag::f_democracy;
|
||||||
|
@ -694,8 +692,8 @@ namespace {
|
||||||
if (d.has_admin_rights() || cdata->hasAdminRights()) {
|
if (d.has_admin_rights() || cdata->hasAdminRights()) {
|
||||||
cdata->setAdminRights(d.has_admin_rights() ? d.vadmin_rights : MTP_channelAdminRights(MTP_flags(0)));
|
cdata->setAdminRights(d.has_admin_rights() ? d.vadmin_rights : MTP_channelAdminRights(MTP_flags(0)));
|
||||||
}
|
}
|
||||||
if (d.has_banned_rights() || cdata->hasBannedRights()) {
|
if (d.has_banned_rights() || cdata->hasRestrictedRights()) {
|
||||||
cdata->setBannedRights(d.has_banned_rights() ? d.vbanned_rights : MTP_channelBannedRights(MTP_flags(0), MTP_int(0)));
|
cdata->setRestrictedRights(d.has_banned_rights() ? d.vbanned_rights : MTP_channelBannedRights(MTP_flags(0), MTP_int(0)));
|
||||||
}
|
}
|
||||||
cdata->inputChannel = MTP_inputChannel(d.vid, d.vaccess_hash);
|
cdata->inputChannel = MTP_inputChannel(d.vid, d.vaccess_hash);
|
||||||
cdata->access = d.vaccess_hash.v;
|
cdata->access = d.vaccess_hash.v;
|
||||||
|
@ -719,14 +717,9 @@ namespace {
|
||||||
cdata->setPhoto(d.vphoto);
|
cdata->setPhoto(d.vphoto);
|
||||||
|
|
||||||
if (wasInChannel != cdata->amIn()) update.flags |= UpdateFlag::ChannelAmIn;
|
if (wasInChannel != cdata->amIn()) update.flags |= UpdateFlag::ChannelAmIn;
|
||||||
if (canEditInformation != cdata->canEditInformation()) update.flags |= UpdateFlag::ChannelCanEditInformation;
|
if (canViewAdmins != cdata->canViewAdmins()
|
||||||
if (canViewAdmins != cdata->canViewAdmins()) update.flags |= UpdateFlag::ChannelCanViewAdmins;
|
|| canViewMembers != cdata->canViewMembers()
|
||||||
if (canViewMembers != cdata->canViewMembers()) update.flags |= UpdateFlag::ChannelCanViewMembers;
|
|| canAddMembers != cdata->canAddMembers()) update.flags |= UpdateFlag::ChannelRightsChanged;
|
||||||
if (canAddMembers != cdata->canAddMembers()) update.flags |= UpdateFlag::ChannelCanAddMembers;
|
|
||||||
if (wasEditor != cdata->amEditor()) {
|
|
||||||
cdata->selfAdminUpdated();
|
|
||||||
update.flags |= (UpdateFlag::ChannelAmEditor | UpdateFlag::AdminsChanged);
|
|
||||||
}
|
|
||||||
} break;
|
} break;
|
||||||
case mtpc_channelForbidden: {
|
case mtpc_channelForbidden: {
|
||||||
auto &d(chat.c_channelForbidden());
|
auto &d(chat.c_channelForbidden());
|
||||||
|
@ -737,11 +730,9 @@ namespace {
|
||||||
|
|
||||||
auto cdata = data->asChannel();
|
auto cdata = data->asChannel();
|
||||||
auto wasInChannel = cdata->amIn();
|
auto wasInChannel = cdata->amIn();
|
||||||
auto canEditInformation = cdata->canEditInformation();
|
|
||||||
auto canViewAdmins = cdata->canViewAdmins();
|
auto canViewAdmins = cdata->canViewAdmins();
|
||||||
auto canViewMembers = cdata->canViewMembers();
|
auto canViewMembers = cdata->canViewMembers();
|
||||||
auto canAddMembers = cdata->canAddMembers();
|
auto canAddMembers = cdata->canAddMembers();
|
||||||
auto wasEditor = cdata->amEditor();
|
|
||||||
|
|
||||||
cdata->inputChannel = MTP_inputChannel(d.vid, d.vaccess_hash);
|
cdata->inputChannel = MTP_inputChannel(d.vid, d.vaccess_hash);
|
||||||
|
|
||||||
|
@ -752,8 +743,8 @@ namespace {
|
||||||
if (cdata->hasAdminRights()) {
|
if (cdata->hasAdminRights()) {
|
||||||
cdata->setAdminRights(MTP_channelAdminRights(MTP_flags(0)));
|
cdata->setAdminRights(MTP_channelAdminRights(MTP_flags(0)));
|
||||||
}
|
}
|
||||||
if (cdata->hasBannedRights()) {
|
if (cdata->hasRestrictedRights()) {
|
||||||
cdata->setBannedRights(MTP_channelBannedRights(MTP_flags(0), MTP_int(0)));
|
cdata->setRestrictedRights(MTP_channelBannedRights(MTP_flags(0), MTP_int(0)));
|
||||||
}
|
}
|
||||||
|
|
||||||
cdata->setName(qs(d.vtitle), QString());
|
cdata->setName(qs(d.vtitle), QString());
|
||||||
|
@ -765,14 +756,9 @@ namespace {
|
||||||
cdata->setIsForbidden(true);
|
cdata->setIsForbidden(true);
|
||||||
|
|
||||||
if (wasInChannel != cdata->amIn()) update.flags |= UpdateFlag::ChannelAmIn;
|
if (wasInChannel != cdata->amIn()) update.flags |= UpdateFlag::ChannelAmIn;
|
||||||
if (canEditInformation != cdata->canEditInformation()) update.flags |= UpdateFlag::ChannelCanEditInformation;
|
if (canViewAdmins != cdata->canViewAdmins()
|
||||||
if (canViewAdmins != cdata->canViewAdmins()) update.flags |= UpdateFlag::ChannelCanViewAdmins;
|
|| canViewMembers != cdata->canViewMembers()
|
||||||
if (canViewMembers != cdata->canViewMembers()) update.flags |= UpdateFlag::ChannelCanViewMembers;
|
|| canAddMembers != cdata->canAddMembers()) update.flags |= UpdateFlag::ChannelRightsChanged;
|
||||||
if (canAddMembers != cdata->canAddMembers()) update.flags |= UpdateFlag::ChannelCanAddMembers;
|
|
||||||
if (wasEditor != cdata->amEditor()) {
|
|
||||||
cdata->selfAdminUpdated();
|
|
||||||
update.flags |= (UpdateFlag::ChannelAmEditor | UpdateFlag::AdminsChanged);
|
|
||||||
}
|
|
||||||
} break;
|
} break;
|
||||||
}
|
}
|
||||||
if (!data) {
|
if (!data) {
|
||||||
|
@ -825,10 +811,10 @@ namespace {
|
||||||
auto &v = d.vparticipants.v;
|
auto &v = d.vparticipants.v;
|
||||||
chat->count = v.size();
|
chat->count = v.size();
|
||||||
int32 pversion = chat->participants.isEmpty() ? 1 : (chat->participants.begin().value() + 1);
|
int32 pversion = chat->participants.isEmpty() ? 1 : (chat->participants.begin().value() + 1);
|
||||||
chat->invitedByMe = ChatData::InvitedByMe();
|
chat->invitedByMe.clear();
|
||||||
chat->admins = ChatData::Admins();
|
chat->admins.clear();
|
||||||
chat->flags &= ~MTPDchat::Flag::f_admin;
|
chat->flags &= ~MTPDchat::Flag::f_admin;
|
||||||
for (QVector<MTPChatParticipant>::const_iterator i = v.cbegin(), e = v.cend(); i != e; ++i) {
|
for (auto i = v.cbegin(), e = v.cend(); i != e; ++i) {
|
||||||
int32 uid = 0, inviter = 0;
|
int32 uid = 0, inviter = 0;
|
||||||
switch (i->type()) {
|
switch (i->type()) {
|
||||||
case mtpc_chatParticipantCreator: {
|
case mtpc_chatParticipantCreator: {
|
||||||
|
@ -870,7 +856,7 @@ namespace {
|
||||||
History *h = App::historyLoaded(chat->id);
|
History *h = App::historyLoaded(chat->id);
|
||||||
bool found = !h || !h->lastKeyboardFrom;
|
bool found = !h || !h->lastKeyboardFrom;
|
||||||
int32 botStatus = -1;
|
int32 botStatus = -1;
|
||||||
for (ChatData::Participants::iterator i = chat->participants.begin(), e = chat->participants.end(); i != e;) {
|
for (auto i = chat->participants.begin(), e = chat->participants.end(); i != e;) {
|
||||||
if (i.value() < pversion) {
|
if (i.value() < pversion) {
|
||||||
i = chat->participants.erase(i);
|
i = chat->participants.erase(i);
|
||||||
} else {
|
} else {
|
||||||
|
@ -976,7 +962,7 @@ namespace {
|
||||||
chat->count--;
|
chat->count--;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
ChatData::Participants::iterator i = chat->participants.find(user);
|
auto i = chat->participants.find(user);
|
||||||
if (i != chat->participants.end()) {
|
if (i != chat->participants.end()) {
|
||||||
chat->participants.erase(i);
|
chat->participants.erase(i);
|
||||||
chat->count--;
|
chat->count--;
|
||||||
|
@ -993,7 +979,7 @@ namespace {
|
||||||
}
|
}
|
||||||
if (chat->botStatus > 0 && user->botInfo) {
|
if (chat->botStatus > 0 && user->botInfo) {
|
||||||
int32 botStatus = -1;
|
int32 botStatus = -1;
|
||||||
for (ChatData::Participants::const_iterator j = chat->participants.cbegin(), e = chat->participants.cend(); j != e; ++j) {
|
for (auto j = chat->participants.cbegin(), e = chat->participants.cend(); j != e; ++j) {
|
||||||
if (j.key()->botInfo) {
|
if (j.key()->botInfo) {
|
||||||
if (true || botStatus > 0/* || !j.key()->botInfo->readsAllHistory*/) {
|
if (true || botStatus > 0/* || !j.key()->botInfo->readsAllHistory*/) {
|
||||||
botStatus = 2;
|
botStatus = 2;
|
||||||
|
|
|
@ -938,7 +938,7 @@ void EditChannelBox::prepare() {
|
||||||
|
|
||||||
connect(_publicLink, SIGNAL(clicked()), this, SLOT(onPublicLink()));
|
connect(_publicLink, SIGNAL(clicked()), this, SLOT(onPublicLink()));
|
||||||
_publicLink->setVisible(_channel->canEditUsername());
|
_publicLink->setVisible(_channel->canEditUsername());
|
||||||
_sign->setVisible(!_channel->isMegagroup());
|
_sign->setVisible(canEditSignatures());
|
||||||
|
|
||||||
updateMaxHeight();
|
updateMaxHeight();
|
||||||
}
|
}
|
||||||
|
@ -969,10 +969,14 @@ void EditChannelBox::onDescriptionResized() {
|
||||||
update();
|
update();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool EditChannelBox::canEditSignatures() const {
|
||||||
|
return _channel->amCreator() && !_channel->isMegagroup();
|
||||||
|
}
|
||||||
|
|
||||||
void EditChannelBox::updateMaxHeight() {
|
void EditChannelBox::updateMaxHeight() {
|
||||||
auto newHeight = st::newGroupInfoPadding.top() + _title->height();
|
auto newHeight = st::newGroupInfoPadding.top() + _title->height();
|
||||||
newHeight += st::newGroupDescriptionPadding.top() + _description->height() + st::newGroupDescriptionPadding.bottom();
|
newHeight += st::newGroupDescriptionPadding.top() + _description->height() + st::newGroupDescriptionPadding.bottom();
|
||||||
if (!_channel->isMegagroup()) {
|
if (canEditSignatures()) {
|
||||||
newHeight += st::newGroupPublicLinkPadding.top() + _sign->heightNoMargins() + st::newGroupPublicLinkPadding.bottom();
|
newHeight += st::newGroupPublicLinkPadding.top() + _sign->heightNoMargins() + st::newGroupPublicLinkPadding.bottom();
|
||||||
}
|
}
|
||||||
if (_channel->canEditUsername()) {
|
if (_channel->canEditUsername()) {
|
||||||
|
@ -993,10 +997,10 @@ void EditChannelBox::resizeEvent(QResizeEvent *e) {
|
||||||
|
|
||||||
_sign->moveToLeft(st::boxPadding.left() + st::newGroupInfoPadding.left(), _description->y() + _description->height() + st::newGroupDescriptionPadding.bottom() + st::newGroupPublicLinkPadding.top());
|
_sign->moveToLeft(st::boxPadding.left() + st::newGroupInfoPadding.left(), _description->y() + _description->height() + st::newGroupDescriptionPadding.bottom() + st::newGroupPublicLinkPadding.top());
|
||||||
|
|
||||||
if (_channel->isMegagroup()) {
|
if (canEditSignatures()) {
|
||||||
_publicLink->moveToLeft(st::boxPadding.left() + st::newGroupInfoPadding.left(), _description->y() + _description->height() + st::newGroupDescriptionPadding.bottom() + st::newGroupPublicLinkPadding.top());
|
|
||||||
} else {
|
|
||||||
_publicLink->moveToLeft(st::boxPadding.left() + st::newGroupInfoPadding.left(), _sign->bottomNoMargins() + st::newGroupDescriptionPadding.bottom() + st::newGroupPublicLinkPadding.top());
|
_publicLink->moveToLeft(st::boxPadding.left() + st::newGroupInfoPadding.left(), _sign->bottomNoMargins() + st::newGroupDescriptionPadding.bottom() + st::newGroupPublicLinkPadding.top());
|
||||||
|
} else {
|
||||||
|
_publicLink->moveToLeft(st::boxPadding.left() + st::newGroupInfoPadding.left(), _description->y() + _description->height() + st::newGroupDescriptionPadding.bottom() + st::newGroupPublicLinkPadding.top());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1031,7 +1035,7 @@ void EditChannelBox::saveDescription() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void EditChannelBox::saveSign() {
|
void EditChannelBox::saveSign() {
|
||||||
if (_channel->isMegagroup() || _channel->addsSignature() == _sign->checked()) {
|
if (!canEditSignatures() || _channel->addsSignature() == _sign->checked()) {
|
||||||
closeBox();
|
closeBox();
|
||||||
} else {
|
} else {
|
||||||
_saveSignRequestId = MTP::send(MTPchannels_ToggleSignatures(_channel->inputChannel, MTP_bool(_sign->checked())), rpcDone(&EditChannelBox::onSaveSignDone), rpcFail(&EditChannelBox::onSaveFail));
|
_saveSignRequestId = MTP::send(MTPchannels_ToggleSignatures(_channel->inputChannel, MTP_bool(_sign->checked())), rpcDone(&EditChannelBox::onSaveSignDone), rpcFail(&EditChannelBox::onSaveFail));
|
||||||
|
|
|
@ -255,6 +255,7 @@ private slots:
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void updateMaxHeight();
|
void updateMaxHeight();
|
||||||
|
bool canEditSignatures() const;
|
||||||
|
|
||||||
void onSaveTitleDone(const MTPUpdates &updates);
|
void onSaveTitleDone(const MTPUpdates &updates);
|
||||||
void onSaveDescriptionDone(const MTPBool &result);
|
void onSaveDescriptionDone(const MTPBool &result);
|
||||||
|
|
|
@ -532,7 +532,7 @@ void DeleteMessagesBox::deleteAndClear() {
|
||||||
|
|
||||||
if (_moderateFrom) {
|
if (_moderateFrom) {
|
||||||
if (_banUser && _banUser->checked()) {
|
if (_banUser && _banUser->checked()) {
|
||||||
// MTP::send(MTPchannels_KickFromChannel(_moderateInChannel->inputChannel, _moderateFrom->inputUser, MTP_boolTrue()), App::main()->rpcDone(&MainWidget::sentUpdatesReceived));
|
App::api()->kickParticipant(_moderateInChannel, _moderateFrom);
|
||||||
}
|
}
|
||||||
if (_reportSpam->checked()) {
|
if (_reportSpam->checked()) {
|
||||||
MTP::send(MTPchannels_ReportSpam(_moderateInChannel->inputChannel, _moderateFrom->inputUser, MTP_vector<MTPint>(1, MTP_int(_ids[0].msg))));
|
MTP::send(MTPchannels_ReportSpam(_moderateInChannel->inputChannel, _moderateFrom->inputUser, MTP_vector<MTPint>(1, MTP_int(_ids[0].msg))));
|
||||||
|
|
|
@ -344,8 +344,8 @@ void ContactsBox::onPeerSelectedChanged(PeerData *peer, bool checked) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void ContactsBox::inviteParticipants() {
|
void ContactsBox::inviteParticipants() {
|
||||||
QVector<UserData*> users(_inner->selected());
|
auto users = _inner->selected();
|
||||||
if (users.isEmpty()) {
|
if (users.empty()) {
|
||||||
_select->entity()->setInnerFocus();
|
_select->entity()->setInnerFocus();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -363,7 +363,7 @@ void ContactsBox::createGroup() {
|
||||||
if (_saveRequestId) return;
|
if (_saveRequestId) return;
|
||||||
|
|
||||||
auto users = _inner->selectedInputs();
|
auto users = _inner->selectedInputs();
|
||||||
if (users.isEmpty() || (users.size() == 1 && users.at(0).type() == mtpc_inputUserSelf)) {
|
if (users.empty() || (users.size() == 1 && users.at(0).type() == mtpc_inputUserSelf)) {
|
||||||
_select->entity()->setInnerFocus();
|
_select->entity()->setInnerFocus();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -396,15 +396,16 @@ void ContactsBox::getAdminsDone(const MTPmessages_ChatFull &result) {
|
||||||
closeBox();
|
closeBox();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
ChatData::Admins curadmins = _inner->chat()->admins;
|
auto curadmins = _inner->chat()->admins;
|
||||||
QVector<UserData*> newadmins = _inner->selected(), appoint;
|
auto newadmins = _inner->selected();
|
||||||
if (!newadmins.isEmpty()) {
|
auto appoint = decltype(newadmins)();
|
||||||
|
if (!newadmins.empty()) {
|
||||||
appoint.reserve(newadmins.size());
|
appoint.reserve(newadmins.size());
|
||||||
for (int32 i = 0, l = newadmins.size(); i < l; ++i) {
|
for (auto &user : newadmins) {
|
||||||
ChatData::Admins::iterator c = curadmins.find(newadmins.at(i));
|
auto c = curadmins.find(user);
|
||||||
if (c == curadmins.cend()) {
|
if (c == curadmins.cend()) {
|
||||||
if (newadmins.at(i)->id != peerFromUser(_inner->chat()->creator)) {
|
if (user->id != peerFromUser(_inner->chat()->creator)) {
|
||||||
appoint.push_back(newadmins.at(i));
|
appoint.push_back(user);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
curadmins.erase(c);
|
curadmins.erase(c);
|
||||||
|
@ -413,10 +414,10 @@ void ContactsBox::getAdminsDone(const MTPmessages_ChatFull &result) {
|
||||||
}
|
}
|
||||||
_saveRequestId = 0;
|
_saveRequestId = 0;
|
||||||
|
|
||||||
for_const (UserData *user, curadmins) {
|
for_const (auto user, curadmins) {
|
||||||
MTP::send(MTPmessages_EditChatAdmin(_inner->chat()->inputChat, user->inputUser, MTP_boolFalse()), rpcDone(&ContactsBox::removeAdminDone, user), rpcFail(&ContactsBox::editAdminFail), 0, 10);
|
MTP::send(MTPmessages_EditChatAdmin(_inner->chat()->inputChat, user->inputUser, MTP_boolFalse()), rpcDone(&ContactsBox::removeAdminDone, user), rpcFail(&ContactsBox::editAdminFail), 0, 10);
|
||||||
}
|
}
|
||||||
for_const (UserData *user, appoint) {
|
for_const (auto user, appoint) {
|
||||||
MTP::send(MTPmessages_EditChatAdmin(_inner->chat()->inputChat, user->inputUser, MTP_boolTrue()), rpcDone(&ContactsBox::setAdminDone, user), rpcFail(&ContactsBox::editAdminFail), 0, 10);
|
MTP::send(MTPmessages_EditChatAdmin(_inner->chat()->inputChat, user->inputUser, MTP_boolTrue()), rpcDone(&ContactsBox::setAdminDone, user), rpcFail(&ContactsBox::editAdminFail), 0, 10);
|
||||||
}
|
}
|
||||||
MTP::sendAnything();
|
MTP::sendAnything();
|
||||||
|
@ -427,7 +428,7 @@ void ContactsBox::getAdminsDone(const MTPmessages_ChatFull &result) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ContactsBox::setAdminDone(UserData *user, const MTPBool &result) {
|
void ContactsBox::setAdminDone(gsl::not_null<UserData*> user, const MTPBool &result) {
|
||||||
if (mtpIsTrue(result)) {
|
if (mtpIsTrue(result)) {
|
||||||
if (_inner->chat()->noParticipantInfo()) {
|
if (_inner->chat()->noParticipantInfo()) {
|
||||||
App::api()->requestFullPeer(_inner->chat());
|
App::api()->requestFullPeer(_inner->chat());
|
||||||
|
@ -442,7 +443,7 @@ void ContactsBox::setAdminDone(UserData *user, const MTPBool &result) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ContactsBox::removeAdminDone(UserData *user, const MTPBool &result) {
|
void ContactsBox::removeAdminDone(gsl::not_null<UserData*> user, const MTPBool &result) {
|
||||||
if (mtpIsTrue(result)) {
|
if (mtpIsTrue(result)) {
|
||||||
_inner->chat()->admins.remove(user);
|
_inner->chat()->admins.remove(user);
|
||||||
}
|
}
|
||||||
|
@ -610,7 +611,7 @@ ContactsBox::Inner::Inner(QWidget *parent, UserData *bot) : TWidget(parent)
|
||||||
addDialogsToList([](PeerData *peer) {
|
addDialogsToList([](PeerData *peer) {
|
||||||
if (peer->isChat() && peer->asChat()->canEdit()) {
|
if (peer->isChat() && peer->asChat()->canEdit()) {
|
||||||
return true;
|
return true;
|
||||||
} else if (peer->isMegagroup() && peer->asChannel()->canAddMembers()) {
|
} else if (peer->isMegagroup()) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
|
@ -719,10 +720,10 @@ void ContactsBox::Inner::addBot() {
|
||||||
} else if (!info->startGroupToken.isEmpty()) {
|
} else if (!info->startGroupToken.isEmpty()) {
|
||||||
MTP::send(MTPmessages_StartBot(_bot->inputUser, _addToPeer->input, MTP_long(rand_value<uint64>()), MTP_string(info->startGroupToken)), App::main()->rpcDone(&MainWidget::sentUpdatesReceived), App::main()->rpcFail(&MainWidget::addParticipantFail, { _bot, _addToPeer }));
|
MTP::send(MTPmessages_StartBot(_bot->inputUser, _addToPeer->input, MTP_long(rand_value<uint64>()), MTP_string(info->startGroupToken)), App::main()->rpcDone(&MainWidget::sentUpdatesReceived), App::main()->rpcFail(&MainWidget::addParticipantFail, { _bot, _addToPeer }));
|
||||||
} else {
|
} else {
|
||||||
App::main()->addParticipants(_addToPeer, QVector<UserData*>(1, _bot));
|
App::main()->addParticipants(_addToPeer, std::vector<gsl::not_null<UserData*>>(1, _bot));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
App::main()->addParticipants(_addToPeer, QVector<UserData*>(1, _bot));
|
App::main()->addParticipants(_addToPeer, std::vector<gsl::not_null<UserData*>>(1, _bot));
|
||||||
}
|
}
|
||||||
Ui::hideLayer();
|
Ui::hideLayer();
|
||||||
Ui::showPeerHistory(_addToPeer, ShowAtUnreadMsgId);
|
Ui::showPeerHistory(_addToPeer, ShowAtUnreadMsgId);
|
||||||
|
@ -742,21 +743,8 @@ void ContactsBox::Inner::addAdminDone(MTPChannelAdminRights rights, const MTPUpd
|
||||||
if (req != _addAdminRequestId) return;
|
if (req != _addAdminRequestId) return;
|
||||||
|
|
||||||
_addAdminRequestId = 0;
|
_addAdminRequestId = 0;
|
||||||
if (_addAdmin && _channel && _channel->isMegagroup()) {
|
if (_addAdmin && _channel) {
|
||||||
Notify::PeerUpdate update(_channel);
|
_channel->applyEditAdmin(_addAdmin, rights);
|
||||||
if (_channel->mgInfo->lastParticipants.indexOf(_addAdmin) < 0) {
|
|
||||||
_channel->mgInfo->lastParticipants.push_front(_addAdmin);
|
|
||||||
update.flags |= Notify::PeerUpdate::Flag::MembersChanged;
|
|
||||||
}
|
|
||||||
_channel->mgInfo->lastAdmins.insert(_addAdmin, rights);
|
|
||||||
update.flags |= Notify::PeerUpdate::Flag::AdminsChanged;
|
|
||||||
if (_addAdmin->botInfo) {
|
|
||||||
_channel->mgInfo->bots.insert(_addAdmin);
|
|
||||||
if (_channel->mgInfo->botStatus != 0 && _channel->mgInfo->botStatus < 2) {
|
|
||||||
_channel->mgInfo->botStatus = 2;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Notify::peerUpdatedDelayed(update);
|
|
||||||
}
|
}
|
||||||
if (_addAdminBox) _addAdminBox->closeBox();
|
if (_addAdminBox) _addAdminBox->closeBox();
|
||||||
emit adminAdded();
|
emit adminAdded();
|
||||||
|
@ -1336,117 +1324,197 @@ void ContactsBox::Inner::setSearchedPressed(int pressed) {
|
||||||
_searchedPressed = pressed;
|
_searchedPressed = pressed;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ContactsBox::Inner::chooseParticipant() {
|
void ContactsBox::Inner::changeMultiSelectCheckState() {
|
||||||
if (_saving) return;
|
_time = unixtime();
|
||||||
bool addingAdmin = (_channel && _membersFilter == MembersFilter::Admins);
|
if (_filter.isEmpty()) {
|
||||||
if (!addingAdmin && usingMultiSelect()) {
|
if (_searchedSelected >= 0 && _searchedSelected < _byUsername.size()) {
|
||||||
_time = unixtime();
|
auto data = d_byUsername[_searchedSelected];
|
||||||
if (_filter.isEmpty()) {
|
auto peer = _byUsername[_searchedSelected];
|
||||||
if (_searchedSelected >= 0 && _searchedSelected < _byUsername.size()) {
|
if (data->disabledChecked) return;
|
||||||
auto data = d_byUsername[_searchedSelected];
|
|
||||||
auto peer = _byUsername[_searchedSelected];
|
|
||||||
if (data->disabledChecked) return;
|
|
||||||
|
|
||||||
changeCheckState(data, peer);
|
changeCheckState(data, peer);
|
||||||
} else if (_selected) {
|
} else if (_selected) {
|
||||||
auto data = contactData(_selected);
|
auto data = contactData(_selected);
|
||||||
auto peer = _selected->history()->peer;
|
auto peer = _selected->history()->peer;
|
||||||
if (data->disabledChecked) return;
|
if (data->disabledChecked) return;
|
||||||
|
|
||||||
changeCheckState(_selected);
|
changeCheckState(_selected);
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (_searchedSelected >= 0 && _searchedSelected < _byUsernameFiltered.size()) {
|
|
||||||
auto data = d_byUsernameFiltered[_searchedSelected];
|
|
||||||
auto peer = _byUsernameFiltered[_searchedSelected];
|
|
||||||
if (data->disabledChecked) return;
|
|
||||||
|
|
||||||
int i = 0, l = d_byUsername.size();
|
|
||||||
for (; i < l; ++i) {
|
|
||||||
if (d_byUsername[i] == data) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (i == l) {
|
|
||||||
d_byUsername.push_back(data);
|
|
||||||
_byUsername.push_back(peer);
|
|
||||||
for (i = 0, l = _byUsernameDatas.size(); i < l;) {
|
|
||||||
if (_byUsernameDatas[i] == data) {
|
|
||||||
_byUsernameDatas.removeAt(i);
|
|
||||||
--l;
|
|
||||||
} else {
|
|
||||||
++i;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
changeCheckState(data, peer);
|
|
||||||
} else if (_filteredSelected >= 0 && _filteredSelected < _filtered.size()) {
|
|
||||||
auto data = contactData(_filtered[_filteredSelected]);
|
|
||||||
auto peer = _filtered[_filteredSelected]->history()->peer;
|
|
||||||
if (data->disabledChecked) return;
|
|
||||||
|
|
||||||
changeCheckState(data, peer);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
PeerData *peer = 0;
|
if (_searchedSelected >= 0 && _searchedSelected < _byUsernameFiltered.size()) {
|
||||||
if (_filter.isEmpty()) {
|
auto data = d_byUsernameFiltered[_searchedSelected];
|
||||||
if (_searchedSelected >= 0 && _searchedSelected < _byUsername.size()) {
|
auto peer = _byUsernameFiltered[_searchedSelected];
|
||||||
peer = _byUsername[_searchedSelected];
|
if (data->disabledChecked) return;
|
||||||
} else if (_selected) {
|
|
||||||
peer = _selected->history()->peer;
|
int i = 0, l = d_byUsername.size();
|
||||||
}
|
for (; i < l; ++i) {
|
||||||
} else {
|
if (d_byUsername[i] == data) {
|
||||||
if (_searchedSelected >= 0 && _searchedSelected < _byUsernameFiltered.size()) {
|
break;
|
||||||
peer = _byUsernameFiltered[_searchedSelected];
|
|
||||||
} else {
|
|
||||||
if (_filteredSelected < 0 || _filteredSelected >= _filtered.size()) return;
|
|
||||||
peer = _filtered[_filteredSelected]->history()->peer;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (peer) {
|
|
||||||
if (addingAdmin) {
|
|
||||||
_addAdmin = peer->asUser();
|
|
||||||
if (_addAdminRequestId) {
|
|
||||||
MTP::cancel(_addAdminRequestId);
|
|
||||||
_addAdminRequestId = 0;
|
|
||||||
}
|
}
|
||||||
if (_addAdminBox) _addAdminBox->deleteLater();
|
|
||||||
using Right = MTPDchannelAdminRights::Flag;
|
|
||||||
auto defaultRights = _channel->isMegagroup()
|
|
||||||
? (Right::f_change_info | Right::f_delete_messages | Right::f_ban_users | Right::f_invite_users | Right::f_invite_link | Right::f_pin_messages)
|
|
||||||
: (Right::f_change_info | Right::f_post_messages | Right::f_edit_messages | Right::f_delete_messages);
|
|
||||||
auto currentRights = (_channel->isMegagroup() ? _channel->mgInfo->lastAdmins : QMap<UserData*, MTPChannelAdminRights>()).value(_addAdmin, MTP_channelAdminRights(MTP_flags(defaultRights)));
|
|
||||||
_addAdminBox = Ui::show(Box<EditAdminBox>(_channel, _addAdmin, currentRights, base::lambda_guarded(this, [this](const MTPChannelAdminRights &rights) {
|
|
||||||
if (_addAdminRequestId) return;
|
|
||||||
_addAdminRequestId = MTP::send(MTPchannels_EditAdmin(_channel->inputChannel, _addAdmin->inputUser, rights), rpcDone(&Inner::addAdminDone, rights), rpcFail(&Inner::addAdminFail));
|
|
||||||
})), KeepOtherLayers);
|
|
||||||
} else if (sharingBotGame()) {
|
|
||||||
_addToPeer = peer;
|
|
||||||
auto confirmText = [peer] {
|
|
||||||
if (peer->isUser()) {
|
|
||||||
return lng_bot_sure_share_game(lt_user, App::peerName(peer));
|
|
||||||
}
|
|
||||||
return lng_bot_sure_share_game_group(lt_group, peer->name);
|
|
||||||
};
|
|
||||||
Ui::show(Box<ConfirmBox>(confirmText(), base::lambda_guarded(this, [this] {
|
|
||||||
addBot();
|
|
||||||
})), KeepOtherLayers);
|
|
||||||
} else if (bot() && (peer->isChat() || peer->isMegagroup())) {
|
|
||||||
_addToPeer = peer;
|
|
||||||
Ui::show(Box<ConfirmBox>(lng_bot_sure_invite(lt_group, peer->name), base::lambda_guarded(this, [this] {
|
|
||||||
addBot();
|
|
||||||
})), KeepOtherLayers);
|
|
||||||
} else {
|
|
||||||
Ui::hideSettingsAndLayer(true);
|
|
||||||
App::main()->choosePeer(peer->id, ShowAtUnreadMsgId);
|
|
||||||
}
|
}
|
||||||
|
if (i == l) {
|
||||||
|
d_byUsername.push_back(data);
|
||||||
|
_byUsername.push_back(peer);
|
||||||
|
for (i = 0, l = _byUsernameDatas.size(); i < l;) {
|
||||||
|
if (_byUsernameDatas[i] == data) {
|
||||||
|
_byUsernameDatas.removeAt(i);
|
||||||
|
--l;
|
||||||
|
} else {
|
||||||
|
++i;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
changeCheckState(data, peer);
|
||||||
|
} else if (_filteredSelected >= 0 && _filteredSelected < _filtered.size()) {
|
||||||
|
auto data = contactData(_filtered[_filteredSelected]);
|
||||||
|
auto peer = _filtered[_filteredSelected]->history()->peer;
|
||||||
|
if (data->disabledChecked) return;
|
||||||
|
|
||||||
|
changeCheckState(data, peer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
PeerData *ContactsBox::Inner::selectedPeer() const {
|
||||||
|
if (_filter.isEmpty()) {
|
||||||
|
if (_searchedSelected >= 0 && _searchedSelected < _byUsername.size()) {
|
||||||
|
return _byUsername[_searchedSelected];
|
||||||
|
} else if (_selected) {
|
||||||
|
return _selected->history()->peer;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (_searchedSelected >= 0 && _searchedSelected < _byUsernameFiltered.size()) {
|
||||||
|
return _byUsernameFiltered[_searchedSelected];
|
||||||
|
} else if (_filteredSelected >= 0 && _filteredSelected < _filtered.size()) {
|
||||||
|
return _filtered[_filteredSelected]->history()->peer;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ContactsBox::Inner::chooseParticipant() {
|
||||||
|
if (_saving) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (usingMultiSelect()) {
|
||||||
|
changeMultiSelectCheckState();
|
||||||
|
} else {
|
||||||
|
if (_channel && _membersFilter == MembersFilter::Admins) {
|
||||||
|
addSelectedAsChannelAdmin();
|
||||||
|
} else if (sharingBotGame()) {
|
||||||
|
shareBotGameToSelected();
|
||||||
|
} else if (bot()) {
|
||||||
|
addBotToSelectedGroup();
|
||||||
|
} else if (auto peer = selectedPeer()) {
|
||||||
|
Ui::hideSettingsAndLayer(true);
|
||||||
|
App::main()->choosePeer(peer->id, ShowAtUnreadMsgId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
update();
|
update();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ContactsBox::Inner::addSelectedAsChannelAdmin() {
|
||||||
|
auto peer = selectedPeer();
|
||||||
|
if (!peer) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
_addAdmin = peer->asUser();
|
||||||
|
t_assert(_addAdmin != nullptr);
|
||||||
|
|
||||||
|
if (_addAdminRequestId) {
|
||||||
|
MTP::cancel(_addAdminRequestId);
|
||||||
|
_addAdminRequestId = 0;
|
||||||
|
}
|
||||||
|
if (_addAdminBox) _addAdminBox->deleteLater();
|
||||||
|
|
||||||
|
auto showBox = [this](auto &¤tRights) {
|
||||||
|
_addAdminBox = Ui::show(Box<EditAdminBox>(_channel, _addAdmin, currentRights, base::lambda_guarded(this, [this](const MTPChannelAdminRights &rights) {
|
||||||
|
if (_addAdminRequestId) return;
|
||||||
|
_addAdminRequestId = MTP::send(MTPchannels_EditAdmin(_channel->inputChannel, _addAdmin->inputUser, rights), rpcDone(&Inner::addAdminDone, rights), rpcFail(&Inner::addAdminFail));
|
||||||
|
})), KeepOtherLayers);
|
||||||
|
};
|
||||||
|
|
||||||
|
auto loadedRights = [this]() -> const MegagroupInfo::Admin * {
|
||||||
|
if (_channel->isMegagroup()) {
|
||||||
|
auto it = _channel->mgInfo->lastAdmins.constFind(_addAdmin);
|
||||||
|
if (it != _channel->mgInfo->lastAdmins.cend()) {
|
||||||
|
return &it.value();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
if (auto rights = loadedRights()) {
|
||||||
|
if (rights->canEdit) {
|
||||||
|
showBox(rights->rights);
|
||||||
|
} else {
|
||||||
|
Ui::show(Box<InformBox>(lang(lng_error_cant_edit_admin)), KeepOtherLayers);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// We don't have current rights yet.
|
||||||
|
_addAdminRequestId = MTP::send(MTPchannels_GetParticipant(_channel->inputChannel, _addAdmin->inputUser), ::rpcDone(base::lambda_guarded(this, [this, showBox](const MTPchannels_ChannelParticipant &result) {
|
||||||
|
Expects(result.type() == mtpc_channels_channelParticipant);
|
||||||
|
auto &participant = result.c_channels_channelParticipant();
|
||||||
|
App::feedUsers(participant.vusers);
|
||||||
|
_addAdminRequestId = 0;
|
||||||
|
if (participant.vparticipant.type() == mtpc_channelParticipantAdmin) {
|
||||||
|
if (participant.vparticipant.c_channelParticipantAdmin().is_can_edit()) {
|
||||||
|
showBox(participant.vparticipant.c_channelParticipantAdmin().vadmin_rights);
|
||||||
|
} else {
|
||||||
|
Ui::show(Box<InformBox>(lang(lng_error_cant_edit_admin)), KeepOtherLayers);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
showBox(EditAdminBox::DefaultRights(_channel));
|
||||||
|
}
|
||||||
|
})), ::rpcFail(base::lambda_guarded(this, [this](const RPCError &error) {
|
||||||
|
if (MTP::isDefaultHandledError(error)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
_addAdminRequestId = 0;
|
||||||
|
return true;
|
||||||
|
})));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ContactsBox::Inner::shareBotGameToSelected() {
|
||||||
|
_addToPeer = selectedPeer();
|
||||||
|
if (!_addToPeer) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto confirmText = [this] {
|
||||||
|
if (_addToPeer->isUser()) {
|
||||||
|
return lng_bot_sure_share_game(lt_user, App::peerName(_addToPeer));
|
||||||
|
}
|
||||||
|
return lng_bot_sure_share_game_group(lt_group, _addToPeer->name);
|
||||||
|
};
|
||||||
|
Ui::show(Box<ConfirmBox>(confirmText(), base::lambda_guarded(this, [this] {
|
||||||
|
addBot();
|
||||||
|
})), KeepOtherLayers);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ContactsBox::Inner::addBotToSelectedGroup() {
|
||||||
|
_addToPeer = selectedPeer();
|
||||||
|
if (!_addToPeer) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (auto megagroup = _addToPeer->asMegagroup()) {
|
||||||
|
if (!megagroup->canAddMembers()) {
|
||||||
|
Ui::show(Box<InformBox>(lang(lng_error_cant_add_member)), KeepOtherLayers);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (_addToPeer->isChat() || _addToPeer->isMegagroup()) {
|
||||||
|
Ui::show(Box<ConfirmBox>(lng_bot_sure_invite(lt_group, _addToPeer->name), base::lambda_guarded(this, [this] {
|
||||||
|
addBot();
|
||||||
|
})), KeepOtherLayers);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void ContactsBox::Inner::changeCheckState(Dialogs::Row *row) {
|
void ContactsBox::Inner::changeCheckState(Dialogs::Row *row) {
|
||||||
changeCheckState(contactData(row), row->history()->peer);
|
changeCheckState(contactData(row), row->history()->peer);
|
||||||
}
|
}
|
||||||
|
@ -1974,8 +2042,8 @@ void ContactsBox::Inner::selectSkipPage(int32 h, int32 dir) {
|
||||||
selectSkip(points * dir);
|
selectSkip(points * dir);
|
||||||
}
|
}
|
||||||
|
|
||||||
QVector<UserData*> ContactsBox::Inner::selected() {
|
std::vector<gsl::not_null<UserData*>> ContactsBox::Inner::selected() {
|
||||||
QVector<UserData*> result;
|
std::vector<gsl::not_null<UserData*>> result;
|
||||||
if (!usingMultiSelect()) {
|
if (!usingMultiSelect()) {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
@ -1987,13 +2055,17 @@ QVector<UserData*> ContactsBox::Inner::selected() {
|
||||||
}
|
}
|
||||||
result.reserve(_contactsData.size());
|
result.reserve(_contactsData.size());
|
||||||
for (auto i = _contactsData.cbegin(), e = _contactsData.cend(); i != e; ++i) {
|
for (auto i = _contactsData.cbegin(), e = _contactsData.cend(); i != e; ++i) {
|
||||||
if (i.value()->checkbox->checked() && i.key()->isUser()) {
|
if (i.value()->checkbox->checked()) {
|
||||||
result.push_back(i.key()->asUser());
|
if (auto user = i.key()->asUser()) {
|
||||||
|
result.push_back(user);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (int i = 0, l = _byUsername.size(); i < l; ++i) {
|
for (int i = 0, l = _byUsername.size(); i < l; ++i) {
|
||||||
if (d_byUsername[i]->checkbox->checked() && _byUsername[i]->isUser()) {
|
if (d_byUsername[i]->checkbox->checked()) {
|
||||||
result.push_back(_byUsername[i]->asUser());
|
if (auto user = _byUsername[i]->asUser()) {
|
||||||
|
result.push_back(user);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
|
|
|
@ -105,8 +105,8 @@ private:
|
||||||
void saveAdminsDone(const MTPUpdates &result);
|
void saveAdminsDone(const MTPUpdates &result);
|
||||||
void saveSelectedAdmins();
|
void saveSelectedAdmins();
|
||||||
void getAdminsDone(const MTPmessages_ChatFull &result);
|
void getAdminsDone(const MTPmessages_ChatFull &result);
|
||||||
void setAdminDone(UserData *user, const MTPBool &result);
|
void setAdminDone(gsl::not_null<UserData*> user, const MTPBool &result);
|
||||||
void removeAdminDone(UserData *user, const MTPBool &result);
|
void removeAdminDone(gsl::not_null<UserData*> user, const MTPBool &result);
|
||||||
bool saveAdminsFail(const RPCError &error);
|
bool saveAdminsFail(const RPCError &error);
|
||||||
bool editAdminFail(const RPCError &error);
|
bool editAdminFail(const RPCError &error);
|
||||||
|
|
||||||
|
@ -163,7 +163,7 @@ public:
|
||||||
void selectSkip(int32 dir);
|
void selectSkip(int32 dir);
|
||||||
void selectSkipPage(int32 h, int32 dir);
|
void selectSkipPage(int32 h, int32 dir);
|
||||||
|
|
||||||
QVector<UserData*> selected();
|
std::vector<gsl::not_null<UserData*>> selected();
|
||||||
QVector<MTPInputUser> selectedInputs();
|
QVector<MTPInputUser> selectedInputs();
|
||||||
bool allAdmins() const;
|
bool allAdmins() const;
|
||||||
void setAllAdminsChangedCallback(base::lambda<void()> allAdminsChangedCallback) {
|
void setAllAdminsChangedCallback(base::lambda<void()> allAdminsChangedCallback) {
|
||||||
|
@ -268,9 +268,14 @@ private:
|
||||||
template <typename FilterCallback>
|
template <typename FilterCallback>
|
||||||
void addDialogsToList(FilterCallback callback);
|
void addDialogsToList(FilterCallback callback);
|
||||||
|
|
||||||
|
PeerData *selectedPeer() const;
|
||||||
bool usingMultiSelect() const {
|
bool usingMultiSelect() const {
|
||||||
return (_chat != nullptr) || (_creating != CreatingGroupNone && (!_channel || _membersFilter != MembersFilter::Admins));
|
return (_chat != nullptr) || (_creating != CreatingGroupNone && (!_channel || _membersFilter != MembersFilter::Admins));
|
||||||
}
|
}
|
||||||
|
void changeMultiSelectCheckState();
|
||||||
|
void addSelectedAsChannelAdmin();
|
||||||
|
void shareBotGameToSelected();
|
||||||
|
void addBotToSelectedGroup();
|
||||||
|
|
||||||
base::lambda<void(PeerData *peer, bool selected)> _peerSelectedChangedCallback;
|
base::lambda<void(PeerData *peer, bool selected)> _peerSelectedChangedCallback;
|
||||||
|
|
||||||
|
|
|
@ -23,16 +23,81 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
||||||
#include "lang/lang_keys.h"
|
#include "lang/lang_keys.h"
|
||||||
#include "ui/widgets/checkbox.h"
|
#include "ui/widgets/checkbox.h"
|
||||||
#include "ui/widgets/labels.h"
|
#include "ui/widgets/labels.h"
|
||||||
|
#include "ui/widgets/buttons.h"
|
||||||
#include "styles/style_boxes.h"
|
#include "styles/style_boxes.h"
|
||||||
|
#include "boxes/calendar_box.h"
|
||||||
|
|
||||||
EditParticipantBox::EditParticipantBox(QWidget*, gsl::not_null<ChannelData*> channel, gsl::not_null<UserData*> user) : BoxContent()
|
namespace {
|
||||||
, _channel(channel)
|
|
||||||
, _user(user) {
|
constexpr auto kMaxRestrictDelayDays = 366;
|
||||||
|
|
||||||
|
template <typename CheckboxesMap, typename DependenciesMap>
|
||||||
|
void ApplyDependencies(CheckboxesMap &checkboxes, DependenciesMap &dependencies, QPointer<Ui::Checkbox> changed) {
|
||||||
|
auto checkAndApply = [&checkboxes](auto &¤t, auto dependency, bool isChecked) {
|
||||||
|
for (auto &&checkbox : checkboxes) {
|
||||||
|
if ((checkbox.first & dependency) && (checkbox.second->checked() == isChecked)) {
|
||||||
|
current->setChecked(isChecked);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
auto applySomeDependency = [&checkboxes, &dependencies, &changed, checkAndApply] {
|
||||||
|
auto result = false;
|
||||||
|
for (auto &&entry : checkboxes) {
|
||||||
|
if (entry.second == changed) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
auto isChecked = entry.second->checked();
|
||||||
|
for (auto &&dependency : dependencies) {
|
||||||
|
if (entry.first & (isChecked ? dependency.first : dependency.second)) {
|
||||||
|
if (checkAndApply(entry.second, (isChecked ? dependency.second : dependency.first), !isChecked)) {
|
||||||
|
result = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
if (!applySomeDependency()) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
void EditParticipantBox::resizeToContent() {
|
} // namespace
|
||||||
auto newWidth = st::boxWideWidth;
|
|
||||||
auto newHeight = 0;
|
class EditParticipantBox::Inner : public TWidget {
|
||||||
|
public:
|
||||||
|
Inner(QWidget *parent, gsl::not_null<ChannelData*> channel, gsl::not_null<UserData*> user) : TWidget(parent)
|
||||||
|
, _channel(channel)
|
||||||
|
, _user(user) {
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename Widget>
|
||||||
|
QPointer<Widget> addControl(object_ptr<Widget> row) {
|
||||||
|
row->setParent(this);
|
||||||
|
_rows.push_back(std::move(row));
|
||||||
|
return static_cast<Widget*>(_rows.back().data());
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
int resizeGetHeight(int newWidth) override;
|
||||||
|
void resizeEvent(QResizeEvent *e) override;
|
||||||
|
void paintEvent(QPaintEvent *e) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
gsl::not_null<ChannelData*> _channel;
|
||||||
|
gsl::not_null<UserData*> _user;
|
||||||
|
std::vector<object_ptr<TWidget>> _rows;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
int EditParticipantBox::Inner::resizeGetHeight(int newWidth) {
|
||||||
|
auto newHeight = st::contactsPhotoSize + st::contactsPadding.bottom();
|
||||||
auto rowWidth = newWidth - st::boxPadding.left() - st::boxPadding.right();
|
auto rowWidth = newWidth - st::boxPadding.left() - st::boxPadding.right();
|
||||||
for (auto &&row : _rows) {
|
for (auto &&row : _rows) {
|
||||||
row->resizeToNaturalWidth(rowWidth);
|
row->resizeToNaturalWidth(rowWidth);
|
||||||
|
@ -41,50 +106,111 @@ void EditParticipantBox::resizeToContent() {
|
||||||
if (!_rows.empty()) {
|
if (!_rows.empty()) {
|
||||||
newHeight += (_rows.size() - 1) * st::boxLittleSkip;
|
newHeight += (_rows.size() - 1) * st::boxLittleSkip;
|
||||||
}
|
}
|
||||||
setDimensions(st::boxWideWidth, newHeight);
|
return newHeight;
|
||||||
}
|
}
|
||||||
|
|
||||||
void EditParticipantBox::resizeEvent(QResizeEvent *e) {
|
void EditParticipantBox::Inner::resizeEvent(QResizeEvent *e) {
|
||||||
auto top = 0;
|
auto top = st::contactsPhotoSize + st::contactsPadding.bottom();
|
||||||
for (auto &&row : _rows) {
|
for (auto &&row : _rows) {
|
||||||
row->moveToLeft(st::boxPadding.left(), top);
|
row->moveToLeft(st::boxPadding.left(), top);
|
||||||
top += row->heightNoMargins() + st::boxLittleSkip;
|
top += row->heightNoMargins() + st::boxLittleSkip;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void EditParticipantBox::paintEvent(QPaintEvent *e) {
|
void EditParticipantBox::Inner::paintEvent(QPaintEvent *e) {
|
||||||
Painter p(this);
|
Painter p(this);
|
||||||
|
|
||||||
p.fillRect(e->rect(), st::boxBg);
|
p.fillRect(e->rect(), st::boxBg);
|
||||||
|
|
||||||
|
_user->paintUserpicLeft(p, st::boxPadding.left(), 0, width(), st::contactsPhotoSize);
|
||||||
|
|
||||||
|
p.setPen(st::contactsNameFg);
|
||||||
|
auto namex = st::contactsPadding.left() + st::contactsPhotoSize + st::contactsPadding.left();
|
||||||
|
auto namew = width() - namex - st::contactsPadding.right();
|
||||||
|
_user->nameText.drawLeftElided(p, namex, st::contactsNameTop, namew, width());
|
||||||
|
auto statusText = [this] {
|
||||||
|
if (_user->botInfo) {
|
||||||
|
auto isAdmin = _channel->mgInfo ? _channel->mgInfo->lastAdmins.contains(_user) : false;
|
||||||
|
auto seesAllMessages = (_user->botInfo->readsAllHistory || isAdmin);
|
||||||
|
return lang(seesAllMessages ? lng_status_bot_reads_all : lng_status_bot_not_reads_all);
|
||||||
|
}
|
||||||
|
return App::onlineText(_user->onlineTill, unixtime());
|
||||||
|
};
|
||||||
|
p.setFont(st::contactsStatusFont);
|
||||||
|
p.setPen(st::contactsStatusFg);
|
||||||
|
p.drawTextLeft(namex, st::contactsStatusTop, width(), statusText());
|
||||||
|
}
|
||||||
|
|
||||||
|
EditParticipantBox::EditParticipantBox(QWidget*, gsl::not_null<ChannelData*> channel, gsl::not_null<UserData*> user) : BoxContent()
|
||||||
|
, _channel(channel)
|
||||||
|
, _user(user) {
|
||||||
|
}
|
||||||
|
|
||||||
|
void EditParticipantBox::prepare() {
|
||||||
|
_inner = setInnerWidget(object_ptr<Inner>(this, _channel, _user));
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename Widget>
|
||||||
|
QPointer<Widget> EditParticipantBox::addControl(object_ptr<Widget> row) {
|
||||||
|
Expects(_inner != nullptr);
|
||||||
|
return _inner->addControl(std::move(row));
|
||||||
|
}
|
||||||
|
|
||||||
|
void EditParticipantBox::resizeToContent() {
|
||||||
|
_inner->resizeToWidth(st::boxWideWidth);
|
||||||
|
setDimensions(_inner->width(), _inner->height());
|
||||||
}
|
}
|
||||||
|
|
||||||
EditAdminBox::EditAdminBox(QWidget*, gsl::not_null<ChannelData*> channel, gsl::not_null<UserData*> user, const MTPChannelAdminRights &rights, base::lambda<void(MTPChannelAdminRights)> callback) : EditParticipantBox(nullptr, channel, user)
|
EditAdminBox::EditAdminBox(QWidget*, gsl::not_null<ChannelData*> channel, gsl::not_null<UserData*> user, const MTPChannelAdminRights &rights, base::lambda<void(MTPChannelAdminRights)> callback) : EditParticipantBox(nullptr, channel, user)
|
||||||
, _rights(rights) {
|
, _rights(rights)
|
||||||
|
, _saveCallback(std::move(callback)) {
|
||||||
|
auto dependency = [this](Flag dependent, Flag dependency) {
|
||||||
|
_dependencies.push_back(std::make_pair(dependent, dependency));
|
||||||
|
};
|
||||||
|
dependency(Flag::f_invite_link, Flag::f_invite_users); // invite_link <-> invite_users
|
||||||
|
dependency(Flag::f_invite_users, Flag::f_invite_link);
|
||||||
|
}
|
||||||
|
|
||||||
|
MTPChannelAdminRights EditAdminBox::DefaultRights(gsl::not_null<ChannelData*> channel) {
|
||||||
|
auto defaultRights = channel->isMegagroup()
|
||||||
|
? (Flag::f_change_info | Flag::f_delete_messages | Flag::f_ban_users | Flag::f_invite_users | Flag::f_invite_link | Flag::f_pin_messages)
|
||||||
|
: (Flag::f_change_info | Flag::f_post_messages | Flag::f_edit_messages | Flag::f_delete_messages);
|
||||||
|
return MTP_channelAdminRights(MTP_flags(defaultRights));
|
||||||
}
|
}
|
||||||
|
|
||||||
void EditAdminBox::prepare() {
|
void EditAdminBox::prepare() {
|
||||||
|
EditParticipantBox::prepare();
|
||||||
|
|
||||||
setTitle(langFactory(lng_rights_edit_admin));
|
setTitle(langFactory(lng_rights_edit_admin));
|
||||||
|
|
||||||
addControl(object_ptr<Ui::FlatLabel>(this, lang(lng_rights_edit_admin_header), Ui::FlatLabel::InitType::Simple, st::boxLabel));
|
addControl(object_ptr<Ui::FlatLabel>(this, lang(lng_rights_edit_admin_header), Ui::FlatLabel::InitType::Simple, st::boxLabel));
|
||||||
|
|
||||||
auto addCheckbox = [this](Flag flag, const QString &text) {
|
auto addCheckbox = [this](Flag flag, const QString &text) {
|
||||||
|
if (!channel()->amCreator()) {
|
||||||
|
if (!(channel()->adminRights().vflags.v & flag)) {
|
||||||
|
return; // Don't add options that we don't have ourselves.
|
||||||
|
}
|
||||||
|
}
|
||||||
auto checked = (_rights.c_channelAdminRights().vflags.v & flag) != 0;
|
auto checked = (_rights.c_channelAdminRights().vflags.v & flag) != 0;
|
||||||
auto control = addControl(object_ptr<Ui::Checkbox>(this, text, checked, st::defaultBoxCheckbox));
|
auto control = addControl(object_ptr<Ui::Checkbox>(this, text, checked, st::defaultBoxCheckbox));
|
||||||
|
connect(control, &Ui::Checkbox::changed, this, [this, control] {
|
||||||
|
applyDependencies(control);
|
||||||
|
}, Qt::QueuedConnection);
|
||||||
_checkboxes.emplace(flag, control);
|
_checkboxes.emplace(flag, control);
|
||||||
};
|
};
|
||||||
if (channel()->isMegagroup()) {
|
if (channel()->isMegagroup()) {
|
||||||
addCheckbox(Flag::f_change_info, lang(lng_rights_group_info));
|
addCheckbox(Flag::f_change_info, lang(lng_rights_group_info));
|
||||||
addCheckbox(Flag::f_delete_messages, lang(lng_rights_delete));
|
addCheckbox(Flag::f_delete_messages, lang(lng_rights_group_delete));
|
||||||
addCheckbox(Flag::f_ban_users, lang(lng_rights_group_ban));
|
addCheckbox(Flag::f_ban_users, lang(lng_rights_group_ban));
|
||||||
addCheckbox(Flag::f_invite_users, lang(lng_rights_group_invite));
|
addCheckbox(Flag::f_invite_users, lang(lng_rights_group_invite));
|
||||||
addCheckbox(Flag::f_invite_link, lang(lng_rights_group_invite_link));
|
// addCheckbox(Flag::f_invite_link, lang(lng_rights_group_invite_link));
|
||||||
addCheckbox(Flag::f_pin_messages, lang(lng_rights_group_pin));
|
addCheckbox(Flag::f_pin_messages, lang(lng_rights_group_pin));
|
||||||
addCheckbox(Flag::f_add_admins, lang(lng_rights_add_admins));
|
addCheckbox(Flag::f_add_admins, lang(lng_rights_add_admins));
|
||||||
} else {
|
} else {
|
||||||
addCheckbox(Flag::f_change_info, lang(lng_rights_channel_info));
|
addCheckbox(Flag::f_change_info, lang(lng_rights_channel_info));
|
||||||
addCheckbox(Flag::f_post_messages, lang(lng_rights_channel_post));
|
addCheckbox(Flag::f_post_messages, lang(lng_rights_channel_post));
|
||||||
addCheckbox(Flag::f_edit_messages, lang(lng_rights_channel_edit));
|
addCheckbox(Flag::f_edit_messages, lang(lng_rights_channel_edit));
|
||||||
addCheckbox(Flag::f_delete_messages, lang(lng_rights_delete));
|
addCheckbox(Flag::f_delete_messages, lang(lng_rights_channel_delete));
|
||||||
addCheckbox(Flag::f_add_admins, lang(lng_rights_add_admins));
|
addCheckbox(Flag::f_add_admins, lang(lng_rights_add_admins));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -96,9 +222,34 @@ void EditAdminBox::prepare() {
|
||||||
});
|
});
|
||||||
refreshAboutAddAdminsText();
|
refreshAboutAddAdminsText();
|
||||||
|
|
||||||
|
addButton(langFactory(lng_settings_save), [this] {
|
||||||
|
if (!_saveCallback) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
auto newFlags = MTPDchannelAdminRights::Flags(0);
|
||||||
|
for (auto &&checkbox : _checkboxes) {
|
||||||
|
if (checkbox.second->checked()) {
|
||||||
|
newFlags |= checkbox.first;
|
||||||
|
} else {
|
||||||
|
newFlags &= ~checkbox.first;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_saveCallback(MTP_channelAdminRights(MTP_flags(newFlags)));
|
||||||
|
});
|
||||||
|
addButton(langFactory(lng_cancel), [this] { closeBox(); });
|
||||||
|
|
||||||
|
applyDependencies(nullptr);
|
||||||
|
for (auto &&checkbox : _checkboxes) {
|
||||||
|
checkbox.second->finishAnimations();
|
||||||
|
}
|
||||||
|
|
||||||
resizeToContent();
|
resizeToContent();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void EditAdminBox::applyDependencies(QPointer<Ui::Checkbox> changed) {
|
||||||
|
ApplyDependencies(_checkboxes, _dependencies, changed);
|
||||||
|
}
|
||||||
|
|
||||||
void EditAdminBox::refreshAboutAddAdminsText() {
|
void EditAdminBox::refreshAboutAddAdminsText() {
|
||||||
auto addAdmins = _checkboxes.find(Flag::f_add_admins);
|
auto addAdmins = _checkboxes.find(Flag::f_add_admins);
|
||||||
t_assert(addAdmins != _checkboxes.end());
|
t_assert(addAdmins != _checkboxes.end());
|
||||||
|
@ -107,23 +258,105 @@ void EditAdminBox::refreshAboutAddAdminsText() {
|
||||||
resizeToContent();
|
resizeToContent();
|
||||||
}
|
}
|
||||||
|
|
||||||
EditRestrictedBox::EditRestrictedBox(QWidget*, gsl::not_null<ChannelData*> channel, gsl::not_null<UserData*> user, const MTPChannelBannedRights &rights, base::lambda<void(MTPChannelBannedRights)> callback) : EditParticipantBox(nullptr, channel, user) {
|
EditRestrictedBox::EditRestrictedBox(QWidget*, gsl::not_null<ChannelData*> channel, gsl::not_null<UserData*> user, const MTPChannelBannedRights &rights, base::lambda<void(MTPChannelBannedRights)> callback) : EditParticipantBox(nullptr, channel, user)
|
||||||
|
, _rights(rights)
|
||||||
|
, _until(rights.c_channelBannedRights().vuntil_date.v)
|
||||||
|
, _saveCallback(std::move(callback)) {
|
||||||
|
auto dependency = [this](Flag dependent, Flag dependency) {
|
||||||
|
_dependencies.push_back(std::make_pair(dependent, dependency));
|
||||||
|
};
|
||||||
|
dependency(Flag::f_send_gifs, Flag::f_send_stickers); // stickers <-> gifs
|
||||||
|
dependency(Flag::f_send_stickers, Flag::f_send_gifs);
|
||||||
|
dependency(Flag::f_send_games, Flag::f_send_stickers); // stickers <-> games
|
||||||
|
dependency(Flag::f_send_stickers, Flag::f_send_games);
|
||||||
|
dependency(Flag::f_send_inline, Flag::f_send_stickers); // stickers <-> inline
|
||||||
|
dependency(Flag::f_send_stickers, Flag::f_send_inline);
|
||||||
|
dependency(Flag::f_send_stickers, Flag::f_send_media); // stickers -> send_media
|
||||||
|
dependency(Flag::f_embed_links, Flag::f_send_media); // embed_links -> send_media
|
||||||
|
dependency(Flag::f_send_media, Flag::f_send_messages); // send_media- > send_messages
|
||||||
|
dependency(Flag::f_send_messages, Flag::f_view_messages); // send_messages -> view_messages
|
||||||
}
|
}
|
||||||
|
|
||||||
void EditRestrictedBox::prepare() {
|
void EditRestrictedBox::prepare() {
|
||||||
|
EditParticipantBox::prepare();
|
||||||
|
|
||||||
setTitle(langFactory(lng_rights_user_restrictions));
|
setTitle(langFactory(lng_rights_user_restrictions));
|
||||||
|
|
||||||
addControl(object_ptr<Ui::FlatLabel>(this, lang(lng_rights_edit_admin_header), Ui::FlatLabel::InitType::Simple, st::boxLabel));
|
addControl(object_ptr<Ui::FlatLabel>(this, lang(lng_rights_user_restrictions_header), Ui::FlatLabel::InitType::Simple, st::boxLabel));
|
||||||
|
|
||||||
auto addCheckbox = [this](Flag flag, const QString &text) {
|
auto addCheckbox = [this](Flags flags, const QString &text) {
|
||||||
auto checked = (_rights.c_channelBannedRights().vflags.v & flag) != 0;
|
auto checked = (_rights.c_channelBannedRights().vflags.v & flags) == 0;
|
||||||
auto control = addControl(object_ptr<Ui::Checkbox>(this, text, checked, st::defaultBoxCheckbox));
|
auto control = addControl(object_ptr<Ui::Checkbox>(this, text, checked, st::defaultBoxCheckbox));
|
||||||
_checkboxes.emplace(flag, control);
|
connect(control, &Ui::Checkbox::changed, this, [this, control] {
|
||||||
|
applyDependencies(control);
|
||||||
|
}, Qt::QueuedConnection);
|
||||||
|
_checkboxes.emplace(flags, control);
|
||||||
};
|
};
|
||||||
addCheckbox(Flag::f_view_messages, lang(lng_rights_chat_read));
|
addCheckbox(Flag::f_view_messages, lang(lng_rights_chat_read));
|
||||||
addCheckbox(Flag::f_send_messages, lang(lng_rights_chat_send_text));
|
addCheckbox(Flag::f_send_messages, lang(lng_rights_chat_send_text));
|
||||||
addCheckbox(Flag::f_send_media, lang(lng_rights_chat_send_media));
|
addCheckbox(Flag::f_send_media, lang(lng_rights_chat_send_media));
|
||||||
addCheckbox(Flag::f_send_stickers, lang(lng_rights_chat_send_stickers));
|
addCheckbox(Flag::f_send_stickers | Flag::f_send_gifs | Flag::f_send_games | Flag::f_send_inline, lang(lng_rights_chat_send_stickers));
|
||||||
addCheckbox(Flag::f_embed_links, lang(lng_rights_chat_send_links));
|
addCheckbox(Flag::f_embed_links, lang(lng_rights_chat_send_links));
|
||||||
|
|
||||||
|
_restrictUntil = addControl(object_ptr<Ui::LinkButton>(this, QString(), st::boxLinkButton));
|
||||||
|
_restrictUntil->setClickedCallback([this] { showRestrictUntil(); });
|
||||||
|
setRestrictUntil(_until);
|
||||||
|
|
||||||
|
//addControl(object_ptr<Ui::LinkButton>(this, lang(lng_rights_chat_banned_block), st::boxLinkButton));
|
||||||
|
|
||||||
|
addButton(langFactory(lng_settings_save), [this] {
|
||||||
|
if (!_saveCallback) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
auto newFlags = MTPDchannelBannedRights::Flags(0);
|
||||||
|
for (auto &&checkbox : _checkboxes) {
|
||||||
|
if (checkbox.second->checked()) {
|
||||||
|
newFlags &= ~checkbox.first;
|
||||||
|
} else {
|
||||||
|
newFlags |= checkbox.first;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_saveCallback(MTP_channelBannedRights(MTP_flags(newFlags), MTP_int(_until)));
|
||||||
|
});
|
||||||
|
addButton(langFactory(lng_cancel), [this] { closeBox(); });
|
||||||
|
|
||||||
|
applyDependencies(nullptr);
|
||||||
|
for (auto &&checkbox : _checkboxes) {
|
||||||
|
checkbox.second->finishAnimations();
|
||||||
|
}
|
||||||
|
|
||||||
resizeToContent();
|
resizeToContent();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void EditRestrictedBox::applyDependencies(QPointer<Ui::Checkbox> changed) {
|
||||||
|
ApplyDependencies(_checkboxes, _dependencies, changed);
|
||||||
|
}
|
||||||
|
|
||||||
|
MTPChannelBannedRights EditRestrictedBox::DefaultRights(gsl::not_null<ChannelData*> channel) {
|
||||||
|
auto defaultRights = Flag::f_send_messages | Flag::f_send_media | Flag::f_embed_links | Flag::f_send_stickers | Flag::f_send_gifs | Flag::f_send_games | Flag::f_send_inline;
|
||||||
|
return MTP_channelBannedRights(MTP_flags(defaultRights), MTP_int(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
void EditRestrictedBox::showRestrictUntil() {
|
||||||
|
auto tomorrow = QDate::currentDate().addDays(1);
|
||||||
|
auto highlighted = isUntilForever() ? tomorrow : date(_until).date();
|
||||||
|
auto month = highlighted;
|
||||||
|
_restrictUntilBox = Ui::show(Box<CalendarBox>(month, highlighted, [this](const QDate &date) { setRestrictUntil(static_cast<int>(QDateTime(date).toTime_t())); }), KeepOtherLayers);
|
||||||
|
_restrictUntilBox->setMaxDate(QDate::currentDate().addDays(kMaxRestrictDelayDays));
|
||||||
|
_restrictUntilBox->setMinDate(tomorrow);
|
||||||
|
_restrictUntilBox->addLeftButton(langFactory(lng_rights_chat_banned_forever), [this] { setRestrictUntil(0); });
|
||||||
|
}
|
||||||
|
|
||||||
|
void EditRestrictedBox::setRestrictUntil(int32 until) {
|
||||||
|
_until = until;
|
||||||
|
if (_restrictUntilBox) {
|
||||||
|
_restrictUntilBox->closeBox();
|
||||||
|
}
|
||||||
|
auto untilText = [this] {
|
||||||
|
if (isUntilForever()) {
|
||||||
|
return lang(lng_rights_chat_banned_forever);
|
||||||
|
}
|
||||||
|
return langDayOfMonthFull(date(_until).date());
|
||||||
|
};
|
||||||
|
_restrictUntil->setText(lng_rights_chat_banned_until(lt_when, untilText()));
|
||||||
|
}
|
||||||
|
|
|
@ -28,15 +28,16 @@ class LinkButton;
|
||||||
class Checkbox;
|
class Checkbox;
|
||||||
} // namespace Ui
|
} // namespace Ui
|
||||||
|
|
||||||
|
class CalendarBox;
|
||||||
|
|
||||||
class EditParticipantBox : public BoxContent {
|
class EditParticipantBox : public BoxContent {
|
||||||
public:
|
public:
|
||||||
EditParticipantBox(QWidget*, gsl::not_null<ChannelData*> channel, gsl::not_null<UserData*> user);
|
EditParticipantBox(QWidget*, gsl::not_null<ChannelData*> channel, gsl::not_null<UserData*> user);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void resizeToContent();
|
void prepare() override;
|
||||||
|
|
||||||
void resizeEvent(QResizeEvent *e) override;
|
void resizeToContent();
|
||||||
void paintEvent(QPaintEvent *e) override;
|
|
||||||
|
|
||||||
gsl::not_null<UserData*> user() const {
|
gsl::not_null<UserData*> user() const {
|
||||||
return _user;
|
return _user;
|
||||||
|
@ -46,16 +47,14 @@ protected:
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Widget>
|
template <typename Widget>
|
||||||
QPointer<Widget> addControl(object_ptr<Widget> row) {
|
QPointer<Widget> addControl(object_ptr<Widget> row);
|
||||||
_rows.push_back(std::move(row));
|
|
||||||
return static_cast<Widget*>(_rows.back().data());
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
gsl::not_null<UserData*> _user;
|
gsl::not_null<UserData*> _user;
|
||||||
gsl::not_null<ChannelData*> _channel;
|
gsl::not_null<ChannelData*> _channel;
|
||||||
|
|
||||||
std::vector<object_ptr<TWidget>> _rows;
|
class Inner;
|
||||||
|
QPointer<Inner> _inner;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -63,33 +62,58 @@ class EditAdminBox : public EditParticipantBox {
|
||||||
public:
|
public:
|
||||||
EditAdminBox(QWidget*, gsl::not_null<ChannelData*> channel, gsl::not_null<UserData*> user, const MTPChannelAdminRights &rights, base::lambda<void(MTPChannelAdminRights)> callback);
|
EditAdminBox(QWidget*, gsl::not_null<ChannelData*> channel, gsl::not_null<UserData*> user, const MTPChannelAdminRights &rights, base::lambda<void(MTPChannelAdminRights)> callback);
|
||||||
|
|
||||||
|
static MTPChannelAdminRights DefaultRights(gsl::not_null<ChannelData*> channel);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void prepare() override;
|
void prepare() override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
using Flag = MTPDchannelAdminRights::Flag;
|
using Flag = MTPDchannelAdminRights::Flag;
|
||||||
|
using Flags = MTPDchannelAdminRights::Flags;
|
||||||
|
|
||||||
|
void applyDependencies(QPointer<Ui::Checkbox> changed);
|
||||||
void refreshAboutAddAdminsText();
|
void refreshAboutAddAdminsText();
|
||||||
|
|
||||||
MTPChannelAdminRights _rights;
|
MTPChannelAdminRights _rights;
|
||||||
|
std::vector<std::pair<Flag, Flag>> _dependencies;
|
||||||
|
base::lambda<void(MTPChannelAdminRights)> _saveCallback;
|
||||||
|
|
||||||
std::map<Flag, QPointer<Ui::Checkbox>> _checkboxes;
|
std::map<Flags, QPointer<Ui::Checkbox>> _checkboxes;
|
||||||
QPointer<Ui::FlatLabel> _aboutAddAdmins;
|
QPointer<Ui::FlatLabel> _aboutAddAdmins;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Restricted box works with flags in the opposite way.
|
||||||
|
// If some flag is set in the rights then the checkbox is unchecked.
|
||||||
|
|
||||||
class EditRestrictedBox : public EditParticipantBox {
|
class EditRestrictedBox : public EditParticipantBox {
|
||||||
public:
|
public:
|
||||||
EditRestrictedBox(QWidget*, gsl::not_null<ChannelData*> channel, gsl::not_null<UserData*> user, const MTPChannelBannedRights &rights, base::lambda<void(MTPChannelBannedRights)> callback);
|
EditRestrictedBox(QWidget*, gsl::not_null<ChannelData*> channel, gsl::not_null<UserData*> user, const MTPChannelBannedRights &rights, base::lambda<void(MTPChannelBannedRights)> callback);
|
||||||
|
|
||||||
|
static MTPChannelBannedRights DefaultRights(gsl::not_null<ChannelData*> channel);
|
||||||
|
static constexpr auto kRestrictUntilForever = TimeId(INT_MAX);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void prepare() override;
|
void prepare() override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
using Flag = MTPDchannelBannedRights::Flag;
|
using Flag = MTPDchannelBannedRights::Flag;
|
||||||
|
using Flags = MTPDchannelBannedRights::Flags;
|
||||||
|
|
||||||
|
void applyDependencies(QPointer<Ui::Checkbox> changed);
|
||||||
|
void showRestrictUntil();
|
||||||
|
void setRestrictUntil(int32 until);
|
||||||
|
bool isUntilForever() {
|
||||||
|
return (_until <= 0) || (_until == kRestrictUntilForever);
|
||||||
|
}
|
||||||
|
|
||||||
MTPChannelBannedRights _rights;
|
MTPChannelBannedRights _rights;
|
||||||
|
int32 _until = 0;
|
||||||
|
std::vector<std::pair<Flag, Flag>> _dependencies;
|
||||||
|
base::lambda<void(MTPChannelBannedRights)> _saveCallback;
|
||||||
|
|
||||||
std::map<Flag, QPointer<Ui::Checkbox>> _checkboxes;
|
std::map<Flags, QPointer<Ui::Checkbox>> _checkboxes;
|
||||||
|
QPointer<Ui::LinkButton> _restrictUntil;
|
||||||
|
QPointer<CalendarBox> _restrictUntilBox;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
|
@ -27,6 +27,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
||||||
#include "mainwindow.h"
|
#include "mainwindow.h"
|
||||||
#include "boxes/contacts_box.h"
|
#include "boxes/contacts_box.h"
|
||||||
#include "boxes/confirm_box.h"
|
#include "boxes/confirm_box.h"
|
||||||
|
#include "boxes/edit_participant_box.h"
|
||||||
#include "ui/widgets/buttons.h"
|
#include "ui/widgets/buttons.h"
|
||||||
#include "ui/widgets/scroll_area.h"
|
#include "ui/widgets/scroll_area.h"
|
||||||
#include "ui/effects/ripple_animation.h"
|
#include "ui/effects/ripple_animation.h"
|
||||||
|
@ -79,9 +80,13 @@ void MembersBox::prepare() {
|
||||||
_inner = setInnerWidget(object_ptr<Inner>(this, _channel, _filter), st::boxLayerScroll);
|
_inner = setInnerWidget(object_ptr<Inner>(this, _channel, _filter), st::boxLayerScroll);
|
||||||
|
|
||||||
setDimensions(st::boxWideWidth, st::boxMaxListHeight);
|
setDimensions(st::boxWideWidth, st::boxMaxListHeight);
|
||||||
addButton(langFactory(lng_close), [this] { closeBox(); });
|
refreshButtons();
|
||||||
if (_channel->amCreator() && (_channel->membersCount() < (_channel->isMegagroup() ? Global::MegagroupSizeMax() : Global::ChatSizeMax()) || (!_channel->isMegagroup() && !_channel->isPublic()) || _filter == MembersFilter::Admins)) {
|
if (_filter == MembersFilter::Admins) {
|
||||||
addLeftButton(langFactory((_filter == MembersFilter::Admins) ? lng_channel_add_admin : lng_channel_add_members), [this] { onAdd(); });
|
subscribe(Notify::PeerUpdated(), Notify::PeerUpdatedHandler(Notify::PeerUpdate::Flag::ChannelRightsChanged, [this](const Notify::PeerUpdate &update) {
|
||||||
|
if (update.peer == _channel) {
|
||||||
|
refreshButtons();
|
||||||
|
}
|
||||||
|
}));
|
||||||
}
|
}
|
||||||
|
|
||||||
connect(_inner, SIGNAL(mustScrollTo(int, int)), this, SLOT(onScrollToY(int, int)));
|
connect(_inner, SIGNAL(mustScrollTo(int, int)), this, SLOT(onScrollToY(int, int)));
|
||||||
|
@ -90,6 +95,18 @@ void MembersBox::prepare() {
|
||||||
connect(_loadTimer, SIGNAL(timeout()), _inner, SLOT(load()));
|
connect(_loadTimer, SIGNAL(timeout()), _inner, SLOT(load()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MembersBox::refreshButtons() {
|
||||||
|
clearButtons();
|
||||||
|
addButton(langFactory(lng_close), [this] { closeBox(); });
|
||||||
|
if (_filter == MembersFilter::Admins) {
|
||||||
|
if (_channel->canAddAdmins()) {
|
||||||
|
addLeftButton(langFactory(lng_channel_add_admin), [this] { onAdd(); });
|
||||||
|
}
|
||||||
|
} else if (_channel->amCreator() && (_channel->membersCount() < (_channel->isMegagroup() ? Global::MegagroupSizeMax() : Global::ChatSizeMax()) || (!_channel->isMegagroup() && !_channel->isPublic()))) {
|
||||||
|
addLeftButton(langFactory(lng_channel_add_members), [this] { onAdd(); });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void MembersBox::keyPressEvent(QKeyEvent *e) {
|
void MembersBox::keyPressEvent(QKeyEvent *e) {
|
||||||
if (e->key() == Qt::Key_Down) {
|
if (e->key() == Qt::Key_Down) {
|
||||||
_inner->selectSkip(1);
|
_inner->selectSkip(1);
|
||||||
|
@ -133,11 +150,20 @@ void MembersBox::onAdminAdded() {
|
||||||
_loadTimer->start(kReloadChannelAdminsTimeout);
|
_loadTimer->start(kReloadChannelAdminsTimeout);
|
||||||
}
|
}
|
||||||
|
|
||||||
MembersBox::Inner::Inner(QWidget *parent, ChannelData *channel, MembersFilter filter) : TWidget(parent)
|
struct MembersBox::Inner::RowData {
|
||||||
|
std::unique_ptr<Ui::RippleAnimation> ripple;
|
||||||
|
int rippleRowTop = 0;
|
||||||
|
Text name;
|
||||||
|
QString online;
|
||||||
|
bool onlineColor;
|
||||||
|
bool canKick;
|
||||||
|
};
|
||||||
|
|
||||||
|
MembersBox::Inner::Inner(QWidget *parent, gsl::not_null<ChannelData*> channel, MembersFilter filter) : TWidget(parent)
|
||||||
, _rowHeight(st::contactsPadding.top() + st::contactsPhotoSize + st::contactsPadding.bottom())
|
, _rowHeight(st::contactsPadding.top() + st::contactsPhotoSize + st::contactsPadding.bottom())
|
||||||
, _channel(channel)
|
, _channel(channel)
|
||||||
, _filter(filter)
|
, _filter(filter)
|
||||||
, _kickText(lang(lng_profile_kick))
|
, _kickText(lang((filter == MembersFilter::Admins) ? lng_profile_edit_admin : lng_profile_kick))
|
||||||
, _kickWidth(st::normalFont->width(_kickText))
|
, _kickWidth(st::normalFont->width(_kickText))
|
||||||
, _aboutWidth(st::boxWideWidth - st::contactsPadding.left() - st::contactsPadding.right())
|
, _aboutWidth(st::boxWideWidth - st::contactsPadding.left() - st::contactsPadding.right())
|
||||||
, _about(_aboutWidth) {
|
, _about(_aboutWidth) {
|
||||||
|
@ -168,7 +194,7 @@ void MembersBox::Inner::paintEvent(QPaintEvent *e) {
|
||||||
auto yFrom = r.y() - st::membersMarginTop;
|
auto yFrom = r.y() - st::membersMarginTop;
|
||||||
auto yTo = r.y() + r.height() - st::membersMarginTop;
|
auto yTo = r.y() + r.height() - st::membersMarginTop;
|
||||||
p.translate(0, st::membersMarginTop);
|
p.translate(0, st::membersMarginTop);
|
||||||
if (_rows.isEmpty()) {
|
if (_rows.empty()) {
|
||||||
p.setFont(st::noContactsFont);
|
p.setFont(st::noContactsFont);
|
||||||
p.setPen(st::noContactsColor);
|
p.setPen(st::noContactsColor);
|
||||||
p.drawText(QRect(0, 0, width(), st::noContactsHeight), lang(lng_contacts_loading), style::al_center);
|
p.drawText(QRect(0, 0, width(), st::noContactsHeight), lang(lng_contacts_loading), style::al_center);
|
||||||
|
@ -179,7 +205,7 @@ void MembersBox::Inner::paintEvent(QPaintEvent *e) {
|
||||||
for (; from < to; ++from) {
|
for (; from < to; ++from) {
|
||||||
auto selected = (_pressed >= 0) ? (from == _pressed) : (from == _selected);
|
auto selected = (_pressed >= 0) ? (from == _pressed) : (from == _selected);
|
||||||
auto kickSelected = (_pressed >= 0) ? (from == _kickPressed && from == _kickSelected) : (from == _kickSelected);
|
auto kickSelected = (_pressed >= 0) ? (from == _kickPressed && from == _kickSelected) : (from == _kickSelected);
|
||||||
paintDialog(p, ms, _rows[from], data(from), selected, kickSelected);
|
paintDialog(p, ms, _rows[from], selected, kickSelected);
|
||||||
p.translate(0, _rowHeight);
|
p.translate(0, _rowHeight);
|
||||||
}
|
}
|
||||||
if (to == _rows.size() && _filter == MembersFilter::Recent && (_rows.size() < _channel->membersCount() || _rows.size() >= Global::ChatSizeMax())) {
|
if (to == _rows.size() && _filter == MembersFilter::Recent && (_rows.size() < _channel->membersCount() || _rows.size() >= Global::ChatSizeMax())) {
|
||||||
|
@ -213,8 +239,9 @@ void MembersBox::Inner::mousePressEvent(QMouseEvent *e) {
|
||||||
updateSelection();
|
updateSelection();
|
||||||
setPressed(_selected);
|
setPressed(_selected);
|
||||||
_kickPressed = _kickSelected;
|
_kickPressed = _kickSelected;
|
||||||
if (_selected >= 0 && _selected < _datas.size() && _kickSelected < 0) {
|
if (_selected >= 0 && _selected < _rows.size() && _kickSelected < 0) {
|
||||||
addRipple(_datas[_selected]);
|
ensureData(_rows[_selected]);
|
||||||
|
addRipple(_rows[_selected].data.get());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -225,18 +252,7 @@ void MembersBox::Inner::mouseReleaseEvent(QMouseEvent *e) {
|
||||||
if (e->button() == Qt::LeftButton) {
|
if (e->button() == Qt::LeftButton) {
|
||||||
if (pressed == _selected && kickPressed == _kickSelected) {
|
if (pressed == _selected && kickPressed == _kickSelected) {
|
||||||
if (kickPressed >= 0) {
|
if (kickPressed >= 0) {
|
||||||
if (!_kickRequestId) {
|
actionPressed(_rows[_kickSelected]);
|
||||||
_kickConfirm = _rows.at(_kickSelected);
|
|
||||||
if (_kickBox) _kickBox->deleteLater();
|
|
||||||
auto text = (_filter == MembersFilter::Recent ? (_channel->isMegagroup() ? lng_profile_sure_kick : lng_profile_sure_kick_channel) : lng_profile_sure_kick_admin)(lt_user, _kickConfirm->firstName);
|
|
||||||
_kickBox = Ui::show(Box<ConfirmBox>(text, base::lambda_guarded(this, [this] {
|
|
||||||
if (_filter == MembersFilter::Recent) {
|
|
||||||
// _kickRequestId = MTP::send(MTPchannels_KickFromChannel(_channel->inputChannel, _kickConfirm->inputUser, MTP_bool(true)), rpcDone(&Inner::kickDone), rpcFail(&Inner::kickFail));
|
|
||||||
} else {
|
|
||||||
// _kickRequestId = MTP::send(MTPchannels_EditAdmin(_channel->inputChannel, _kickConfirm->inputUser, MTP_channelRoleEmpty()), rpcDone(&Inner::kickAdminDone), rpcFail(&Inner::kickFail));
|
|
||||||
}
|
|
||||||
})), KeepOtherLayers);
|
|
||||||
}
|
|
||||||
} else if (pressed >= 0) {
|
} else if (pressed >= 0) {
|
||||||
chooseParticipant();
|
chooseParticipant();
|
||||||
}
|
}
|
||||||
|
@ -244,7 +260,42 @@ void MembersBox::Inner::mouseReleaseEvent(QMouseEvent *e) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void MembersBox::Inner::addRipple(MemberData *data) {
|
void MembersBox::Inner::actionPressed(Member &row) {
|
||||||
|
auto user = row.user;
|
||||||
|
if (_kickBox) _kickBox->closeBox();
|
||||||
|
if (_filter == MembersFilter::Recent) {
|
||||||
|
auto text = (_channel->isMegagroup() ? lng_profile_sure_kick : lng_profile_sure_kick_channel)(lt_user, user->firstName);
|
||||||
|
_kickBox = Ui::show(Box<ConfirmBox>(text, base::lambda_guarded(this, [this, user] {
|
||||||
|
MTP::send(MTPchannels_EditBanned(_channel->inputChannel, user->inputUser, ChannelData::KickedRestrictedRights()), ::rpcDone(base::lambda_guarded(this, [this, user](const MTPUpdates &result) {
|
||||||
|
App::main()->sentUpdatesReceived(result);
|
||||||
|
removeKicked(user);
|
||||||
|
if (_kickBox) _kickBox->closeBox();
|
||||||
|
})), rpcFail(&Inner::kickFail));
|
||||||
|
})), KeepOtherLayers);
|
||||||
|
} else {
|
||||||
|
auto currentRights = _rows[_kickSelected].adminRights;
|
||||||
|
_kickBox = Ui::show(Box<EditAdminBox>(_channel, user, currentRights, base::lambda_guarded(this, [this, user](const MTPChannelAdminRights &rights) {
|
||||||
|
if (_kickBox) _kickBox->closeBox();
|
||||||
|
MTP::send(MTPchannels_EditAdmin(_channel->inputChannel, user->inputUser, rights), ::rpcDone(base::lambda_guarded(this, [this, user, rights](const MTPUpdates &result, mtpRequestId req) {
|
||||||
|
if (App::main()) App::main()->sentUpdatesReceived(result);
|
||||||
|
_channel->applyEditAdmin(user, rights);
|
||||||
|
if (rights.c_channelAdminRights().vflags.v == 0) {
|
||||||
|
removeKicked(user);
|
||||||
|
} else {
|
||||||
|
auto it = std::find_if(_rows.begin(), _rows.end(), [this, user](auto &&row) {
|
||||||
|
return (row.user == user);
|
||||||
|
});
|
||||||
|
if (it != _rows.end()) {
|
||||||
|
it->adminRights = rights;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (_kickBox) _kickBox->closeBox();
|
||||||
|
})), rpcFail(&Inner::kickFail));
|
||||||
|
})), KeepOtherLayers);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void MembersBox::Inner::addRipple(gsl::not_null<RowData*> data) {
|
||||||
auto rowTop = getSelectedRowTop();
|
auto rowTop = getSelectedRowTop();
|
||||||
if (!data->ripple) {
|
if (!data->ripple) {
|
||||||
auto mask = Ui::RippleAnimation::rectMask(QSize(width(), _rowHeight));
|
auto mask = Ui::RippleAnimation::rectMask(QSize(width(), _rowHeight));
|
||||||
|
@ -256,21 +307,26 @@ void MembersBox::Inner::addRipple(MemberData *data) {
|
||||||
data->ripple->add(mapFromGlobal(QCursor::pos()) - QPoint(0, rowTop));
|
data->ripple->add(mapFromGlobal(QCursor::pos()) - QPoint(0, rowTop));
|
||||||
}
|
}
|
||||||
|
|
||||||
void MembersBox::Inner::stopLastRipple(MemberData *data) {
|
void MembersBox::Inner::stopLastRipple(gsl::not_null<RowData*> data) {
|
||||||
if (data->ripple) {
|
if (data->ripple) {
|
||||||
data->ripple->lastStop();
|
data->ripple->lastStop();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void MembersBox::Inner::setPressed(int pressed) {
|
void MembersBox::Inner::setPressed(int pressed) {
|
||||||
if (_pressed >= 0 && _pressed < _datas.size()) {
|
if (_pressed >= 0 && _pressed < _rows.size()) {
|
||||||
stopLastRipple(_datas[_pressed]);
|
if (_rows[_pressed].data) {
|
||||||
|
stopLastRipple(_rows[_pressed].data.get());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
_pressed = pressed;
|
_pressed = pressed;
|
||||||
}
|
}
|
||||||
|
|
||||||
void MembersBox::Inner::paintDialog(Painter &p, TimeMs ms, PeerData *peer, MemberData *data, bool selected, bool kickSelected) {
|
void MembersBox::Inner::paintDialog(Painter &p, TimeMs ms, Member &row, bool selected, bool kickSelected) {
|
||||||
UserData *user = peer->asUser();
|
ensureData(row);
|
||||||
|
|
||||||
|
auto user = row.user;
|
||||||
|
auto &data = row.data;
|
||||||
|
|
||||||
p.fillRect(0, 0, width(), _rowHeight, selected ? st::contactsBgOver : st::contactsBg);
|
p.fillRect(0, 0, width(), _rowHeight, selected ? st::contactsBgOver : st::contactsBg);
|
||||||
if (data->ripple) {
|
if (data->ripple) {
|
||||||
|
@ -279,13 +335,13 @@ void MembersBox::Inner::paintDialog(Painter &p, TimeMs ms, PeerData *peer, Membe
|
||||||
data->ripple.reset();
|
data->ripple.reset();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
peer->paintUserpicLeft(p, st::contactsPadding.left(), st::contactsPadding.top(), width(), st::contactsPhotoSize);
|
user->paintUserpicLeft(p, st::contactsPadding.left(), st::contactsPadding.top(), width(), st::contactsPhotoSize);
|
||||||
|
|
||||||
p.setPen(st::contactsNameFg);
|
p.setPen(st::contactsNameFg);
|
||||||
|
|
||||||
int32 namex = st::contactsPadding.left() + st::contactsPhotoSize + st::contactsPadding.left();
|
auto namex = st::contactsPadding.left() + st::contactsPhotoSize + st::contactsPadding.left();
|
||||||
int32 namew = width() - namex - st::contactsPadding.right() - (data->canKick ? (_kickWidth + st::contactsCheckPosition.x() * 2) : 0);
|
auto namew = width() - namex - st::contactsPadding.right() - (data->canKick ? (_kickWidth + st::contactsCheckPosition.x() * 2) : 0);
|
||||||
if (peer->isVerified()) {
|
if (user->isVerified()) {
|
||||||
auto icon = &st::dialogsVerifiedIcon;
|
auto icon = &st::dialogsVerifiedIcon;
|
||||||
namew -= icon->width();
|
namew -= icon->width();
|
||||||
icon->paint(p, namex + qMin(data->name.maxWidth(), namew), st::contactsPadding.top() + st::contactsNameTop, width());
|
icon->paint(p, namex + qMin(data->name.maxWidth(), namew), st::contactsPadding.top() + st::contactsNameTop, width());
|
||||||
|
@ -313,7 +369,7 @@ void MembersBox::Inner::selectSkip(int32 dir) {
|
||||||
}
|
}
|
||||||
cur += dir;
|
cur += dir;
|
||||||
if (cur <= 0) {
|
if (cur <= 0) {
|
||||||
_selected = _rows.isEmpty() ? -1 : 0;
|
_selected = _rows.empty() ? -1 : 0;
|
||||||
} else if (cur >= _rows.size()) {
|
} else if (cur >= _rows.size()) {
|
||||||
_selected = -1;
|
_selected = -1;
|
||||||
} else {
|
} else {
|
||||||
|
@ -324,7 +380,7 @@ void MembersBox::Inner::selectSkip(int32 dir) {
|
||||||
_selected = -1;
|
_selected = -1;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (!_rows.isEmpty()) {
|
if (!_rows.empty()) {
|
||||||
if (_selected < 0) _selected = _rows.size() - 1;
|
if (_selected < 0) _selected = _rows.size() - 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -341,9 +397,14 @@ void MembersBox::Inner::selectSkipPage(int32 h, int32 dir) {
|
||||||
selectSkip(points * dir);
|
selectSkip(points * dir);
|
||||||
}
|
}
|
||||||
|
|
||||||
MembersBox::Inner::MemberData::MemberData() = default;
|
MembersBox::Inner::Member::Member(gsl::not_null<UserData*> user) : user(user) {
|
||||||
|
}
|
||||||
|
|
||||||
MembersBox::Inner::MemberData::~MemberData() = default;
|
MembersBox::Inner::Member::Member(Member &&other) = default;
|
||||||
|
|
||||||
|
MembersBox::Inner::Member &MembersBox::Inner::Member::operator=(Member &&other) = default;
|
||||||
|
|
||||||
|
MembersBox::Inner::Member::~Member() = default;
|
||||||
|
|
||||||
void MembersBox::Inner::loadProfilePhotos() {
|
void MembersBox::Inner::loadProfilePhotos() {
|
||||||
if (_visibleTop >= _visibleBottom) return;
|
if (_visibleTop >= _visibleBottom) return;
|
||||||
|
@ -355,7 +416,7 @@ void MembersBox::Inner::loadProfilePhotos() {
|
||||||
if (yTo < 0) return;
|
if (yTo < 0) return;
|
||||||
if (yFrom < 0) yFrom = 0;
|
if (yFrom < 0) yFrom = 0;
|
||||||
|
|
||||||
if (!_rows.isEmpty()) {
|
if (!_rows.empty()) {
|
||||||
int32 from = yFrom / _rowHeight;
|
int32 from = yFrom / _rowHeight;
|
||||||
if (from < 0) from = 0;
|
if (from < 0) from = 0;
|
||||||
if (from < _rows.size()) {
|
if (from < _rows.size()) {
|
||||||
|
@ -363,7 +424,7 @@ void MembersBox::Inner::loadProfilePhotos() {
|
||||||
if (to > _rows.size()) to = _rows.size();
|
if (to > _rows.size()) to = _rows.size();
|
||||||
|
|
||||||
for (; from < to; ++from) {
|
for (; from < to; ++from) {
|
||||||
_rows[from]->loadUserpic();
|
_rows[from].user->loadUserpic();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -371,14 +432,14 @@ void MembersBox::Inner::loadProfilePhotos() {
|
||||||
|
|
||||||
void MembersBox::Inner::chooseParticipant() {
|
void MembersBox::Inner::chooseParticipant() {
|
||||||
if (_selected < 0 || _selected >= _rows.size()) return;
|
if (_selected < 0 || _selected >= _rows.size()) return;
|
||||||
if (auto peer = _rows[_selected]) {
|
if (auto peer = _rows[_selected].user) {
|
||||||
Ui::hideLayer();
|
Ui::hideLayer();
|
||||||
Ui::showPeerProfile(peer);
|
Ui::showPeerProfile(peer);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void MembersBox::Inner::refresh() {
|
void MembersBox::Inner::refresh() {
|
||||||
if (_rows.isEmpty()) {
|
if (_rows.empty()) {
|
||||||
resize(width(), st::membersMarginTop + st::noContactsHeight + st::membersMarginBottom);
|
resize(width(), st::membersMarginTop + st::noContactsHeight + st::membersMarginBottom);
|
||||||
_aboutHeight = 0;
|
_aboutHeight = 0;
|
||||||
} else {
|
} else {
|
||||||
|
@ -402,10 +463,8 @@ MembersFilter MembersBox::Inner::filter() const {
|
||||||
|
|
||||||
MembersAlreadyIn MembersBox::Inner::already() const {
|
MembersAlreadyIn MembersBox::Inner::already() const {
|
||||||
MembersAlreadyIn result;
|
MembersAlreadyIn result;
|
||||||
for_const (auto peer, _rows) {
|
for_const (auto &&row, _rows) {
|
||||||
if (peer->isUser()) {
|
result.insert(row.user);
|
||||||
result.insert(peer->asUser());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
@ -423,34 +482,27 @@ void MembersBox::Inner::clearSel() {
|
||||||
updateSelection();
|
updateSelection();
|
||||||
}
|
}
|
||||||
|
|
||||||
MembersBox::Inner::MemberData *MembersBox::Inner::data(int32 index) {
|
void MembersBox::Inner::ensureData(Member &row) {
|
||||||
if (MemberData *result = _datas.at(index)) {
|
if (row.data) {
|
||||||
return result;
|
return;
|
||||||
}
|
}
|
||||||
MemberData *result = _datas[index] = new MemberData();
|
row.data = std::make_unique<RowData>();
|
||||||
result->name.setText(st::contactsNameStyle, _rows[index]->name, _textNameOptions);
|
row.data->name.setText(st::contactsNameStyle, row.user->name, _textNameOptions);
|
||||||
int32 t = unixtime();
|
auto now = unixtime();
|
||||||
result->online = App::onlineText(_rows[index], t);// lng_mediaview_date_time(lt_date, _dates[index].date().toString(qsl("dd.MM.yy")), lt_time, _dates[index].time().toString(cTimeFormat()));
|
row.data->online = App::onlineText(row.user, now);// lng_mediaview_date_time(lt_date, _dates[index].date().toString(qsl("dd.MM.yy")), lt_time, _dates[index].time().toString(cTimeFormat()));
|
||||||
result->onlineColor = App::onlineColorUse(_rows[index], t);
|
row.data->onlineColor = App::onlineColorUse(row.user, now);
|
||||||
if (_filter == MembersFilter::Recent) {
|
if (_filter == MembersFilter::Recent) {
|
||||||
result->canKick = _channel->canBanMembers() ? (_roles[index] == MemberRole::None) : false;
|
row.data->canKick = _channel->canBanMembers() ? (row.role == MemberRole::None) : false;
|
||||||
} else if (_filter == MembersFilter::Admins) {
|
} else if (_filter == MembersFilter::Admins) {
|
||||||
result->canKick = _channel->amCreator() ? (_roles[index] == MemberRole::Admin) : false;
|
row.data->canKick = _channel->amCreator() ? (row.role == MemberRole::Admin) : row.adminCanEdit;
|
||||||
} else {
|
} else {
|
||||||
result->canKick = false;
|
row.data->canKick = false;
|
||||||
}
|
}
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void MembersBox::Inner::clear() {
|
void MembersBox::Inner::clear() {
|
||||||
for (int32 i = 0, l = _datas.size(); i < l; ++i) {
|
|
||||||
delete _datas.at(i);
|
|
||||||
}
|
|
||||||
_datas.clear();
|
|
||||||
_rows.clear();
|
_rows.clear();
|
||||||
_dates.clear();
|
if (_kickBox) _kickBox->closeBox();
|
||||||
_roles.clear();
|
|
||||||
if (_kickBox) _kickBox->deleteLater();
|
|
||||||
clearSel();
|
clearSel();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -461,12 +513,15 @@ MembersBox::Inner::~Inner() {
|
||||||
void MembersBox::Inner::updateSelection() {
|
void MembersBox::Inner::updateSelection() {
|
||||||
if (!_mouseSelection) return;
|
if (!_mouseSelection) return;
|
||||||
|
|
||||||
QPoint p(mapFromGlobal(_lastMousePos));
|
auto p = mapFromGlobal(_lastMousePos);
|
||||||
p.setY(p.y() - st::membersMarginTop);
|
p.setY(p.y() - st::membersMarginTop);
|
||||||
bool in = parentWidget()->rect().contains(parentWidget()->mapFromGlobal(_lastMousePos));
|
auto in = parentWidget()->rect().contains(parentWidget()->mapFromGlobal(_lastMousePos));
|
||||||
auto selected = (in && p.y() >= 0 && p.y() < _rows.size() * _rowHeight) ? (p.y() / _rowHeight) : -1;
|
auto selected = (in && p.y() >= 0 && p.y() < _rows.size() * _rowHeight) ? (p.y() / _rowHeight) : -1;
|
||||||
auto kickSelected = selected;
|
auto kickSelected = selected;
|
||||||
if (selected >= 0 && (!data(selected)->canKick || !QRect(width() - _kickWidth - st::contactsPadding.right() - st::contactsCheckPosition.x(), selected * _rowHeight + st::contactsPadding.top() + (st::contactsPhotoSize - st::normalFont->height) / 2, _kickWidth, st::normalFont->height).contains(p))) {
|
if (selected >= 0) {
|
||||||
|
ensureData(_rows[selected]);
|
||||||
|
}
|
||||||
|
if (selected >= 0 && (!_rows[selected].data->canKick || !QRect(width() - _kickWidth - st::contactsPadding.right() - st::contactsCheckPosition.x(), selected * _rowHeight + st::contactsPadding.top() + (st::contactsPhotoSize - st::normalFont->height) / 2, _kickWidth, st::normalFont->height).contains(p))) {
|
||||||
kickSelected = -1;
|
kickSelected = -1;
|
||||||
}
|
}
|
||||||
if (_selected != selected || _kickSelected != kickSelected) {
|
if (_selected != selected || _kickSelected != kickSelected) {
|
||||||
|
@ -501,14 +556,14 @@ void MembersBox::Inner::updateSelectedRow() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void MembersBox::Inner::onPeerNameChanged(PeerData *peer, const PeerData::Names &oldNames, const PeerData::NameFirstChars &oldChars) {
|
void MembersBox::Inner::onPeerNameChanged(PeerData *peer, const PeerData::Names &oldNames, const PeerData::NameFirstChars &oldChars) {
|
||||||
for (int32 i = 0, l = _rows.size(); i < l; ++i) {
|
for (auto i= 0, l = int(_rows.size()); i != l; ++i) {
|
||||||
if (_rows.at(i) == peer) {
|
auto &row = _rows[i];
|
||||||
if (_datas.at(i)) {
|
if (row.user == peer) {
|
||||||
_datas.at(i)->name.setText(st::contactsNameStyle, peer->name, _textNameOptions);
|
if (row.data) {
|
||||||
|
row.data->name.setText(st::contactsNameStyle, peer->name, _textNameOptions);
|
||||||
update(0, st::membersMarginTop + i * _rowHeight, width(), _rowHeight);
|
update(0, st::membersMarginTop + i * _rowHeight, width(), _rowHeight);
|
||||||
} else {
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -522,9 +577,6 @@ void MembersBox::Inner::membersReceived(const MTPchannels_ChannelParticipants &r
|
||||||
auto &d = result.c_channels_channelParticipants();
|
auto &d = result.c_channels_channelParticipants();
|
||||||
auto &v = d.vparticipants.v;
|
auto &v = d.vparticipants.v;
|
||||||
_rows.reserve(v.size());
|
_rows.reserve(v.size());
|
||||||
_datas.reserve(v.size());
|
|
||||||
_dates.reserve(v.size());
|
|
||||||
_roles.reserve(v.size());
|
|
||||||
|
|
||||||
if (_filter == MembersFilter::Recent && _channel->membersCount() < d.vcount.v) {
|
if (_filter == MembersFilter::Recent && _channel->membersCount() < d.vcount.v) {
|
||||||
_channel->setMembersCount(d.vcount.v);
|
_channel->setMembersCount(d.vcount.v);
|
||||||
|
@ -535,11 +587,15 @@ void MembersBox::Inner::membersReceived(const MTPchannels_ChannelParticipants &r
|
||||||
}
|
}
|
||||||
App::feedUsers(d.vusers);
|
App::feedUsers(d.vusers);
|
||||||
|
|
||||||
auto emptyRights = MTP_channelAdminRights(MTP_flags(0));
|
auto emptyAdminRights = MTP_channelAdminRights(MTP_flags(0));
|
||||||
for (QVector<MTPChannelParticipant>::const_iterator i = v.cbegin(), e = v.cend(); i != e; ++i) {
|
auto emptyRestrictedRights = MTP_channelBannedRights(MTP_flags(0), MTP_int(0));
|
||||||
int32 userId = 0, addedTime = 0;
|
for (auto i = v.cbegin(), e = v.cend(); i != e; ++i) {
|
||||||
|
auto userId = UserId(0);
|
||||||
|
auto addedTime = TimeId(0);
|
||||||
auto role = MemberRole::None;
|
auto role = MemberRole::None;
|
||||||
auto rights = emptyRights;
|
auto adminCanEdit = false;
|
||||||
|
auto adminRights = emptyAdminRights;
|
||||||
|
auto restrictedRights = emptyRestrictedRights;
|
||||||
switch (i->type()) {
|
switch (i->type()) {
|
||||||
case mtpc_channelParticipant:
|
case mtpc_channelParticipant:
|
||||||
userId = i->c_channelParticipant().vuser_id.v;
|
userId = i->c_channelParticipant().vuser_id.v;
|
||||||
|
@ -554,7 +610,8 @@ void MembersBox::Inner::membersReceived(const MTPchannels_ChannelParticipants &r
|
||||||
role = MemberRole::Admin;
|
role = MemberRole::Admin;
|
||||||
userId = i->c_channelParticipantAdmin().vuser_id.v;
|
userId = i->c_channelParticipantAdmin().vuser_id.v;
|
||||||
addedTime = i->c_channelParticipantAdmin().vdate.v;
|
addedTime = i->c_channelParticipantAdmin().vdate.v;
|
||||||
rights = i->c_channelParticipantAdmin().vadmin_rights;
|
adminRights = i->c_channelParticipantAdmin().vadmin_rights;
|
||||||
|
adminCanEdit = i->c_channelParticipantAdmin().is_can_edit();
|
||||||
break;
|
break;
|
||||||
case mtpc_channelParticipantCreator:
|
case mtpc_channelParticipantCreator:
|
||||||
userId = i->c_channelParticipantCreator().vuser_id.v;
|
userId = i->c_channelParticipantCreator().vuser_id.v;
|
||||||
|
@ -564,34 +621,42 @@ void MembersBox::Inner::membersReceived(const MTPchannels_ChannelParticipants &r
|
||||||
case mtpc_channelParticipantBanned:
|
case mtpc_channelParticipantBanned:
|
||||||
userId = i->c_channelParticipantBanned().vuser_id.v;
|
userId = i->c_channelParticipantBanned().vuser_id.v;
|
||||||
addedTime = i->c_channelParticipantBanned().vdate.v;
|
addedTime = i->c_channelParticipantBanned().vdate.v;
|
||||||
role = MemberRole::Kicked;
|
restrictedRights = i->c_channelParticipantBanned().vbanned_rights;
|
||||||
|
role = MemberRole::Restricted;
|
||||||
}
|
}
|
||||||
if (auto user = App::userLoaded(userId)) {
|
if (auto user = App::userLoaded(userId)) {
|
||||||
_rows.push_back(user);
|
auto row = Member(user);
|
||||||
_dates.push_back(date(addedTime));
|
row.adminCanEdit = adminCanEdit;
|
||||||
_roles.push_back(role);
|
row.adminRights = adminRights;
|
||||||
_adminRights.push_back(rights);
|
row.restrictedRights = restrictedRights;
|
||||||
_datas.push_back(nullptr);
|
row.date = date(addedTime);
|
||||||
|
row.role = role;
|
||||||
|
_rows.push_back(std::move(row));
|
||||||
|
if (role == MemberRole::Creator && _channel->mgInfo) {
|
||||||
|
_channel->mgInfo->creator = user;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// update admins if we got all of them
|
// update admins if we got all of them
|
||||||
if (_filter == MembersFilter::Admins && _channel->isMegagroup() && _rows.size() < Global::ChatSizeMax()) {
|
if (_filter == MembersFilter::Admins && _channel->isMegagroup() && _rows.size() < Global::ChatSizeMax()) {
|
||||||
_channel->mgInfo->lastAdmins.clear();
|
_channel->mgInfo->lastAdmins.clear();
|
||||||
for (int32 i = 0, l = _rows.size(); i != l; ++i) {
|
for (auto &&row : _rows) {
|
||||||
if (_roles[i] == MemberRole::Admin) {
|
if (row.role == MemberRole::Admin) {
|
||||||
_channel->mgInfo->lastAdmins.insert(_rows[i], _adminRights[i]);
|
_channel->mgInfo->lastAdmins.insert(row.user, MegagroupInfo::Admin { row.adminRights, row.adminCanEdit });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Notify::peerUpdatedDelayed(_channel, Notify::PeerUpdate::Flag::AdminsChanged);
|
Notify::peerUpdatedDelayed(_channel, Notify::PeerUpdate::Flag::AdminsChanged);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_rows.isEmpty()) {
|
if (_rows.empty()) {
|
||||||
_rows.push_back(App::self());
|
auto row = Member(App::self());
|
||||||
_dates.push_back(date(MTP_int(_channel->date)));
|
row.date = date(MTP_int(_channel->date));
|
||||||
_roles.push_back(MemberRole::Self);
|
row.role = MemberRole::Self;
|
||||||
_datas.push_back(nullptr);
|
row.adminRights = _channel->adminRightsBoxed();
|
||||||
|
row.restrictedRights = _channel->restrictedRightsBoxed();
|
||||||
|
_rows.push_back(std::move(row));
|
||||||
}
|
}
|
||||||
|
|
||||||
clearSel();
|
clearSel();
|
||||||
|
@ -608,22 +673,7 @@ bool MembersBox::Inner::membersFailed(const RPCError &error, mtpRequestId req) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void MembersBox::Inner::kickDone(const MTPUpdates &result, mtpRequestId req) {
|
bool MembersBox::Inner::kickFail(const RPCError &error) {
|
||||||
App::main()->sentUpdatesReceived(result);
|
|
||||||
|
|
||||||
if (_kickRequestId != req) return;
|
|
||||||
removeKicked();
|
|
||||||
if (_kickBox) _kickBox->closeBox();
|
|
||||||
}
|
|
||||||
|
|
||||||
void MembersBox::Inner::kickAdminDone(const MTPUpdates &result, mtpRequestId req) {
|
|
||||||
if (_kickRequestId != req) return;
|
|
||||||
if (App::main()) App::main()->sentUpdatesReceived(result);
|
|
||||||
removeKicked();
|
|
||||||
if (_kickBox) _kickBox->closeBox();
|
|
||||||
}
|
|
||||||
|
|
||||||
bool MembersBox::Inner::kickFail(const RPCError &error, mtpRequestId req) {
|
|
||||||
if (MTP::isDefaultHandledError(error)) return false;
|
if (MTP::isDefaultHandledError(error)) return false;
|
||||||
|
|
||||||
if (_kickBox) _kickBox->closeBox();
|
if (_kickBox) _kickBox->closeBox();
|
||||||
|
@ -631,15 +681,12 @@ bool MembersBox::Inner::kickFail(const RPCError &error, mtpRequestId req) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void MembersBox::Inner::removeKicked() {
|
void MembersBox::Inner::removeKicked(UserData *kicked) {
|
||||||
_kickRequestId = 0;
|
auto it = std::find_if(_rows.begin(), _rows.end(), [this, kicked](auto &&row) {
|
||||||
int32 index = _rows.indexOf(_kickConfirm);
|
return (row.user == kicked);
|
||||||
if (index >= 0) {
|
});
|
||||||
_rows.removeAt(index);
|
if (it != _rows.end()) {
|
||||||
delete _datas.at(index);
|
_rows.erase(it);
|
||||||
_datas.removeAt(index);
|
|
||||||
_dates.removeAt(index);
|
|
||||||
_roles.removeAt(index);
|
|
||||||
clearSel();
|
clearSel();
|
||||||
if (_filter == MembersFilter::Recent && _channel->membersCount() > 1) {
|
if (_filter == MembersFilter::Recent && _channel->membersCount() > 1) {
|
||||||
_channel->setMembersCount(_channel->membersCount() - 1);
|
_channel->setMembersCount(_channel->membersCount() - 1);
|
||||||
|
@ -650,5 +697,4 @@ void MembersBox::Inner::removeKicked() {
|
||||||
}
|
}
|
||||||
refresh();
|
refresh();
|
||||||
}
|
}
|
||||||
_kickConfirm = 0;
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -68,6 +68,7 @@ protected:
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void onAdd();
|
void onAdd();
|
||||||
|
void refreshButtons();
|
||||||
|
|
||||||
ChannelData *_channel = nullptr;
|
ChannelData *_channel = nullptr;
|
||||||
MembersFilter _filter = MembersFilter::Recent;
|
MembersFilter _filter = MembersFilter::Recent;
|
||||||
|
@ -86,13 +87,11 @@ class MembersBox::Inner : public TWidget, public RPCSender, private base::Subscr
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
Inner(QWidget *parent, ChannelData *channel, MembersFilter filter);
|
Inner(QWidget *parent, gsl::not_null<ChannelData*> channel, MembersFilter filter);
|
||||||
|
|
||||||
void selectSkip(int32 dir);
|
void selectSkip(int32 dir);
|
||||||
void selectSkipPage(int32 h, int32 dir);
|
void selectSkipPage(int32 h, int32 dir);
|
||||||
|
|
||||||
void chooseParticipant();
|
|
||||||
|
|
||||||
void refresh();
|
void refresh();
|
||||||
|
|
||||||
ChannelData *channel() const;
|
ChannelData *channel() const;
|
||||||
|
@ -127,20 +126,34 @@ protected:
|
||||||
void mouseReleaseEvent(QMouseEvent *e) override;
|
void mouseReleaseEvent(QMouseEvent *e) override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
struct MemberData {
|
struct RowData;
|
||||||
MemberData();
|
enum class MemberRole {
|
||||||
~MemberData();
|
None,
|
||||||
|
Self,
|
||||||
std::unique_ptr<Ui::RippleAnimation> ripple;
|
Creator,
|
||||||
int rippleRowTop = 0;
|
Admin,
|
||||||
Text name;
|
Restricted,
|
||||||
QString online;
|
Kicked,
|
||||||
bool onlineColor;
|
|
||||||
bool canKick;
|
|
||||||
};
|
};
|
||||||
void addRipple(MemberData *data);
|
struct Member {
|
||||||
void stopLastRipple(MemberData *data);
|
Member(gsl::not_null<UserData*> user);
|
||||||
|
Member(Member &&other);
|
||||||
|
Member &operator=(Member &&other);
|
||||||
|
~Member();
|
||||||
|
|
||||||
|
gsl::not_null<UserData*> user;
|
||||||
|
QDateTime date;
|
||||||
|
MemberRole role = MemberRole::None;
|
||||||
|
bool adminCanEdit = false;
|
||||||
|
MTPChannelAdminRights adminRights;
|
||||||
|
MTPChannelBannedRights restrictedRights;
|
||||||
|
std::unique_ptr<RowData> data;
|
||||||
|
};
|
||||||
|
void addRipple(gsl::not_null<RowData*> data);
|
||||||
|
void stopLastRipple(gsl::not_null<RowData*> data);
|
||||||
void setPressed(int pressed);
|
void setPressed(int pressed);
|
||||||
|
void chooseParticipant();
|
||||||
|
void actionPressed(Member &row);
|
||||||
|
|
||||||
void updateSelection();
|
void updateSelection();
|
||||||
void loadProfilePhotos();
|
void loadProfilePhotos();
|
||||||
|
@ -148,17 +161,15 @@ private:
|
||||||
void updateRowWithTop(int rowTop);
|
void updateRowWithTop(int rowTop);
|
||||||
int getSelectedRowTop() const;
|
int getSelectedRowTop() const;
|
||||||
void updateSelectedRow();
|
void updateSelectedRow();
|
||||||
MemberData *data(int32 index);
|
void ensureData(Member &row);
|
||||||
|
|
||||||
void paintDialog(Painter &p, TimeMs ms, PeerData *peer, MemberData *data, bool selected, bool kickSelected);
|
void paintDialog(Painter &p, TimeMs ms, Member &row, bool selected, bool kickSelected);
|
||||||
|
|
||||||
void membersReceived(const MTPchannels_ChannelParticipants &result, mtpRequestId req);
|
void membersReceived(const MTPchannels_ChannelParticipants &result, mtpRequestId req);
|
||||||
bool membersFailed(const RPCError &error, mtpRequestId req);
|
bool membersFailed(const RPCError &error, mtpRequestId req);
|
||||||
|
|
||||||
void kickDone(const MTPUpdates &result, mtpRequestId req);
|
bool kickFail(const RPCError &error);
|
||||||
void kickAdminDone(const MTPUpdates &result, mtpRequestId req);
|
void removeKicked(UserData *kicked);
|
||||||
bool kickFail(const RPCError &error, mtpRequestId req);
|
|
||||||
void removeKicked();
|
|
||||||
|
|
||||||
void clear();
|
void clear();
|
||||||
|
|
||||||
|
@ -166,7 +177,7 @@ private:
|
||||||
int _visibleTop = 0;
|
int _visibleTop = 0;
|
||||||
int _visibleBottom = 0;
|
int _visibleBottom = 0;
|
||||||
|
|
||||||
ChannelData *_channel = nullptr;
|
gsl::not_null<ChannelData*> _channel;
|
||||||
MembersFilter _filter;
|
MembersFilter _filter;
|
||||||
|
|
||||||
QString _kickText;
|
QString _kickText;
|
||||||
|
@ -179,26 +190,11 @@ private:
|
||||||
int _kickPressed = -1;
|
int _kickPressed = -1;
|
||||||
bool _mouseSelection = false;
|
bool _mouseSelection = false;
|
||||||
|
|
||||||
UserData *_kickConfirm = nullptr;
|
QPointer<BoxContent> _kickBox;
|
||||||
mtpRequestId _kickRequestId = 0;
|
|
||||||
|
|
||||||
QPointer<ConfirmBox> _kickBox;
|
|
||||||
|
|
||||||
enum class MemberRole {
|
|
||||||
None,
|
|
||||||
Self,
|
|
||||||
Creator,
|
|
||||||
Admin,
|
|
||||||
Kicked,
|
|
||||||
};
|
|
||||||
|
|
||||||
bool _loading = true;
|
bool _loading = true;
|
||||||
mtpRequestId _loadingRequestId = 0;
|
mtpRequestId _loadingRequestId = 0;
|
||||||
QVector<UserData*> _rows;
|
std::vector<Member> _rows;
|
||||||
QVector<QDateTime> _dates;
|
|
||||||
QVector<MemberRole> _roles;
|
|
||||||
QVector<MTPChannelAdminRights> _adminRights;
|
|
||||||
QVector<MemberData*> _datas;
|
|
||||||
|
|
||||||
int _aboutWidth = 0;
|
int _aboutWidth = 0;
|
||||||
Text _about;
|
Text _about;
|
||||||
|
|
|
@ -900,9 +900,9 @@ HistoryItem *History::createItem(const MTPMessage &msg, bool applyServiceAction,
|
||||||
if (peer->isMegagroup()) {
|
if (peer->isMegagroup()) {
|
||||||
if (auto user = App::userLoaded(uid)) {
|
if (auto user = App::userLoaded(uid)) {
|
||||||
auto channel = peer->asChannel();
|
auto channel = peer->asChannel();
|
||||||
auto megagroupInfo = channel->mgInfo;
|
auto &megagroupInfo = channel->mgInfo;
|
||||||
|
|
||||||
int32 index = megagroupInfo->lastParticipants.indexOf(user);
|
auto index = megagroupInfo->lastParticipants.indexOf(user);
|
||||||
if (index >= 0) {
|
if (index >= 0) {
|
||||||
megagroupInfo->lastParticipants.removeAt(index);
|
megagroupInfo->lastParticipants.removeAt(index);
|
||||||
Notify::peerUpdatedDelayed(peer, Notify::PeerUpdate::Flag::MembersChanged);
|
Notify::peerUpdatedDelayed(peer, Notify::PeerUpdate::Flag::MembersChanged);
|
||||||
|
@ -1117,20 +1117,24 @@ HistoryItem *History::addNewItem(HistoryItem *adding, bool newMsg) {
|
||||||
|
|
||||||
adding->addToOverview(AddToOverviewNew);
|
adding->addToOverview(AddToOverviewNew);
|
||||||
if (adding->from()->id) {
|
if (adding->from()->id) {
|
||||||
if (adding->from()->isUser()) {
|
if (auto user = adding->from()->asUser()) {
|
||||||
QList<UserData*> *lastAuthors = 0;
|
auto getLastAuthors = [this]() -> QList<gsl::not_null<UserData*>>* {
|
||||||
if (peer->isChat()) {
|
if (auto chat = peer->asChat()) {
|
||||||
lastAuthors = &peer->asChat()->lastAuthors;
|
return &chat->lastAuthors;
|
||||||
} else if (peer->isMegagroup()) {
|
} else if (auto channel = peer->asMegagroup()) {
|
||||||
lastAuthors = &peer->asChannel()->mgInfo->lastParticipants;
|
return &channel->mgInfo->lastParticipants;
|
||||||
|
}
|
||||||
|
return nullptr;
|
||||||
|
};
|
||||||
|
if (auto channel = peer->asMegagroup()) {
|
||||||
if (adding->from()->asUser()->botInfo) {
|
if (adding->from()->asUser()->botInfo) {
|
||||||
peer->asChannel()->mgInfo->bots.insert(adding->from()->asUser());
|
channel->mgInfo->bots.insert(adding->from()->asUser());
|
||||||
if (peer->asChannel()->mgInfo->botStatus != 0 && peer->asChannel()->mgInfo->botStatus < 2) {
|
if (channel->mgInfo->botStatus != 0 && channel->mgInfo->botStatus < 2) {
|
||||||
peer->asChannel()->mgInfo->botStatus = 2;
|
channel->mgInfo->botStatus = 2;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (lastAuthors) {
|
if (auto lastAuthors = getLastAuthors()) {
|
||||||
int prev = lastAuthors->indexOf(adding->from()->asUser());
|
int prev = lastAuthors->indexOf(adding->from()->asUser());
|
||||||
if (prev > 0) {
|
if (prev > 0) {
|
||||||
lastAuthors->removeAt(prev);
|
lastAuthors->removeAt(prev);
|
||||||
|
@ -1146,15 +1150,17 @@ HistoryItem *History::addNewItem(HistoryItem *adding, bool newMsg) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (adding->definesReplyKeyboard()) {
|
if (adding->definesReplyKeyboard()) {
|
||||||
MTPDreplyKeyboardMarkup::Flags markupFlags = adding->replyKeyboardFlags();
|
auto markupFlags = adding->replyKeyboardFlags();
|
||||||
if (!(markupFlags & MTPDreplyKeyboardMarkup::Flag::f_selective) || adding->mentionsMe()) {
|
if (!(markupFlags & MTPDreplyKeyboardMarkup::Flag::f_selective) || adding->mentionsMe()) {
|
||||||
OrderedSet<PeerData*> *markupSenders = 0;
|
auto getMarkupSenders = [this]() -> OrderedSet<gsl::not_null<PeerData*>>* {
|
||||||
if (peer->isChat()) {
|
if (auto chat = peer->asChat()) {
|
||||||
markupSenders = &peer->asChat()->markupSenders;
|
return &chat->markupSenders;
|
||||||
} else if (peer->isMegagroup()) {
|
} else if (auto channel = peer->asMegagroup()) {
|
||||||
markupSenders = &peer->asChannel()->mgInfo->markupSenders;
|
return &channel->mgInfo->markupSenders;
|
||||||
}
|
}
|
||||||
if (markupSenders) {
|
return nullptr;
|
||||||
|
};
|
||||||
|
if (auto markupSenders = getMarkupSenders()) {
|
||||||
markupSenders->insert(adding->from());
|
markupSenders->insert(adding->from());
|
||||||
}
|
}
|
||||||
if (markupFlags & MTPDreplyKeyboardMarkup_ClientFlag::f_zero) { // zero markup means replyKeyboardHide
|
if (markupFlags & MTPDreplyKeyboardMarkup_ClientFlag::f_zero) { // zero markup means replyKeyboardHide
|
||||||
|
@ -1299,8 +1305,8 @@ void History::addOlderSlice(const QVector<MTPMessage> &slice) {
|
||||||
} else if (loadedAtBottom()) { // add photos to overview and authors to lastAuthors
|
} else if (loadedAtBottom()) { // add photos to overview and authors to lastAuthors
|
||||||
bool channel = isChannel();
|
bool channel = isChannel();
|
||||||
int32 mask = 0;
|
int32 mask = 0;
|
||||||
QList<UserData*> *lastAuthors = nullptr;
|
QList<gsl::not_null<UserData*>> *lastAuthors = nullptr;
|
||||||
OrderedSet<PeerData*> *markupSenders = nullptr;
|
OrderedSet<gsl::not_null<PeerData*>> *markupSenders = nullptr;
|
||||||
if (peer->isChat()) {
|
if (peer->isChat()) {
|
||||||
lastAuthors = &peer->asChat()->lastAuthors;
|
lastAuthors = &peer->asChat()->lastAuthors;
|
||||||
markupSenders = &peer->asChat()->markupSenders;
|
markupSenders = &peer->asChat()->markupSenders;
|
||||||
|
|
|
@ -795,11 +795,18 @@ bool HistoryItem::canEdit(const QDateTime &cur) const {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (out() || messageToMyself) {
|
if (messageToMyself) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (auto channel = _history->peer->asChannel()) {
|
if (auto channel = _history->peer->asChannel()) {
|
||||||
return channel->canDeleteMessages();
|
if (channel->canEditMessages()) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (out()) {
|
||||||
|
return !isPost() || channel->canPublish();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return out();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
|
@ -814,16 +821,13 @@ bool HistoryItem::canDelete() const {
|
||||||
if (id == 1) {
|
if (id == 1) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (channel->amCreator()) {
|
if (channel->canDeleteMessages()) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (isPost()) {
|
if (out() && toHistoryMessage()) {
|
||||||
if (out()) {
|
return isPost() ? channel->canPublish() : true;
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
return (channel->canDeleteMessages() || out());
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool HistoryItem::canDeleteForEveryone(const QDateTime &cur) const {
|
bool HistoryItem::canDeleteForEveryone(const QDateTime &cur) const {
|
||||||
|
@ -857,10 +861,11 @@ bool HistoryItem::canDeleteForEveryone(const QDateTime &cur) const {
|
||||||
|
|
||||||
bool HistoryItem::suggestBanReport() const {
|
bool HistoryItem::suggestBanReport() const {
|
||||||
auto channel = history()->peer->asChannel();
|
auto channel = history()->peer->asChannel();
|
||||||
if (!channel || !channel->canBanMembers()) {
|
auto fromUser = from()->asUser();
|
||||||
|
if (!channel || !fromUser || !channel->canRestrictUser(fromUser)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return !isPost() && !out() && from()->isUser() && toHistoryMessage();
|
return !isPost() && !out() && toHistoryMessage();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool HistoryItem::suggestDeleteAllReport() const {
|
bool HistoryItem::suggestDeleteAllReport() const {
|
||||||
|
|
|
@ -4111,7 +4111,7 @@ void HistoryWidget::updateOnlineDisplayTimer() {
|
||||||
ChatData *chat = _peer->asChat();
|
ChatData *chat = _peer->asChat();
|
||||||
if (chat->participants.isEmpty()) return;
|
if (chat->participants.isEmpty()) return;
|
||||||
|
|
||||||
for (ChatData::Participants::const_iterator i = chat->participants.cbegin(), e = chat->participants.cend(); i != e; ++i) {
|
for (auto i = chat->participants.cbegin(), e = chat->participants.cend(); i != e; ++i) {
|
||||||
int32 onlineWillChangeIn = App::onlineWillChangeIn(i.key(), t);
|
int32 onlineWillChangeIn = App::onlineWillChangeIn(i.key(), t);
|
||||||
if (onlineWillChangeIn < minIn) {
|
if (onlineWillChangeIn < minIn) {
|
||||||
minIn = onlineWillChangeIn;
|
minIn = onlineWillChangeIn;
|
||||||
|
|
|
@ -1226,7 +1226,7 @@ void MainWidget::clearHistory(PeerData *peer) {
|
||||||
MTP::send(MTPmessages_DeleteHistory(MTP_flags(flags), peer->input, MTP_int(0)), rpcDone(&MainWidget::deleteHistoryPart, request));
|
MTP::send(MTPmessages_DeleteHistory(MTP_flags(flags), peer->input, MTP_int(0)), rpcDone(&MainWidget::deleteHistoryPart, request));
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWidget::addParticipants(PeerData *chatOrChannel, const QVector<UserData*> &users) {
|
void MainWidget::addParticipants(PeerData *chatOrChannel, const std::vector<gsl::not_null<UserData*>> &users) {
|
||||||
if (chatOrChannel->isChat()) {
|
if (chatOrChannel->isChat()) {
|
||||||
auto chat = chatOrChannel->asChat();
|
auto chat = chatOrChannel->asChat();
|
||||||
for_const (auto user, users) {
|
for_const (auto user, users) {
|
||||||
|
@ -1234,8 +1234,8 @@ void MainWidget::addParticipants(PeerData *chatOrChannel, const QVector<UserData
|
||||||
}
|
}
|
||||||
} else if (chatOrChannel->isChannel()) {
|
} else if (chatOrChannel->isChannel()) {
|
||||||
QVector<MTPInputUser> inputUsers;
|
QVector<MTPInputUser> inputUsers;
|
||||||
inputUsers.reserve(qMin(users.size(), int(MaxUsersPerInvite)));
|
inputUsers.reserve(qMin(int(users.size()), int(MaxUsersPerInvite)));
|
||||||
for (QVector<UserData*>::const_iterator i = users.cbegin(), e = users.cend(); i != e; ++i) {
|
for (auto i = users.cbegin(), e = users.cend(); i != e; ++i) {
|
||||||
inputUsers.push_back((*i)->inputUser);
|
inputUsers.push_back((*i)->inputUser);
|
||||||
if (inputUsers.size() == MaxUsersPerInvite) {
|
if (inputUsers.size() == MaxUsersPerInvite) {
|
||||||
MTP::send(MTPchannels_InviteToChannel(chatOrChannel->asChannel()->inputChannel, MTP_vector<MTPInputUser>(inputUsers)), rpcDone(&MainWidget::inviteToChannelDone, chatOrChannel->asChannel()), rpcFail(&MainWidget::addParticipantsFail, chatOrChannel->asChannel()), 0, 5);
|
MTP::send(MTPchannels_InviteToChannel(chatOrChannel->asChannel()->inputChannel, MTP_vector<MTPInputUser>(inputUsers)), rpcDone(&MainWidget::inviteToChannelDone, chatOrChannel->asChannel()), rpcFail(&MainWidget::addParticipantsFail, chatOrChannel->asChannel()), 0, 5);
|
||||||
|
|
|
@ -273,7 +273,7 @@ public:
|
||||||
void clearHistory(PeerData *peer);
|
void clearHistory(PeerData *peer);
|
||||||
void deleteAllFromUser(ChannelData *channel, UserData *from);
|
void deleteAllFromUser(ChannelData *channel, UserData *from);
|
||||||
|
|
||||||
void addParticipants(PeerData *chatOrChannel, const QVector<UserData*> &users);
|
void addParticipants(PeerData *chatOrChannel, const std::vector<gsl::not_null<UserData*>> &users);
|
||||||
struct UserAndPeer {
|
struct UserAndPeer {
|
||||||
UserData *user;
|
UserData *user;
|
||||||
PeerData *peer;
|
PeerData *peer;
|
||||||
|
|
|
@ -34,6 +34,8 @@ struct PeerUpdate {
|
||||||
PeerData *peer;
|
PeerData *peer;
|
||||||
|
|
||||||
enum class Flag {
|
enum class Flag {
|
||||||
|
None = 0x00000000U,
|
||||||
|
|
||||||
// Common flags
|
// Common flags
|
||||||
NameChanged = 0x00000001U,
|
NameChanged = 0x00000001U,
|
||||||
UsernameChanged = 0x00000002U,
|
UsernameChanged = 0x00000002U,
|
||||||
|
@ -66,11 +68,7 @@ struct PeerUpdate {
|
||||||
|
|
||||||
// For channels
|
// For channels
|
||||||
ChannelAmIn = 0x00010000U,
|
ChannelAmIn = 0x00010000U,
|
||||||
ChannelAmEditor = 0x00020000U,
|
ChannelRightsChanged = 0x00020000U,
|
||||||
ChannelCanEditInformation = 0x00040000U,
|
|
||||||
ChannelCanAddMembers = 0x00080000U,
|
|
||||||
ChannelCanViewAdmins = 0x00100000U,
|
|
||||||
ChannelCanViewMembers = 0x00200000U,
|
|
||||||
};
|
};
|
||||||
Q_DECLARE_FLAGS(Flags, Flag);
|
Q_DECLARE_FLAGS(Flags, Flag);
|
||||||
Flags flags = 0;
|
Flags flags = 0;
|
||||||
|
|
|
@ -31,8 +31,7 @@ namespace Profile {
|
||||||
using UpdateFlag = Notify::PeerUpdate::Flag;
|
using UpdateFlag = Notify::PeerUpdate::Flag;
|
||||||
|
|
||||||
ChannelMembersWidget::ChannelMembersWidget(QWidget *parent, PeerData *peer) : BlockWidget(parent, peer, lang(lng_profile_participants_section)) {
|
ChannelMembersWidget::ChannelMembersWidget(QWidget *parent, PeerData *peer) : BlockWidget(parent, peer, lang(lng_profile_participants_section)) {
|
||||||
auto observeEvents = UpdateFlag::ChannelCanViewAdmins
|
auto observeEvents = UpdateFlag::ChannelRightsChanged
|
||||||
| UpdateFlag::ChannelCanViewMembers
|
|
||||||
| UpdateFlag::AdminsChanged
|
| UpdateFlag::AdminsChanged
|
||||||
| UpdateFlag::MembersChanged;
|
| UpdateFlag::MembersChanged;
|
||||||
subscribe(Notify::PeerUpdated(), Notify::PeerUpdatedHandler(observeEvents, [this](const Notify::PeerUpdate &update) {
|
subscribe(Notify::PeerUpdated(), Notify::PeerUpdatedHandler(observeEvents, [this](const Notify::PeerUpdate &update) {
|
||||||
|
@ -47,10 +46,10 @@ void ChannelMembersWidget::notifyPeerUpdated(const Notify::PeerUpdate &update) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (update.flags & (UpdateFlag::ChannelCanViewAdmins | UpdateFlag::AdminsChanged)) {
|
if (update.flags & (UpdateFlag::ChannelRightsChanged | UpdateFlag::AdminsChanged)) {
|
||||||
refreshAdmins();
|
refreshAdmins();
|
||||||
}
|
}
|
||||||
if (update.flags & (UpdateFlag::ChannelCanViewMembers | UpdateFlag::MembersChanged)) {
|
if (update.flags & (UpdateFlag::ChannelRightsChanged | UpdateFlag::MembersChanged)) {
|
||||||
refreshMembers();
|
refreshMembers();
|
||||||
}
|
}
|
||||||
refreshVisibility();
|
refreshVisibility();
|
||||||
|
|
|
@ -67,41 +67,35 @@ GroupMembersWidget::GroupMembersWidget(QWidget *parent, PeerData *peer, TitleVis
|
||||||
refreshMembers();
|
refreshMembers();
|
||||||
}
|
}
|
||||||
|
|
||||||
void GroupMembersWidget::addAdmin(gsl::not_null<UserData*> user) {
|
void GroupMembersWidget::editAdmin(gsl::not_null<UserData*> user) {
|
||||||
auto megagroup = peer()->asMegagroup();
|
auto megagroup = peer()->asMegagroup();
|
||||||
if (!megagroup) {
|
if (!megagroup) {
|
||||||
return; // not supported
|
return; // not supported
|
||||||
}
|
}
|
||||||
auto currentRights = megagroup->mgInfo->lastAdmins.value(user, MTP_channelAdminRights(MTP_flags(0)));
|
auto defaultAdmin = MegagroupInfo::Admin { EditAdminBox::DefaultRights(megagroup) };
|
||||||
|
auto currentRights = megagroup->mgInfo->lastAdmins.value(user, defaultAdmin).rights;
|
||||||
Ui::show(Box<EditAdminBox>(megagroup, user, currentRights, base::lambda_guarded(this, [this, megagroup, user](const MTPChannelAdminRights &rights) {
|
Ui::show(Box<EditAdminBox>(megagroup, user, currentRights, base::lambda_guarded(this, [this, megagroup, user](const MTPChannelAdminRights &rights) {
|
||||||
Ui::hideLayer();
|
Ui::hideLayer();
|
||||||
MTP::send(MTPchannels_EditAdmin(megagroup->inputChannel, user->inputUser, rights), rpcDone(base::lambda_guarded(this, [this, megagroup, user, rights](const MTPUpdates &result) {
|
MTP::send(MTPchannels_EditAdmin(megagroup->inputChannel, user->inputUser, rights), rpcDone(base::lambda_guarded(this, [this, megagroup, user, rights](const MTPUpdates &result) {
|
||||||
if (App::main()) App::main()->sentUpdatesReceived(result);
|
if (App::main()) App::main()->sentUpdatesReceived(result);
|
||||||
megagroup->mgInfo->lastAdmins.insert(user, rights);
|
megagroup->applyEditAdmin(user, rights);
|
||||||
megagroup->setAdminsCount(megagroup->adminsCount() + 1);
|
|
||||||
if (App::main()) emit App::main()->peerUpdated(megagroup);
|
|
||||||
Notify::peerUpdatedDelayed(megagroup, Notify::PeerUpdate::Flag::AdminsChanged);
|
|
||||||
})));
|
})));
|
||||||
})));
|
})));
|
||||||
}
|
}
|
||||||
|
|
||||||
void GroupMembersWidget::removeAdmin(gsl::not_null<UserData*> user) {
|
void GroupMembersWidget::restrictUser(gsl::not_null<UserData*> user) {
|
||||||
auto text = lng_profile_sure_kick_admin(lt_user, user->firstName);
|
auto megagroup = peer()->asMegagroup();
|
||||||
Ui::show(Box<ConfirmBox>(text, lang(lng_box_remove), base::lambda_guarded(this, [this, user] {
|
if (!megagroup) {
|
||||||
|
return; // not supported
|
||||||
|
}
|
||||||
|
auto defaultRestricted = MegagroupInfo::Restricted { EditRestrictedBox::DefaultRights(megagroup) };
|
||||||
|
auto currentRights = megagroup->mgInfo->lastRestricted.value(user, defaultRestricted).rights;
|
||||||
|
Ui::show(Box<EditRestrictedBox>(megagroup, user, currentRights, base::lambda_guarded(this, [this, megagroup, user](const MTPChannelBannedRights &rights) {
|
||||||
Ui::hideLayer();
|
Ui::hideLayer();
|
||||||
if (auto chat = peer()->asChat()) {
|
MTP::send(MTPchannels_EditBanned(megagroup->inputChannel, user->inputUser, rights), rpcDone(base::lambda_guarded(this, [this, megagroup, user, rights](const MTPUpdates &result) {
|
||||||
// not supported
|
if (App::main()) App::main()->sentUpdatesReceived(result);
|
||||||
} else if (auto channel = peer()->asMegagroup()) {
|
megagroup->applyEditBanned(user, rights);
|
||||||
MTP::send(MTPchannels_EditAdmin(channel->inputChannel, user->inputUser, MTP_channelAdminRights(MTP_flags(0))), rpcDone(base::lambda_guarded(this, [this, channel, user](const MTPUpdates &result) {
|
})));
|
||||||
if (App::main()) App::main()->sentUpdatesReceived(result);
|
|
||||||
channel->mgInfo->lastAdmins.remove(user);
|
|
||||||
if (channel->adminsCount() > 1) {
|
|
||||||
channel->setAdminsCount(channel->adminsCount() - 1);
|
|
||||||
if (App::main()) emit App::main()->peerUpdated(channel);
|
|
||||||
}
|
|
||||||
Notify::peerUpdatedDelayed(channel, Notify::PeerUpdate::Flag::AdminsChanged);
|
|
||||||
})));
|
|
||||||
}
|
|
||||||
})));
|
})));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -163,7 +157,7 @@ void GroupMembersWidget::refreshUserOnline(UserData *user) {
|
||||||
|
|
||||||
void GroupMembersWidget::preloadMore() {
|
void GroupMembersWidget::preloadMore() {
|
||||||
if (auto megagroup = peer()->asMegagroup()) {
|
if (auto megagroup = peer()->asMegagroup()) {
|
||||||
auto megagroupInfo = megagroup->mgInfo;
|
auto &megagroupInfo = megagroup->mgInfo;
|
||||||
if (!megagroupInfo->lastParticipants.isEmpty() && megagroupInfo->lastParticipants.size() < megagroup->membersCount()) {
|
if (!megagroupInfo->lastParticipants.isEmpty() && megagroupInfo->lastParticipants.size() < megagroup->membersCount()) {
|
||||||
App::api()->requestLastParticipants(megagroup, false);
|
App::api()->requestLastParticipants(megagroup, false);
|
||||||
}
|
}
|
||||||
|
@ -222,16 +216,22 @@ Ui::PopupMenu *GroupMembersWidget::fillPeerMenu(PeerData *selectedPeer) {
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
};
|
};
|
||||||
if (channel && channel->amCreator() && !item->hasAdminStar) {
|
if (channel) {
|
||||||
result->addAction(lang(lng_context_promote_admin), base::lambda_guarded(this, [this, user] {
|
if (channel->canEditAdmin(user)) {
|
||||||
addAdmin(user);
|
auto label = lang(item->hasAdminStar ? lng_context_edit_permissions : lng_context_promote_admin);
|
||||||
}));
|
result->addAction(label, base::lambda_guarded(this, [this, user] {
|
||||||
} else if (canRemoveAdmin()) {
|
editAdmin(user);
|
||||||
result->addAction(lang(lng_context_remove_admin), base::lambda_guarded(this, [this, user] {
|
}));
|
||||||
removeAdmin(user);
|
}
|
||||||
}));
|
if (channel->canRestrictUser(user)) {
|
||||||
}
|
result->addAction(lang(lng_context_restrict_user), base::lambda_guarded(this, [this, user] {
|
||||||
if (item->hasRemoveLink) {
|
restrictUser(user);
|
||||||
|
}));
|
||||||
|
result->addAction(lang(lng_context_remove_from_group), base::lambda_guarded(this, [this, selectedPeer] {
|
||||||
|
removePeer(selectedPeer);
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
} else if (item->hasRemoveLink) {
|
||||||
result->addAction(lang(lng_context_remove_from_group), base::lambda_guarded(this, [this, selectedPeer] {
|
result->addAction(lang(lng_context_remove_from_group), base::lambda_guarded(this, [this, selectedPeer] {
|
||||||
removePeer(selectedPeer);
|
removePeer(selectedPeer);
|
||||||
}));
|
}));
|
||||||
|
@ -280,8 +280,7 @@ void GroupMembersWidget::refreshMembers() {
|
||||||
fillChatMembers(chat);
|
fillChatMembers(chat);
|
||||||
refreshLimitReached();
|
refreshLimitReached();
|
||||||
} else if (auto megagroup = peer()->asMegagroup()) {
|
} else if (auto megagroup = peer()->asMegagroup()) {
|
||||||
checkSelfAdmin(megagroup);
|
auto &megagroupInfo = megagroup->mgInfo;
|
||||||
auto megagroupInfo = megagroup->mgInfo;
|
|
||||||
if (megagroupInfo->lastParticipants.isEmpty() || megagroup->lastParticipantsCountOutdated()) {
|
if (megagroupInfo->lastParticipants.isEmpty() || megagroup->lastParticipantsCountOutdated()) {
|
||||||
App::api()->requestLastParticipants(megagroup);
|
App::api()->requestLastParticipants(megagroup);
|
||||||
}
|
}
|
||||||
|
@ -324,18 +323,6 @@ void GroupMembersWidget::checkSelfAdmin(ChatData *chat) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void GroupMembersWidget::checkSelfAdmin(ChannelData *megagroup) {
|
|
||||||
if (megagroup->mgInfo->lastParticipants.isEmpty()) return;
|
|
||||||
|
|
||||||
auto amAdmin = megagroup->hasAdminRights();
|
|
||||||
auto self = App::self();
|
|
||||||
if (amAdmin && !megagroup->mgInfo->lastAdmins.contains(self)) {
|
|
||||||
megagroup->selfAdminUpdated();
|
|
||||||
} else if (!amAdmin && megagroup->mgInfo->lastAdmins.contains(self)) {
|
|
||||||
megagroup->selfAdminUpdated();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void GroupMembersWidget::sortMembers() {
|
void GroupMembersWidget::sortMembers() {
|
||||||
if (!_sortByOnline || !itemsCount()) return;
|
if (!_sortByOnline || !itemsCount()) return;
|
||||||
|
|
||||||
|
@ -470,11 +457,14 @@ bool GroupMembersWidget::addUsersToEnd(ChannelData *megagroup) {
|
||||||
|
|
||||||
void GroupMembersWidget::setItemFlags(Item *item, ChannelData *megagroup) {
|
void GroupMembersWidget::setItemFlags(Item *item, ChannelData *megagroup) {
|
||||||
auto amCreatorOrAdmin = item->peer->isSelf() && (megagroup->hasAdminRights() || megagroup->amCreator());
|
auto amCreatorOrAdmin = item->peer->isSelf() && (megagroup->hasAdminRights() || megagroup->amCreator());
|
||||||
auto isAdmin = megagroup->mgInfo->lastAdmins.contains(getMember(item)->user());
|
auto adminIt = megagroup->mgInfo->lastAdmins.constFind(getMember(item)->user());
|
||||||
item->hasAdminStar = amCreatorOrAdmin || isAdmin;
|
auto isAdmin = (adminIt != megagroup->mgInfo->lastAdmins.cend());
|
||||||
|
auto isCreator = megagroup->mgInfo->creator == item->peer;
|
||||||
|
auto adminCanEdit = isAdmin && adminIt->canEdit;
|
||||||
|
item->hasAdminStar = amCreatorOrAdmin || isAdmin || isCreator;
|
||||||
if (item->peer->isSelf()) {
|
if (item->peer->isSelf()) {
|
||||||
item->hasRemoveLink = false;
|
item->hasRemoveLink = false;
|
||||||
} else if (megagroup->amCreator() || (megagroup->canBanMembers() && !item->hasAdminStar)) {
|
} else if (megagroup->amCreator() || (megagroup->canBanMembers() && (!item->hasAdminStar || adminCanEdit))) {
|
||||||
item->hasRemoveLink = true;
|
item->hasRemoveLink = true;
|
||||||
} else {
|
} else {
|
||||||
item->hasRemoveLink = false;
|
item->hasRemoveLink = false;
|
||||||
|
|
|
@ -67,8 +67,8 @@ private:
|
||||||
// Observed notifications.
|
// Observed notifications.
|
||||||
void notifyPeerUpdated(const Notify::PeerUpdate &update);
|
void notifyPeerUpdated(const Notify::PeerUpdate &update);
|
||||||
|
|
||||||
void addAdmin(gsl::not_null<UserData*> user);
|
void editAdmin(gsl::not_null<UserData*> user);
|
||||||
void removeAdmin(gsl::not_null<UserData*> user);
|
void restrictUser(gsl::not_null<UserData*> user);
|
||||||
void removePeer(PeerData *selectedPeer);
|
void removePeer(PeerData *selectedPeer);
|
||||||
void refreshMembers();
|
void refreshMembers();
|
||||||
void fillChatMembers(ChatData *chat);
|
void fillChatMembers(ChatData *chat);
|
||||||
|
@ -76,7 +76,6 @@ private:
|
||||||
void sortMembers();
|
void sortMembers();
|
||||||
void updateOnlineCount();
|
void updateOnlineCount();
|
||||||
void checkSelfAdmin(ChatData *chat);
|
void checkSelfAdmin(ChatData *chat);
|
||||||
void checkSelfAdmin(ChannelData *megagroup);
|
|
||||||
void refreshLimitReached();
|
void refreshLimitReached();
|
||||||
|
|
||||||
void preloadMore();
|
void preloadMore();
|
||||||
|
|
|
@ -172,10 +172,7 @@ SettingsWidget::SettingsWidget(QWidget *parent, PeerData *peer) : BlockWidget(pa
|
||||||
observeEvents |= UpdateFlag::ChatCanEdit | UpdateFlag::InviteLinkChanged;
|
observeEvents |= UpdateFlag::ChatCanEdit | UpdateFlag::InviteLinkChanged;
|
||||||
}
|
}
|
||||||
} else if (auto channel = peer->asChannel()) {
|
} else if (auto channel = peer->asChannel()) {
|
||||||
if (channel->amCreator()) {
|
observeEvents |= UpdateFlag::ChannelRightsChanged | UpdateFlag::BlockedUsersChanged | UpdateFlag::UsernameChanged | UpdateFlag::InviteLinkChanged;
|
||||||
observeEvents |= UpdateFlag::UsernameChanged | UpdateFlag::InviteLinkChanged;
|
|
||||||
}
|
|
||||||
observeEvents |= UpdateFlag::ChannelAmEditor | UpdateFlag::BlockedUsersChanged;
|
|
||||||
}
|
}
|
||||||
subscribe(Notify::PeerUpdated(), Notify::PeerUpdatedHandler(observeEvents, [this](const Notify::PeerUpdate &update) {
|
subscribe(Notify::PeerUpdated(), Notify::PeerUpdatedHandler(observeEvents, [this](const Notify::PeerUpdate &update) {
|
||||||
notifyPeerUpdated(update);
|
notifyPeerUpdated(update);
|
||||||
|
@ -195,13 +192,13 @@ void SettingsWidget::notifyPeerUpdated(const Notify::PeerUpdate &update) {
|
||||||
if (update.flags & UpdateFlag::NotificationsEnabled) {
|
if (update.flags & UpdateFlag::NotificationsEnabled) {
|
||||||
refreshEnableNotifications();
|
refreshEnableNotifications();
|
||||||
}
|
}
|
||||||
if (update.flags & (UpdateFlag::ChatCanEdit | UpdateFlag::UsernameChanged | UpdateFlag::InviteLinkChanged)) {
|
if (update.flags & (UpdateFlag::ChannelRightsChanged | UpdateFlag::ChatCanEdit | UpdateFlag::UsernameChanged | UpdateFlag::InviteLinkChanged)) {
|
||||||
refreshInviteLinkButton();
|
refreshInviteLinkButton();
|
||||||
}
|
}
|
||||||
if (update.flags & (UpdateFlag::ChatCanEdit)) {
|
if (update.flags & (UpdateFlag::ChannelRightsChanged | UpdateFlag::ChatCanEdit)) {
|
||||||
refreshManageAdminsButton();
|
refreshManageAdminsButton();
|
||||||
}
|
}
|
||||||
if ((update.flags & UpdateFlag::ChannelAmEditor) || (update.flags & UpdateFlag::BlockedUsersChanged)) {
|
if (update.flags & (UpdateFlag::ChannelRightsChanged | UpdateFlag::BlockedUsersChanged)) {
|
||||||
refreshManageBlockedUsersButton();
|
refreshManageBlockedUsersButton();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -254,7 +251,7 @@ void SettingsWidget::refreshManageAdminsButton() {
|
||||||
if (auto chat = peer()->asChat()) {
|
if (auto chat = peer()->asChat()) {
|
||||||
return (chat->amCreator() && chat->canEdit());
|
return (chat->amCreator() && chat->canEdit());
|
||||||
} else if (auto channel = peer()->asMegagroup()) {
|
} else if (auto channel = peer()->asMegagroup()) {
|
||||||
return channel->amCreator();
|
return channel->hasAdminRights() || channel->amCreator();
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
};
|
};
|
||||||
|
@ -269,7 +266,7 @@ void SettingsWidget::refreshManageAdminsButton() {
|
||||||
void SettingsWidget::refreshManageBlockedUsersButton() {
|
void SettingsWidget::refreshManageBlockedUsersButton() {
|
||||||
auto hasManageBlockedUsers = [this] {
|
auto hasManageBlockedUsers = [this] {
|
||||||
if (auto channel = peer()->asMegagroup()) {
|
if (auto channel = peer()->asMegagroup()) {
|
||||||
return channel->canBanMembers() && (channel->kickedCount() > 0);
|
return channel->canBanMembers() && (channel->kickedCount() > 0 || channel->restrictedCount() > 0);
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
};
|
};
|
||||||
|
@ -288,7 +285,7 @@ void SettingsWidget::refreshInviteLinkButton() {
|
||||||
return lang(chat->inviteLink().isEmpty() ? lng_group_invite_create : lng_group_invite_create_new);
|
return lang(chat->inviteLink().isEmpty() ? lng_group_invite_create : lng_group_invite_create_new);
|
||||||
}
|
}
|
||||||
} else if (auto channel = peer()->asChannel()) {
|
} else if (auto channel = peer()->asChannel()) {
|
||||||
if (channel->amCreator() && !channel->isPublic()) {
|
if (channel->canHaveInviteLink() && !channel->isPublic()) {
|
||||||
return lang(channel->inviteLink().isEmpty() ? lng_group_invite_create : lng_group_invite_create_new);
|
return lang(channel->inviteLink().isEmpty() ? lng_group_invite_create : lng_group_invite_create_new);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -45,8 +45,7 @@ using UpdateFlag = Notify::PeerUpdate::Flag;
|
||||||
const auto ButtonsUpdateFlags = UpdateFlag::UserCanShareContact
|
const auto ButtonsUpdateFlags = UpdateFlag::UserCanShareContact
|
||||||
| UpdateFlag::BotCanAddToGroups
|
| UpdateFlag::BotCanAddToGroups
|
||||||
| UpdateFlag::ChatCanEdit
|
| UpdateFlag::ChatCanEdit
|
||||||
| UpdateFlag::ChannelCanEditInformation
|
| UpdateFlag::ChannelRightsChanged
|
||||||
| UpdateFlag::ChannelCanAddMembers
|
|
||||||
| UpdateFlag::ChannelAmIn;
|
| UpdateFlag::ChannelAmIn;
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
@ -437,7 +436,7 @@ void CoverWidget::setChatButtons() {
|
||||||
|
|
||||||
void CoverWidget::setMegagroupButtons() {
|
void CoverWidget::setMegagroupButtons() {
|
||||||
if (_peerMegagroup->amIn()) {
|
if (_peerMegagroup->amIn()) {
|
||||||
if (_peerMegagroup->canEditInformation()) {
|
if (canEditPhoto()) {
|
||||||
addButton(langFactory(lng_profile_set_group_photo), SLOT(onSetPhoto()));
|
addButton(langFactory(lng_profile_set_group_photo), SLOT(onSetPhoto()));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -449,7 +448,7 @@ void CoverWidget::setMegagroupButtons() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void CoverWidget::setChannelButtons() {
|
void CoverWidget::setChannelButtons() {
|
||||||
if (_peerChannel->amCreator()) {
|
if (canEditPhoto()) {
|
||||||
addButton(langFactory(lng_profile_set_group_photo), SLOT(onSetPhoto()));
|
addButton(langFactory(lng_profile_set_group_photo), SLOT(onSetPhoto()));
|
||||||
} else if (_peerChannel->amIn()) {
|
} else if (_peerChannel->amIn()) {
|
||||||
addButton(langFactory(lng_profile_view_channel), SLOT(onViewChannel()));
|
addButton(langFactory(lng_profile_view_channel), SLOT(onViewChannel()));
|
||||||
|
|
|
@ -38,7 +38,7 @@ using UpdateFlag = Notify::PeerUpdate::Flag;
|
||||||
const auto ButtonsUpdateFlags = UpdateFlag::UserCanShareContact
|
const auto ButtonsUpdateFlags = UpdateFlag::UserCanShareContact
|
||||||
| UpdateFlag::UserIsContact
|
| UpdateFlag::UserIsContact
|
||||||
| UpdateFlag::ChatCanEdit
|
| UpdateFlag::ChatCanEdit
|
||||||
| UpdateFlag::ChannelAmEditor;
|
| UpdateFlag::ChannelRightsChanged;
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
|
@ -117,13 +117,13 @@ void FixedBar::setChatActions() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void FixedBar::setMegagroupActions() {
|
void FixedBar::setMegagroupActions() {
|
||||||
if (_peerMegagroup->amCreator() || _peerMegagroup->canEditInformation()) {
|
if (_peerMegagroup->canEditInformation()) {
|
||||||
addRightAction(RightActionType::EditChannel, langFactory(lng_profile_edit_contact), SLOT(onEditChannel()));
|
addRightAction(RightActionType::EditChannel, langFactory(lng_profile_edit_contact), SLOT(onEditChannel()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void FixedBar::setChannelActions() {
|
void FixedBar::setChannelActions() {
|
||||||
if (_peerChannel->amCreator()) {
|
if (_peerChannel->canEditInformation()) {
|
||||||
addRightAction(RightActionType::EditChannel, langFactory(lng_profile_edit_contact), SLOT(onEditChannel()));
|
addRightAction(RightActionType::EditChannel, langFactory(lng_profile_edit_contact), SLOT(onEditChannel()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -647,10 +647,10 @@ void ChatData::setName(const QString &newName) {
|
||||||
|
|
||||||
void ChatData::invalidateParticipants() {
|
void ChatData::invalidateParticipants() {
|
||||||
auto wasCanEdit = canEdit();
|
auto wasCanEdit = canEdit();
|
||||||
participants = ChatData::Participants();
|
participants.clear();
|
||||||
admins = ChatData::Admins();
|
admins.clear();
|
||||||
flags &= ~MTPDchat::Flag::f_admin;
|
flags &= ~MTPDchat::Flag::f_admin;
|
||||||
invitedByMe = ChatData::InvitedByMe();
|
invitedByMe.clear();
|
||||||
botStatus = 0;
|
botStatus = 0;
|
||||||
if (wasCanEdit != canEdit()) {
|
if (wasCanEdit != canEdit()) {
|
||||||
Notify::peerUpdatedDelayed(this, Notify::PeerUpdate::Flag::ChatCanEdit);
|
Notify::peerUpdatedDelayed(this, Notify::PeerUpdate::Flag::ChatCanEdit);
|
||||||
|
@ -748,6 +748,13 @@ void ChannelData::setAdminsCount(int newAdminsCount) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ChannelData::setRestrictedCount(int newRestrictedCount) {
|
||||||
|
if (_restrictedCount != newRestrictedCount) {
|
||||||
|
_restrictedCount = newRestrictedCount;
|
||||||
|
Notify::peerUpdatedDelayed(this, Notify::PeerUpdate::Flag::BlockedUsersChanged);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void ChannelData::setKickedCount(int newKickedCount) {
|
void ChannelData::setKickedCount(int newKickedCount) {
|
||||||
if (_kickedCount != newKickedCount) {
|
if (_kickedCount != newKickedCount) {
|
||||||
_kickedCount = newKickedCount;
|
_kickedCount = newKickedCount;
|
||||||
|
@ -755,29 +762,187 @@ void ChannelData::setKickedCount(int newKickedCount) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
MTPChannelBannedRights ChannelData::KickedRestrictedRights() {
|
||||||
|
using Flag = MTPDchannelBannedRights::Flag;
|
||||||
|
auto flags = Flag::f_view_messages | Flag::f_send_messages | Flag::f_send_media | Flag::f_embed_links | Flag::f_send_stickers | Flag::f_send_gifs | Flag::f_send_games | Flag::f_send_inline;
|
||||||
|
return MTP_channelBannedRights(MTP_flags(flags), MTP_int(std::numeric_limits<int32>::max()));
|
||||||
|
}
|
||||||
|
|
||||||
|
void ChannelData::applyEditAdmin(gsl::not_null<UserData*> user, const MTPChannelAdminRights &rights) {
|
||||||
|
auto flags = Notify::PeerUpdate::Flag::AdminsChanged | Notify::PeerUpdate::Flag::None;
|
||||||
|
if (mgInfo) {
|
||||||
|
if (!mgInfo->lastParticipants.contains(user)) { // If rights are empty - still add participant? TODO check
|
||||||
|
mgInfo->lastParticipants.push_front(user);
|
||||||
|
setMembersCount(membersCount() + 1);
|
||||||
|
if (user->botInfo && !mgInfo->bots.contains(user)) {
|
||||||
|
mgInfo->bots.insert(user);
|
||||||
|
if (mgInfo->botStatus != 0 && mgInfo->botStatus < 2) {
|
||||||
|
mgInfo->botStatus = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (mgInfo->lastRestricted.contains(user)) { // If rights are empty - still remove restrictions? TODO check
|
||||||
|
mgInfo->lastRestricted.remove(user);
|
||||||
|
if (restrictedCount() > 0) {
|
||||||
|
setRestrictedCount(restrictedCount() - 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
auto it = mgInfo->lastAdmins.find(user);
|
||||||
|
if (rights.c_channelAdminRights().vflags.v != 0) {
|
||||||
|
auto lastAdmin = MegagroupInfo::Admin { rights };
|
||||||
|
lastAdmin.canEdit = true;
|
||||||
|
if (it == mgInfo->lastAdmins.cend()) {
|
||||||
|
mgInfo->lastAdmins.insert(user, lastAdmin);
|
||||||
|
setAdminsCount(adminsCount() + 1);
|
||||||
|
} else {
|
||||||
|
it.value() = lastAdmin;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (it != mgInfo->lastAdmins.cend()) {
|
||||||
|
mgInfo->lastAdmins.erase(it);
|
||||||
|
if (adminsCount() > 0) {
|
||||||
|
setAdminsCount(adminsCount() - 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Notify::peerUpdatedDelayed(this, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ChannelData::applyEditBanned(gsl::not_null<UserData*> user, const MTPChannelBannedRights &rights) {
|
||||||
|
auto flags = Notify::PeerUpdate::Flag::BlockedUsersChanged | Notify::PeerUpdate::Flag::None;
|
||||||
|
if (mgInfo) {
|
||||||
|
if (mgInfo->lastAdmins.contains(user)) { // If rights are empty - still remove admin? TODO check
|
||||||
|
mgInfo->lastAdmins.remove(user);
|
||||||
|
if (adminsCount() > 1) {
|
||||||
|
setAdminsCount(adminsCount() - 1);
|
||||||
|
} else {
|
||||||
|
flags |= Notify::PeerUpdate::Flag::AdminsChanged;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
auto isKicked = (rights.c_channelBannedRights().vflags.v & MTPDchannelBannedRights::Flag::f_view_messages);
|
||||||
|
auto isRestricted = !isKicked && (rights.c_channelBannedRights().vflags.v != 0);
|
||||||
|
auto it = mgInfo->lastRestricted.find(user);
|
||||||
|
if (isRestricted) {
|
||||||
|
if (it == mgInfo->lastRestricted.cend()) {
|
||||||
|
mgInfo->lastRestricted.insert(user, MegagroupInfo::Restricted { rights });
|
||||||
|
setRestrictedCount(restrictedCount() + 1);
|
||||||
|
} else {
|
||||||
|
it->rights = rights;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (it != mgInfo->lastRestricted.cend()) {
|
||||||
|
mgInfo->lastRestricted.erase(it);
|
||||||
|
if (restrictedCount() > 0) {
|
||||||
|
setRestrictedCount(restrictedCount() - 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (isKicked) {
|
||||||
|
auto i = mgInfo->lastParticipants.indexOf(user);
|
||||||
|
if (i >= 0) {
|
||||||
|
mgInfo->lastParticipants.removeAt(i);
|
||||||
|
}
|
||||||
|
if (membersCount() > 1) {
|
||||||
|
setMembersCount(membersCount() - 1);
|
||||||
|
} else {
|
||||||
|
mgInfo->lastParticipantsStatus |= MegagroupInfo::LastParticipantsCountOutdated;
|
||||||
|
mgInfo->lastParticipantsCount = 0;
|
||||||
|
}
|
||||||
|
setKickedCount(kickedCount() + 1);
|
||||||
|
if (mgInfo->bots.contains(user)) {
|
||||||
|
mgInfo->bots.remove(user);
|
||||||
|
if (mgInfo->bots.isEmpty() && mgInfo->botStatus > 0) {
|
||||||
|
mgInfo->botStatus = -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
flags |= Notify::PeerUpdate::Flag::MembersChanged;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Notify::peerUpdatedDelayed(this, flags);
|
||||||
|
}
|
||||||
|
|
||||||
void ChannelData::flagsUpdated() {
|
void ChannelData::flagsUpdated() {
|
||||||
if (isMegagroup()) {
|
if (isMegagroup()) {
|
||||||
if (!mgInfo) {
|
if (!mgInfo) {
|
||||||
mgInfo = new MegagroupInfo();
|
mgInfo = std::make_unique<MegagroupInfo>();
|
||||||
}
|
}
|
||||||
} else if (mgInfo) {
|
} else if (mgInfo) {
|
||||||
delete mgInfo;
|
|
||||||
mgInfo = nullptr;
|
mgInfo = nullptr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ChannelData::selfAdminUpdated() {
|
bool ChannelData::canNotEditLastAdmin(gsl::not_null<UserData*> user) const {
|
||||||
|
if (mgInfo) {
|
||||||
|
auto i = mgInfo->lastAdmins.constFind(user);
|
||||||
|
if (i != mgInfo->lastAdmins.cend()) {
|
||||||
|
return !i->canEdit;
|
||||||
|
}
|
||||||
|
return (user == mgInfo->creator);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ChannelData::canEditAdmin(gsl::not_null<UserData*> user) const {
|
||||||
|
if (user->isSelf()) {
|
||||||
|
return false;
|
||||||
|
} else if (amCreator()) {
|
||||||
|
return true;
|
||||||
|
} else if (canNotEditLastAdmin(user)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return adminRights().is_add_admins();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ChannelData::canRestrictUser(gsl::not_null<UserData*> user) const {
|
||||||
|
if (user->isSelf()) {
|
||||||
|
return false;
|
||||||
|
} else if (amCreator()) {
|
||||||
|
return true;
|
||||||
|
} else if (canNotEditLastAdmin(user)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return adminRights().is_ban_users();
|
||||||
|
}
|
||||||
|
|
||||||
|
void ChannelData::setAdminRights(const MTPChannelAdminRights &rights) {
|
||||||
|
if (rights.c_channelAdminRights().vflags.v == _adminRights.c_channelAdminRights().vflags.v) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_adminRights = rights;
|
||||||
if (isMegagroup()) {
|
if (isMegagroup()) {
|
||||||
if (hasAdminRights()) {
|
if (hasAdminRights()) {
|
||||||
mgInfo->lastAdmins.insert(App::self(), _adminRights);
|
if (!amCreator()) {
|
||||||
|
auto me = MegagroupInfo::Admin { _adminRights };
|
||||||
|
me.canEdit = false;
|
||||||
|
mgInfo->lastAdmins.insert(App::self(), me);
|
||||||
|
}
|
||||||
|
mgInfo->lastRestricted.remove(App::self());
|
||||||
} else {
|
} else {
|
||||||
mgInfo->lastAdmins.remove(App::self());
|
mgInfo->lastAdmins.remove(App::self());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Notify::peerUpdatedDelayed(this, UpdateFlag::ChannelRightsChanged | UpdateFlag::AdminsChanged | UpdateFlag::BlockedUsersChanged);
|
||||||
}
|
}
|
||||||
|
|
||||||
ChannelData::~ChannelData() {
|
void ChannelData::setRestrictedRights(const MTPChannelBannedRights &rights) {
|
||||||
delete mgInfo;
|
if (rights.c_channelBannedRights().vflags.v == _restrictedRights.c_channelBannedRights().vflags.v
|
||||||
|
&& rights.c_channelBannedRights().vuntil_date.v == _restrictedRights.c_channelBannedRights().vuntil_date.v) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_restrictedRights = rights;
|
||||||
|
if (isMegagroup()) {
|
||||||
|
if (hasRestrictedRights()) {
|
||||||
|
if (!amCreator()) {
|
||||||
|
auto me = MegagroupInfo::Restricted { _restrictedRights };
|
||||||
|
mgInfo->lastRestricted.insert(App::self(), me);
|
||||||
|
}
|
||||||
|
mgInfo->lastAdmins.remove(App::self());
|
||||||
|
} else {
|
||||||
|
mgInfo->lastRestricted.remove(App::self());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Notify::peerUpdatedDelayed(this, UpdateFlag::ChannelRightsChanged | UpdateFlag::AdminsChanged | UpdateFlag::BlockedUsersChanged);
|
||||||
}
|
}
|
||||||
|
|
||||||
uint64 PtsWaiter::ptsKey(PtsSkippedQueue queue) {
|
uint64 PtsWaiter::ptsKey(PtsSkippedQueue queue) {
|
||||||
|
|
|
@ -619,16 +619,11 @@ public:
|
||||||
bool isMigrated() const {
|
bool isMigrated() const {
|
||||||
return flags & MTPDchat::Flag::f_migrated_to;
|
return flags & MTPDchat::Flag::f_migrated_to;
|
||||||
}
|
}
|
||||||
typedef QMap<UserData*, int> Participants;
|
QMap<gsl::not_null<UserData*>, int> participants;
|
||||||
Participants participants;
|
OrderedSet<gsl::not_null<UserData*>> invitedByMe;
|
||||||
typedef OrderedSet<UserData*> InvitedByMe;
|
OrderedSet<gsl::not_null<UserData*>> admins;
|
||||||
InvitedByMe invitedByMe;
|
QList<gsl::not_null<UserData*>> lastAuthors;
|
||||||
typedef OrderedSet<UserData*> Admins;
|
OrderedSet<gsl::not_null<PeerData*>> markupSenders;
|
||||||
Admins admins;
|
|
||||||
typedef QList<UserData*> LastAuthors;
|
|
||||||
LastAuthors lastAuthors;
|
|
||||||
typedef OrderedSet<PeerData*> MarkupSenders;
|
|
||||||
MarkupSenders markupSenders;
|
|
||||||
int botStatus = 0; // -1 - no bots, 0 - unknown, 1 - one bot, that sees all history, 2 - other
|
int botStatus = 0; // -1 - no bots, 0 - unknown, 1 - one bot, that sees all history, 2 - other
|
||||||
// ImagePtr photoFull;
|
// ImagePtr photoFull;
|
||||||
|
|
||||||
|
@ -705,14 +700,26 @@ private:
|
||||||
};
|
};
|
||||||
|
|
||||||
struct MegagroupInfo {
|
struct MegagroupInfo {
|
||||||
typedef QList<UserData*> LastParticipants;
|
struct Admin {
|
||||||
LastParticipants lastParticipants;
|
explicit Admin(MTPChannelAdminRights rights) : rights(rights) {
|
||||||
QMap<UserData*, MTPChannelAdminRights> lastAdmins;
|
}
|
||||||
typedef OrderedSet<PeerData*> MarkupSenders;
|
Admin(MTPChannelAdminRights rights, bool canEdit) : rights(rights), canEdit(canEdit) {
|
||||||
MarkupSenders markupSenders;
|
}
|
||||||
typedef OrderedSet<UserData*> Bots;
|
MTPChannelAdminRights rights;
|
||||||
Bots bots;
|
bool canEdit = false;
|
||||||
|
};
|
||||||
|
struct Restricted {
|
||||||
|
explicit Restricted(MTPChannelBannedRights rights) : rights(rights) {
|
||||||
|
}
|
||||||
|
MTPChannelBannedRights rights;
|
||||||
|
};
|
||||||
|
QList<gsl::not_null<UserData*>> lastParticipants;
|
||||||
|
QMap<gsl::not_null<UserData*>, Admin> lastAdmins;
|
||||||
|
QMap<gsl::not_null<UserData*>, Restricted> lastRestricted;
|
||||||
|
OrderedSet<gsl::not_null<PeerData*>> markupSenders;
|
||||||
|
OrderedSet<gsl::not_null<UserData*>> bots;
|
||||||
|
|
||||||
|
UserData *creator = nullptr; // nullptr means unknown
|
||||||
int botStatus = 0; // -1 - no bots, 0 - unknown, 1 - one bot, that sees all history, 2 - other
|
int botStatus = 0; // -1 - no bots, 0 - unknown, 1 - one bot, that sees all history, 2 - other
|
||||||
MsgId pinnedMsgId = 0;
|
MsgId pinnedMsgId = 0;
|
||||||
bool joinedMessageFound = false;
|
bool joinedMessageFound = false;
|
||||||
|
@ -764,6 +771,11 @@ public:
|
||||||
}
|
}
|
||||||
void setAdminsCount(int newAdminsCount);
|
void setAdminsCount(int newAdminsCount);
|
||||||
|
|
||||||
|
int restrictedCount() const {
|
||||||
|
return _restrictedCount;
|
||||||
|
}
|
||||||
|
void setRestrictedCount(int newRestrictedCount);
|
||||||
|
|
||||||
int kickedCount() const {
|
int kickedCount() const {
|
||||||
return _kickedCount;
|
return _kickedCount;
|
||||||
}
|
}
|
||||||
|
@ -788,11 +800,15 @@ public:
|
||||||
return flags & MTPDchannel::Flag::f_verified;
|
return flags & MTPDchannel::Flag::f_verified;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static MTPChannelBannedRights KickedRestrictedRights();
|
||||||
|
void applyEditAdmin(gsl::not_null<UserData*> user, const MTPChannelAdminRights &rights);
|
||||||
|
void applyEditBanned(gsl::not_null<UserData*> user, const MTPChannelBannedRights &rights);
|
||||||
|
|
||||||
int32 date = 0;
|
int32 date = 0;
|
||||||
int version = 0;
|
int version = 0;
|
||||||
MTPDchannel::Flags flags = { 0 };
|
MTPDchannel::Flags flags = { 0 };
|
||||||
MTPDchannelFull::Flags flagsFull = { 0 };
|
MTPDchannelFull::Flags flagsFull = { 0 };
|
||||||
MegagroupInfo *mgInfo = nullptr;
|
std::unique_ptr<MegagroupInfo> mgInfo;
|
||||||
bool lastParticipantsCountOutdated() const {
|
bool lastParticipantsCountOutdated() const {
|
||||||
if (!mgInfo || !(mgInfo->lastParticipantsStatus & MegagroupInfo::LastParticipantsCountOutdated)) {
|
if (!mgInfo || !(mgInfo->lastParticipantsStatus & MegagroupInfo::LastParticipantsCountOutdated)) {
|
||||||
return false;
|
return false;
|
||||||
|
@ -804,7 +820,6 @@ public:
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
void flagsUpdated();
|
void flagsUpdated();
|
||||||
void selfAdminUpdated();
|
|
||||||
bool isMegagroup() const {
|
bool isMegagroup() const {
|
||||||
return flags & MTPDchannel::Flag::f_megagroup;
|
return flags & MTPDchannel::Flag::f_megagroup;
|
||||||
}
|
}
|
||||||
|
@ -817,39 +832,44 @@ public:
|
||||||
bool amCreator() const {
|
bool amCreator() const {
|
||||||
return flags & MTPDchannel::Flag::f_creator;
|
return flags & MTPDchannel::Flag::f_creator;
|
||||||
}
|
}
|
||||||
bool amEditor() const {
|
const MTPChannelAdminRights &adminRightsBoxed() const {
|
||||||
return hasAdminRights();
|
return _adminRights;
|
||||||
}
|
}
|
||||||
const MTPDchannelAdminRights &adminRights() const {
|
const MTPDchannelAdminRights &adminRights() const {
|
||||||
return _adminRights.c_channelAdminRights();
|
return _adminRights.c_channelAdminRights();
|
||||||
}
|
}
|
||||||
void setAdminRights(const MTPChannelAdminRights &rights) {
|
void setAdminRights(const MTPChannelAdminRights &rights);
|
||||||
_adminRights = rights;
|
|
||||||
}
|
|
||||||
bool hasAdminRights() const {
|
bool hasAdminRights() const {
|
||||||
return (adminRights().vflags.v != 0);
|
return (adminRights().vflags.v != 0);
|
||||||
}
|
}
|
||||||
const MTPDchannelBannedRights &bannedRights() const {
|
const MTPChannelBannedRights &restrictedRightsBoxed() const {
|
||||||
return _bannedRights.c_channelBannedRights();
|
return _restrictedRights;
|
||||||
}
|
}
|
||||||
void setBannedRights(const MTPChannelBannedRights &rights) {
|
const MTPDchannelBannedRights &restrictedRights() const {
|
||||||
_bannedRights = rights;
|
return _restrictedRights.c_channelBannedRights();
|
||||||
}
|
}
|
||||||
bool hasBannedRights() const {
|
void setRestrictedRights(const MTPChannelBannedRights &rights);
|
||||||
return (bannedRights().vflags.v != 0);
|
bool hasRestrictedRights() const {
|
||||||
|
return (restrictedRights().vflags.v != 0);
|
||||||
}
|
}
|
||||||
bool hasBannedRights(int32 now) const {
|
bool hasRestrictedRights(int32 now) const {
|
||||||
return hasBannedRights() && (bannedRights().vuntil_date.v > now);
|
return hasRestrictedRights() && (restrictedRights().vuntil_date.v > now);
|
||||||
}
|
}
|
||||||
bool canBanMembers() const {
|
bool canBanMembers() const {
|
||||||
return adminRights().is_ban_users() || amCreator();
|
return adminRights().is_ban_users() || amCreator();
|
||||||
}
|
}
|
||||||
|
bool canEditMessages() const {
|
||||||
|
return adminRights().is_edit_messages() || amCreator();
|
||||||
|
}
|
||||||
bool canDeleteMessages() const {
|
bool canDeleteMessages() const {
|
||||||
return adminRights().is_delete_messages() || amCreator();
|
return adminRights().is_delete_messages() || amCreator();
|
||||||
}
|
}
|
||||||
bool canAddMembers() const {
|
bool canAddMembers() const {
|
||||||
return adminRights().is_invite_users() || amCreator() || (amIn() && (flags & MTPDchannel::Flag::f_democracy));
|
return adminRights().is_invite_users() || amCreator() || (amIn() && (flags & MTPDchannel::Flag::f_democracy));
|
||||||
}
|
}
|
||||||
|
bool canAddAdmins() const {
|
||||||
|
return adminRights().is_add_admins() || amCreator();
|
||||||
|
}
|
||||||
bool canPinMessages() const {
|
bool canPinMessages() const {
|
||||||
return adminRights().is_pin_messages() || amCreator();
|
return adminRights().is_pin_messages() || amCreator();
|
||||||
}
|
}
|
||||||
|
@ -857,7 +877,7 @@ public:
|
||||||
return adminRights().is_post_messages() || amCreator();
|
return adminRights().is_post_messages() || amCreator();
|
||||||
}
|
}
|
||||||
bool canWrite() const {
|
bool canWrite() const {
|
||||||
return amIn() && (canPublish() || !isBroadcast());
|
return amIn() && (canPublish() || (!isBroadcast() && !restrictedRights().is_send_messages()));
|
||||||
}
|
}
|
||||||
bool canViewMembers() const {
|
bool canViewMembers() const {
|
||||||
return flagsFull & MTPDchannelFull::Flag::f_can_view_participants;
|
return flagsFull & MTPDchannelFull::Flag::f_can_view_participants;
|
||||||
|
@ -875,11 +895,16 @@ public:
|
||||||
constexpr auto kDeleteChannelMembersLimit = 1000;
|
constexpr auto kDeleteChannelMembersLimit = 1000;
|
||||||
return amCreator() && (membersCount() <= kDeleteChannelMembersLimit);
|
return amCreator() && (membersCount() <= kDeleteChannelMembersLimit);
|
||||||
}
|
}
|
||||||
|
bool canEditAdmin(gsl::not_null<UserData*> user) const;
|
||||||
|
bool canRestrictUser(gsl::not_null<UserData*> user) const;
|
||||||
|
|
||||||
void setInviteLink(const QString &newInviteLink);
|
void setInviteLink(const QString &newInviteLink);
|
||||||
QString inviteLink() const {
|
QString inviteLink() const {
|
||||||
return _inviteLink;
|
return _inviteLink;
|
||||||
}
|
}
|
||||||
|
bool canHaveInviteLink() const {
|
||||||
|
return adminRights().is_invite_link() || amCreator();
|
||||||
|
}
|
||||||
|
|
||||||
int32 inviter = 0; // > 0 - user who invited me to channel, < 0 - not in channel
|
int32 inviter = 0; // > 0 - user who invited me to channel, < 0 - not in channel
|
||||||
QDateTime inviteDate;
|
QDateTime inviteDate;
|
||||||
|
@ -930,19 +955,20 @@ public:
|
||||||
_restrictionReason = reason;
|
_restrictionReason = reason;
|
||||||
}
|
}
|
||||||
|
|
||||||
~ChannelData();
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
bool canNotEditLastAdmin(gsl::not_null<UserData*> user) const;
|
||||||
|
|
||||||
PtsWaiter _ptsWaiter;
|
PtsWaiter _ptsWaiter;
|
||||||
TimeMs _lastFullUpdate = 0;
|
TimeMs _lastFullUpdate = 0;
|
||||||
|
|
||||||
bool _isForbidden = true;
|
bool _isForbidden = true;
|
||||||
int _membersCount = 1;
|
int _membersCount = 1;
|
||||||
int _adminsCount = 1;
|
int _adminsCount = 1;
|
||||||
|
int _restrictedCount = 0;
|
||||||
int _kickedCount = 0;
|
int _kickedCount = 0;
|
||||||
|
|
||||||
MTPChannelAdminRights _adminRights = MTP_channelAdminRights(MTP_flags(0));
|
MTPChannelAdminRights _adminRights = MTP_channelAdminRights(MTP_flags(0));
|
||||||
MTPChannelBannedRights _bannedRights = MTP_channelBannedRights(MTP_flags(0), MTP_int(0));
|
MTPChannelBannedRights _restrictedRights = MTP_channelBannedRights(MTP_flags(0), MTP_int(0));
|
||||||
|
|
||||||
QString _restrictionReason;
|
QString _restrictionReason;
|
||||||
QString _about;
|
QString _about;
|
||||||
|
|
Loading…
Reference in New Issue