| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082 |
- /*
- 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_links.h"
- #include "data/data_peer.h"
- #include "data/data_user.h"
- #include "data/data_chat.h"
- #include "data/data_channel.h"
- #include "data/data_session.h"
- #include "main/session/session_show.h"
- #include "main/main_session.h"
- #include "api/api_invite_links.h"
- #include "settings/settings_credits_graphics.h"
- #include "ui/wrap/slide_wrap.h"
- #include "ui/widgets/buttons.h"
- #include "ui/widgets/popup_menu.h"
- #include "ui/painter.h"
- #include "ui/rect.h"
- #include "ui/vertical_list.h"
- #include "lang/lang_keys.h"
- #include "ui/boxes/confirm_box.h"
- #include "boxes/peer_list_controllers.h"
- #include "boxes/peers/edit_peer_invite_link.h"
- #include "apiwrap.h"
- #include "base/weak_ptr.h"
- #include "base/unixtime.h"
- #include "styles/style_info.h"
- #include "styles/style_layers.h" // st::boxDividerLabel
- #include "styles/style_menu_icons.h"
- #include <QtSvg/QSvgRenderer>
- namespace {
- enum class Color {
- Permanent,
- Expiring,
- ExpireSoon,
- Expired,
- Revoked,
- Subscription,
- Count,
- };
- using InviteLinkData = Api::InviteLink;
- using InviteLinksSlice = Api::PeerInviteLinks;
- struct InviteLinkAction {
- enum class Type {
- Copy,
- Share,
- Edit,
- Revoke,
- Delete,
- };
- QString link;
- Type type = Type::Copy;
- };
- class Row;
- using SubscriptionRightLabel = Settings::SubscriptionRightLabel;
- class RowDelegate {
- public:
- virtual std::optional<SubscriptionRightLabel> rightLabel(
- int credits) const = 0;
- virtual void rowUpdateRow(not_null<Row*> row) = 0;
- virtual void rowPaintIcon(
- QPainter &p,
- int x,
- int y,
- int size,
- float64 progress,
- Color color) = 0;
- };
- class Row final : public PeerListRow {
- public:
- Row(
- not_null<RowDelegate*> delegate,
- const InviteLinkData &data,
- TimeId now);
- void update(const InviteLinkData &data, TimeId now);
- void updateExpireProgress(TimeId now);
- [[nodiscard]] InviteLinkData data() const;
- [[nodiscard]] crl::time updateExpireIn() const;
- QString generateName() override;
- QString generateShortName() override;
- PaintRoundImageCallback generatePaintUserpicCallback(
- bool forceRound) override;
- QSize rightActionSize() const override;
- bool rightActionDisabled() const override;
- QMargins rightActionMargins() const override;
- void rightActionPaint(
- Painter &p,
- int x,
- int y,
- int outerWidth,
- bool selected,
- bool actionSelected) override;
- private:
- const not_null<RowDelegate*> _delegate;
- std::optional<SubscriptionRightLabel> _rightLabel;
- InviteLinkData _data;
- QString _status;
- float64 _progressTillExpire = 0.;
- Color _color = Color::Permanent;
- };
- [[nodiscard]] uint64 ComputeRowId(const InviteLinkData &data) {
- return UniqueRowIdFromString(data.link);
- }
- [[nodiscard]] float64 ComputeProgress(
- const InviteLinkData &link,
- TimeId now) {
- const auto startDate = link.startDate ? link.startDate : link.date;
- if (link.expireDate <= startDate && link.usageLimit <= 0) {
- return -1;
- }
- const auto expireProgress = (link.expireDate <= startDate
- || now <= startDate)
- ? 0.
- : (link.expireDate <= now)
- ? 1.
- : (now - startDate) / float64(link.expireDate - startDate);
- const auto usageProgress = (link.usageLimit <= 0 || link.usage <= 0)
- ? 0.
- : (link.usageLimit <= link.usage)
- ? 1.
- : link.usage / float64(link.usageLimit);
- return std::max(expireProgress, usageProgress);
- }
- [[nodiscard]] Color ComputeColor(
- const InviteLinkData &link,
- float64 progress) {
- return link.subscription
- ? Color::Subscription
- : link.revoked
- ? Color::Revoked
- : (progress >= 1.)
- ? Color::Expired
- : (progress >= 3 / 4.)
- ? Color::ExpireSoon
- : (progress >= 0.)
- ? Color::Expiring
- : Color::Permanent;
- }
- [[nodiscard]] QString ComputeStatus(const InviteLinkData &link, TimeId now) {
- const auto expired = IsExpiredLink(link, now);
- const auto revoked = link.revoked;
- auto result = link.usage
- ? tr::lng_group_invite_joined(tr::now, lt_count_decimal, link.usage)
- : (!expired && !revoked && link.usageLimit > 0)
- ? tr::lng_group_invite_can_join(
- tr::now,
- lt_count_decimal,
- link.usageLimit)
- : tr::lng_group_invite_no_joined(tr::now);
- const auto add = [&](const QString &text) {
- result += QString::fromUtf8(" \xE2\x80\xA2 ") + text;
- };
- if (revoked) {
- return result;
- } else if (expired) {
- add(tr::lng_group_invite_link_expired(tr::now));
- return result;
- }
- if (link.usage > 0 && link.usageLimit > link.usage) {
- result += ", " + tr::lng_group_invite_remaining(
- tr::now,
- lt_count_decimal,
- link.usageLimit - link.usage);
- } else if (link.usage > 0 && link.requested > 0) {
- result += ", " + tr::lng_group_invite_requested(
- tr::now,
- lt_count_decimal,
- link.requested);
- } else if (link.requested > 0) {
- result = tr::lng_group_invite_requested_full(
- tr::now,
- lt_count_decimal,
- link.requested);
- }
- if (link.expireDate > now) {
- const auto left = (link.expireDate - now);
- if (left >= 86400) {
- add(tr::lng_group_invite_days_left(
- tr::now,
- lt_count,
- left / 86400));
- } else {
- const auto time = base::unixtime::parse(link.expireDate).time();
- add(QLocale().toString(time, QLocale::LongFormat));
- }
- }
- return result;
- }
- object_ptr<Ui::BoxContent> DeleteAllRevokedBox(
- not_null<PeerData*> peer,
- not_null<UserData*> admin) {
- const auto sure = [=](Fn<void()> &&close) {
- peer->session().api().inviteLinks().destroyAllRevoked(
- peer,
- admin,
- std::move(close));
- };
- return Ui::MakeConfirmBox({
- tr::lng_group_invite_delete_all_sure(),
- sure
- });
- }
- [[nodiscard]] not_null<Ui::SettingsButton*> AddCreateLinkButton(
- not_null<Ui::VerticalLayout*> container) {
- return container->add(
- MakeCreateLinkButton(container, tr::lng_group_invite_add()),
- style::margins(0, st::inviteLinkCreateSkip, 0, 0));
- }
- Row::Row(
- not_null<RowDelegate*> delegate,
- const InviteLinkData &data,
- TimeId now)
- : PeerListRow(ComputeRowId(data))
- , _delegate(delegate)
- , _data(data)
- , _progressTillExpire(ComputeProgress(data, now))
- , _color(ComputeColor(data, _progressTillExpire)) {
- _rightLabel = _delegate->rightLabel(_data.subscription.credits);
- setCustomStatus(ComputeStatus(data, now));
- }
- void Row::update(const InviteLinkData &data, TimeId now) {
- _data = data;
- _rightLabel = _delegate->rightLabel(_data.subscription.credits);
- _progressTillExpire = ComputeProgress(data, now);
- _color = ComputeColor(data, _progressTillExpire);
- setCustomStatus(ComputeStatus(data, now));
- refreshName(st::inviteLinkList.item);
- _delegate->rowUpdateRow(this);
- }
- void Row::updateExpireProgress(TimeId now) {
- const auto updated = ComputeProgress(_data, now);
- if (base::SafeRound(_progressTillExpire * 360)
- != base::SafeRound(updated * 360)) {
- _progressTillExpire = updated;
- const auto color = ComputeColor(_data, _progressTillExpire);
- if (_color != color) {
- _color = color;
- setCustomStatus(ComputeStatus(_data, now));
- }
- _delegate->rowUpdateRow(this);
- }
- }
- InviteLinkData Row::data() const {
- return _data;
- }
- crl::time Row::updateExpireIn() const {
- if (_color != Color::Expiring && _color != Color::ExpireSoon) {
- return 0;
- }
- const auto start = _data.startDate ? _data.startDate : _data.date;
- if (_data.expireDate <= start) {
- return 0;
- }
- return base::SafeRound(
- (_data.expireDate - start) * crl::time(1000) / 720.);
- }
- QString Row::generateName() {
- if (!_data.label.isEmpty()) {
- return _data.label;
- }
- auto result = _data.link;
- return result.replace(
- u"https://"_q,
- QString()
- ).replace(
- u"t.me/+"_q,
- QString()
- ).replace(
- u"t.me/joinchat/"_q,
- QString()
- );
- }
- QString Row::generateShortName() {
- return generateName();
- }
- PaintRoundImageCallback Row::generatePaintUserpicCallback(bool forceRound) {
- return [=](
- QPainter &p,
- int x,
- int y,
- int outerWidth,
- int size) {
- _delegate->rowPaintIcon(p, x, y, size, _progressTillExpire, _color);
- };
- }
- QSize Row::rightActionSize() const {
- if (_rightLabel) {
- return _rightLabel->size;
- }
- return QSize(
- st::inviteLinkThreeDotsIcon.width(),
- st::inviteLinkThreeDotsIcon.height());
- }
- bool Row::rightActionDisabled() const {
- return _rightLabel.has_value();
- }
- QMargins Row::rightActionMargins() const {
- if (_rightLabel) {
- return QMargins(0, 0, st::boxRowPadding.right(), 0);
- }
- return QMargins(
- 0,
- (st::inviteLinkList.item.height - rightActionSize().height()) / 2,
- st::inviteLinkThreeDotsSkip,
- 0);
- }
- void Row::rightActionPaint(
- Painter &p,
- int x,
- int y,
- int outerWidth,
- bool selected,
- bool actionSelected) {
- if (_rightLabel) {
- return _rightLabel->draw(p, x, y, st::inviteLinkList.item.height);
- }
- (actionSelected
- ? st::inviteLinkThreeDotsIconOver
- : st::inviteLinkThreeDotsIcon).paint(p, x, y, outerWidth);
- }
- class LinksController final
- : public PeerListController
- , public RowDelegate
- , public base::has_weak_ptr {
- public:
- LinksController(
- not_null<PeerData*> peer,
- not_null<UserData*> admin,
- int count,
- bool revoked);
- [[nodiscard]] rpl::producer<int> fullCountValue() const {
- return _count.value();
- }
- void prepare() override;
- void loadMoreRows() override;
- void rowClicked(not_null<PeerListRow*> row) override;
- void rowRightActionClicked(not_null<PeerListRow*> row) override;
- base::unique_qptr<Ui::PopupMenu> rowContextMenu(
- QWidget *parent,
- not_null<PeerListRow*> row) override;
- Main::Session &session() const override;
- std::optional<SubscriptionRightLabel> rightLabel(int) const override;
- void rowUpdateRow(not_null<Row*> row) override;
- void rowPaintIcon(
- QPainter &p,
- int x,
- int y,
- int size,
- float64 progress,
- Color color) override;
- [[nodiscard]] rpl::producer<InviteLinkData> permanentFound() const {
- return _permanentFound.events();
- }
- private:
- void appendRow(const InviteLinkData &data, TimeId now);
- void prependRow(const InviteLinkData &data, TimeId now);
- void updateRow(const InviteLinkData &data, TimeId now);
- bool removeRow(const QString &link);
- void appendSlice(const InviteLinksSlice &slice);
- void checkExpiringTimer(not_null<Row*> row);
- void expiringProgressTimer();
- [[nodiscard]] base::unique_qptr<Ui::PopupMenu> createRowContextMenu(
- QWidget *parent,
- not_null<PeerListRow*> row);
- const not_null<PeerData*> _peer;
- const not_null<UserData*> _admin;
- const bool _revoked = false;
- rpl::variable<int> _count;
- base::unique_qptr<Ui::PopupMenu> _menu;
- QString _offsetLink;
- TimeId _offsetDate = 0;
- bool _requesting = false;
- bool _allLoaded = false;
- rpl::event_stream<InviteLinkData> _permanentFound;
- base::flat_set<not_null<Row*>> _expiringRows;
- base::Timer _updateExpiringTimer;
- std::array<QImage, int(Color::Count)> _icons;
- rpl::lifetime _lifetime;
- };
- LinksController::LinksController(
- not_null<PeerData*> peer,
- not_null<UserData*> admin,
- int count,
- bool revoked)
- : _peer(peer)
- , _admin(admin)
- , _revoked(revoked)
- , _count(count)
- , _updateExpiringTimer([=] { expiringProgressTimer(); }) {
- style::PaletteChanged(
- ) | rpl::start_with_next([=] {
- for (auto &image : _icons) {
- image = QImage();
- }
- }, _lifetime);
- peer->session().api().inviteLinks().updates(
- peer,
- admin
- ) | rpl::start_with_next([=](const Api::InviteLinkUpdate &update) {
- const auto now = base::unixtime::now();
- if (!update.now || update.now->revoked != _revoked) {
- if (removeRow(update.was)) {
- delegate()->peerListRefreshRows();
- }
- } else if (update.was.isEmpty()) {
- if (update.now->permanent && !update.now->revoked) {
- _permanentFound.fire_copy(*update.now);
- } else {
- prependRow(*update.now, now);
- delegate()->peerListRefreshRows();
- }
- } else {
- updateRow(*update.now, now);
- }
- }, _lifetime);
- if (_revoked) {
- peer->session().api().inviteLinks().allRevokedDestroyed(
- peer,
- admin
- ) | rpl::start_with_next([=] {
- _requesting = false;
- _allLoaded = true;
- while (delegate()->peerListFullRowsCount()) {
- delegate()->peerListRemoveRow(delegate()->peerListRowAt(0));
- }
- delegate()->peerListRefreshRows();
- }, _lifetime);
- }
- }
- void LinksController::prepare() {
- if (!_revoked && _admin->isSelf()) {
- appendSlice(_peer->session().api().inviteLinks().myLinks(_peer));
- }
- if (!delegate()->peerListFullRowsCount()) {
- loadMoreRows();
- }
- }
- void LinksController::loadMoreRows() {
- if (_requesting || _allLoaded) {
- return;
- }
- _requesting = true;
- const auto done = [=](const InviteLinksSlice &slice) {
- if (!_requesting) {
- return;
- }
- _requesting = false;
- if (slice.links.empty()) {
- _allLoaded = true;
- return;
- }
- appendSlice(slice);
- };
- _peer->session().api().inviteLinks().requestMoreLinks(
- _peer,
- _admin,
- _offsetDate,
- _offsetLink,
- _revoked,
- crl::guard(this, done));
- }
- void LinksController::appendSlice(const InviteLinksSlice &slice) {
- const auto now = base::unixtime::now();
- for (const auto &link : slice.links) {
- if (link.permanent && !link.revoked) {
- _permanentFound.fire_copy(link);
- } else {
- appendRow(link, now);
- }
- _offsetLink = link.link;
- _offsetDate = link.date;
- }
- if (slice.links.size() >= slice.count) {
- _allLoaded = true;
- }
- const auto rowsCount = delegate()->peerListFullRowsCount();
- const auto minimalCount = _revoked ? rowsCount : (rowsCount + 1);
- _count = _allLoaded ? minimalCount : std::max(slice.count, minimalCount);
- delegate()->peerListRefreshRows();
- }
- void LinksController::rowClicked(not_null<PeerListRow*> row) {
- delegate()->peerListUiShow()->showBox(
- ShowInviteLinkBox(_peer, static_cast<Row*>(row.get())->data()));
- }
- void LinksController::rowRightActionClicked(not_null<PeerListRow*> row) {
- delegate()->peerListShowRowMenu(row, true);
- }
- base::unique_qptr<Ui::PopupMenu> LinksController::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> LinksController::createRowContextMenu(
- QWidget *parent,
- not_null<PeerListRow*> row) {
- const auto real = static_cast<Row*>(row.get());
- const auto data = real->data();
- const auto link = data.link;
- auto result = base::make_unique_q<Ui::PopupMenu>(
- parent,
- st::popupMenuWithIcons);
- if (data.revoked) {
- result->addAction(tr::lng_group_invite_context_delete(tr::now), [=] {
- delegate()->peerListUiShow()->showBox(
- DeleteLinkBox(_peer, _admin, link));
- }, &st::menuIconDelete);
- } else {
- result->addAction(tr::lng_group_invite_context_copy(tr::now), [=] {
- CopyInviteLink(delegate()->peerListUiShow(), link);
- }, &st::menuIconCopy);
- result->addAction(tr::lng_group_invite_context_share(tr::now), [=] {
- delegate()->peerListUiShow()->showBox(
- ShareInviteLinkBox(_peer, link));
- }, &st::menuIconShare);
- result->addAction(tr::lng_group_invite_context_qr(tr::now), [=] {
- delegate()->peerListUiShow()->showBox(InviteLinkQrBox(
- nullptr,
- link,
- tr::lng_group_invite_qr_title(),
- tr::lng_group_invite_qr_about()));
- }, &st::menuIconQrCode);
- result->addAction(tr::lng_group_invite_context_edit(tr::now), [=] {
- delegate()->peerListUiShow()->showBox(EditLinkBox(_peer, data));
- }, &st::menuIconEdit);
- result->addAction(tr::lng_group_invite_context_revoke(tr::now), [=] {
- delegate()->peerListUiShow()->showBox(
- RevokeLinkBox(_peer, _admin, link));
- }, &st::menuIconRemove);
- }
- return result;
- }
- Main::Session &LinksController::session() const {
- return _peer->session();
- }
- void LinksController::appendRow(const InviteLinkData &data, TimeId now) {
- delegate()->peerListAppendRow(std::make_unique<Row>(this, data, now));
- }
- void LinksController::prependRow(const InviteLinkData &data, TimeId now) {
- delegate()->peerListPrependRow(std::make_unique<Row>(this, data, now));
- }
- void LinksController::updateRow(const InviteLinkData &data, TimeId now) {
- if (const auto row = delegate()->peerListFindRow(ComputeRowId(data))) {
- const auto real = static_cast<Row*>(row);
- real->update(data, now);
- checkExpiringTimer(real);
- delegate()->peerListUpdateRow(row);
- } else if (_revoked) {
- prependRow(data, now);
- delegate()->peerListRefreshRows();
- }
- }
- bool LinksController::removeRow(const QString &link) {
- const auto id = UniqueRowIdFromString(link);
- if (const auto row = delegate()->peerListFindRow(id)) {
- delegate()->peerListRemoveRow(row);
- return true;
- }
- return false;
- }
- void LinksController::checkExpiringTimer(not_null<Row*> row) {
- const auto updateIn = row->updateExpireIn();
- if (updateIn > 0) {
- _expiringRows.emplace(row);
- if (!_updateExpiringTimer.isActive()
- || updateIn < _updateExpiringTimer.remainingTime()) {
- _updateExpiringTimer.callOnce(updateIn);
- }
- } else {
- _expiringRows.remove(row);
- }
- }
- void LinksController::expiringProgressTimer() {
- const auto now = base::unixtime::now();
- auto minimalIn = 0;
- for (auto i = begin(_expiringRows); i != end(_expiringRows);) {
- (*i)->updateExpireProgress(now);
- const auto updateIn = (*i)->updateExpireIn();
- if (!updateIn) {
- i = _expiringRows.erase(i);
- } else {
- ++i;
- if (!minimalIn || minimalIn > updateIn) {
- minimalIn = updateIn;
- }
- }
- }
- if (minimalIn) {
- _updateExpiringTimer.callOnce(minimalIn);
- }
- }
- std::optional<SubscriptionRightLabel> LinksController::rightLabel(
- int credits) const {
- if (credits > 0) {
- return Settings::PaintSubscriptionRightLabelCallback(
- &session(),
- st::inviteLinkList.item,
- credits);
- }
- return std::nullopt;
- }
- void LinksController::rowUpdateRow(not_null<Row*> row) {
- delegate()->peerListUpdateRow(row);
- }
- void LinksController::rowPaintIcon(
- QPainter &p,
- int x,
- int y,
- int size,
- float64 progress,
- Color color) {
- const auto skip = st::inviteLinkIconSkip;
- const auto inner = size - 2 * skip;
- const auto bg = [&] {
- switch (color) {
- case Color::Permanent: return &st::msgFile1Bg;
- case Color::Expiring: return &st::msgFile2Bg;
- case Color::ExpireSoon: return &st::msgFile4Bg;
- case Color::Expired: return &st::msgFile3Bg;
- case Color::Revoked: return &st::windowSubTextFg;
- case Color::Subscription: return &st::msgFile2Bg;
- }
- Unexpected("Color in LinksController::rowPaintIcon.");
- }();
- const auto stroke = st::inviteLinkIconStroke;
- auto &icon = _icons[int(color)];
- if (icon.isNull()) {
- icon = QImage(
- QSize(inner, inner) * style::DevicePixelRatio(),
- QImage::Format_ARGB32_Premultiplied);
- icon.fill(Qt::transparent);
- icon.setDevicePixelRatio(style::DevicePixelRatio());
- auto p = QPainter(&icon);
- p.setPen(Qt::NoPen);
- p.setBrush(*bg);
- {
- auto hq = PainterHighQualityEnabler(p);
- const auto rect = QRect(0, 0, inner, inner)
- - ((color == Color::Expiring || color == Color::ExpireSoon)
- ? Margins(stroke)
- : Margins(0));
- p.drawEllipse(rect);
- }
- if (color == Color::Subscription) {
- auto svg = QSvgRenderer(u":/gui/links_subscription.svg"_q);
- const auto r = QRect(
- (inner - st::inviteLinkSubscriptionSize) / 2,
- (inner - st::inviteLinkSubscriptionSize) / 2,
- st::inviteLinkSubscriptionSize,
- st::inviteLinkSubscriptionSize);
- svg.render(&p, r);
- } else {
- (color == Color::Revoked
- ? st::inviteLinkRevokedIcon
- : st::inviteLinkIcon).paintInCenter(p, Rect(Size(inner)));
- }
- }
- p.drawImage(x + skip, y + skip, icon);
- if (progress >= 0. && progress < 1.) {
- auto hq = PainterHighQualityEnabler(p);
- auto pen = QPen((*bg)->c);
- pen.setWidth(stroke);
- pen.setCapStyle(Qt::RoundCap);
- p.setPen(pen);
- p.setBrush(Qt::NoBrush);
- const auto margins = .5 * stroke;
- p.drawArc(QRectF(x + skip, y + skip, inner, inner).marginsAdded({
- margins,
- margins,
- margins,
- margins,
- }), arc::kQuarterLength, arc::kFullLength * (1. - progress));
- }
- }
- class AdminsController final
- : public PeerListController
- , public base::has_weak_ptr {
- public:
- AdminsController(not_null<PeerData*> peer, not_null<UserData*> admin);
- ~AdminsController();
- void prepare() override;
- void loadMoreRows() override;
- void rowClicked(not_null<PeerListRow*> row) override;
- Main::Session &session() const override;
- private:
- void appendRow(not_null<UserData*> user, int count);
- const not_null<PeerData*> _peer;
- const not_null<UserData*> _admin;
- mtpRequestId _requestId = 0;
- };
- AdminsController::AdminsController(
- not_null<PeerData*> peer,
- not_null<UserData*> admin)
- : _peer(peer)
- , _admin(admin) {
- }
- AdminsController::~AdminsController() {
- session().api().request(base::take(_requestId)).cancel();
- }
- void AdminsController::prepare() {
- if (const auto chat = _peer->asChat()) {
- if (!chat->amCreator()) {
- return;
- }
- } else if (const auto channel = _peer->asChannel()) {
- if (!channel->amCreator()) {
- return;
- }
- }
- if (!_admin->isSelf()) {
- return;
- }
- _requestId = session().api().request(MTPmessages_GetAdminsWithInvites(
- _peer->input
- )).done([=](const MTPmessages_ChatAdminsWithInvites &result) {
- result.match([&](const MTPDmessages_chatAdminsWithInvites &data) {
- auto &owner = _peer->owner();
- owner.processUsers(data.vusers());
- for (const auto &admin : data.vadmins().v) {
- admin.match([&](const MTPDchatAdminWithInvites &data) {
- const auto adminId = data.vadmin_id();
- if (const auto user = owner.userLoaded(adminId)) {
- if (!user->isSelf()) {
- appendRow(user, data.vinvites_count().v);
- }
- }
- });
- }
- delegate()->peerListRefreshRows();
- });
- }).send();
- }
- void AdminsController::loadMoreRows() {
- }
- void AdminsController::rowClicked(not_null<PeerListRow*> row) {
- delegate()->peerListUiShow()->showBox(
- Box(ManageInviteLinksBox, _peer, row->peer()->asUser(), 0, 0));
- }
- Main::Session &AdminsController::session() const {
- return _peer->session();
- }
- void AdminsController::appendRow(not_null<UserData*> user, int count) {
- auto row = std::make_unique<PeerListRow>(user);
- row->setCustomStatus(
- tr::lng_group_invite_other_count(tr::now, lt_count, count));
- delegate()->peerListAppendRow(std::move(row));
- }
- } // namespace
- struct LinksList {
- not_null<Ui::RpWidget*> widget;
- not_null<LinksController*> controller;
- };
- LinksList AddLinksList(
- std::shared_ptr<Main::SessionShow> show,
- not_null<Ui::VerticalLayout*> container,
- not_null<PeerData*> peer,
- not_null<UserData*> admin,
- int count,
- bool revoked) {
- auto &lifetime = container->lifetime();
- const auto delegate = lifetime.make_state<PeerListContentDelegateShow>(
- show);
- const auto controller = lifetime.make_state<LinksController>(
- peer,
- admin,
- count,
- revoked);
- controller->setStyleOverrides(&st::inviteLinkList);
- const auto content = container->add(object_ptr<PeerListContent>(
- container,
- controller));
- delegate->setContent(content);
- controller->setDelegate(delegate);
- return { content, controller };
- }
- not_null<Ui::RpWidget*> AddAdminsList(
- std::shared_ptr<Main::SessionShow> show,
- not_null<Ui::VerticalLayout*> container,
- not_null<PeerData*> peer,
- not_null<UserData*> admin) {
- auto &lifetime = container->lifetime();
- const auto delegate = lifetime.make_state<PeerListContentDelegateShow>(
- show);
- const auto controller = lifetime.make_state<AdminsController>(
- peer,
- admin);
- controller->setStyleOverrides(&st::inviteLinkAdminsList);
- const auto content = container->add(object_ptr<PeerListContent>(
- container,
- controller));
- delegate->setContent(content);
- controller->setDelegate(delegate);
- return content;
- }
- void ManageInviteLinksBox(
- not_null<Ui::GenericBox*> box,
- not_null<PeerData*> peer,
- not_null<UserData*> admin,
- int count,
- int revokedCount) {
- const auto show = Main::MakeSessionShow(
- box->uiShow(),
- &peer->session());
- box->setTitle(tr::lng_group_invite_title());
- box->setWidth(st::boxWideWidth);
- const auto container = box->verticalLayout();
- const auto permanentFromList = box->lifetime().make_state<
- rpl::event_stream<InviteLinkData>
- >();
- const auto countValue = box->lifetime().make_state<rpl::variable<int>>(
- count);
- if (!admin->isSelf()) {
- auto status = tr::lng_group_invite_links_count(
- lt_count,
- countValue->value() | tr::to_count());
- AddSinglePeerRow(
- container,
- admin,
- std::move(status));
- }
- Ui::AddSubsectionTitle(container, tr::lng_create_permanent_link_title());
- AddPermanentLinkBlock(
- show,
- container,
- peer,
- admin,
- permanentFromList->events());
- Ui::AddDivider(container);
- auto otherHeader = (Ui::SlideWrap<>*)nullptr;
- if (admin->isSelf()) {
- const auto add = AddCreateLinkButton(container);
- add->setClickedCallback([=] {
- show->showBox(
- EditLinkBox(peer, InviteLinkData{ .admin = admin }));
- });
- } else {
- otherHeader = container->add(object_ptr<Ui::SlideWrap<>>(
- container,
- object_ptr<Ui::FlatLabel>(
- container,
- tr::lng_group_invite_other_list(),
- st::defaultSubsectionTitle),
- st::inviteLinkRevokedTitlePadding));
- }
- auto [list, controller] = AddLinksList(
- show,
- container,
- peer,
- admin,
- count,
- false);
- *countValue = controller->fullCountValue();
- controller->permanentFound(
- ) | rpl::start_with_next([=](InviteLinkData &&data) {
- permanentFromList->fire(std::move(data));
- }, container->lifetime());
- const auto dividerAbout = container->add(object_ptr<Ui::SlideWrap<>>(
- container,
- object_ptr<Ui::DividerLabel>(
- container,
- object_ptr<Ui::FlatLabel>(
- container,
- tr::lng_group_invite_add_about(),
- st::boxDividerLabel),
- st::defaultBoxDividerLabelPadding)),
- style::margins(0, st::inviteLinkCreateSkip, 0, 0));
- const auto adminsDivider = container->add(object_ptr<Ui::SlideWrap<>>(
- container,
- object_ptr<Ui::BoxContentDivider>(container)));
- const auto adminsHeader = container->add(object_ptr<Ui::SlideWrap<>>(
- container,
- object_ptr<Ui::FlatLabel>(
- container,
- tr::lng_group_invite_other_title(),
- st::defaultSubsectionTitle),
- st::inviteLinkRevokedTitlePadding));
- const auto admins = AddAdminsList(show, container, peer, admin);
- const auto revokedDivider = container->add(object_ptr<Ui::SlideWrap<>>(
- container,
- object_ptr<Ui::BoxContentDivider>(container)));
- const auto revokedHeader = container->add(object_ptr<Ui::SlideWrap<>>(
- container,
- object_ptr<Ui::FlatLabel>(
- container,
- tr::lng_group_invite_revoked_title(),
- st::defaultSubsectionTitle),
- st::inviteLinkRevokedTitlePadding));
- const auto revoked = AddLinksList(
- show,
- container,
- peer,
- admin,
- revokedCount,
- true).widget;
- const auto deleteAll = Ui::CreateChild<Ui::LinkButton>(
- container.get(),
- tr::lng_group_invite_context_delete_all(tr::now),
- st::defaultLinkButton);
- rpl::combine(
- revokedHeader->topValue(),
- container->widthValue()
- ) | rpl::start_with_next([=](int top, int outerWidth) {
- deleteAll->moveToRight(
- st::inviteLinkRevokedTitlePadding.left(),
- top + st::inviteLinkRevokedTitlePadding.top(),
- outerWidth);
- }, deleteAll->lifetime());
- deleteAll->setClickedCallback([=, show = box->uiShow()] {
- show->showBox(DeleteAllRevokedBox(peer, admin));
- });
- rpl::combine(
- list->heightValue(),
- admins->heightValue(),
- revoked->heightValue()
- ) | rpl::start_with_next([=](int list, int admins, int revoked) {
- if (otherHeader) {
- otherHeader->toggle(list > 0, anim::type::instant);
- }
- dividerAbout->toggle(!list && !otherHeader, anim::type::instant);
- adminsDivider->toggle(admins > 0 && list > 0, anim::type::instant);
- adminsHeader->toggle(admins > 0, anim::type::instant);
- revokedDivider->toggle(revoked > 0 && (list > 0 || admins > 0), anim::type::instant);
- revokedHeader->toggle(revoked > 0, anim::type::instant);
- deleteAll->setVisible(revoked > 0);
- }, revokedHeader->lifetime());
- box->addButton(tr::lng_about_done(), [=] { box->closeBox(); });
- }
- object_ptr<Ui::SettingsButton> MakeCreateLinkButton(
- not_null<QWidget*> parent,
- rpl::producer<QString> text) {
- auto result = object_ptr<Ui::SettingsButton>(
- parent,
- std::move(text),
- st::inviteLinkCreate);
- const auto raw = result.data();
- const auto icon = Ui::CreateChild<Ui::RpWidget>(raw);
- icon->setAttribute(Qt::WA_TransparentForMouseEvents);
- const auto size = st::inviteLinkCreateIconSize;
- icon->resize(size, size);
- raw->heightValue(
- ) | rpl::start_with_next([=](int height) {
- const auto &st = st::inviteLinkList.item;
- icon->move(
- st.photoPosition.x() + (st.photoSize - size) / 2,
- (height - size) / 2);
- }, icon->lifetime());
- icon->paintRequest(
- ) | rpl::start_with_next([=] {
- auto p = QPainter(icon);
- p.setPen(Qt::NoPen);
- p.setBrush(st::windowBgActive);
- const auto rect = icon->rect();
- {
- auto hq = PainterHighQualityEnabler(p);
- p.drawEllipse(rect);
- }
- st::inviteLinkCreateIcon.paintInCenter(p, rect);
- }, icon->lifetime());
- return result;
- }
|