1354 lines
		
	
	
	
		
			36 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1354 lines
		
	
	
	
		
			36 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_channel.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/toasts/common_toasts.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_bar.h" // GenerateUserpics...
 | 
						|
#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 "styles/style_menu_icons.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 RequestedRow final : public PeerListRow {
 | 
						|
public:
 | 
						|
	RequestedRow(not_null<PeerData*> peer, TimeId date);
 | 
						|
 | 
						|
	QSize rightActionSize() const override;
 | 
						|
	QMargins rightActionMargins() const override;
 | 
						|
	void rightActionPaint(
 | 
						|
		Painter &p,
 | 
						|
		int x,
 | 
						|
		int y,
 | 
						|
		int outerWidth,
 | 
						|
		bool selected,
 | 
						|
		bool actionSelected) override;
 | 
						|
 | 
						|
};
 | 
						|
 | 
						|
RequestedRow::RequestedRow(not_null<PeerData*> peer, TimeId date)
 | 
						|
: PeerListRow(peer) {
 | 
						|
	setCustomStatus(PrepareRequestedRowStatus(date));
 | 
						|
}
 | 
						|
 | 
						|
QSize RequestedRow::rightActionSize() const {
 | 
						|
	return QSize(
 | 
						|
		st::inviteLinkThreeDotsIcon.width(),
 | 
						|
		st::inviteLinkThreeDotsIcon.height());
 | 
						|
}
 | 
						|
 | 
						|
QMargins RequestedRow::rightActionMargins() const {
 | 
						|
	return QMargins(
 | 
						|
		0,
 | 
						|
		(st::peerListBoxItem.height - rightActionSize().height()) / 2,
 | 
						|
		st::inviteLinkThreeDotsSkip,
 | 
						|
		0);
 | 
						|
}
 | 
						|
 | 
						|
void RequestedRow::rightActionPaint(
 | 
						|
		Painter &p,
 | 
						|
		int x,
 | 
						|
		int y,
 | 
						|
		int outerWidth,
 | 
						|
		bool selected,
 | 
						|
		bool actionSelected) {
 | 
						|
	(actionSelected
 | 
						|
		? st::inviteLinkThreeDotsIconOver
 | 
						|
		: st::inviteLinkThreeDotsIcon).paint(p, x, y, outerWidth);
 | 
						|
}
 | 
						|
 | 
						|
class Controller final
 | 
						|
	: public PeerListController
 | 
						|
	, public base::has_weak_ptr {
 | 
						|
public:
 | 
						|
	enum class Role {
 | 
						|
		Requested,
 | 
						|
		Joined,
 | 
						|
	};
 | 
						|
	Controller(
 | 
						|
		not_null<PeerData*> peer,
 | 
						|
		not_null<UserData*> admin,
 | 
						|
		rpl::producer<LinkData> data,
 | 
						|
		Role role);
 | 
						|
 | 
						|
	void prepare() override;
 | 
						|
	void loadMoreRows() override;
 | 
						|
	void rowClicked(not_null<PeerListRow*> row) override;
 | 
						|
	void rowRightActionClicked(not_null<PeerListRow*> row) override;
 | 
						|
	Main::Session &session() const override;
 | 
						|
 | 
						|
	rpl::producer<int> boxHeightValue() const override;
 | 
						|
	int descriptionTopSkipMin() const override;
 | 
						|
 | 
						|
	struct Processed {
 | 
						|
		not_null<UserData*> user;
 | 
						|
		bool approved = false;
 | 
						|
	};
 | 
						|
	[[nodiscard]] rpl::producer<Processed> processed() const {
 | 
						|
		return _processed.events();
 | 
						|
	}
 | 
						|
 | 
						|
private:
 | 
						|
	base::unique_qptr<Ui::PopupMenu> rowContextMenu(
 | 
						|
		QWidget *parent,
 | 
						|
		not_null<PeerListRow*> row) override;
 | 
						|
 | 
						|
	void setupAboveJoinedWidget();
 | 
						|
	void appendSlice(const Api::JoinedByLinkSlice &slice);
 | 
						|
	void addHeaderBlock(not_null<Ui::VerticalLayout*> container);
 | 
						|
	not_null<Ui::SlideWrap<>*> addRequestedListBlock(
 | 
						|
		not_null<Ui::VerticalLayout*> container);
 | 
						|
	void updateWithProcessed(Processed processed);
 | 
						|
 | 
						|
	[[nodiscard]] rpl::producer<LinkData> dataValue() const;
 | 
						|
 | 
						|
	[[nodiscard]] base::unique_qptr<Ui::PopupMenu> createRowContextMenu(
 | 
						|
		QWidget *parent,
 | 
						|
		not_null<PeerListRow*> row);
 | 
						|
	void processRequest(not_null<UserData*> user, bool approved);
 | 
						|
 | 
						|
	const not_null<PeerData*> _peer;
 | 
						|
	const Role _role = Role::Joined;
 | 
						|
	rpl::variable<LinkData> _data;
 | 
						|
 | 
						|
	base::unique_qptr<Ui::PopupMenu> _menu;
 | 
						|
	rpl::event_stream<Processed> _processed;
 | 
						|
 | 
						|
	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,
 | 
						|
	Role role)
 | 
						|
: _peer(peer)
 | 
						|
, _role(role)
 | 
						|
, _data(LinkData{ .admin = admin })
 | 
						|
, _api(&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,
 | 
						|
			st::popupMenuWithIcons);
 | 
						|
		if (revoked) {
 | 
						|
			result->addAction(
 | 
						|
				tr::lng_group_invite_context_delete(tr::now),
 | 
						|
				deleteLink,
 | 
						|
				&st::menuIconDelete);
 | 
						|
		} else {
 | 
						|
			result->addAction(
 | 
						|
				tr::lng_group_invite_context_copy(tr::now),
 | 
						|
				copyLink,
 | 
						|
				&st::menuIconCopy);
 | 
						|
			result->addAction(
 | 
						|
				tr::lng_group_invite_context_share(tr::now),
 | 
						|
				shareLink,
 | 
						|
				&st::menuIconShare);
 | 
						|
			result->addAction(
 | 
						|
				tr::lng_group_invite_context_qr(tr::now),
 | 
						|
				getLinkQr,
 | 
						|
				&st::menuIconQrCode);
 | 
						|
			if (!admin->isBot()) {
 | 
						|
				result->addAction(
 | 
						|
					tr::lng_group_invite_context_edit(tr::now),
 | 
						|
					editLink,
 | 
						|
					&st::menuIconEdit);
 | 
						|
				result->addAction(
 | 
						|
					tr::lng_group_invite_context_revoke(tr::now),
 | 
						|
					revokeLink,
 | 
						|
					&st::menuIconRemove);
 | 
						|
			}
 | 
						|
		}
 | 
						|
		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());
 | 
						|
}
 | 
						|
 | 
						|
not_null<Ui::SlideWrap<>*> Controller::addRequestedListBlock(
 | 
						|
		not_null<Ui::VerticalLayout*> container) {
 | 
						|
	using namespace Settings;
 | 
						|
 | 
						|
	auto result = container->add(
 | 
						|
		object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
 | 
						|
			container,
 | 
						|
			object_ptr<Ui::VerticalLayout>(
 | 
						|
				container)));
 | 
						|
	const auto wrap = result->entity();
 | 
						|
	// Make this container occupy full width.
 | 
						|
	wrap->add(object_ptr<Ui::RpWidget>(wrap));
 | 
						|
	AddDivider(wrap);
 | 
						|
	AddSkip(wrap);
 | 
						|
	auto requestedCount = dataValue(
 | 
						|
	) | rpl::filter([](const LinkData &data) {
 | 
						|
		return data.requested > 0;
 | 
						|
	}) | rpl::map([=](const LinkData &data) {
 | 
						|
		return float64(data.requested);
 | 
						|
	});
 | 
						|
	AddSubsectionTitle(
 | 
						|
		wrap,
 | 
						|
		tr::lng_group_invite_requested_full(
 | 
						|
			lt_count_decimal,
 | 
						|
			std::move(requestedCount)));
 | 
						|
 | 
						|
	const auto delegate = container->lifetime().make_state<
 | 
						|
		PeerListContentDelegateSimple
 | 
						|
	>();
 | 
						|
	const auto controller = container->lifetime().make_state<
 | 
						|
		Controller
 | 
						|
	>(_peer, _data.current().admin, _data.value(), Role::Requested);
 | 
						|
	const auto content = container->add(object_ptr<PeerListContent>(
 | 
						|
		container,
 | 
						|
		controller));
 | 
						|
	delegate->setContent(content);
 | 
						|
	controller->setDelegate(delegate);
 | 
						|
 | 
						|
	controller->processed(
 | 
						|
	) | rpl::start_with_next([=](Processed processed) {
 | 
						|
		updateWithProcessed(processed);
 | 
						|
	}, lifetime());
 | 
						|
 | 
						|
	return result;
 | 
						|
}
 | 
						|
 | 
						|
void Controller::prepare() {
 | 
						|
	if (_role == Role::Joined) {
 | 
						|
		setupAboveJoinedWidget();
 | 
						|
 | 
						|
		_allLoaded = (_data.current().usage == 0);
 | 
						|
 | 
						|
		const auto &inviteLinks = session().api().inviteLinks();
 | 
						|
		const auto slice = inviteLinks.joinedFirstSliceLoaded(_peer, _link);
 | 
						|
		if (slice) {
 | 
						|
			appendSlice(*slice);
 | 
						|
		}
 | 
						|
	} else {
 | 
						|
		_allLoaded = (_data.current().requested == 0);
 | 
						|
	}
 | 
						|
	loadMoreRows();
 | 
						|
}
 | 
						|
 | 
						|
void Controller::updateWithProcessed(Processed processed) {
 | 
						|
	const auto user = processed.user;
 | 
						|
	auto updated = _data.current();
 | 
						|
	if (processed.approved) {
 | 
						|
		++updated.usage;
 | 
						|
		if (!delegate()->peerListFindRow(user->id.value)) {
 | 
						|
			delegate()->peerListPrependRow(
 | 
						|
				std::make_unique<PeerListRow>(user));
 | 
						|
			delegate()->peerListRefreshRows();
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if (updated.requested > 0) {
 | 
						|
		--updated.requested;
 | 
						|
	}
 | 
						|
	session().api().inviteLinks().applyExternalUpdate(_peer, updated);
 | 
						|
}
 | 
						|
 | 
						|
void Controller::setupAboveJoinedWidget() {
 | 
						|
	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);
 | 
						|
 | 
						|
	auto requestedWrap = addRequestedListBlock(container);
 | 
						|
 | 
						|
	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();
 | 
						|
		}
 | 
						|
 | 
						|
		requestedWrap->toggle(data.requested > 0, anim::type::instant);
 | 
						|
	}, 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));
 | 
						|
}
 | 
						|
 | 
						|
void Controller::loadMoreRows() {
 | 
						|
	if (_requestId || _allLoaded) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	using Flag = MTPmessages_GetChatInviteImporters::Flag;
 | 
						|
	_requestId = _api.request(MTPmessages_GetChatInviteImporters(
 | 
						|
		MTP_flags(Flag::f_link
 | 
						|
			| (_role == Role::Requested ? Flag::f_requested : Flag(0))),
 | 
						|
		_peer->input,
 | 
						|
		MTP_string(_link),
 | 
						|
		MTPstring(), // q
 | 
						|
		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([=] {
 | 
						|
		_requestId = 0;
 | 
						|
		_allLoaded = true;
 | 
						|
	}).send();
 | 
						|
}
 | 
						|
 | 
						|
void Controller::appendSlice(const Api::JoinedByLinkSlice &slice) {
 | 
						|
	for (const auto &user : slice.users) {
 | 
						|
		_lastUser = user;
 | 
						|
		delegate()->peerListAppendRow((_role == Role::Requested)
 | 
						|
			? std::make_unique<RequestedRow>(user.user, user.date)
 | 
						|
			: 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());
 | 
						|
}
 | 
						|
 | 
						|
void Controller::rowRightActionClicked(not_null<PeerListRow*> row) {
 | 
						|
	if (_role != Role::Requested) {
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	delegate()->peerListShowRowMenu(row, true);
 | 
						|
}
 | 
						|
 | 
						|
base::unique_qptr<Ui::PopupMenu> Controller::rowContextMenu(
 | 
						|
		QWidget *parent,
 | 
						|
		not_null<PeerListRow*> row) {
 | 
						|
	auto result = createRowContextMenu(parent, row);
 | 
						|
 | 
						|
	if (result) {
 | 
						|
		// First clear _menu value, so that we don't check row positions yet.
 | 
						|
		base::take(_menu);
 | 
						|
 | 
						|
		// Here unique_qptr is used like a shared pointer, where
 | 
						|
		// not the last destroyed pointer destroys the object, but the first.
 | 
						|
		_menu = base::unique_qptr<Ui::PopupMenu>(result.get());
 | 
						|
	}
 | 
						|
 | 
						|
	return result;
 | 
						|
}
 | 
						|
 | 
						|
base::unique_qptr<Ui::PopupMenu> Controller::createRowContextMenu(
 | 
						|
		QWidget *parent,
 | 
						|
		not_null<PeerListRow*> row) {
 | 
						|
	const auto user = row->peer()->asUser();
 | 
						|
	Assert(user != nullptr);
 | 
						|
 | 
						|
	auto result = base::make_unique_q<Ui::PopupMenu>(
 | 
						|
		parent,
 | 
						|
		st::popupMenuWithIcons);
 | 
						|
	const auto add = _peer->isBroadcast()
 | 
						|
		? tr::lng_group_requests_add_channel(tr::now)
 | 
						|
		: tr::lng_group_requests_add(tr::now);
 | 
						|
	result->addAction(add, [=] {
 | 
						|
		processRequest(user, true);
 | 
						|
	}, &st::menuIconInvite);
 | 
						|
	result->addAction(tr::lng_group_requests_dismiss(tr::now), [=] {
 | 
						|
		processRequest(user, false);
 | 
						|
	}, &st::menuIconRemove);
 | 
						|
	return result;
 | 
						|
}
 | 
						|
 | 
						|
void Controller::processRequest(
 | 
						|
		not_null<UserData*> user,
 | 
						|
		bool approved) {
 | 
						|
	const auto done = crl::guard(this, [=] {
 | 
						|
		_processed.fire({ user, approved });
 | 
						|
		if (const auto row = delegate()->peerListFindRow(user->id.value)) {
 | 
						|
			delegate()->peerListRemoveRow(row);
 | 
						|
			delegate()->peerListRefreshRows();
 | 
						|
		}
 | 
						|
		if (approved) {
 | 
						|
			Ui::ShowMultilineToast({
 | 
						|
				.text = (_peer->isBroadcast()
 | 
						|
					? tr::lng_group_requests_was_added_channel
 | 
						|
					: tr::lng_group_requests_was_added)(
 | 
						|
						tr::now,
 | 
						|
						lt_user,
 | 
						|
						Ui::Text::Bold(user->name),
 | 
						|
						Ui::Text::WithEntities)
 | 
						|
			});
 | 
						|
		}
 | 
						|
	});
 | 
						|
	const auto fail = crl::guard(this, [=] {
 | 
						|
		_processed.fire({ user, false });
 | 
						|
	});
 | 
						|
	session().api().inviteLinks().processRequest(
 | 
						|
		_peer,
 | 
						|
		_data.current().link,
 | 
						|
		user,
 | 
						|
		approved,
 | 
						|
		done,
 | 
						|
		fail);
 | 
						|
}
 | 
						|
 | 
						|
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>
 | 
						|
	>();
 | 
						|
	const auto currentLinkFields = container->lifetime().make_state<
 | 
						|
		Api::InviteLink
 | 
						|
	>(Api::InviteLink{ .admin = admin });
 | 
						|
	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();
 | 
						|
			if (link && link->permanent && !link->revoked) {
 | 
						|
				*currentLinkFields = *link;
 | 
						|
				return LinkData{ link->link, link->usage };
 | 
						|
			}
 | 
						|
			return LinkData();
 | 
						|
		});
 | 
						|
	} else {
 | 
						|
		rpl::duplicate(
 | 
						|
			fromList
 | 
						|
		) | rpl::start_with_next([=](const Api::InviteLink &link) {
 | 
						|
			*currentLinkFields = link;
 | 
						|
		}, container->lifetime());
 | 
						|
 | 
						|
		*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 done = crl::guard(weak, [=](Fn<void()> &&close) {
 | 
						|
			peer->session().api().inviteLinks().revokePermanent(
 | 
						|
				peer,
 | 
						|
				admin,
 | 
						|
				value->current().link,
 | 
						|
				std::move(close));
 | 
						|
		});
 | 
						|
		Ui::show(
 | 
						|
			Ui::MakeConfirmBox({
 | 
						|
				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,
 | 
						|
			st::popupMenuWithIcons);
 | 
						|
		result->addAction(
 | 
						|
			tr::lng_group_invite_context_copy(tr::now),
 | 
						|
			copyLink,
 | 
						|
			&st::menuIconCopy);
 | 
						|
		result->addAction(
 | 
						|
			tr::lng_group_invite_context_share(tr::now),
 | 
						|
			shareLink,
 | 
						|
			&st::menuIconShare);
 | 
						|
		result->addAction(
 | 
						|
			tr::lng_group_invite_context_qr(tr::now),
 | 
						|
			getLinkQr,
 | 
						|
			&st::menuIconQrCode);
 | 
						|
		if (!admin->isBot()) {
 | 
						|
			result->addAction(
 | 
						|
				tr::lng_group_invite_context_revoke(tr::now),
 | 
						|
				revokeLink,
 | 
						|
				&st::menuIconRemove);
 | 
						|
		}
 | 
						|
		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([=] {
 | 
						|
		if (!currentLinkFields->link.isEmpty()) {
 | 
						|
			ShowInviteLinkBox(peer, *currentLinkFields);
 | 
						|
		}
 | 
						|
	});
 | 
						|
 | 
						|
	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,
 | 
						|
			Data::ForwardOptions) {
 | 
						|
		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(
 | 
						|
				Ui::MakeInformBox(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 = Api::MessageToSend(Api::SendAction(history, options));
 | 
						|
			message.textWithTags = comment;
 | 
						|
			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.label,
 | 
						|
				result.expireDate,
 | 
						|
				result.usageLimit,
 | 
						|
				result.requestApproval);
 | 
						|
		} else {
 | 
						|
			peer->session().api().inviteLinks().edit(
 | 
						|
				peer,
 | 
						|
				data.admin,
 | 
						|
				result.link,
 | 
						|
				result.label,
 | 
						|
				result.expireDate,
 | 
						|
				result.usageLimit,
 | 
						|
				result.requestApproval,
 | 
						|
				finish);
 | 
						|
		}
 | 
						|
	};
 | 
						|
	const auto isGroup = !peer->isBroadcast();
 | 
						|
	const auto isPublic = peer->isChannel() && peer->asChannel()->isPublic();
 | 
						|
	*box = Ui::show(
 | 
						|
		(creating
 | 
						|
			? Box(Ui::CreateInviteLinkBox, isGroup, isPublic, done)
 | 
						|
			: Box(
 | 
						|
				Ui::EditInviteLinkBox,
 | 
						|
				Fields{
 | 
						|
					.link = data.link,
 | 
						|
					.label = data.label,
 | 
						|
					.expireDate = data.expireDate,
 | 
						|
					.usageLimit = data.usageLimit,
 | 
						|
					.requestApproval = data.requestApproval,
 | 
						|
					.isGroup = isGroup,
 | 
						|
					.isPublic = isPublic,
 | 
						|
				},
 | 
						|
				done)),
 | 
						|
		Ui::LayerOption::KeepOther);
 | 
						|
}
 | 
						|
 | 
						|
void RevokeLink(
 | 
						|
		not_null<PeerData*> peer,
 | 
						|
		not_null<UserData*> admin,
 | 
						|
		const QString &link) {
 | 
						|
	const auto revoke = [=](Fn<void()> &&close) {
 | 
						|
		const auto done = [close = std::move(close)](const LinkData &data) {
 | 
						|
			close();
 | 
						|
		};
 | 
						|
		peer->session().api().inviteLinks().revoke(peer, admin, link, done);
 | 
						|
	};
 | 
						|
	Ui::show(
 | 
						|
		Ui::MakeConfirmBox({
 | 
						|
			tr::lng_group_invite_revoke_about(),
 | 
						|
			revoke
 | 
						|
		}),
 | 
						|
		Ui::LayerOption::KeepOther);
 | 
						|
}
 | 
						|
 | 
						|
void DeleteLink(
 | 
						|
		not_null<PeerData*> peer,
 | 
						|
		not_null<UserData*> admin,
 | 
						|
		const QString &link) {
 | 
						|
	const auto sure = [=](Fn<void()> &&close) {
 | 
						|
		peer->session().api().inviteLinks().destroy(
 | 
						|
			peer,
 | 
						|
			admin,
 | 
						|
			link,
 | 
						|
			std::move(close));
 | 
						|
	};
 | 
						|
	Ui::show(
 | 
						|
		Ui::MakeConfirmBox({
 | 
						|
			tr::lng_group_invite_delete_sure(),
 | 
						|
			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.label.isEmpty()
 | 
						|
				? rpl::single(link.label)
 | 
						|
				: 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),
 | 
						|
				Controller::Role::Joined),
 | 
						|
			std::move(initBox)),
 | 
						|
		Ui::LayerOption::KeepOther);
 | 
						|
}
 | 
						|
 | 
						|
QString PrepareRequestedRowStatus(TimeId date) {
 | 
						|
	const auto now = QDateTime::currentDateTime();
 | 
						|
	const auto parsed = base::unixtime::parse(date);
 | 
						|
	const auto parsedDate = parsed.date();
 | 
						|
	const auto time = parsed.time().toString(cTimeFormat());
 | 
						|
	const auto generic = [&] {
 | 
						|
		return tr::lng_group_requests_status_date_time(
 | 
						|
			tr::now,
 | 
						|
			lt_date,
 | 
						|
			langDayOfMonth(parsedDate),
 | 
						|
			lt_time,
 | 
						|
			time);
 | 
						|
	};
 | 
						|
	return (parsedDate.addDays(1) < now.date())
 | 
						|
		? generic()
 | 
						|
		: (parsedDate.addDays(1) == now.date())
 | 
						|
		? tr::lng_group_requests_status_yesterday(tr::now, lt_time, time)
 | 
						|
		: (now.date() == parsedDate)
 | 
						|
		? tr::lng_group_requests_status_today(tr::now, lt_time, time)
 | 
						|
		: generic();
 | 
						|
}
 |