tdesktop/Telegram/SourceFiles/window/window_main_menu.cpp

1226 lines
36 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_main_menu.h"
#include "window/themes/window_theme.h"
#include "window/window_peer_menu.h"
#include "window/window_session_controller.h"
#include "ui/widgets/buttons.h"
#include "ui/widgets/labels.h"
#include "ui/widgets/menu/menu.h"
#include "ui/widgets/menu/menu_common.h"
#include "ui/widgets/menu/menu_toggle.h"
#include "ui/widgets/popup_menu.h"
#include "ui/widgets/scroll_area.h"
#include "ui/widgets/shadow.h"
#include "ui/wrap/slide_wrap.h"
#include "ui/wrap/vertical_layout.h"
#include "ui/text/text_utilities.h"
#include "ui/special_buttons.h"
#include "ui/empty_userpic.h"
#include "dialogs/dialogs_layout.h"
#include "base/call_delayed.h"
#include "mainwindow.h"
#include "storage/localstorage.h"
#include "storage/storage_account.h"
#include "support/support_templates.h"
#include "settings/settings_common.h"
#include "base/qt_signal_producer.h"
#include "boxes/about_box.h"
#include "boxes/confirm_box.h"
#include "boxes/peer_list_controllers.h"
#include "calls/calls_box_controller.h"
#include "lang/lang_keys.h"
#include "core/click_handler_types.h"
#include "core/core_settings.h"
#include "core/application.h"
#include "main/main_session.h"
#include "main/main_session_settings.h"
#include "main/main_account.h"
#include "main/main_domain.h"
#include "mtproto/mtp_instance.h"
#include "mtproto/mtproto_config.h"
#include "data/data_folder.h"
#include "data/data_session.h"
#include "data/data_user.h"
#include "data/data_changes.h"
#include "mainwidget.h"
#include "app.h"
#include "styles/style_window.h"
#include "styles/style_widgets.h"
#include "styles/style_dialogs.h"
#include "styles/style_settings.h"
#include "styles/style_boxes.h"
#include "styles/style_info.h" // infoTopBarMenu
#include "styles/style_layers.h"
#include <QtGui/QWindow>
#include <QtGui/QScreen>
namespace {
constexpr auto kMinDiffIntensity = 0.25;
[[nodicard]] float64 IntensityOfColor(QColor color) {
return (0.299 * color.red()
+ 0.587 * color.green()
+ 0.114 * color.blue()) / 255.0;
}
[[nodiscard]] bool IsShadowShown(const QImage &img, const QRect r, float64 intensityText) {
for (auto x = r.x(); x < r.x() + r.width(); x++) {
for (auto y = r.y(); y < r.y() + r.height(); y++) {
const auto intensity = IntensityOfColor(QColor(img.pixel(x, y)));
if ((std::abs(intensity - intensityText)) < kMinDiffIntensity) {
return true;
}
}
}
return false;
}
[[nodiscard]] bool IsFilledCover() {
const auto background = Window::Theme::Background();
return background->tile()
|| background->colorForFill().has_value()
|| background->isMonoColorImage()
|| background->paper().isPattern()
|| Data::IsLegacy1DefaultWallPaper(background->paper());
}
[[nodiscard]] bool IsAltShift(Qt::KeyboardModifiers modifiers) {
return (modifiers & Qt::ShiftModifier) && (modifiers & Qt::AltModifier);
}
void ShowCallsBox(not_null<Window::SessionController*> window) {
auto controller = std::make_unique<Calls::BoxController>(window);
const auto initBox = [
window,
controller = controller.get()
](not_null<PeerListBox*> box) {
box->addButton(tr::lng_close(), [=] {
box->closeBox();
});
using MenuPointer = base::unique_qptr<Ui::PopupMenu>;
const auto menu = std::make_shared<MenuPointer>();
const auto menuButton = box->addTopButton(st::infoTopBarMenu);
menuButton->setClickedCallback([=] {
*menu = base::make_unique_q<Ui::PopupMenu>(menuButton);
const auto showSettings = [=] {
window->showSettings(
Settings::Type::Calls,
Window::SectionShow(anim::type::instant));
};
const auto clearAll = crl::guard(box, [=] {
box->getDelegate()->show(Box(Calls::ClearCallsBox, window));
});
(*menu)->addAction(
tr::lng_settings_section_call_settings(tr::now),
showSettings);
if (controller->delegate()->peerListFullRowsCount() > 0) {
(*menu)->addAction(
tr::lng_call_box_clear_all(tr::now),
clearAll);
}
(*menu)->popup(QCursor::pos());
return true;
});
};
Ui::show(Box<PeerListBox>(std::move(controller), initBox));
}
} // namespace
namespace Window {
class MainMenu::AccountButton final
: public Ui::RippleButton
, public base::Subscriber {
public:
AccountButton(QWidget *parent, not_null<Main::Account*> account);
private:
void paintEvent(QPaintEvent *e) override;
void contextMenuEvent(QContextMenuEvent *e) override;
void paintUserpic(Painter &p);
const not_null<Main::Session*> _session;
const style::Menu &_st;
std::shared_ptr<Data::CloudImageView> _userpicView;
InMemoryKey _userpicKey = {};
QImage _userpicCache;
base::unique_qptr<Ui::PopupMenu> _menu;
Dialogs::Layout::UnreadBadgeStyle _unreadSt;
int _unreadBadge = 0;
bool _unreadBadgeMuted = true;
};
class MainMenu::ToggleAccountsButton final : public Ui::AbstractButton {
public:
explicit ToggleAccountsButton(QWidget *parent);
[[nodiscard]] int rightSkip() const {
return _rightSkip.current();
}
private:
void paintEvent(QPaintEvent *e) override;
void paintUnreadBadge(QPainter &p);
void validateUnreadBadge();
[[nodiscard]] QString computeUnreadBadge() const;
rpl::variable<int> _rightSkip;
Ui::Animations::Simple _toggledAnimation;
bool _toggled = false;
QString _unreadBadge;
int _unreadBadgeWidth = 0;
bool _unreadBadgeStale = false;
};
class MainMenu::ResetScaleButton final : public Ui::AbstractButton {
public:
ResetScaleButton(QWidget *parent);
protected:
void paintEvent(QPaintEvent *e) override;
static constexpr auto kText = "100%";
};
MainMenu::AccountButton::AccountButton(
QWidget *parent,
not_null<Main::Account*> account)
: RippleButton(parent, st::defaultRippleAnimation)
, _session(&account->session())
, _st(st::mainMenu){
const auto height = _st.itemPadding.top()
+ _st.itemStyle.font->height
+ _st.itemPadding.bottom();
resize(width(), height);
subscribe(Window::Theme::Background(), [=](
const Window::Theme::BackgroundUpdate &update) {
if (update.paletteChanged()) {
_userpicKey = {};
}
});
rpl::single(
rpl::empty_value()
) | rpl::then(
_session->data().unreadBadgeChanges()
) | rpl::start_with_next([=] {
_unreadBadge = _session->data().unreadBadge();
_unreadBadgeMuted = _session->data().unreadBadgeMuted();
update();
}, lifetime());
}
void MainMenu::AccountButton::paintUserpic(Painter &p) {
const auto size = st::mainMenuAccountSize;
const auto iconSize = height() - 2 * _st.itemIconPosition.y();
const auto shift = (size - iconSize) / 2;
const auto x = _st.itemIconPosition.x() - shift;
const auto y = (height() - size) / 2;
const auto check = (&_session->account()
== &Core::App().domain().active());
const auto user = _session->user();
if (!check) {
user->paintUserpicLeft(p, _userpicView, x, y, width(), size);
return;
}
const auto added = st::mainMenuAccountCheck.size;
const auto cacheSize = QSize(size + added, size + added)
* cIntRetinaFactor();
const auto key = user->userpicUniqueKey(_userpicView);
if (_userpicKey != key) {
_userpicKey = key;
if (_userpicCache.size() != cacheSize) {
_userpicCache = QImage(cacheSize, QImage::Format_ARGB32_Premultiplied);
_userpicCache.setDevicePixelRatio(cRetinaFactor());
}
_userpicCache.fill(Qt::transparent);
auto q = Painter(&_userpicCache);
user->paintUserpicLeft(q, _userpicView, 0, 0, width(), size);
const auto iconDiameter = st::mainMenuAccountCheck.size;
const auto iconLeft = size + st::mainMenuAccountCheckPosition.x() - iconDiameter;
const auto iconTop = size + st::mainMenuAccountCheckPosition.y() - iconDiameter;
const auto iconEllipse = QRect(iconLeft, iconTop, iconDiameter, iconDiameter);
auto iconBorderPen = QPen(Qt::transparent);
const auto line = st::mainMenuAccountCheckLine;
iconBorderPen.setWidth(line);
PainterHighQualityEnabler hq(q);
q.setCompositionMode(QPainter::CompositionMode_Source);
q.setPen(iconBorderPen);
q.setBrush(st::dialogsUnreadBg);
q.drawEllipse(iconEllipse);
q.setCompositionMode(QPainter::CompositionMode_SourceOver);
st::mainMenuAccountCheck.check.paintInCenter(q, iconEllipse);
}
p.drawImage(x, y, _userpicCache);
}
void MainMenu::AccountButton::paintEvent(QPaintEvent *e) {
auto p = Painter(this);
const auto over = isOver();
p.fillRect(rect(), over ? _st.itemBgOver : _st.itemBg);
paintRipple(p, 0, 0);
paintUserpic(p);
auto available = width() - _st.itemPadding.left();
if (_unreadBadge
&& (&_session->account() != &Core::App().activeAccount())) {
_unreadSt.muted = _unreadBadgeMuted;
const auto string = (_unreadBadge > 99)
? "99+"
: QString::number(_unreadBadge);
auto unreadWidth = 0;
const auto skip = _st.itemPadding.right()
- st::mainMenu.itemToggleShift;
const auto unreadRight = width() - skip;
const auto unreadTop = (height() - _unreadSt.size) / 2;
Dialogs::Layout::paintUnreadCount(
p,
string,
unreadRight,
unreadTop,
_unreadSt,
&unreadWidth);
available -= unreadWidth + skip + st::mainMenu.itemStyle.font->spacew;
} else {
available -= _st.itemPadding.right();
}
p.setPen(over ? _st.itemFgOver : _st.itemFg);
_session->user()->nameText().drawElided(
p,
_st.itemPadding.left(),
_st.itemPadding.top(),
available);
}
void MainMenu::AccountButton::contextMenuEvent(QContextMenuEvent *e) {
if (!_menu && IsAltShift(e->modifiers())) {
_menu = base::make_unique_q<Ui::PopupMenu>(this);
const auto addAction = [&](const QString &text, Fn<void()> callback) {
return _menu->addAction(
text,
crl::guard(this, std::move(callback)));
};
MenuAddMarkAsReadAllChatsAction(&_session->data(), addAction);
_menu->popup(QCursor::pos());
return;
}
if (&_session->account() == &Core::App().activeAccount() || _menu) {
return;
}
_menu = base::make_unique_q<Ui::PopupMenu>(this);
_menu->addAction(tr::lng_menu_activate(tr::now), crl::guard(this, [=] {
Core::App().domain().activate(&_session->account());
}));
_menu->addAction(tr::lng_settings_logout(tr::now), crl::guard(this, [=] {
const auto session = _session;
const auto callback = [=](Fn<void()> &&close) {
close();
Core::App().logout(&session->account());
};
Ui::show(Box<ConfirmBox>(
tr::lng_sure_logout(tr::now),
tr::lng_settings_logout(tr::now),
st::attentionBoxButton,
crl::guard(session, callback)));
}));
_menu->popup(QCursor::pos());
}
MainMenu::ToggleAccountsButton::ToggleAccountsButton(QWidget *parent)
: AbstractButton(parent) {
rpl::single(
rpl::empty_value()
) | rpl::then(
Core::App().unreadBadgeChanges()
) | rpl::start_with_next([=] {
_unreadBadgeStale = true;
if (!_toggled) {
validateUnreadBadge();
update();
}
}, lifetime());
auto &settings = Core::App().settings();
if (Core::App().domain().accounts().size() < 2
&& settings.mainMenuAccountsShown()) {
settings.setMainMenuAccountsShown(false);
}
settings.mainMenuAccountsShownValue(
) | rpl::filter([=](bool value) {
return (_toggled != value);
}) | rpl::start_with_next([=](bool value) {
_toggled = value;
_toggledAnimation.start(
[=] { update(); },
_toggled ? 0. : 1.,
_toggled ? 1. : 0.,
st::slideWrapDuration);
validateUnreadBadge();
}, lifetime());
_toggledAnimation.stop();
}
void MainMenu::ToggleAccountsButton::paintEvent(QPaintEvent *e) {
auto p = Painter(this);
const auto toggled = _toggledAnimation.value(_toggled ? 1. : 0.);
const auto x = 0. + width() - st::mainMenuTogglePosition.x();
const auto y = 0. + height() - st::mainMenuTogglePosition.y();
const auto size = st::mainMenuToggleSize;
const auto size2 = size / 2.;
const auto sqrt2 = sqrt(2.);
const auto stroke = (st::mainMenuToggleFourStrokes / 4.) / sqrt2;
const auto left = x - size;
const auto right = x + size;
const auto bottom = y + size2;
const auto top = y - size2;
constexpr auto kPointCount = 6;
std::array<QPointF, kPointCount> points = { {
{ left - stroke, bottom - stroke },
{ x, bottom - stroke - size - stroke },
{ right + stroke, bottom - stroke },
{ right - stroke, bottom + stroke },
{ x, bottom + stroke - size + stroke },
{ left + stroke, bottom + stroke }
} };
const auto alpha = (toggled - 1.) * M_PI;
const auto cosalpha = cos(alpha);
const auto sinalpha = sin(alpha);
for (auto &point : points) {
auto px = point.x() - x;
auto py = point.y() - y;
point.setX(x + px * cosalpha - py * sinalpha);
point.setY(y + py * cosalpha + px * sinalpha);
}
QPainterPath path;
path.moveTo(points[0]);
for (int i = 1; i != kPointCount; ++i) {
path.lineTo(points[i]);
}
path.lineTo(points[0]);
auto hq = PainterHighQualityEnabler(p);
p.fillPath(path, st::mainMenuCoverFg);
paintUnreadBadge(p);
}
void MainMenu::ToggleAccountsButton::paintUnreadBadge(QPainter &p) {
const auto progress = 1. - _toggledAnimation.value(_toggled ? 1. : 0.);
if (!progress) {
return;
}
validateUnreadBadge();
if (_unreadBadge.isEmpty()) {
return;
}
Dialogs::Layout::UnreadBadgeStyle st;
const auto right = width() - st::mainMenuTogglePosition.x() - st::mainMenuToggleSize * 2;
const auto top = height() - st::mainMenuTogglePosition.y() - st::mainMenuToggleSize;
const auto width = _unreadBadgeWidth;
const auto rectHeight = st.size;
const auto rectWidth = std::max(width + 2 * st.padding, rectHeight);
const auto left = right - rectWidth;
const auto textLeft = left + (rectWidth - width) / 2;
const auto textTop = top + (st.textTop ? st.textTop : (rectHeight - st.font->height) / 2);
const auto isFill = IsFilledCover();
auto hq = PainterHighQualityEnabler(p);
auto brush = (isFill ? st::mainMenuCloudBg : st::msgServiceBg)->c;
brush.setAlphaF(progress * brush.alphaF());
p.setBrush(brush);
p.setPen(Qt::NoPen);
p.drawRoundedRect(left, top, rectWidth, rectHeight, rectHeight / 2, rectHeight / 2);
p.setFont(st.font);
auto pen = (isFill ? st::mainMenuCloudFg : st::msgServiceFg)->c;
pen.setAlphaF(progress * pen.alphaF());
p.setPen(pen);
p.drawText(textLeft, textTop + st.font->ascent, _unreadBadge);
}
void MainMenu::ToggleAccountsButton::validateUnreadBadge() {
const auto base = st::mainMenuTogglePosition.x()
+ 2 * st::mainMenuToggleSize;
if (_toggled) {
_rightSkip = base;
return;
} else if (!_unreadBadgeStale) {
return;
}
_unreadBadge = computeUnreadBadge();
Dialogs::Layout::UnreadBadgeStyle st;
_unreadBadgeWidth = st.font->width(_unreadBadge);
const auto rectHeight = st.size;
const auto rectWidth = std::max(
_unreadBadgeWidth + 2 * st.padding,
rectHeight);
_rightSkip = base + rectWidth + st::mainMenuToggleSize;
}
QString MainMenu::ToggleAccountsButton::computeUnreadBadge() const {
const auto state = OtherAccountsUnreadStateCurrent();
return state.allMuted
? QString()
: (state.count > 99)
? u"99+"_q
: (state.count > 0)
? QString::number(state.count)
: QString();
}
MainMenu::ResetScaleButton::ResetScaleButton(QWidget *parent)
: AbstractButton(parent) {
const auto margin = st::mainMenuCloudButton.height
- st::mainMenuCloudSize;
const auto textWidth = st::mainMenuResetScaleFont->width(kText);
const auto innerWidth = st::mainMenuResetScaleLeft
+ textWidth
+ st::mainMenuResetScaleRight;
const auto width = margin + innerWidth;
resize(width, st::mainMenuCloudButton.height);
}
void MainMenu::ResetScaleButton::paintEvent(QPaintEvent *e) {
Painter p(this);
const auto innerHeight = st::mainMenuCloudSize;
const auto radius = innerHeight / 2;
const auto margin = st::mainMenuCloudButton.height
- st::mainMenuCloudSize;
const auto textWidth = st::mainMenuResetScaleFont->width(kText);
const auto innerWidth = st::mainMenuResetScaleLeft
+ textWidth
+ st::mainMenuResetScaleRight;
const auto left = margin / 2;
const auto top = margin / 2;
p.setPen(Qt::NoPen);
p.setBrush(st::mainMenuCloudBg);
p.drawRoundedRect(left, top, innerWidth, innerHeight, radius, radius);
st::settingsIconInterfaceScale.paint(
p,
left + st::mainMenuResetScaleIconLeft,
top + ((innerHeight - st::settingsIconInterfaceScale.height()) / 2),
width(),
st::mainMenuCloudFg->c);
p.setFont(st::mainMenuResetScaleFont);
p.setPen(st::mainMenuCloudFg);
p.drawText(
left + st::mainMenuResetScaleLeft,
top + st::mainMenuResetScaleTop + st::mainMenuResetScaleFont->ascent,
kText);
}
MainMenu::MainMenu(
QWidget *parent,
not_null<SessionController*> controller)
: LayerWidget(parent)
, _controller(controller)
, _userpicButton(
this,
_controller,
_controller->session().user(),
Ui::UserpicButton::Role::Custom,
st::mainMenuUserpic)
, _toggleAccounts(this)
, _archiveButton(this, st::mainMenuCloudButton)
, _cloudButton(this, st::mainMenuCloudButton)
, _scroll(this, st::defaultSolidScroll)
, _inner(_scroll->setOwnedWidget(
object_ptr<Ui::VerticalLayout>(_scroll.data())))
, _accounts(_inner->add(object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
_inner.get(),
object_ptr<Ui::VerticalLayout>(_inner.get()))))
, _shadow(_inner->add(object_ptr<Ui::SlideWrap<Ui::PlainShadow>>(
_inner.get(),
object_ptr<Ui::PlainShadow>(_inner.get()))))
, _menu(_inner->add(
object_ptr<Ui::Menu::Menu>(_inner.get(), st::mainMenu),
{ 0, st::mainMenuSkip, 0, 0 }))
, _footer(_inner->add(object_ptr<Ui::RpWidget>(_inner.get())))
, _telegram(
Ui::CreateChild<Ui::FlatLabel>(_footer.get(), st::mainMenuTelegramLabel))
, _version(
Ui::CreateChild<Ui::FlatLabel>(
_footer.get(),
st::mainMenuVersionLabel)) {
setAttribute(Qt::WA_OpaquePaintEvent);
setupArchiveButton();
setupCloudButton();
setupUserpicButton();
setupAccountsToggle();
setupAccounts();
_nightThemeSwitch.setCallback([this] {
if (const auto action = *_nightThemeAction) {
const auto nightMode = Window::Theme::IsNightMode();
if (action->isChecked() != nightMode) {
Window::Theme::ToggleNightMode();
Window::Theme::KeepApplied();
}
}
});
_footer->heightValue(
) | rpl::start_with_next([=] {
_telegram->moveToLeft(st::mainMenuFooterLeft, _footer->height() - st::mainMenuTelegramBottom - _telegram->height());
_version->moveToLeft(st::mainMenuFooterLeft, _footer->height() - st::mainMenuVersionBottom - _version->height());
}, _footer->lifetime());
rpl::combine(
heightValue(),
_inner->heightValue()
) | rpl::start_with_next([=] {
updateInnerControlsGeometry();
}, _inner->lifetime());
parentResized();
_menu->setTriggeredCallback([](const Ui::Menu::CallbackData &data) {
data.action->triggered();
});
refreshMenu();
refreshBackground();
_telegram->setMarkedText(Ui::Text::Link(
qsl("Telegram Desktop"),
qsl("https://desktop.telegram.org")));
_telegram->setLinksTrusted();
_version->setRichText(textcmdLink(1, tr::lng_settings_current_version(tr::now, lt_version, currentVersionText())) + QChar(' ') + QChar(8211) + QChar(' ') + textcmdLink(2, tr::lng_menu_about(tr::now)));
_version->setLink(1, std::make_shared<UrlClickHandler>(Core::App().changelogLink()));
_version->setLink(2, std::make_shared<LambdaClickHandler>([] { Ui::show(Box<AboutBox>()); }));
_controller->session().downloaderTaskFinished(
) | rpl::start_with_next([=] {
update();
}, lifetime());
_controller->session().changes().peerUpdates(
_controller->session().user(),
Data::PeerUpdate::Flag::PhoneNumber
) | rpl::start_with_next([=] {
updatePhone();
}, lifetime());
_controller->session().serverConfig().phoneCallsEnabled.changes(
) | rpl::start_with_next([=] {
refreshMenu();
}, lifetime());
subscribe(Window::Theme::Background(), [this](const Window::Theme::BackgroundUpdate &update) {
if (update.type == Window::Theme::BackgroundUpdate::Type::ApplyingTheme) {
if (const auto action = *_nightThemeAction) {
const auto nightMode = Window::Theme::IsNightMode();
if (action->isChecked() != nightMode) {
action->setChecked(nightMode);
_menu->finishAnimating();
}
}
}
if (update.type == Window::Theme::BackgroundUpdate::Type::New) {
refreshBackground();
}
});
updatePhone();
initResetScaleButton();
}
void MainMenu::setupArchiveButton() {
const auto controller = _controller;
const auto folder = [=] {
return controller->session().data().folderLoaded(Data::Folder::kId);
};
const auto showArchive = [=] {
if (const auto f = folder()) {
controller->openFolder(f);
Ui::hideSettingsAndLayer();
}
};
const auto checkArchive = [=] {
const auto f = folder();
return f
&& !f->chatsList()->empty()
&& controller->session().settings().archiveInMainMenu();
};
_archiveButton->setVisible(checkArchive());
_archiveButton->setAcceptBoth(true);
_archiveButton->clicks(
) | rpl::start_with_next([=](Qt::MouseButton which) {
if (which == Qt::LeftButton) {
showArchive();
return;
} else if (which != Qt::RightButton) {
return;
}
_contextMenu = base::make_unique_q<Ui::PopupMenu>(this);
const auto addAction = [&](const QString &text, Fn<void()> callback) {
return _contextMenu->addAction(text, std::move(callback));
};
const auto hide = [=] {
controller->session().settings().setArchiveInMainMenu(false);
controller->session().saveSettingsDelayed();
Ui::hideSettingsAndLayer();
};
addAction(tr::lng_context_archive_to_list(tr::now), std::move(hide));
MenuAddMarkAsReadChatListAction(
[f = folder()] { return f->chatsList(); },
addAction);
_contextMenu->popup(QCursor::pos());
}, _archiveButton->lifetime());
controller->session().data().chatsListChanges(
) | rpl::filter([](Data::Folder *folder) {
return folder && (folder->id() == Data::Folder::kId);
}) | rpl::start_with_next([=](Data::Folder *folder) {
const auto isArchiveVisible = checkArchive();
_archiveButton->setVisible(isArchiveVisible);
if (!isArchiveVisible) {
_contextMenu = nullptr;
}
update();
}, lifetime());
}
void MainMenu::setupCloudButton() {
_cloudButton->setClickedCallback([=] {
_controller->content()->choosePeer(
_controller->session().userPeerId(),
ShowAtUnreadMsgId);
});
_cloudButton->show();
}
void MainMenu::setupUserpicButton() {
_userpicButton->setClickedCallback([=] { toggleAccounts(); });
_userpicButton->show();
}
void MainMenu::toggleAccounts() {
auto &settings = Core::App().settings();
const auto shown = !settings.mainMenuAccountsShown();
settings.setMainMenuAccountsShown(shown);
Core::App().saveSettingsDelayed();
}
void MainMenu::setupAccounts() {
const auto inner = _accounts->entity();
inner->add(object_ptr<Ui::FixedHeightWidget>(inner, st::mainMenuSkip));
_addAccount = setupAddAccount(inner);
inner->add(object_ptr<Ui::FixedHeightWidget>(inner, st::mainMenuSkip));
rpl::single(
rpl::empty_value()
) | rpl::then(Core::App().domain().accountsChanges(
)) | rpl::start_with_next([=] {
const auto &list = Core::App().domain().accounts();
const auto exists = [&](not_null<Main::Account*> account) {
for (const auto &[index, existing] : list) {
if (account == existing.get()) {
return true;
}
}
return false;
};
for (auto i = _watched.begin(); i != _watched.end();) {
if (!exists(i->first)) {
i = _watched.erase(i);
} else {
++i;
}
}
for (const auto &[index, account] : list) {
if (_watched.emplace(account.get()).second) {
account->sessionChanges(
) | rpl::start_with_next([=](Main::Session *session) {
rebuildAccounts();
}, lifetime());
}
}
rebuildAccounts();
}, lifetime());
_accounts->toggleOn(Core::App().settings().mainMenuAccountsShownValue());
_accounts->finishAnimating();
_shadow->setDuration(0)->toggleOn(_accounts->shownValue());
}
void MainMenu::rebuildAccounts() {
const auto inner = _accounts->entity();
auto count = 0;
for (const auto &[index, pointer] : Core::App().domain().accounts()) {
const auto account = pointer.get();
auto i = _watched.find(account);
Assert(i != _watched.end());
auto &button = i->second;
if (!account->sessionExists()) {
button = nullptr;
} else if (!button) {
button.reset(inner->insert(
++count,
object_ptr<AccountButton>(inner, account)));
button->setClickedCallback([=] {
if (account == &Core::App().domain().active()) {
closeLayer();
return;
}
auto activate = [=, guard = _accountSwitchGuard.make_guard()]{
if (guard) {
Core::App().domain().maybeActivate(account);
}
};
base::call_delayed(
st::defaultRippleAnimation.hideDuration,
account,
std::move(activate));
});
} else {
++count;
}
}
inner->resizeToWidth(_accounts->width());
_addAccount->toggle(
(count < Main::Domain::kMaxAccounts),
anim::type::instant);
}
not_null<Ui::SlideWrap<Ui::RippleButton>*> MainMenu::setupAddAccount(
not_null<Ui::VerticalLayout*> container) {
const auto result = container->add(
object_ptr<Ui::SlideWrap<Ui::RippleButton>>(
container.get(),
object_ptr<Ui::RippleButton>(
container.get(),
st::defaultRippleAnimation)))->setDuration(0);
const auto st = &st::mainMenu;
const auto height = st->itemPadding.top()
+ st->itemStyle.font->height
+ st->itemPadding.bottom();
const auto button = result->entity();
button->resize(button->width(), height);
button->paintRequest(
) | rpl::start_with_next([=] {
auto p = Painter(button);
const auto over = button->isOver();
p.fillRect(button->rect(), over ? st->itemBgOver : st->itemBg);
button->paintRipple(p, 0, 0);
const auto &icon = over
? st::mainMenuAddAccountOver
: st::mainMenuAddAccount;
icon.paint(p, st->itemIconPosition, width());
p.setPen(over ? st->itemFgOver : st->itemFg);
p.setFont(st->itemStyle.font);
p.drawTextLeft(
st->itemPadding.left(),
st->itemPadding.top(),
width(),
tr::lng_menu_add_account(tr::now));
}, button->lifetime());
const auto add = [=](MTP::Environment environment) {
Core::App().preventOrInvoke([=] {
Core::App().domain().addActivated(environment);
});
};
button->setAcceptBoth(true);
button->clicks(
) | rpl::start_with_next([=](Qt::MouseButton which) {
if (which == Qt::LeftButton) {
add(MTP::Environment::Production);
return;
} else if (which != Qt::RightButton
|| !IsAltShift(button->clickModifiers())) {
return;
}
_contextMenu = base::make_unique_q<Ui::PopupMenu>(this);
_contextMenu->addAction("Production Server", [=] {
add(MTP::Environment::Production);
});
_contextMenu->addAction("Test Server", [=] {
add(MTP::Environment::Test);
});
_contextMenu->popup(QCursor::pos());
}, button->lifetime());
return result;
}
void MainMenu::setupAccountsToggle() {
_toggleAccounts->show();
_toggleAccounts->setClickedCallback([=] { toggleAccounts(); });
}
void MainMenu::parentResized() {
resize(st::mainMenuWidth, parentWidget()->height());
}
void MainMenu::refreshMenu() {
_menu->clearActions();
const auto controller = _controller;
if (!_controller->session().supportMode()) {
_menu->addAction(tr::lng_create_group_title(tr::now), [=] {
controller->showNewGroup();
}, &st::mainMenuNewGroup, &st::mainMenuNewGroupOver);
_menu->addAction(tr::lng_create_channel_title(tr::now), [=] {
controller->showNewChannel();
}, &st::mainMenuNewChannel, &st::mainMenuNewChannelOver);
_menu->addAction(tr::lng_menu_contacts(tr::now), [=] {
Ui::show(PrepareContactsBox(controller));
}, &st::mainMenuContacts, &st::mainMenuContactsOver);
if (_controller->session().serverConfig().phoneCallsEnabled.current()) {
_menu->addAction(tr::lng_menu_calls(tr::now), [=] {
ShowCallsBox(controller);
}, &st::mainMenuCalls, &st::mainMenuCallsOver);
}
} else {
_menu->addAction(tr::lng_profile_add_contact(tr::now), [=] {
controller->showAddContact();
}, &st::mainMenuContacts, &st::mainMenuContactsOver);
const auto fix = std::make_shared<QPointer<QAction>>();
*fix = _menu->addAction(qsl("Fix chats order"), [=] {
(*fix)->setChecked(!(*fix)->isChecked());
_controller->session().settings().setSupportFixChatsOrder(
(*fix)->isChecked());
_controller->session().saveSettings();
}, &st::mainMenuFixOrder, &st::mainMenuFixOrderOver);
(*fix)->setCheckable(true);
(*fix)->setChecked(
_controller->session().settings().supportFixChatsOrder());
_menu->addAction(qsl("Reload templates"), [=] {
_controller->session().supportTemplates().reload();
}, &st::mainMenuReload, &st::mainMenuReloadOver);
}
_menu->addAction(tr::lng_menu_settings(tr::now), [=] {
controller->showSettings();
}, &st::mainMenuSettings, &st::mainMenuSettingsOver);
_nightThemeAction = std::make_shared<QPointer<QAction>>();
auto nightCallback = [=] {
if (Window::Theme::Background()->editingTheme()) {
Ui::show(Box<InformBox>(
tr::lng_theme_editor_cant_change_theme(tr::now)));
return;
}
const auto weak = MakeWeak(this);
const auto toggle = [=] {
if (!weak) {
Window::Theme::ToggleNightMode();
Window::Theme::KeepApplied();
} else if (auto action = *_nightThemeAction) {
action->setChecked(!action->isChecked());
_nightThemeSwitch.callOnce(st::mainMenu.itemToggle.duration);
}
};
Window::Theme::ToggleNightModeWithConfirmation(
&_controller->window(),
toggle);
};
auto item = base::make_unique_q<Ui::Menu::Toggle>(
_menu,
st::mainMenu,
tr::lng_menu_night_mode(tr::now),
std::move(nightCallback),
&st::mainMenuNightMode,
&st::mainMenuNightModeOver);
auto action = _menu->addAction(std::move(item));
*_nightThemeAction = action;
action->setCheckable(true);
action->setChecked(Window::Theme::IsNightMode());
Core::App().settings().systemDarkModeValue(
) | rpl::start_with_next([=](std::optional<bool> darkMode) {
const auto darkModeEnabled = Core::App().settings().systemDarkModeEnabled();
if (darkModeEnabled && darkMode.has_value()) {
action->setChecked(*darkMode);
}
}, lifetime());
_menu->finishAnimating();
updatePhone();
}
void MainMenu::refreshBackground() {
const auto fill = QRect(0, 0, width(), st::mainMenuCoverHeight);
const auto intensityText = IntensityOfColor(st::mainMenuCoverFg->c);
QImage backgroundImage(
st::mainMenuWidth * cIntRetinaFactor(),
st::mainMenuCoverHeight * cIntRetinaFactor(),
QImage::Format_ARGB32_Premultiplied);
QPainter p(&backgroundImage);
const auto drawShadow = [](QPainter &p) {
st::mainMenuShadow.paint(
p,
0,
st::mainMenuCoverHeight - st::mainMenuShadow.height(),
st::mainMenuWidth,
IntensityOfColor(st::mainMenuCoverFg->c) < 0.5
? Qt::white
: Qt::black);
};
// Solid color.
if (const auto color = Window::Theme::Background()->colorForFill()) {
const auto intensity = IntensityOfColor(*color);
p.fillRect(fill, *color);
if (std::abs(intensity - intensityText) < kMinDiffIntensity) {
drawShadow(p);
}
_background = backgroundImage;
return;
}
// Background image.
const auto &pixmap = Window::Theme::Background()->pixmap();
QRect to, from;
Window::Theme::ComputeBackgroundRects(fill, pixmap.size(), to, from);
// Cut off the part of the background that is under text.
const QRect underText(
st::mainMenuCoverTextLeft,
st::mainMenuCoverNameTop,
std::max(
st::semiboldFont->width(
_controller->session().user()->nameText().toString()),
st::normalFont->width(_phoneText)),
st::semiboldFont->height * 2);
p.drawPixmap(to, pixmap, from);
if (IsShadowShown(backgroundImage, underText, intensityText)) {
drawShadow(p);
}
_background = backgroundImage;
}
void MainMenu::resizeEvent(QResizeEvent *e) {
_menu->setForceWidth(width());
_inner->resizeToWidth(width());
updateControlsGeometry();
}
void MainMenu::updateControlsGeometry() {
_userpicButton->moveToLeft(
st::mainMenuUserpicLeft,
st::mainMenuUserpicTop);
if (_resetScaleButton) {
_resetScaleButton->moveToRight(0, 0);
_cloudButton->moveToRight(_resetScaleButton->width(), 0);
_archiveButton->moveToRight(
_resetScaleButton->width() + _cloudButton->width(),
0);
} else {
const auto right = st::mainMenuTogglePosition.x()
- (_cloudButton->width() / 2);
const auto top = st::mainMenuUserpicTop
- (_cloudButton->height() - st::mainMenuCloudSize) / 2;
_cloudButton->moveToRight(right, top);
_archiveButton->moveToRight(right + _cloudButton->width(), top);
}
_toggleAccounts->setGeometry(
0,
st::mainMenuCoverNameTop,
width(),
st::mainMenuCoverHeight - st::mainMenuCoverNameTop);
const auto top = st::mainMenuCoverHeight;
_scroll->setGeometry(0, top, width(), height() - top);
updateInnerControlsGeometry();
}
void MainMenu::updateInnerControlsGeometry() {
const auto contentHeight = _accounts->height()
+ _shadow->height()
+ st::mainMenuSkip
+ _menu->height();
const auto available = height() - st::mainMenuCoverHeight - contentHeight;
const auto footerHeight = std::max(
available,
st::mainMenuTelegramBottom + _telegram->height() + st::mainMenuSkip);
if (_footer->height() != footerHeight) {
_footer->resize(_footer->width(), footerHeight);
}
}
void MainMenu::updatePhone() {
_phoneText = App::formatPhone(_controller->session().user()->phone());
update();
}
void MainMenu::paintEvent(QPaintEvent *e) {
Painter p(this);
const auto clip = e->rect();
const auto cover = QRect(0, 0, width(), st::mainMenuCoverHeight)
.intersected(e->rect());
const auto isFill = IsFilledCover();
if (!isFill && !_background.isNull()) {
PainterHighQualityEnabler hq(p);
p.drawImage(0, 0, _background);
}
if (!cover.isEmpty()) {
const auto widthText = width()
- st::mainMenuCoverTextLeft
- _toggleAccounts->rightSkip();
if (isFill) {
p.fillRect(cover, st::mainMenuCoverBg);
}
p.setPen(st::mainMenuCoverFg);
p.setFont(st::semiboldFont);
_controller->session().user()->nameText().drawLeftElided(
p,
st::mainMenuCoverTextLeft,
st::mainMenuCoverNameTop,
widthText,
width());
p.setFont(st::normalFont);
p.drawTextLeft(st::mainMenuCoverTextLeft, st::mainMenuCoverStatusTop, width(), _phoneText);
// Draw Saved Messages button.
if (!_cloudButton->isHidden()) {
Ui::EmptyUserpic::PaintSavedMessages(
p,
_cloudButton->x() + (_cloudButton->width() - st::mainMenuCloudSize) / 2,
_cloudButton->y() + (_cloudButton->height() - st::mainMenuCloudSize) / 2,
width(),
st::mainMenuCloudSize,
isFill ? st::mainMenuCloudBg : st::msgServiceBg,
isFill ? st::mainMenuCloudFg : st::msgServiceFg);
}
// Draw Archive button.
if (!_archiveButton->isHidden()) {
const auto folder = _controller->session().data().folderLoaded(
Data::Folder::kId);
if (folder) {
folder->paintUserpic(
p,
_archiveButton->x() + (_archiveButton->width() - st::mainMenuCloudSize) / 2,
_archiveButton->y() + (_archiveButton->height() - st::mainMenuCloudSize) / 2,
st::mainMenuCloudSize,
isFill ? st::mainMenuCloudBg : st::msgServiceBg,
isFill ? st::mainMenuCloudFg : st::msgServiceFg);
}
}
}
auto other = QRect(0, st::mainMenuCoverHeight, width(), height() - st::mainMenuCoverHeight).intersected(clip);
if (!other.isEmpty()) {
p.fillRect(other, st::mainMenuBg);
}
}
void MainMenu::initResetScaleButton() {
if (!window() || !window()->windowHandle()) {
return;
}
const auto handle = window()->windowHandle();
rpl::single(
handle->screen()
) | rpl::then(
base::qt_signal_producer(handle, &QWindow::screenChanged)
) | rpl::filter([](QScreen *screen) {
return screen != nullptr;
}) | rpl::map([](QScreen * screen) {
return rpl::single(
screen->availableGeometry()
) | rpl::then(
#ifdef OS_MAC_OLD
base::qt_signal_producer(screen, &QScreen::virtualGeometryChanged)
#else // OS_MAC_OLD
base::qt_signal_producer(screen, &QScreen::availableGeometryChanged)
#endif // OS_MAC_OLD
);
}) | rpl::flatten_latest(
) | rpl::map([](QRect available) {
return (available.width() >= st::windowMinWidth)
&& (available.height() >= st::windowMinHeight);
}) | rpl::distinct_until_changed(
) | rpl::start_with_next([=](bool good) {
if (good) {
_resetScaleButton.destroy();
} else {
_resetScaleButton.create(this);
_resetScaleButton->addClickHandler([] {
cSetConfigScale(style::kScaleDefault);
Local::writeSettings();
App::restart();
});
_resetScaleButton->show();
updateControlsGeometry();
}
}, lifetime());
}
OthersUnreadState OtherAccountsUnreadStateCurrent() {
auto &app = Core::App();
const auto active = &app.activeAccount();
auto allMuted = true;
for (const auto &[index, account] : app.domain().accounts()) {
if (account.get() == active) {
continue;
} else if (const auto session = account->maybeSession()) {
if (!session->data().unreadBadgeMuted()) {
allMuted = false;
break;
}
}
}
return {
.count = (app.unreadBadge() - active->session().data().unreadBadge()),
.allMuted = allMuted,
};
}
rpl::producer<OthersUnreadState> OtherAccountsUnreadState() {
return rpl::single(
rpl::empty_value()
) | rpl::then(
Core::App().unreadBadgeChanges()
) | rpl::map(OtherAccountsUnreadStateCurrent);
}
} // namespace Window