/* 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 namespace Window { class MainMenu; } // namespace Window class LayerWidget : public TWidget { Q_OBJECT public: using TWidget::TWidget; virtual void parentResized() = 0; virtual void showFinished() { } void setInnerFocus(); bool setClosing() { if (!_closing) { _closing = true; closeHook(); return true; } return false; } bool overlaps(const QRect &globalRect); void setClosedCallback(base::lambda &&callback) { _closedCallback = std_::move(callback); } void setResizedCallback(base::lambda &&callback) { _resizedCallback = std_::move(callback); } protected: void closeLayer() { if (_closedCallback) { _closedCallback(); } } void mousePressEvent(QMouseEvent *e) override { e->accept(); } void resizeEvent(QResizeEvent *e) override { if (_resizedCallback) { _resizedCallback(); } } virtual void doSetInnerFocus() { setFocus(); } virtual void closeHook() { } private: bool _closing = false; base::lambda _closedCallback; base::lambda _resizedCallback; }; class LayerStackWidget : public TWidget { Q_OBJECT public: LayerStackWidget(QWidget *parent); void finishAnimation(); void showBox(object_ptr box); void showSpecialLayer(object_ptr layer); void showMainMenu(); void appendBox(object_ptr box); void prependBox(object_ptr box); bool canSetFocus() const; void setInnerFocus(); bool contentOverlapped(const QRect &globalRect); void hideLayers(); void hideAll(); void hideTopLayer(); bool layerShown() const; ~LayerStackWidget(); protected: void keyPressEvent(QKeyEvent *e) override; void mousePressEvent(QMouseEvent *e) override; void resizeEvent(QResizeEvent *e) override; private slots: void onLayerDestroyed(QObject *obj); void onLayerClosed(LayerWidget *layer); void onLayerResized(); private: LayerWidget *pushBox(object_ptr box); void showFinished(); void hideCurrent(); enum class Action { ShowMainMenu, ShowSpecialLayer, ShowLayer, HideLayer, HideAll, }; template void startAnimation(SetupNew setupNewWidgets, ClearOld clearOldWidgets, Action action); void prepareForAnimation(); void animationDone(); void setCacheImages(); void clearLayers(); void clearSpecialLayer(); void initChildLayer(LayerWidget *layer); void updateLayerBoxes(); void fixOrder(); void sendFakeMouseEvent(); LayerWidget *currentLayer() { return _layers.empty() ? nullptr : _layers.back(); } const LayerWidget *currentLayer() const { return const_cast(this)->currentLayer(); } using Layers = QList; Layers _layers; object_ptr _specialLayer = { nullptr }; object_ptr _mainMenu = { nullptr }; class BackgroundWidget; object_ptr _background; }; class MediaPreviewWidget : public TWidget, private base::Subscriber { Q_OBJECT public: MediaPreviewWidget(QWidget *parent); void showPreview(DocumentData *document); void showPreview(PhotoData *photo); void hidePreview(); ~MediaPreviewWidget(); protected: void paintEvent(QPaintEvent *e) override; void resizeEvent(QResizeEvent *e) override; private: QSize currentDimensions() const; QPixmap currentImage() const; void startShow(); void fillEmojiString(); void resetGifAndCache(); Animation _a_shown; bool _hiding = false; DocumentData *_document = nullptr; PhotoData *_photo = nullptr; Media::Clip::ReaderPointer _gif; int _emojiSize; QList _emojiList; void clipCallback(Media::Clip::Notification notification); enum CacheStatus { CacheNotLoaded, CacheThumbLoaded, CacheLoaded, }; mutable CacheStatus _cacheStatus = CacheNotLoaded; mutable QPixmap _cache; mutable QSize _cachedSize; };