From 3deea14559441ded8b29d29252c2c22dc704c5f2 Mon Sep 17 00:00:00 2001 From: John Preston Date: Mon, 13 Nov 2017 12:05:56 +0400 Subject: [PATCH] Remove old shared media overview section. Also move window/top_bar_widget to history/history_top_bar_widget. --- Telegram/Resources/langs/lang.strings | 2 - .../SourceFiles/data/data_shared_media.cpp | 8 +- Telegram/SourceFiles/facades.cpp | 6 - Telegram/SourceFiles/facades.h | 8 - Telegram/SourceFiles/history/history.cpp | 1 - .../history/history_inner_widget.cpp | 4 +- .../history/history_inner_widget.h | 4 +- .../history_top_bar_widget.cpp} | 358 ++- .../history_top_bar_widget.h} | 41 +- .../SourceFiles/history/history_widget.cpp | 163 +- Telegram/SourceFiles/history/history_widget.h | 16 +- Telegram/SourceFiles/mainwidget.cpp | 310 +-- Telegram/SourceFiles/mainwidget.h | 20 +- Telegram/SourceFiles/mediaview.h | 2 - Telegram/SourceFiles/overviewwidget.cpp | 2432 ----------------- Telegram/SourceFiles/overviewwidget.h | 399 --- .../profile/profile_back_button.cpp | 4 +- .../profile/profile_block_shared_media.cpp | 2 +- Telegram/gyp/telegram_sources.txt | 6 +- 19 files changed, 361 insertions(+), 3425 deletions(-) rename Telegram/SourceFiles/{window/top_bar_widget.cpp => history/history_top_bar_widget.cpp} (56%) rename Telegram/SourceFiles/{window/top_bar_widget.h => history/history_top_bar_widget.h} (77%) delete mode 100644 Telegram/SourceFiles/overviewwidget.cpp delete mode 100644 Telegram/SourceFiles/overviewwidget.h diff --git a/Telegram/Resources/langs/lang.strings b/Telegram/Resources/langs/lang.strings index 105b633e7..862bcf6b5 100644 --- a/Telegram/Resources/langs/lang.strings +++ b/Telegram/Resources/langs/lang.strings @@ -854,8 +854,6 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org "lng_attach_file" = "File"; "lng_attach_photo" = "Photo"; -"lng_media_type" = "Media type"; - "lng_media_open_with" = "Open With"; "lng_media_download" = "Download"; "lng_media_cancel" = "Cancel"; diff --git a/Telegram/SourceFiles/data/data_shared_media.cpp b/Telegram/SourceFiles/data/data_shared_media.cpp index d7720318c..36ef3d5d3 100644 --- a/Telegram/SourceFiles/data/data_shared_media.cpp +++ b/Telegram/SourceFiles/data/data_shared_media.cpp @@ -27,6 +27,10 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org #include "storage/storage_shared_media.h" #include "history/history_media_types.h" #include "data/data_sparse_ids.h" +#include "info/info_memento.h" +#include "info/info_controller.h" +#include "window/window_controller.h" +#include "mainwindow.h" namespace { @@ -59,7 +63,9 @@ void SharedMediaShowOverview( Storage::SharedMediaType type, not_null history) { if (SharedMediaOverviewType(type)) { - Ui::showPeerOverview(history, SharedMediaTypeToOverview(type)); + App::wnd()->controller()->showSection(Info::Memento( + history->peer->id, + Info::Section(type))); } } diff --git a/Telegram/SourceFiles/facades.cpp b/Telegram/SourceFiles/facades.cpp index 1249ffdb5..e546134ed 100644 --- a/Telegram/SourceFiles/facades.cpp +++ b/Telegram/SourceFiles/facades.cpp @@ -272,12 +272,6 @@ void showPeerProfile(const PeerId &peer) { } } -void showPeerOverview(const PeerId &peer, MediaOverviewType type) { - if (auto m = App::main()) { - m->showMediaOverview(App::peer(peer), type); - } -} - void showPeerHistory( const PeerId &peer, MsgId msgId) { diff --git a/Telegram/SourceFiles/facades.h b/Telegram/SourceFiles/facades.h index abe936159..ab50fc737 100644 --- a/Telegram/SourceFiles/facades.h +++ b/Telegram/SourceFiles/facades.h @@ -192,14 +192,6 @@ inline void showPeerProfile(const History *history) { showPeerProfile(history->peer->id); } -void showPeerOverview(const PeerId &peer, MediaOverviewType type); -inline void showPeerOverview(const PeerData *peer, MediaOverviewType type) { - showPeerOverview(peer->id, type); -} -inline void showPeerOverview(const History *history, MediaOverviewType type) { - showPeerOverview(history->peer->id, type); -} - void showPeerHistory(const PeerId &peer, MsgId msgId); inline void showPeerHistory(const PeerData *peer, MsgId msgId) { diff --git a/Telegram/SourceFiles/history/history.cpp b/Telegram/SourceFiles/history/history.cpp index b2b81de04..e17ef2482 100644 --- a/Telegram/SourceFiles/history/history.cpp +++ b/Telegram/SourceFiles/history/history.cpp @@ -31,7 +31,6 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org #include "mainwidget.h" #include "mainwindow.h" #include "storage/localstorage.h" -#include "window/top_bar_widget.h" #include "observer_peer.h" #include "auth_session.h" #include "window/notifications_manager.h" diff --git a/Telegram/SourceFiles/history/history_inner_widget.cpp b/Telegram/SourceFiles/history/history_inner_widget.cpp index a07d2bb2a..e795749cc 100644 --- a/Telegram/SourceFiles/history/history_inner_widget.cpp +++ b/Telegram/SourceFiles/history/history_inner_widget.cpp @@ -1953,8 +1953,8 @@ bool HistoryInner::canDeleteSelected() const { return (selectedState.count > 0) && (selectedState.count == selectedState.canDeleteCount); } -Window::TopBarWidget::SelectedState HistoryInner::getSelectionState() const { - auto result = Window::TopBarWidget::SelectedState {}; +HistoryTopBarWidget::SelectedState HistoryInner::getSelectionState() const { + auto result = HistoryTopBarWidget::SelectedState {}; for (auto &selected : _selected) { if (selected.second == FullSelection) { ++result.count; diff --git a/Telegram/SourceFiles/history/history_inner_widget.h b/Telegram/SourceFiles/history/history_inner_widget.h index 3ac202071..202e4f4db 100644 --- a/Telegram/SourceFiles/history/history_inner_widget.h +++ b/Telegram/SourceFiles/history/history_inner_widget.h @@ -23,7 +23,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org #include "ui/rp_widget.h" #include "ui/widgets/tooltip.h" #include "ui/widgets/scroll_area.h" -#include "window/top_bar_widget.h" +#include "history/history_top_bar_widget.h" namespace Window { class Controller; @@ -63,7 +63,7 @@ public: bool canCopySelected() const; bool canDeleteSelected() const; - Window::TopBarWidget::SelectedState getSelectionState() const; + HistoryTopBarWidget::SelectedState getSelectionState() const; void clearSelectedItems(bool onlyTextSelection = false); SelectedItemSet getSelectedItems() const; void selectItem(HistoryItem *item); diff --git a/Telegram/SourceFiles/window/top_bar_widget.cpp b/Telegram/SourceFiles/history/history_top_bar_widget.cpp similarity index 56% rename from Telegram/SourceFiles/window/top_bar_widget.cpp rename to Telegram/SourceFiles/history/history_top_bar_widget.cpp index 6a59b95c7..c61d38a84 100644 --- a/Telegram/SourceFiles/window/top_bar_widget.cpp +++ b/Telegram/SourceFiles/history/history_top_bar_widget.cpp @@ -18,11 +18,13 @@ to link the code of portions of this program with the OpenSSL library. Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org */ -#include "window/top_bar_widget.h" +#include "history/history_top_bar_widget.h" #include #include #include "styles/style_window.h" +#include "styles/style_dialogs.h" +#include "styles/style_history.h" #include "boxes/add_contact_box.h" #include "boxes/confirm_box.h" #include "info/info_memento.h" @@ -39,10 +41,9 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org #include "window/window_peer_menu.h" #include "calls/calls_instance.h" #include "observer_peer.h" +#include "apiwrap.h" -namespace Window { - -TopBarWidget::TopBarWidget( +HistoryTopBarWidget::HistoryTopBarWidget( QWidget *parent, not_null controller) : RpWidget(parent) @@ -51,11 +52,11 @@ TopBarWidget::TopBarWidget( , _forward(this, langFactory(lng_selected_forward), st::defaultActiveButton) , _delete(this, langFactory(lng_selected_delete), st::defaultActiveButton) , _info(this, nullptr, st::topBarInfoButton) -, _mediaType(this, langFactory(lng_media_type), st::topBarButton) , _call(this, st::topBarCall) , _search(this, st::topBarSearch) , _infoToggle(this, st::topBarInfo) -, _menuToggle(this, st::topBarMenuToggle) { +, _menuToggle(this, st::topBarMenuToggle) +, _onlineUpdater([this] { updateOnlineDisplay(); }) { subscribe(Lang::Current().updated(), [this] { refreshLang(); }); _forward->setClickedCallback([this] { onForwardSelection(); }); @@ -99,13 +100,21 @@ TopBarWidget::TopBarWidget( }); } subscribe(App::histories().sendActionAnimationUpdated(), [this](const Histories::SendActionAnimationUpdate &update) { - if (update.history->peer == _controller->historyPeer.current()) { + if (update.history->peer == _historyPeer) { rtlupdate(0, 0, width(), height()); } }); - subscribe(Notify::PeerUpdated(), Notify::PeerUpdatedHandler(Notify::PeerUpdate::Flag::UserHasCalls, [this](const Notify::PeerUpdate &update) { - if (update.peer->isUser()) { - updateControlsVisibility(); + using UpdateFlag = Notify::PeerUpdate::Flag; + auto flags = UpdateFlag::UserHasCalls + | UpdateFlag::UserOnlineChanged + | UpdateFlag::MembersChanged; + subscribe(Notify::PeerUpdated(), Notify::PeerUpdatedHandler(flags, [this](const Notify::PeerUpdate &update) { + if (update.flags & UpdateFlag::UserHasCalls) { + if (update.peer->isUser()) { + updateControlsVisibility(); + } + } else { + updateOnlineDisplay(); } })); subscribe(Global::RefPhoneCallsEnabledChanged(), [this] { @@ -122,107 +131,98 @@ TopBarWidget::TopBarWidget( updateControlsVisibility(); } -void TopBarWidget::refreshLang() { +void HistoryTopBarWidget::refreshLang() { InvokeQueued(this, [this] { updateControlsGeometry(); }); } -void TopBarWidget::onForwardSelection() { +void HistoryTopBarWidget::onForwardSelection() { if (App::main()) App::main()->forwardSelectedItems(); } -void TopBarWidget::onDeleteSelection() { +void HistoryTopBarWidget::onDeleteSelection() { if (App::main()) App::main()->confirmDeleteSelectedItems(); } -void TopBarWidget::onClearSelection() { +void HistoryTopBarWidget::onClearSelection() { if (App::main()) App::main()->clearSelectedItems(); } -void TopBarWidget::onInfoClicked() { - if (auto peer = _controller->historyPeer.current()) { - _controller->showPeerInfo(peer); +void HistoryTopBarWidget::onInfoClicked() { + if (_historyPeer) { + _controller->showPeerInfo(_historyPeer); } } -void TopBarWidget::onSearch() { - if (auto main = App::main()) { - if (auto peer = main->peer()) { - main->searchInPeer(peer); +void HistoryTopBarWidget::onSearch() { + if (_historyPeer) { + App::main()->searchInPeer(_historyPeer); + } +} + +void HistoryTopBarWidget::onCall() { + if (auto user = _historyPeer->asUser()) { + Calls::Current().startOutgoingCall(user); + } +} + +void HistoryTopBarWidget::showMenu() { + if (!_historyPeer || _menu) { + return; + } + _menu.create(parentWidget()); + _menu->setHiddenCallback([that = weak(this), menu = _menu.data()] { + menu->deleteLater(); + if (that && that->_menu == menu) { + that->_menu = nullptr; + that->_menuToggle->setForceRippled(false); } - } -} - -void TopBarWidget::onCall() { - if (auto main = App::main()) { - if (auto peer = main->peer()) { - if (auto user = peer->asUser()) { - Calls::Current().startOutgoingCall(user); - } + }); + _menu->setShowStartCallback(base::lambda_guarded(this, [this, menu = _menu.data()] { + if (_menu == menu) { + _menuToggle->setForceRippled(true); } - } -} - -void TopBarWidget::showMenu() { - if (auto main = App::main()) { - if (auto peer = main->peer()) { - if (!_menu) { - _menu.create(parentWidget()); - _menu->setHiddenCallback([that = weak(this), menu = _menu.data()] { - menu->deleteLater(); - if (that && that->_menu == menu) { - that->_menu = nullptr; - that->_menuToggle->setForceRippled(false); - } - }); - _menu->setShowStartCallback(base::lambda_guarded(this, [this, menu = _menu.data()] { - if (_menu == menu) { - _menuToggle->setForceRippled(true); - } - })); - _menu->setHideStartCallback(base::lambda_guarded(this, [this, menu = _menu.data()] { - if (_menu == menu) { - _menuToggle->setForceRippled(false); - } - })); - _menuToggle->installEventFilter(_menu); - Window::FillPeerMenu( - _controller, - peer, - [this](const QString &text, base::lambda callback) { - return _menu->addAction(text, std::move(callback)); - }, - Window::PeerMenuSource::History); - _menu->moveToRight((parentWidget()->width() - width()) + st::topBarMenuPosition.x(), st::topBarMenuPosition.y()); - _menu->showAnimated(Ui::PanelAnimation::Origin::TopRight); - } + })); + _menu->setHideStartCallback(base::lambda_guarded(this, [this, menu = _menu.data()] { + if (_menu == menu) { + _menuToggle->setForceRippled(false); } - } + })); + _menuToggle->installEventFilter(_menu); + Window::FillPeerMenu( + _controller, + _historyPeer, + [this](const QString &text, base::lambda callback) { + return _menu->addAction(text, std::move(callback)); + }, + Window::PeerMenuSource::History); + _menu->moveToRight((parentWidget()->width() - width()) + st::topBarMenuPosition.x(), st::topBarMenuPosition.y()); + _menu->showAnimated(Ui::PanelAnimation::Origin::TopRight); } -void TopBarWidget::toggleInfoSection() { +void HistoryTopBarWidget::toggleInfoSection() { if (Adaptive::ThreeColumn() && (Auth().data().thirdSectionInfoEnabled() || Auth().data().tabbedReplacedWithInfo())) { _controller->closeThirdSection(); - } else if (auto peer = _controller->historyPeer.current()) { + } else if (_historyPeer) { if (_controller->canShowThirdSection()) { Auth().data().setThirdSectionInfoEnabled(true); Auth().saveDataDelayed(); if (Adaptive::ThreeColumn()) { - _controller->showSection(Info::Memento(peer->id)); + _controller->showSection(Info::Memento(_historyPeer->id)); } else { _controller->resizeForThirdSection(); _controller->updateColumnLayout(); } } else { - _controller->showSection(Info::Memento(peer->id)); + _controller->showSection(Info::Memento(_historyPeer->id)); } } else { updateControlsVisibility(); } } -bool TopBarWidget::eventFilter(QObject *obj, QEvent *e) { +bool HistoryTopBarWidget::eventFilter(QObject *obj, QEvent *e) { if (obj == _membersShowArea) { switch (e->type()) { case QEvent::MouseButtonPress: @@ -230,18 +230,18 @@ bool TopBarWidget::eventFilter(QObject *obj, QEvent *e) { return true; case QEvent::Enter: - App::main()->setMembersShowAreaActive(true); + _membersShowAreaActive.fire(true); break; case QEvent::Leave: - App::main()->setMembersShowAreaActive(false); + _membersShowAreaActive.fire(false); break; } } return TWidget::eventFilter(obj, e); } -void TopBarWidget::paintEvent(QPaintEvent *e) { +void HistoryTopBarWidget::paintEvent(QPaintEvent *e) { Painter p(this); auto ms = getms(); @@ -271,16 +271,61 @@ void TopBarWidget::paintEvent(QPaintEvent *e) { if (!_call->isHidden()) { decreaseWidth += st::topBarCallSkip + _call->width(); } - auto paintCounter = App::main()->paintTopBar(p, decreaseWidth, ms); + paintTopBar(p, decreaseWidth, ms); p.restore(); - if (paintCounter) { - paintUnreadCounter(p, width()); - } + paintUnreadCounter(p, width(), _historyPeer); } } -void TopBarWidget::paintUnreadCounter(Painter &p, int outerWidth) { +void HistoryTopBarWidget::paintTopBar( + Painter &p, + int decreaseWidth, + TimeMs ms) { + auto history = App::historyLoaded(_historyPeer); + if (!history) return; + + auto increaseLeft = (Adaptive::OneColumn() || !App::main()->stackIsEmpty()) + ? (st::topBarArrowPadding.left() - st::topBarArrowPadding.right()) + : 0; + auto nameleft = st::topBarArrowPadding.right() + increaseLeft; + auto nametop = st::topBarArrowPadding.top(); + auto statustop = st::topBarHeight - st::topBarArrowPadding.bottom() - st::dialogsTextFont->height; + auto namewidth = width() - decreaseWidth - nameleft - st::topBarArrowPadding.right(); + p.setFont(st::dialogsTextFont); + if (!history->paintSendAction(p, nameleft, statustop, namewidth, width(), st::historyStatusFgTyping, ms)) { + p.setPen(_titlePeerTextOnline ? st::historyStatusFgActive : st::historyStatusFg); + p.drawText(nameleft, statustop + st::dialogsTextFont->ascent, _titlePeerText); + } + + p.setPen(st::dialogsNameFg); + _historyPeer->dialogName().drawElided(p, nameleft, nametop, namewidth); + + if (Adaptive::OneColumn() || !App::main()->stackIsEmpty()) { + st::topBarBackward.paint( + p, + (st::topBarArrowPadding.left() - st::topBarBackward.width()) / 2, + (st::topBarHeight - st::topBarBackward.height()) / 2, + width()); + } +} + +QRect HistoryTopBarWidget::getMembersShowAreaGeometry() const { + int increaseLeft = (Adaptive::OneColumn() || !App::main()->stackIsEmpty()) + ? (st::topBarArrowPadding.left() - st::topBarArrowPadding.right()) + : 0; + int membersTextLeft = st::topBarArrowPadding.right() + increaseLeft; + int membersTextTop = st::topBarHeight - st::topBarArrowPadding.bottom() - st::dialogsTextFont->height; + int membersTextWidth = _titlePeerTextWidth; + int membersTextHeight = st::topBarHeight - membersTextTop; + + return myrtlrect(membersTextLeft, membersTextTop, membersTextWidth, membersTextHeight); +} + +void HistoryTopBarWidget::paintUnreadCounter( + Painter &p, + int outerWidth, + PeerData *substractPeer) { if (!Adaptive::OneColumn()) { return; } @@ -288,7 +333,7 @@ void TopBarWidget::paintUnreadCounter(Painter &p, int outerWidth) { auto fullCounter = App::histories().unreadBadge() + (Global::IncludeMuted() ? 0 : mutedCount); // Do not include currently shown chat in the top bar unread counter. - if (auto historyShown = App::historyLoaded(App::wnd()->controller()->historyPeer.current())) { + if (auto historyShown = App::historyLoaded(substractPeer)) { auto shownUnreadCount = historyShown->unreadCount(); if (!historyShown->mute() || Global::IncludeMuted()) { fullCounter -= shownUnreadCount; @@ -309,21 +354,31 @@ void TopBarWidget::paintUnreadCounter(Painter &p, int outerWidth) { } } -void TopBarWidget::mousePressEvent(QMouseEvent *e) { - if (e->button() == Qt::LeftButton && e->pos().y() < st::topBarHeight && !_selectedCount) { - emit clicked(); +void HistoryTopBarWidget::mousePressEvent(QMouseEvent *e) { + if (e->button() == Qt::LeftButton + && e->pos().y() < st::topBarHeight + && !_selectedCount) { + clicked(); } } -void TopBarWidget::resizeEvent(QResizeEvent *e) { +void HistoryTopBarWidget::clicked() { + if (Adaptive::OneColumn() || !App::main()->stackIsEmpty()) { + _controller->showBackFromStack(); + } else if (_historyPeer) { + _controller->showPeerInfo(_historyPeer); + } +} + +void HistoryTopBarWidget::resizeEvent(QResizeEvent *e) { updateControlsGeometry(); } -int TopBarWidget::countSelectedButtonsTop(float64 selectedShown) { +int HistoryTopBarWidget::countSelectedButtonsTop(float64 selectedShown) { return (1. - selectedShown) * (-st::topBarHeight); } -void TopBarWidget::updateControlsGeometry() { +void HistoryTopBarWidget::updateControlsGeometry() { auto hasSelected = (_selectedCount > 0); auto selectedButtonsTop = countSelectedButtonsTop(_selectedShown.current(hasSelected ? 1. : 0.)); auto otherButtonsTop = selectedButtonsTop + st::topBarHeight; @@ -348,7 +403,6 @@ void TopBarWidget::updateControlsGeometry() { auto right = 0; _info->moveToRight(right, otherButtonsTop); _menuToggle->moveToRight(right, otherButtonsTop); - _mediaType->moveToRight(right, otherButtonsTop); if (_info->isHidden()) { right += _menuToggle->width() + st::topBarSkip; } else { @@ -363,20 +417,17 @@ void TopBarWidget::updateControlsGeometry() { _call->moveToRight(right, otherButtonsTop); } -void TopBarWidget::animationFinished() { +void HistoryTopBarWidget::animationFinished() { updateMembersShowArea(); updateControlsVisibility(); } -void TopBarWidget::updateControlsVisibility() { - auto overviewPeer = App::main() ? App::main()->overviewPeer() : nullptr; - +void HistoryTopBarWidget::updateControlsVisibility() { _clearSelection->show(); _delete->setVisible(_canDelete); _forward->setVisible(_canForward); - _mediaType->setVisible(App::main() ? App::main()->showMediaTypeSwitch() : false); - if (_historyPeer && !overviewPeer) { + if (_historyPeer) { if (Adaptive::OneColumn() || !App::main()->stackIsEmpty()) { _info->setPeer(_historyPeer); _info->show(); @@ -408,10 +459,10 @@ void TopBarWidget::updateControlsVisibility() { updateControlsGeometry(); } -void TopBarWidget::updateMembersShowArea() { +void HistoryTopBarWidget::updateMembersShowArea() { auto membersShowAreaNeeded = [this]() { auto peer = App::main()->peer(); - if ((_selectedCount > 0) || !peer || App::main()->overviewPeer()) { + if ((_selectedCount > 0) || !peer) { return false; } if (auto chat = peer->asChat()) { @@ -424,7 +475,7 @@ void TopBarWidget::updateMembersShowArea() { }; if (!membersShowAreaNeeded()) { if (_membersShowArea) { - App::main()->setMembersShowAreaActive(false); + _membersShowAreaActive.fire(false); _membersShowArea.destroy(); } return; @@ -433,10 +484,10 @@ void TopBarWidget::updateMembersShowArea() { _membersShowArea->show(); _membersShowArea->installEventFilter(this); } - _membersShowArea->setGeometry(App::main()->getMembersShowAreaGeometry()); + _membersShowArea->setGeometry(getMembersShowAreaGeometry()); } -void TopBarWidget::showSelected(SelectedState state) { +void HistoryTopBarWidget::showSelected(SelectedState state) { auto canDelete = (state.count > 0 && state.count == state.canDeleteCount); auto canForward = (state.count > 0 && state.count == state.canForwardCount); if (_selectedCount == state.count && _canDelete == canDelete && _canForward == canForward) { @@ -474,12 +525,12 @@ void TopBarWidget::showSelected(SelectedState state) { } } -void TopBarWidget::selectedShowCallback() { +void HistoryTopBarWidget::selectedShowCallback() { updateControlsGeometry(); update(); } -void TopBarWidget::updateAdaptiveLayout() { +void HistoryTopBarWidget::updateAdaptiveLayout() { updateMembersShowArea(); updateControlsVisibility(); if (!Adaptive::OneColumn()) { @@ -492,7 +543,7 @@ void TopBarWidget::updateAdaptiveLayout() { updateInfoToggleActive(); } -void TopBarWidget::updateInfoToggleActive() { +void HistoryTopBarWidget::updateInfoToggleActive() { auto infoThirdActive = Adaptive::ThreeColumn() && (Auth().data().thirdSectionInfoEnabled() || Auth().data().tabbedReplacedWithInfo()); @@ -506,8 +557,105 @@ void TopBarWidget::updateInfoToggleActive() { _infoToggle->setRippleColorOverride(rippleOverride); } -Ui::RoundButton *TopBarWidget::mediaTypeButton() { - return _mediaType; +void HistoryTopBarWidget::updateOnlineDisplay() { + if (!_historyPeer) return; + + QString text; + int32 t = unixtime(); + bool titlePeerTextOnline = false; + if (auto user = _historyPeer->asUser()) { + text = App::onlineText(user, t); + titlePeerTextOnline = App::onlineColorUse(user, t); + } else if (auto chat = _historyPeer->asChat()) { + if (!chat->amIn()) { + text = lang(lng_chat_status_unaccessible); + } else if (chat->participants.isEmpty()) { + if (!_titlePeerText.isEmpty()) { + text = _titlePeerText; + } else if (chat->count <= 0) { + text = lang(lng_group_status); + } else { + text = lng_chat_status_members(lt_count, chat->count); + } + } else { + auto online = 0; + auto onlyMe = true; + for (auto i = chat->participants.cbegin(), e = chat->participants.cend(); i != e; ++i) { + if (i.key()->onlineTill > t) { + ++online; + if (onlyMe && i.key() != App::self()) onlyMe = false; + } + } + if (online > 0 && !onlyMe) { + auto membersCount = lng_chat_status_members(lt_count, chat->participants.size()); + auto onlineCount = lng_chat_status_online(lt_count, online); + text = lng_chat_status_members_online(lt_members_count, membersCount, lt_online_count, onlineCount); + } else if (chat->participants.size() > 0) { + text = lng_chat_status_members(lt_count, chat->participants.size()); + } else { + text = lang(lng_group_status); + } + } + } else if (auto channel = _historyPeer->asChannel()) { + if (channel->isMegagroup() && channel->membersCount() > 0 && channel->membersCount() <= Global::ChatSizeMax()) { + if (channel->mgInfo->lastParticipants.size() < channel->membersCount() || channel->lastParticipantsCountOutdated()) { + Auth().api().requestLastParticipants(channel); + } + auto online = 0; + bool onlyMe = true; + for (auto &participant : std::as_const(channel->mgInfo->lastParticipants)) { + if (participant->onlineTill > t) { + ++online; + if (onlyMe && participant != App::self()) { + onlyMe = false; + } + } + } + if (online && !onlyMe) { + auto membersCount = lng_chat_status_members(lt_count, channel->membersCount()); + auto onlineCount = lng_chat_status_online(lt_count, online); + text = lng_chat_status_members_online(lt_members_count, membersCount, lt_online_count, onlineCount); + } else if (channel->membersCount() > 0) { + text = lng_chat_status_members(lt_count, channel->membersCount()); + } else { + text = lang(lng_group_status); + } + } else if (channel->membersCount() > 0) { + text = lng_chat_status_members(lt_count, channel->membersCount()); + } else { + text = lang(channel->isMegagroup() ? lng_group_status : lng_channel_status); + } + } + if (_titlePeerText != text) { + _titlePeerText = text; + _titlePeerTextOnline = titlePeerTextOnline; + _titlePeerTextWidth = st::dialogsTextFont->width(_titlePeerText); + updateMembersShowArea(); + update(); + } + updateOnlineDisplayTimer(); } -} // namespace Window +void HistoryTopBarWidget::updateOnlineDisplayTimer() { + if (!_historyPeer) return; + + int32 t = unixtime(), minIn = 86400; + if (auto user = _historyPeer->asUser()) { + minIn = App::onlineWillChangeIn(user, t); + } else if (auto chat = _historyPeer->asChat()) { + if (chat->participants.isEmpty()) return; + + for (auto i = chat->participants.cbegin(), e = chat->participants.cend(); i != e; ++i) { + int32 onlineWillChangeIn = App::onlineWillChangeIn(i.key(), t); + if (onlineWillChangeIn < minIn) { + minIn = onlineWillChangeIn; + } + } + } else if (_historyPeer->isChannel()) { + } + updateOnlineDisplayIn(minIn * 1000); +} + +void HistoryTopBarWidget::updateOnlineDisplayIn(TimeMs timeout) { + _onlineUpdater.callOnce(timeout); +} diff --git a/Telegram/SourceFiles/window/top_bar_widget.h b/Telegram/SourceFiles/history/history_top_bar_widget.h similarity index 77% rename from Telegram/SourceFiles/window/top_bar_widget.h rename to Telegram/SourceFiles/history/history_top_bar_widget.h index 0e1ba7c71..62490c7df 100644 --- a/Telegram/SourceFiles/window/top_bar_widget.h +++ b/Telegram/SourceFiles/history/history_top_bar_widget.h @@ -21,6 +21,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org #pragma once #include "ui/rp_widget.h" +#include "base/timer.h" namespace Ui { class PeerAvatarButton; @@ -30,14 +31,14 @@ class DropdownMenu; } // namespace Ui namespace Window { - class Controller; +} // namespace Window -class TopBarWidget : public Ui::RpWidget, private base::Subscriber { - Q_OBJECT - +class HistoryTopBarWidget : public Ui::RpWidget, private base::Subscriber { public: - TopBarWidget(QWidget *parent, not_null controller); + HistoryTopBarWidget( + QWidget *parent, + not_null controller); struct SelectedState { bool textSelected = false; @@ -49,14 +50,19 @@ public: void updateControlsVisibility(); void showSelected(SelectedState state); void animationFinished(); - void updateMembersShowArea(); + rpl::producer membersShowAreaActive() const { + return _membersShowAreaActive.events(); + } - Ui::RoundButton *mediaTypeButton(); void setHistoryPeer(not_null historyPeer) { _historyPeer = historyPeer; } - static void paintUnreadCounter(Painter &p, int outerWidth); + void clicked(); + static void paintUnreadCounter( + Painter &p, + int outerWidth, + PeerData *substractPeer = nullptr); protected: void paintEvent(QPaintEvent *e) override; @@ -64,9 +70,6 @@ protected: void resizeEvent(QResizeEvent *e) override; bool eventFilter(QObject *obj, QEvent *e) override; -signals: - void clicked(); - private: void refreshLang(); void updateControlsGeometry(); @@ -85,6 +88,13 @@ private: void updateAdaptiveLayout(); int countSelectedButtonsTop(float64 selectedShown); + void paintTopBar(Painter &p, int decreaseWidth, TimeMs ms); + QRect getMembersShowAreaGeometry() const; + void updateMembersShowArea(); + void updateOnlineDisplay(); + void updateOnlineDisplayTimer(); + void updateOnlineDisplayIn(TimeMs timeout); + not_null _controller; PeerData *_historyPeer = nullptr; @@ -98,7 +108,6 @@ private: object_ptr _forward, _delete; object_ptr _info; - object_ptr _mediaType; object_ptr _call; object_ptr _search; @@ -107,9 +116,13 @@ private: object_ptr _menu = { nullptr }; object_ptr _membersShowArea = { nullptr }; + rpl::event_stream _membersShowAreaActive; + + QString _titlePeerText; + bool _titlePeerTextOnline = false; + int _titlePeerTextWidth = 0; int _unreadCounterSubscription = 0; + base::Timer _onlineUpdater; }; - -} // namespace Window diff --git a/Telegram/SourceFiles/history/history_widget.cpp b/Telegram/SourceFiles/history/history_widget.cpp index e4964e268..ffda023fa 100644 --- a/Telegram/SourceFiles/history/history_widget.cpp +++ b/Telegram/SourceFiles/history/history_widget.cpp @@ -64,7 +64,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org #include "media/player/media_player_instance.h" #include "storage/localstorage.h" #include "apiwrap.h" -#include "window/top_bar_widget.h" +#include "history/history_top_bar_widget.h" #include "window/themes/window_theme.h" #include "observer_peer.h" #include "base/qthelp_regex.h" @@ -547,7 +547,6 @@ HistoryWidget::HistoryWidget(QWidget *parent, not_null cont setAcceptDrops(true); subscribe(Auth().downloaderTaskFinished(), [this] { update(); }); - connect(_topBar, &Window::TopBarWidget::clicked, this, [this] { topBarClick(); }); connect(_scroll, SIGNAL(scrolled()), this, SLOT(onScroll())); _historyDown->setClickedCallback([this] { historyDownClicked(); }); _unreadMentions->setClickedCallback([this] { showNextUnreadMention(); }); @@ -760,6 +759,10 @@ HistoryWidget::HistoryWidget(QWidget *parent, not_null cont } } }, lifetime()); + _topBar->membersShowAreaActive() + | rpl::start_with_next([this](bool active) { + setMembersShowAreaActive(active); + }, _topBar->lifetime()); orderWidgets(); } @@ -1753,12 +1756,9 @@ void HistoryWidget::showHistory(const PeerId &peerId, MsgId showAtMsgId, bool re } _reportSpamSettingRequestId = ReportSpamRequestNeeded; - _titlePeerText = QString(); - _titlePeerTextWidth = 0; - noSelectingScroll(); _nonEmptySelection = false; - _topBar->showSelected(Window::TopBarWidget::SelectedState {}); + _topBar->showSelected(HistoryTopBarWidget::SelectedState {}); App::hoveredItem(nullptr); App::pressedItem(nullptr); @@ -2975,7 +2975,9 @@ MsgId HistoryWidget::msgId() const { return _showAtMsgId; } -void HistoryWidget::showAnimated(Window::SlideDirection direction, const Window::SectionSlideParams ¶ms) { +void HistoryWidget::showAnimated( + Window::SlideDirection direction, + const Window::SectionSlideParams ¶ms) { _showDirection = direction; _a_show.finish(); @@ -2997,7 +2999,7 @@ void HistoryWidget::showAnimated(Window::SlideDirection direction, const Window: _a_show.start([this] { animationCallback(); }, 0., 1., st::slideDuration, Window::SlideAnimation::transition()); if (_history) { _backAnimationButton.create(this); - _backAnimationButton->setClickedCallback([this] { topBarClick(); }); + _backAnimationButton->setClickedCallback([this] { _topBar->clicked(); }); _backAnimationButton->setGeometry(_topBar->geometry()); _backAnimationButton->show(); } @@ -3716,39 +3718,6 @@ void HistoryWidget::selectMessage() { if (_list) _list->selectItem(item); } -bool HistoryWidget::paintTopBar(Painter &p, int decreaseWidth, TimeMs ms) { - if (!_history) return false; - - auto increaseLeft = (Adaptive::OneColumn() || !App::main()->stackIsEmpty()) ? (st::topBarArrowPadding.left() - st::topBarArrowPadding.right()) : 0; - auto nameleft = st::topBarArrowPadding.right() + increaseLeft; - auto nametop = st::topBarArrowPadding.top(); - auto statustop = st::topBarHeight - st::topBarArrowPadding.bottom() - st::dialogsTextFont->height; - auto namewidth = width() - decreaseWidth - nameleft - st::topBarArrowPadding.right(); - p.setFont(st::dialogsTextFont); - if (!_history->paintSendAction(p, nameleft, statustop, namewidth, width(), st::historyStatusFgTyping, ms)) { - p.setPen(_titlePeerTextOnline ? st::historyStatusFgActive : st::historyStatusFg); - p.drawText(nameleft, statustop + st::dialogsTextFont->ascent, _titlePeerText); - } - - p.setPen(st::dialogsNameFg); - _peer->dialogName().drawElided(p, nameleft, nametop, namewidth); - - if (Adaptive::OneColumn() || !App::main()->stackIsEmpty()) { - st::topBarBackward.paint(p, (st::topBarArrowPadding.left() - st::topBarBackward.width()) / 2, (st::topBarHeight - st::topBarBackward.height()) / 2, width()); - } - return true; -} - -QRect HistoryWidget::getMembersShowAreaGeometry() const { - int increaseLeft = (Adaptive::OneColumn() || !App::main()->stackIsEmpty()) ? (st::topBarArrowPadding.left() - st::topBarArrowPadding.right()) : 0; - int membersTextLeft = st::topBarArrowPadding.right() + increaseLeft; - int membersTextTop = st::topBarHeight - st::topBarArrowPadding.bottom() - st::dialogsTextFont->height; - int membersTextWidth = _titlePeerTextWidth; - int membersTextHeight = st::topBarHeight - membersTextTop; - - return myrtlrect(membersTextLeft, membersTextTop, membersTextWidth, membersTextHeight); -} - void HistoryWidget::setMembersShowAreaActive(bool active) { if (!active) { _membersDropdownShowTimer.stop(); @@ -3781,14 +3750,6 @@ void HistoryWidget::onModerateKeyActivate(int index, bool *outHandled) { *outHandled = _keyboard->isHidden() ? false : _keyboard->moderateKeyActivate(index); } -void HistoryWidget::topBarClick() { - if (Adaptive::OneColumn() || !App::main()->stackIsEmpty()) { - controller()->showBackFromStack(); - } else if (_peer) { - controller()->showPeerInfo(_peer); - } -} - void HistoryWidget::pushTabbedSelectorToThirdSection( const Window::SectionShow ¶ms) { if (!_history || !_tabbedPanel) { @@ -3862,107 +3823,6 @@ void HistoryWidget::recountChatWidth() { } } -void HistoryWidget::updateOnlineDisplay() { - if (!_history) return; - - QString text; - int32 t = unixtime(); - bool titlePeerTextOnline = false; - if (auto user = _peer->asUser()) { - text = App::onlineText(user, t); - titlePeerTextOnline = App::onlineColorUse(user, t); - } else if (_peer->isChat()) { - auto chat = _peer->asChat(); - if (!chat->amIn()) { - text = lang(lng_chat_status_unaccessible); - } else if (chat->participants.isEmpty()) { - if (!_titlePeerText.isEmpty()) { - text = _titlePeerText; - } else if (chat->count <= 0) { - text = lang(lng_group_status); - } else { - text = lng_chat_status_members(lt_count, chat->count); - } - } else { - auto online = 0; - auto onlyMe = true; - for (auto i = chat->participants.cbegin(), e = chat->participants.cend(); i != e; ++i) { - if (i.key()->onlineTill > t) { - ++online; - if (onlyMe && i.key() != App::self()) onlyMe = false; - } - } - if (online > 0 && !onlyMe) { - auto membersCount = lng_chat_status_members(lt_count, chat->participants.size()); - auto onlineCount = lng_chat_status_online(lt_count, online); - text = lng_chat_status_members_online(lt_members_count, membersCount, lt_online_count, onlineCount); - } else if (chat->participants.size() > 0) { - text = lng_chat_status_members(lt_count, chat->participants.size()); - } else { - text = lang(lng_group_status); - } - } - } else if (_peer->isChannel()) { - if (_peer->isMegagroup() && _peer->asChannel()->membersCount() > 0 && _peer->asChannel()->membersCount() <= Global::ChatSizeMax()) { - if (_peer->asChannel()->mgInfo->lastParticipants.size() < _peer->asChannel()->membersCount() || _peer->asChannel()->lastParticipantsCountOutdated()) { - Auth().api().requestLastParticipants(_peer->asChannel()); - } - auto online = 0; - bool onlyMe = true; - for (auto i = _peer->asChannel()->mgInfo->lastParticipants.cbegin(), e = _peer->asChannel()->mgInfo->lastParticipants.cend(); i != e; ++i) { - if ((*i)->onlineTill > t) { - ++online; - if (onlyMe && (*i) != App::self()) onlyMe = false; - } - } - if (online && !onlyMe) { - auto membersCount = lng_chat_status_members(lt_count, _peer->asChannel()->membersCount()); - auto onlineCount = lng_chat_status_online(lt_count, online); - text = lng_chat_status_members_online(lt_members_count, membersCount, lt_online_count, onlineCount); - } else if (_peer->asChannel()->membersCount() > 0) { - text = lng_chat_status_members(lt_count, _peer->asChannel()->membersCount()); - } else { - text = lang(lng_group_status); - } - } else if (_peer->asChannel()->membersCount() > 0) { - text = lng_chat_status_members(lt_count, _peer->asChannel()->membersCount()); - } else { - text = lang(_peer->isMegagroup() ? lng_group_status : lng_channel_status); - } - } - if (_titlePeerText != text) { - _titlePeerText = text; - _titlePeerTextOnline = titlePeerTextOnline; - _titlePeerTextWidth = st::dialogsTextFont->width(_titlePeerText); - if (App::main()) { - _topBar->updateMembersShowArea(); - _topBar->update(); - } - } - updateOnlineDisplayTimer(); -} - -void HistoryWidget::updateOnlineDisplayTimer() { - if (!_history) return; - - int32 t = unixtime(), minIn = 86400; - if (_peer->isUser()) { - minIn = App::onlineWillChangeIn(_peer->asUser(), t); - } else if (_peer->isChat()) { - ChatData *chat = _peer->asChat(); - if (chat->participants.isEmpty()) return; - - for (auto i = chat->participants.cbegin(), e = chat->participants.cend(); i != e; ++i) { - int32 onlineWillChangeIn = App::onlineWillChangeIn(i.key(), t); - if (onlineWillChangeIn < minIn) { - minIn = onlineWillChangeIn; - } - } - } else if (_peer->isChannel()) { - } - App::main()->updateOnlineDisplayIn(minIn * 1000); -} - void HistoryWidget::moveFieldControls() { auto keyboardHeight = 0; auto bottom = height(); @@ -6005,7 +5865,6 @@ void HistoryWidget::handlePeerUpdate() { updateControlsGeometry(); } } - App::main()->updateOnlineDisplay(); } void HistoryWidget::onForwardSelected() { @@ -6113,7 +5972,7 @@ SelectedItemSet HistoryWidget::getSelectedItems() const { void HistoryWidget::updateTopBarSelection() { if (!_list) { - _topBar->showSelected(Window::TopBarWidget::SelectedState {}); + _topBar->showSelected(HistoryTopBarWidget::SelectedState {}); return; } diff --git a/Telegram/SourceFiles/history/history_widget.h b/Telegram/SourceFiles/history/history_widget.h index a140b622a..4d597e081 100644 --- a/Telegram/SourceFiles/history/history_widget.h +++ b/Telegram/SourceFiles/history/history_widget.h @@ -58,7 +58,6 @@ class SuggestionsController; namespace Window { class Controller; -class TopBarWidget; } // namespace Window namespace ChatHelpers { @@ -73,6 +72,7 @@ class SendFilesBox; class BotKeyboard; class MessageField; class HistoryInner; +class HistoryTopBarWidget; class ReportSpamPanel : public TWidget { Q_OBJECT @@ -191,10 +191,6 @@ public: bool isItemCompletelyHidden(HistoryItem *item) const; void updateTopBarSelection(); - bool paintTopBar(Painter &p, int decreaseWidth, TimeMs ms); - QRect getMembersShowAreaGeometry() const; - void setMembersShowAreaActive(bool active); - void loadMessages(); void loadMessagesDown(); void firstLoadMessages(); @@ -237,8 +233,6 @@ public: void updateControlsVisibility(); void updateControlsGeometry(); - void updateOnlineDisplay(); - void updateOnlineDisplayTimer(); void onShareContact(const PeerId &peer, UserData *contact); @@ -483,7 +477,6 @@ private: void repaintHistoryItem(not_null item); void handlePendingHistoryUpdate(); void fullPeerUpdated(PeerData *peer); - void topBarClick(); void toggleTabbedSelectorMode(); void returnTabbedSelector(object_ptr selector); void recountChatWidth(); @@ -491,6 +484,7 @@ private: void historyDownClicked(); void showNextUnreadMention(); void handlePeerUpdate(); + void setMembersShowAreaActive(bool active); void highlightMessage(MsgId universalMessageId); void adjustHighlightedMessageToMigrated(); @@ -735,7 +729,7 @@ private: mtpRequestId _delayedShowAtRequest = 0; object_ptr _backAnimationButton = { nullptr }; - object_ptr _topBar; + object_ptr _topBar; object_ptr _scroll; QPointer _list; History *_migrated = nullptr; @@ -836,10 +830,6 @@ private: int64 _serviceImageCacheSize = 0; QString _confirmSource; - QString _titlePeerText; - bool _titlePeerTextOnline = false; - int _titlePeerTextWidth = 0; - Animation _a_show; Window::SlideDirection _showDirection; QPixmap _cacheUnder, _cacheOver; diff --git a/Telegram/SourceFiles/mainwidget.cpp b/Telegram/SourceFiles/mainwidget.cpp index cdf19e3a1..ac9f8f842 100644 --- a/Telegram/SourceFiles/mainwidget.cpp +++ b/Telegram/SourceFiles/mainwidget.cpp @@ -35,7 +35,6 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org #include "ui/widgets/shadow.h" #include "window/section_memento.h" #include "window/section_widget.h" -#include "window/top_bar_widget.h" #include "data/data_drafts.h" #include "ui/widgets/dropdown_menu.h" #include "ui/focus_persister.h" @@ -49,7 +48,6 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org #include "history/history_message.h" #include "history/history_media.h" #include "history/history_service_layout.h" -#include "overviewwidget.h" #include "lang/lang_keys.h" #include "lang/lang_cloud_manager.h" #include "boxes/add_contact_box.h" @@ -112,7 +110,6 @@ MTPMessagesFilter TypeToMediaFilter(MediaOverviewType &type) { enum StackItemType { HistoryStackItem, SectionStackItem, - OverviewStackItem, }; class StackItem { @@ -176,28 +173,6 @@ private: }; -class StackItemOverview : public StackItem { -public: - StackItemOverview( - PeerData *peer, - MediaOverviewType mediaType, - int32 lastWidth, - int32 lastScrollTop) - : StackItem(peer) - , mediaType(mediaType) - , lastWidth(lastWidth) - , lastScrollTop(lastScrollTop) { - } - - StackItemType type() const { - return OverviewStackItem; - } - - MediaOverviewType mediaType; - int32 lastWidth, lastScrollTop; - -}; - void StackItem::setThirdSectionMemento( std::unique_ptr &&memento) { _thirdSectionMemento = std::move(memento); @@ -244,7 +219,6 @@ MainWidget::MainWidget( connect(_history, SIGNAL(cancelled()), _dialogs, SLOT(activate())); connect(&noUpdatesTimer, SIGNAL(timeout()), this, SLOT(mtpPing())); connect(&_onlineTimer, SIGNAL(timeout()), this, SLOT(updateOnline())); - connect(&_onlineUpdater, SIGNAL(timeout()), this, SLOT(updateOnlineDisplay())); connect(&_idleFinishTimer, SIGNAL(timeout()), this, SLOT(checkIdleFinish())); connect(&_bySeqTimer, SIGNAL(timeout()), this, SLOT(getDifference())); connect(&_byPtsTimer, SIGNAL(timeout()), this, SLOT(onGetDifferenceTimeByPts())); @@ -344,11 +318,6 @@ MainWidget::MainWidget( subscribe(Adaptive::Changed(), [this]() { handleAdaptiveLayoutUpdate(); }); - auto observeEvents = Notify::PeerUpdate::Flag::SharedMediaChanged; - subscribe(Notify::PeerUpdated(), Notify::PeerUpdatedHandler(observeEvents, [this](const Notify::PeerUpdate &update) { - mediaOverviewUpdated(update); - })); - _dialogs->show(); if (Adaptive::OneColumn()) { _history->hide(); @@ -529,9 +498,6 @@ Window::AbstractSectionWidget *MainWidget::getFloatPlayerSection(Window::Column } else if (Adaptive::Normal()) { if (column == Window::Column::First) { return _dialogs; - } - if (_overview) { - return _overview; } else if (_mainSection) { return _mainSection; } @@ -539,8 +505,6 @@ Window::AbstractSectionWidget *MainWidget::getFloatPlayerSection(Window::Column } if (Adaptive::OneColumn() && selectingPeer()) { return _dialogs; - } else if (_overview) { - return _overview; } else if (_mainSection) { return _mainSection; } else if (!Adaptive::OneColumn() || _history->peer()) { @@ -589,9 +553,7 @@ void MainWidget::updateFloatPlayerColumnCorner(QPoint center) { } } else if (Adaptive::Normal()) { checkSection(_dialogs, Window::Column::First); - if (_overview) { - checkSection(_overview, Window::Column::Second); - } else if (_mainSection) { + if (_mainSection) { checkSection(_mainSection, Window::Column::Second); } else { checkSection(_history, Window::Column::Second); @@ -599,8 +561,6 @@ void MainWidget::updateFloatPlayerColumnCorner(QPoint center) { } else { if (Adaptive::OneColumn() && selectingPeer()) { checkSection(_dialogs, Window::Column::First); - } else if (_overview) { - checkSection(_overview, Window::Column::Second); } else if (_mainSection) { checkSection(_mainSection, Window::Column::Second); } else if (!Adaptive::OneColumn() || _history->peer()) { @@ -648,7 +608,7 @@ void MainWidget::finishFloatPlayerDrag(not_null instance, bool closed) { bool MainWidget::setForwardDraft(PeerId peerId, ForwardWhatMessages what) { auto toForward = SelectedItemSet(); if (what == ForwardSelectedMessages) { - toForward = _overview ? _overview->getSelectedItems() : _history->getSelectedItems(); + toForward = _history->getSelectedItems(); } else { auto item = (HistoryItem*)nullptr; if (what == ForwardContextMessage) { @@ -969,19 +929,15 @@ void MainWidget::noHider(HistoryHider *destroyed) { _forwardConfirm = nullptr; } onHistoryShown(_history->history(), _history->msgId()); - if (_mainSection || _overview || (_history->peer() && _history->peer()->id)) { + if (_mainSection || (_history->peer() && _history->peer()->id)) { auto animationParams = ([this] { - if (_overview) { - return prepareOverviewAnimation(); - } else if (_mainSection) { + if (_mainSection) { return prepareMainSectionAnimation(_mainSection); } return prepareHistoryAnimation(_history->peer() ? _history->peer()->id : 0); })(); _dialogs->hide(); - if (_overview) { - _overview->showAnimated(Window::SlideDirection::FromRight, animationParams); - } else if (_mainSection) { + if (_mainSection) { _mainSection->showAnimated(Window::SlideDirection::FromRight, animationParams); } else { _history->showAnimated(Window::SlideDirection::FromRight, animationParams); @@ -1011,9 +967,7 @@ void MainWidget::hiderLayer(object_ptr h) { auto animationParams = prepareDialogsAnimation(); onHistoryShown(0, 0); - if (_overview) { - _overview->hide(); - } else if (_mainSection) { + if (_mainSection) { _mainSection->hide(); } else { _history->hide(); @@ -1042,12 +996,12 @@ void MainWidget::showSendPathsLayer() { void MainWidget::deleteLayer(int selectedCount) { if (selectedCount) { auto forDelete = true; - auto selected = _overview ? _overview->getSelectedItems() : _history->getSelectedItems(); + auto selected = _history->getSelectedItems(); if (!selected.isEmpty()) { Ui::show(Box(selected)); } } else if (auto item = App::contextItem()) { - auto suggestModerateActions = !_overview; + auto suggestModerateActions = true; Ui::show(Box(item, suggestModerateActions)); } } @@ -1060,11 +1014,7 @@ void MainWidget::cancelUploadLayer() { Auth().uploader().pause(item->fullId()); Ui::show(Box(lang(lng_selected_cancel_sure_this), lang(lng_selected_upload_stop), lang(lng_continue), base::lambda_guarded(this, [this] { - if (_overview) { - _overview->deleteContextItem(false); - } else { - _history->deleteContextItem(false); - } + _history->deleteContextItem(false); Auth().uploader().unpause(); }), base::lambda_guarded(this, [] { Auth().uploader().unpause(); @@ -1473,27 +1423,15 @@ void MainWidget::onCacheBackground() { } void MainWidget::forwardSelectedItems() { - if (_overview) { - _overview->onForwardSelected(); - } else { - _history->onForwardSelected(); - } + _history->onForwardSelected(); } void MainWidget::confirmDeleteSelectedItems() { - if (_overview) { - _overview->confirmDeleteSelectedItems(); - } else { - _history->confirmDeleteSelectedItems(); - } + _history->confirmDeleteSelectedItems(); } void MainWidget::clearSelectedItems() { - if (_overview) { - _overview->onClearSelected(); - } else { - _history->onClearSelected(); - } + _history->onClearSelected(); } Dialogs::IndexedList *MainWidget::contactsList() { @@ -1682,11 +1620,32 @@ bool MainWidget::preloadOverview(PeerData *peer, MediaOverviewType type) { return false; } - _overviewPreload[type].insert(peer, MTP::send(MTPmessages_Search(MTP_flags(0), peer->input, MTP_string(""), MTP_inputUserEmpty(), filter, MTP_int(0), MTP_int(0), MTP_int(0), MTP_int(0), MTP_int(0), MTP_int(0), MTP_int(0)), rpcDone(&MainWidget::overviewPreloaded, peer), rpcFail(&MainWidget::overviewFailed, peer), 0, 10)); + auto request = MTPmessages_Search( + MTP_flags(0), + peer->input, + MTP_string(""), + MTP_inputUserEmpty(), + filter, + MTP_int(0), + MTP_int(0), + MTP_int(0), + MTP_int(0), + MTP_int(0), + MTP_int(0), + MTP_int(0)); + _overviewPreload[type].insert(peer, MTP::send( + request, + rpcDone(&MainWidget::overviewPreloaded, peer), + rpcFail(&MainWidget::overviewFailed, peer), + 0, + 10)); return true; } -void MainWidget::overviewPreloaded(PeerData *peer, const MTPmessages_Messages &result, mtpRequestId req) { +void MainWidget::overviewPreloaded( + PeerData *peer, + const MTPmessages_Messages &result, + mtpRequestId req) { MediaOverviewType type = OverviewCount; for (int32 i = 0; i < OverviewCount; ++i) { OverviewsPreload::iterator j = _overviewPreload[i].find(peer); @@ -1705,13 +1664,6 @@ void MainWidget::overviewPreloaded(PeerData *peer, const MTPmessages_Messages &r Notify::mediaOverviewUpdated(peer, type); } -void MainWidget::mediaOverviewUpdated(const Notify::PeerUpdate &update) { - auto peer = update.peer; - if (_overview && (_overview->peer() == peer || _overview->peer()->migrateFrom() == peer)) { - _overview->mediaOverviewUpdated(update); - } -} - void MainWidget::itemEdited(HistoryItem *item) { if (_history->peer() == item->history()->peer || (_history->peer() && _history->peer() == item->history()->peer->migrateTo())) { _history->itemEdited(item); @@ -2123,11 +2075,6 @@ void MainWidget::mediaMarkRead(not_null item) { } } -void MainWidget::updateOnlineDisplay() { - if (this != App::main()) return; - _history->updateOnlineDisplay(); -} - void MainWidget::onSendFileConfirm(const FileLoadResultPtr &file) { _history->sendFileConfirmed(file); } @@ -2274,8 +2221,6 @@ void MainWidget::setInnerFocus() { if (_hider || !_history->peer()) { if (_hider && _hider->wasOffered()) { _hider->setFocus(); - } else if (!_hider && _overview) { - _overview->activate(); } else if (!_hider && _mainSection) { _mainSection->setInnerFocus(); } else if (!_hider && _thirdSection) { @@ -2283,8 +2228,6 @@ void MainWidget::setInnerFocus() { } else { dialogsActivate(); } - } else if (_overview) { - _overview->activate(); } else if (_mainSection) { _mainSection->setInnerFocus(); } else if (_history->peer() || !_thirdSection) { @@ -2465,7 +2408,6 @@ void MainWidget::ui_showPeerHistory( } if (_history->isHidden()) { return (_mainSection != nullptr) - || (_overview != nullptr) || (Adaptive::OneColumn() && !_dialogs->isHidden()); } if (back || way == Way::Forward) { @@ -2493,19 +2435,10 @@ void MainWidget::ui_showPeerHistory( auto noPeer = !_history->peer(); auto onlyDialogs = noPeer && Adaptive::OneColumn(); - if (_mainSection || _overview) { - if (_mainSection) { - _mainSection->hide(); - _mainSection->deleteLater(); - _mainSection = nullptr; - } - if (_overview) { - _overview->hide(); - _overview->clear(); - _overview->deleteLater(); - _overview->rpcClear(); - _overview = nullptr; - } + if (_mainSection) { + _mainSection->hide(); + _mainSection->deleteLater(); + _mainSection = nullptr; } updateControlsGeometry(); @@ -2584,7 +2517,7 @@ void MainWidget::peerAfter(const PeerData *inPeer, MsgId inMsg, PeerData *&outPe } PeerData *MainWidget::peer() { - return _overview ? _overview->peer() : _history->peer(); + return _history->peer(); } PeerData *MainWidget::activePeer() { @@ -2595,22 +2528,8 @@ MsgId MainWidget::activeMsgId() { return _history->peer() ? _history->msgId() : _msgIdInStack; } -PeerData *MainWidget::overviewPeer() { - return _overview ? _overview->peer() : 0; -} - -bool MainWidget::showMediaTypeSwitch() const { - return _overview ? _overview->showMediaTypeSwitch() : false; -} - void MainWidget::saveSectionInStack() { - if (_overview) { - _stack.push_back(std::make_unique( - _overview->peer(), - _overview->type(), - _overview->lastWidth(), - _overview->lastScrollTop())); - } else if (_mainSection) { + if (_mainSection) { if (auto memento = _mainSection->createMemento()) { _stack.push_back(std::make_unique( std::move(memento))); @@ -2625,77 +2544,6 @@ void MainWidget::saveSectionInStack() { } } -void MainWidget::showMediaOverview(PeerData *peer, MediaOverviewType type, bool back, int32 lastScrollTop) { - if (peer->migrateTo()) { - peer = peer->migrateTo(); - } - - Ui::hideSettingsAndLayer(); - if (_overview && _overview->peer() == peer) { - if (_overview->type() != type) { - _overview->switchType(type); - } else if (type == OverviewMusicFiles) { // hack for player - _controller->showBackFromStack(); - } - return; - } - - _controller->dialogsListFocused().set(false, true); - _a_dialogsWidth.finish(); - - auto animatedShow = [this] { - if (_a_show.animating() || App::passcoded()) { - return false; - } - if (Adaptive::OneColumn() || isMainSectionShown()) { - return true; - } - return false; - }; - auto animationParams = animatedShow() ? prepareOverviewAnimation() : Window::SectionSlideParams(); - setFocus(); // otherwise dialogs widget could be focused. - - if (!back) { - saveSectionInStack(); - } - if (_overview) { - _overview->hide(); - _overview->clear(); - _overview->deleteLater(); - _overview->rpcClear(); - } - if (_mainSection) { - _mainSection->hide(); - _mainSection->deleteLater(); - _mainSection = nullptr; - } - _overview.create(this, _controller, peer, type); - updateControlsGeometry(); - - // Send a fake update. - Notify::PeerUpdate update(peer); - update.flags |= Notify::PeerUpdate::Flag::SharedMediaChanged; - update.mediaTypesMask |= (1 << type); - mediaOverviewUpdated(update); - - _overview->setLastScrollTop(lastScrollTop); - if (!animationParams.oldContentCache.isNull()) { - _overview->showAnimated(back ? Window::SlideDirection::FromLeft : Window::SlideDirection::FromRight, animationParams); - } else { - _overview->fastShow(); - } - _history->finishAnimating(); - if (back) { - clearBotStartToken(_history->peer()); - } - _history->showHistory(0, 0); - _history->hide(); - if (Adaptive::OneColumn()) _dialogs->hide(); - - checkFloatPlayerVisibility(); - orderWidgets(); -} - void MainWidget::showSection( Window::SectionMemento &&memento, const SectionShow ¶ms) { @@ -2754,7 +2602,7 @@ Window::SectionSlideParams MainWidget::prepareShowAnimation( if (!_mainSection->hasTopBarShadow()) { result.withTopBarShadow = false; } - } else if (!_overview && !_history->peer()) { + } else if (!_history->peer()) { result.withTopBarShadow = false; } @@ -2788,10 +2636,8 @@ Window::SectionSlideParams MainWidget::prepareShowAnimation( result.oldContentCache = _mainSection->grabForShowAnimation(result); } else { if (result.withTopBarShadow) { - if (_overview) _overview->grapWithoutTopBarShadow(); _history->grapWithoutTopBarShadow(); } else { - if (_overview) _overview->grabStart(); _history->grabStart(); } if (Adaptive::OneColumn()) { @@ -2815,7 +2661,6 @@ Window::SectionSlideParams MainWidget::prepareShowAnimation( _thirdShadow->show(); } } - if (_overview) _overview->grabFinish(); _history->grabFinish(); } @@ -2848,10 +2693,6 @@ Window::SectionSlideParams MainWidget::prepareHistoryAnimation(PeerId historyPee return prepareShowAnimation(historyPeerId != 0); } -Window::SectionSlideParams MainWidget::prepareOverviewAnimation() { - return prepareShowAnimation(true); -} - Window::SectionSlideParams MainWidget::prepareDialogsAnimation() { return prepareShowAnimation(false); } @@ -2934,13 +2775,6 @@ void MainWidget::showNewSection( // This may modify the current section, for example remove its contents. saveSectionInStack(); } - if (_overview) { - _overview->hide(); - _overview->clear(); - _overview->deleteLater(); - _overview->rpcClear(); - _overview = nullptr; - } auto &settingSection = newThirdSection ? _thirdSection : _mainSection; @@ -3009,7 +2843,7 @@ void MainWidget::dropMainSection(Window::SectionWidget *widget) { } bool MainWidget::isMainSectionShown() const { - return _mainSection || _overview || _history->peer(); + return _mainSection || _history->peer(); } bool MainWidget::isThirdSectionShown() const { @@ -3057,13 +2891,6 @@ void MainWidget::showBackFromStack( showNewSection( std::move(*sectionItem->memento()), params.withWay(SectionShow::Way::Backward)); - } else if (item->type() == OverviewStackItem) { - auto overviewItem = static_cast(item.get()); - showMediaOverview( - overviewItem->peer(), - overviewItem->mediaType, - true, - overviewItem->lastScrollTop); } if (auto memento = item->thirdSectionMemento()) { if (_thirdSection) { @@ -3299,9 +3126,6 @@ void MainWidget::hideAll() { if (_thirdSection) { _thirdSection->hide(); } - if (_overview) { - _overview->hide(); - } _sideShadow->hide(); if (_thirdShadow) { _thirdShadow->hide(); @@ -3337,10 +3161,7 @@ void MainWidget::showAll() { if (selectingPeer()) { _dialogs->showFast(); _history->hide(); - if (_overview) _overview->hide(); if (_mainSection) _mainSection->hide(); - } else if (_overview) { - _overview->show(); } else if (_mainSection) { _mainSection->show(); } else if (_history->peer()) { @@ -3370,9 +3191,7 @@ void MainWidget::showAll() { } } _dialogs->showFast(); - if (_overview) { - _overview->show(); - } else if (_mainSection) { + if (_mainSection) { _mainSection->show(); } else { _history->show(); @@ -3487,7 +3306,6 @@ void MainWidget::updateControlsGeometry() { auto mainSectionGeometry = QRect(_history->x(), mainSectionTop, _history->width(), height() - mainSectionTop); _mainSection->setGeometryWithTopMoved(mainSectionGeometry, _contentScrollAddToY); } - if (_overview) _overview->setGeometry(_history->geometry()); refreshResizeAreas(); updateMediaPlayerPosition(); updateMediaPlaylistPosition(_playerPlaylist->x()); @@ -3689,7 +3507,6 @@ bool MainWidget::eventFilter(QObject *o, QEvent *e) { if (e->type() == QEvent::FocusIn) { if (auto widget = qobject_cast(o)) { if (_history == widget || _history->isAncestorOf(widget) - || (_overview && (_overview == widget || _overview->isAncestorOf(widget))) || (_mainSection && (_mainSection == widget || _mainSection->isAncestorOf(widget))) || (_thirdSection && (_thirdSection == widget || _thirdSection->isAncestorOf(widget)))) { _controller->dialogsListFocused().set(false); @@ -3782,28 +3599,6 @@ void MainWidget::updateWindowAdaptiveLayout() { } } -bool MainWidget::paintTopBar(Painter &p, int decreaseWidth, TimeMs ms) { - if (_overview) { - return _overview->paintTopBar(p, decreaseWidth); - } else if (!_mainSection) { - return _history->paintTopBar(p, decreaseWidth, ms); - } - return false; -} - -QRect MainWidget::getMembersShowAreaGeometry() const { - if (!_overview && !_mainSection) { - return _history->getMembersShowAreaGeometry(); - } - return QRect(); -} - -void MainWidget::setMembersShowAreaActive(bool active) { - if (!active || (!_overview && !_mainSection)) { - _history->setMembersShowAreaActive(active); - } -} - int MainWidget::backgroundFromY() const { return -getMainSectionTop(); } @@ -4406,7 +4201,6 @@ void MainWidget::onSelfParticipantUpdated(ChannelData *channel) { bool MainWidget::contentOverlapped(const QRect &globalRect) { return (_history->contentOverlapped(globalRect) - || (_overview && _overview->contentOverlapped(globalRect)) || _playerPanel->overlaps(globalRect) || _playerPlaylist->overlaps(globalRect) || (_playerVolume && _playerVolume->overlaps(globalRect))); @@ -4745,7 +4539,7 @@ void MainWidget::incrementSticker(DocumentData *sticker) { void MainWidget::activate() { if (_a_show.animating()) return; - if (!_mainSection && !_overview) { + if (!_mainSection) { if (_hider) { if (_hider->wasOffered()) { _hider->setFocus(); @@ -4770,20 +4564,16 @@ void MainWidget::destroyData() { _dialogs->destroyData(); } -void MainWidget::updateOnlineDisplayIn(int32 msecs) { - _onlineUpdater.start(msecs); -} - bool MainWidget::isActive() const { return !_isIdle && isVisible() && !_a_show.animating(); } bool MainWidget::doWeReadServerHistory() const { - return isActive() && !_mainSection && !_overview && _history->doWeReadServerHistory(); + return isActive() && !_mainSection && _history->doWeReadServerHistory(); } bool MainWidget::doWeReadMentions() const { - return isActive() && !_mainSection && !_overview && _history->doWeReadMentions(); + return isActive() && !_mainSection && _history->doWeReadMentions(); } bool MainWidget::lastWasOnline() const { @@ -4827,7 +4617,9 @@ void MainWidget::updateOnline(bool gotOtherOffline) { } } auto ms = getms(true); - if (isOnline != _lastWasOnline || (isOnline && _lastSetOnline + Global::OnlineUpdatePeriod() <= ms) || (isOnline && gotOtherOffline)) { + if (isOnline != _lastWasOnline + || (isOnline && _lastSetOnline + Global::OnlineUpdatePeriod() <= ms) + || (isOnline && gotOtherOffline)) { if (_onlineRequest) { MTP::cancel(_onlineRequest); _onlineRequest = 0; @@ -4846,8 +4638,6 @@ void MainWidget::updateOnline(bool gotOtherOffline) { } _lastSetOnline = ms; - - updateOnlineDisplay(); } else if (isOnline) { updateIn = qMin(updateIn, int(_lastSetOnline + Global::OnlineUpdatePeriod() - ms)); } diff --git a/Telegram/SourceFiles/mainwidget.h b/Telegram/SourceFiles/mainwidget.h index 036622f3f..5b1f1fedf 100644 --- a/Telegram/SourceFiles/mainwidget.h +++ b/Telegram/SourceFiles/mainwidget.h @@ -54,7 +54,6 @@ class SlideWrap; namespace Window { class Controller; class PlayerWrapWidget; -class TopBarWidget; class SectionMemento; class SectionWidget; class AbstractSectionWidget; @@ -72,7 +71,6 @@ class MainWindow; class ConfirmBox; class DialogsWidget; class HistoryWidget; -class OverviewWidget; class HistoryHider; class StackItem; @@ -105,11 +103,6 @@ public: bool isMainSectionShown() const; bool isThirdSectionShown() const; - // Temporary methods, while top bar was not done inside HistoryWidget / OverviewWidget. - bool paintTopBar(Painter &, int decreaseWidth, TimeMs ms); - QRect getMembersShowAreaGeometry() const; - void setMembersShowAreaActive(bool active); - int contentScrollAddToY() const; void showAnimated(const QPixmap &bgAnimCache, bool back = false); @@ -156,17 +149,10 @@ public: MsgId activeMsgId(); int backgroundFromY() const; - PeerData *overviewPeer(); - bool showMediaTypeSwitch() const; void showSection( Window::SectionMemento &&memento, const SectionShow ¶ms); void updateColumnLayout(); - void showMediaOverview( - PeerData *peer, - MediaOverviewType type, - bool back = false, - int32 lastScrollTop = -1); bool stackIsEmpty() const; void showBackFromStack( const SectionShow ¶ms); @@ -381,7 +367,6 @@ public slots: void updateOnline(bool gotOtherOffline = false); void checkIdleFinish(); - void updateOnlineDisplay(); void onHistoryShown(History *history, MsgId atMsgId); @@ -472,7 +457,6 @@ private: std::pair, MsgId> historyAndStartMsgId, const MTPmessages_Messages &result, mtpRequestId req); - void mediaOverviewUpdated(const Notify::PeerUpdate &update); Window::SectionSlideParams prepareShowAnimation( bool willHaveTopBarShadow); @@ -486,7 +470,6 @@ private: // All this methods use the prepareShowAnimation(). Window::SectionSlideParams prepareMainSectionAnimation(Window::SectionWidget *section); Window::SectionSlideParams prepareHistoryAnimation(PeerId historyPeerId); - Window::SectionSlideParams prepareOverviewAnimation(); Window::SectionSlideParams prepareDialogsAnimation(); void startWithSelf(const MTPUserFull &user); @@ -595,7 +578,6 @@ private: object_ptr _history; object_ptr _mainSection = { nullptr }; object_ptr _thirdSection = { nullptr }; - object_ptr _overview = { nullptr }; base::weak_unique_ptr _currentCall; object_ptr> _callTopBar = { nullptr }; @@ -636,7 +618,7 @@ private: SingleTimer _byMinChannelTimer; mtpRequestId _onlineRequest = 0; - SingleTimer _onlineTimer, _onlineUpdater, _idleFinishTimer; + SingleTimer _onlineTimer, _idleFinishTimer; bool _lastWasOnline = false; TimeMs _lastSetOnline = 0; bool _isIdle = false; diff --git a/Telegram/SourceFiles/mediaview.h b/Telegram/SourceFiles/mediaview.h index 78901d619..be131318c 100644 --- a/Telegram/SourceFiles/mediaview.h +++ b/Telegram/SourceFiles/mediaview.h @@ -69,8 +69,6 @@ public: updateOver(mapFromGlobal(QCursor::pos())); } - void mediaOverviewUpdated(const Notify::PeerUpdate &update); - void close(); void activateControls(); diff --git a/Telegram/SourceFiles/overviewwidget.cpp b/Telegram/SourceFiles/overviewwidget.cpp deleted file mode 100644 index 8dc56437c..000000000 --- a/Telegram/SourceFiles/overviewwidget.cpp +++ /dev/null @@ -1,2432 +0,0 @@ -/* -This file is part of Telegram Desktop, -the official desktop version of Telegram messaging app, see https://telegram.org - -Telegram Desktop is free software: you can redistribute it and/or modify -it under the terms of the GNU General Public License as published by -the Free Software Foundation, either version 3 of the License, or -(at your option) any later version. - -It is distributed in the hope that it will be useful, -but WITHOUT ANY WARRANTY; without even the implied warranty of -MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -GNU General Public License for more details. - -In addition, as a special exception, the copyright holders give permission -to link the code of portions of this program with the OpenSSL library. - -Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE -Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org -*/ -#include "styles/style_overview.h" -#include "styles/style_dialogs.h" -#include "styles/style_window.h" -#include "styles/style_settings.h" -#include "boxes/add_contact_box.h" -#include "boxes/confirm_box.h" -#include "boxes/photo_crop_box.h" -#include "core/file_utilities.h" -#include "ui/widgets/popup_menu.h" -#include "ui/widgets/tooltip.h" -#include "ui/widgets/buttons.h" -#include "ui/widgets/input_fields.h" -#include "ui/widgets/shadow.h" -#include "window/top_bar_widget.h" -#include "window/window_slide_animation.h" -#include "window/themes/window_theme.h" -#include "window/window_controller.h" -#include "lang/lang_keys.h" -#include "mainwindow.h" -#include "mainwidget.h" -#include "overviewwidget.h" -#include "application.h" -#include "overview/overview_layout.h" -#include "history/history_message.h" -#include "history/history_media_types.h" -#include "history/history_service_layout.h" -#include "media/media_audio.h" -#include "observer_peer.h" -#include "auth_session.h" -#include "storage/file_download.h" -#include "ui/widgets/dropdown_menu.h" - -// flick scroll taken from http://qt-project.org/doc/qt-4.8/demos-embedded-anomaly-src-flickcharm-cpp.html - -OverviewInner::OverviewInner(OverviewWidget *overview, Ui::ScrollArea *scroll, PeerData *peer, MediaOverviewType type) : TWidget(nullptr) -, _overview(overview) -, _scroll(scroll) -, _peer(peer->migrateTo() ? peer->migrateTo() : peer) -, _type(type) -, _reversed(_type != OverviewFiles && _type != OverviewLinks) -, _history(App::history(_peer)) -, _migrated(_history->migrateFrom()) -, _channel(peerToChannel(_peer->id)) -, _rowWidth(st::msgMinWidth) -, _search(this, st::overviewFilter, langFactory(lng_dlg_filter)) -, _cancelSearch(this, st::dialogsCancelSearch) -, _itemsToBeLoaded(LinksOverviewPerPage * 2) -, _width(st::columnMinimalWidthMain) { - subscribe(Auth().downloader().taskFinished(), [this] { update(); }); - - resize(_width, st::windowMinHeight); - - App::contextItem(0); - - _touchSelectTimer.setSingleShot(true); - connect(&_touchSelectTimer, SIGNAL(timeout()), this, SLOT(onTouchSelect())); - - setAttribute(Qt::WA_AcceptTouchEvents); - connect(&_touchScrollTimer, SIGNAL(timeout()), this, SLOT(onTouchScrollTimer())); - - mediaOverviewUpdated(); - setMouseTracking(true); - - connect(_cancelSearch, SIGNAL(clicked()), this, SLOT(onCancelSearch())); - connect(_search, SIGNAL(cancelled()), this, SLOT(onCancel())); - connect(_search, SIGNAL(changed()), this, SLOT(onSearchUpdate())); - - _searchTimer.setSingleShot(true); - connect(&_searchTimer, SIGNAL(timeout()), this, SLOT(onSearchMessages())); - - subscribe(Window::Theme::Background(), [this](const Window::Theme::BackgroundUpdate &update) { - if (update.paletteChanged()) { - invalidateCache(); - } - }); - subscribe(App::wnd()->dragFinished(), [this] { - dragActionUpdate(QCursor::pos()); - }); - subscribe(Auth().messageIdChanging, [this](std::pair, MsgId> update) { - changingMsgId(update.first, update.second); - }); - - if (_type == OverviewLinks || _type == OverviewFiles) { - _search->show(); - } else { - _search->hide(); - } -} - -void OverviewInner::invalidateCache() { - for_const (auto item, _layoutItems) { - item->invalidateCache(); - } - for_const (auto item, _layoutDates) { - item->invalidateCache(); - } -} - -bool OverviewInner::event(QEvent *e) { - if (e->type() == QEvent::TouchBegin || e->type() == QEvent::TouchUpdate || e->type() == QEvent::TouchEnd || e->type() == QEvent::TouchCancel) { - QTouchEvent *ev = static_cast(e); - if (ev->device()->type() == QTouchDevice::TouchScreen) { - touchEvent(ev); - return true; - } - } - - return QWidget::event(e); -} - -void OverviewInner::touchUpdateSpeed() { - const auto nowTime = getms(); - if (_touchPrevPosValid) { - const int elapsed = nowTime - _touchSpeedTime; - if (elapsed) { - const QPoint newPixelDiff = (_touchPos - _touchPrevPos); - const QPoint pixelsPerSecond = newPixelDiff * (1000 / elapsed); - - // fingers are inacurates, we ignore small changes to avoid stopping the autoscroll because - // of a small horizontal offset when scrolling vertically - const int newSpeedY = (qAbs(pixelsPerSecond.y()) > FingerAccuracyThreshold) ? pixelsPerSecond.y() : 0; - const int newSpeedX = (qAbs(pixelsPerSecond.x()) > FingerAccuracyThreshold) ? pixelsPerSecond.x() : 0; - if (_touchScrollState == Ui::TouchScrollState::Auto) { - const int oldSpeedY = _touchSpeed.y(); - const int oldSpeedX = _touchSpeed.x(); - if ((oldSpeedY <= 0 && newSpeedY <= 0) || ((oldSpeedY >= 0 && newSpeedY >= 0) - && (oldSpeedX <= 0 && newSpeedX <= 0)) || (oldSpeedX >= 0 && newSpeedX >= 0)) { - _touchSpeed.setY(snap((oldSpeedY + (newSpeedY / 4)), -MaxScrollAccelerated, +MaxScrollAccelerated)); - _touchSpeed.setX(snap((oldSpeedX + (newSpeedX / 4)), -MaxScrollAccelerated, +MaxScrollAccelerated)); - } else { - _touchSpeed = QPoint(); - } - } else { - // we average the speed to avoid strange effects with the last delta - if (!_touchSpeed.isNull()) { - _touchSpeed.setX(snap((_touchSpeed.x() / 4) + (newSpeedX * 3 / 4), -MaxScrollFlick, +MaxScrollFlick)); - _touchSpeed.setY(snap((_touchSpeed.y() / 4) + (newSpeedY * 3 / 4), -MaxScrollFlick, +MaxScrollFlick)); - } else { - _touchSpeed = QPoint(newSpeedX, newSpeedY); - } - } - } - } else { - _touchPrevPosValid = true; - } - _touchSpeedTime = nowTime; - _touchPrevPos = _touchPos; -} - -bool OverviewInner::itemMigrated(MsgId msgId) const { - return _migrated && (msgId < 0) && (-msgId < ServerMaxMsgId); -} - -ChannelId OverviewInner::itemChannel(MsgId msgId) const { - return itemMigrated(msgId) ? _migrated->channelId() : _channel; -} - -MsgId OverviewInner::itemMsgId(MsgId msgId) const { - return itemMigrated(msgId) ? -msgId : msgId; -} - -int32 OverviewInner::migratedIndexSkip() const { - return (_migrated && _history->overviewLoaded(_type)) ? _migrated->overview(_type).size() : 0; -} - -void OverviewInner::fixItemIndex(int32 ¤t, MsgId msgId) const { - if (!msgId) { - current = -1; - } else { - int32 l = _items.size(); - if (current < 0 || current >= l || complexMsgId(_items.at(current)->getItem()) != msgId) { - current = -1; - for (int32 i = 0; i < l; ++i) { - if (complexMsgId(_items.at(i)->getItem()) == msgId) { - current = i; - break; - } - } - } - } -} - -void OverviewInner::searchReceived(SearchRequestType type, const MTPmessages_Messages &result, mtpRequestId req) { - if (!_search->text().isEmpty()) { - if (type == SearchFromStart) { - SearchQueries::iterator i = _searchQueries.find(req); - if (i != _searchQueries.cend()) { - _searchCache[i.value()] = result; - _searchQueries.erase(i); - } - } - } - - if (_searchRequest == req) { - const QVector *messages = nullptr; - switch (result.type()) { - case mtpc_messages_messages: { - auto &d = result.c_messages_messages(); - App::feedUsers(d.vusers); - App::feedChats(d.vchats); - messages = &d.vmessages.v; - _searchedCount = messages->size(); - } break; - - case mtpc_messages_messagesSlice: { - auto &d = result.c_messages_messagesSlice(); - App::feedUsers(d.vusers); - App::feedChats(d.vchats); - messages = &d.vmessages.v; - _searchedCount = d.vcount.v; - } break; - - case mtpc_messages_channelMessages: { - auto &d = result.c_messages_channelMessages(); - if (_peer && _peer->isChannel()) { - _peer->asChannel()->ptsReceived(d.vpts.v); - } else { - LOG(("API Error: received messages.channelMessages when no channel was passed! (OverviewInner::searchReceived)")); - } - App::feedUsers(d.vusers); - App::feedChats(d.vchats); - messages = &d.vmessages.v; - _searchedCount = d.vcount.v; - } break; - } - if (messages) { - bool migratedSearch = (type == SearchMigratedFromStart || type == SearchMigratedFromOffset); - if (messages->isEmpty()) { - if (migratedSearch) { - _searchFullMigrated = true; - } else { - _searchFull = true; - } - } - if (type == SearchFromStart) { - _searchResults.clear(); - _lastSearchId = _lastSearchMigratedId = 0; - _itemsToBeLoaded = LinksOverviewPerPage * 2; - } - if (type == SearchMigratedFromStart) { - _lastSearchMigratedId = 0; - } - for (auto i = messages->cbegin(), e = messages->cend(); i != e; ++i) { - auto item = App::histories().addNewMessage(*i, NewMessageExisting); - auto msgId = item ? item->id : idFromMessage(*i); - if (migratedSearch) { - if (item) _searchResults.push_front(-item->id); - _lastSearchMigratedId = msgId; - } else { - if (item) _searchResults.push_front(item->id); - _lastSearchId = msgId; - } - } - mediaOverviewUpdated(); - if (messages->isEmpty()) { - update(); - } - } - - _searchRequest = 0; - _overview->onScroll(); - } -} - -bool OverviewInner::searchFailed(SearchRequestType type, const RPCError &error, mtpRequestId req) { - if (MTP::isDefaultHandledError(error)) return false; - - if (_searchRequest == req) { - _searchRequest = 0; - if (type == SearchFromStart || type == SearchFromOffset) { - _searchFull = true; - } else if (type == SearchMigratedFromStart || type == SearchMigratedFromOffset) { - _searchFullMigrated = true; - } - } - return true; -} - -bool OverviewInner::itemHasPoint(MsgId msgId, int32 index, int32 x, int32 y) const { - fixItemIndex(index, msgId); - if (index < 0) return false; - - if (_type == OverviewPhotos || _type == OverviewVideos) { - if (x >= 0 && x < _rowWidth && y >= 0 && y < _rowWidth) { - return true; - } - } else { - if (x >= _rowsLeft && x < _rowsLeft + _rowWidth && y >= 0 && y < itemHeight(msgId, index)) { - return true; - } - } - return false; -} - -int32 OverviewInner::itemHeight(MsgId msgId, int32 index) const { - if (_type == OverviewPhotos || _type == OverviewVideos) { - return _rowWidth; - } - - fixItemIndex(index, msgId); - return (index < 0) ? 0 : _items.at(index)->height(); -} - -void OverviewInner::moveToNextItem(MsgId &msgId, int32 &index, MsgId upTo, int32 delta) const { - fixItemIndex(index, msgId); - if (msgId == upTo || index < 0) { - msgId = 0; - index = -1; - return; - } - - index += delta; - while (index >= 0 && index < _items.size() && !_items.at(index)->toMediaItem()) { - index += (delta > 0) ? 1 : -1; - } - if (index < 0 || index >= _items.size()) { - msgId = 0; - index = -1; - } else { - msgId = complexMsgId(_items.at(index)->getItem()); - } -} - -void OverviewInner::repaintItem(MsgId itemId, int32 itemIndex) { - fixItemIndex(itemIndex, itemId); - if (itemIndex >= 0) { - if (_type == OverviewPhotos || _type == OverviewVideos) { - float64 w = (float64(_width - st::overviewPhotoSkip) / _photosInRow); - int32 vsize = (_rowWidth + st::overviewPhotoSkip); - int32 row = itemIndex / _photosInRow, col = itemIndex % _photosInRow; - update(int32(col * w), _marginTop + int32(row * vsize), qCeil(w), vsize); - } else { - int32 top = _items.at(itemIndex)->Get()->top; - if (_reversed) top = _height - top; - update(_rowsLeft, _marginTop + top, _rowWidth, _items.at(itemIndex)->height()); - } - } -} - -void OverviewInner::touchResetSpeed() { - _touchSpeed = QPoint(); - _touchPrevPosValid = false; -} - -void OverviewInner::touchDeaccelerate(int32 elapsed) { - int32 x = _touchSpeed.x(); - int32 y = _touchSpeed.y(); - _touchSpeed.setX((x == 0) ? x : (x > 0) ? qMax(0, x - elapsed) : qMin(0, x + elapsed)); - _touchSpeed.setY((y == 0) ? y : (y > 0) ? qMax(0, y - elapsed) : qMin(0, y + elapsed)); -} - -void OverviewInner::touchEvent(QTouchEvent *e) { - const Qt::TouchPointStates &states(e->touchPointStates()); - if (e->type() == QEvent::TouchCancel) { // cancel - if (!_touchInProgress) return; - _touchInProgress = false; - _touchSelectTimer.stop(); - _touchScroll = _touchSelect = false; - _touchScrollState = Ui::TouchScrollState::Manual; - dragActionCancel(); - return; - } - - if (!e->touchPoints().isEmpty()) { - _touchPrevPos = _touchPos; - _touchPos = e->touchPoints().cbegin()->screenPos().toPoint(); - } - - switch (e->type()) { - case QEvent::TouchBegin: - if (_touchInProgress) return; - if (e->touchPoints().isEmpty()) return; - - _touchInProgress = true; - if (_touchScrollState == Ui::TouchScrollState::Auto) { - _touchScrollState = Ui::TouchScrollState::Acceleration; - _touchWaitingAcceleration = true; - _touchAccelerationTime = getms(); - touchUpdateSpeed(); - _touchStart = _touchPos; - } else { - _touchScroll = false; - _touchSelectTimer.start(QApplication::startDragTime()); - } - _touchSelect = false; - _touchStart = _touchPrevPos = _touchPos; - break; - - case QEvent::TouchUpdate: - if (!_touchInProgress) return; - if (_touchSelect) { - dragActionUpdate(_touchPos); - } else if (!_touchScroll && (_touchPos - _touchStart).manhattanLength() >= QApplication::startDragDistance()) { - _touchSelectTimer.stop(); - _touchScroll = true; - touchUpdateSpeed(); - } - if (_touchScroll) { - if (_touchScrollState == Ui::TouchScrollState::Manual) { - touchScrollUpdated(_touchPos); - } else if (_touchScrollState == Ui::TouchScrollState::Acceleration) { - touchUpdateSpeed(); - _touchAccelerationTime = getms(); - if (_touchSpeed.isNull()) { - _touchScrollState = Ui::TouchScrollState::Manual; - } - } - } - break; - - case QEvent::TouchEnd: - if (!_touchInProgress) return; - _touchInProgress = false; - if (_touchSelect) { - dragActionFinish(_touchPos, Qt::RightButton); - QContextMenuEvent contextMenu(QContextMenuEvent::Mouse, mapFromGlobal(_touchPos), _touchPos); - showContextMenu(&contextMenu, true); - _touchScroll = false; - } else if (_touchScroll) { - if (_touchScrollState == Ui::TouchScrollState::Manual) { - _touchScrollState = Ui::TouchScrollState::Auto; - _touchPrevPosValid = false; - _touchScrollTimer.start(15); - _touchTime = getms(); - } else if (_touchScrollState == Ui::TouchScrollState::Auto) { - _touchScrollState = Ui::TouchScrollState::Manual; - _touchScroll = false; - touchResetSpeed(); - } else if (_touchScrollState == Ui::TouchScrollState::Acceleration) { - _touchScrollState = Ui::TouchScrollState::Auto; - _touchWaitingAcceleration = false; - _touchPrevPosValid = false; - } - } else { // one short tap -- like mouse click - dragActionStart(_touchPos); - dragActionFinish(_touchPos); - } - _touchSelectTimer.stop(); - _touchSelect = false; - break; - } -} - -void OverviewInner::dragActionUpdate(const QPoint &screenPos) { - _dragPos = screenPos; - onUpdateSelected(); -} - -void OverviewInner::dragActionStart(const QPoint &screenPos, Qt::MouseButton button) { - dragActionUpdate(screenPos); - if (button != Qt::LeftButton) return; - - ClickHandler::pressed(); - - _dragAction = NoDrag; - _dragItem = _mousedItem; - _dragItemIndex = _mousedItemIndex; - _dragStartPos = mapMouseToItem(mapFromGlobal(screenPos), _dragItem, _dragItemIndex); - _dragWasInactive = App::wnd()->wasInactivePress(); - if (_dragWasInactive) App::wnd()->setInactivePress(false); - if (ClickHandler::getPressed() && _selected.isEmpty()) { - _dragAction = PrepareDrag; - } else if (!_selected.isEmpty()) { - if (_selected.cbegin().value() == FullSelection) { - if (_selected.constFind(_dragItem) != _selected.cend() && ClickHandler::getPressed()) { - _dragAction = PrepareDrag; // start items drag - } else { - _dragAction = PrepareSelect; // start items select - } - } - } - if (_dragAction == NoDrag && _dragItem) { - if (!_dragWasInactive) { - _dragAction = PrepareSelect; - } - } - - if (!_dragItem) { - _dragAction = NoDrag; - } else if (_dragAction == NoDrag) { - _dragItem = 0; - } -} - -void OverviewInner::dragActionCancel() { - _dragItem = 0; - _dragItemIndex = -1; - _dragAction = NoDrag; - _dragSelFrom = _dragSelTo = 0; - _dragSelFromIndex = _dragSelToIndex = -1; - _dragStartPos = QPoint(0, 0); - _overview->noSelectingScroll(); -} - -void OverviewInner::dragActionFinish(const QPoint &screenPos, Qt::MouseButton button) { - dragActionUpdate(screenPos); - - ClickHandlerPtr activated = ClickHandler::unpressed(); - if (_dragAction == Dragging || _selMode) { - activated.clear(); - } - if (!ClickHandler::getActive() && _cursor != style::cur_default) { - _cursor = style::cur_default; - setCursor(_cursor); - } - if (activated) { - dragActionCancel(); - App::activateClickHandler(activated, button); - return; - } - if (_dragAction == PrepareSelect && !_dragWasInactive && !_selected.isEmpty() && _selected.cbegin().value() == FullSelection) { - SelectedItems::iterator i = _selected.find(_dragItem); - if (i == _selected.cend() && itemMsgId(_dragItem) > 0) { - if (_selected.size() < MaxSelectedItems) { - if (!_selected.isEmpty() && _selected.cbegin().value() != FullSelection) { - _selected.clear(); - } - _selected.insert(_dragItem, FullSelection); - } - } else { - _selected.erase(i); - } - repaintItem(_dragItem, _dragItemIndex); - } else if (_dragAction == PrepareDrag && !_dragWasInactive && button != Qt::RightButton) { - SelectedItems::iterator i = _selected.find(_dragItem); - if (i != _selected.cend() && i.value() == FullSelection) { - _selected.erase(i); - repaintItem(_dragItem, _dragItemIndex); - } else if (i == _selected.cend() && itemMsgId(_dragItem) > 0 && !_selected.isEmpty() && _selected.cbegin().value() == FullSelection) { - if (_selected.size() < MaxSelectedItems) { - _selected.insert(_dragItem, FullSelection); - repaintItem(_dragItem, _dragItemIndex); - } - } else { - _selected.clear(); - update(); - } - } else if (_dragAction == Selecting) { - if (_dragSelFrom && _dragSelTo) { - applyDragSelection(); - } else if (!_selected.isEmpty() && !_dragWasInactive) { - auto sel = _selected.cbegin().value(); - if (sel != FullSelection && sel.from == sel.to) { - _selected.clear(); - App::wnd()->setInnerFocus(); - } - } - } - _dragAction = NoDrag; - _overview->noSelectingScroll(); - _overview->updateTopBarSelection(); -} - -void OverviewInner::performDrag() { - if (_dragAction != Dragging) return; - - bool uponSelected = false; - if (_dragItem) { - if (!_selected.isEmpty() && _selected.cbegin().value() == FullSelection) { - uponSelected = _selected.contains(_dragItem); - } else { - uponSelected = false; - } - } - ClickHandlerPtr pressedHandler = ClickHandler::getPressed(); - QString sel; - QList urls; - bool forwardSelected = false; - if (uponSelected) { - if (!Adaptive::OneColumn()) { - auto selectedState = getSelectionState(); - if (selectedState.count > 0 && selectedState.count == selectedState.canForwardCount) { - forwardSelected = true; - } - } - } else if (pressedHandler) { - sel = pressedHandler->dragText(); - //if (!sel.isEmpty() && sel.at(0) != '/' && sel.at(0) != '@' && sel.at(0) != '#') { - // urls.push_back(QUrl::fromEncoded(sel.toUtf8())); // Google Chrome crashes in Mac OS X O_o - //} - } - if (!sel.isEmpty() || forwardSelected) { - updateDragSelection(0, -1, 0, -1, false); - _overview->noSelectingScroll(); - - auto mimeData = std::make_unique(); - if (!sel.isEmpty()) mimeData->setText(sel); - if (!urls.isEmpty()) mimeData->setUrls(urls); - if (forwardSelected) { - mimeData->setData(qsl("application/x-td-forward-selected"), "1"); - } - - // This call enters event loop and can destroy any QObject. - App::wnd()->launchDrag(std::move(mimeData)); - return; - } else { - QString forwardMimeType; - HistoryMedia *pressedMedia = nullptr; - if (auto pressedLnkItem = App::pressedLinkItem()) { - if ((pressedMedia = pressedLnkItem->getMedia())) { - if (forwardMimeType.isEmpty() && pressedMedia->dragItemByHandler(pressedHandler)) { - forwardMimeType = qsl("application/x-td-forward-pressed-link"); - } - } - } - if (!forwardMimeType.isEmpty()) { - auto mimeData = std::make_unique(); - mimeData->setData(qsl("application/x-td-forward-pressed-link"), "1"); - if (auto document = (pressedMedia ? pressedMedia->getDocument() : nullptr)) { - auto filepath = document->filepath(DocumentData::FilePathResolveChecked); - if (!filepath.isEmpty()) { - QList urls; - urls.push_back(QUrl::fromLocalFile(filepath)); - mimeData->setUrls(urls); - } - } - - // This call enters event loop and can destroy any QObject. - App::wnd()->launchDrag(std::move(mimeData)); - return; - } - } -} - -void OverviewInner::touchScrollUpdated(const QPoint &screenPos) { - _touchPos = screenPos; - _overview->touchScroll(_touchPos - _touchPrevPos); - touchUpdateSpeed(); -} - -void OverviewInner::addSelectionRange(int32 selFrom, int32 selTo, History *history) { - if (selFrom < 0 || selTo < 0) return; - for (int32 i = selFrom; i <= selTo; ++i) { - MsgId msgid = complexMsgId(_items.at(i)->getItem()); - if (!msgid) continue; - - SelectedItems::iterator j = _selected.find(msgid); - if (_dragSelecting && itemMsgId(msgid) > 0) { - if (j == _selected.cend()) { - if (_selected.size() >= MaxSelectedItems) break; - _selected.insert(msgid, FullSelection); - } else if (j.value() != FullSelection) { - *j = FullSelection; - } - } else { - if (j != _selected.cend()) { - _selected.erase(j); - } - } - } -} - -void OverviewInner::applyDragSelection() { - if (_dragSelFromIndex < 0 || _dragSelToIndex < 0) return; - - if (!_selected.isEmpty() && _selected.cbegin().value() != FullSelection) { - _selected.clear(); - } - addSelectionRange(_dragSelToIndex, _dragSelFromIndex, _history); - - _dragSelFrom = _dragSelTo = 0; - _dragSelFromIndex = _dragSelToIndex = -1; - _overview->updateTopBarSelection(); -} - -QPoint OverviewInner::mapMouseToItem(QPoint p, MsgId itemId, int32 itemIndex) { - fixItemIndex(itemIndex, itemId); - if (itemIndex < 0) return QPoint(0, 0); - - if (_type == OverviewPhotos || _type == OverviewVideos) { - int32 row = itemIndex / _photosInRow, col = itemIndex % _photosInRow; - float64 w = (_width - st::overviewPhotoSkip) / float64(_photosInRow); - p.setX(p.x() - int32(col * w) - st::overviewPhotoSkip); - p.setY(p.y() - _marginTop - row * (_rowWidth + st::overviewPhotoSkip) - st::overviewPhotoSkip); - } else { - int32 top = _items.at(itemIndex)->Get()->top; - if (_reversed) top = _height - top; - p.setY(p.y() - _marginTop - top); - } - return p; -} - -void OverviewInner::activate() { - if (_type == OverviewLinks || _type == OverviewFiles) { - _search->setFocus(); - } else { - setFocus(); - } -} - -void OverviewInner::clear() { - _selected.clear(); - _dragItemIndex = _mousedItemIndex = _dragSelFromIndex = _dragSelToIndex = -1; - _dragItem = _mousedItem = _dragSelFrom = _dragSelTo = 0; - _dragAction = NoDrag; - for (LayoutItems::const_iterator i = _layoutItems.cbegin(), e = _layoutItems.cend(); i != e; ++i) { - delete i.value(); - } - _layoutItems.clear(); - for (LayoutDates::const_iterator i = _layoutDates.cbegin(), e = _layoutDates.cend(); i != e; ++i) { - delete i.value(); - } - _layoutDates.clear(); - _items.clear(); - - App::clearMousedItems(); -} - -int32 OverviewInner::itemTop(const FullMsgId &msgId) const { - if (_type == OverviewMusicFiles) { - int32 itemIndex = -1; - fixItemIndex(itemIndex, (msgId.channel == _channel) ? msgId.msg : ((_migrated && msgId.channel == _migrated->channelId()) ? -msgId.msg : 0)); - if (itemIndex >= 0) { - int32 top = _items.at(itemIndex)->Get()->top; - if (_reversed) top = _height - top; - return _marginTop + top; - } - } - return -1; -} - -void OverviewInner::preloadMore() { - if (_inSearch) { - if (!_searchRequest) { - auto filter = (_type == OverviewLinks) ? MTP_inputMessagesFilterUrl() : MTP_inputMessagesFilterDocument(); - if (!_searchFull) { - _searchRequest = MTP::send(MTPmessages_Search(MTP_flags(0), _history->peer->input, MTP_string(_searchQuery), MTP_inputUserEmpty(), filter, MTP_int(0), MTP_int(0), MTP_int(_lastSearchId), MTP_int(0), MTP_int(SearchPerPage), MTP_int(0), MTP_int(0)), rpcDone(&OverviewInner::searchReceived, _lastSearchId ? SearchFromOffset : SearchFromStart), rpcFail(&OverviewInner::searchFailed, _lastSearchId ? SearchFromOffset : SearchFromStart)); - if (!_lastSearchId) { - _searchQueries.insert(_searchRequest, _searchQuery); - } - } else if (_migrated && !_searchFullMigrated) { - _searchRequest = MTP::send(MTPmessages_Search(MTP_flags(0), _migrated->peer->input, MTP_string(_searchQuery), MTP_inputUserEmpty(), filter, MTP_int(0), MTP_int(0), MTP_int(_lastSearchMigratedId), MTP_int(0), MTP_int(SearchPerPage), MTP_int(0), MTP_int(0)), rpcDone(&OverviewInner::searchReceived, _lastSearchMigratedId ? SearchMigratedFromOffset : SearchMigratedFromStart), rpcFail(&OverviewInner::searchFailed, _lastSearchMigratedId ? SearchMigratedFromOffset : SearchMigratedFromStart)); - } - } - } else if (App::main()) { - if (_migrated && _history->overviewLoaded(_type)) { - App::main()->loadMediaBack(_migrated->peer, _type, true); - } else { - App::main()->loadMediaBack(_history->peer, _type, true); - } - } -} - -bool OverviewInner::preloadLocal() { - if (_itemsToBeLoaded >= migratedIndexSkip() + _history->overview(_type).size()) return false; - _itemsToBeLoaded += LinksOverviewPerPage; - mediaOverviewUpdated(); - return true; -} - -TextSelection OverviewInner::itemSelectedValue(int32 index) const { - int32 selfrom = -1, selto = -1; - if (_dragSelFromIndex >= 0 && _dragSelToIndex >= 0) { - selfrom = _dragSelToIndex; - selto = _dragSelFromIndex; - } - if (_items.at(index)->toMediaItem()) { // draw item - if (index >= _dragSelToIndex && index <= _dragSelFromIndex && _dragSelToIndex >= 0) { - return (_dragSelecting && _items.at(index)->msgId() > 0) ? FullSelection : TextSelection{ 0, 0 }; - } else if (!_selected.isEmpty()) { - SelectedItems::const_iterator j = _selected.constFind(complexMsgId(_items.at(index)->getItem())); - if (j != _selected.cend()) { - return j.value(); - } - } - } - return { 0, 0 }; -} - -void OverviewInner::paintEvent(QPaintEvent *e) { - if (App::wnd() && App::wnd()->contentOverlapped(this, e)) return; - - Painter p(this); - - QRect r(e->rect()); - bool trivial = (r == rect()); - if (!trivial) { - p.setClipRect(r); - } - auto ms = getms(); - Overview::Layout::PaintContext context(ms, _selMode); - - if (_history->overview(_type).isEmpty() && (!_migrated || !_history->overviewLoaded(_type) || _migrated->overview(_type).isEmpty())) { - HistoryLayout::paintEmpty(p, _width, height()); - return; - } else if (_inSearch && _searchResults.isEmpty() && _searchFull && (!_migrated || _searchFullMigrated) && !_searchTimer.isActive()) { - p.setFont(st::noContactsFont->f); - p.setPen(st::noContactsColor->p); - p.drawText(QRect(_rowsLeft, _marginTop, _rowWidth, _marginTop), lang(lng_search_no_results), style::al_center); - return; - } - - int32 selfrom = -1, selto = -1; - if (_dragSelFromIndex >= 0 && _dragSelToIndex >= 0) { - selfrom = _dragSelToIndex; - selto = _dragSelFromIndex; - } - - SelectedItems::const_iterator selEnd = _selected.cend(); - bool hasSel = !_selected.isEmpty(); - - if (_type == OverviewPhotos || _type == OverviewVideos) { - int32 count = _items.size(), rowsCount = count / _photosInRow + ((count % _photosInRow) ? 1 : 0); - int32 rowFrom = floorclamp(r.y() - _marginTop - st::overviewPhotoSkip, _rowWidth + st::overviewPhotoSkip, 0, rowsCount); - int32 rowTo = ceilclamp(r.y() + r.height() - _marginTop - st::overviewPhotoSkip, _rowWidth + st::overviewPhotoSkip, 0, rowsCount); - float64 w = float64(_width - st::overviewPhotoSkip) / _photosInRow; - for (int32 row = rowFrom; row < rowTo; ++row) { - if (row * _photosInRow >= count) break; - for (int32 col = 0; col < _photosInRow; ++col) { - int32 i = row * _photosInRow + col; - if (i < 0) continue; - if (i >= count) break; - - QPoint pos(int32(col * w + st::overviewPhotoSkip), _marginTop + row * (_rowWidth + st::overviewPhotoSkip) + st::overviewPhotoSkip); - p.translate(pos.x(), pos.y()); - _items.at(i)->paint(p, r.translated(-pos.x(), -pos.y()), itemSelectedValue(i), &context); - p.translate(-pos.x(), -pos.y()); - } - } - } else { - p.translate(_rowsLeft, _marginTop); - int32 y = 0, w = _rowWidth; - for (int32 j = 0, l = _items.size(); j < l; ++j) { - int32 i = _reversed ? (l - j - 1) : j, nexti = _reversed ? (i - 1) : (i + 1); - int32 nextItemTop = (j + 1 == l) ? (_reversed ? 0 : _height) : _items.at(nexti)->Get()->top; - if (_reversed) nextItemTop = _height - nextItemTop; - if (_marginTop + nextItemTop > r.top()) { - auto info = _items.at(i)->Get(); - int32 curY = info->top; - if (_reversed) curY = _height - curY; - if (_marginTop + curY >= r.y() + r.height()) break; - - context.isAfterDate = (j > 0) ? !_items.at(j - 1)->toMediaItem() : false; - p.translate(0, curY - y); - _items.at(i)->paint(p, r.translated(-_rowsLeft, -_marginTop - curY), itemSelectedValue(i), &context); - y = curY; - } - } - } -} - -void OverviewInner::mouseMoveEvent(QMouseEvent *e) { - if (!(e->buttons() & (Qt::LeftButton | Qt::MiddleButton)) && _dragAction != NoDrag) { - mouseReleaseEvent(e); - } - dragActionUpdate(e->globalPos()); -} - -void OverviewInner::onUpdateSelected() { - if (isHidden()) return; - - QPoint mousePos(mapFromGlobal(_dragPos)); - QPoint m(_overview->clampMousePosition(mousePos)); - - ClickHandlerPtr lnk; - ClickHandlerHost *lnkhost = nullptr; - HistoryItem *item = 0; - int32 index = -1; - int32 newsel = 0; - HistoryCursorState cursorState = HistoryDefaultCursorState; - if (_type == OverviewPhotos || _type == OverviewVideos) { - float64 w = (float64(_width - st::overviewPhotoSkip) / _photosInRow); - int32 col = int32((m.x() - (st::overviewPhotoSkip / 2)) / w), vsize = (_rowWidth + st::overviewPhotoSkip); - int32 row = int32((m.y() - _marginTop - (st::overviewPhotoSkip / 2)) / vsize); - if (col < 0) col = 0; - if (row < 0) row = 0; - bool upon = true; - - int32 count = _items.size(), i = row * _photosInRow + col; - if (i < 0) { - i = 0; - upon = false; - } - if (i >= count) { - i = count - 1; - upon = false; - } - if (i >= 0) { - if (auto media = _items.at(i)->toMediaItem()) { - item = media->getItem(); - index = i; - if (upon) { - auto result = media->getState(m - QPoint(col * w + st::overviewPhotoSkip, _marginTop + row * vsize + st::overviewPhotoSkip), HistoryStateRequest()); - lnk = result.link; - cursorState = result.cursor; - lnkhost = media; - } - } - } - } else { - for (int32 j = 0, l = _items.size(); j < l; ++j) { - bool lastItem = (j + 1 == l); - int32 i = _reversed ? (l - j - 1) : j, nexti = _reversed ? (i - 1) : (i + 1); - int32 nextItemTop = lastItem ? (_reversed ? 0 : _height) : _items.at(nexti)->Get()->top; - if (_reversed) nextItemTop = _height - nextItemTop; - if (_marginTop + nextItemTop > m.y() || lastItem) { - int32 top = _items.at(i)->Get()->top; - if (_reversed) top = _height - top; - if (!_items.at(i)->toMediaItem()) { // day item - int32 h = _items.at(i)->height(); - bool beforeItem = (_marginTop + top + h / 2) >= m.y(); - if (_reversed) beforeItem = !beforeItem; - if (i > 0 && (beforeItem || i == _items.size() - 1)) { - --i; - if (!_items.at(i)->toMediaItem()) break; // wtf - top = _items.at(i)->Get()->top; - } else if (i < _items.size() - 1 && (!beforeItem || !i)) { - ++i; - if (!_items.at(i)->toMediaItem()) break; // wtf - top = _items.at(i)->Get()->top; - } else { - break; // wtf - } - if (_reversed) top = _height - top; - j = _reversed ? (l - i - 1) : i; - } - - if (auto media = _items.at(i)->toMediaItem()) { - item = media->getItem(); - index = i; - auto result = media->getState(m - QPoint(_rowsLeft, _marginTop + top), HistoryStateRequest()); - lnk = result.link; - cursorState = result.cursor; - lnkhost = media; - } - break; - } - } - } - - MsgId oldMousedItem = _mousedItem; - int32 oldMousedItemIndex = _mousedItemIndex; - _mousedItem = item ? ((item->history() == _migrated) ? -item->id : item->id) : 0; - _mousedItemIndex = index; - m = mapMouseToItem(m, _mousedItem, _mousedItemIndex); - - Qt::CursorShape cur = style::cur_default; - bool lnkChanged = ClickHandler::setActive(lnk, lnkhost); - if (lnkChanged) { - Ui::Tooltip::Hide(); - } - App::mousedItem(item); - if (_mousedItem != oldMousedItem) { - Ui::Tooltip::Hide(); - if (oldMousedItem) repaintItem(oldMousedItem, oldMousedItemIndex); - if (item) repaintItem(item); - } - if (_cursorState == HistoryInDateCursorState && cursorState != HistoryInDateCursorState) { - Ui::Tooltip::Hide(); - } - if (cursorState != _cursorState) { - _cursorState = cursorState; - } - if (lnk || cursorState == HistoryInDateCursorState) { - Ui::Tooltip::Show(1000, this); - } - - fixItemIndex(_dragItemIndex, _dragItem); - fixItemIndex(_mousedItemIndex, _mousedItem); - if (_dragAction == NoDrag) { - if (lnk) { - cur = style::cur_pointer; - } - } else { - if (_dragItemIndex < 0 || _mousedItemIndex < 0) { - _dragAction = NoDrag; - return; - } - if (_mousedItem != _dragItem || (m - _dragStartPos).manhattanLength() >= QApplication::startDragDistance()) { - if (_dragAction == PrepareDrag) { - _dragAction = Dragging; - InvokeQueued(this, [this] { performDrag(); }); - } else if (_dragAction == PrepareSelect) { - _dragAction = Selecting; - } - } - if (_dragAction == Selecting) { - bool canSelectMany = (_peer != 0); - if (_mousedItem == _dragItem && lnk && !_selected.isEmpty() && _selected.cbegin().value() != FullSelection) { - bool afterSymbol = false, uponSymbol = false; - uint16 second = 0; - _selected[_dragItem] = { 0, 0 }; - updateDragSelection(0, -1, 0, -1, false); - } else if (canSelectMany) { - bool selectingDown = (_reversed ? (_mousedItemIndex < _dragItemIndex) : (_mousedItemIndex > _dragItemIndex)) || (_mousedItemIndex == _dragItemIndex && ((_type == OverviewPhotos || _type == OverviewVideos) ? (_dragStartPos.x() < m.x()) : (_dragStartPos.y() < m.y()))); - MsgId dragSelFrom = _dragItem, dragSelTo = _mousedItem; - int32 dragSelFromIndex = _dragItemIndex, dragSelToIndex = _mousedItemIndex; - if (!itemHasPoint(dragSelFrom, dragSelFromIndex, _dragStartPos.x(), _dragStartPos.y())) { // maybe exclude dragSelFrom - if (selectingDown) { - if (_type == OverviewPhotos || _type == OverviewVideos) { - if (_dragStartPos.x() >= _rowWidth || ((_mousedItem == dragSelFrom) && (m.x() < _dragStartPos.x() + QApplication::startDragDistance()))) { - moveToNextItem(dragSelFrom, dragSelFromIndex, dragSelTo, 1); - } - } else { - if (_dragStartPos.y() >= itemHeight(dragSelFrom, dragSelFromIndex) || ((_mousedItem == dragSelFrom) && (m.y() < _dragStartPos.y() + QApplication::startDragDistance()))) { - moveToNextItem(dragSelFrom, dragSelFromIndex, dragSelTo, _reversed ? -1 : 1); - } - } - } else { - if (_type == OverviewPhotos || _type == OverviewVideos) { - if (_dragStartPos.x() < 0 || ((_mousedItem == dragSelFrom) && (m.x() >= _dragStartPos.x() - QApplication::startDragDistance()))) { - moveToNextItem(dragSelFrom, dragSelFromIndex, dragSelTo, -1); - } - } else { - if (_dragStartPos.y() < 0 || ((_mousedItem == dragSelFrom) && (m.y() >= _dragStartPos.y() - QApplication::startDragDistance()))) { - moveToNextItem(dragSelFrom, dragSelFromIndex, dragSelTo, _reversed ? 1 : -1); - } - } - } - } - if (_dragItem != _mousedItem) { // maybe exclude dragSelTo - if (selectingDown) { - if (_type == OverviewPhotos || _type == OverviewVideos) { - if (m.x() < 0) { - moveToNextItem(dragSelTo, dragSelToIndex, dragSelFrom, -1); - } - } else { - if (m.y() < 0) { - moveToNextItem(dragSelTo, dragSelToIndex, dragSelFrom, _reversed ? 1 : -1); - } - } - } else { - if (_type == OverviewPhotos || _type == OverviewVideos) { - if (m.x() >= _rowWidth) { - moveToNextItem(dragSelTo, dragSelToIndex, dragSelFrom, 1); - } - } else { - if (m.y() >= itemHeight(dragSelTo, dragSelToIndex)) { - moveToNextItem(dragSelTo, dragSelToIndex, dragSelFrom, _reversed ? -1 : 1); - } - } - } - } - bool dragSelecting = false; - MsgId dragFirstAffected = dragSelFrom; - int32 dragFirstAffectedIndex = dragSelFromIndex; - while (dragFirstAffectedIndex >= 0 && itemMsgId(dragFirstAffected) <= 0) { - moveToNextItem(dragFirstAffected, dragFirstAffectedIndex, dragSelTo, selectingDown ? (_reversed ? -1 : 1) : (_reversed ? 1 : -1)); - } - if (dragFirstAffectedIndex >= 0) { - SelectedItems::const_iterator i = _selected.constFind(dragFirstAffected); - dragSelecting = (i == _selected.cend() || i.value() != FullSelection); - } - updateDragSelection(dragSelFrom, dragSelFromIndex, dragSelTo, dragSelToIndex, dragSelecting); - } - } else if (_dragAction == Dragging) { - } - - if (ClickHandler::getPressed()) { - cur = style::cur_pointer; - } else if (_dragAction == Selecting && !_selected.isEmpty() && _selected.cbegin().value() != FullSelection) { - if (!_dragSelFrom || !_dragSelTo) { - cur = style::cur_text; - } - } - } - if (_dragAction == Selecting) { - _overview->checkSelectingScroll(mousePos); - } else { - updateDragSelection(0, -1, 0, -1, false); - _overview->noSelectingScroll(); - } - - if (lnkChanged || cur != _cursor) { - setCursor(_cursor = cur); - } -} - -QPoint OverviewInner::tooltipPos() const { - return _dragPos; -} - -QString OverviewInner::tooltipText() const { - if (_cursorState == HistoryInDateCursorState && _dragAction == NoDrag && _mousedItem) { - if (HistoryItem *item = App::histItemById(itemChannel(_mousedItem), itemMsgId(_mousedItem))) { - return item->date.toString(QLocale::system().dateTimeFormat(QLocale::LongFormat)); - } - } else if (ClickHandlerPtr lnk = ClickHandler::getActive()) { - return lnk->tooltip(); - } - return QString(); -} - -void OverviewInner::updateDragSelection(MsgId dragSelFrom, int32 dragSelFromIndex, MsgId dragSelTo, int32 dragSelToIndex, bool dragSelecting) { - if (_dragSelFrom != dragSelFrom || _dragSelFromIndex != dragSelFromIndex || _dragSelTo != dragSelTo || _dragSelToIndex != dragSelToIndex || _dragSelecting != dragSelecting) { - _dragSelFrom = dragSelFrom; - _dragSelFromIndex = dragSelFromIndex; - _dragSelTo = dragSelTo; - _dragSelToIndex = dragSelToIndex; - if (_dragSelFromIndex >= 0 && _dragSelToIndex >= 0 && _dragSelFromIndex < _dragSelToIndex) { - qSwap(_dragSelFrom, _dragSelTo); - qSwap(_dragSelFromIndex, _dragSelToIndex); - } - _dragSelecting = dragSelecting; - update(); - } -} - -void OverviewInner::mousePressEvent(QMouseEvent *e) { - if (_menu) { - e->accept(); - return; // ignore mouse press, that was hiding context menu - } - dragActionStart(e->globalPos(), e->button()); -} - -void OverviewInner::mouseReleaseEvent(QMouseEvent *e) { - dragActionFinish(e->globalPos(), e->button()); - if (!rect().contains(e->pos())) { - leaveEvent(e); - } -} - -void OverviewInner::keyPressEvent(QKeyEvent *e) { - if ((_search->isHidden() || !_search->hasFocus()) && !_overview->isHidden() && e->key() == Qt::Key_Escape) { - onCancel(); - } else if (e->key() == Qt::Key_Back) { - App::main()->showBackFromStack(Window::SectionShow()); - } else if (e->key() == Qt::Key_Return || e->key() == Qt::Key_Enter) { - onSearchMessages(); - } -} - -void OverviewInner::enterEventHook(QEvent *e) { - return TWidget::enterEventHook(e); -} - -void OverviewInner::leaveEventHook(QEvent *e) { - if (auto selectedMsgId = base::take(_selectedMsgId)) { - repaintItem(selectedMsgId, -1); - } - ClickHandler::clearActive(); - if (!ClickHandler::getPressed() && _cursor != style::cur_default) { - _cursor = style::cur_default; - setCursor(_cursor); - } - return TWidget::leaveEventHook(e); -} - -void OverviewInner::resizeEvent(QResizeEvent *e) { - onUpdateSelected(); - update(); -} - -void OverviewInner::showContextMenu(QContextMenuEvent *e, bool showFromTouch) { - if (_menu) { - _menu->deleteLater(); - _menu = 0; - repaintItem(App::contextItem()); - if (_selectedMsgId) repaintItem(_selectedMsgId, -1); - } - if (e->reason() == QContextMenuEvent::Mouse) { - dragActionUpdate(e->globalPos()); - } - - bool ignoreMousedItem = false; - if (itemMsgId(_mousedItem) > 0) { - QPoint m = mapMouseToItem(mapFromGlobal(e->globalPos()), _mousedItem, _mousedItemIndex); - if (m.y() < 0 || m.y() >= itemHeight(_mousedItem, _mousedItemIndex)) { - ignoreMousedItem = true; - } - } - - auto selectedState = getSelectionState(); - - // -2 - has full selected items, but not over, 0 - no selection, 2 - over full selected items - int32 isUponSelected = 0, hasSelected = 0; - if (!_selected.isEmpty()) { - isUponSelected = -1; - if (_selected.cbegin().value() == FullSelection) { - hasSelected = 2; - if (!ignoreMousedItem && App::mousedItem() && _selected.constFind(App::mousedItem()->history() == _migrated ? -App::mousedItem()->id : App::mousedItem()->id) != _selected.cend()) { - isUponSelected = 2; - } else { - isUponSelected = -2; - } - } - } - if (showFromTouch && hasSelected && isUponSelected < hasSelected) { - isUponSelected = hasSelected; - } - - _contextMenuLnk = ClickHandler::getActive(); - PhotoClickHandler *lnkPhoto = dynamic_cast(_contextMenuLnk.data()); - DocumentClickHandler *lnkDocument = dynamic_cast(_contextMenuLnk.data()); - bool lnkIsVideo = lnkDocument ? lnkDocument->document()->isVideo() : false; - bool lnkIsAudio = lnkDocument ? (lnkDocument->document()->voice() != nullptr) : false; - bool lnkIsSong = lnkDocument ? (lnkDocument->document()->song() != nullptr) : false; - if (lnkPhoto || lnkDocument) { - _menu = new Ui::PopupMenu(nullptr); - if (App::hoveredLinkItem()) { - _menu->addAction(lang(lng_context_to_msg), this, SLOT(goToMessage()))->setEnabled(true); - } - if (lnkPhoto) { - } else { - if (auto document = lnkDocument->document()) { - if (document->loading()) { - _menu->addAction(lang(lng_context_cancel_download), this, SLOT(cancelContextDownload()))->setEnabled(true); - } else { - if (!document->filepath(DocumentData::FilePathResolveChecked).isEmpty()) { - _menu->addAction(lang((cPlatform() == dbipMac || cPlatform() == dbipMacOld) ? lng_context_show_in_finder : lng_context_show_in_folder), this, SLOT(showContextInFolder()))->setEnabled(true); - } - _menu->addAction(lang(lnkIsVideo ? lng_context_save_video : (lnkIsAudio ? lng_context_save_audio : (lnkIsSong ? lng_context_save_audio_file : lng_context_save_file))), App::LambdaDelayed(st::defaultDropdownMenu.menu.ripple.hideDuration, this, [this, document] { - saveDocumentToFile(document); - }))->setEnabled(true); - } - } - } - if (isUponSelected > 1) { - if (selectedState.count > 0 && selectedState.count == selectedState.canForwardCount) { - _menu->addAction(lang(lng_context_forward_selected), _overview, SLOT(onForwardSelected())); - } - if (selectedState.count > 0 && selectedState.count == selectedState.canDeleteCount) { - _menu->addAction(lang(lng_context_delete_selected), base::lambda_guarded(this, [this] { - _overview->confirmDeleteSelectedItems(); - })); - } - _menu->addAction(lang(lng_context_clear_selection), _overview, SLOT(onClearSelected())); - } else if (App::hoveredLinkItem()) { - if (isUponSelected != -2) { - if (App::hoveredLinkItem()->canForward()) { - _menu->addAction(lang(lng_context_forward_msg), this, SLOT(forwardMessage()))->setEnabled(true); - } - if (App::hoveredLinkItem()->canDelete()) { - _menu->addAction(lang(lng_context_delete_msg), base::lambda_guarded(this, [this] { - _overview->confirmDeleteContextItem(); - })); - } - } - if (App::hoveredLinkItem()->id > 0) { - _menu->addAction(lang(lng_context_select_msg), this, SLOT(selectMessage()))->setEnabled(true); - } - } - App::contextItem(App::hoveredLinkItem()); - repaintItem(App::contextItem()); - if (_selectedMsgId) repaintItem(_selectedMsgId, -1); - } else if (!ignoreMousedItem && App::mousedItem() && App::mousedItem()->channelId() == itemChannel(_mousedItem) && App::mousedItem()->id == itemMsgId(_mousedItem)) { - _menu = new Ui::PopupMenu(nullptr); - QString linkCopyToClipboardText = _contextMenuLnk ? _contextMenuLnk->copyToClipboardContextItemText() : QString(); - if (!linkCopyToClipboardText.isEmpty()) { - _menu->addAction(linkCopyToClipboardText, this, SLOT(copyContextUrl()))->setEnabled(true); - } - _menu->addAction(lang(lng_context_to_msg), this, SLOT(goToMessage()))->setEnabled(true); - if (isUponSelected > 1) { - if (selectedState.count > 0 && selectedState.count == selectedState.canForwardCount) { - _menu->addAction(lang(lng_context_forward_selected), _overview, SLOT(onForwardSelected())); - } - if (selectedState.count > 0 && selectedState.count == selectedState.canDeleteCount) { - _menu->addAction(lang(lng_context_delete_selected), base::lambda_guarded(this, [this] { - _overview->confirmDeleteSelectedItems(); - })); - } - _menu->addAction(lang(lng_context_clear_selection), _overview, SLOT(onClearSelected())); - } else { - if (isUponSelected != -2) { - if (App::mousedItem()->canForward()) { - _menu->addAction(lang(lng_context_forward_msg), this, SLOT(forwardMessage()))->setEnabled(true); - } - if (App::mousedItem()->canDelete()) { - _menu->addAction(lang(lng_context_delete_msg), base::lambda_guarded(this, [this] { - _overview->confirmDeleteContextItem(); - })); - } - } - if (App::mousedItem()->id > 0) { - _menu->addAction(lang(lng_context_select_msg), this, SLOT(selectMessage()))->setEnabled(true); - } - } - App::contextItem(App::mousedItem()); - repaintItem(App::contextItem()); - if (_selectedMsgId) repaintItem(_selectedMsgId, -1); - } - if (_menu) { - connect(_menu, SIGNAL(destroyed(QObject*)), this, SLOT(onMenuDestroy(QObject*))); - _menu->popup(e->globalPos()); - e->accept(); - } -} - -int32 OverviewInner::resizeToWidth(int32 nwidth, int32 scrollTop, int32 minHeight, bool force) { - if (!force && _width == nwidth && minHeight == _minHeight) return scrollTop; - - if ((_type == OverviewPhotos || _type == OverviewVideos) && _resizeIndex < 0) { - _resizeIndex = _photosInRow * ((scrollTop + minHeight) / int32(_rowWidth + st::overviewPhotoSkip)) + _photosInRow - 1; - _resizeSkip = (scrollTop + minHeight) - ((scrollTop + minHeight) / int32(_rowWidth + st::overviewPhotoSkip)) * int32(_rowWidth + st::overviewPhotoSkip); - } - - _width = nwidth; - _minHeight = minHeight; - if (_type == OverviewPhotos || _type == OverviewVideos) { - _photosInRow = int32(_width - st::overviewPhotoSkip) / int32(st::overviewPhotoMinSize + st::overviewPhotoSkip); - _rowWidth = (int32(_width - st::overviewPhotoSkip) / _photosInRow) - st::overviewPhotoSkip; - _rowsLeft = st::overviewPhotoSkip; - } else { - auto contentLeftMin = st::overviewLeftMin; - auto contentLeftMax = st::overviewLeftMax; - if (_type == OverviewMusicFiles || _type == OverviewVoiceFiles) { - contentLeftMin -= st::overviewFileLayout.songPadding.left(); - contentLeftMax -= st::overviewFileLayout.songPadding.left(); - } - auto widthWithMin = st::columnMinimalWidthMain; - auto widthWithMax = st::overviewFileLayout.maxWidth + 2 * contentLeftMax; - _rowsLeft = anim::interpolate(contentLeftMax, contentLeftMin, qMax(widthWithMax - _width, 0) / float64(widthWithMax - widthWithMin)); - _rowWidth = qMin(_width - 2 * _rowsLeft, st::overviewFileLayout.maxWidth); - } - - _search->setGeometry(_rowsLeft, st::linksSearchTop, _rowWidth, _search->height()); - _cancelSearch->moveToLeft(_rowsLeft + _rowWidth - _cancelSearch->width(), _search->y()); - - resizeItems(); - recountMargins(); - - resize(_width, _marginTop + _height + _marginBottom); - - if (_type == OverviewPhotos || _type == OverviewVideos) { - int32 newRow = _resizeIndex / _photosInRow; - return newRow * int32(_rowWidth + st::overviewPhotoSkip) + _resizeSkip - minHeight; - } - return scrollTop; -} - -void OverviewInner::resizeItems() { - if (_type == OverviewPhotos || _type == OverviewVideos || _type == OverviewLinks) { - resizeAndRepositionItems(); - } else { - for (auto i = 0, l = _items.size(); i != l; ++i) { - _items.at(i)->resizeGetHeight(_rowWidth); - } - } -} - -void OverviewInner::resizeAndRepositionItems() { - if (_type == OverviewPhotos || _type == OverviewVideos) { - for (auto i = 0, l = _items.size(); i < l; ++i) { - _items.at(i)->resizeGetHeight(_rowWidth); - } - _height = countHeight(); - } else { - _height = 0; - for (auto i = 0, l = _items.size(); i < l; ++i) { - auto h = _items.at(i)->resizeGetHeight(_rowWidth); - _items.at(i)->Get()->top = _height + (_reversed ? h : 0); - _height += h; - } - } -} - -void OverviewInner::dropResizeIndex() { - _resizeIndex = -1; -} - -PeerData *OverviewInner::peer() const { - return _peer; -} - -PeerData *OverviewInner::migratePeer() const { - return _migrated ? _migrated->peer : 0; -} - -MediaOverviewType OverviewInner::type() const { - return _type; -} - -void OverviewInner::switchType(MediaOverviewType type) { - if (_type != type) { - clear(); - _type = type; - _reversed = (_type != OverviewLinks && _type != OverviewFiles); - if (_type == OverviewLinks || _type == OverviewFiles) { - _search->show(); - } else { - _search->hide(); - } - - if (!_search->getLastText().isEmpty()) { - _search->setText(QString()); - _search->updatePlaceholder(); - onSearchUpdate(); - } - _cancelSearch->hideFast(); - - resizeToWidth(_width, 0, _minHeight, true); - } - mediaOverviewUpdated(); - if (App::wnd()) App::wnd()->update(); -} - -void OverviewInner::setSelectMode(bool enabled) { - _selMode = enabled; -} - -void OverviewInner::copyContextUrl() { - if (_contextMenuLnk) { - _contextMenuLnk->copyToClipboard(); - } -} - -void OverviewInner::goToMessage() { - HistoryItem *item = App::contextItem(); - if (!item) return; - - Ui::showPeerHistoryAtItem(item); -} - -void OverviewInner::forwardMessage() { - auto item = App::contextItem(); - if (!item || item->id < 0 || item->serviceMsg()) return; - - auto items = SelectedItemSet(); - items.insert(item->id, item); - App::main()->showForwardLayer(items); -} - -MsgId OverviewInner::complexMsgId(const HistoryItem *item) const { - return item ? ((item->history() == _migrated) ? -item->id : item->id) : 0; -} - -void OverviewInner::selectMessage() { - auto item = App::contextItem(); - if (!item || item->id < 0) return; - - if (!_selected.isEmpty() && _selected.cbegin().value() != FullSelection) { - _selected.clear(); - } else if (_selected.size() == MaxSelectedItems && _selected.constFind(complexMsgId(item)) == _selected.cend()) { - return; - } - _selected.insert(complexMsgId(item), FullSelection); - _overview->updateTopBarSelection(); - _overview->update(); -} - -void OverviewInner::cancelContextDownload() { - DocumentClickHandler *lnkDocument = dynamic_cast(_contextMenuLnk.data()); - if (lnkDocument) { - lnkDocument->document()->cancel(); - } -} - -void OverviewInner::showContextInFolder() { - if (auto lnkDocument = dynamic_cast(_contextMenuLnk.data())) { - auto filepath = lnkDocument->document()->filepath(DocumentData::FilePathResolveChecked); - if (!filepath.isEmpty()) { - File::ShowInFolder(filepath); - } - } -} - -void OverviewInner::saveDocumentToFile(DocumentData *document) { - DocumentSaveClickHandler::doSave(document, true); -} - -bool OverviewInner::onSearchMessages(bool searchCache) { - _searchTimer.stop(); - auto q = _search->text().trimmed(); - if (q.isEmpty()) { - MTP::cancel(base::take(_searchRequest)); - return true; - } - if (searchCache) { - auto i = _searchCache.constFind(q); - if (i != _searchCache.cend()) { - _searchQuery = q; - _searchFull = _searchFullMigrated = false; - _searchRequest = 0; - searchReceived(SearchFromStart, i.value(), 0); - return true; - } - } else if (_searchQuery != q) { - _searchQuery = q; - _searchFull = _searchFullMigrated = false; - auto filter = (_type == OverviewLinks) ? MTP_inputMessagesFilterUrl() : MTP_inputMessagesFilterDocument(); - _searchRequest = MTP::send(MTPmessages_Search(MTP_flags(0), _history->peer->input, MTP_string(_searchQuery), MTP_inputUserEmpty(), filter, MTP_int(0), MTP_int(0), MTP_int(0), MTP_int(0), MTP_int(SearchPerPage), MTP_int(0), MTP_int(0)), rpcDone(&OverviewInner::searchReceived, SearchFromStart), rpcFail(&OverviewInner::searchFailed, SearchFromStart)); - _searchQueries.insert(_searchRequest, _searchQuery); - } - return false; -} - -void OverviewInner::onNeedSearchMessages() { - if (!onSearchMessages(true)) { - _searchTimer.start(AutoSearchTimeout); - if (_inSearch && _searchFull && (!_migrated || _searchFullMigrated) && _searchResults.isEmpty()) { - update(); - } - } -} - -void OverviewInner::onSearchUpdate() { - QString filterText = (_type == OverviewLinks || _type == OverviewFiles) ? _search->text().trimmed() : QString(); - bool inSearch = !filterText.isEmpty(), changed = (inSearch != _inSearch); - _inSearch = inSearch; - - onNeedSearchMessages(); - - if (!_inSearch) { - _searchCache.clear(); - _searchQueries.clear(); - _searchQuery = QString(); - _searchResults.clear(); - _cancelSearch->hideAnimated(); - } else { - _cancelSearch->showAnimated(); - } - - if (changed) { - _itemsToBeLoaded = LinksOverviewPerPage * 2; - mediaOverviewUpdated(); - } - _overview->scrollReset(); -} - -void OverviewInner::onCancel() { - if (_selected.isEmpty()) { - if (onCancelSearch()) return; - App::main()->showBackFromStack(Window::SectionShow()); - } else { - _overview->onClearSelected(); - } -} - -bool OverviewInner::onCancelSearch() { - if (_search->isHidden()) return false; - bool clearing = !_search->text().isEmpty(); - _cancelSearch->hideAnimated(); - _search->clear(); - _search->updatePlaceholder(); - onSearchUpdate(); - return clearing; -} - -void OverviewInner::onMenuDestroy(QObject *obj) { - if (_menu == obj) { - _menu = 0; - dragActionUpdate(QCursor::pos()); - repaintItem(App::contextItem()); - if (_selectedMsgId) repaintItem(_selectedMsgId, -1); - } -} - -Window::TopBarWidget::SelectedState OverviewInner::getSelectionState() const { - auto result = Window::TopBarWidget::SelectedState {}; - for (auto i = _selected.cbegin(), e = _selected.cend(); i != e; ++i) { - if (i.value() == FullSelection) { - if (auto item = App::histItemById(itemChannel(i.key()), itemMsgId(i.key()))) { - ++result.count; - if (item->canForward()) { - ++result.canForwardCount; - } - if (item->canDelete()) { - ++result.canDeleteCount; - } - } - } - } - return result; -} - -void OverviewInner::clearSelectedItems(bool onlyTextSelection) { - if (!_selected.isEmpty() && (!onlyTextSelection || _selected.cbegin().value() != FullSelection)) { - _selected.clear(); - _overview->updateTopBarSelection(); - _overview->update(); - } -} - -SelectedItemSet OverviewInner::getSelectedItems() const { - auto result = SelectedItemSet(); - if (_selected.isEmpty() || _selected.cbegin().value() != FullSelection) { - return result; - } - - for (auto i = _selected.cbegin(), e = _selected.cend(); i != e; ++i) { - auto item = App::histItemById(itemChannel(i.key()), itemMsgId(i.key())); - if (item && item->toHistoryMessage() && item->id > 0) { - if (item->history() == _migrated) { - result.insert(item->id - ServerMaxMsgId, item); - } else { - result.insert(item->id, item); - } - } - } - return result; -} - -void OverviewInner::onTouchSelect() { - _touchSelect = true; - dragActionStart(_touchPos); -} - -void OverviewInner::onTouchScrollTimer() { - auto nowTime = getms(); - if (_touchScrollState == Ui::TouchScrollState::Acceleration && _touchWaitingAcceleration && (nowTime - _touchAccelerationTime) > 40) { - _touchScrollState = Ui::TouchScrollState::Manual; - touchResetSpeed(); - } else if (_touchScrollState == Ui::TouchScrollState::Auto || _touchScrollState == Ui::TouchScrollState::Acceleration) { - int32 elapsed = int32(nowTime - _touchTime); - QPoint delta = _touchSpeed * elapsed / 1000; - bool hasScrolled = _overview->touchScroll(delta); - - if (_touchSpeed.isNull() || !hasScrolled) { - _touchScrollState = Ui::TouchScrollState::Manual; - _touchScroll = false; - _touchScrollTimer.stop(); - } else { - _touchTime = nowTime; - } - touchDeaccelerate(elapsed); - } -} - -void OverviewInner::mediaOverviewUpdated() { - if (_type == OverviewPhotos || _type == OverviewVideos) { - auto &o = _history->overview(_type); - auto migratedOverview = _migrated ? &_migrated->overview(_type) : nullptr; - auto migrateCount = migratedIndexSkip(); - auto wasCount = _items.size(); - auto fullCount = (migrateCount + o.size()); - auto tocheck = qMin(fullCount, _itemsToBeLoaded); - _items.reserve(tocheck); - - auto index = 0; - auto allGood = true; - auto migrateIt = migratedOverview ? migratedOverview->end() : o.end(); - auto it = o.end(); - for (auto i = fullCount, l = fullCount - tocheck; i > l;) { - --i; - auto msgid = MsgId(0); - if (i < migrateCount) { - --migrateIt; - msgid = -(*migrateIt); - } else { - --it; - msgid = *it; - } - if (allGood) { - if (_items.size() > index && complexMsgId(_items.at(index)->getItem()) == msgid) { - ++index; - continue; - } - allGood = false; - } - HistoryItem *item = App::histItemById(itemChannel(msgid), itemMsgId(msgid)); - auto layout = layoutPrepare(item); - if (!layout) continue; - - setLayoutItem(index, layout, 0); - ++index; - } - if (_items.size() > index) _items.resize(index); - - _height = countHeight(); - } else { - bool dateEveryMonth = (_type == OverviewFiles), dateEveryDay = (_type == OverviewLinks); - bool withDates = (dateEveryMonth || dateEveryDay); - - auto &o = _history->overview(_type); - auto migratedOverview = _migrated ? &_migrated->overview(_type) : nullptr; - auto migrateCount = migratedIndexSkip(); - auto l = _inSearch ? _searchResults.size() : (migrateCount + o.size()); - auto tocheck = qMin(l, _itemsToBeLoaded); - _items.reserve((withDates ? 2 : 1) * tocheck); // day items - - auto migrateIt = migratedOverview ? migratedOverview->end() : o.end(); - auto it = o.end(); - - auto top = 0; - auto count = 0; - bool allGood = true; - QDate prevDate; - for (auto i = 0; i < tocheck; ++i) { - auto msgid = MsgId(0); - auto index = l - i - 1; - if (_inSearch) { - msgid = _searchResults[index]; - } else if (index < migrateCount) { - --migrateIt; - msgid = -(*migrateIt); - } else { - --it; - msgid = *it; - } - if (allGood) { - if (_items.size() > count && complexMsgId(_items.at(count)->getItem()) == msgid) { - if (withDates) prevDate = _items.at(count)->getItem()->date.date(); - top = _items.at(count)->Get()->top; - if (!_reversed) { - top += _items.at(count)->height(); - } - ++count; - continue; - } - if (_items.size() > count + 1 && !_items.at(count)->toMediaItem() && complexMsgId(_items.at(count + 1)->getItem()) == msgid) { // day item - ++count; - if (withDates) prevDate = _items.at(count)->getItem()->date.date(); - top = _items.at(count)->Get()->top; - if (!_reversed) { - top += _items.at(count)->height(); - } - ++count; - continue; - } - allGood = false; - } - auto item = App::histItemById(itemChannel(msgid), itemMsgId(msgid)); - auto layout = layoutPrepare(item); - if (!layout) continue; - - if (withDates) { - QDate date = item->date.date(); - if (!count || (count > 0 && (dateEveryMonth ? (date.month() != prevDate.month() || date.year() != prevDate.year()) : (date != prevDate)))) { - top += setLayoutItem(count, layoutPrepare(date, dateEveryMonth), top); - ++count; - prevDate = date; - } - } - top += setLayoutItem(count, layout, top); - ++count; - } - if (_items.size() > count) _items.resize(count); - - _height = top; - } - - fixItemIndex(_dragSelFromIndex, _dragSelFrom); - fixItemIndex(_dragSelToIndex, _dragSelTo); - fixItemIndex(_mousedItemIndex, _mousedItem); - fixItemIndex(_dragItemIndex, _dragItem); - - recountMargins(); - int32 newHeight = _marginTop + _height + _marginBottom, deltaHeight = newHeight - height(); - if (deltaHeight) { - resize(_width, newHeight); - if (_type == OverviewMusicFiles || _type == OverviewVoiceFiles) { - _overview->scrollBy(deltaHeight); - } - } else { - onUpdateSelected(); - update(); - } -} - -void OverviewInner::changingMsgId(HistoryItem *row, MsgId newId) { - if (peer() != row->history()->peer && migratePeer() != row->history()->peer) { - return; - } - - MsgId oldId = complexMsgId(row); - if (row->history() == _migrated) newId = -newId; - - if (_dragSelFrom == oldId) _dragSelFrom = newId; - if (_dragSelTo == oldId) _dragSelTo = newId; - if (_mousedItem == oldId) _mousedItem = newId; - if (_dragItem == oldId) _dragItem = newId; - if (_selectedMsgId == oldId) _selectedMsgId = newId; - for (SelectedItems::iterator i = _selected.begin(), e = _selected.end(); i != e; ++i) { - if (i.key() == oldId) { - auto sel = i.value(); - _selected.erase(i); - _selected.insert(newId, sel); - break; - } - } -} - -void OverviewInner::itemRemoved(HistoryItem *item) { - if (_history != item->history() && _migrated != item->history()) { - return; - } - - MsgId msgId = (item->history() == _migrated) ? -item->id : item->id; - if (_dragItem == msgId) { - dragActionCancel(); - } - if (_selectedMsgId == msgId) { - _selectedMsgId = 0; - } - - auto i = _selected.find(msgId); - if (i != _selected.cend()) { - _selected.erase(i); - _overview->updateTopBarSelection(); - } - - auto j = _layoutItems.find(item); - if (j != _layoutItems.cend()) { - int32 index = _items.indexOf(j.value()); - if (index >= 0) { - _items.remove(index); - } - delete j.value(); - _layoutItems.erase(j); - - resizeAndRepositionItems(); - } - - if (_dragSelFrom == msgId || _dragSelTo == msgId) { - _dragSelFrom = 0; - _dragSelFromIndex = -1; - _dragSelTo = 0; - _dragSelToIndex = -1; - update(); - } - - onUpdateSelected(); -} - -void OverviewInner::repaintItem(const HistoryItem *msg) { - if (!msg) return; - - History *history = (msg->history() == _history) ? _history : (msg->history() == _migrated ? _migrated : 0); - if (!history) return; - - int32 migrateindex = migratedIndexSkip(); - MsgId msgid = msg->id; - if ((history == _history || migrateindex > 0) && (_inSearch || history->overviewHasMsgId(_type, msgid))) { - if (_type == OverviewPhotos || _type == OverviewVideos) { - if (history == _migrated) msgid = -msgid; - for (int32 i = 0, l = _items.size(); i != l; ++i) { - if (complexMsgId(_items.at(i)->getItem()) == msgid) { - float64 w = (float64(width() - st::overviewPhotoSkip) / _photosInRow); - int32 vsize = (_rowWidth + st::overviewPhotoSkip); - int32 row = i / _photosInRow, col = i % _photosInRow; - update(int32(col * w), _marginTop + int32(row * vsize), qCeil(w), vsize); - break; - } - } - } else { - if (history == _migrated) msgid = -msgid; - for (int32 i = 0, l = _items.size(); i != l; ++i) { - if (complexMsgId(_items.at(i)->getItem()) == msgid) { - int32 top = _items.at(i)->Get()->top; - if (_reversed) top = _height - top; - update(_rowsLeft, _marginTop + top, _rowWidth, _items.at(i)->height()); - break; - } - } - } - } -} - -int OverviewInner::countHeight() { - if (_type == OverviewPhotos || _type == OverviewVideos) { - auto count = _items.size(); - auto migratedFullCount = _migrated ? _migrated->overviewCount(_type) : 0; - auto fullCount = migratedFullCount + _history->overviewCount(_type); - auto rows = (count / _photosInRow) + ((count % _photosInRow) ? 1 : 0); - return (_rowWidth + st::overviewPhotoSkip) * rows + st::overviewPhotoSkip; - } - return _height; -} - -void OverviewInner::recountMargins() { - if (_type == OverviewPhotos || _type == OverviewVideos) { - _marginBottom = qMax(_minHeight - _height - _marginTop, 0); - _marginTop = 0; - } else if (_type == OverviewMusicFiles) { - _marginTop = st::playlistPadding; - _marginBottom = qMax(_minHeight - _height - _marginTop, int32(st::playlistPadding)); - } else if (_type == OverviewLinks || _type == OverviewFiles) { - _marginTop = st::linksSearchTop + _search->height(); - _marginBottom = qMax(_minHeight - _height - _marginTop, int32(st::playlistPadding)); - } else { - _marginBottom = st::playlistPadding; - _marginTop = qMax(_minHeight - _height - _marginBottom, int32(st::playlistPadding)); - } -} - -Overview::Layout::ItemBase *OverviewInner::layoutPrepare(HistoryItem *item) { - if (!item) return nullptr; - - LayoutItems::const_iterator i = _layoutItems.cend(); - HistoryMedia *media = item->getMedia(); - if (_type == OverviewPhotos) { - if (media && media->type() == MediaTypePhoto) { - if ((i = _layoutItems.constFind(item)) == _layoutItems.cend()) { - i = _layoutItems.insert(item, new Overview::Layout::Photo(item, static_cast(media)->photo())); - i.value()->initDimensions(); - } - } - } else if (_type == OverviewVideos) { - if (media && media->type() == MediaTypeVideo) { - if ((i = _layoutItems.constFind(item)) == _layoutItems.cend()) { - i = _layoutItems.insert(item, new Overview::Layout::Video(item, media->getDocument())); - i.value()->initDimensions(); - } - } - } else if (_type == OverviewVoiceFiles) { - if (media && (media->type() == MediaTypeVoiceFile)) { - if ((i = _layoutItems.constFind(item)) == _layoutItems.cend()) { - i = _layoutItems.insert(item, new Overview::Layout::Voice(item, media->getDocument(), st::overviewFileLayout)); - i.value()->initDimensions(); - } - } - } else if (_type == OverviewFiles || _type == OverviewMusicFiles) { - if (media && (media->type() == MediaTypeFile || media->type() == MediaTypeMusicFile || media->type() == MediaTypeGif)) { - if ((i = _layoutItems.constFind(item)) == _layoutItems.cend()) { - i = _layoutItems.insert(item, new Overview::Layout::Document(item, media->getDocument(), st::overviewFileLayout)); - i.value()->initDimensions(); - } - } - } else if (_type == OverviewLinks) { - if ((i = _layoutItems.constFind(item)) == _layoutItems.cend()) { - i = _layoutItems.insert(item, new Overview::Layout::Link(item, media)); - i.value()->initDimensions(); - } - } - return (i == _layoutItems.cend()) ? nullptr : i.value(); -} - -Overview::Layout::AbstractItem *OverviewInner::layoutPrepare(const QDate &date, bool month) { - int32 key = date.year() * 100 + date.month(); - if (!month) key = key * 100 + date.day(); - LayoutDates::const_iterator i = _layoutDates.constFind(key); - if (i == _layoutDates.cend()) { - i = _layoutDates.insert(key, new Overview::Layout::Date(date, month)); - i.value()->initDimensions(); - } - return i.value(); -} - -int32 OverviewInner::setLayoutItem(int32 index, Overview::Layout::AbstractItem *item, int32 top) { - if (_items.size() > index) { - _items[index] = item; - } else { - _items.push_back(item); - } - int32 h = item->resizeGetHeight(_rowWidth); - if (auto info = item->Get()) { - info->top = top + (_reversed ? h : 0); - } - return h; -} - -OverviewInner::~OverviewInner() { - clear(); -} - -OverviewWidget::OverviewWidget(QWidget *parent, not_null controller, PeerData *peer, MediaOverviewType type) : Window::AbstractSectionWidget(parent, controller) -, _topBar(this, controller) -, _scroll(this, st::settingsScroll, false) -, _mediaType(this, st::defaultDropdownMenu) -, _topShadow(this) { - _inner = _scroll->setOwnedWidget(object_ptr(this, _scroll, peer, type)); - _scroll->move(0, 0); - _inner->move(0, 0); - - connect(_topBar, &Window::TopBarWidget::clicked, this, [this] { topBarClick(); }); - - _mediaType->hide(); - _mediaType->setOrigin(Ui::PanelAnimation::Origin::TopRight); - _topBar->mediaTypeButton()->installEventFilter(_mediaType); - - _topBar->show(); - _scroll->show(); - connect(_scroll, SIGNAL(scrolled()), this, SLOT(onScroll())); - - connect(&_scrollTimer, SIGNAL(timeout()), this, SLOT(onScrollTimer())); - _scrollTimer.setSingleShot(false); - - switchType(type); -} - -void OverviewWidget::clear() { - _inner->clear(); -} - -void OverviewWidget::onScroll() { - Auth().downloader().clearPriorities(); - int32 preloadThreshold = _scroll->height() * 5; - bool needToPreload = false; - do { - needToPreload = (type() == OverviewMusicFiles || type() == OverviewVoiceFiles) ? (_scroll->scrollTop() < preloadThreshold) : (_scroll->scrollTop() + preloadThreshold > _scroll->scrollTopMax()); - if (!needToPreload || !_inner->preloadLocal()) { - break; - } - } while (true); - if (needToPreload) { - _inner->preloadMore(); - } - if (!_noDropResizeIndex) { - _inner->dropResizeIndex(); - } -} - -void OverviewWidget::resizeEvent(QResizeEvent *e) { - _topBar->setGeometryToLeft(0, 0, width(), st::topBarHeight); - auto scrollAreaTop = _topBar->bottomNoMargins(); - - _noDropResizeIndex = true; - auto st = _scroll->scrollTop(); - _scroll->setGeometryToLeft(0, scrollAreaTop, width(), height() - scrollAreaTop); - auto newScrollTop = _inner->resizeToWidth(width(), st, height() - _topBar->height()); - if (auto addToY = App::main() ? App::main()->contentScrollAddToY() : 0) { - newScrollTop += addToY; - } - if (newScrollTop != _scroll->scrollTop()) { - _scroll->scrollToY(newScrollTop); - } - _noDropResizeIndex = false; - - _topShadow->resize(width() - ((!Adaptive::OneColumn() && !_inGrab) ? st::lineWidth : 0), st::lineWidth); - _topShadow->moveToLeft((!Adaptive::OneColumn() && !_inGrab) ? st::lineWidth : 0, _topBar->bottomNoMargins()); - - _mediaType->moveToRight(0, scrollAreaTop); -} - -void OverviewWidget::paintEvent(QPaintEvent *e) { - if (App::wnd() && App::wnd()->contentOverlapped(this, e)) return; - - Painter p(this); - auto progress = _a_show.current(getms(), 1.); - if (_a_show.animating()) { - auto retina = cIntRetinaFactor(); - auto fromLeft = (_showDirection == Window::SlideDirection::FromLeft); - auto coordUnder = fromLeft ? anim::interpolate(-st::slideShift, 0, progress) : anim::interpolate(0, -st::slideShift, progress); - auto coordOver = fromLeft ? anim::interpolate(0, width(), progress) : anim::interpolate(width(), 0, progress); - auto shadow = fromLeft ? (1. - progress) : progress; - if (coordOver > 0) { - p.drawPixmap(QRect(0, 0, coordOver, height()), _cacheUnder, QRect(-coordUnder * retina, 0, coordOver * retina, height() * retina)); - p.setOpacity(shadow); - p.fillRect(0, 0, coordOver, height(), st::slideFadeOutBg); - p.setOpacity(1); - } - p.drawPixmap(QRect(coordOver, 0, _cacheOver.width() / retina, height()), _cacheOver, QRect(0, 0, _cacheOver.width(), height() * retina)); - p.setOpacity(shadow); - st::slideShadow.fill(p, QRect(coordOver - st::slideShadow.width(), 0, st::slideShadow.width(), height())); - return; - } - - p.fillRect(e->rect(), st::windowBg); -} - -void OverviewWidget::contextMenuEvent(QContextMenuEvent *e) { - return _inner->showContextMenu(e); -} - -void OverviewWidget::scrollBy(int32 add) { - if (_scroll->isHidden()) { - _scrollSetAfterShow += add; - } else { - _scroll->scrollToY(_scroll->scrollTop() + add); - } -} - -void OverviewWidget::scrollReset() { - _scroll->scrollToY((type() == OverviewMusicFiles || type() == OverviewVoiceFiles) ? _scroll->scrollTopMax() : 0); -} - -bool OverviewWidget::paintTopBar(Painter &p, int decreaseWidth) { - st::topBarBack.paint(p, (st::topBarArrowPadding.left() - st::topBarBack.width()) / 2, (st::topBarHeight - st::topBarBack.height()) / 2, width()); - p.setFont(st::defaultLightButton.font); - p.setPen(st::defaultLightButton.textFg); - p.drawTextLeft(st::topBarArrowPadding.left(), st::topBarButton.padding.top() + st::topBarButton.textTop, width(), _header); - return true; -} - -void OverviewWidget::topBarClick() { - App::main()->showBackFromStack(Window::SectionShow()); -} - -PeerData *OverviewWidget::peer() const { - return _inner->peer(); -} - -PeerData *OverviewWidget::migratePeer() const { - return _inner->migratePeer(); -} - -MediaOverviewType OverviewWidget::type() const { - return _inner->type(); -} - -void OverviewWidget::switchType(MediaOverviewType type) { - disconnect(_scroll, SIGNAL(scrolled()), this, SLOT(onScroll())); - - _inner->setSelectMode(false); - _inner->switchType(type); - switch (type) { - case OverviewPhotos: _header = lang(lng_profile_photos_header); break; - case OverviewVideos: _header = lang(lng_profile_videos_header); break; - case OverviewMusicFiles: _header = lang(lng_profile_songs_header); break; - case OverviewFiles: _header = lang(lng_profile_files_header); break; - case OverviewVoiceFiles: _header = lang(lng_profile_audios_header); break; - case OverviewLinks: _header = lang(lng_profile_shared_links_header); break; - } - _header = _header.toUpper(); - - noSelectingScroll(); - _topBar->showSelected(Window::TopBarWidget::SelectedState {}); - updateTopBarSelection(); - scrollReset(); - - connect(_scroll, SIGNAL(scrolled()), this, SLOT(onScroll())); - - onScroll(); - activate(); -} - -bool OverviewWidget::showMediaTypeSwitch() const { - for (int32 i = 0; i < OverviewCount; ++i) { - if (!(_mediaTypeMask & ~(1 << i))) { - return false; - } - } - return true; -} - -bool OverviewWidget::contentOverlapped(const QRect &globalRect) { - return _mediaType->overlaps(globalRect); -} - -void OverviewWidget::updateTopBarSelection() { - auto selectedState = _inner->getSelectionState(); - _inner->setSelectMode(selectedState.count > 0); - if (App::main()) { - _topBar->showSelected(selectedState); - _topBar->update(); - } - if (App::wnd() && !Ui::isLayerShown()) { - _inner->activate(); - } - update(); -} - -int32 OverviewWidget::lastWidth() const { - return width(); -} - -int32 OverviewWidget::lastScrollTop() const { - return _scroll->scrollTop(); -} - -bool OverviewWidget::wheelEventFromFloatPlayer(QEvent *e) { - return _scroll->viewportEvent(e); -} - -QRect OverviewWidget::rectForFloatPlayer() const { - return mapToGlobal(_scroll->geometry()); -} - -int32 OverviewWidget::countBestScroll() const { - if (type() == OverviewMusicFiles) { - auto state = Media::Player::mixer()->currentState(AudioMsgId::Type::Song); - if (state.id) { - int32 top = _inner->itemTop(state.id.contextId()); - if (top >= 0) { - return snap(top - int(_scroll->height() - (st::msgPadding.top() + st::mediaThumbSize + st::msgPadding.bottom())) / 2, 0, _scroll->scrollTopMax()); - } - } - } else if (type() == OverviewLinks || type() == OverviewFiles) { - return 0; - } - return _scroll->scrollTopMax(); -} - -void OverviewWidget::fastShow(bool back, int32 lastScrollTop) { - resizeEvent(0); - _scrollSetAfterShow = (lastScrollTop < 0 ? countBestScroll() : lastScrollTop); - show(); - _inner->activate(); - doneShow(); -} - -void OverviewWidget::setLastScrollTop(int lastScrollTop) { - resizeEvent(0); - _scroll->scrollToY(lastScrollTop < 0 ? countBestScroll() : lastScrollTop); -} - -void OverviewWidget::showAnimated(Window::SlideDirection direction, const Window::SectionSlideParams ¶ms) { - _showDirection = direction; - resizeEvent(0); - - _a_show.finish(); - - _cacheUnder = params.oldContentCache; - show(); - _topBar->updateControlsVisibility(); - _topShadow->setVisible(params.withTopBarShadow ? false : true); - _cacheOver = App::main()->grabForShowAnimation(params); - _topShadow->setVisible(params.withTopBarShadow ? true : false); - - _topBar->hide(); - _scrollSetAfterShow = _scroll->scrollTop(); - _scroll->hide(); - - if (_showDirection == Window::SlideDirection::FromLeft) { - std::swap(_cacheUnder, _cacheOver); - } - _a_show.start([this] { animationCallback(); }, 0., 1., st::slideDuration, Window::SlideAnimation::transition()); - - _backAnimationButton.create(this); - _backAnimationButton->setClickedCallback([this] { topBarClick(); }); - _backAnimationButton->setGeometry(_topBar->geometry()); - _backAnimationButton->show(); - - activate(); -} - -void OverviewWidget::animationCallback() { - update(); - if (!_a_show.animating()) { - _topShadow->show(); - _cacheUnder = _cacheOver = QPixmap(); - doneShow(); - } -} - -void OverviewWidget::doneShow() { - _topBar->animationFinished(); - _backAnimationButton.destroy(); - _topBar->show(); - _scroll->show(); - _scroll->scrollToY(_scrollSetAfterShow); - activate(); - onScroll(); -} - -void OverviewWidget::mediaOverviewUpdated(const Notify::PeerUpdate &update) { - if ((peer() == update.peer || migratePeer() == update.peer) && (update.mediaTypesMask & (1 << type()))) { - _inner->mediaOverviewUpdated(); - onScroll(); - updateTopBarSelection(); - } - int32 mask = 0; - History *h = update.peer ? App::historyLoaded(update.peer->migrateTo() ? update.peer->migrateTo() : update.peer) : nullptr; - History *m = (update.peer && update.peer->migrateFrom()) ? App::historyLoaded(update.peer->migrateFrom()->id) : 0; - if (h) { - for (int32 i = 0; i < OverviewCount; ++i) { - if (!h->overview(i).isEmpty() || h->overviewCount(i) > 0 || i == type()) { - mask |= (1 << i); - } else if (m && (!m->overview(i).isEmpty() || m->overviewCount(i) > 0)) { - mask |= (1 << i); - } - } - } - if (mask != _mediaTypeMask) { - auto typeLabel = [](MediaOverviewType type) -> QString { - switch (type) { - case OverviewPhotos: return lang(lng_media_type_photos); - case OverviewVideos: return lang(lng_media_type_videos); - case OverviewMusicFiles: return lang(lng_media_type_songs); - case OverviewFiles: return lang(lng_media_type_files); - case OverviewVoiceFiles: return lang(lng_media_type_audios); - case OverviewLinks: return lang(lng_media_type_links); - } - return QString(); - }; - _mediaType->clearActions(); - for (auto i = 0; i != OverviewCount; ++i) { - if (mask & (1 << i)) { - auto type = static_cast(i); - auto label = typeLabel(type); - if (!label.isEmpty()) { - _mediaType->addAction(label, [this, type] { - switchType(type); - _mediaType->hideAnimated(); - }); - } - } - } - _mediaTypeMask = mask; - _mediaType->move(width() - _mediaType->width(), st::topBarHeight); - updateTopBarSelection(); - } -} - -void OverviewWidget::grapWithoutTopBarShadow() { - grabStart(); - _topShadow->hide(); -} - -void OverviewWidget::grabFinish() { - _inGrab = false; - resizeEvent(0); - _topShadow->show(); -} - -SelectedItemSet OverviewWidget::getSelectedItems() const { - return _inner->getSelectedItems(); -} - -OverviewWidget::~OverviewWidget() { - onClearSelected(); - updateTopBarSelection(); -} - -void OverviewWidget::activate() { - if (_scroll->isHidden()) { - setFocus(); - } else { - _inner->activate(); - } -} - -QPoint OverviewWidget::clampMousePosition(QPoint point) { - if (point.x() < 0) { - point.setX(0); - } else if (point.x() >= _scroll->width()) { - point.setX(_scroll->width() - 1); - } - if (point.y() < _scroll->scrollTop()) { - point.setY(_scroll->scrollTop()); - } else if (point.y() >= _scroll->scrollTop() + _scroll->height()) { - point.setY(_scroll->scrollTop() + _scroll->height() - 1); - } - return point; -} - -void OverviewWidget::onScrollTimer() { - int32 d = (_scrollDelta > 0) ? qMin(_scrollDelta * 3 / 20 + 1, int32(MaxScrollSpeed)) : qMax(_scrollDelta * 3 / 20 - 1, -int32(MaxScrollSpeed)); - _scroll->scrollToY(_scroll->scrollTop() + d); -} - -void OverviewWidget::checkSelectingScroll(QPoint point) { - if (point.y() < _scroll->scrollTop()) { - _scrollDelta = point.y() - _scroll->scrollTop(); - } else if (point.y() >= _scroll->scrollTop() + _scroll->height()) { - _scrollDelta = point.y() - _scroll->scrollTop() - _scroll->height() + 1; - } else { - _scrollDelta = 0; - } - if (_scrollDelta) { - _scrollTimer.start(15); - } else { - _scrollTimer.stop(); - } -} - -void OverviewWidget::noSelectingScroll() { - _scrollTimer.stop(); -} - -bool OverviewWidget::touchScroll(const QPoint &delta) { - int32 scTop = _scroll->scrollTop(), scMax = _scroll->scrollTopMax(), scNew = snap(scTop - delta.y(), 0, scMax); - if (scNew == scTop) return false; - - _scroll->scrollToY(scNew); - return true; -} - -void OverviewWidget::onForwardSelected() { - App::main()->showForwardLayer(getSelectedItems()); -} - -void OverviewWidget::confirmDeleteContextItem() { - auto item = App::contextItem(); - if (!item) return; - - if (auto message = item->toHistoryMessage()) { - if (message->uploading()) { - App::main()->cancelUploadLayer(); - return; - } - } - App::main()->deleteLayer(); -} - -void OverviewWidget::confirmDeleteSelectedItems() { - auto selected = _inner->getSelectedItems(); - if (selected.isEmpty()) return; - - App::main()->deleteLayer(selected.size()); -} - -void OverviewWidget::deleteContextItem(bool forEveryone) { - Ui::hideLayer(); - - auto item = App::contextItem(); - if (!item) { - return; - } - - auto toDelete = QVector(1, MTP_int(item->id)); - auto history = item->history(); - auto wasOnServer = (item->id > 0); - auto wasLast = (history->lastMsg == item); - item->destroy(); - - if (!wasOnServer && wasLast && !history->lastMsg) { - App::main()->checkPeerHistory(history->peer); - } - - if (wasOnServer) { - App::main()->deleteMessages(history->peer, toDelete, forEveryone); - } -} - -void OverviewWidget::deleteSelectedItems(bool forEveryone) { - Ui::hideLayer(); - - auto selected = _inner->getSelectedItems(); - if (selected.isEmpty()) return; - - QMap> idsByPeer; - for_const (auto item, selected) { - if (item->id > 0) { - idsByPeer[item->history()->peer].push_back(MTP_int(item->id)); - } - } - - onClearSelected(); - for_const (auto item, selected) { - item->destroy(); - } - - for (auto i = idsByPeer.cbegin(), e = idsByPeer.cend(); i != e; ++i) { - App::main()->deleteMessages(i.key(), i.value(), forEveryone); - } -} - -void OverviewWidget::onClearSelected() { - _inner->clearSelectedItems(); -} diff --git a/Telegram/SourceFiles/overviewwidget.h b/Telegram/SourceFiles/overviewwidget.h deleted file mode 100644 index 23ed01b06..000000000 --- a/Telegram/SourceFiles/overviewwidget.h +++ /dev/null @@ -1,399 +0,0 @@ -/* -This file is part of Telegram Desktop, -the official desktop version of Telegram messaging app, see https://telegram.org - -Telegram Desktop is free software: you can redistribute it and/or modify -it under the terms of the GNU General Public License as published by -the Free Software Foundation, either version 3 of the License, or -(at your option) any later version. - -It is distributed in the hope that it will be useful, -but WITHOUT ANY WARRANTY; without even the implied warranty of -MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -GNU General Public License for more details. - -In addition, as a special exception, the copyright holders give permission -to link the code of portions of this program with the OpenSSL library. - -Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE -Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org -*/ -#pragma once - -#include "window/section_widget.h" -#include "window/top_bar_widget.h" -#include "ui/widgets/tooltip.h" -#include "ui/widgets/scroll_area.h" - -namespace Overview { -namespace Layout { -class AbstractItem; -class ItemBase; -class Date; -} // namespace Layout -} // namespace Overview - -namespace Ui { -class AbstractButton; -class PlainShadow; -class PopupMenu; -class IconButton; -class FlatInput; -class CrossButton; -class DropdownMenu; -} // namespace Ui - -namespace Notify { -struct PeerUpdate; -} // namespace Notify - -namespace Window { -class Controller; -class TopBarWidget; -} // namespace Window - -class OverviewWidget; -class OverviewInner : public TWidget, public Ui::AbstractTooltipShower, public RPCSender, private base::Subscriber { - Q_OBJECT - -public: - OverviewInner(OverviewWidget *overview, Ui::ScrollArea *scroll, PeerData *peer, MediaOverviewType type); - - void activate(); - - void clear(); - int32 itemTop(const FullMsgId &msgId) const; - - bool preloadLocal(); - void preloadMore(); - - void showContextMenu(QContextMenuEvent *e, bool showFromTouch = false); - - void dragActionStart(const QPoint &screenPos, Qt::MouseButton button = Qt::LeftButton); - void dragActionUpdate(const QPoint &screenPos); - void dragActionFinish(const QPoint &screenPos, Qt::MouseButton button = Qt::LeftButton); - void dragActionCancel(); - - void touchScrollUpdated(const QPoint &screenPos); - QPoint mapMouseToItem(QPoint p, MsgId itemId, int32 itemIndex); - - int32 resizeToWidth(int32 nwidth, int32 scrollTop, int32 minHeight, bool force = false); // returns new scroll top - void dropResizeIndex(); - - PeerData *peer() const; - PeerData *migratePeer() const; - MediaOverviewType type() const; - void switchType(MediaOverviewType type); - - void setSelectMode(bool enabled); - - void mediaOverviewUpdated(); - void repaintItem(const HistoryItem *msg); - - Window::TopBarWidget::SelectedState getSelectionState() const; - void clearSelectedItems(bool onlyTextSelection = false); - SelectedItemSet getSelectedItems() const; - - // AbstractTooltipShower interface - QString tooltipText() const override; - QPoint tooltipPos() const override; - - ~OverviewInner(); - -protected: - bool event(QEvent *e) override; - void touchEvent(QTouchEvent *e); - void paintEvent(QPaintEvent *e) override; - void mouseMoveEvent(QMouseEvent *e) override; - void mousePressEvent(QMouseEvent *e) override; - void mouseReleaseEvent(QMouseEvent *e) override; - void keyPressEvent(QKeyEvent *e) override; - void enterEventHook(QEvent *e) override; - void leaveEventHook(QEvent *e) override; - void resizeEvent(QResizeEvent *e) override; - -public slots: - void onUpdateSelected(); - - void copyContextUrl(); - void cancelContextDownload(); - void showContextInFolder(); - - void goToMessage(); - void forwardMessage(); - void selectMessage(); - - void onSearchUpdate(); - void onCancel(); - bool onCancelSearch(); - - void onMenuDestroy(QObject *obj); - void onTouchSelect(); - void onTouchScrollTimer(); - - bool onSearchMessages(bool searchCache = false); - void onNeedSearchMessages(); - -private: - void saveDocumentToFile(DocumentData *document); - void invalidateCache(); - void resizeItems(); - void resizeAndRepositionItems(); - void performDrag(); - - void itemRemoved(HistoryItem *item); - MsgId complexMsgId(const HistoryItem *item) const; - void changingMsgId(HistoryItem *row, MsgId newId); - - bool itemMigrated(MsgId msgId) const; - ChannelId itemChannel(MsgId msgId) const; - MsgId itemMsgId(MsgId msgId) const; - int32 migratedIndexSkip() const; - - void fixItemIndex(int32 ¤t, MsgId msgId) const; - bool itemHasPoint(MsgId msgId, int32 index, int32 x, int32 y) const; - int32 itemHeight(MsgId msgId, int32 index) const; - void moveToNextItem(MsgId &msgId, int32 &index, MsgId upTo, int32 delta) const; - - void updateDragSelection(MsgId dragSelFrom, int32 dragSelFromIndex, MsgId dragSelTo, int32 dragSelToIndex, bool dragSelecting); - - void repaintItem(MsgId itemId, int32 itemIndex); - - void touchResetSpeed(); - void touchUpdateSpeed(); - void touchDeaccelerate(int32 elapsed); - - void applyDragSelection(); - void addSelectionRange(int32 selFrom, int32 selTo, History *history); - - void recountMargins(); - int countHeight(); - - OverviewWidget *_overview; - Ui::ScrollArea *_scroll; - int _resizeIndex = -1; - int _resizeSkip = 0; - - PeerData *_peer; - MediaOverviewType _type; - bool _reversed; - History *_history; - History *_migrated; - ChannelId _channel; - - bool _selMode = false; - TextSelection itemSelectedValue(int32 index) const; - - int _rowsLeft = 0; - int _rowWidth = 0; - - typedef QVector Items; - Items _items; - typedef QMap LayoutItems; - LayoutItems _layoutItems; - typedef QMap LayoutDates; - LayoutDates _layoutDates; - Overview::Layout::ItemBase *layoutPrepare(HistoryItem *item); - Overview::Layout::AbstractItem *layoutPrepare(const QDate &date, bool month); - int32 setLayoutItem(int32 index, Overview::Layout::AbstractItem *item, int32 top); - - object_ptr _search; - object_ptr _cancelSearch; - QVector _results; - int32 _itemsToBeLoaded; - - // photos - int32 _photosInRow = 1; - - QTimer _searchTimer; - QString _searchQuery; - bool _inSearch = false; - bool _searchFull = false; - bool _searchFullMigrated = false; - mtpRequestId _searchRequest = 0; - QList _searchResults; - MsgId _lastSearchId = 0; - MsgId _lastSearchMigratedId = 0; - int _searchedCount = 0; - enum SearchRequestType { - SearchFromStart, - SearchFromOffset, - SearchMigratedFromStart, - SearchMigratedFromOffset - }; - void searchReceived(SearchRequestType type, const MTPmessages_Messages &result, mtpRequestId req); - bool searchFailed(SearchRequestType type, const RPCError &error, mtpRequestId req); - - typedef QMap SearchCache; - SearchCache _searchCache; - - typedef QMap SearchQueries; - SearchQueries _searchQueries; - - int _width = 0; - int _height = 0; - int _minHeight = 0; - int _marginTop = 0; - int _marginBottom = 0; - - QTimer _linkTipTimer; - - // selection support, like in HistoryWidget - style::cursor _cursor = style::cur_default; - HistoryCursorState _cursorState = HistoryDefaultCursorState; - using SelectedItems = QMap; - SelectedItems _selected; - enum DragAction { - NoDrag = 0x00, - PrepareDrag = 0x01, - Dragging = 0x02, - PrepareSelect = 0x03, - Selecting = 0x04, - }; - DragAction _dragAction = NoDrag; - QPoint _dragStartPos, _dragPos; - MsgId _dragItem = 0; - MsgId _selectedMsgId = 0; - int _dragItemIndex = -1; - MsgId _mousedItem = 0; - int _mousedItemIndex = -1; - uint16 _dragSymbol; - bool _dragWasInactive = false; - - ClickHandlerPtr _contextMenuLnk; - - MsgId _dragSelFrom = 0; - MsgId _dragSelTo = 0; - int _dragSelFromIndex = -1; - int _dragSelToIndex = -1; - bool _dragSelecting = false; - - bool _touchScroll = false; - bool _touchSelect = false; - bool _touchInProgress = false; - QPoint _touchStart, _touchPrevPos, _touchPos; - QTimer _touchSelectTimer; - - Ui::TouchScrollState _touchScrollState = Ui::TouchScrollState::Manual; - bool _touchPrevPosValid = false; - bool _touchWaitingAcceleration = false; - QPoint _touchSpeed; - TimeMs _touchSpeedTime = 0; - TimeMs _touchAccelerationTime = 0; - TimeMs _touchTime = 0; - QTimer _touchScrollTimer; - - Ui::PopupMenu *_menu = nullptr; -}; - -class OverviewWidget : public Window::AbstractSectionWidget, public RPCSender { - Q_OBJECT - -public: - OverviewWidget(QWidget *parent, not_null controller, PeerData *peer, MediaOverviewType type); - - void clear(); - - void scrollBy(int32 add); - void scrollReset(); - - bool paintTopBar(Painter &p, int decreaseWidth); - - PeerData *peer() const; - PeerData *migratePeer() const; - MediaOverviewType type() const; - void switchType(MediaOverviewType type); - bool showMediaTypeSwitch() const; - void updateTopBarSelection(); - bool contentOverlapped(const QRect &globalRect); - - int32 lastWidth() const; - int32 lastScrollTop() const; - int32 countBestScroll() const; - - void fastShow(bool back = false, int32 lastScrollTop = -1); - bool hasTopBarShadow() const { - return true; - } - void setLastScrollTop(int lastScrollTop); - void showAnimated(Window::SlideDirection direction, const Window::SectionSlideParams ¶ms); - - void doneShow(); - - void mediaOverviewUpdated(const Notify::PeerUpdate &update); - void itemRemoved(HistoryItem *item); - - QPoint clampMousePosition(QPoint point); - - void checkSelectingScroll(QPoint point); - void noSelectingScroll(); - - bool touchScroll(const QPoint &delta); - - SelectedItemSet getSelectedItems() const; - - void grabStart() override { - _inGrab = true; - resizeEvent(0); - } - void grapWithoutTopBarShadow(); - void grabFinish() override; - void rpcClear() override { - _inner->rpcClear(); - RPCSender::rpcClear(); - } - - void confirmDeleteContextItem(); - void confirmDeleteSelectedItems(); - void deleteContextItem(bool forEveryone); - void deleteSelectedItems(bool forEveryone); - - // Float player interface. - bool wheelEventFromFloatPlayer(QEvent *e) override; - QRect rectForFloatPlayer() const override; - - ~OverviewWidget(); - -protected: - void resizeEvent(QResizeEvent *e) override; - void paintEvent(QPaintEvent *e) override; - void contextMenuEvent(QContextMenuEvent *e) override; - -public slots: - void activate(); - void onScroll(); - - void onScrollTimer(); - - void onForwardSelected(); - void onClearSelected(); - -private: - void topBarClick(); - void animationCallback(); - - object_ptr _backAnimationButton = { nullptr }; - object_ptr _topBar; - object_ptr _scroll; - QPointer _inner; - bool _noDropResizeIndex = false; - - object_ptr _mediaType; - int32 _mediaTypeMask = 0; - - QString _header; - - Animation _a_show; - Window::SlideDirection _showDirection; - QPixmap _cacheUnder, _cacheOver; - - int32 _scrollSetAfterShow = 0; - - QTimer _scrollTimer; - int32 _scrollDelta = 0; - - object_ptr _topShadow; - bool _inGrab = false; - -}; - diff --git a/Telegram/SourceFiles/profile/profile_back_button.cpp b/Telegram/SourceFiles/profile/profile_back_button.cpp index 9293a388d..da09d8b78 100644 --- a/Telegram/SourceFiles/profile/profile_back_button.cpp +++ b/Telegram/SourceFiles/profile/profile_back_button.cpp @@ -20,7 +20,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org */ #include "profile/profile_back_button.h" -#include "window/top_bar_widget.h" +#include "history/history_top_bar_widget.h" #include "styles/style_widgets.h" #include "styles/style_window.h" #include "styles/style_profile.h" @@ -54,7 +54,7 @@ void BackButton::paintEvent(QPaintEvent *e) { p.setPen(st::topBarButton.textFg); p.drawTextLeft(st::topBarArrowPadding.left(), st::topBarButton.padding.top() + st::topBarButton.textTop, width(), _text); - Window::TopBarWidget::paintUnreadCounter(p, width()); + HistoryTopBarWidget::paintUnreadCounter(p, width()); } void BackButton::onStateChanged(State was, StateChangeSource source) { diff --git a/Telegram/SourceFiles/profile/profile_block_shared_media.cpp b/Telegram/SourceFiles/profile/profile_block_shared_media.cpp index 5976db50d..41249f57c 100644 --- a/Telegram/SourceFiles/profile/profile_block_shared_media.cpp +++ b/Telegram/SourceFiles/profile/profile_block_shared_media.cpp @@ -145,7 +145,7 @@ void SharedMediaWidget::onMediaChosen() { for (int i = 0; i < OverviewCount; ++i) { auto button = _mediaButtons[i]; if (button && button == sender()) { - App::main()->showMediaOverview(peer(), static_cast(i)); + // SharedMediaShowOverview(); return; } } diff --git a/Telegram/gyp/telegram_sources.txt b/Telegram/gyp/telegram_sources.txt index fd7312ea8..68ca49811 100644 --- a/Telegram/gyp/telegram_sources.txt +++ b/Telegram/gyp/telegram_sources.txt @@ -215,6 +215,8 @@ <(src_loc)/history/history_service.h <(src_loc)/history/history_service_layout.cpp <(src_loc)/history/history_service_layout.h +<(src_loc)/history/history_top_bar_widget.cpp +<(src_loc)/history/history_top_bar_widget.h <(src_loc)/history/history_widget.cpp <(src_loc)/history/history_widget.h <(src_loc)/info/info_content_widget.cpp @@ -632,8 +634,6 @@ <(src_loc)/window/section_memento.h <(src_loc)/window/section_widget.cpp <(src_loc)/window/section_widget.h -<(src_loc)/window/top_bar_widget.cpp -<(src_loc)/window/top_bar_widget.h <(src_loc)/window/window_controller.cpp <(src_loc)/window/window_controller.h <(src_loc)/window/window_main_menu.cpp @@ -682,8 +682,6 @@ <(src_loc)/messenger.h <(src_loc)/observer_peer.cpp <(src_loc)/observer_peer.h -<(src_loc)/overviewwidget.cpp -<(src_loc)/overviewwidget.h <(src_loc)/passcodewidget.cpp <(src_loc)/passcodewidget.h <(src_loc)/qt_static_plugins.cpp