mirror of
				https://github.com/kotatogram/kotatogram-desktop
				synced 2025-10-23 15:06:09 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			506 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			506 lines
		
	
	
		
			14 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/dragging_scroll_manager.h"
 | |
| #include "ui/effects/animations.h"
 | |
| #include "ui/rp_widget.h"
 | |
| #include "ui/userpic_view.h"
 | |
| #include "base/flags.h"
 | |
| #include "base/object_ptr.h"
 | |
| 
 | |
| namespace style {
 | |
| struct DialogRow;
 | |
| } // namespace style
 | |
| 
 | |
| namespace MTP {
 | |
| class Error;
 | |
| } // namespace MTP
 | |
| 
 | |
| namespace Main {
 | |
| class Session;
 | |
| } // namespace Main
 | |
| 
 | |
| namespace Ui {
 | |
| class IconButton;
 | |
| class PopupMenu;
 | |
| class FlatLabel;
 | |
| struct ScrollToRequest;
 | |
| } // namespace Ui
 | |
| 
 | |
| namespace Window {
 | |
| class SessionController;
 | |
| } // namespace Window
 | |
| 
 | |
| namespace Data {
 | |
| class Thread;
 | |
| class Folder;
 | |
| class Forum;
 | |
| } // namespace Data
 | |
| 
 | |
| namespace Dialogs::Ui {
 | |
| using namespace ::Ui;
 | |
| class VideoUserpic;
 | |
| struct PaintContext;
 | |
| struct TopicJumpCache;
 | |
| } // namespace Dialogs::Ui
 | |
| 
 | |
| namespace Dialogs {
 | |
| 
 | |
| class Row;
 | |
| class FakeRow;
 | |
| class IndexedList;
 | |
| 
 | |
| struct ChosenRow {
 | |
| 	Key key;
 | |
| 	Data::MessagePosition message;
 | |
| 	bool filteredRow = false;
 | |
| 	bool newWindow = false;
 | |
| };
 | |
| 
 | |
| enum class SearchRequestType {
 | |
| 	FromStart,
 | |
| 	FromOffset,
 | |
| 	PeerFromStart,
 | |
| 	PeerFromOffset,
 | |
| 	MigratedFromStart,
 | |
| 	MigratedFromOffset,
 | |
| };
 | |
| 
 | |
| enum class WidgetState {
 | |
| 	Default,
 | |
| 	Filtered,
 | |
| };
 | |
| 
 | |
| class InnerWidget final : public Ui::RpWidget {
 | |
| public:
 | |
| 	struct ChildListShown {
 | |
| 		PeerId peerId = 0;
 | |
| 		float64 shown = 0.;
 | |
| 	};
 | |
| 	InnerWidget(
 | |
| 		QWidget *parent,
 | |
| 		not_null<Window::SessionController*> controller,
 | |
| 		rpl::producer<ChildListShown> childListShown);
 | |
| 
 | |
| 	void searchReceived(
 | |
| 		std::vector<not_null<HistoryItem*>> 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 changeOpenedForum(Data::Forum *forum);
 | |
| 	void selectSkip(int32 direction);
 | |
| 	void selectSkipPage(int32 pixels, int32 direction);
 | |
| 
 | |
| 	void dragLeft();
 | |
| 	void setNarrowRatio(float64 narrowRatio);
 | |
| 
 | |
| 	void clearFilter();
 | |
| 	void refresh(bool toTop = false);
 | |
| 	void refreshEmptyLabel();
 | |
| 	void resizeEmptyLabel();
 | |
| 
 | |
| 	bool chooseRow(
 | |
| 		Qt::KeyboardModifiers modifiers = {},
 | |
| 		MsgId pressedTopicRootId = {});
 | |
| 
 | |
| 	void scrollToEntry(const RowDescriptor &entry);
 | |
| 
 | |
| 	[[nodiscard]] Data::Folder *shownFolder() const;
 | |
| 	[[nodiscard]] Data::Forum *shownForum() const;
 | |
| 
 | |
| 	[[nodiscard]] WidgetState state() const;
 | |
| 	[[nodiscard]] not_null<const style::DialogRow*> st() const {
 | |
| 		return _st;
 | |
| 	}
 | |
| 	[[nodiscard]] bool waitingForSearch() const {
 | |
| 		return _waitingForSearch;
 | |
| 	}
 | |
| 	[[nodiscard]] bool hasFilteredResults() const;
 | |
| 
 | |
| 	void searchInChat(Key key, PeerData *from);
 | |
| 
 | |
| 	void applyFilterUpdate(QString newFilter, bool force = false);
 | |
| 	void onHashtagFilterUpdate(QStringView newFilter);
 | |
| 	void appendToFiltered(Key key);
 | |
| 
 | |
| 	Data::Thread *updateFromParentDrag(QPoint globalPosition);
 | |
| 
 | |
| 	void setLoadMoreCallback(Fn<void()> callback);
 | |
| 	void setLoadMoreFilteredCallback(Fn<void()> callback);
 | |
| 	[[nodiscard]] rpl::producer<> listBottomReached() const;
 | |
| 	[[nodiscard]] rpl::producer<> cancelSearchFromUserRequests() const;
 | |
| 	[[nodiscard]] rpl::producer<ChosenRow> chosenRow() const;
 | |
| 	[[nodiscard]] rpl::producer<> updated() const;
 | |
| 
 | |
| 	[[nodiscard]] rpl::producer<int> scrollByDeltaRequests() const;
 | |
| 	[[nodiscard]] rpl::producer<Ui::ScrollToRequest> mustScrollTo() const;
 | |
| 	[[nodiscard]] rpl::producer<Ui::ScrollToRequest> dialogMoved() const;
 | |
| 	[[nodiscard]] rpl::producer<> searchMessages() const;
 | |
| 	[[nodiscard]] rpl::producer<> cancelSearchInChatRequests() const;
 | |
| 	[[nodiscard]] rpl::producer<QString> completeHashtagRequests() const;
 | |
| 	[[nodiscard]] rpl::producer<> refreshHashtagsRequests() const;
 | |
| 
 | |
| 	[[nodiscard]] RowDescriptor resolveChatNext(RowDescriptor from = {}) const;
 | |
| 	[[nodiscard]] RowDescriptor resolveChatPrevious(RowDescriptor from = {}) const;
 | |
| 
 | |
| 	~InnerWidget();
 | |
| 
 | |
| 	void parentGeometryChanged();
 | |
| 
 | |
| 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,
 | |
| 		EmptyForum,
 | |
| 	};
 | |
| 
 | |
| 	struct PinnedRow {
 | |
| 		anim::value yadd;
 | |
| 		crl::time animStartTime = 0;
 | |
| 	};
 | |
| 
 | |
| 	struct FilterResult {
 | |
| 		FilterResult(not_null<Row*> row) : row(row) {
 | |
| 		}
 | |
| 
 | |
| 		not_null<Row*> row;
 | |
| 		int top = 0;
 | |
| 
 | |
| 		[[nodiscard]] Key key() const;
 | |
| 		[[nodiscard]] int bottom() const;
 | |
| 	};
 | |
| 
 | |
| 	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);
 | |
| 	bool updateEntryHeight(not_null<Entry*> entry);
 | |
| 
 | |
| 	void clearMouseSelection(bool clearSelection = false);
 | |
| 	void mousePressReleased(
 | |
| 		QPoint globalPosition,
 | |
| 		Qt::MouseButton button,
 | |
| 		Qt::KeyboardModifiers modifiers);
 | |
| 	void clearIrrelevantState();
 | |
| 	void selectByMouse(QPoint globalPosition);
 | |
| 	void preloadRowsData();
 | |
| 	void scrollToItem(int top, int height);
 | |
| 	void scrollToDefaultSelected();
 | |
| 	void setCollapsedPressed(int pressed);
 | |
| 	void setPressed(Row *pressed, bool pressedTopicJump);
 | |
| 	void clearPressed();
 | |
| 	void setHashtagPressed(int pressed);
 | |
| 	void setFilteredPressed(int pressed, bool pressedTopicJump);
 | |
| 	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 updateRowCornerStatusShown(not_null<History*> history);
 | |
| 	void repaintDialogRowCornerStatus(not_null<History*> history);
 | |
| 
 | |
| 	void setupShortcuts();
 | |
| 	RowDescriptor computeJump(
 | |
| 		const RowDescriptor &to,
 | |
| 		JumpSkip skip) const;
 | |
| 	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);
 | |
| 
 | |
| 	void refreshShownList();
 | |
| 	[[nodiscard]] int skipTopHeight() const;
 | |
| 	[[nodiscard]] int dialogsOffset() const;
 | |
| 	[[nodiscard]] int shownHeight(int till = -1) const;
 | |
| 	[[nodiscard]] int fixedOnTopCount() const;
 | |
| 	[[nodiscard]] int pinnedOffset() const;
 | |
| 	[[nodiscard]] int filteredOffset() const;
 | |
| 	[[nodiscard]] int filteredIndex(int y) const;
 | |
| 	[[nodiscard]] int filteredHeight(int till = -1) const;
 | |
| 	[[nodiscard]] int peerSearchOffset() const;
 | |
| 	[[nodiscard]] int searchedOffset() const;
 | |
| 	[[nodiscard]] 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,
 | |
| 		const Ui::PaintContext &context);
 | |
| 	void paintSearchInChat(
 | |
| 		Painter &p,
 | |
| 		const Ui::PaintContext &context) const;
 | |
| 	void paintSearchInPeer(
 | |
| 		Painter &p,
 | |
| 		not_null<PeerData*> peer,
 | |
| 		Ui::PeerUserpicView &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;
 | |
| 	void paintSearchInTopic(
 | |
| 		Painter &p,
 | |
| 		const Ui::PaintContext &context,
 | |
| 		not_null<Data::ForumTopic*> topic,
 | |
| 		Ui::PeerUserpicView &userpic,
 | |
| 		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 repaintSearchResult(int index);
 | |
| 
 | |
| 	Ui::VideoUserpic *validateVideoUserpic(not_null<Row*> row);
 | |
| 	Ui::VideoUserpic *validateVideoUserpic(not_null<History*> history);
 | |
| 
 | |
| 	Row *shownRowByKey(Key key);
 | |
| 	void clearSearchResults(bool clearPeerSearchResults = true);
 | |
| 	void updateSelectedRow(Key key = Key());
 | |
| 	void trackSearchResultsHistory(not_null<History*> history);
 | |
| 	void trackSearchResultsForum(Data::Forum *forum);
 | |
| 
 | |
| 	[[nodiscard]] QBrush currentBg() const;
 | |
| 
 | |
| 	[[nodiscard]] const std::vector<Key> &pinnedChatsOrder() 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();
 | |
| 	void moveCancelSearchButtons();
 | |
| 
 | |
| 	void saveChatsFilterScrollState(FilterId filterId);
 | |
| 	void restoreChatsFilterScrollState(FilterId filterId);
 | |
| 
 | |
| 	const not_null<Window::SessionController*> _controller;
 | |
| 
 | |
| 	not_null<IndexedList*> _shownList;
 | |
| 	FilterId _filterId = 0;
 | |
| 	bool _mouseSelection = false;
 | |
| 	std::optional<QPoint> _lastMousePosition;
 | |
| 	Qt::MouseButton _pressButton = Qt::LeftButton;
 | |
| 
 | |
| 	Data::Folder *_openedFolder = nullptr;
 | |
| 	Data::Forum *_openedForum = nullptr;
 | |
| 	rpl::lifetime _openedForumLifetime;
 | |
| 
 | |
| 	std::vector<std::unique_ptr<CollapsedRow>> _collapsedRows;
 | |
| 	not_null<const style::DialogRow*> _st;
 | |
| 	mutable std::unique_ptr<Ui::TopicJumpCache> _topicJumpCache;
 | |
| 	int _collapsedSelected = -1;
 | |
| 	int _collapsedPressed = -1;
 | |
| 	bool _skipTopDialog = false;
 | |
| 	Row *_selected = nullptr;
 | |
| 	Row *_pressed = nullptr;
 | |
| 	MsgId _pressedTopicJumpRootId;
 | |
| 	bool _selectedTopicJump = false;
 | |
| 	bool _pressedTopicJump = false;
 | |
| 
 | |
| 	Row *_dragging = nullptr;
 | |
| 	int _draggingIndex = -1;
 | |
| 	int _aboveIndex = -1;
 | |
| 	QPoint _dragStart;
 | |
| 	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 _narrowWidth = 0;
 | |
| 
 | |
| 	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<FilterResult> _filterResults;
 | |
| 	base::flat_map<Key, 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;
 | |
| 	base::flat_set<not_null<History*>> _searchResultsHistories;
 | |
| 	rpl::lifetime _searchResultsLifetime;
 | |
| 	int _searchedCount = 0;
 | |
| 	int _searchedMigratedCount = 0;
 | |
| 	int _searchedSelected = -1;
 | |
| 	int _searchedPressed = -1;
 | |
| 
 | |
| 	WidgetState _state = WidgetState::Default;
 | |
| 
 | |
| 	object_ptr<Ui::FlatLabel> _empty = { nullptr };
 | |
| 	object_ptr<Ui::IconButton> _cancelSearchInChat;
 | |
| 	object_ptr<Ui::IconButton> _cancelSearchFromUser;
 | |
| 
 | |
| 	Ui::DraggingScrollManager _draggingScroll;
 | |
| 
 | |
| 	Key _searchInChat;
 | |
| 	History *_searchInMigrated = nullptr;
 | |
| 	PeerData *_searchFromPeer = nullptr;
 | |
| 	mutable Ui::PeerUserpicView _searchInChatUserpic;
 | |
| 	mutable Ui::PeerUserpicView _searchFromUserUserpic;
 | |
| 	Ui::Text::String _searchInChatText;
 | |
| 	Ui::Text::String _searchFromUserText;
 | |
| 	RowDescriptor _menuRow;
 | |
| 
 | |
| 	base::flat_map<
 | |
| 		not_null<PeerData*>,
 | |
| 		std::unique_ptr<Ui::VideoUserpic>> _videoUserpics;
 | |
| 
 | |
| 	base::flat_map<FilterId, int> _chatsFilterScrollStates;
 | |
| 
 | |
| 	Fn<void()> _loadMoreCallback;
 | |
| 	Fn<void()> _loadMoreFilteredCallback;
 | |
| 	rpl::event_stream<> _listBottomReached;
 | |
| 	rpl::event_stream<ChosenRow> _chosenRow;
 | |
| 	rpl::event_stream<> _updated;
 | |
| 
 | |
| 	rpl::event_stream<Ui::ScrollToRequest> _mustScrollTo;
 | |
| 	rpl::event_stream<Ui::ScrollToRequest> _dialogMoved;
 | |
| 	rpl::event_stream<> _searchMessages;
 | |
| 	rpl::event_stream<QString> _completeHashtagRequests;
 | |
| 	rpl::event_stream<> _refreshHashtagsRequests;
 | |
| 
 | |
| 	rpl::variable<ChildListShown> _childListShown;
 | |
| 	float64 _narrowRatio = 0.;
 | |
| 	bool _geometryInited = false;
 | |
| 
 | |
| 	base::unique_qptr<Ui::PopupMenu> _menu;
 | |
| 
 | |
| };
 | |
| 
 | |
| } // namespace Dialogs
 |