1057 lines
28 KiB
C++
1057 lines
28 KiB
C++
/*
|
|
This file is part of Telegram Desktop,
|
|
the official desktop application for the Telegram messaging service.
|
|
|
|
For license and copyright information please follow this link:
|
|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|
*/
|
|
#include "boxes/peers/edit_peer_invite_link.h"
|
|
|
|
#include "data/data_peer.h"
|
|
#include "data/data_user.h"
|
|
#include "data/data_changes.h"
|
|
#include "data/data_session.h"
|
|
#include "data/data_histories.h"
|
|
#include "main/main_session.h"
|
|
#include "api/api_invite_links.h"
|
|
#include "base/unixtime.h"
|
|
#include "apiwrap.h"
|
|
#include "ui/controls/invite_link_buttons.h"
|
|
#include "ui/controls/invite_link_label.h"
|
|
#include "ui/wrap/vertical_layout.h"
|
|
#include "ui/wrap/slide_wrap.h"
|
|
#include "ui/wrap/padding_wrap.h"
|
|
#include "ui/widgets/popup_menu.h"
|
|
#include "ui/abstract_button.h"
|
|
#include "ui/toast/toast.h"
|
|
#include "ui/text/text_utilities.h"
|
|
#include "ui/boxes/edit_invite_link.h"
|
|
#include "boxes/share_box.h"
|
|
#include "history/view/history_view_group_call_tracker.h" // GenerateUs...
|
|
#include "history/history_message.h" // GetErrorTextForSending.
|
|
#include "history/history.h"
|
|
#include "ui/boxes/confirm_box.h"
|
|
#include "boxes/peer_list_box.h"
|
|
#include "mainwindow.h"
|
|
#include "facades.h" // Ui::showPerProfile.
|
|
#include "lang/lang_keys.h"
|
|
#include "window/window_session_controller.h"
|
|
#include "settings/settings_common.h"
|
|
#include "mtproto/sender.h"
|
|
#include "qr/qr_generate.h"
|
|
#include "intro/intro_qr.h" // TelegramLogoImage
|
|
#include "styles/style_boxes.h"
|
|
#include "styles/style_layers.h" // st::boxDividerLabel.
|
|
#include "styles/style_info.h"
|
|
#include "styles/style_settings.h"
|
|
|
|
#include <QtGui/QGuiApplication>
|
|
#include <QtCore/QMimeData>
|
|
|
|
namespace {
|
|
|
|
constexpr auto kFirstPage = 20;
|
|
constexpr auto kPerPage = 100;
|
|
constexpr auto kShareQrSize = 768;
|
|
constexpr auto kShareQrPadding = 16;
|
|
|
|
using LinkData = Api::InviteLink;
|
|
|
|
class Controller final : public PeerListController {
|
|
public:
|
|
Controller(
|
|
not_null<PeerData*> peer,
|
|
not_null<UserData*> admin,
|
|
rpl::producer<LinkData> data);
|
|
|
|
void prepare() override;
|
|
void loadMoreRows() override;
|
|
void rowClicked(not_null<PeerListRow*> row) override;
|
|
Main::Session &session() const override;
|
|
|
|
rpl::producer<int> boxHeightValue() const override;
|
|
int descriptionTopSkipMin() const override;
|
|
|
|
private:
|
|
void appendSlice(const Api::JoinedByLinkSlice &slice);
|
|
void addHeaderBlock(not_null<Ui::VerticalLayout*> container);
|
|
|
|
[[nodiscard]] rpl::producer<LinkData> dataValue() const;
|
|
|
|
const not_null<PeerData*> _peer;
|
|
rpl::variable<LinkData> _data;
|
|
|
|
QString _link;
|
|
bool _revoked = false;
|
|
|
|
mtpRequestId _requestId = 0;
|
|
std::optional<Api::JoinedByLinkUser> _lastUser;
|
|
bool _allLoaded = false;
|
|
|
|
Ui::RpWidget *_headerWidget = nullptr;
|
|
rpl::variable<int> _addedHeight;
|
|
|
|
MTP::Sender _api;
|
|
rpl::lifetime _lifetime;
|
|
|
|
};
|
|
|
|
class SingleRowController final : public PeerListController {
|
|
public:
|
|
SingleRowController(
|
|
not_null<PeerData*> peer,
|
|
rpl::producer<QString> status);
|
|
|
|
void prepare() override;
|
|
void loadMoreRows() override;
|
|
void rowClicked(not_null<PeerListRow*> row) override;
|
|
Main::Session &session() const override;
|
|
|
|
private:
|
|
const not_null<PeerData*> _peer;
|
|
rpl::producer<QString> _status;
|
|
rpl::lifetime _lifetime;
|
|
|
|
};
|
|
|
|
[[nodiscard]] bool ClosingLinkBox(const LinkData &updated, bool revoked) {
|
|
return updated.link.isEmpty() || (!revoked && updated.revoked);
|
|
}
|
|
|
|
QImage QrExact(const Qr::Data &data, int pixel, QColor color) {
|
|
const auto image = [](int size) {
|
|
auto result = QImage(
|
|
size,
|
|
size,
|
|
QImage::Format_ARGB32_Premultiplied);
|
|
result.fill(Qt::transparent);
|
|
{
|
|
QPainter p(&result);
|
|
const auto skip = size / 12;
|
|
const auto logoSize = size - 2 * skip;
|
|
p.drawImage(
|
|
skip,
|
|
skip,
|
|
Intro::details::TelegramLogoImage().scaled(
|
|
logoSize * cIntRetinaFactor(),
|
|
logoSize * cIntRetinaFactor(),
|
|
Qt::IgnoreAspectRatio,
|
|
Qt::SmoothTransformation));
|
|
}
|
|
return result;
|
|
};
|
|
return Qr::ReplaceCenter(
|
|
Qr::Generate(data, pixel, color),
|
|
image(Qr::ReplaceSize(data, pixel)));
|
|
}
|
|
|
|
QImage Qr(const Qr::Data &data, int pixel, int max = 0) {
|
|
Expects(data.size > 0);
|
|
|
|
if (max > 0 && data.size * pixel > max) {
|
|
pixel = std::max(max / data.size, 1);
|
|
}
|
|
return QrExact(data, pixel * style::DevicePixelRatio(), st::windowFg->c);
|
|
}
|
|
|
|
QImage Qr(const QString &text, int pixel, int max) {
|
|
return Qr(Qr::Encode(text), pixel, max);
|
|
}
|
|
|
|
QImage QrForShare(const QString &text) {
|
|
const auto data = Qr::Encode(text);
|
|
const auto size = (kShareQrSize - 2 * kShareQrPadding);
|
|
const auto image = QrExact(data, size / data.size, Qt::black);
|
|
auto result = QImage(
|
|
kShareQrPadding * 2 + image.width(),
|
|
kShareQrPadding * 2 + image.height(),
|
|
QImage::Format_ARGB32_Premultiplied);
|
|
result.fill(Qt::white);
|
|
{
|
|
auto p = QPainter(&result);
|
|
p.drawImage(kShareQrPadding, kShareQrPadding, image);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void QrBox(
|
|
not_null<Ui::GenericBox*> box,
|
|
const QString &link,
|
|
Fn<void(QImage)> share) {
|
|
box->setTitle(tr::lng_group_invite_qr_title());
|
|
|
|
box->addButton(tr::lng_about_done(), [=] { box->closeBox(); });
|
|
|
|
const auto qr = Qr(
|
|
link,
|
|
st::inviteLinkQrPixel,
|
|
st::boxWidth - st::boxRowPadding.left() - st::boxRowPadding.right());
|
|
const auto size = qr.width() / style::DevicePixelRatio();
|
|
const auto height = st::inviteLinkQrSkip * 2 + size;
|
|
const auto container = box->addRow(
|
|
object_ptr<Ui::BoxContentDivider>(box, height),
|
|
st::inviteLinkQrMargin);
|
|
const auto button = Ui::CreateChild<Ui::AbstractButton>(container);
|
|
button->resize(size, size);
|
|
button->paintRequest(
|
|
) | rpl::start_with_next([=] {
|
|
QPainter(button).drawImage(QRect(0, 0, size, size), qr);
|
|
}, button->lifetime());
|
|
container->widthValue(
|
|
) | rpl::start_with_next([=](int width) {
|
|
button->move((width - size) / 2, st::inviteLinkQrSkip);
|
|
}, button->lifetime());
|
|
button->setClickedCallback([=] {
|
|
share(QrForShare(link));
|
|
});
|
|
|
|
box->addRow(
|
|
object_ptr<Ui::FlatLabel>(
|
|
box,
|
|
tr::lng_group_invite_qr_about(),
|
|
st::boxLabel),
|
|
st::inviteLinkQrValuePadding);
|
|
|
|
box->addLeftButton(
|
|
tr::lng_group_invite_context_copy(),
|
|
[=] { share(QrForShare(link)); });
|
|
}
|
|
|
|
Controller::Controller(
|
|
not_null<PeerData*> peer,
|
|
not_null<UserData*> admin,
|
|
rpl::producer<LinkData> data)
|
|
: _peer(peer)
|
|
, _data(LinkData{ .admin = admin })
|
|
, _api(&_peer->session().api().instance()) {
|
|
_data = std::move(data);
|
|
const auto current = _data.current();
|
|
_link = current.link;
|
|
_revoked = current.revoked;
|
|
}
|
|
|
|
rpl::producer<LinkData> Controller::dataValue() const {
|
|
return _data.value(
|
|
) | rpl::filter([=](const LinkData &data) {
|
|
return !ClosingLinkBox(data, _revoked);
|
|
});
|
|
}
|
|
|
|
void Controller::addHeaderBlock(not_null<Ui::VerticalLayout*> container) {
|
|
using namespace Settings;
|
|
|
|
const auto current = _data.current();
|
|
const auto revoked = current.revoked;
|
|
const auto link = current.link;
|
|
const auto admin = current.admin;
|
|
const auto weak = Ui::MakeWeak(container);
|
|
const auto copyLink = crl::guard(weak, [=] {
|
|
CopyInviteLink(link);
|
|
});
|
|
const auto shareLink = crl::guard(weak, [=] {
|
|
ShareInviteLinkBox(_peer, link);
|
|
});
|
|
const auto getLinkQr = crl::guard(weak, [=] {
|
|
InviteLinkQrBox(link);
|
|
});
|
|
const auto revokeLink = crl::guard(weak, [=] {
|
|
RevokeLink(_peer, admin, link);
|
|
});
|
|
const auto editLink = crl::guard(weak, [=] {
|
|
EditLink(_peer, _data.current());
|
|
});
|
|
const auto deleteLink = crl::guard(weak, [=] {
|
|
DeleteLink(_peer, admin, link);
|
|
});
|
|
|
|
const auto createMenu = [=] {
|
|
auto result = base::make_unique_q<Ui::PopupMenu>(container);
|
|
if (revoked) {
|
|
result->addAction(
|
|
tr::lng_group_invite_context_delete(tr::now),
|
|
deleteLink);
|
|
} else {
|
|
result->addAction(
|
|
tr::lng_group_invite_context_copy(tr::now),
|
|
copyLink);
|
|
result->addAction(
|
|
tr::lng_group_invite_context_share(tr::now),
|
|
shareLink);
|
|
result->addAction(
|
|
tr::lng_group_invite_context_qr(tr::now),
|
|
getLinkQr);
|
|
if (!admin->isBot()) {
|
|
result->addAction(
|
|
tr::lng_group_invite_context_edit(tr::now),
|
|
editLink);
|
|
result->addAction(
|
|
tr::lng_group_invite_context_revoke(tr::now),
|
|
revokeLink);
|
|
}
|
|
}
|
|
return result;
|
|
};
|
|
|
|
const auto prefix = qstr("https://");
|
|
const auto label = container->lifetime().make_state<Ui::InviteLinkLabel>(
|
|
container,
|
|
rpl::single(link.startsWith(prefix)
|
|
? link.mid(prefix.size())
|
|
: link),
|
|
createMenu);
|
|
container->add(
|
|
label->take(),
|
|
st::inviteLinkFieldPadding);
|
|
|
|
label->clicks(
|
|
) | rpl::start_with_next(copyLink, label->lifetime());
|
|
|
|
const auto reactivateWrap = container->add(
|
|
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
|
|
container,
|
|
object_ptr<Ui::VerticalLayout>(
|
|
container)));
|
|
const auto copyShareWrap = container->add(
|
|
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
|
|
container,
|
|
object_ptr<Ui::VerticalLayout>(
|
|
container)));
|
|
|
|
AddReactivateLinkButton(reactivateWrap->entity(), editLink);
|
|
AddCopyShareLinkButtons(copyShareWrap->entity(), copyLink, shareLink);
|
|
if (revoked) {
|
|
AddDeleteLinkButton(container, deleteLink);
|
|
}
|
|
|
|
AddSkip(container, st::inviteLinkJoinedRowPadding.bottom() * 2);
|
|
|
|
auto grayLabelText = dataValue(
|
|
) | rpl::map([=](const LinkData &data) {
|
|
const auto usageExpired = (data.usageLimit > 0)
|
|
&& (data.usageLimit <= data.usage);
|
|
return usageExpired
|
|
? tr::lng_group_invite_used_about()
|
|
: tr::lng_group_invite_expires_at(
|
|
lt_when,
|
|
rpl::single(langDateTime(
|
|
base::unixtime::parse(data.expireDate))));
|
|
}) | rpl::flatten_latest();
|
|
|
|
const auto redLabelWrap = container->add(
|
|
object_ptr<Ui::SlideWrap<Ui::DividerLabel>>(
|
|
container,
|
|
object_ptr<Ui::DividerLabel>(
|
|
container,
|
|
object_ptr<Ui::FlatLabel>(
|
|
container,
|
|
tr::lng_group_invite_expired_about(),
|
|
st::boxAttentionDividerLabel),
|
|
st::settingsDividerLabelPadding)));
|
|
const auto grayLabelWrap = container->add(
|
|
object_ptr<Ui::SlideWrap<Ui::DividerLabel>>(
|
|
container,
|
|
object_ptr<Ui::DividerLabel>(
|
|
container,
|
|
object_ptr<Ui::FlatLabel>(
|
|
container,
|
|
std::move(grayLabelText),
|
|
st::boxDividerLabel),
|
|
st::settingsDividerLabelPadding)));
|
|
const auto justDividerWrap = container->add(
|
|
object_ptr<Ui::SlideWrap<>>(
|
|
container,
|
|
object_ptr<Ui::BoxContentDivider>(container)));
|
|
AddSkip(container);
|
|
|
|
dataValue(
|
|
) | rpl::start_with_next([=](const LinkData &data) {
|
|
const auto now = base::unixtime::now();
|
|
const auto expired = IsExpiredLink(data, now);
|
|
reactivateWrap->toggle(
|
|
!revoked && expired && !admin->isBot(),
|
|
anim::type::instant);
|
|
copyShareWrap->toggle(!revoked && !expired, anim::type::instant);
|
|
|
|
const auto timeExpired = (data.expireDate > 0)
|
|
&& (data.expireDate <= now);
|
|
const auto usageExpired = (data.usageLimit > 0)
|
|
&& (data.usageLimit <= data.usage);
|
|
redLabelWrap->toggle(!revoked && timeExpired, anim::type::instant);
|
|
grayLabelWrap->toggle(
|
|
!revoked && !timeExpired && (data.expireDate > 0 || usageExpired),
|
|
anim::type::instant);
|
|
justDividerWrap->toggle(
|
|
revoked || (!data.expireDate && !expired),
|
|
anim::type::instant);
|
|
}, lifetime());
|
|
}
|
|
|
|
void Controller::prepare() {
|
|
using namespace Settings;
|
|
|
|
auto header = object_ptr<Ui::VerticalLayout>((QWidget*)nullptr);
|
|
const auto container = header.data();
|
|
|
|
const auto current = _data.current();
|
|
const auto revoked = current.revoked;
|
|
if (revoked || !current.permanent) {
|
|
addHeaderBlock(container);
|
|
}
|
|
AddSubsectionTitle(
|
|
container,
|
|
tr::lng_group_invite_created_by());
|
|
AddSinglePeerRow(
|
|
container,
|
|
current.admin,
|
|
rpl::single(langDateTime(base::unixtime::parse(current.date))));
|
|
AddSkip(container, st::membersMarginBottom);
|
|
|
|
const auto listHeaderWrap = container->add(
|
|
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
|
|
container,
|
|
object_ptr<Ui::VerticalLayout>(
|
|
container)));
|
|
const auto listHeader = listHeaderWrap->entity();
|
|
|
|
// Make this container occupy full width.
|
|
listHeader->add(object_ptr<Ui::RpWidget>(listHeader));
|
|
|
|
AddDivider(listHeader);
|
|
AddSkip(listHeader);
|
|
|
|
auto listHeaderText = dataValue(
|
|
) | rpl::map([=](const LinkData &data) {
|
|
const auto now = base::unixtime::now();
|
|
const auto timeExpired = (data.expireDate > 0)
|
|
&& (data.expireDate <= now);
|
|
if (!revoked && !data.usage && data.usageLimit > 0 && !timeExpired) {
|
|
auto description = object_ptr<Ui::FlatLabel>(
|
|
nullptr,
|
|
tr::lng_group_invite_can_join_via_link(
|
|
tr::now,
|
|
lt_count,
|
|
data.usageLimit),
|
|
computeListSt().about);
|
|
if (!delegate()->peerListFullRowsCount()) {
|
|
using namespace rpl::mappers;
|
|
_addedHeight = description->heightValue(
|
|
) | rpl::map(_1
|
|
+ st::membersAboutLimitPadding.top()
|
|
+ st::membersAboutLimitPadding.bottom());
|
|
}
|
|
delegate()->peerListSetDescription(std::move(description));
|
|
} else {
|
|
_addedHeight = std::max(
|
|
data.usage,
|
|
delegate()->peerListFullRowsCount()
|
|
) * computeListSt().item.height;
|
|
delegate()->peerListSetDescription(nullptr);
|
|
}
|
|
listHeaderWrap->toggle(
|
|
!revoked && (data.usage || (data.usageLimit > 0 && !timeExpired)),
|
|
anim::type::instant);
|
|
delegate()->peerListRefreshRows();
|
|
return data.usage
|
|
? tr::lng_group_invite_joined(
|
|
lt_count,
|
|
rpl::single(float64(data.usage)))
|
|
: tr::lng_group_invite_no_joined();
|
|
}) | rpl::flatten_latest();
|
|
const auto listTitle = AddSubsectionTitle(
|
|
listHeader,
|
|
std::move(listHeaderText));
|
|
auto remainingText = dataValue(
|
|
) | rpl::map([=](const LinkData &data) {
|
|
return !data.usageLimit
|
|
? QString()
|
|
: tr::lng_group_invite_remaining(
|
|
tr::now,
|
|
lt_count_decimal,
|
|
std::max(data.usageLimit - data.usage, 0));
|
|
});
|
|
const auto remaining = Ui::CreateChild<Ui::FlatLabel>(
|
|
listHeader,
|
|
std::move(remainingText),
|
|
st::settingsSubsectionTitleRight);
|
|
dataValue(
|
|
) | rpl::start_with_next([=](const LinkData &data) {
|
|
remaining->setTextColorOverride(
|
|
(data.usageLimit && (data.usageLimit <= data.usage)
|
|
? std::make_optional(st::boxTextFgError->c)
|
|
: std::nullopt));
|
|
if (revoked || (!data.usage && data.usageLimit > 0)) {
|
|
remaining->hide();
|
|
} else {
|
|
remaining->show();
|
|
}
|
|
}, remaining->lifetime());
|
|
|
|
rpl::combine(
|
|
listTitle->positionValue(),
|
|
remaining->widthValue(),
|
|
listHeader->widthValue()
|
|
) | rpl::start_with_next([=](
|
|
QPoint position,
|
|
int width,
|
|
int outerWidth) {
|
|
remaining->moveToRight(position.x(), position.y(), outerWidth);
|
|
}, remaining->lifetime());
|
|
|
|
_headerWidget = header.data();
|
|
|
|
delegate()->peerListSetAboveWidget(std::move(header));
|
|
_allLoaded = (current.usage == 0);
|
|
|
|
const auto &inviteLinks = _peer->session().api().inviteLinks();
|
|
const auto slice = inviteLinks.joinedFirstSliceLoaded(_peer, _link);
|
|
if (slice) {
|
|
appendSlice(*slice);
|
|
}
|
|
loadMoreRows();
|
|
}
|
|
|
|
void Controller::loadMoreRows() {
|
|
if (_requestId || _allLoaded) {
|
|
return;
|
|
}
|
|
_requestId = _api.request(MTPmessages_GetChatInviteImporters(
|
|
MTP_flags(0),
|
|
_peer->input,
|
|
MTP_string(_link),
|
|
MTPstring(), // q // #TODO requests
|
|
MTP_int(_lastUser ? _lastUser->date : 0),
|
|
_lastUser ? _lastUser->user->inputUser : MTP_inputUserEmpty(),
|
|
MTP_int(_lastUser ? kPerPage : kFirstPage)
|
|
)).done([=](const MTPmessages_ChatInviteImporters &result) {
|
|
_requestId = 0;
|
|
auto slice = Api::ParseJoinedByLinkSlice(_peer, result);
|
|
_allLoaded = slice.users.empty();
|
|
appendSlice(slice);
|
|
}).fail([=](const MTP::Error &error) {
|
|
_requestId = 0;
|
|
_allLoaded = true;
|
|
}).send();
|
|
}
|
|
|
|
void Controller::appendSlice(const Api::JoinedByLinkSlice &slice) {
|
|
for (const auto &user : slice.users) {
|
|
_lastUser = user;
|
|
delegate()->peerListAppendRow(
|
|
std::make_unique<PeerListRow>(user.user));
|
|
}
|
|
delegate()->peerListRefreshRows();
|
|
if (delegate()->peerListFullRowsCount() > 0) {
|
|
_addedHeight = std::max(
|
|
_data.current().usage,
|
|
delegate()->peerListFullRowsCount()
|
|
) * computeListSt().item.height;
|
|
}
|
|
}
|
|
|
|
void Controller::rowClicked(not_null<PeerListRow*> row) {
|
|
Ui::showPeerProfile(row->peer());
|
|
}
|
|
|
|
Main::Session &Controller::session() const {
|
|
return _peer->session();
|
|
}
|
|
|
|
rpl::producer<int> Controller::boxHeightValue() const {
|
|
Expects(_headerWidget != nullptr);
|
|
|
|
return rpl::combine(
|
|
_headerWidget->heightValue(),
|
|
_addedHeight.value()
|
|
) | rpl::map([=](int header, int description) {
|
|
const auto wrapped = description
|
|
? (computeListSt().padding.top()
|
|
+ description
|
|
+ computeListSt().padding.bottom())
|
|
: 0;
|
|
return std::min(header + wrapped, st::boxMaxListHeight);
|
|
});
|
|
}
|
|
|
|
int Controller::descriptionTopSkipMin() const {
|
|
return 0;
|
|
}
|
|
|
|
SingleRowController::SingleRowController(
|
|
not_null<PeerData*> peer,
|
|
rpl::producer<QString> status)
|
|
: _peer(peer)
|
|
, _status(std::move(status)) {
|
|
}
|
|
|
|
void SingleRowController::prepare() {
|
|
auto row = std::make_unique<PeerListRow>(_peer);
|
|
|
|
const auto raw = row.get();
|
|
std::move(
|
|
_status
|
|
) | rpl::start_with_next([=](const QString &status) {
|
|
raw->setCustomStatus(status);
|
|
delegate()->peerListUpdateRow(raw);
|
|
}, _lifetime);
|
|
|
|
delegate()->peerListAppendRow(std::move(row));
|
|
delegate()->peerListRefreshRows();
|
|
}
|
|
|
|
void SingleRowController::loadMoreRows() {
|
|
}
|
|
|
|
void SingleRowController::rowClicked(not_null<PeerListRow*> row) {
|
|
Ui::showPeerProfile(row->peer());
|
|
}
|
|
|
|
Main::Session &SingleRowController::session() const {
|
|
return _peer->session();
|
|
}
|
|
|
|
} // namespace
|
|
|
|
bool IsExpiredLink(const Api::InviteLink &data, TimeId now) {
|
|
return (data.expireDate > 0 && data.expireDate <= now)
|
|
|| (data.usageLimit > 0 && data.usageLimit <= data.usage);
|
|
}
|
|
|
|
void AddSinglePeerRow(
|
|
not_null<Ui::VerticalLayout*> container,
|
|
not_null<PeerData*> peer,
|
|
rpl::producer<QString> status) {
|
|
const auto delegate = container->lifetime().make_state<
|
|
PeerListContentDelegateSimple
|
|
>();
|
|
const auto controller = container->lifetime().make_state<
|
|
SingleRowController
|
|
>(peer, std::move(status));
|
|
controller->setStyleOverrides(&st::peerListSingleRow);
|
|
const auto content = container->add(object_ptr<PeerListContent>(
|
|
container,
|
|
controller));
|
|
delegate->setContent(content);
|
|
controller->setDelegate(delegate);
|
|
}
|
|
|
|
void AddPermanentLinkBlock(
|
|
not_null<Ui::VerticalLayout*> container,
|
|
not_null<PeerData*> peer,
|
|
not_null<UserData*> admin,
|
|
rpl::producer<Api::InviteLink> fromList) {
|
|
struct LinkData {
|
|
QString link;
|
|
int usage = 0;
|
|
};
|
|
const auto value = container->lifetime().make_state<
|
|
rpl::variable<LinkData>
|
|
>();
|
|
if (admin->isSelf()) {
|
|
*value = peer->session().changes().peerFlagsValue(
|
|
peer,
|
|
Data::PeerUpdate::Flag::InviteLinks
|
|
) | rpl::map([=] {
|
|
const auto &links = peer->session().api().inviteLinks().myLinks(
|
|
peer).links;
|
|
const auto link = links.empty() ? nullptr : &links.front();
|
|
return (link && link->permanent && !link->revoked)
|
|
? LinkData{ link->link, link->usage }
|
|
: LinkData();
|
|
});
|
|
} else {
|
|
*value = std::move(
|
|
fromList
|
|
) | rpl::map([](const Api::InviteLink &link) {
|
|
return LinkData{ link.link, link.usage };
|
|
});
|
|
}
|
|
const auto weak = Ui::MakeWeak(container);
|
|
const auto copyLink = crl::guard(weak, [=] {
|
|
if (const auto current = value->current(); !current.link.isEmpty()) {
|
|
CopyInviteLink(current.link);
|
|
}
|
|
});
|
|
const auto shareLink = crl::guard(weak, [=] {
|
|
if (const auto current = value->current(); !current.link.isEmpty()) {
|
|
ShareInviteLinkBox(peer, current.link);
|
|
}
|
|
});
|
|
const auto getLinkQr = crl::guard(weak, [=] {
|
|
if (const auto current = value->current(); !current.link.isEmpty()) {
|
|
InviteLinkQrBox(current.link);
|
|
}
|
|
});
|
|
const auto revokeLink = crl::guard(weak, [=] {
|
|
const auto box = std::make_shared<QPointer<Ui::ConfirmBox>>();
|
|
const auto done = crl::guard(weak, [=] {
|
|
const auto close = [=] {
|
|
if (*box) {
|
|
(*box)->closeBox();
|
|
}
|
|
};
|
|
peer->session().api().inviteLinks().revokePermanent(
|
|
peer,
|
|
admin,
|
|
value->current().link,
|
|
close);
|
|
});
|
|
*box = Ui::show(
|
|
Box<Ui::ConfirmBox>(
|
|
tr::lng_group_invite_about_new(tr::now),
|
|
done),
|
|
Ui::LayerOption::KeepOther);
|
|
});
|
|
|
|
auto link = value->value(
|
|
) | rpl::map([=](const LinkData &data) {
|
|
const auto prefix = qstr("https://");
|
|
return data.link.startsWith(prefix)
|
|
? data.link.mid(prefix.size())
|
|
: data.link;
|
|
});
|
|
const auto createMenu = [=] {
|
|
auto result = base::make_unique_q<Ui::PopupMenu>(container);
|
|
result->addAction(
|
|
tr::lng_group_invite_context_copy(tr::now),
|
|
copyLink);
|
|
result->addAction(
|
|
tr::lng_group_invite_context_share(tr::now),
|
|
shareLink);
|
|
result->addAction(
|
|
tr::lng_group_invite_context_qr(tr::now),
|
|
getLinkQr);
|
|
if (!admin->isBot()) {
|
|
result->addAction(
|
|
tr::lng_group_invite_context_revoke(tr::now),
|
|
revokeLink);
|
|
}
|
|
return result;
|
|
};
|
|
const auto label = container->lifetime().make_state<Ui::InviteLinkLabel>(
|
|
container,
|
|
std::move(link),
|
|
createMenu);
|
|
container->add(
|
|
label->take(),
|
|
st::inviteLinkFieldPadding);
|
|
|
|
label->clicks(
|
|
) | rpl::start_with_next(copyLink, label->lifetime());
|
|
|
|
AddCopyShareLinkButtons(container, copyLink, shareLink);
|
|
|
|
struct JoinedState {
|
|
QImage cachedUserpics;
|
|
std::vector<HistoryView::UserpicInRow> list;
|
|
int count = 0;
|
|
bool allUserpicsLoaded = false;
|
|
rpl::variable<Ui::JoinedCountContent> content;
|
|
rpl::lifetime lifetime;
|
|
};
|
|
const auto state = container->lifetime().make_state<JoinedState>();
|
|
const auto push = [=] {
|
|
HistoryView::GenerateUserpicsInRow(
|
|
state->cachedUserpics,
|
|
state->list,
|
|
st::inviteLinkUserpics,
|
|
0);
|
|
state->allUserpicsLoaded = ranges::all_of(
|
|
state->list,
|
|
[](const HistoryView::UserpicInRow &element) {
|
|
return !element.peer->hasUserpic() || element.view->image();
|
|
});
|
|
state->content = Ui::JoinedCountContent{
|
|
.count = state->count,
|
|
.userpics = state->cachedUserpics
|
|
};
|
|
};
|
|
value->value(
|
|
) | rpl::map([=](const LinkData &data) {
|
|
return peer->session().api().inviteLinks().joinedFirstSliceValue(
|
|
peer,
|
|
data.link,
|
|
data.usage);
|
|
}) | rpl::flatten_latest(
|
|
) | rpl::start_with_next([=](const Api::JoinedByLinkSlice &slice) {
|
|
auto list = std::vector<HistoryView::UserpicInRow>();
|
|
list.reserve(slice.users.size());
|
|
for (const auto &item : slice.users) {
|
|
const auto i = ranges::find(
|
|
state->list,
|
|
item.user,
|
|
&HistoryView::UserpicInRow::peer);
|
|
if (i != end(state->list)) {
|
|
list.push_back(std::move(*i));
|
|
} else {
|
|
list.push_back({ item.user });
|
|
}
|
|
}
|
|
state->count = slice.count;
|
|
state->list = std::move(list);
|
|
push();
|
|
}, state->lifetime);
|
|
|
|
peer->session().downloaderTaskFinished(
|
|
) | rpl::filter([=] {
|
|
return !state->allUserpicsLoaded;
|
|
}) | rpl::start_with_next([=] {
|
|
auto pushing = false;
|
|
state->allUserpicsLoaded = true;
|
|
for (const auto &element : state->list) {
|
|
if (!element.peer->hasUserpic()) {
|
|
continue;
|
|
} else if (element.peer->userpicUniqueKey(element.view)
|
|
!= element.uniqueKey) {
|
|
pushing = true;
|
|
} else if (!element.view->image()) {
|
|
state->allUserpicsLoaded = false;
|
|
}
|
|
}
|
|
if (pushing) {
|
|
push();
|
|
}
|
|
}, state->lifetime);
|
|
|
|
Ui::AddJoinedCountButton(
|
|
container,
|
|
state->content.value(),
|
|
st::inviteLinkJoinedRowPadding
|
|
)->setClickedCallback([=] {
|
|
});
|
|
|
|
container->add(object_ptr<Ui::SlideWrap<Ui::FixedHeightWidget>>(
|
|
container,
|
|
object_ptr<Ui::FixedHeightWidget>(
|
|
container,
|
|
st::inviteLinkJoinedRowPadding.bottom()))
|
|
)->setDuration(0)->toggleOn(state->content.value(
|
|
) | rpl::map([=](const Ui::JoinedCountContent &content) {
|
|
return (content.count <= 0);
|
|
}));
|
|
}
|
|
|
|
void CopyInviteLink(const QString &link) {
|
|
QGuiApplication::clipboard()->setText(link);
|
|
Ui::Toast::Show(tr::lng_group_invite_copied(tr::now));
|
|
}
|
|
|
|
void ShareInviteLinkBox(not_null<PeerData*> peer, const QString &link) {
|
|
const auto sending = std::make_shared<bool>();
|
|
const auto box = std::make_shared<QPointer<ShareBox>>();
|
|
|
|
auto copyCallback = [=] {
|
|
QGuiApplication::clipboard()->setText(link);
|
|
Ui::Toast::Show(tr::lng_group_invite_copied(tr::now));
|
|
};
|
|
auto submitCallback = [=](
|
|
std::vector<not_null<PeerData*>> &&result,
|
|
TextWithTags &&comment,
|
|
Api::SendOptions options) {
|
|
if (*sending || result.empty()) {
|
|
return;
|
|
}
|
|
|
|
const auto error = [&] {
|
|
for (const auto peer : result) {
|
|
const auto error = GetErrorTextForSending(
|
|
peer,
|
|
{},
|
|
comment);
|
|
if (!error.isEmpty()) {
|
|
return std::make_pair(error, peer);
|
|
}
|
|
}
|
|
return std::make_pair(QString(), result.front());
|
|
}();
|
|
if (!error.first.isEmpty()) {
|
|
auto text = TextWithEntities();
|
|
if (result.size() > 1) {
|
|
text.append(
|
|
Ui::Text::Bold(error.second->name)
|
|
).append("\n\n");
|
|
}
|
|
text.append(error.first);
|
|
Ui::show(
|
|
Box<Ui::InformBox>(text),
|
|
Ui::LayerOption::KeepOther);
|
|
return;
|
|
}
|
|
|
|
*sending = true;
|
|
if (!comment.text.isEmpty()) {
|
|
comment.text = link + "\n" + comment.text;
|
|
const auto add = link.size() + 1;
|
|
for (auto &tag : comment.tags) {
|
|
tag.offset += add;
|
|
}
|
|
} else {
|
|
comment.text = link;
|
|
}
|
|
const auto owner = &peer->owner();
|
|
auto &api = peer->session().api();
|
|
for (const auto peer : result) {
|
|
const auto history = owner->history(peer);
|
|
auto message = ApiWrap::MessageToSend(history);
|
|
message.textWithTags = comment;
|
|
message.action.options = options;
|
|
message.action.clearDraft = false;
|
|
api.sendMessage(std::move(message));
|
|
}
|
|
Ui::Toast::Show(tr::lng_share_done(tr::now));
|
|
if (*box) {
|
|
(*box)->closeBox();
|
|
}
|
|
};
|
|
*box = Ui::show(
|
|
Box<ShareBox>(ShareBox::Descriptor{
|
|
.session = &peer->session(),
|
|
.copyCallback = std::move(copyCallback),
|
|
.submitCallback = std::move(submitCallback),
|
|
.filterCallback = [](auto peer) { return peer->canWrite(); },
|
|
.navigation = App::wnd()->sessionController() }),
|
|
Ui::LayerOption::KeepOther);
|
|
}
|
|
|
|
void InviteLinkQrBox(const QString &link) {
|
|
Ui::show(Box(QrBox, link, [=](const QImage &image) {
|
|
auto mime = std::make_unique<QMimeData>();
|
|
mime->setImageData(image);
|
|
QGuiApplication::clipboard()->setMimeData(mime.release());
|
|
|
|
Ui::Toast::Show(tr::lng_group_invite_qr_copied(tr::now));
|
|
}));
|
|
}
|
|
|
|
void EditLink(
|
|
not_null<PeerData*> peer,
|
|
const Api::InviteLink &data) {
|
|
const auto creating = data.link.isEmpty();
|
|
const auto box = std::make_shared<QPointer<Ui::GenericBox>>();
|
|
using Fields = Ui::InviteLinkFields;
|
|
const auto done = [=](Fields result) {
|
|
const auto finish = [=](Api::InviteLink finished) {
|
|
if (creating) {
|
|
ShowInviteLinkBox(peer, finished);
|
|
}
|
|
if (*box) {
|
|
(*box)->closeBox();
|
|
}
|
|
};
|
|
if (creating) {
|
|
Assert(data.admin->isSelf());
|
|
peer->session().api().inviteLinks().create(
|
|
peer,
|
|
finish,
|
|
result.expireDate,
|
|
result.usageLimit);
|
|
} else {
|
|
peer->session().api().inviteLinks().edit(
|
|
peer,
|
|
data.admin,
|
|
result.link,
|
|
result.expireDate,
|
|
result.usageLimit,
|
|
finish);
|
|
}
|
|
};
|
|
*box = Ui::show(
|
|
(creating
|
|
? Box(Ui::CreateInviteLinkBox, done)
|
|
: Box(
|
|
Ui::EditInviteLinkBox,
|
|
Fields{
|
|
.link = data.link,
|
|
.expireDate = data.expireDate,
|
|
.usageLimit = data.usageLimit
|
|
},
|
|
done)),
|
|
Ui::LayerOption::KeepOther);
|
|
}
|
|
|
|
void RevokeLink(
|
|
not_null<PeerData*> peer,
|
|
not_null<UserData*> admin,
|
|
const QString &link) {
|
|
const auto box = std::make_shared<QPointer<Ui::ConfirmBox>>();
|
|
const auto revoke = [=] {
|
|
const auto done = [=](const LinkData &data) {
|
|
if (*box) {
|
|
(*box)->closeBox();
|
|
}
|
|
};
|
|
peer->session().api().inviteLinks().revoke(peer, admin, link, done);
|
|
};
|
|
*box = Ui::show(
|
|
Box<Ui::ConfirmBox>(
|
|
tr::lng_group_invite_revoke_about(tr::now),
|
|
revoke),
|
|
Ui::LayerOption::KeepOther);
|
|
}
|
|
|
|
void DeleteLink(
|
|
not_null<PeerData*> peer,
|
|
not_null<UserData*> admin,
|
|
const QString &link) {
|
|
const auto box = std::make_shared<QPointer<Ui::ConfirmBox>>();
|
|
const auto sure = [=] {
|
|
const auto finish = [=] {
|
|
if (*box) {
|
|
(*box)->closeBox();
|
|
}
|
|
};
|
|
peer->session().api().inviteLinks().destroy(
|
|
peer,
|
|
admin,
|
|
link,
|
|
finish);
|
|
};
|
|
*box = Ui::show(
|
|
Box<Ui::ConfirmBox>(tr::lng_group_invite_delete_sure(tr::now), sure),
|
|
Ui::LayerOption::KeepOther);
|
|
}
|
|
|
|
void ShowInviteLinkBox(
|
|
not_null<PeerData*> peer,
|
|
const Api::InviteLink &link) {
|
|
const auto admin = link.admin;
|
|
const auto linkText = link.link;
|
|
const auto revoked = link.revoked;
|
|
|
|
auto updates = peer->session().api().inviteLinks().updates(
|
|
peer,
|
|
admin
|
|
) | rpl::filter([=](const Api::InviteLinkUpdate &update) {
|
|
return (update.was == linkText);
|
|
}) | rpl::map([=](const Api::InviteLinkUpdate &update) {
|
|
return update.now ? *update.now : LinkData{ .admin = admin };
|
|
});
|
|
auto data = rpl::single(link) | rpl::then(std::move(updates));
|
|
|
|
auto initBox = [=, data = rpl::duplicate(data)](
|
|
not_null<Ui::BoxContent*> box) {
|
|
rpl::duplicate(
|
|
data
|
|
) | rpl::start_with_next([=](const LinkData &link) {
|
|
if (ClosingLinkBox(link, revoked)) {
|
|
box->closeBox();
|
|
return;
|
|
}
|
|
const auto now = base::unixtime::now();
|
|
box->setTitle(link.revoked
|
|
? tr::lng_manage_peer_link_invite()
|
|
: IsExpiredLink(link, now)
|
|
? tr::lng_manage_peer_link_expired()
|
|
: link.permanent
|
|
? tr::lng_manage_peer_link_permanent()
|
|
: tr::lng_manage_peer_link_invite());
|
|
}, box->lifetime());
|
|
|
|
box->addButton(tr::lng_about_done(), [=] { box->closeBox(); });
|
|
};
|
|
Ui::show(
|
|
Box<PeerListBox>(
|
|
std::make_unique<Controller>(peer, link.admin, std::move(data)),
|
|
std::move(initBox)),
|
|
Ui::LayerOption::KeepOther);
|
|
}
|