Moved out functions to create expandable moderate list to separate file.

This commit is contained in:
23rd 2024-08-21 11:44:31 +03:00
parent b364c4f23a
commit eb268102fc
4 changed files with 272 additions and 194 deletions

View file

@ -1509,6 +1509,8 @@ PRIVATE
ui/image/image_location_factory.h ui/image/image_location_factory.h
ui/text/format_song_document_name.cpp ui/text/format_song_document_name.cpp
ui/text/format_song_document_name.h ui/text/format_song_document_name.h
ui/widgets/expandable_peer_list.cpp
ui/widgets/expandable_peer_list.h
ui/widgets/label_with_custom_emoji.cpp ui/widgets/label_with_custom_emoji.cpp
ui/widgets/label_with_custom_emoji.h ui/widgets/label_with_custom_emoji.h
ui/countryinput.cpp ui/countryinput.cpp

View file

@ -12,7 +12,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "apiwrap.h" #include "apiwrap.h"
#include "base/event_filter.h" #include "base/event_filter.h"
#include "base/timer.h" #include "base/timer.h"
#include "ui/widgets/participants_check_view.h"
#include "boxes/delete_messages_box.h" #include "boxes/delete_messages_box.h"
#include "boxes/peers/edit_peer_permissions_box.h" #include "boxes/peers/edit_peer_permissions_box.h"
#include "core/application.h" #include "core/application.h"
@ -32,7 +31,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "ui/boxes/confirm_box.h" #include "ui/boxes/confirm_box.h"
#include "ui/controls/userpic_button.h" #include "ui/controls/userpic_button.h"
#include "ui/effects/ripple_animation.h" #include "ui/effects/ripple_animation.h"
#include "ui/effects/toggle_arrow.h"
#include "ui/layers/generic_box.h" #include "ui/layers/generic_box.h"
#include "ui/painter.h" #include "ui/painter.h"
#include "ui/rect.h" #include "ui/rect.h"
@ -40,6 +38,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "ui/text/text_utilities.h" #include "ui/text/text_utilities.h"
#include "ui/vertical_list.h" #include "ui/vertical_list.h"
#include "ui/widgets/checkbox.h" #include "ui/widgets/checkbox.h"
#include "ui/widgets/expandable_peer_list.h"
#include "ui/widgets/participants_check_view.h"
#include "ui/wrap/slide_wrap.h" #include "ui/wrap/slide_wrap.h"
#include "styles/style_boxes.h" #include "styles/style_boxes.h"
#include "styles/style_layers.h" #include "styles/style_layers.h"
@ -47,15 +47,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
namespace { namespace {
using Participants = std::vector<not_null<PeerData*>>;
struct Controller final {
rpl::event_stream<bool> toggleRequestsFromTop;
rpl::event_stream<bool> toggleRequestsFromInner;
rpl::event_stream<bool> checkAllRequests;
Fn<Participants()> collectRequests;
};
struct ModerateOptions final { struct ModerateOptions final {
bool allCanBan = false; bool allCanBan = false;
bool allCanDelete = false; bool allCanDelete = false;
@ -111,192 +102,14 @@ ModerateOptions CalculateModerateOptions(const HistoryItemsList &items) {
}; };
} }
class Button final : public Ui::RippleButton {
public:
Button(not_null<QWidget*> parent, int count);
[[nodiscard]] not_null<Ui::AbstractCheckView*> checkView() const;
private:
void paintEvent(QPaintEvent *event) override;
QImage prepareRippleMask() const override;
QPoint prepareRippleStartPosition() const override;
std::unique_ptr<Ui::AbstractCheckView> _view;
};
Button::Button(not_null<QWidget*> parent, int count)
: Ui::RippleButton(parent, st::defaultRippleAnimation)
, _view(std::make_unique<Ui::ParticipantsCheckView>(
count,
st::slideWrapDuration,
false,
[=] { update(); })) {
}
not_null<Ui::AbstractCheckView*> Button::checkView() const {
return _view.get();
}
QImage Button::prepareRippleMask() const {
return _view->prepareRippleMask();
}
QPoint Button::prepareRippleStartPosition() const {
return mapFromGlobal(QCursor::pos());
}
void Button::paintEvent(QPaintEvent *event) {
auto p = QPainter(this);
Ui::RippleButton::paintRipple(p, QPoint());
_view->paint(p, 0, 0, width());
}
void CreateParticipantsList(
not_null<Controller*> controller,
not_null<Ui::VerticalLayout*> inner,
const Participants &participants) {
const auto wrap = inner->add(
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
inner,
object_ptr<Ui::VerticalLayout>(inner)));
wrap->toggle(false, anim::type::instant);
controller->toggleRequestsFromTop.events(
) | rpl::start_with_next([=](bool toggled) {
wrap->toggle(toggled, anim::type::normal);
}, wrap->lifetime());
const auto container = wrap->entity();
Ui::AddSkip(container);
auto &lifetime = wrap->lifetime();
const auto clicks = lifetime.make_state<rpl::event_stream<>>();
const auto checkboxes = ranges::views::all(
participants
) | ranges::views::transform([&](not_null<PeerData*> peer) {
const auto line = container->add(
object_ptr<Ui::AbstractButton>(container));
const auto &st = st::moderateBoxUserpic;
line->resize(line->width(), st.size.height());
const auto userpic = Ui::CreateChild<Ui::UserpicButton>(
line,
peer,
st);
const auto checkbox = Ui::CreateChild<Ui::Checkbox>(
line,
peer->name(),
false,
st::defaultBoxCheckbox);
line->widthValue(
) | rpl::start_with_next([=](int width) {
userpic->moveToLeft(
st::boxRowPadding.left()
+ checkbox->checkRect().width()
+ st::defaultBoxCheckbox.textPosition.x(),
0);
const auto skip = st::defaultBoxCheckbox.textPosition.x();
checkbox->resizeToWidth(width
- rect::right(userpic)
- skip
- st::boxRowPadding.right());
checkbox->moveToLeft(
rect::right(userpic) + skip,
((userpic->height() - checkbox->height()) / 2)
+ st::defaultBoxCheckbox.margin.top());
}, checkbox->lifetime());
userpic->setAttribute(Qt::WA_TransparentForMouseEvents);
checkbox->setAttribute(Qt::WA_TransparentForMouseEvents);
line->setClickedCallback([=] {
checkbox->setChecked(!checkbox->checked());
clicks->fire({});
});
return checkbox;
}) | ranges::to_vector;
clicks->events(
) | rpl::start_with_next([=] {
controller->toggleRequestsFromInner.fire_copy(
ranges::any_of(checkboxes, &Ui::Checkbox::checked));
}, container->lifetime());
controller->checkAllRequests.events(
) | rpl::start_with_next([=](bool checked) {
for (const auto &c : checkboxes) {
c->setChecked(checked);
}
}, container->lifetime());
controller->collectRequests = [=] {
auto result = Participants();
for (auto i = 0; i < checkboxes.size(); i++) {
if (checkboxes[i]->checked()) {
result.push_back(participants[i]);
}
}
return result;
};
}
void AppendList(
not_null<Ui::Checkbox*> checkbox,
not_null<Controller*> controller,
not_null<Ui::VerticalLayout*> inner,
const Participants &participants,
bool handleSingle) {
const auto isSingle = handleSingle ? (participants.size() == 1) : false;
if (isSingle) {
const auto p = participants.front();
controller->collectRequests = [=] { return Participants{ p }; };
return;
}
const auto count = int(participants.size());
const auto button = Ui::CreateChild<Button>(inner, count);
button->resize(Ui::ParticipantsCheckView::ComputeSize(count));
const auto overlay = Ui::CreateChild<Ui::AbstractButton>(inner);
checkbox->geometryValue(
) | rpl::start_with_next([=](const QRect &rect) {
overlay->setGeometry(rect);
overlay->raise();
button->moveToRight(
st::moderateBoxExpandRight,
rect.top() + (rect.height() - button->height()) / 2,
inner->width());
button->raise();
}, button->lifetime());
controller->toggleRequestsFromInner.events(
) | rpl::start_with_next([=](bool toggled) {
checkbox->setChecked(toggled);
}, checkbox->lifetime());
button->setClickedCallback([=] {
button->checkView()->setChecked(
!button->checkView()->checked(),
anim::type::normal);
controller->toggleRequestsFromTop.fire_copy(
button->checkView()->checked());
});
overlay->setClickedCallback([=] {
checkbox->setChecked(!checkbox->checked());
controller->checkAllRequests.fire_copy(checkbox->checked());
});
CreateParticipantsList(controller, inner, participants);
}
} // namespace } // namespace
void CreateModerateMessagesBox( void CreateModerateMessagesBox(
not_null<Ui::GenericBox*> box, not_null<Ui::GenericBox*> box,
const HistoryItemsList &items, const HistoryItemsList &items,
Fn<void()> confirmed) { Fn<void()> confirmed) {
using Controller = Ui::ExpandablePeerListController;
const auto [allCanBan, allCanDelete, participants] const auto [allCanBan, allCanDelete, participants]
= CalculateModerateOptions(items); = CalculateModerateOptions(items);
const auto inner = box->verticalLayout(); const auto inner = box->verticalLayout();
@ -417,7 +230,13 @@ void CreateModerateMessagesBox(
st::defaultBoxCheckbox), st::defaultBoxCheckbox),
st::boxRowPadding + buttonPadding); st::boxRowPadding + buttonPadding);
const auto controller = box->lifetime().make_state<Controller>(); const auto controller = box->lifetime().make_state<Controller>();
AppendList(report, controller, inner, participants, true); Ui::AddExpandablePeerList(
report,
controller,
inner,
participants,
true,
false);
handleSubmition(report); handleSubmition(report);
const auto ids = items.front()->from()->owner().itemsToIds(items); const auto ids = items.front()->from()->owner().itemsToIds(items);
@ -475,7 +294,13 @@ void CreateModerateMessagesBox(
} }
const auto controller = box->lifetime().make_state<Controller>(); const auto controller = box->lifetime().make_state<Controller>();
AppendList(deleteAll, controller, inner, participants, true); Ui::AddExpandablePeerList(
deleteAll,
controller,
inner,
participants,
true,
false);
handleSubmition(deleteAll); handleSubmition(deleteAll);
handleConfirmation(deleteAll, controller, [=]( handleConfirmation(deleteAll, controller, [=](
@ -505,7 +330,13 @@ void CreateModerateMessagesBox(
st::defaultBoxCheckbox), st::defaultBoxCheckbox),
st::boxRowPadding + buttonPadding); st::boxRowPadding + buttonPadding);
const auto controller = box->lifetime().make_state<Controller>(); const auto controller = box->lifetime().make_state<Controller>();
AppendList(ban, controller, inner, participants, true); Ui::AddExpandablePeerList(
ban,
controller,
inner,
participants,
true,
false);
handleSubmition(ban); handleSubmition(ban);
Ui::AddSkip(inner); Ui::AddSkip(inner);

View file

@ -0,0 +1,211 @@
/*
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 "ui/widgets/expandable_peer_list.h"
#include "data/data_peer.h"
#include "ui/controls/userpic_button.h"
#include "ui/rect.h"
#include "ui/vertical_list.h"
#include "ui/widgets/buttons.h"
#include "ui/widgets/participants_check_view.h"
#include "ui/wrap/slide_wrap.h"
#include "ui/wrap/vertical_layout.h"
#include "styles/style_boxes.h"
#include "styles/style_layers.h"
#include "styles/style_widgets.h"
namespace Ui {
namespace {
class Button final : public Ui::RippleButton {
public:
Button(not_null<QWidget*> parent, int count);
[[nodiscard]] not_null<Ui::AbstractCheckView*> checkView() const;
private:
void paintEvent(QPaintEvent *event) override;
QImage prepareRippleMask() const override;
QPoint prepareRippleStartPosition() const override;
std::unique_ptr<Ui::AbstractCheckView> _view;
};
Button::Button(not_null<QWidget*> parent, int count)
: Ui::RippleButton(parent, st::defaultRippleAnimation)
, _view(std::make_unique<Ui::ParticipantsCheckView>(
count,
st::slideWrapDuration,
false,
[=] { update(); })) {
}
not_null<Ui::AbstractCheckView*> Button::checkView() const {
return _view.get();
}
QImage Button::prepareRippleMask() const {
return _view->prepareRippleMask();
}
QPoint Button::prepareRippleStartPosition() const {
return mapFromGlobal(QCursor::pos());
}
void Button::paintEvent(QPaintEvent *event) {
auto p = QPainter(this);
Ui::RippleButton::paintRipple(p, QPoint());
_view->paint(p, 0, 0, width());
}
} // namespace
void AddExpandablePeerList(
not_null<Ui::Checkbox*> checkbox,
not_null<ExpandablePeerListController*> controller,
not_null<Ui::VerticalLayout*> inner,
const Participants &participants,
bool handleSingle,
bool hideRightButton) {
const auto isSingle = handleSingle ? (participants.size() == 1) : false;
if (isSingle) {
const auto p = participants.front();
controller->collectRequests = [=] { return Participants{ p }; };
return;
}
const auto count = int(participants.size());
const auto button = !hideRightButton
? Ui::CreateChild<Button>(inner, count)
: nullptr;
if (button) {
button->resize(Ui::ParticipantsCheckView::ComputeSize(count));
}
const auto overlay = Ui::CreateChild<Ui::AbstractButton>(inner);
checkbox->geometryValue(
) | rpl::start_with_next([=](const QRect &rect) {
overlay->setGeometry(rect);
overlay->raise();
if (button) {
button->moveToRight(
st::moderateBoxExpandRight,
rect.top() + (rect.height() - button->height()) / 2,
inner->width());
button->raise();
}
}, overlay->lifetime());
controller->toggleRequestsFromInner.events(
) | rpl::start_with_next([=](bool toggled) {
checkbox->setChecked(toggled);
}, checkbox->lifetime());
if (button) {
button->setClickedCallback([=] {
button->checkView()->setChecked(
!button->checkView()->checked(),
anim::type::normal);
controller->toggleRequestsFromTop.fire_copy(
button->checkView()->checked());
});
}
overlay->setClickedCallback([=] {
checkbox->setChecked(!checkbox->checked());
controller->checkAllRequests.fire_copy(checkbox->checked());
});
{
const auto wrap = inner->add(
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
inner,
object_ptr<Ui::VerticalLayout>(inner)));
wrap->toggle(hideRightButton, anim::type::instant);
controller->toggleRequestsFromTop.events(
) | rpl::start_with_next([=](bool toggled) {
wrap->toggle(toggled, anim::type::normal);
}, wrap->lifetime());
const auto container = wrap->entity();
Ui::AddSkip(container);
auto &lifetime = wrap->lifetime();
const auto clicks = lifetime.make_state<rpl::event_stream<>>();
const auto checkboxes = ranges::views::all(
participants
) | ranges::views::transform([&](not_null<PeerData*> peer) {
const auto line = container->add(
object_ptr<Ui::AbstractButton>(container));
const auto &st = st::moderateBoxUserpic;
line->resize(line->width(), st.size.height());
const auto userpic = Ui::CreateChild<Ui::UserpicButton>(
line,
peer,
st);
const auto checkbox = Ui::CreateChild<Ui::Checkbox>(
line,
peer->name(),
false,
st::defaultBoxCheckbox);
line->widthValue(
) | rpl::start_with_next([=](int width) {
userpic->moveToLeft(
st::boxRowPadding.left()
+ checkbox->checkRect().width()
+ st::defaultBoxCheckbox.textPosition.x(),
0);
const auto skip = st::defaultBoxCheckbox.textPosition.x();
checkbox->resizeToWidth(width
- rect::right(userpic)
- skip
- st::boxRowPadding.right());
checkbox->moveToLeft(
rect::right(userpic) + skip,
((userpic->height() - checkbox->height()) / 2)
+ st::defaultBoxCheckbox.margin.top());
}, checkbox->lifetime());
userpic->setAttribute(Qt::WA_TransparentForMouseEvents);
checkbox->setAttribute(Qt::WA_TransparentForMouseEvents);
line->setClickedCallback([=] {
checkbox->setChecked(!checkbox->checked());
clicks->fire({});
});
return checkbox;
}) | ranges::to_vector;
clicks->events(
) | rpl::start_with_next([=] {
controller->toggleRequestsFromInner.fire_copy(
ranges::any_of(checkboxes, &Ui::Checkbox::checked));
}, container->lifetime());
controller->checkAllRequests.events(
) | rpl::start_with_next([=](bool checked) {
for (const auto &c : checkboxes) {
c->setChecked(checked);
}
}, container->lifetime());
controller->collectRequests = [=] {
auto result = Participants();
for (auto i = 0; i < checkboxes.size(); i++) {
if (checkboxes[i]->checked()) {
result.push_back(participants[i]);
}
}
return result;
};
}
}
} // namespace Ui

View file

@ -0,0 +1,34 @@
/*
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
*/
#pragma once
class PeerData;
using Participants = std::vector<not_null<PeerData*>>;
namespace Ui {
class Checkbox;
class VerticalLayout;
struct ExpandablePeerListController final {
rpl::event_stream<bool> toggleRequestsFromTop;
rpl::event_stream<bool> toggleRequestsFromInner;
rpl::event_stream<bool> checkAllRequests;
Fn<Participants()> collectRequests;
};
void AddExpandablePeerList(
not_null<Ui::Checkbox*> checkbox,
not_null<ExpandablePeerListController*> controller,
not_null<Ui::VerticalLayout*> inner,
const Participants &participants,
bool handleSingle,
bool hideRightButton);
} // namespace Ui