403 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			403 lines
		
	
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
| 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<Overview::Layout::AbstractItem*> Items;
 | |
| 	Items _items;
 | |
| 	typedef QMap<HistoryItem*, Overview::Layout::ItemBase*> LayoutItems;
 | |
| 	LayoutItems _layoutItems;
 | |
| 	typedef QMap<int32, Overview::Layout::Date*> 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<Ui::FlatInput> _search;
 | |
| 	object_ptr<Ui::CrossButton> _cancelSearch;
 | |
| 	QVector<MsgId> _results;
 | |
| 	int32 _itemsToBeLoaded;
 | |
| 
 | |
| 	// photos
 | |
| 	int32 _photosInRow = 1;
 | |
| 
 | |
| 	QTimer _searchTimer;
 | |
| 	QString _searchQuery;
 | |
| 	bool _inSearch = false;
 | |
| 	bool _searchFull = false;
 | |
| 	bool _searchFullMigrated = false;
 | |
| 	mtpRequestId _searchRequest = 0;
 | |
| 	QList<MsgId> _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<QString, MTPmessages_Messages> SearchCache;
 | |
| 	SearchCache _searchCache;
 | |
| 
 | |
| 	typedef QMap<mtpRequestId, QString> 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<MsgId, TextSelection>;
 | |
| 	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<Window::Controller*> 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;
 | |
| 
 | |
| 	void ui_repaintHistoryItem(not_null<const HistoryItem*> item);
 | |
| 
 | |
| 	void notify_historyItemLayoutChanged(const HistoryItem *item);
 | |
| 
 | |
| 	~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<Ui::AbstractButton> _backAnimationButton = { nullptr };
 | |
| 	object_ptr<Window::TopBarWidget> _topBar;
 | |
| 	object_ptr<Ui::ScrollArea> _scroll;
 | |
| 	QPointer<OverviewInner> _inner;
 | |
| 	bool _noDropResizeIndex = false;
 | |
| 
 | |
| 	object_ptr<Ui::DropdownMenu> _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<Ui::PlainShadow> _topShadow;
 | |
| 	bool _inGrab = false;
 | |
| 
 | |
| };
 | |
| 
 | 
