416 lines
11 KiB
C++
416 lines
11 KiB
C++
/*
|
|
This file is part of Telegram Desktop,
|
|
the official desktop application for the Telegram messaging service.
|
|
|
|
For license and copyright information please follow this link:
|
|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|
*/
|
|
#pragma once
|
|
|
|
#include "dialogs/dialogs_key.h"
|
|
#include "data/data_messages.h"
|
|
#include "ui/effects/animations.h"
|
|
#include "ui/rp_widget.h"
|
|
#include "base/flags.h"
|
|
#include "base/object_ptr.h"
|
|
|
|
namespace MTP {
|
|
class Error;
|
|
} // namespace MTP
|
|
|
|
namespace Main {
|
|
class Session;
|
|
} // namespace Main
|
|
|
|
namespace Ui {
|
|
class IconButton;
|
|
class PopupMenu;
|
|
class FlatLabel;
|
|
} // namespace Ui
|
|
|
|
namespace Window {
|
|
class SessionController;
|
|
} // namespace Window
|
|
|
|
namespace Data {
|
|
class CloudImageView;
|
|
} // namespace Data
|
|
|
|
namespace Dialogs {
|
|
|
|
class Row;
|
|
class FakeRow;
|
|
class IndexedList;
|
|
|
|
struct ChosenRow {
|
|
Key key;
|
|
Data::MessagePosition message;
|
|
bool filteredRow = false;
|
|
};
|
|
|
|
enum class SearchRequestType {
|
|
FromStart,
|
|
FromOffset,
|
|
PeerFromStart,
|
|
PeerFromOffset,
|
|
MigratedFromStart,
|
|
MigratedFromOffset,
|
|
};
|
|
|
|
enum class WidgetState {
|
|
Default,
|
|
Filtered,
|
|
};
|
|
|
|
class InnerWidget final : public Ui::RpWidget {
|
|
Q_OBJECT
|
|
|
|
public:
|
|
InnerWidget(
|
|
QWidget *parent,
|
|
not_null<Window::SessionController*> controller);
|
|
|
|
bool searchReceived(
|
|
const QVector<MTPMessage> &result,
|
|
HistoryItem *inject,
|
|
SearchRequestType type,
|
|
int fullCount);
|
|
void peerSearchReceived(
|
|
const QString &query,
|
|
const QVector<MTPPeer> &my,
|
|
const QVector<MTPPeer> &result);
|
|
|
|
[[nodiscard]] FilterId filterId() const;
|
|
|
|
void clearSelection();
|
|
|
|
void changeOpenedFolder(Data::Folder *folder);
|
|
void selectSkip(int32 direction);
|
|
void selectSkipPage(int32 pixels, int32 direction);
|
|
|
|
void dragLeft();
|
|
|
|
void clearFilter();
|
|
void refresh(bool toTop = false);
|
|
void refreshEmptyLabel();
|
|
void resizeEmptyLabel();
|
|
|
|
bool chooseRow();
|
|
|
|
void scrollToEntry(const RowDescriptor &entry);
|
|
|
|
Data::Folder *shownFolder() const;
|
|
int32 lastSearchDate() const;
|
|
PeerData *lastSearchPeer() const;
|
|
MsgId lastSearchId() const;
|
|
MsgId lastSearchMigratedId() const;
|
|
|
|
WidgetState state() const;
|
|
bool waitingForSearch() const {
|
|
return _waitingForSearch;
|
|
}
|
|
bool hasFilteredResults() const;
|
|
|
|
void searchInChat(Key key, PeerData *from);
|
|
|
|
void applyFilterUpdate(QString newFilter, bool force = false);
|
|
void onHashtagFilterUpdate(QStringView newFilter);
|
|
|
|
PeerData *updateFromParentDrag(QPoint globalPosition);
|
|
|
|
void setLoadMoreCallback(Fn<void()> callback);
|
|
[[nodiscard]] rpl::producer<> listBottomReached() const;
|
|
[[nodiscard]] rpl::producer<> cancelSearchFromUserRequests() const;
|
|
[[nodiscard]] rpl::producer<ChosenRow> chosenRow() const;
|
|
[[nodiscard]] rpl::producer<> updated() const;
|
|
|
|
~InnerWidget();
|
|
|
|
public Q_SLOTS:
|
|
void onParentGeometryChanged();
|
|
|
|
Q_SIGNALS:
|
|
void draggingScrollDelta(int delta);
|
|
void mustScrollTo(int scrollToTop, int scrollToBottom);
|
|
void dialogMoved(int movedFrom, int movedTo);
|
|
void searchMessages();
|
|
void cancelSearchInChat();
|
|
void completeHashtag(QString tag);
|
|
void refreshHashtags();
|
|
|
|
protected:
|
|
void visibleTopBottomUpdated(
|
|
int visibleTop,
|
|
int visibleBottom) override;
|
|
|
|
void paintEvent(QPaintEvent *e) override;
|
|
void mouseMoveEvent(QMouseEvent *e) override;
|
|
void mousePressEvent(QMouseEvent *e) override;
|
|
void mouseReleaseEvent(QMouseEvent *e) override;
|
|
void resizeEvent(QResizeEvent *e) override;
|
|
void enterEventHook(QEnterEvent *e) override;
|
|
void leaveEventHook(QEvent *e) override;
|
|
void contextMenuEvent(QContextMenuEvent *e) override;
|
|
|
|
private:
|
|
struct CollapsedRow;
|
|
struct HashtagResult;
|
|
struct PeerSearchResult;
|
|
|
|
enum class JumpSkip {
|
|
PreviousOrBegin,
|
|
NextOrEnd,
|
|
PreviousOrOriginal,
|
|
NextOrOriginal,
|
|
};
|
|
|
|
enum class EmptyState : uchar {
|
|
None,
|
|
Loading,
|
|
NoContacts,
|
|
EmptyFolder,
|
|
};
|
|
|
|
Main::Session &session() const;
|
|
|
|
void dialogRowReplaced(Row *oldRow, Row *newRow);
|
|
|
|
void editOpenedFilter();
|
|
void repaintCollapsedFolderRow(not_null<Data::Folder*> folder);
|
|
void refreshWithCollapsedRows(bool toTop = false);
|
|
bool needCollapsedRowsRefresh() const;
|
|
bool chooseCollapsedRow();
|
|
void switchToFilter(FilterId filterId);
|
|
bool chooseHashtag();
|
|
ChosenRow computeChosenRow() const;
|
|
bool isSearchResultActive(
|
|
not_null<FakeRow*> result,
|
|
const RowDescriptor &entry) const;
|
|
|
|
void repaintDialogRow(FilterId filterId, not_null<Row*> row);
|
|
void repaintDialogRow(RowDescriptor row);
|
|
void refreshDialogRow(RowDescriptor row);
|
|
|
|
void clearMouseSelection(bool clearSelection = false);
|
|
void mousePressReleased(QPoint globalPosition, Qt::MouseButton button);
|
|
void clearIrrelevantState();
|
|
void selectByMouse(QPoint globalPosition);
|
|
void loadPeerPhotos();
|
|
void setCollapsedPressed(int pressed);
|
|
void setPressed(Row *pressed);
|
|
void setHashtagPressed(int pressed);
|
|
void setFilteredPressed(int pressed);
|
|
void setPeerSearchPressed(int pressed);
|
|
void setSearchedPressed(int pressed);
|
|
bool isPressed() const {
|
|
return (_collapsedPressed >= 0)
|
|
|| _pressed
|
|
|| (_hashtagPressed >= 0)
|
|
|| (_filteredPressed >= 0)
|
|
|| (_peerSearchPressed >= 0)
|
|
|| (_searchedPressed >= 0);
|
|
}
|
|
bool isSelected() const {
|
|
return (_collapsedSelected >= 0)
|
|
|| _selected
|
|
|| (_hashtagSelected >= 0)
|
|
|| (_filteredSelected >= 0)
|
|
|| (_peerSearchSelected >= 0)
|
|
|| (_searchedSelected >= 0);
|
|
}
|
|
bool uniqueSearchResults() const;
|
|
bool hasHistoryInResults(not_null<History*> history) const;
|
|
|
|
int defaultRowTop(not_null<Row*> row) const;
|
|
void setupOnlineStatusCheck();
|
|
void userOnlineUpdated(not_null<PeerData*> peer);
|
|
void groupHasCallUpdated(not_null<PeerData*> peer);
|
|
|
|
void updateRowCornerStatusShown(
|
|
not_null<History*> history,
|
|
bool shown);
|
|
void updateDialogRowCornerStatus(not_null<History*> history);
|
|
|
|
void setupShortcuts();
|
|
RowDescriptor computeJump(
|
|
const RowDescriptor &to,
|
|
JumpSkip skip);
|
|
bool jumpToDialogRow(RowDescriptor to);
|
|
|
|
RowDescriptor chatListEntryBefore(const RowDescriptor &which) const;
|
|
RowDescriptor chatListEntryAfter(const RowDescriptor &which) const;
|
|
RowDescriptor chatListEntryFirst() const;
|
|
RowDescriptor chatListEntryLast() const;
|
|
|
|
void itemRemoved(not_null<const HistoryItem*> item);
|
|
enum class UpdateRowSection {
|
|
Default = (1 << 0),
|
|
Filtered = (1 << 1),
|
|
PeerSearch = (1 << 2),
|
|
MessageSearch = (1 << 3),
|
|
All = Default | Filtered | PeerSearch | MessageSearch,
|
|
};
|
|
using UpdateRowSections = base::flags<UpdateRowSection>;
|
|
friend inline constexpr auto is_flag_type(UpdateRowSection) { return true; };
|
|
|
|
void updateSearchResult(not_null<PeerData*> peer);
|
|
void updateDialogRow(
|
|
RowDescriptor row,
|
|
QRect updateRect = QRect(),
|
|
UpdateRowSections sections = UpdateRowSection::All);
|
|
void fillSupportSearchMenu(not_null<Ui::PopupMenu*> menu);
|
|
void fillArchiveSearchMenu(not_null<Ui::PopupMenu*> menu);
|
|
|
|
int dialogsOffset() const;
|
|
int fixedOnTopCount() const;
|
|
int pinnedOffset() const;
|
|
int filteredOffset() const;
|
|
int peerSearchOffset() const;
|
|
int searchedOffset() const;
|
|
int searchInChatSkip() const;
|
|
|
|
void paintCollapsedRows(
|
|
Painter &p,
|
|
QRect clip) const;
|
|
void paintCollapsedRow(
|
|
Painter &p,
|
|
not_null<const CollapsedRow*> row,
|
|
bool selected) const;
|
|
void paintPeerSearchResult(
|
|
Painter &p,
|
|
not_null<const PeerSearchResult*> result,
|
|
int fullWidth,
|
|
bool active,
|
|
bool selected) const;
|
|
void paintSearchInChat(Painter &p) const;
|
|
void paintSearchInPeer(
|
|
Painter &p,
|
|
not_null<PeerData*> peer,
|
|
std::shared_ptr<Data::CloudImageView> &userpic,
|
|
int top,
|
|
const Ui::Text::String &text) const;
|
|
void paintSearchInSaved(
|
|
Painter &p,
|
|
int top,
|
|
const Ui::Text::String &text) const;
|
|
void paintSearchInReplies(
|
|
Painter &p,
|
|
int top,
|
|
const Ui::Text::String &text) const;
|
|
template <typename PaintUserpic>
|
|
void paintSearchInFilter(
|
|
Painter &p,
|
|
PaintUserpic paintUserpic,
|
|
int top,
|
|
const style::icon *icon,
|
|
const Ui::Text::String &text) const;
|
|
void refreshSearchInChatLabel();
|
|
|
|
void clearSearchResults(bool clearPeerSearchResults = true);
|
|
void updateSelectedRow(Key key = Key());
|
|
|
|
not_null<IndexedList*> shownDialogs() const;
|
|
|
|
void checkReorderPinnedStart(QPoint localPosition);
|
|
int updateReorderIndexGetCount();
|
|
bool updateReorderPinned(QPoint localPosition);
|
|
void finishReorderPinned();
|
|
void stopReorderPinned();
|
|
int countPinnedIndex(Row *ofRow);
|
|
void savePinnedOrder();
|
|
bool pinnedShiftAnimationCallback(crl::time now);
|
|
void handleChatListEntryRefreshes();
|
|
|
|
not_null<Window::SessionController*> _controller;
|
|
|
|
FilterId _filterId = 0;
|
|
bool _mouseSelection = false;
|
|
std::optional<QPoint> _lastMousePosition;
|
|
Qt::MouseButton _pressButton = Qt::LeftButton;
|
|
|
|
Data::Folder *_openedFolder = nullptr;
|
|
|
|
std::vector<std::unique_ptr<CollapsedRow>> _collapsedRows;
|
|
int _collapsedSelected = -1;
|
|
int _collapsedPressed = -1;
|
|
int _skipTopDialogs = 0;
|
|
Row *_selected = nullptr;
|
|
Row *_pressed = nullptr;
|
|
|
|
Row *_dragging = nullptr;
|
|
int _draggingIndex = -1;
|
|
int _aboveIndex = -1;
|
|
QPoint _dragStart;
|
|
struct PinnedRow {
|
|
anim::value yadd;
|
|
crl::time animStartTime = 0;
|
|
};
|
|
std::vector<PinnedRow> _pinnedRows;
|
|
Ui::Animations::Basic _pinnedShiftAnimation;
|
|
base::flat_set<Key> _pinnedOnDragStart;
|
|
|
|
// Remember the last currently dragged row top shift for updating area.
|
|
int _aboveTopShift = -1;
|
|
|
|
int _visibleTop = 0;
|
|
int _visibleBottom = 0;
|
|
QString _filter, _hashtagFilter;
|
|
|
|
std::vector<std::unique_ptr<HashtagResult>> _hashtagResults;
|
|
int _hashtagSelected = -1;
|
|
int _hashtagPressed = -1;
|
|
bool _hashtagDeleteSelected = false;
|
|
bool _hashtagDeletePressed = false;
|
|
|
|
std::vector<not_null<Row*>> _filterResults;
|
|
base::flat_map<
|
|
not_null<PeerData*>,
|
|
std::unique_ptr<Row>> _filterResultsGlobal;
|
|
int _filteredSelected = -1;
|
|
int _filteredPressed = -1;
|
|
|
|
bool _waitingForSearch = false;
|
|
EmptyState _emptyState = EmptyState::None;
|
|
|
|
QString _peerSearchQuery;
|
|
std::vector<std::unique_ptr<PeerSearchResult>> _peerSearchResults;
|
|
int _peerSearchSelected = -1;
|
|
int _peerSearchPressed = -1;
|
|
|
|
std::vector<std::unique_ptr<FakeRow>> _searchResults;
|
|
int _searchedCount = 0;
|
|
int _searchedMigratedCount = 0;
|
|
int _searchedSelected = -1;
|
|
int _searchedPressed = -1;
|
|
|
|
int _lastSearchDate = 0;
|
|
PeerData *_lastSearchPeer = nullptr;
|
|
MsgId _lastSearchId = 0;
|
|
MsgId _lastSearchMigratedId = 0;
|
|
|
|
WidgetState _state = WidgetState::Default;
|
|
|
|
object_ptr<Ui::FlatLabel> _empty = { nullptr };
|
|
object_ptr<Ui::IconButton> _cancelSearchInChat;
|
|
object_ptr<Ui::IconButton> _cancelSearchFromUser;
|
|
|
|
Key _searchInChat;
|
|
History *_searchInMigrated = nullptr;
|
|
PeerData *_searchFromPeer = nullptr;
|
|
mutable std::shared_ptr<Data::CloudImageView> _searchInChatUserpic;
|
|
mutable std::shared_ptr<Data::CloudImageView> _searchFromUserUserpic;
|
|
Ui::Text::String _searchInChatText;
|
|
Ui::Text::String _searchFromUserText;
|
|
RowDescriptor _menuRow;
|
|
|
|
Fn<void()> _loadMoreCallback;
|
|
rpl::event_stream<> _listBottomReached;
|
|
rpl::event_stream<ChosenRow> _chosenRow;
|
|
rpl::event_stream<> _updated;
|
|
|
|
base::unique_qptr<Ui::PopupMenu> _menu;
|
|
|
|
};
|
|
|
|
} // namespace Dialogs
|