mirror of https://github.com/procxx/kepka.git
Edit chat types in filters.
This commit is contained in:
parent
13fe0b6272
commit
b88f0108ad
|
@ -185,7 +185,7 @@ BackgroundBox::Inner::Inner(
|
||||||
, _session(session)
|
, _session(session)
|
||||||
, _api(_session->api().instance())
|
, _api(_session->api().instance())
|
||||||
, _check(std::make_unique<Ui::RoundCheckbox>(st::overviewCheck, [=] { update(); })) {
|
, _check(std::make_unique<Ui::RoundCheckbox>(st::overviewCheck, [=] { update(); })) {
|
||||||
_check->setChecked(true, Ui::RoundCheckbox::SetStyle::Fast);
|
_check->setChecked(true, anim::type::instant);
|
||||||
if (_session->data().wallpapers().empty()) {
|
if (_session->data().wallpapers().empty()) {
|
||||||
resize(st::boxWideWidth, 2 * (st::backgroundSize.height() + st::backgroundPadding) + st::backgroundPadding);
|
resize(st::boxWideWidth, 2 * (st::backgroundSize.height() + st::backgroundPadding) + st::backgroundPadding);
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -216,26 +216,9 @@ contactsMultiSelect: MultiSelect {
|
||||||
fieldCancel: contactsSearchCancel;
|
fieldCancel: contactsSearchCancel;
|
||||||
fieldCancelSkip: 40px;
|
fieldCancelSkip: 40px;
|
||||||
}
|
}
|
||||||
contactsPhotoCheckIcon: icon {{
|
contactsPhotoCheckIcon: defaultPeerListCheckIcon;
|
||||||
"default_checkbox_check",
|
contactsPhotoCheck: defaultPeerListCheck;
|
||||||
overviewCheckFgActive,
|
contactsPhotoCheckbox: defaultPeerListCheckbox;
|
||||||
point(3px, 6px)
|
|
||||||
}};
|
|
||||||
contactsPhotoCheck: RoundCheckbox(defaultRoundCheckbox) {
|
|
||||||
size: 20px;
|
|
||||||
sizeSmall: 0.3;
|
|
||||||
bgInactive: overviewCheckBg;
|
|
||||||
bgActive: overviewCheckBgActive;
|
|
||||||
check: contactsPhotoCheckIcon;
|
|
||||||
}
|
|
||||||
contactsPhotoCheckbox: RoundImageCheckbox {
|
|
||||||
imageRadius: 21px;
|
|
||||||
imageSmallRadius: 18px;
|
|
||||||
selectWidth: 2px;
|
|
||||||
selectFg: windowBgActive;
|
|
||||||
selectDuration: 150;
|
|
||||||
check: contactsPhotoCheck;
|
|
||||||
}
|
|
||||||
contactsPhotoDisabledCheckFg: menuIconFg;
|
contactsPhotoDisabledCheckFg: menuIconFg;
|
||||||
contactsNameCheckedFg: windowActiveTextFg;
|
contactsNameCheckedFg: windowActiveTextFg;
|
||||||
contactsRipple: defaultRippleAnimation;
|
contactsRipple: defaultRippleAnimation;
|
||||||
|
|
|
@ -69,7 +69,7 @@ Main::Session &PrivacyExceptionsBoxController::session() const {
|
||||||
|
|
||||||
void PrivacyExceptionsBoxController::prepareViewHook() {
|
void PrivacyExceptionsBoxController::prepareViewHook() {
|
||||||
delegate()->peerListSetTitle(std::move(_title));
|
delegate()->peerListSetTitle(std::move(_title));
|
||||||
delegate()->peerListAddSelectedRows(_selected);
|
delegate()->peerListAddSelectedPeers(_selected);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<not_null<PeerData*>> PrivacyExceptionsBoxController::getResult() const {
|
std::vector<not_null<PeerData*>> PrivacyExceptionsBoxController::getResult() const {
|
||||||
|
|
|
@ -77,13 +77,9 @@ private:
|
||||||
|
|
||||||
void paintEvent(QPaintEvent *e) override;
|
void paintEvent(QPaintEvent *e) override;
|
||||||
|
|
||||||
void setup(
|
|
||||||
Flags flags,
|
|
||||||
const base::flat_set<not_null<History*>> &peers);
|
|
||||||
void refresh();
|
void refresh();
|
||||||
void removeFlag(Flag flag);
|
void removeFlag(Flag flag);
|
||||||
void removePeer(not_null<History*> history);
|
void removePeer(not_null<History*> history);
|
||||||
void paintFlagIcon(QPainter &p, int left, int top, Flag flag) const;
|
|
||||||
|
|
||||||
std::vector<FlagButton> _removeFlag;
|
std::vector<FlagButton> _removeFlag;
|
||||||
std::vector<PeerButton> _removePeer;
|
std::vector<PeerButton> _removePeer;
|
||||||
|
@ -187,21 +183,6 @@ int FilterChatsPreview::resizeGetHeight(int newWidth) {
|
||||||
return top;
|
return top;
|
||||||
}
|
}
|
||||||
|
|
||||||
[[nodiscard]] QString TypeName(Flag flag) {
|
|
||||||
switch (flag) {
|
|
||||||
case Flag::Contacts: return tr::lng_filters_type_contacts(tr::now);
|
|
||||||
case Flag::NonContacts:
|
|
||||||
return tr::lng_filters_type_non_contacts(tr::now);
|
|
||||||
case Flag::Groups: return tr::lng_filters_type_groups(tr::now);
|
|
||||||
case Flag::Channels: return tr::lng_filters_type_channels(tr::now);
|
|
||||||
case Flag::Bots: return tr::lng_filters_type_bots(tr::now);
|
|
||||||
case Flag::NoMuted: return tr::lng_filters_type_no_muted(tr::now);
|
|
||||||
case Flag::NoArchived: return tr::lng_filters_type_no_archived(tr::now);
|
|
||||||
case Flag::NoRead: return tr::lng_filters_type_no_read(tr::now);
|
|
||||||
}
|
|
||||||
Unexpected("Flag in TypeName.");
|
|
||||||
}
|
|
||||||
|
|
||||||
void FilterChatsPreview::paintEvent(QPaintEvent *e) {
|
void FilterChatsPreview::paintEvent(QPaintEvent *e) {
|
||||||
auto p = Painter(this);
|
auto p = Painter(this);
|
||||||
auto top = 0;
|
auto top = 0;
|
||||||
|
@ -212,10 +193,20 @@ void FilterChatsPreview::paintEvent(QPaintEvent *e) {
|
||||||
p.setFont(st::windowFilterSmallItem.nameStyle.font);
|
p.setFont(st::windowFilterSmallItem.nameStyle.font);
|
||||||
const auto nameTop = st.namePosition.y();
|
const auto nameTop = st.namePosition.y();
|
||||||
for (const auto &[flag, button] : _removeFlag) {
|
for (const auto &[flag, button] : _removeFlag) {
|
||||||
paintFlagIcon(p, iconLeft, top + iconTop, flag);
|
PaintFilterChatsTypeIcon(
|
||||||
|
p,
|
||||||
|
flag,
|
||||||
|
iconLeft,
|
||||||
|
top + iconTop,
|
||||||
|
width(),
|
||||||
|
st.photoSize);
|
||||||
|
|
||||||
p.setPen(st::contactsNameFg);
|
p.setPen(st::contactsNameFg);
|
||||||
p.drawTextLeft(nameLeft, top + nameTop, width(), TypeName(flag));
|
p.drawTextLeft(
|
||||||
|
nameLeft,
|
||||||
|
top + nameTop,
|
||||||
|
width(),
|
||||||
|
FilterChatsTypeName(flag));
|
||||||
top += st.height;
|
top += st.height;
|
||||||
}
|
}
|
||||||
for (const auto &[history, button] : _removePeer) {
|
for (const auto &[history, button] : _removePeer) {
|
||||||
|
@ -236,46 +227,6 @@ void FilterChatsPreview::paintEvent(QPaintEvent *e) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void FilterChatsPreview::paintFlagIcon(
|
|
||||||
QPainter &p,
|
|
||||||
int left,
|
|
||||||
int top,
|
|
||||||
Flag flag) const {
|
|
||||||
const auto &color = [&]() -> const style::color& {
|
|
||||||
switch (flag) {
|
|
||||||
case Flag::Contacts: return st::historyPeer4UserpicBg;
|
|
||||||
case Flag::NonContacts: return st::historyPeer7UserpicBg;
|
|
||||||
case Flag::Groups: return st::historyPeer2UserpicBg;
|
|
||||||
case Flag::Channels: return st::historyPeer1UserpicBg;
|
|
||||||
case Flag::Bots: return st::historyPeer6UserpicBg;
|
|
||||||
case Flag::NoMuted: return st::historyPeer6UserpicBg;
|
|
||||||
case Flag::NoArchived: return st::historyPeer4UserpicBg;
|
|
||||||
case Flag::NoRead: return st::historyPeer7UserpicBg;
|
|
||||||
}
|
|
||||||
Unexpected("Flag in color paintFlagIcon.");
|
|
||||||
}();
|
|
||||||
const auto &icon = [&]() -> const style::icon& {
|
|
||||||
switch (flag) {
|
|
||||||
case Flag::Contacts: return st::windowFilterTypeContacts;
|
|
||||||
case Flag::NonContacts: return st::windowFilterTypeNonContacts;
|
|
||||||
case Flag::Groups: return st::windowFilterTypeGroups;
|
|
||||||
case Flag::Channels: return st::windowFilterTypeChannels;
|
|
||||||
case Flag::Bots: return st::windowFilterTypeBots;
|
|
||||||
case Flag::NoMuted: return st::windowFilterTypeNoMuted;
|
|
||||||
case Flag::NoArchived: return st::windowFilterTypeNoArchived;
|
|
||||||
case Flag::NoRead: return st::windowFilterTypeNoRead;
|
|
||||||
}
|
|
||||||
Unexpected("Flag in icon paintFlagIcon.");
|
|
||||||
}();
|
|
||||||
const auto size = st::windowFilterSmallItem.photoSize;
|
|
||||||
const auto rect = QRect(left, top, size, size);
|
|
||||||
auto hq = PainterHighQualityEnabler(p);
|
|
||||||
p.setBrush(color->b);
|
|
||||||
p.setPen(Qt::NoPen);
|
|
||||||
p.drawEllipse(rect);
|
|
||||||
icon.paintInCenter(p, rect);
|
|
||||||
}
|
|
||||||
|
|
||||||
void FilterChatsPreview::removeFlag(Flag flag) {
|
void FilterChatsPreview::removeFlag(Flag flag) {
|
||||||
const auto i = ranges::find(_removeFlag, flag, &FlagButton::flag);
|
const auto i = ranges::find(_removeFlag, flag, &FlagButton::flag);
|
||||||
Assert(i != end(_removeFlag));
|
Assert(i != end(_removeFlag));
|
||||||
|
@ -307,7 +258,16 @@ void EditExceptions(
|
||||||
not_null<Data::ChatFilter*> data,
|
not_null<Data::ChatFilter*> data,
|
||||||
Fn<void()> refresh) {
|
Fn<void()> refresh) {
|
||||||
const auto include = (options & Flag::Contacts) != Flags(0);
|
const auto include = (options & Flag::Contacts) != Flags(0);
|
||||||
const auto initBox = [=](not_null<PeerListBox*> box) {
|
auto controller = std::make_unique<EditFilterChatsListController>(
|
||||||
|
window,
|
||||||
|
(include
|
||||||
|
? tr::lng_filters_include_title()
|
||||||
|
: tr::lng_filters_exclude_title()),
|
||||||
|
options,
|
||||||
|
data->flags() & options,
|
||||||
|
include ? data->always() : data->never());
|
||||||
|
const auto rawController = controller.get();
|
||||||
|
auto initBox = [=](not_null<PeerListBox*> box) {
|
||||||
box->addButton(tr::lng_settings_save(), crl::guard(context, [=] {
|
box->addButton(tr::lng_settings_save(), crl::guard(context, [=] {
|
||||||
const auto peers = box->peerListCollectSelectedRows();
|
const auto peers = box->peerListCollectSelectedRows();
|
||||||
auto &&histories = ranges::view::all(
|
auto &&histories = ranges::view::all(
|
||||||
|
@ -326,7 +286,7 @@ void EditExceptions(
|
||||||
*data = Data::ChatFilter(
|
*data = Data::ChatFilter(
|
||||||
data->id(),
|
data->id(),
|
||||||
data->title(),
|
data->title(),
|
||||||
data->flags(),
|
(data->flags() & ~options) | rawController->chosenOptions(),
|
||||||
include ? std::move(changed) : std::move(removeFrom),
|
include ? std::move(changed) : std::move(removeFrom),
|
||||||
include ? std::move(removeFrom) : std::move(changed));
|
include ? std::move(removeFrom) : std::move(changed));
|
||||||
refresh();
|
refresh();
|
||||||
|
@ -336,14 +296,7 @@ void EditExceptions(
|
||||||
};
|
};
|
||||||
window->window().show(
|
window->window().show(
|
||||||
Box<PeerListBox>(
|
Box<PeerListBox>(
|
||||||
std::make_unique<EditFilterChatsListController>(
|
std::move(controller),
|
||||||
window,
|
|
||||||
(include
|
|
||||||
? tr::lng_filters_include_title()
|
|
||||||
: tr::lng_filters_exclude_title()),
|
|
||||||
options,
|
|
||||||
data->flags() & options,
|
|
||||||
include ? data->always() : data->never()),
|
|
||||||
std::move(initBox)),
|
std::move(initBox)),
|
||||||
Ui::LayerOption::KeepOther);
|
Ui::LayerOption::KeepOther);
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,13 +10,291 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "history/history.h"
|
#include "history/history.h"
|
||||||
#include "window/window_session_controller.h"
|
#include "window/window_session_controller.h"
|
||||||
#include "lang/lang_keys.h"
|
#include "lang/lang_keys.h"
|
||||||
|
#include "ui/widgets/labels.h"
|
||||||
|
#include "ui/wrap/vertical_layout.h"
|
||||||
|
#include "base/object_ptr.h"
|
||||||
|
#include "styles/style_window.h"
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
constexpr auto kMaxExceptions = 100;
|
constexpr auto kMaxExceptions = 100;
|
||||||
|
|
||||||
|
using Flag = Data::ChatFilter::Flag;
|
||||||
|
using Flags = Data::ChatFilter::Flags;
|
||||||
|
|
||||||
|
constexpr auto kAllTypes = {
|
||||||
|
Flag::Contacts,
|
||||||
|
Flag::NonContacts,
|
||||||
|
Flag::Groups,
|
||||||
|
Flag::Channels,
|
||||||
|
Flag::Bots,
|
||||||
|
Flag::NoMuted,
|
||||||
|
Flag::NoArchived,
|
||||||
|
Flag::NoRead
|
||||||
|
};
|
||||||
|
|
||||||
|
class TypeRow final : public PeerListRow {
|
||||||
|
public:
|
||||||
|
explicit TypeRow(Flag flag);
|
||||||
|
|
||||||
|
QString generateName() override;
|
||||||
|
QString generateShortName() override;
|
||||||
|
void paintEntityUserpicLeft(
|
||||||
|
Painter &p,
|
||||||
|
int x,
|
||||||
|
int y,
|
||||||
|
int outerWidth,
|
||||||
|
int size) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
[[nodiscard]] Flag flag() const;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
class TypeDelegate final : public PeerListContentDelegate {
|
||||||
|
public:
|
||||||
|
void peerListSetTitle(rpl::producer<QString> title) override;
|
||||||
|
void peerListSetAdditionalTitle(rpl::producer<QString> title) override;
|
||||||
|
bool peerListIsRowChecked(not_null<PeerListRow*> row) override;
|
||||||
|
int peerListSelectedRowsCount() override;
|
||||||
|
std::vector<not_null<PeerData*>> peerListCollectSelectedRows() override;
|
||||||
|
void peerListScrollToTop() override;
|
||||||
|
void peerListAddSelectedPeerInBunch(
|
||||||
|
not_null<PeerData*> peer) override;
|
||||||
|
void peerListAddSelectedRowInBunch(not_null<PeerListRow*> row) override;
|
||||||
|
void peerListFinishSelectedRowsBunch() override;
|
||||||
|
void peerListSetDescription(
|
||||||
|
object_ptr<Ui::FlatLabel> description) override;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
class TypeController final : public PeerListController {
|
||||||
|
public:
|
||||||
|
TypeController(
|
||||||
|
not_null<Main::Session*> session,
|
||||||
|
Flags options,
|
||||||
|
Flags selected);
|
||||||
|
|
||||||
|
Main::Session &session() const override;
|
||||||
|
void prepare() override;
|
||||||
|
void rowClicked(not_null<PeerListRow*> row) override;
|
||||||
|
|
||||||
|
[[nodiscard]] rpl::producer<Flags> selectedOptions() const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
[[nodiscard]] std::unique_ptr<PeerListRow> createRow(Flag flag) const;
|
||||||
|
[[nodiscard]] Flags collectSelectedOptions() const;
|
||||||
|
|
||||||
|
const not_null<Main::Session*> _session;
|
||||||
|
Flags _options;
|
||||||
|
|
||||||
|
rpl::event_stream<> _selectionChanged;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
[[nodiscard]] object_ptr<Ui::RpWidget> CreateSectionSubtitle(
|
||||||
|
not_null<QWidget*> parent,
|
||||||
|
rpl::producer<QString> text) {
|
||||||
|
auto result = object_ptr<Ui::FixedHeightWidget>(
|
||||||
|
parent,
|
||||||
|
st::searchedBarHeight);
|
||||||
|
|
||||||
|
const auto raw = result.data();
|
||||||
|
raw->paintRequest(
|
||||||
|
) | rpl::start_with_next([=](QRect clip) {
|
||||||
|
auto p = QPainter(raw);
|
||||||
|
p.fillRect(clip, st::searchedBarBg);
|
||||||
|
}, raw->lifetime());
|
||||||
|
|
||||||
|
const auto label = Ui::CreateChild<Ui::FlatLabel>(
|
||||||
|
raw,
|
||||||
|
std::move(text),
|
||||||
|
st::windowFilterChatsSectionSubtitle);
|
||||||
|
raw->widthValue(
|
||||||
|
) | rpl::start_with_next([=](int width) {
|
||||||
|
const auto padding = st::windowFilterChatsSectionSubtitlePadding;
|
||||||
|
const auto available = width - padding.left() - padding.right();
|
||||||
|
label->resizeToNaturalWidth(available);
|
||||||
|
label->moveToLeft(padding.left(), padding.top(), width);
|
||||||
|
}, label->lifetime());
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
[[nodiscard]] uint64 TypeId(Flag flag) {
|
||||||
|
return PeerIdFakeShift | static_cast<uint64>(flag);
|
||||||
|
}
|
||||||
|
|
||||||
|
TypeRow::TypeRow(Flag flag) : PeerListRow(TypeId(flag)) {
|
||||||
|
}
|
||||||
|
|
||||||
|
QString TypeRow::generateName() {
|
||||||
|
return FilterChatsTypeName(flag());
|
||||||
|
}
|
||||||
|
|
||||||
|
QString TypeRow::generateShortName() {
|
||||||
|
return generateName();
|
||||||
|
}
|
||||||
|
|
||||||
|
void TypeRow::paintEntityUserpicLeft(
|
||||||
|
Painter &p,
|
||||||
|
int x,
|
||||||
|
int y,
|
||||||
|
int outerWidth,
|
||||||
|
int size) {
|
||||||
|
PaintFilterChatsTypeIcon(p, flag(), x, y, outerWidth, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
Flag TypeRow::flag() const {
|
||||||
|
return static_cast<Flag>(id() & 0xFF);
|
||||||
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto TypeDelegate::peerListCollectSelectedRows()
|
||||||
|
-> std::vector<not_null<PeerData*>> {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
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(
|
||||||
|
not_null<Main::Session*> session,
|
||||||
|
Flags options,
|
||||||
|
Flags selected)
|
||||||
|
: _session(session)
|
||||||
|
, _options(options) {
|
||||||
|
}
|
||||||
|
|
||||||
|
Main::Session &TypeController::session() const {
|
||||||
|
return *_session;
|
||||||
|
}
|
||||||
|
|
||||||
|
void TypeController::prepare() {
|
||||||
|
for (const auto flag : kAllTypes) {
|
||||||
|
if (_options & flag) {
|
||||||
|
delegate()->peerListAppendRow(createRow(flag));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
delegate()->peerListRefreshRows();
|
||||||
|
}
|
||||||
|
|
||||||
|
Flags TypeController::collectSelectedOptions() const {
|
||||||
|
auto result = Flags();
|
||||||
|
for (const auto flag : kAllTypes) {
|
||||||
|
if (const auto row = delegate()->peerListFindRow(TypeId(flag))) {
|
||||||
|
if (row->checked()) {
|
||||||
|
result |= flag;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
void TypeController::rowClicked(not_null<PeerListRow*> row) {
|
||||||
|
delegate()->peerListSetRowChecked(row, !row->checked());
|
||||||
|
_selectionChanged.fire({});
|
||||||
|
}
|
||||||
|
|
||||||
|
std::unique_ptr<PeerListRow> TypeController::createRow(Flag flag) const {
|
||||||
|
return std::make_unique<TypeRow>(flag);
|
||||||
|
}
|
||||||
|
|
||||||
|
rpl::producer<Flags> TypeController::selectedOptions() const {
|
||||||
|
return _selectionChanged.events_starting_with(
|
||||||
|
{}
|
||||||
|
) | rpl::map([=] {
|
||||||
|
return collectSelectedOptions();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
|
[[nodiscard]] QString FilterChatsTypeName(Flag flag) {
|
||||||
|
switch (flag) {
|
||||||
|
case Flag::Contacts: return tr::lng_filters_type_contacts(tr::now);
|
||||||
|
case Flag::NonContacts:
|
||||||
|
return tr::lng_filters_type_non_contacts(tr::now);
|
||||||
|
case Flag::Groups: return tr::lng_filters_type_groups(tr::now);
|
||||||
|
case Flag::Channels: return tr::lng_filters_type_channels(tr::now);
|
||||||
|
case Flag::Bots: return tr::lng_filters_type_bots(tr::now);
|
||||||
|
case Flag::NoMuted: return tr::lng_filters_type_no_muted(tr::now);
|
||||||
|
case Flag::NoArchived: return tr::lng_filters_type_no_archived(tr::now);
|
||||||
|
case Flag::NoRead: return tr::lng_filters_type_no_read(tr::now);
|
||||||
|
}
|
||||||
|
Unexpected("Flag in TypeName.");
|
||||||
|
}
|
||||||
|
|
||||||
|
void PaintFilterChatsTypeIcon(
|
||||||
|
Painter &p,
|
||||||
|
Data::ChatFilter::Flag flag,
|
||||||
|
int x,
|
||||||
|
int y,
|
||||||
|
int outerWidth,
|
||||||
|
int size) {
|
||||||
|
const auto &color = [&]() -> const style::color& {
|
||||||
|
switch (flag) {
|
||||||
|
case Flag::Contacts: return st::historyPeer4UserpicBg;
|
||||||
|
case Flag::NonContacts: return st::historyPeer7UserpicBg;
|
||||||
|
case Flag::Groups: return st::historyPeer2UserpicBg;
|
||||||
|
case Flag::Channels: return st::historyPeer1UserpicBg;
|
||||||
|
case Flag::Bots: return st::historyPeer6UserpicBg;
|
||||||
|
case Flag::NoMuted: return st::historyPeer6UserpicBg;
|
||||||
|
case Flag::NoArchived: return st::historyPeer4UserpicBg;
|
||||||
|
case Flag::NoRead: return st::historyPeer7UserpicBg;
|
||||||
|
}
|
||||||
|
Unexpected("Flag in color paintFlagIcon.");
|
||||||
|
}();
|
||||||
|
const auto &icon = [&]() -> const style::icon& {
|
||||||
|
switch (flag) {
|
||||||
|
case Flag::Contacts: return st::windowFilterTypeContacts;
|
||||||
|
case Flag::NonContacts: return st::windowFilterTypeNonContacts;
|
||||||
|
case Flag::Groups: return st::windowFilterTypeGroups;
|
||||||
|
case Flag::Channels: return st::windowFilterTypeChannels;
|
||||||
|
case Flag::Bots: return st::windowFilterTypeBots;
|
||||||
|
case Flag::NoMuted: return st::windowFilterTypeNoMuted;
|
||||||
|
case Flag::NoArchived: return st::windowFilterTypeNoArchived;
|
||||||
|
case Flag::NoRead: return st::windowFilterTypeNoRead;
|
||||||
|
}
|
||||||
|
Unexpected("Flag in icon paintFlagIcon.");
|
||||||
|
}();
|
||||||
|
const auto rect = style::rtlrect(x, y, size, size, outerWidth);
|
||||||
|
auto hq = PainterHighQualityEnabler(p);
|
||||||
|
p.setBrush(color->b);
|
||||||
|
p.setPen(Qt::NoPen);
|
||||||
|
p.drawEllipse(rect);
|
||||||
|
icon.paintInCenter(p, rect);
|
||||||
|
}
|
||||||
|
|
||||||
EditFilterChatsListController::EditFilterChatsListController(
|
EditFilterChatsListController::EditFilterChatsListController(
|
||||||
not_null<Window::SessionNavigation*> navigation,
|
not_null<Window::SessionNavigation*> navigation,
|
||||||
rpl::producer<QString> title,
|
rpl::producer<QString> title,
|
||||||
|
@ -26,7 +304,9 @@ EditFilterChatsListController::EditFilterChatsListController(
|
||||||
: ChatsListBoxController(navigation)
|
: ChatsListBoxController(navigation)
|
||||||
, _navigation(navigation)
|
, _navigation(navigation)
|
||||||
, _title(std::move(title))
|
, _title(std::move(title))
|
||||||
, _peers(peers) {
|
, _peers(peers)
|
||||||
|
, _options(options)
|
||||||
|
, _selected(selected) {
|
||||||
}
|
}
|
||||||
|
|
||||||
Main::Session &EditFilterChatsListController::session() const {
|
Main::Session &EditFilterChatsListController::session() const {
|
||||||
|
@ -48,8 +328,45 @@ void EditFilterChatsListController::itemDeselectedHook(
|
||||||
|
|
||||||
void EditFilterChatsListController::prepareViewHook() {
|
void EditFilterChatsListController::prepareViewHook() {
|
||||||
delegate()->peerListSetTitle(std::move(_title));
|
delegate()->peerListSetTitle(std::move(_title));
|
||||||
delegate()->peerListAddSelectedRows(
|
delegate()->peerListAddSelectedPeers(
|
||||||
_peers | ranges::view::transform(&History::peer));
|
_peers | ranges::view::transform(&History::peer));
|
||||||
|
delegate()->peerListSetAboveWidget(prepareTypesList());
|
||||||
|
}
|
||||||
|
|
||||||
|
object_ptr<Ui::RpWidget> EditFilterChatsListController::prepareTypesList() {
|
||||||
|
auto result = object_ptr<Ui::VerticalLayout>((QWidget*)nullptr);
|
||||||
|
const auto container = result.data();
|
||||||
|
container->add(CreateSectionSubtitle(
|
||||||
|
container,
|
||||||
|
tr::lng_filters_edit_types()));
|
||||||
|
const auto delegate = container->lifetime().make_state<TypeDelegate>();
|
||||||
|
const auto controller = container->lifetime().make_state<TypeController>(
|
||||||
|
&session(),
|
||||||
|
_options,
|
||||||
|
_selected);
|
||||||
|
const auto content = result->add(object_ptr<PeerListContent>(
|
||||||
|
container,
|
||||||
|
controller,
|
||||||
|
st::windowFilterSmallList));
|
||||||
|
delegate->setContent(content);
|
||||||
|
controller->setDelegate(delegate);
|
||||||
|
for (const auto flag : kAllTypes) {
|
||||||
|
if (_selected & flag) {
|
||||||
|
if (const auto row = delegate->peerListFindRow(TypeId(flag))) {
|
||||||
|
content->changeCheckState(row, true, anim::type::instant);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
container->add(CreateSectionSubtitle(
|
||||||
|
container,
|
||||||
|
tr::lng_filters_edit_chats()));
|
||||||
|
|
||||||
|
controller->selectedOptions(
|
||||||
|
) | rpl::start_with_next([=](Flags selected) {
|
||||||
|
_selected = selected;
|
||||||
|
}, _lifetime);
|
||||||
|
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto EditFilterChatsListController::createRow(not_null<History*> history)
|
auto EditFilterChatsListController::createRow(not_null<History*> history)
|
||||||
|
@ -60,7 +377,5 @@ auto EditFilterChatsListController::createRow(not_null<History*> history)
|
||||||
void EditFilterChatsListController::updateTitle() {
|
void EditFilterChatsListController::updateTitle() {
|
||||||
const auto count = delegate()->peerListSelectedRowsCount();
|
const auto count = delegate()->peerListSelectedRowsCount();
|
||||||
const auto additional = qsl("%1 / %2").arg(count).arg(kMaxExceptions);
|
const auto additional = qsl("%1 / %2").arg(count).arg(kMaxExceptions);
|
||||||
delegate()->peerListSetTitle(tr::lng_profile_add_participant());
|
|
||||||
delegate()->peerListSetAdditionalTitle(rpl::single(additional));
|
delegate()->peerListSetAdditionalTitle(rpl::single(additional));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,6 +24,17 @@ namespace Main {
|
||||||
class Session;
|
class Session;
|
||||||
} // namespace Main
|
} // namespace Main
|
||||||
|
|
||||||
|
class Painter;
|
||||||
|
|
||||||
|
[[nodiscard]] QString FilterChatsTypeName(Data::ChatFilter::Flag flag);
|
||||||
|
void PaintFilterChatsTypeIcon(
|
||||||
|
Painter &p,
|
||||||
|
Data::ChatFilter::Flag flag,
|
||||||
|
int x,
|
||||||
|
int y,
|
||||||
|
int outerWidth,
|
||||||
|
int size);
|
||||||
|
|
||||||
class EditFilterChatsListController final : public ChatsListBoxController {
|
class EditFilterChatsListController final : public ChatsListBoxController {
|
||||||
public:
|
public:
|
||||||
using Flag = Data::ChatFilter::Flag;
|
using Flag = Data::ChatFilter::Flag;
|
||||||
|
@ -36,20 +47,27 @@ public:
|
||||||
Flags selected,
|
Flags selected,
|
||||||
const base::flat_set<not_null<History*>> &peers);
|
const base::flat_set<not_null<History*>> &peers);
|
||||||
|
|
||||||
Main::Session &session() const override;
|
[[nodiscard]] Main::Session &session() const override;
|
||||||
|
[[nodiscard]] Flags chosenOptions() const {
|
||||||
|
return _selected;
|
||||||
|
}
|
||||||
|
|
||||||
void rowClicked(not_null<PeerListRow*> row) override;
|
void rowClicked(not_null<PeerListRow*> row) override;
|
||||||
void itemDeselectedHook(not_null<PeerData*> peer) override;
|
void itemDeselectedHook(not_null<PeerData*> peer) override;
|
||||||
|
|
||||||
protected:
|
private:
|
||||||
void prepareViewHook() override;
|
void prepareViewHook() override;
|
||||||
std::unique_ptr<Row> createRow(not_null<History*> history) override;
|
std::unique_ptr<Row> createRow(not_null<History*> history) override;
|
||||||
|
[[nodiscard]] object_ptr<Ui::RpWidget> prepareTypesList();
|
||||||
|
|
||||||
private:
|
|
||||||
void updateTitle();
|
void updateTitle();
|
||||||
|
|
||||||
const not_null<Window::SessionNavigation*> _navigation;
|
const not_null<Window::SessionNavigation*> _navigation;
|
||||||
rpl::producer<QString> _title;
|
rpl::producer<QString> _title;
|
||||||
base::flat_set<not_null<History*>> _peers;
|
base::flat_set<not_null<History*>> _peers;
|
||||||
|
Flags _options;
|
||||||
|
Flags _selected;
|
||||||
|
|
||||||
|
rpl::lifetime _lifetime;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
|
@ -33,10 +33,9 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
|
|
||||||
#include <rpl/range.h>
|
#include <rpl/range.h>
|
||||||
|
|
||||||
auto PaintUserpicCallback(
|
PaintRoundImageCallback PaintUserpicCallback(
|
||||||
not_null<PeerData*> peer,
|
not_null<PeerData*> peer,
|
||||||
bool respectSavedMessagesChat)
|
bool respectSavedMessagesChat) {
|
||||||
-> Fn<void(Painter &p, int x, int y, int outerWidth, int size)> {
|
|
||||||
if (respectSavedMessagesChat && peer->isSelf()) {
|
if (respectSavedMessagesChat && peer->isSelf()) {
|
||||||
return [](Painter &p, int x, int y, int outerWidth, int size) {
|
return [](Painter &p, int x, int y, int outerWidth, int size) {
|
||||||
Ui::EmptyUserpic::PaintSavedMessages(p, x, y, outerWidth, size);
|
Ui::EmptyUserpic::PaintSavedMessages(p, x, y, outerWidth, size);
|
||||||
|
@ -77,7 +76,7 @@ void PeerListBox::createMultiSelect() {
|
||||||
_select->entity()->setItemRemovedCallback([=](uint64 itemId) {
|
_select->entity()->setItemRemovedCallback([=](uint64 itemId) {
|
||||||
if (const auto peer = _controller->session().data().peerLoaded(itemId)) {
|
if (const auto peer = _controller->session().data().peerLoaded(itemId)) {
|
||||||
if (const auto row = peerListFindRow(peer->id)) {
|
if (const auto row = peerListFindRow(peer->id)) {
|
||||||
content()->changeCheckState(row, false, PeerListRow::SetStyle::Animated);
|
content()->changeCheckState(row, false, anim::type::normal);
|
||||||
update();
|
update();
|
||||||
}
|
}
|
||||||
_controller->itemDeselectedHook(peer);
|
_controller->itemDeselectedHook(peer);
|
||||||
|
@ -185,9 +184,8 @@ void PeerListBox::setInnerFocus() {
|
||||||
void PeerListBox::peerListSetRowChecked(
|
void PeerListBox::peerListSetRowChecked(
|
||||||
not_null<PeerListRow*> row,
|
not_null<PeerListRow*> row,
|
||||||
bool checked) {
|
bool checked) {
|
||||||
auto peer = row->peer();
|
|
||||||
if (checked) {
|
if (checked) {
|
||||||
addSelectItem(peer, PeerListRow::SetStyle::Animated);
|
addSelectItem(row, anim::type::normal);
|
||||||
PeerListContentDelegate::peerListSetRowChecked(row, checked);
|
PeerListContentDelegate::peerListSetRowChecked(row, checked);
|
||||||
peerListUpdateRow(row);
|
peerListUpdateRow(row);
|
||||||
|
|
||||||
|
@ -195,7 +193,7 @@ void PeerListBox::peerListSetRowChecked(
|
||||||
_select->entity()->clearQuery();
|
_select->entity()->clearQuery();
|
||||||
} else {
|
} else {
|
||||||
// The itemRemovedCallback will call changeCheckState() here.
|
// The itemRemovedCallback will call changeCheckState() here.
|
||||||
_select->entity()->removeItem(peer->id);
|
_select->entity()->removeItem(row->id());
|
||||||
peerListUpdateRow(row);
|
peerListUpdateRow(row);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -295,38 +293,62 @@ int PeerListController::contentWidth() const {
|
||||||
return st::boxWideWidth;
|
return st::boxWideWidth;
|
||||||
}
|
}
|
||||||
|
|
||||||
void PeerListBox::addSelectItem(not_null<PeerData*> peer, PeerListRow::SetStyle style) {
|
void PeerListBox::addSelectItem(
|
||||||
if (!_select) {
|
not_null<PeerData*> peer,
|
||||||
createMultiSelect();
|
anim::type animated) {
|
||||||
_select->hide(anim::type::instant);
|
|
||||||
}
|
|
||||||
const auto respect = _controller->respectSavedMessagesChat();
|
const auto respect = _controller->respectSavedMessagesChat();
|
||||||
const auto text = (respect && peer->isSelf())
|
const auto text = (respect && peer->isSelf())
|
||||||
? tr::lng_saved_short(tr::now)
|
? tr::lng_saved_short(tr::now)
|
||||||
: peer->shortName();
|
: peer->shortName();
|
||||||
const auto callback = PaintUserpicCallback(peer, respect);
|
addSelectItem(
|
||||||
if (style == PeerListRow::SetStyle::Fast) {
|
peer->id,
|
||||||
|
text,
|
||||||
|
PaintUserpicCallback(peer, respect),
|
||||||
|
animated);
|
||||||
|
}
|
||||||
|
|
||||||
|
void PeerListBox::addSelectItem(
|
||||||
|
not_null<PeerListRow*> row,
|
||||||
|
anim::type animated) {
|
||||||
|
addSelectItem(
|
||||||
|
row->id(),
|
||||||
|
row->generateShortName(),
|
||||||
|
row->generatePaintUserpicCallback(),
|
||||||
|
animated);
|
||||||
|
}
|
||||||
|
|
||||||
|
void PeerListBox::addSelectItem(
|
||||||
|
uint64 itemId,
|
||||||
|
const QString &text,
|
||||||
|
Ui::MultiSelect::PaintRoundImage paintUserpic,
|
||||||
|
anim::type animated) {
|
||||||
|
if (!_select) {
|
||||||
|
createMultiSelect();
|
||||||
|
_select->hide(anim::type::instant);
|
||||||
|
}
|
||||||
|
if (animated == anim::type::instant) {
|
||||||
_select->entity()->addItemInBunch(
|
_select->entity()->addItemInBunch(
|
||||||
peer->id,
|
itemId,
|
||||||
text,
|
text,
|
||||||
st::activeButtonBg,
|
st::activeButtonBg,
|
||||||
std::move(callback));
|
std::move(paintUserpic));
|
||||||
} else {
|
} else {
|
||||||
_select->entity()->addItem(
|
_select->entity()->addItem(
|
||||||
peer->id,
|
itemId,
|
||||||
text,
|
text,
|
||||||
st::activeButtonBg,
|
st::activeButtonBg,
|
||||||
std::move(callback));
|
std::move(paintUserpic));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void PeerListBox::peerListFinishSelectedRowsBunch() {
|
void PeerListBox::peerListFinishSelectedRowsBunch() {
|
||||||
Expects(_select != nullptr);
|
Expects(_select != nullptr);
|
||||||
|
|
||||||
_select->entity()->finishItemsBunch();
|
_select->entity()->finishItemsBunch();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool PeerListBox::peerListIsRowSelected(not_null<PeerData*> peer) {
|
bool PeerListBox::peerListIsRowChecked(not_null<PeerListRow*> row) {
|
||||||
return _select ? _select->entity()->hasItem(peer->id) : false;
|
return _select ? _select->entity()->hasItem(row->id()) : false;
|
||||||
}
|
}
|
||||||
|
|
||||||
int PeerListBox::peerListSelectedRowsCount() {
|
int PeerListBox::peerListSelectedRowsCount() {
|
||||||
|
@ -348,7 +370,8 @@ auto PeerListBox::peerListCollectSelectedRows()
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
PeerListRow::PeerListRow(not_null<PeerData*> peer) : PeerListRow(peer, peer->id) {
|
PeerListRow::PeerListRow(not_null<PeerData*> peer)
|
||||||
|
: PeerListRow(peer, peer->id) {
|
||||||
}
|
}
|
||||||
|
|
||||||
PeerListRow::PeerListRow(not_null<PeerData*> peer, PeerListRowId id)
|
PeerListRow::PeerListRow(not_null<PeerData*> peer, PeerListRowId id)
|
||||||
|
@ -359,6 +382,15 @@ PeerListRow::PeerListRow(not_null<PeerData*> peer, PeerListRowId id)
|
||||||
, _isSavedMessagesChat(false) {
|
, _isSavedMessagesChat(false) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
PeerListRow::PeerListRow(PeerListRowId id)
|
||||||
|
: _id(id)
|
||||||
|
, _initialized(false)
|
||||||
|
, _isSearchResult(false)
|
||||||
|
, _isSavedMessagesChat(false) {
|
||||||
|
}
|
||||||
|
|
||||||
|
PeerListRow::~PeerListRow() = default;
|
||||||
|
|
||||||
bool PeerListRow::checked() const {
|
bool PeerListRow::checked() const {
|
||||||
return _checkbox && _checkbox->checked();
|
return _checkbox && _checkbox->checked();
|
||||||
}
|
}
|
||||||
|
@ -375,7 +407,7 @@ void PeerListRow::clearCustomStatus() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void PeerListRow::refreshStatus() {
|
void PeerListRow::refreshStatus() {
|
||||||
if (!_initialized || _statusType == StatusType::Custom) {
|
if (!_initialized || special() || _statusType == StatusType::Custom) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
_statusType = StatusType::LastSeen;
|
_statusType = StatusType::LastSeen;
|
||||||
|
@ -417,11 +449,38 @@ void PeerListRow::refreshName(const style::PeerListItem &st) {
|
||||||
}
|
}
|
||||||
const auto text = _isSavedMessagesChat
|
const auto text = _isSavedMessagesChat
|
||||||
? tr::lng_saved_messages(tr::now)
|
? tr::lng_saved_messages(tr::now)
|
||||||
: peer()->name;
|
: generateName();
|
||||||
_name.setText(st.nameStyle, text, Ui::NameTextOptions());
|
_name.setText(st.nameStyle, text, Ui::NameTextOptions());
|
||||||
}
|
}
|
||||||
|
|
||||||
PeerListRow::~PeerListRow() = default;
|
QString PeerListRow::generateName() {
|
||||||
|
return peer()->name;
|
||||||
|
}
|
||||||
|
|
||||||
|
QString PeerListRow::generateShortName() {
|
||||||
|
return _isSavedMessagesChat
|
||||||
|
? tr::lng_saved_short(tr::now)
|
||||||
|
: peer()->shortName();
|
||||||
|
}
|
||||||
|
|
||||||
|
PaintRoundImageCallback PeerListRow::generatePaintUserpicCallback() {
|
||||||
|
return [=](Painter &p, int x, int y, int outerWidth, int size) {
|
||||||
|
paintEntityUserpicLeft(p, x, y, outerWidth, size);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
void PeerListRow::paintEntityUserpicLeft(
|
||||||
|
Painter &p,
|
||||||
|
int x,
|
||||||
|
int y,
|
||||||
|
int outerWidth,
|
||||||
|
int size) {
|
||||||
|
if (_isSavedMessagesChat) {
|
||||||
|
Ui::EmptyUserpic::PaintSavedMessages(p, x, y, outerWidth, size);
|
||||||
|
} else {
|
||||||
|
peer()->paintUserpicLeft(p, x, y, outerWidth, size);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void PeerListRow::invalidatePixmapsCache() {
|
void PeerListRow::invalidatePixmapsCache() {
|
||||||
if (_checkbox) {
|
if (_checkbox) {
|
||||||
|
@ -430,7 +489,9 @@ void PeerListRow::invalidatePixmapsCache() {
|
||||||
}
|
}
|
||||||
|
|
||||||
int PeerListRow::nameIconWidth() const {
|
int PeerListRow::nameIconWidth() const {
|
||||||
return _peer->isVerified() ? st::dialogsVerifiedIcon.width() : 0;
|
return (special() || !_peer->isVerified())
|
||||||
|
? 0
|
||||||
|
: st::dialogsVerifiedIcon.width();
|
||||||
}
|
}
|
||||||
|
|
||||||
void PeerListRow::paintNameIcon(
|
void PeerListRow::paintNameIcon(
|
||||||
|
@ -490,10 +551,8 @@ void PeerListRow::paintUserpic(
|
||||||
paintDisabledCheckUserpic(p, st, x, y, outerWidth);
|
paintDisabledCheckUserpic(p, st, x, y, outerWidth);
|
||||||
} else if (_checkbox) {
|
} else if (_checkbox) {
|
||||||
_checkbox->paint(p, x, y, outerWidth);
|
_checkbox->paint(p, x, y, outerWidth);
|
||||||
} else if (_isSavedMessagesChat) {
|
|
||||||
Ui::EmptyUserpic::PaintSavedMessages(p, x, y, outerWidth, st.photoSize);
|
|
||||||
} else {
|
} else {
|
||||||
peer()->paintUserpicLeft(p, x, y, outerWidth, st.photoSize);
|
paintEntityUserpicLeft(p, x, y, outerWidth, st.photoSize);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -558,18 +617,19 @@ void PeerListRow::lazyInitialize(const style::PeerListItem &st) {
|
||||||
refreshStatus();
|
refreshStatus();
|
||||||
}
|
}
|
||||||
|
|
||||||
void PeerListRow::createCheckbox(Fn<void()> updateCallback) {
|
void PeerListRow::createCheckbox(
|
||||||
|
const style::RoundImageCheckbox &st,
|
||||||
|
Fn<void()> updateCallback) {
|
||||||
_checkbox = std::make_unique<Ui::RoundImageCheckbox>(
|
_checkbox = std::make_unique<Ui::RoundImageCheckbox>(
|
||||||
st::contactsPhotoCheckbox,
|
st,
|
||||||
std::move(updateCallback),
|
std::move(updateCallback),
|
||||||
PaintUserpicCallback(_peer, _isSavedMessagesChat));
|
generatePaintUserpicCallback());
|
||||||
}
|
}
|
||||||
|
|
||||||
void PeerListRow::setCheckedInternal(bool checked, SetStyle style) {
|
void PeerListRow::setCheckedInternal(bool checked, anim::type animated) {
|
||||||
Expects(_checkbox != nullptr);
|
Expects(_checkbox != nullptr);
|
||||||
using CheckboxStyle = Ui::RoundCheckbox::SetStyle;
|
|
||||||
auto speed = (style == SetStyle::Animated) ? CheckboxStyle::Animated : CheckboxStyle::Fast;
|
_checkbox->setChecked(checked, animated);
|
||||||
_checkbox->setChecked(checked, speed);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
PeerListContent::PeerListContent(
|
PeerListContent::PeerListContent(
|
||||||
|
@ -636,25 +696,30 @@ void PeerListContent::appendFoundRow(not_null<PeerListRow*> row) {
|
||||||
void PeerListContent::changeCheckState(
|
void PeerListContent::changeCheckState(
|
||||||
not_null<PeerListRow*> row,
|
not_null<PeerListRow*> row,
|
||||||
bool checked,
|
bool checked,
|
||||||
PeerListRow::SetStyle style) {
|
anim::type animated) {
|
||||||
row->setChecked(
|
row->setChecked(
|
||||||
checked,
|
checked,
|
||||||
style,
|
_st.item.checkbox,
|
||||||
[this, row] { updateRow(row); });
|
animated,
|
||||||
|
[=] { updateRow(row); });
|
||||||
}
|
}
|
||||||
|
|
||||||
void PeerListContent::addRowEntry(not_null<PeerListRow*> row) {
|
void PeerListContent::addRowEntry(not_null<PeerListRow*> row) {
|
||||||
if (_controller->respectSavedMessagesChat() && row->peer()->isSelf()) {
|
if (_controller->respectSavedMessagesChat()
|
||||||
|
&& !row->special()
|
||||||
|
&& row->peer()->isSelf()) {
|
||||||
row->setIsSavedMessagesChat(true);
|
row->setIsSavedMessagesChat(true);
|
||||||
}
|
}
|
||||||
_rowsById.emplace(row->id(), row);
|
_rowsById.emplace(row->id(), row);
|
||||||
_rowsByPeer[row->peer()].push_back(row);
|
if (!row->special()) {
|
||||||
|
_rowsByPeer[row->peer()].push_back(row);
|
||||||
|
}
|
||||||
if (addingToSearchIndex()) {
|
if (addingToSearchIndex()) {
|
||||||
addToSearchIndex(row);
|
addToSearchIndex(row);
|
||||||
}
|
}
|
||||||
if (_controller->isRowSelected(row->peer())) {
|
if (_controller->isRowSelected(row)) {
|
||||||
Assert(row->id() == row->peer()->id);
|
Assert(row->special() || row->id() == row->peer()->id);
|
||||||
changeCheckState(row, true, PeerListRow::SetStyle::Fast);
|
changeCheckState(row, true, anim::type::instant);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -670,7 +735,7 @@ bool PeerListContent::addingToSearchIndex() const {
|
||||||
}
|
}
|
||||||
|
|
||||||
void PeerListContent::addToSearchIndex(not_null<PeerListRow*> row) {
|
void PeerListContent::addToSearchIndex(not_null<PeerListRow*> row) {
|
||||||
if (row->isSearchResult()) {
|
if (row->isSearchResult() || row->special()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -763,8 +828,10 @@ void PeerListContent::removeRow(not_null<PeerListRow*> row) {
|
||||||
setContexted(Selected());
|
setContexted(Selected());
|
||||||
|
|
||||||
_rowsById.erase(row->id());
|
_rowsById.erase(row->id());
|
||||||
auto &byPeer = _rowsByPeer[row->peer()];
|
if (!row->special()) {
|
||||||
byPeer.erase(ranges::remove(byPeer, row), end(byPeer));
|
auto &byPeer = _rowsByPeer[row->peer()];
|
||||||
|
byPeer.erase(ranges::remove(byPeer, row), end(byPeer));
|
||||||
|
}
|
||||||
removeFromSearchIndex(row);
|
removeFromSearchIndex(row);
|
||||||
_filterResults.erase(
|
_filterResults.erase(
|
||||||
ranges::remove(_filterResults, row),
|
ranges::remove(_filterResults, row),
|
||||||
|
@ -1107,8 +1174,11 @@ void PeerListContent::setPressed(Selected pressed) {
|
||||||
_pressed = pressed;
|
_pressed = pressed;
|
||||||
}
|
}
|
||||||
|
|
||||||
crl::time PeerListContent::paintRow(Painter &p, crl::time ms, RowIndex index) {
|
crl::time PeerListContent::paintRow(
|
||||||
auto row = getRow(index);
|
Painter &p,
|
||||||
|
crl::time ms,
|
||||||
|
RowIndex index) {
|
||||||
|
const auto row = getRow(index);
|
||||||
Assert(row != nullptr);
|
Assert(row != nullptr);
|
||||||
|
|
||||||
row->lazyInitialize(_st.item);
|
row->lazyInitialize(_st.item);
|
||||||
|
@ -1119,17 +1189,17 @@ crl::time PeerListContent::paintRow(Painter &p, crl::time ms, RowIndex index) {
|
||||||
refreshStatusAt = row->refreshStatusTime();
|
refreshStatusAt = row->refreshStatusTime();
|
||||||
}
|
}
|
||||||
|
|
||||||
auto peer = row->peer();
|
const auto peer = row->special() ? nullptr : row->peer().get();
|
||||||
auto user = peer->asUser();
|
const auto user = peer ? peer->asUser() : nullptr;
|
||||||
auto active = (_contexted.index.value >= 0)
|
const auto active = (_contexted.index.value >= 0)
|
||||||
? _contexted
|
? _contexted
|
||||||
: (_pressed.index.value >= 0)
|
: (_pressed.index.value >= 0)
|
||||||
? _pressed
|
? _pressed
|
||||||
: _selected;
|
: _selected;
|
||||||
auto selected = (active.index == index);
|
const auto selected = (active.index == index);
|
||||||
auto actionSelected = (selected && active.action);
|
const auto actionSelected = (selected && active.action);
|
||||||
|
|
||||||
auto &bg = selected
|
const auto &bg = selected
|
||||||
? _st.item.button.textBgOver
|
? _st.item.button.textBgOver
|
||||||
: _st.item.button.textBg;
|
: _st.item.button.textBg;
|
||||||
p.fillRect(0, 0, width(), _rowHeight, bg);
|
p.fillRect(0, 0, width(), _rowHeight, bg);
|
||||||
|
@ -1184,12 +1254,17 @@ crl::time PeerListContent::paintRow(Painter &p, crl::time ms, RowIndex index) {
|
||||||
}
|
}
|
||||||
|
|
||||||
p.setFont(st::contactsStatusFont);
|
p.setFont(st::contactsStatusFont);
|
||||||
if (row->isSearchResult() && !_mentionHighlight.isEmpty() && peer->userName().startsWith(_mentionHighlight, Qt::CaseInsensitive)) {
|
if (row->isSearchResult()
|
||||||
auto username = peer->userName();
|
&& !_mentionHighlight.isEmpty()
|
||||||
auto availableWidth = statusw;
|
&& peer
|
||||||
|
&& peer->userName().startsWith(
|
||||||
|
_mentionHighlight,
|
||||||
|
Qt::CaseInsensitive)) {
|
||||||
|
const auto username = peer->userName();
|
||||||
|
const auto availableWidth = statusw;
|
||||||
auto highlightedPart = '@' + username.mid(0, _mentionHighlight.size());
|
auto highlightedPart = '@' + username.mid(0, _mentionHighlight.size());
|
||||||
auto grayedPart = username.mid(_mentionHighlight.size());
|
auto grayedPart = username.mid(_mentionHighlight.size());
|
||||||
auto highlightedWidth = st::contactsStatusFont->width(highlightedPart);
|
const auto highlightedWidth = st::contactsStatusFont->width(highlightedPart);
|
||||||
if (highlightedWidth >= availableWidth || grayedPart.isEmpty()) {
|
if (highlightedWidth >= availableWidth || grayedPart.isEmpty()) {
|
||||||
if (highlightedWidth > availableWidth) {
|
if (highlightedWidth > availableWidth) {
|
||||||
highlightedPart = st::contactsStatusFont->elided(highlightedPart, availableWidth);
|
highlightedPart = st::contactsStatusFont->elided(highlightedPart, availableWidth);
|
||||||
|
@ -1300,7 +1375,10 @@ void PeerListContent::loadProfilePhotos() {
|
||||||
if (to > rowsCount) to = rowsCount;
|
if (to > rowsCount) to = rowsCount;
|
||||||
|
|
||||||
for (auto index = from; index != to; ++index) {
|
for (auto index = from; index != to; ++index) {
|
||||||
getRow(RowIndex(index))->peer()->loadUserpic();
|
const auto row = getRow(RowIndex(index));
|
||||||
|
if (!row->special()) {
|
||||||
|
row->peer()->loadUserpic();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1313,13 +1391,13 @@ void PeerListContent::checkScrollForPreload() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void PeerListContent::searchQueryChanged(QString query) {
|
void PeerListContent::searchQueryChanged(QString query) {
|
||||||
auto searchWordsList = TextUtilities::PrepareSearchWords(query);
|
const auto searchWordsList = TextUtilities::PrepareSearchWords(query);
|
||||||
auto normalizedQuery = searchWordsList.join(' ');
|
const auto normalizedQuery = searchWordsList.join(' ');
|
||||||
if (_normalizedSearchQuery != normalizedQuery) {
|
if (_normalizedSearchQuery != normalizedQuery) {
|
||||||
setSearchQuery(query, normalizedQuery);
|
setSearchQuery(query, normalizedQuery);
|
||||||
if (_controller->searchInLocal() && !searchWordsList.isEmpty()) {
|
if (_controller->searchInLocal() && !searchWordsList.isEmpty()) {
|
||||||
auto minimalList = (const std::vector<not_null<PeerListRow*>>*)nullptr;
|
auto minimalList = (const std::vector<not_null<PeerListRow*>>*)nullptr;
|
||||||
for_const (auto &searchWord, searchWordsList) {
|
for (const auto &searchWord : searchWordsList) {
|
||||||
auto searchWordStart = searchWord[0].toLower();
|
auto searchWordStart = searchWord[0].toLower();
|
||||||
auto it = _searchIndex.find(searchWordStart);
|
auto it = _searchIndex.find(searchWordStart);
|
||||||
if (it == _searchIndex.cend()) {
|
if (it == _searchIndex.cend()) {
|
||||||
|
@ -1343,7 +1421,7 @@ void PeerListContent::searchQueryChanged(QString query) {
|
||||||
};
|
};
|
||||||
auto allSearchWordsInNames = [&](
|
auto allSearchWordsInNames = [&](
|
||||||
not_null<PeerData*> peer) {
|
not_null<PeerData*> peer) {
|
||||||
for_const (auto &searchWord, searchWordsList) {
|
for (const auto &searchWord : searchWordsList) {
|
||||||
if (!searchWordInNames(peer, searchWord)) {
|
if (!searchWordInNames(peer, searchWord)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -1352,8 +1430,8 @@ void PeerListContent::searchQueryChanged(QString query) {
|
||||||
};
|
};
|
||||||
|
|
||||||
_filterResults.reserve(minimalList->size());
|
_filterResults.reserve(minimalList->size());
|
||||||
for_const (auto row, *minimalList) {
|
for (const auto row : *minimalList) {
|
||||||
if (allSearchWordsInNames(row->peer())) {
|
if (!row->special() && allSearchWordsInNames(row->peer())) {
|
||||||
_filterResults.push_back(row);
|
_filterResults.push_back(row);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -38,22 +38,31 @@ namespace Notify {
|
||||||
struct PeerUpdate;
|
struct PeerUpdate;
|
||||||
} // namespace Notify
|
} // namespace Notify
|
||||||
|
|
||||||
auto PaintUserpicCallback(
|
using PaintRoundImageCallback = Fn<void(
|
||||||
|
Painter &p,
|
||||||
|
int x,
|
||||||
|
int y,
|
||||||
|
int outerWidth,
|
||||||
|
int size)>;
|
||||||
|
|
||||||
|
[[nodiscard]] PaintRoundImageCallback PaintUserpicCallback(
|
||||||
not_null<PeerData*> peer,
|
not_null<PeerData*> peer,
|
||||||
bool respectSavedMessagesChat)
|
bool respectSavedMessagesChat);
|
||||||
-> Fn<void(Painter &p, int x, int y, int outerWidth, int size)>;
|
|
||||||
|
|
||||||
using PeerListRowId = uint64;
|
using PeerListRowId = uint64;
|
||||||
class PeerListRow {
|
class PeerListRow {
|
||||||
public:
|
public:
|
||||||
PeerListRow(not_null<PeerData*> peer);
|
|
||||||
PeerListRow(not_null<PeerData*> peer, PeerListRowId id);
|
|
||||||
|
|
||||||
enum class State {
|
enum class State {
|
||||||
Active,
|
Active,
|
||||||
Disabled,
|
Disabled,
|
||||||
DisabledChecked,
|
DisabledChecked,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
explicit PeerListRow(not_null<PeerData*> peer);
|
||||||
|
PeerListRow(not_null<PeerData*> peer, PeerListRowId id);
|
||||||
|
|
||||||
|
virtual ~PeerListRow();
|
||||||
|
|
||||||
void setDisabledState(State state) {
|
void setDisabledState(State state) {
|
||||||
_disabledState = state;
|
_disabledState = state;
|
||||||
}
|
}
|
||||||
|
@ -62,20 +71,27 @@ public:
|
||||||
// not by the row itself, so there is no setChecked() method.
|
// not by the row itself, so there is no setChecked() method.
|
||||||
// We can query the checked state from row, but before it is
|
// We can query the checked state from row, but before it is
|
||||||
// added to the box it is always false.
|
// added to the box it is always false.
|
||||||
bool checked() const;
|
[[nodiscard]] bool checked() const;
|
||||||
|
|
||||||
|
[[nodiscard]] bool special() const {
|
||||||
|
return !_peer;
|
||||||
|
}
|
||||||
|
[[nodiscard]] not_null<PeerData*> peer() const {
|
||||||
|
Expects(!special());
|
||||||
|
|
||||||
not_null<PeerData*> peer() const {
|
|
||||||
return _peer;
|
return _peer;
|
||||||
}
|
}
|
||||||
PeerListRowId id() const {
|
[[nodiscard]] PeerListRowId id() const {
|
||||||
return _id;
|
return _id;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[[nodiscard]] virtual QString generateName();
|
||||||
|
[[nodiscard]] virtual QString generateShortName();
|
||||||
|
[[nodiscard]] PaintRoundImageCallback generatePaintUserpicCallback();
|
||||||
|
|
||||||
void setCustomStatus(const QString &status);
|
void setCustomStatus(const QString &status);
|
||||||
void clearCustomStatus();
|
void clearCustomStatus();
|
||||||
|
|
||||||
virtual ~PeerListRow();
|
|
||||||
|
|
||||||
// Box interface.
|
// Box interface.
|
||||||
virtual int nameIconWidth() const;
|
virtual int nameIconWidth() const;
|
||||||
virtual void paintNameIcon(
|
virtual void paintNameIcon(
|
||||||
|
@ -138,19 +154,16 @@ public:
|
||||||
_isSavedMessagesChat = isSavedMessagesChat;
|
_isSavedMessagesChat = isSavedMessagesChat;
|
||||||
}
|
}
|
||||||
|
|
||||||
enum class SetStyle {
|
|
||||||
Animated,
|
|
||||||
Fast,
|
|
||||||
};
|
|
||||||
template <typename UpdateCallback>
|
template <typename UpdateCallback>
|
||||||
void setChecked(
|
void setChecked(
|
||||||
bool checked,
|
bool checked,
|
||||||
SetStyle style,
|
const style::RoundImageCheckbox &st,
|
||||||
|
anim::type animated,
|
||||||
UpdateCallback callback) {
|
UpdateCallback callback) {
|
||||||
if (checked && !_checkbox) {
|
if (checked && !_checkbox) {
|
||||||
createCheckbox(std::move(callback));
|
createCheckbox(st, std::move(callback));
|
||||||
}
|
}
|
||||||
setCheckedInternal(checked, style);
|
setCheckedInternal(checked, animated);
|
||||||
}
|
}
|
||||||
void invalidatePixmapsCache();
|
void invalidatePixmapsCache();
|
||||||
|
|
||||||
|
@ -192,9 +205,20 @@ protected:
|
||||||
return _initialized;
|
return _initialized;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
explicit PeerListRow(PeerListRowId id);
|
||||||
|
|
||||||
|
virtual void paintEntityUserpicLeft(
|
||||||
|
Painter &p,
|
||||||
|
int x,
|
||||||
|
int y,
|
||||||
|
int outerWidth,
|
||||||
|
int size);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void createCheckbox(Fn<void()> updateCallback);
|
void createCheckbox(
|
||||||
void setCheckedInternal(bool checked, SetStyle style);
|
const style::RoundImageCheckbox &st,
|
||||||
|
Fn<void()> updateCallback);
|
||||||
|
void setCheckedInternal(bool checked, anim::type animated);
|
||||||
void paintDisabledCheckUserpic(
|
void paintDisabledCheckUserpic(
|
||||||
Painter &p,
|
Painter &p,
|
||||||
const style::PeerListItem &st,
|
const style::PeerListItem &st,
|
||||||
|
@ -204,7 +228,7 @@ private:
|
||||||
void setStatusText(const QString &text);
|
void setStatusText(const QString &text);
|
||||||
|
|
||||||
PeerListRowId _id = 0;
|
PeerListRowId _id = 0;
|
||||||
not_null<PeerData*> _peer;
|
PeerData *_peer = nullptr;
|
||||||
std::unique_ptr<Ui::RippleAnimation> _ripple;
|
std::unique_ptr<Ui::RippleAnimation> _ripple;
|
||||||
std::unique_ptr<Ui::RoundImageCheckbox> _checkbox;
|
std::unique_ptr<Ui::RoundImageCheckbox> _checkbox;
|
||||||
Ui::Text::String _name;
|
Ui::Text::String _name;
|
||||||
|
@ -245,7 +269,7 @@ public:
|
||||||
virtual void peerListUpdateRow(not_null<PeerListRow*> row) = 0;
|
virtual void peerListUpdateRow(not_null<PeerListRow*> row) = 0;
|
||||||
virtual void peerListRemoveRow(not_null<PeerListRow*> row) = 0;
|
virtual void peerListRemoveRow(not_null<PeerListRow*> row) = 0;
|
||||||
virtual void peerListConvertRowToSearchResult(not_null<PeerListRow*> row) = 0;
|
virtual void peerListConvertRowToSearchResult(not_null<PeerListRow*> row) = 0;
|
||||||
virtual bool peerListIsRowSelected(not_null<PeerData*> peer) = 0;
|
virtual bool peerListIsRowChecked(not_null<PeerListRow*> row) = 0;
|
||||||
virtual void peerListSetRowChecked(not_null<PeerListRow*> row, bool checked) = 0;
|
virtual void peerListSetRowChecked(not_null<PeerListRow*> row, bool checked) = 0;
|
||||||
virtual not_null<PeerListRow*> peerListRowAt(int index) = 0;
|
virtual not_null<PeerListRow*> peerListRowAt(int index) = 0;
|
||||||
virtual void peerListRefreshRows() = 0;
|
virtual void peerListRefreshRows() = 0;
|
||||||
|
@ -256,9 +280,17 @@ public:
|
||||||
virtual int peerListPartitionRows(Fn<bool(const PeerListRow &a)> border) = 0;
|
virtual int peerListPartitionRows(Fn<bool(const PeerListRow &a)> border) = 0;
|
||||||
|
|
||||||
template <typename PeerDataRange>
|
template <typename PeerDataRange>
|
||||||
void peerListAddSelectedRows(PeerDataRange &&range) {
|
void peerListAddSelectedPeers(PeerDataRange &&range) {
|
||||||
for (auto peer : range) {
|
for (const auto peer : range) {
|
||||||
peerListAddSelectedRowInBunch(peer);
|
peerListAddSelectedPeerInBunch(peer);
|
||||||
|
}
|
||||||
|
peerListFinishSelectedRowsBunch();
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename PeerListRowRange>
|
||||||
|
void peerListAddSelectedRows(PeerListRowRange &&range) {
|
||||||
|
for (const auto row : range) {
|
||||||
|
peerListAddSelectedRowInBunch(row);
|
||||||
}
|
}
|
||||||
peerListFinishSelectedRowsBunch();
|
peerListFinishSelectedRowsBunch();
|
||||||
}
|
}
|
||||||
|
@ -271,7 +303,8 @@ public:
|
||||||
virtual ~PeerListDelegate() = default;
|
virtual ~PeerListDelegate() = default;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
virtual void peerListAddSelectedRowInBunch(not_null<PeerData*> peer) = 0;
|
virtual void peerListAddSelectedPeerInBunch(not_null<PeerData*> peer) = 0;
|
||||||
|
virtual void peerListAddSelectedRowInBunch(not_null<PeerListRow*> row) = 0;
|
||||||
virtual void peerListFinishSelectedRowsBunch() = 0;
|
virtual void peerListFinishSelectedRowsBunch() = 0;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
@ -366,8 +399,8 @@ public:
|
||||||
|
|
||||||
virtual int contentWidth() const;
|
virtual int contentWidth() const;
|
||||||
|
|
||||||
bool isRowSelected(not_null<PeerData*> peer) {
|
bool isRowSelected(not_null<PeerListRow*> row) {
|
||||||
return delegate()->peerListIsRowSelected(peer);
|
return delegate()->peerListIsRowChecked(row);
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual bool searchInLocal() {
|
virtual bool searchInLocal() {
|
||||||
|
@ -470,7 +503,10 @@ public:
|
||||||
void refreshRows();
|
void refreshRows();
|
||||||
|
|
||||||
void setSearchMode(PeerListSearchMode mode);
|
void setSearchMode(PeerListSearchMode mode);
|
||||||
void changeCheckState(not_null<PeerListRow*> row, bool checked, PeerListRow::SetStyle style);
|
void changeCheckState(
|
||||||
|
not_null<PeerListRow*> row,
|
||||||
|
bool checked,
|
||||||
|
anim::type animated);
|
||||||
|
|
||||||
template <typename ReorderCallback>
|
template <typename ReorderCallback>
|
||||||
void reorderRows(ReorderCallback &&callback) {
|
void reorderRows(ReorderCallback &&callback) {
|
||||||
|
@ -677,10 +713,7 @@ public:
|
||||||
void peerListSetRowChecked(
|
void peerListSetRowChecked(
|
||||||
not_null<PeerListRow*> row,
|
not_null<PeerListRow*> row,
|
||||||
bool checked) override {
|
bool checked) override {
|
||||||
_content->changeCheckState(
|
_content->changeCheckState(row, checked, anim::type::normal);
|
||||||
row,
|
|
||||||
checked,
|
|
||||||
PeerListRow::SetStyle::Animated);
|
|
||||||
}
|
}
|
||||||
int peerListFullRowsCount() override {
|
int peerListFullRowsCount() override {
|
||||||
return _content->fullRowsCount();
|
return _content->fullRowsCount();
|
||||||
|
@ -770,7 +803,7 @@ public:
|
||||||
void peerListSetRowChecked(
|
void peerListSetRowChecked(
|
||||||
not_null<PeerListRow*> row,
|
not_null<PeerListRow*> row,
|
||||||
bool checked) override;
|
bool checked) override;
|
||||||
bool peerListIsRowSelected(not_null<PeerData*> peer) override;
|
bool peerListIsRowChecked(not_null<PeerListRow*> row) override;
|
||||||
int peerListSelectedRowsCount() override;
|
int peerListSelectedRowsCount() override;
|
||||||
std::vector<not_null<PeerData*>> peerListCollectSelectedRows() override;
|
std::vector<not_null<PeerData*>> peerListCollectSelectedRows() override;
|
||||||
void peerListScrollToTop() override;
|
void peerListScrollToTop() override;
|
||||||
|
@ -784,15 +817,26 @@ protected:
|
||||||
void paintEvent(QPaintEvent *e) override;
|
void paintEvent(QPaintEvent *e) override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void peerListAddSelectedRowInBunch(
|
void peerListAddSelectedPeerInBunch(
|
||||||
not_null<PeerData*> peer) override {
|
not_null<PeerData*> peer) override {
|
||||||
addSelectItem(peer, PeerListRow::SetStyle::Fast);
|
addSelectItem(peer, anim::type::instant);
|
||||||
|
}
|
||||||
|
void peerListAddSelectedRowInBunch(not_null<PeerListRow*> row) override {
|
||||||
|
addSelectItem(row, anim::type::instant);
|
||||||
}
|
}
|
||||||
void peerListFinishSelectedRowsBunch() override;
|
void peerListFinishSelectedRowsBunch() override;
|
||||||
|
|
||||||
void addSelectItem(
|
void addSelectItem(
|
||||||
not_null<PeerData*> peer,
|
not_null<PeerData*> peer,
|
||||||
PeerListRow::SetStyle style);
|
anim::type animated);
|
||||||
|
void addSelectItem(
|
||||||
|
not_null<PeerListRow*> row,
|
||||||
|
anim::type animated);
|
||||||
|
void addSelectItem(
|
||||||
|
uint64 itemId,
|
||||||
|
const QString &text,
|
||||||
|
PaintRoundImageCallback paintUserpic,
|
||||||
|
anim::type animated);
|
||||||
void createMultiSelect();
|
void createMultiSelect();
|
||||||
int getTopScrollSkip() const;
|
int getTopScrollSkip() const;
|
||||||
void updateScrollSkips();
|
void updateScrollSkips();
|
||||||
|
|
|
@ -135,10 +135,11 @@ constexpr auto NoChannel = ChannelId(0);
|
||||||
using PeerId = uint64;
|
using PeerId = uint64;
|
||||||
|
|
||||||
constexpr auto PeerIdMask = PeerId(0xFFFFFFFFULL);
|
constexpr auto PeerIdMask = PeerId(0xFFFFFFFFULL);
|
||||||
constexpr auto PeerIdTypeMask = PeerId(0x300000000ULL);
|
constexpr auto PeerIdTypeMask = PeerId(0xF00000000ULL);
|
||||||
constexpr auto PeerIdUserShift = PeerId(0x000000000ULL);
|
constexpr auto PeerIdUserShift = PeerId(0x000000000ULL);
|
||||||
constexpr auto PeerIdChatShift = PeerId(0x100000000ULL);
|
constexpr auto PeerIdChatShift = PeerId(0x100000000ULL);
|
||||||
constexpr auto PeerIdChannelShift = PeerId(0x200000000ULL);
|
constexpr auto PeerIdChannelShift = PeerId(0x200000000ULL);
|
||||||
|
constexpr auto PeerIdFakeShift = PeerId(0xF00000000ULL);
|
||||||
|
|
||||||
inline constexpr bool peerIsUser(const PeerId &id) {
|
inline constexpr bool peerIsUser(const PeerId &id) {
|
||||||
return (id & PeerIdTypeMask) == PeerIdUserShift;
|
return (id & PeerIdTypeMask) == PeerIdUserShift;
|
||||||
|
|
|
@ -251,7 +251,7 @@ void InnerWidget::peerListSetTitle(rpl::producer<QString> title) {
|
||||||
void InnerWidget::peerListSetAdditionalTitle(rpl::producer<QString> title) {
|
void InnerWidget::peerListSetAdditionalTitle(rpl::producer<QString> title) {
|
||||||
}
|
}
|
||||||
|
|
||||||
bool InnerWidget::peerListIsRowSelected(not_null<PeerData*> peer) {
|
bool InnerWidget::peerListIsRowChecked(not_null<PeerListRow*> row) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -267,7 +267,11 @@ void InnerWidget::peerListScrollToTop() {
|
||||||
_scrollToRequests.fire({ -1, -1 });
|
_scrollToRequests.fire({ -1, -1 });
|
||||||
}
|
}
|
||||||
|
|
||||||
void InnerWidget::peerListAddSelectedRowInBunch(not_null<PeerData*> peer) {
|
void InnerWidget::peerListAddSelectedPeerInBunch(not_null<PeerData*> peer) {
|
||||||
|
Unexpected("Item selection in Info::Profile::Members.");
|
||||||
|
}
|
||||||
|
|
||||||
|
void InnerWidget::peerListAddSelectedRowInBunch(not_null<PeerListRow*> row) {
|
||||||
Unexpected("Item selection in Info::Profile::Members.");
|
Unexpected("Item selection in Info::Profile::Members.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -50,12 +50,14 @@ private:
|
||||||
// PeerListContentDelegate interface.
|
// PeerListContentDelegate interface.
|
||||||
void peerListSetTitle(rpl::producer<QString> title) override;
|
void peerListSetTitle(rpl::producer<QString> title) override;
|
||||||
void peerListSetAdditionalTitle(rpl::producer<QString> title) override;
|
void peerListSetAdditionalTitle(rpl::producer<QString> title) override;
|
||||||
bool peerListIsRowSelected(not_null<PeerData*> peer) override;
|
bool peerListIsRowChecked(not_null<PeerListRow*> row) override;
|
||||||
int peerListSelectedRowsCount() override;
|
int peerListSelectedRowsCount() override;
|
||||||
std::vector<not_null<PeerData*>> peerListCollectSelectedRows() override;
|
std::vector<not_null<PeerData*>> peerListCollectSelectedRows() override;
|
||||||
void peerListScrollToTop() override;
|
void peerListScrollToTop() override;
|
||||||
void peerListAddSelectedRowInBunch(
|
void peerListAddSelectedPeerInBunch(
|
||||||
not_null<PeerData*> peer) override;
|
not_null<PeerData*> peer) override;
|
||||||
|
void peerListAddSelectedRowInBunch(
|
||||||
|
not_null<PeerListRow*> row) override;
|
||||||
void peerListFinishSelectedRowsBunch() override;
|
void peerListFinishSelectedRowsBunch() override;
|
||||||
void peerListSetDescription(
|
void peerListSetDescription(
|
||||||
object_ptr<Ui::FlatLabel> description) override;
|
object_ptr<Ui::FlatLabel> description) override;
|
||||||
|
|
|
@ -56,12 +56,14 @@ class ListDelegate final : public PeerListContentDelegate {
|
||||||
public:
|
public:
|
||||||
void peerListSetTitle(rpl::producer<QString> title) override;
|
void peerListSetTitle(rpl::producer<QString> title) override;
|
||||||
void peerListSetAdditionalTitle(rpl::producer<QString> title) override;
|
void peerListSetAdditionalTitle(rpl::producer<QString> title) override;
|
||||||
bool peerListIsRowSelected(not_null<PeerData*> peer) override;
|
bool peerListIsRowChecked(not_null<PeerListRow*> row) override;
|
||||||
int peerListSelectedRowsCount() override;
|
int peerListSelectedRowsCount() override;
|
||||||
std::vector<not_null<PeerData*>> peerListCollectSelectedRows() override;
|
std::vector<not_null<PeerData*>> peerListCollectSelectedRows() override;
|
||||||
void peerListScrollToTop() override;
|
void peerListScrollToTop() override;
|
||||||
void peerListAddSelectedRowInBunch(
|
void peerListAddSelectedPeerInBunch(
|
||||||
not_null<PeerData*> peer) override;
|
not_null<PeerData*> peer) override;
|
||||||
|
void peerListAddSelectedRowInBunch(
|
||||||
|
not_null<PeerListRow*> row) override;
|
||||||
void peerListFinishSelectedRowsBunch() override;
|
void peerListFinishSelectedRowsBunch() override;
|
||||||
void peerListSetDescription(
|
void peerListSetDescription(
|
||||||
object_ptr<Ui::FlatLabel> description) override;
|
object_ptr<Ui::FlatLabel> description) override;
|
||||||
|
@ -121,7 +123,7 @@ void ListDelegate::peerListSetTitle(rpl::producer<QString> title) {
|
||||||
void ListDelegate::peerListSetAdditionalTitle(rpl::producer<QString> title) {
|
void ListDelegate::peerListSetAdditionalTitle(rpl::producer<QString> title) {
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ListDelegate::peerListIsRowSelected(not_null<PeerData*> peer) {
|
bool ListDelegate::peerListIsRowChecked(not_null<PeerListRow*> row) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -137,7 +139,11 @@ auto ListDelegate::peerListCollectSelectedRows()
|
||||||
void ListDelegate::peerListScrollToTop() {
|
void ListDelegate::peerListScrollToTop() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void ListDelegate::peerListAddSelectedRowInBunch(not_null<PeerData*> peer) {
|
void ListDelegate::peerListAddSelectedPeerInBunch(not_null<PeerData*> peer) {
|
||||||
|
Unexpected("Item selection in Info::Profile::Members.");
|
||||||
|
}
|
||||||
|
|
||||||
|
void ListDelegate::peerListAddSelectedRowInBunch(not_null<PeerListRow*> row) {
|
||||||
Unexpected("Item selection in Info::Profile::Members.");
|
Unexpected("Item selection in Info::Profile::Members.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -415,7 +415,7 @@ void Members::peerListSetTitle(rpl::producer<QString> title) {
|
||||||
void Members::peerListSetAdditionalTitle(rpl::producer<QString> title) {
|
void Members::peerListSetAdditionalTitle(rpl::producer<QString> title) {
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Members::peerListIsRowSelected(not_null<PeerData*> peer) {
|
bool Members::peerListIsRowChecked(not_null<PeerListRow*> row) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -431,7 +431,11 @@ void Members::peerListScrollToTop() {
|
||||||
_scrollToRequests.fire({ -1, -1 });
|
_scrollToRequests.fire({ -1, -1 });
|
||||||
}
|
}
|
||||||
|
|
||||||
void Members::peerListAddSelectedRowInBunch(not_null<PeerData*> peer) {
|
void Members::peerListAddSelectedPeerInBunch(not_null<PeerData*> peer) {
|
||||||
|
Unexpected("Item selection in Info::Profile::Members.");
|
||||||
|
}
|
||||||
|
|
||||||
|
void Members::peerListAddSelectedRowInBunch(not_null<PeerListRow*> row) {
|
||||||
Unexpected("Item selection in Info::Profile::Members.");
|
Unexpected("Item selection in Info::Profile::Members.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -61,12 +61,14 @@ private:
|
||||||
// PeerListContentDelegate interface.
|
// PeerListContentDelegate interface.
|
||||||
void peerListSetTitle(rpl::producer<QString> title) override;
|
void peerListSetTitle(rpl::producer<QString> title) override;
|
||||||
void peerListSetAdditionalTitle(rpl::producer<QString> title) override;
|
void peerListSetAdditionalTitle(rpl::producer<QString> title) override;
|
||||||
bool peerListIsRowSelected(not_null<PeerData*> peer) override;
|
bool peerListIsRowChecked(not_null<PeerListRow*> row) override;
|
||||||
int peerListSelectedRowsCount() override;
|
int peerListSelectedRowsCount() override;
|
||||||
std::vector<not_null<PeerData*>> peerListCollectSelectedRows() override;
|
std::vector<not_null<PeerData*>> peerListCollectSelectedRows() override;
|
||||||
void peerListScrollToTop() override;
|
void peerListScrollToTop() override;
|
||||||
void peerListAddSelectedRowInBunch(
|
void peerListAddSelectedPeerInBunch(
|
||||||
not_null<PeerData*> peer) override;
|
not_null<PeerData*> peer) override;
|
||||||
|
void peerListAddSelectedRowInBunch(
|
||||||
|
not_null<PeerListRow*> row) override;
|
||||||
void peerListFinishSelectedRowsBunch() override;
|
void peerListFinishSelectedRowsBunch() override;
|
||||||
void peerListSetDescription(
|
void peerListSetDescription(
|
||||||
object_ptr<Ui::FlatLabel> description) override;
|
object_ptr<Ui::FlatLabel> description) override;
|
||||||
|
|
|
@ -323,15 +323,15 @@ void RoundCheckbox::paint(Painter &p, int x, int y, int outerWidth, float64 mast
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void RoundCheckbox::setChecked(bool newChecked, SetStyle speed) {
|
void RoundCheckbox::setChecked(bool newChecked, anim::type animated) {
|
||||||
if (_checked == newChecked) {
|
if (_checked == newChecked) {
|
||||||
if (speed != SetStyle::Animated) {
|
if (animated == anim::type::instant) {
|
||||||
_checkedProgress.stop();
|
_checkedProgress.stop();
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
_checked = newChecked;
|
_checked = newChecked;
|
||||||
if (speed == SetStyle::Animated) {
|
if (animated == anim::type::normal) {
|
||||||
_checkedProgress.start(
|
_checkedProgress.start(
|
||||||
_updateCallback,
|
_updateCallback,
|
||||||
_checked ? 0. : 1.,
|
_checked ? 0. : 1.,
|
||||||
|
@ -441,16 +441,16 @@ float64 RoundImageCheckbox::checkedAnimationRatio() const {
|
||||||
return snap(_selection.value(checked() ? 1. : 0.), 0., 1.);
|
return snap(_selection.value(checked() ? 1. : 0.), 0., 1.);
|
||||||
}
|
}
|
||||||
|
|
||||||
void RoundImageCheckbox::setChecked(bool newChecked, SetStyle speed) {
|
void RoundImageCheckbox::setChecked(bool newChecked, anim::type animated) {
|
||||||
auto changed = (checked() != newChecked);
|
auto changed = (checked() != newChecked);
|
||||||
_check.setChecked(newChecked, speed);
|
_check.setChecked(newChecked, animated);
|
||||||
if (!changed) {
|
if (!changed) {
|
||||||
if (speed != SetStyle::Animated) {
|
if (animated == anim::type::instant) {
|
||||||
_selection.stop();
|
_selection.stop();
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (speed == SetStyle::Animated) {
|
if (animated == anim::type::normal) {
|
||||||
prepareWideCache();
|
prepareWideCache();
|
||||||
_selection.start(_updateCallback, checked() ? 0 : 1, checked() ? 1 : 0, _st.selectDuration, anim::bumpy(1.25));
|
_selection.start(_updateCallback, checked() ? 0 : 1, checked() ? 1 : 0, _st.selectDuration, anim::bumpy(1.25));
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -22,11 +22,9 @@ public:
|
||||||
bool checked() const {
|
bool checked() const {
|
||||||
return _checked;
|
return _checked;
|
||||||
}
|
}
|
||||||
enum class SetStyle {
|
void setChecked(
|
||||||
Animated,
|
bool newChecked,
|
||||||
Fast,
|
anim::type animated = anim::type::normal);
|
||||||
};
|
|
||||||
void setChecked(bool newChecked, SetStyle speed = SetStyle::Animated);
|
|
||||||
|
|
||||||
void invalidateCache();
|
void invalidateCache();
|
||||||
|
|
||||||
|
@ -55,8 +53,9 @@ public:
|
||||||
bool checked() const {
|
bool checked() const {
|
||||||
return _check.checked();
|
return _check.checked();
|
||||||
}
|
}
|
||||||
using SetStyle = RoundCheckbox::SetStyle;
|
void setChecked(
|
||||||
void setChecked(bool newChecked, SetStyle speed = SetStyle::Animated);
|
bool newChecked,
|
||||||
|
anim::type animated = anim::type::normal);
|
||||||
|
|
||||||
void invalidateCache() {
|
void invalidateCache() {
|
||||||
_check.invalidateCache();
|
_check.invalidateCache();
|
||||||
|
|
|
@ -304,6 +304,13 @@ windowFilterSmallItem: PeerListItem(defaultPeerListItem) {
|
||||||
photoPosition: point(15px, 5px);
|
photoPosition: point(15px, 5px);
|
||||||
namePosition: point(62px, 14px);
|
namePosition: point(62px, 14px);
|
||||||
photoSize: 34px;
|
photoSize: 34px;
|
||||||
|
checkbox: RoundImageCheckbox(defaultPeerListCheckbox) {
|
||||||
|
imageRadius: 17px;
|
||||||
|
imageSmallRadius: 14px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
windowFilterSmallList: PeerList(defaultPeerList) {
|
||||||
|
item: windowFilterSmallItem;
|
||||||
}
|
}
|
||||||
windowFilterSmallRemove: IconButton(notifyClose) {
|
windowFilterSmallRemove: IconButton(notifyClose) {
|
||||||
}
|
}
|
||||||
|
@ -316,6 +323,13 @@ windowFilterTypeBots: icon {{ "filters_type_bots", historyPeerUserpicFg }};
|
||||||
windowFilterTypeNoMuted: icon {{ "filters_type_muted", historyPeerUserpicFg }};
|
windowFilterTypeNoMuted: icon {{ "filters_type_muted", historyPeerUserpicFg }};
|
||||||
windowFilterTypeNoArchived: icon {{ "filters_type_archived", historyPeerUserpicFg }};
|
windowFilterTypeNoArchived: icon {{ "filters_type_archived", historyPeerUserpicFg }};
|
||||||
windowFilterTypeNoRead: icon {{ "filters_type_read", historyPeerUserpicFg }};
|
windowFilterTypeNoRead: icon {{ "filters_type_read", historyPeerUserpicFg }};
|
||||||
|
windowFilterChatsSectionSubtitle: FlatLabel(defaultFlatLabel) {
|
||||||
|
style: TextStyle(defaultTextStyle) {
|
||||||
|
font: searchedBarFont;
|
||||||
|
}
|
||||||
|
textFg: searchedBarFg;
|
||||||
|
}
|
||||||
|
windowFilterChatsSectionSubtitlePadding: margins(17px, 7px, 17px, 7px);
|
||||||
|
|
||||||
// Mac specific
|
// Mac specific
|
||||||
|
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
Subproject commit c96119dcd18bff5974348524b1e15b3d396426dc
|
Subproject commit d2611d7e8588759c9ecc129ce70ef0d7b2e24d6c
|
Loading…
Reference in New Issue