Files
tdesktop/Telegram/SourceFiles/window/window_chat_switch_process.cpp
2025-12-10 21:28:33 +03:00

497 lines
13 KiB
C++

/*
This file is part of Telegram Desktop,
the official desktop application for the Telegram messaging service.
For license and copyright information please follow this link:
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#include "window/window_chat_switch_process.h"
#include "core/application.h"
#include "core/shortcuts.h"
#include "data/components/recent_peers.h"
#include "data/data_forum_topic.h"
#include "data/data_peer.h"
#include "data/data_saved_sublist.h"
#include "data/data_thread.h"
#include "info/profile/info_profile_cover.h"
#include "lang/lang_keys.h"
#include "main/main_session.h"
#include "ui/widgets/labels.h"
#include "ui/widgets/shadow.h"
#include "ui/controls/userpic_button.h"
#include "ui/painter.h"
#include "ui/rp_widget.h"
#include "window/window_separate_id.h"
#include "window/window_session_controller.h"
#include "styles/style_boxes.h"
#include "styles/style_layers.h"
#include "styles/style_window.h"
namespace Window {
namespace {
class Button final : public Ui::AbstractButton {
public:
Button(
QWidget *parent,
not_null<Data::Thread*> thread,
base::flat_map<not_null<PeerData*>, Ui::PeerUserpicView> &userpics);
[[nodiscard]] rpl::producer<> selectRequests() const;
void setSelected(
bool selected,
anim::type animated = anim::type::normal);
private:
void setup(
not_null<Data::Thread*> thread,
base::flat_map<not_null<PeerData*>, Ui::PeerUserpicView> &userpics);
void paintEvent(QPaintEvent *e) override;
void mouseMoveEvent(QMouseEvent *e) override;
rpl::event_stream<> _selectRequests;
Ui::Animations::Simple _overAnimation;
bool _selected = false;
};
void CloseInWindows(not_null<Data::Thread*> thread) {
using WindowPointer = base::weak_ptr<Window::SessionController>;
auto closing = std::vector<WindowPointer>();
auto clearing = std::vector<WindowPointer>();
for (const auto window : thread->session().windows()) {
if (window->windowId().chat() == thread) {
closing.push_back(base::make_weak(window));
} else if (window->activeChatCurrent().thread() == thread) {
clearing.push_back(base::make_weak(window));
}
}
for (const auto &window : closing) {
if (const auto strong = window.get()) {
Core::App().closeWindow(&strong->window());
}
}
for (const auto &window : clearing) {
if (const auto strong = window.get()) {
strong->clearSectionStack();
}
}
}
Button::Button(
QWidget *parent,
not_null<Data::Thread*> thread,
base::flat_map<not_null<PeerData*>, Ui::PeerUserpicView> &userpics)
: AbstractButton(parent) {
setup(thread, userpics);
}
void Button::setup(
not_null<Data::Thread*> thread,
base::flat_map<not_null<PeerData*>, Ui::PeerUserpicView> &userpics) {
resize(st::chatSwitchSize);
auto userpicSt = &st::chatSwitchUserpic;
const auto userpicSize = userpicSt->size;
if (const auto topic = thread->asTopic()) {
using namespace Info::Profile;
const auto userpic = Ui::CreateChild<TopicIconButton>(
this,
topic,
[] { return true; }); // paused
userpic->show();
userpic->move(
((width() - userpic->width()) / 2),
st::chatSwitchUserpicTop);
userpic->setAttribute(Qt::WA_TransparentForMouseEvents);
userpicSt = &st::chatSwitchUserpicSmall;
} else if (const auto sublist = thread->asSublist()) {
const auto sublistPeer = sublist->sublistPeer();
const auto userpic = Ui::CreateChild<Ui::UserpicButton>(
this,
sublistPeer,
st::chatSwitchUserpicSublist);
userpic->showMyNotesOnSelf(true);
userpic->show();
userpic->move(
((width() - userpicSize.width()) / 2),
st::chatSwitchUserpicTop);
userpic->setAttribute(Qt::WA_TransparentForMouseEvents);
userpics.emplace(sublistPeer, sublistPeer->createUserpicView());
userpicSt = &st::chatSwitchUserpicSmall;
}
const auto peer = thread->peer();
const auto userpic = Ui::CreateChild<Ui::UserpicButton>(
this,
peer,
*userpicSt);
userpic->showSavedMessagesOnSelf(true);
userpic->show();
userpic->move(
(((width() - userpicSize.width()) / 2)
+ (userpicSize.width() - userpicSt->size.width())),
(st::chatSwitchUserpicTop
+ (userpicSize.height() - userpicSt->size.height())));
userpic->setAttribute(Qt::WA_TransparentForMouseEvents);
userpics.emplace(peer, peer->createUserpicView());
const auto label = Ui::CreateChild<Ui::FlatLabel>(
this,
((thread->asHistory() && peer->isSelf())
? tr::lng_saved_messages(tr::now)
: thread->chatListName()),
st::chatSwitchNameLabel);
label->setBreakEverywhere(true);
label->show();
label->resizeToNaturalWidth(
width() - 2 * st::chatSwitchNameSkip);
label->move(
(width() - label->width()) / 2,
(height() + userpic->y() + userpic->height() - label->height()) / 2);
show();
}
rpl::producer<> Button::selectRequests() const {
return _selectRequests.events();
}
void Button::setSelected(bool selected, anim::type animated) {
if (_selected == selected) {
if (animated == anim::type::instant) {
_overAnimation.stop();
}
return;
}
_selected = selected;
if (animated == anim::type::instant) {
_overAnimation.stop();
update();
} else {
_overAnimation.start(
[=] { update(); },
selected ? 0. : 1.,
selected ? 1. : 0.,
st::slideWrapDuration);
}
}
void Button::paintEvent(QPaintEvent *e) {
const auto selected = _selected ? 1. : 0.;
const auto selection = _overAnimation.value(selected);
if (selection <= 0.) {
return;
}
auto p = QPainter(this);
auto hq = PainterHighQualityEnabler(p);
const auto radius = st::boxRadius;
const auto line = st::chatSwitchSelectLine;
auto pen = st::defaultRoundCheckbox.bgActive->p;
pen.setWidthF(line * selection);
p.setPen(pen);
const auto r = QRectF(rect()).marginsRemoved(
{ line / 2., line / 2., line / 2., line / 2. });
p.drawRoundedRect(r, radius, radius);
}
void Button::mouseMoveEvent(QMouseEvent *e) {
if (!_selected) {
_selectRequests.fire({});
}
}
} // namespace
struct ChatSwitchProcess::Entry {
not_null<Data::Thread*> thread;
std::unique_ptr<Button> button;
};
ChatSwitchProcess::ChatSwitchProcess(
not_null<Ui::RpWidget*> geometry,
not_null<Main::Session*> session,
Data::Thread *opened)
: _session(session)
, _widget(std::make_unique<Ui::RpWidget>(
geometry->parentWidget() ? geometry->parentWidget() : geometry))
, _view(Ui::CreateChild<Ui::RpWidget>(_widget.get()))
, _bg(st::boxRadius, st::boxBg) {
setupWidget(geometry);
setupContent(opened);
setupView();
}
ChatSwitchProcess::~ChatSwitchProcess() {
_session->recentPeers().chatOpenKeepUserpics(std::move(_userpics));
}
rpl::producer<not_null<Data::Thread*>> ChatSwitchProcess::chosen() const {
return _chosen.events();
}
rpl::producer<> ChatSwitchProcess::closeRequests() const {
return _closeRequests.events();
}
void ChatSwitchProcess::process(const Request &request) {
Expects(_selected < _shownCount);
if (request.action == Qt::Key_Escape) {
_closeRequests.fire({});
} else if (request.action == Qt::Key_Enter) {
if (_selected >= 0) {
_chosen.fire_copy(_list[_selected]);
} else {
_closeRequests.fire({});
}
} else if (request.action == Qt::Key_Tab
|| request.action == Qt::Key_Right) {
if (_selected < 0 || _selected + 1 >= _shownCount) {
setSelected(0);
} else {
setSelected(_selected + 1);
}
} else if (request.action == Qt::Key_Backtab
|| request.action == Qt::Key_Left) {
if (_selected <= 0) {
setSelected(_shownCount - 1);
} else {
setSelected(_selected - 1);
}
} else if (request.action == Qt::Key_Up) {
const auto now = std::max(_selected, 0) - _shownPerRow;
const auto bound = (now < 0) ? (_shownCount + now) : now;
setSelected(bound);
} else if (request.action == Qt::Key_Down) {
const auto now = std::max(_selected, 0) + _shownPerRow;
const auto bound = (now >= _shownCount) ? (now - _shownCount) : now;
setSelected(bound);
} else if (request.action == Qt::Key_Q) {
if (_selected >= 0) {
const auto thread = _list[_selected];
thread->session().recentPeers().chatOpenRemove(thread);
remove(thread);
CloseInWindows(thread);
}
}
}
void ChatSwitchProcess::setSelected(int index) {
if (_selected == index || _list.size() < 2) {
return;
}
if (_selected >= 0) {
_entries[_selected].button->setSelected(false);
}
_selected = index;
if (_selected >= 0) {
_entries[_selected].button->setSelected(true);
}
}
void ChatSwitchProcess::setupWidget(not_null<Ui::RpWidget*> geometry) {
geometry->geometryValue(
) | rpl::on_next([=](QRect value) {
const auto parent = geometry->parentWidget();
_widget->setGeometry((parent == _widget->parentWidget())
? value
: QRect(QPoint(), value.size()));
}, _widget->lifetime());
_widget->events() | rpl::on_next([=](not_null<QEvent*> e) {
if (e->type() == QEvent::MouseButtonPress) {
crl::on_main(_widget.get(), [=] {
_closeRequests.fire({});
});
}
}, _widget->lifetime());
_widget->show();
}
void ChatSwitchProcess::setupContent(Data::Thread *opened) {
_list = _session->recentPeers().collectChatOpenHistory();
if (_list.size() < 2) {
return;
}
if (opened) {
const auto i = ranges::find(_list, not_null(opened));
if (i == end(_list)) {
_list.insert(begin(_list), opened);
} else if (i != begin(_list)) {
ranges::rotate(begin(_list), i, i + 1);
}
_selected = 0;
}
const auto find = [=](Button *button) {
return ranges::find(_entries, button, [](const Entry &entry) {
return entry.button.get();
});
};
for (const auto &thread : _list) {
auto button = std::make_unique<Button>(
_view.get(),
thread,
_userpics);
const auto raw = button.get();
raw->selectRequests() | rpl::on_next([=] {
const auto i = find(raw);
setSelected(int(i - begin(_entries)));
}, raw->lifetime());
raw->setClickedCallback([=] {
_chosen.fire_copy(thread);
});
_entries.push_back({ thread, std::move(button) });
auto destroyed = thread->asTopic()
? thread->asTopic()->destroyed()
: thread->asSublist()
? thread->asSublist()->destroyed()
: nullptr;
if (!destroyed) {
continue;
}
std::move(destroyed) | rpl::on_next([=] {
remove(thread);
}, raw->lifetime());
}
}
void ChatSwitchProcess::remove(not_null<Data::Thread*> thread) {
_list.erase(ranges::remove(_list, thread), end(_list));
const auto i = ranges::find(_entries, thread, &Entry::thread);
if (i != end(_entries)) {
const auto selected = _selected;
const auto index = int(i - begin(_entries));
if (_selected > index) {
--_selected;
} else if (_selected == index) {
_selected = -1;
}
_entries.erase(i);
const auto weak = base::make_weak(_widget.get());
layout(_widget->size());
if (weak && _selected < 0 && selected > 0) {
if (_entries.empty()) {
_closeRequests.fire({});
} else {
setSelected(std::min(selected - 1, _shownCount - 1));
}
}
}
}
void ChatSwitchProcess::setupView() {
_widget->sizeValue() | rpl::on_next([=](QSize size) {
layout(size);
}, _view->lifetime());
_view->show();
_view->paintRequest() | rpl::on_next([=](QRect clip) {
if (_outer.isEmpty()) {
return;
}
auto p = QPainter(_view);
p.translate(-_shadowed.topLeft());
Ui::Shadow::paint(p, _outer, _view->width(), st::boxRoundShadow);
_bg.paint(p, _outer);
}, _view->lifetime());
_view->events() | rpl::on_next([=](not_null<QEvent*> e) {
if (e->type() == QEvent::MouseButtonPress) {
e->accept();
}
}, _view->lifetime());
}
void ChatSwitchProcess::layout(QSize size) {
const auto full = QRect(QPoint(), size);
const auto outer = full.marginsRemoved(st::chatSwitchMargins);
auto inner = outer.marginsRemoved(st::chatSwitchPadding);
const auto available = inner.width();
const auto canPerRow = (available / st::chatSwitchSize.width());
const auto canRows = (canPerRow > 2 * 7)
? 1
: (canPerRow > 3 * 4)
? 2
: 3;
if (canPerRow < 1) {
return;
} else if (_list.size() < 2) {
_closeRequests.fire({});
return;
}
const auto count = int(_list.size());
_shownRows = std::min(canRows, (count + canPerRow - 1) / canPerRow);
_shownPerRow = std::min(count / _shownRows, canPerRow);
if (_shownRows > 2) {
if (_shownPerRow * 2 > _shownRows * 4) {
_shownRows = 2;
} else if (_shownPerRow > 4) {
_shownPerRow = 4;
}
}
if (_shownRows > 1) {
if (_shownPerRow > _shownRows * 7) {
_shownRows = 1;
} else if (_shownPerRow > 7) {
_shownPerRow = 7;
}
}
_shownCount = _shownPerRow * _shownRows;
if (_selected >= _shownCount) {
_selected = -1;
}
const auto width = _shownPerRow * st::chatSwitchSize.width();
const auto height = _shownRows * st::chatSwitchSize.height();
size = QSize(width, height);
_inner = QRect(
(full.width() - width) / 2,
(full.height() - height) / 2,
width,
height);
_outer = _inner.marginsAdded(st::chatSwitchPadding);
const auto padding = st::boxRoundShadow.extend + st::chatSwitchPadding;
auto index = 0;
auto top = padding.top();
for (auto row = 0; row != _shownRows; ++row) {
auto left = padding.left();
for (auto column = 0; column != _shownPerRow; ++column) {
auto &entry = _entries[index++];
entry.button->moveToLeft(left, top, _inner.width());
entry.button->show();
left += st::chatSwitchSize.width();
}
top += st::chatSwitchSize.height();
}
for (auto i = _shownRows * _shownPerRow; i < count; ++i) {
_entries[i].button->hide();
}
_shadowed = _outer.marginsAdded(st::boxRoundShadow.extend);
_view->setGeometry(_shadowed);
}
rpl::lifetime &ChatSwitchProcess::lifetime() {
return _lifetime;
}
} // namespace Window