mirror of
https://github.com/AyuGram/AyuGramDesktop.git
synced 2025-06-05 06:33:57 +02:00
Show invite links list with context menu.
This commit is contained in:
parent
40e90af76d
commit
01ecf0ca93
10 changed files with 600 additions and 54 deletions
|
@ -1184,6 +1184,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
"lng_group_invite_about_permanent_channel" = "Anyone who has Telegram installed will be able to join your channel by following this link.";
|
"lng_group_invite_about_permanent_channel" = "Anyone who has Telegram installed will be able to join your channel by following this link.";
|
||||||
"lng_group_invite_manage" = "Manage Invite Links";
|
"lng_group_invite_manage" = "Manage Invite Links";
|
||||||
"lng_group_invite_manage_about" = "You can create additional invite links that have limited time or number of usages.";
|
"lng_group_invite_manage_about" = "You can create additional invite links that have limited time or number of usages.";
|
||||||
|
"lng_group_invite_title" = "Invite links";
|
||||||
"lng_group_invite_add" = "Create a New Link";
|
"lng_group_invite_add" = "Create a New Link";
|
||||||
"lng_group_invite_expires_at" = "This link expires {when}.";
|
"lng_group_invite_expires_at" = "This link expires {when}.";
|
||||||
"lng_group_invite_created_by" = "Link created by";
|
"lng_group_invite_created_by" = "Link created by";
|
||||||
|
|
|
@ -218,6 +218,12 @@ void InviteLinks::revoke(
|
||||||
performEdit(peer, link, std::move(done), true);
|
performEdit(peer, link, std::move(done), true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void InviteLinks::revokePermanent(
|
||||||
|
not_null<PeerData*> peer,
|
||||||
|
Fn<void(Link)> done) {
|
||||||
|
performCreate(peer, std::move(done), true);
|
||||||
|
}
|
||||||
|
|
||||||
void InviteLinks::requestLinks(not_null<PeerData*> peer) {
|
void InviteLinks::requestLinks(not_null<PeerData*> peer) {
|
||||||
if (_firstSliceRequests.contains(peer)) {
|
if (_firstSliceRequests.contains(peer)) {
|
||||||
return;
|
return;
|
||||||
|
@ -411,6 +417,7 @@ auto InviteLinks::parse(
|
||||||
.link = qs(data.vlink()),
|
.link = qs(data.vlink()),
|
||||||
.admin = peer->session().data().user(data.vadmin_id().v),
|
.admin = peer->session().data().user(data.vadmin_id().v),
|
||||||
.date = data.vdate().v,
|
.date = data.vdate().v,
|
||||||
|
.startDate = data.vstart_date().value_or_empty(),
|
||||||
.expireDate = data.vexpire_date().value_or_empty(),
|
.expireDate = data.vexpire_date().value_or_empty(),
|
||||||
.usageLimit = data.vusage_limit().value_or_empty(),
|
.usageLimit = data.vusage_limit().value_or_empty(),
|
||||||
.usage = data.vusage().value_or_empty(),
|
.usage = data.vusage().value_or_empty(),
|
||||||
|
|
|
@ -14,7 +14,8 @@ namespace Api {
|
||||||
struct InviteLink {
|
struct InviteLink {
|
||||||
QString link;
|
QString link;
|
||||||
not_null<UserData*> admin;
|
not_null<UserData*> admin;
|
||||||
TimeId date;
|
TimeId date = 0;
|
||||||
|
TimeId startDate = 0;
|
||||||
TimeId expireDate = 0;
|
TimeId expireDate = 0;
|
||||||
int usageLimit = 0;
|
int usageLimit = 0;
|
||||||
int usage = 0;
|
int usage = 0;
|
||||||
|
@ -60,6 +61,9 @@ public:
|
||||||
not_null<PeerData*> peer,
|
not_null<PeerData*> peer,
|
||||||
const QString &link,
|
const QString &link,
|
||||||
Fn<void(Link)> done = nullptr);
|
Fn<void(Link)> done = nullptr);
|
||||||
|
void revokePermanent(
|
||||||
|
not_null<PeerData*> peer,
|
||||||
|
Fn<void(Link)> done = nullptr);
|
||||||
|
|
||||||
void setPermanent(
|
void setPermanent(
|
||||||
not_null<PeerData*> peer,
|
not_null<PeerData*> peer,
|
||||||
|
|
|
@ -194,39 +194,6 @@ PaintRoundImageCallback ExceptionRow::generatePaintUserpicCallback() {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeDelegate::peerListSetTitle(rpl::producer<QString> title) {
|
|
||||||
}
|
|
||||||
|
|
||||||
void TypeDelegate::peerListSetAdditionalTitle(rpl::producer<QString> title) {
|
|
||||||
}
|
|
||||||
|
|
||||||
bool TypeDelegate::peerListIsRowChecked(not_null<PeerListRow*> row) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
int TypeDelegate::peerListSelectedRowsCount() {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
void TypeDelegate::peerListScrollToTop() {
|
|
||||||
}
|
|
||||||
|
|
||||||
void TypeDelegate::peerListAddSelectedPeerInBunch(not_null<PeerData*> peer) {
|
|
||||||
Unexpected("Item selection in Info::Profile::Members.");
|
|
||||||
}
|
|
||||||
|
|
||||||
void TypeDelegate::peerListAddSelectedRowInBunch(not_null<PeerListRow*> row) {
|
|
||||||
Unexpected("Item selection in Info::Profile::Members.");
|
|
||||||
}
|
|
||||||
|
|
||||||
void TypeDelegate::peerListFinishSelectedRowsBunch() {
|
|
||||||
}
|
|
||||||
|
|
||||||
void TypeDelegate::peerListSetDescription(
|
|
||||||
object_ptr<Ui::FlatLabel> description) {
|
|
||||||
description.destroy();
|
|
||||||
}
|
|
||||||
|
|
||||||
TypeController::TypeController(
|
TypeController::TypeController(
|
||||||
not_null<Main::Session*> session,
|
not_null<Main::Session*> session,
|
||||||
Flags options,
|
Flags options,
|
||||||
|
@ -412,7 +379,9 @@ object_ptr<Ui::RpWidget> EditFilterChatsListController::prepareTypesList() {
|
||||||
container->add(object_ptr<Ui::FixedHeightWidget>(
|
container->add(object_ptr<Ui::FixedHeightWidget>(
|
||||||
container,
|
container,
|
||||||
st::membersMarginTop));
|
st::membersMarginTop));
|
||||||
const auto delegate = container->lifetime().make_state<TypeDelegate>();
|
const auto delegate = container->lifetime().make_state<
|
||||||
|
PeerListContentDelegateSimple
|
||||||
|
>();
|
||||||
const auto controller = container->lifetime().make_state<TypeController>(
|
const auto controller = container->lifetime().make_state<TypeController>(
|
||||||
&session(),
|
&session(),
|
||||||
_options,
|
_options,
|
||||||
|
|
|
@ -302,7 +302,7 @@ public:
|
||||||
|
|
||||||
virtual void peerListShowRowMenu(
|
virtual void peerListShowRowMenu(
|
||||||
not_null<PeerListRow*> row,
|
not_null<PeerListRow*> row,
|
||||||
Fn<void(not_null<Ui::PopupMenu*>)> destroyed) = 0;
|
Fn<void(not_null<Ui::PopupMenu*>)> destroyed = nullptr) = 0;
|
||||||
virtual int peerListSelectedRowsCount() = 0;
|
virtual int peerListSelectedRowsCount() = 0;
|
||||||
virtual std::unique_ptr<PeerListState> peerListSaveState() const = 0;
|
virtual std::unique_ptr<PeerListState> peerListSaveState() const = 0;
|
||||||
virtual void peerListRestoreState(
|
virtual void peerListRestoreState(
|
||||||
|
@ -837,7 +837,7 @@ public:
|
||||||
}
|
}
|
||||||
void peerListShowRowMenu(
|
void peerListShowRowMenu(
|
||||||
not_null<PeerListRow*> row,
|
not_null<PeerListRow*> row,
|
||||||
Fn<void(not_null<Ui::PopupMenu*>)> destroyed) override {
|
Fn<void(not_null<Ui::PopupMenu*>)> destroyed = nullptr) override {
|
||||||
_content->showRowMenu(row, std::move(destroyed));
|
_content->showRowMenu(row, std::move(destroyed));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -851,6 +851,38 @@ private:
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class PeerListContentDelegateSimple : public PeerListContentDelegate {
|
||||||
|
public:
|
||||||
|
void peerListSetTitle(rpl::producer<QString> title) override {
|
||||||
|
}
|
||||||
|
void peerListSetAdditionalTitle(rpl::producer<QString> title) override {
|
||||||
|
}
|
||||||
|
bool peerListIsRowChecked(not_null<PeerListRow*> row) override {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
int peerListSelectedRowsCount() override {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
void peerListScrollToTop() override {
|
||||||
|
}
|
||||||
|
void peerListAddSelectedPeerInBunch(
|
||||||
|
not_null<PeerData*> peer) override {
|
||||||
|
Unexpected("...DelegateSimple::peerListAddSelectedPeerInBunch");
|
||||||
|
}
|
||||||
|
void peerListAddSelectedRowInBunch(not_null<PeerListRow*> row) override {
|
||||||
|
Unexpected("...DelegateSimple::peerListAddSelectedRowInBunch");
|
||||||
|
}
|
||||||
|
void peerListFinishSelectedRowsBunch() override {
|
||||||
|
Unexpected("...DelegateSimple::peerListFinishSelectedRowsBunch");
|
||||||
|
}
|
||||||
|
void peerListSetDescription(
|
||||||
|
object_ptr<Ui::FlatLabel> description) override {
|
||||||
|
description.destroy();
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
class PeerListBox
|
class PeerListBox
|
||||||
: public Ui::BoxContent
|
: public Ui::BoxContent
|
||||||
, public PeerListContentDelegate {
|
, public PeerListContentDelegate {
|
||||||
|
|
|
@ -16,6 +16,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "boxes/peers/edit_peer_type_box.h"
|
#include "boxes/peers/edit_peer_type_box.h"
|
||||||
#include "boxes/peers/edit_peer_history_visibility_box.h"
|
#include "boxes/peers/edit_peer_history_visibility_box.h"
|
||||||
#include "boxes/peers/edit_peer_permissions_box.h"
|
#include "boxes/peers/edit_peer_permissions_box.h"
|
||||||
|
#include "boxes/peers/edit_peer_invite_links.h"
|
||||||
#include "boxes/peers/edit_linked_chat_box.h"
|
#include "boxes/peers/edit_linked_chat_box.h"
|
||||||
#include "boxes/stickers_box.h"
|
#include "boxes/stickers_box.h"
|
||||||
#include "chat_helpers/emoji_suggestions_widget.h"
|
#include "chat_helpers/emoji_suggestions_widget.h"
|
||||||
|
@ -1003,7 +1004,7 @@ void Controller::fillManageSection() {
|
||||||
});
|
});
|
||||||
}) | rpl::flatten_latest(
|
}) | rpl::flatten_latest(
|
||||||
) | ToPositiveNumberString(),
|
) | ToPositiveNumberString(),
|
||||||
[=] { },
|
[=] { Ui::show(Box(ManageInviteLinksBox, _peer)); },
|
||||||
st::infoIconInviteLinks);
|
st::infoIconInviteLinks);
|
||||||
}
|
}
|
||||||
if (canViewAdmins) {
|
if (canViewAdmins) {
|
||||||
|
|
|
@ -15,8 +15,10 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "main/main_session.h"
|
#include "main/main_session.h"
|
||||||
#include "api/api_invite_links.h"
|
#include "api/api_invite_links.h"
|
||||||
#include "ui/wrap/vertical_layout.h"
|
#include "ui/wrap/vertical_layout.h"
|
||||||
|
#include "ui/wrap/slide_wrap.h"
|
||||||
#include "ui/wrap/padding_wrap.h"
|
#include "ui/wrap/padding_wrap.h"
|
||||||
#include "ui/abstract_button.h"
|
#include "ui/abstract_button.h"
|
||||||
|
#include "ui/widgets/buttons.h"
|
||||||
#include "ui/widgets/popup_menu.h"
|
#include "ui/widgets/popup_menu.h"
|
||||||
#include "ui/controls/invite_link_label.h"
|
#include "ui/controls/invite_link_label.h"
|
||||||
#include "ui/controls/invite_link_buttons.h"
|
#include "ui/controls/invite_link_buttons.h"
|
||||||
|
@ -29,17 +31,152 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "boxes/confirm_box.h"
|
#include "boxes/confirm_box.h"
|
||||||
#include "boxes/peer_list_box.h"
|
#include "boxes/peer_list_box.h"
|
||||||
#include "boxes/peer_list_controllers.h"
|
#include "boxes/peer_list_controllers.h"
|
||||||
|
#include "settings/settings_common.h" // AddDivider.
|
||||||
#include "apiwrap.h"
|
#include "apiwrap.h"
|
||||||
#include "mainwindow.h"
|
#include "mainwindow.h"
|
||||||
#include "boxes/share_box.h"
|
#include "boxes/share_box.h"
|
||||||
|
#include "base/weak_ptr.h"
|
||||||
|
#include "base/unixtime.h"
|
||||||
#include "window/window_session_controller.h"
|
#include "window/window_session_controller.h"
|
||||||
#include "api/api_common.h"
|
#include "api/api_common.h"
|
||||||
#include "styles/style_info.h"
|
#include "styles/style_info.h"
|
||||||
|
|
||||||
|
#include <xxhash.h>
|
||||||
|
|
||||||
#include <QtGui/QGuiApplication>
|
#include <QtGui/QGuiApplication>
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
|
constexpr auto kPreloadPages = 2;
|
||||||
|
constexpr auto kExpireSoonNominator = 3;
|
||||||
|
constexpr auto kExpireSoonDenominator = 4;
|
||||||
|
|
||||||
|
enum class Color {
|
||||||
|
Permanent,
|
||||||
|
Expiring,
|
||||||
|
ExpireSoon,
|
||||||
|
Expired,
|
||||||
|
Revoked,
|
||||||
|
|
||||||
|
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;
|
||||||
|
|
||||||
|
class RowDelegate {
|
||||||
|
public:
|
||||||
|
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);
|
||||||
|
|
||||||
|
[[nodiscard]] InviteLinkData data() const;
|
||||||
|
|
||||||
|
QString generateName() override;
|
||||||
|
QString generateShortName() override;
|
||||||
|
PaintRoundImageCallback generatePaintUserpicCallback() override;
|
||||||
|
|
||||||
|
QSize actionSize() const override;
|
||||||
|
QMargins actionMargins() const override;
|
||||||
|
void paintAction(
|
||||||
|
Painter &p,
|
||||||
|
int x,
|
||||||
|
int y,
|
||||||
|
int outerWidth,
|
||||||
|
bool selected,
|
||||||
|
bool actionSelected) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
const not_null<RowDelegate*> _delegate;
|
||||||
|
InviteLinkData _data;
|
||||||
|
QString _status;
|
||||||
|
float64 _progressTillExpire = 0.;
|
||||||
|
Color _color = Color::Permanent;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
[[nodiscard]] uint64 ComputeRowId(const QString &link) {
|
||||||
|
return XXH64(link.data(), link.size() * sizeof(ushort), 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
[[nodiscard]] uint64 ComputeRowId(const InviteLinkData &data) {
|
||||||
|
return ComputeRowId(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) {
|
||||||
|
const auto startDate = link.startDate ? link.startDate : link.date;
|
||||||
|
return 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) {
|
||||||
|
return "nothing";
|
||||||
|
}
|
||||||
|
|
||||||
|
void CopyLink(const QString &link) {
|
||||||
|
QGuiApplication::clipboard()->setText(link);
|
||||||
|
Ui::Toast::Show(tr::lng_group_invite_copied(tr::now));
|
||||||
|
}
|
||||||
|
|
||||||
void ShareLinkBox(not_null<PeerData*> peer, const QString &link) {
|
void ShareLinkBox(not_null<PeerData*> peer, const QString &link) {
|
||||||
const auto session = &peer->session();
|
const auto session = &peer->session();
|
||||||
const auto sending = std::make_shared<bool>();
|
const auto sending = std::make_shared<bool>();
|
||||||
|
@ -118,6 +255,323 @@ void ShareLinkBox(not_null<PeerData*> peer, const QString &link) {
|
||||||
std::move(filterCallback)));
|
std::move(filterCallback)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
not_null<Ui::SettingsButton*> AddCreateLinkButton(
|
||||||
|
not_null<Ui::VerticalLayout*> container) {
|
||||||
|
const auto result = container->add(
|
||||||
|
object_ptr<Ui::SettingsButton>(
|
||||||
|
container,
|
||||||
|
tr::lng_group_invite_add(),
|
||||||
|
st::inviteLinkCreate),
|
||||||
|
style::margins(0, st::inviteLinkCreateSkip, 0, 0));
|
||||||
|
const auto icon = Ui::CreateChild<Ui::RpWidget>(result);
|
||||||
|
icon->setAttribute(Qt::WA_TransparentForMouseEvents);
|
||||||
|
const auto size = st::inviteLinkCreateIconSize;
|
||||||
|
icon->resize(size, size);
|
||||||
|
result->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;
|
||||||
|
}
|
||||||
|
|
||||||
|
void EditLinkBox(
|
||||||
|
not_null<Ui::GenericBox*> box,
|
||||||
|
not_null<PeerData*> peer,
|
||||||
|
const InviteLinkData &data) {
|
||||||
|
box->setTitle(data.link.isEmpty()
|
||||||
|
? tr::lng_group_invite_new_title()
|
||||||
|
: tr::lng_group_invite_edit_title());
|
||||||
|
}
|
||||||
|
|
||||||
|
void CreateLinkBox(
|
||||||
|
not_null<Ui::GenericBox*> box,
|
||||||
|
not_null<PeerData*> peer) {
|
||||||
|
EditLinkBox(
|
||||||
|
box,
|
||||||
|
peer,
|
||||||
|
InviteLinkData{ .admin = peer->session().user() });
|
||||||
|
}
|
||||||
|
|
||||||
|
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)) {
|
||||||
|
setCustomStatus(ComputeStatus(data));
|
||||||
|
}
|
||||||
|
|
||||||
|
void Row::update(const InviteLinkData &data) {
|
||||||
|
_data = data;
|
||||||
|
_progressTillExpire = ComputeProgress(data, base::unixtime::now());
|
||||||
|
_color = ComputeColor(data, _progressTillExpire);
|
||||||
|
setCustomStatus(ComputeStatus(data));
|
||||||
|
_delegate->rowUpdateRow(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
InviteLinkData Row::data() const {
|
||||||
|
return _data;
|
||||||
|
}
|
||||||
|
|
||||||
|
QString Row::generateName() {
|
||||||
|
auto result = _data.link;
|
||||||
|
return result.replace(qstr("https://"), QString());
|
||||||
|
}
|
||||||
|
|
||||||
|
QString Row::generateShortName() {
|
||||||
|
return generateName();
|
||||||
|
}
|
||||||
|
|
||||||
|
PaintRoundImageCallback Row::generatePaintUserpicCallback() {
|
||||||
|
return [=](
|
||||||
|
Painter &p,
|
||||||
|
int x,
|
||||||
|
int y,
|
||||||
|
int outerWidth,
|
||||||
|
int size) {
|
||||||
|
_delegate->rowPaintIcon(p, x, y, size, _progressTillExpire, _color);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
QSize Row::actionSize() const {
|
||||||
|
return QSize(
|
||||||
|
st::inviteLinkThreeDotsIcon.width(),
|
||||||
|
st::inviteLinkThreeDotsIcon.height());
|
||||||
|
}
|
||||||
|
|
||||||
|
QMargins Row::actionMargins() const {
|
||||||
|
return QMargins(
|
||||||
|
0,
|
||||||
|
(st::inviteLinkList.item.height - actionSize().height()) / 2,
|
||||||
|
st::inviteLinkThreeDotsSkip,
|
||||||
|
0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Row::paintAction(
|
||||||
|
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 RowDelegate
|
||||||
|
, public base::has_weak_ptr {
|
||||||
|
public:
|
||||||
|
explicit Controller(not_null<PeerData*> peer);
|
||||||
|
|
||||||
|
void prepare() override;
|
||||||
|
void rowClicked(not_null<PeerListRow*> row) override;
|
||||||
|
void rowActionClicked(not_null<PeerListRow*> row) override;
|
||||||
|
base::unique_qptr<Ui::PopupMenu> rowContextMenu(
|
||||||
|
QWidget *parent,
|
||||||
|
not_null<PeerListRow*> row) override;
|
||||||
|
Main::Session &session() const override;
|
||||||
|
|
||||||
|
void rowUpdateRow(not_null<Row*> row) override;
|
||||||
|
void rowPaintIcon(
|
||||||
|
QPainter &p,
|
||||||
|
int x,
|
||||||
|
int y,
|
||||||
|
int size,
|
||||||
|
float64 progress,
|
||||||
|
Color color) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
void appendRow(const InviteLinkData &data, TimeId now);
|
||||||
|
[[nodiscard]] base::unique_qptr<Ui::PopupMenu> createRowContextMenu(
|
||||||
|
QWidget *parent,
|
||||||
|
not_null<PeerListRow*> row);
|
||||||
|
|
||||||
|
not_null<PeerData*> _peer;
|
||||||
|
base::unique_qptr<Ui::PopupMenu> _menu;
|
||||||
|
|
||||||
|
std::array<QImage, int(Color::Count)> _icons;
|
||||||
|
rpl::lifetime _lifetime;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
Controller::Controller(not_null<PeerData*> peer)
|
||||||
|
: _peer(peer) {
|
||||||
|
style::PaletteChanged(
|
||||||
|
) | rpl::start_with_next([=] {
|
||||||
|
for (auto &image : _icons) {
|
||||||
|
image = QImage();
|
||||||
|
}
|
||||||
|
}, _lifetime);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Controller::prepare() {
|
||||||
|
const auto now = base::unixtime::now();
|
||||||
|
const auto &links = _peer->session().api().inviteLinks().links(_peer);
|
||||||
|
for (const auto &link : links.links) {
|
||||||
|
if (!link.permanent || link.revoked) {
|
||||||
|
appendRow(link, now);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
delegate()->peerListRefreshRows();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Controller::rowClicked(not_null<PeerListRow*> row) {
|
||||||
|
// #TODO links show
|
||||||
|
}
|
||||||
|
|
||||||
|
void Controller::rowActionClicked(not_null<PeerListRow*> row) {
|
||||||
|
delegate()->peerListShowRowMenu(row, nullptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
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 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);
|
||||||
|
if (data.revoked) {
|
||||||
|
//result->addAction(tr::lng_group_invite_context_delete(tr::now), [=] {
|
||||||
|
// // #TODO links delete
|
||||||
|
//});
|
||||||
|
} else {
|
||||||
|
result->addAction(tr::lng_group_invite_context_copy(tr::now), [=] {
|
||||||
|
CopyLink(link);
|
||||||
|
});
|
||||||
|
result->addAction(tr::lng_group_invite_context_share(tr::now), [=] {
|
||||||
|
ShareLinkBox(_peer, link);
|
||||||
|
});
|
||||||
|
result->addAction(tr::lng_group_invite_context_edit(tr::now), [=] {
|
||||||
|
Ui::show(
|
||||||
|
Box(EditLinkBox, _peer, data),
|
||||||
|
Ui::LayerOption::KeepOther);
|
||||||
|
});
|
||||||
|
result->addAction(tr::lng_group_invite_context_revoke(tr::now), [=] {
|
||||||
|
const auto box = std::make_shared<QPointer<ConfirmBox>>();
|
||||||
|
const auto revoke = crl::guard(this, [=] {
|
||||||
|
const auto done = crl::guard(this, [=](InviteLinkData data) {
|
||||||
|
// #TODO links add to revoked, remove from list
|
||||||
|
if (*box) {
|
||||||
|
(*box)->closeBox();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
_peer->session().api().inviteLinks().revoke(
|
||||||
|
_peer,
|
||||||
|
link,
|
||||||
|
done);
|
||||||
|
});
|
||||||
|
*box = Ui::show(
|
||||||
|
Box<ConfirmBox>(
|
||||||
|
tr::lng_group_invite_revoke_about(tr::now),
|
||||||
|
revoke),
|
||||||
|
Ui::LayerOption::KeepOther);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
Main::Session &Controller::session() const {
|
||||||
|
return _peer->session();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Controller::appendRow(const InviteLinkData &data, TimeId now) {
|
||||||
|
delegate()->peerListAppendRow(std::make_unique<Row>(this, data, now));
|
||||||
|
}
|
||||||
|
|
||||||
|
void Controller::rowUpdateRow(not_null<Row*> row) {
|
||||||
|
delegate()->peerListUpdateRow(row);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Controller::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;
|
||||||
|
}
|
||||||
|
Unexpected("Color in Controller::rowPaintIcon.");
|
||||||
|
}();
|
||||||
|
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);
|
||||||
|
p.drawEllipse(0, 0, inner, inner);
|
||||||
|
st::inviteLinkIcon.paintInCenter(p, { 0, 0, inner, inner });
|
||||||
|
}
|
||||||
|
p.drawImage(x + skip, y + skip, icon);
|
||||||
|
if (progress >= 0. && progress < 1.) {
|
||||||
|
const auto kFullArcLength = 360 * 16;
|
||||||
|
const auto stroke = st::inviteLinkIconStroke;
|
||||||
|
auto hq = PainterHighQualityEnabler(p);
|
||||||
|
auto pen = QPen((*bg)->c);
|
||||||
|
pen.setWidth(stroke);
|
||||||
|
p.setPen(pen);
|
||||||
|
p.setBrush(Qt::NoBrush);
|
||||||
|
|
||||||
|
const auto margins = 1.5 * stroke;
|
||||||
|
p.drawArc(QRectF(x + skip, y + skip, inner, inner).marginsAdded({
|
||||||
|
margins,
|
||||||
|
margins,
|
||||||
|
margins,
|
||||||
|
margins,
|
||||||
|
}), 0, kFullArcLength * progress);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
void AddPermanentLinkBlock(
|
void AddPermanentLinkBlock(
|
||||||
|
@ -143,8 +597,7 @@ void AddPermanentLinkBlock(
|
||||||
const auto weak = Ui::MakeWeak(container);
|
const auto weak = Ui::MakeWeak(container);
|
||||||
const auto copyLink = crl::guard(weak, [=] {
|
const auto copyLink = crl::guard(weak, [=] {
|
||||||
if (const auto link = computePermanentLink()) {
|
if (const auto link = computePermanentLink()) {
|
||||||
QGuiApplication::clipboard()->setText(link->link);
|
CopyLink(link->link);
|
||||||
Ui::Toast::Show(tr::lng_group_invite_copied(tr::now));
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
const auto shareLink = crl::guard(weak, [=] {
|
const auto shareLink = crl::guard(weak, [=] {
|
||||||
|
@ -155,17 +608,12 @@ void AddPermanentLinkBlock(
|
||||||
const auto revokeLink = crl::guard(weak, [=] {
|
const auto revokeLink = crl::guard(weak, [=] {
|
||||||
const auto box = std::make_shared<QPointer<ConfirmBox>>();
|
const auto box = std::make_shared<QPointer<ConfirmBox>>();
|
||||||
const auto done = crl::guard(weak, [=] {
|
const auto done = crl::guard(weak, [=] {
|
||||||
if (const auto link = computePermanentLink()) {
|
const auto close = [=](auto&&) {
|
||||||
const auto close = [=](auto&&) {
|
if (*box) {
|
||||||
if (*box) {
|
(*box)->closeBox();
|
||||||
(*box)->closeBox();
|
}
|
||||||
}
|
};
|
||||||
};
|
peer->session().api().inviteLinks().revokePermanent(peer, close);
|
||||||
peer->session().api().inviteLinks().revoke(
|
|
||||||
peer,
|
|
||||||
link->link,
|
|
||||||
close);
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
*box = Ui::show(
|
*box = Ui::show(
|
||||||
Box<ConfirmBox>(tr::lng_group_invite_about_new(tr::now), done),
|
Box<ConfirmBox>(tr::lng_group_invite_about_new(tr::now), done),
|
||||||
|
@ -286,4 +734,40 @@ void AddPermanentLinkBlock(
|
||||||
st::inviteLinkJoinedRowPadding
|
st::inviteLinkJoinedRowPadding
|
||||||
)->setClickedCallback([=] {
|
)->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 ManageInviteLinksBox(
|
||||||
|
not_null<Ui::GenericBox*> box,
|
||||||
|
not_null<PeerData*> peer) {
|
||||||
|
box->setTitle(tr::lng_group_invite_title());
|
||||||
|
|
||||||
|
const auto container = box->verticalLayout();
|
||||||
|
AddPermanentLinkBlock(container, peer);
|
||||||
|
Settings::AddDivider(container);
|
||||||
|
|
||||||
|
const auto add = AddCreateLinkButton(container);
|
||||||
|
add->setClickedCallback([=] {
|
||||||
|
box->getDelegate()->show(Box(CreateLinkBox, peer));
|
||||||
|
});
|
||||||
|
|
||||||
|
const auto delegate = box->lifetime().make_state<
|
||||||
|
PeerListContentDelegateSimple
|
||||||
|
>();
|
||||||
|
const auto controller = box->lifetime().make_state<Controller>(peer);
|
||||||
|
controller->setStyleOverrides(&st::inviteLinkList);
|
||||||
|
const auto content = container->add(object_ptr<PeerListContent>(
|
||||||
|
container,
|
||||||
|
controller));
|
||||||
|
delegate->setContent(content);
|
||||||
|
controller->setDelegate(delegate);
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,6 +7,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
*/
|
*/
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include "ui/layers/generic_box.h"
|
||||||
|
|
||||||
class PeerData;
|
class PeerData;
|
||||||
|
|
||||||
namespace Ui {
|
namespace Ui {
|
||||||
|
@ -16,3 +18,7 @@ class VerticalLayout;
|
||||||
void AddPermanentLinkBlock(
|
void AddPermanentLinkBlock(
|
||||||
not_null<Ui::VerticalLayout*> container,
|
not_null<Ui::VerticalLayout*> container,
|
||||||
not_null<PeerData*> peer);
|
not_null<PeerData*> peer);
|
||||||
|
|
||||||
|
void ManageInviteLinksBox(
|
||||||
|
not_null<Ui::GenericBox*> box,
|
||||||
|
not_null<PeerData*> peer);
|
||||||
|
|
|
@ -185,7 +185,7 @@ void Controller::createContent() {
|
||||||
_wrap.get(),
|
_wrap.get(),
|
||||||
tr::lng_group_invite_manage(),
|
tr::lng_group_invite_manage(),
|
||||||
rpl::single(QString()),
|
rpl::single(QString()),
|
||||||
[=] { /*ShowEditInviteLinks(_navigation, _peer);*/ },
|
[=] { Ui::show(Box(ManageInviteLinksBox, _peer)); },
|
||||||
st::manageGroupButton,
|
st::manageGroupButton,
|
||||||
&st::infoIconInviteLinks));
|
&st::infoIconInviteLinks));
|
||||||
AddSkip(_wrap.get());
|
AddSkip(_wrap.get());
|
||||||
|
|
|
@ -839,12 +839,14 @@ inviteLinkField: FlatInput(defaultFlatInput) {
|
||||||
height: 44px;
|
height: 44px;
|
||||||
textMrg: margins(14px, 12px, 36px, 9px);
|
textMrg: margins(14px, 12px, 36px, 9px);
|
||||||
}
|
}
|
||||||
|
inviteLinkThreeDotsIcon: icon {{ "info/edit/dotsmini", dialogsMenuIconFg }};
|
||||||
|
inviteLinkThreeDotsIconOver: icon {{ "info/edit/dotsmini", dialogsMenuIconFgOver }};
|
||||||
inviteLinkThreeDots: IconButton(defaultIconButton) {
|
inviteLinkThreeDots: IconButton(defaultIconButton) {
|
||||||
width: 36px;
|
width: 36px;
|
||||||
height: 44px;
|
height: 44px;
|
||||||
|
|
||||||
icon: icon {{ "info/edit/dotsmini", dialogsMenuIconFg }};
|
icon: inviteLinkThreeDotsIcon;
|
||||||
iconOver: icon {{ "info/edit/dotsmini", dialogsMenuIconFgOver }};
|
iconOver: inviteLinkThreeDotsIconOver;
|
||||||
iconPosition: point(-1px, -1px);
|
iconPosition: point(-1px, -1px);
|
||||||
|
|
||||||
rippleAreaSize: 0px;
|
rippleAreaSize: 0px;
|
||||||
|
@ -875,3 +877,43 @@ inviteLinkUserpics: GroupCallUserpics {
|
||||||
inviteLinkUserpicsSkip: 8px;
|
inviteLinkUserpicsSkip: 8px;
|
||||||
inviteLinkJoinedFont: font(14px);
|
inviteLinkJoinedFont: font(14px);
|
||||||
inviteLinkJoinedRowPadding: margins(0px, 18px, 0px, 8px);
|
inviteLinkJoinedRowPadding: margins(0px, 18px, 0px, 8px);
|
||||||
|
|
||||||
|
inviteLinkCreateSkip: 8px;
|
||||||
|
inviteLinkCreate: SettingsButton(defaultSettingsButton) {
|
||||||
|
textFg: lightButtonFg;
|
||||||
|
textFgOver: lightButtonFgOver;
|
||||||
|
textBg: windowBg;
|
||||||
|
textBgOver: windowBgOver;
|
||||||
|
|
||||||
|
font: semiboldFont;
|
||||||
|
|
||||||
|
height: 20px;
|
||||||
|
padding: margins(58px, 7px, 12px, 5px);
|
||||||
|
|
||||||
|
toggle: infoProfileToggle;
|
||||||
|
toggleOver: infoProfileToggleOver;
|
||||||
|
toggleSkip: 22px;
|
||||||
|
|
||||||
|
ripple: defaultRippleAnimation;
|
||||||
|
}
|
||||||
|
inviteLinkCreateIcon: icon {{ "info/edit/roundbtn_plus", windowFgActive }};
|
||||||
|
inviteLinkCreateIconSize: 20px;
|
||||||
|
inviteLinkListItem: PeerListItem(defaultPeerListItem) {
|
||||||
|
button: OutlineButton(defaultPeerListButton) {
|
||||||
|
font: normalFont;
|
||||||
|
padding: margins(11px, 5px, 11px, 5px);
|
||||||
|
}
|
||||||
|
height: 52px;
|
||||||
|
photoPosition: point(8px, 6px);
|
||||||
|
namePosition: point(58px, 6px);
|
||||||
|
statusPosition: point(58px, 25px);
|
||||||
|
photoSize: 40px;
|
||||||
|
}
|
||||||
|
inviteLinkList: PeerList(defaultPeerList) {
|
||||||
|
item: inviteLinkListItem;
|
||||||
|
padding: margins(0px, 4px, 0px, 4px);
|
||||||
|
}
|
||||||
|
inviteLinkIconSkip: 7px;
|
||||||
|
inviteLinkIconStroke: 2px;
|
||||||
|
inviteLinkIcon: icon {{ "info/edit/links_link", mediaviewFileExtFg }};
|
||||||
|
inviteLinkThreeDotsSkip: 8px;
|
||||||
|
|
Loading…
Add table
Reference in a new issue