#pragma once #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "common/Aliases.hpp" #include "common/Common.hpp" namespace chatterino { namespace detail { template struct Frame { Image image; int duration; }; class Frames : boost::noncopyable { public: Frames(); Frames(QVector> &&frames); ~Frames(); void clear(); bool empty() const; bool animated() const; void advance(); boost::optional current() const; boost::optional first() const; private: void processOffset(); QVector> items_; int index_{0}; int durationOffset_{0}; pajlada::Signals::Connection gifTimerConnection_; }; } // namespace detail class Image; using ImagePtr = std::shared_ptr; /// This class is thread safe. class Image : public std::enable_shared_from_this, boost::noncopyable { public: // Maximum amount of RAM used by the image in bytes. static constexpr int maxBytesRam = 20 * 1024 * 1024; ~Image(); static ImagePtr fromUrl(const Url &url, qreal scale = 1); static ImagePtr fromResourcePixmap(const QPixmap &pixmap, qreal scale = 1); static ImagePtr getEmpty(); const Url &url() const; bool loaded() const; // either returns the current pixmap, or triggers loading it (lazy loading) boost::optional pixmapOrLoad() const; void load() const; qreal scale() const; bool isEmpty() const; int width() const; int height() const; bool animated() const; bool operator==(const Image &image) const; bool operator!=(const Image &image) const; private: Image(); Image(const Url &url, qreal scale); Image(qreal scale); void setPixmap(const QPixmap &pixmap); void actuallyLoad(); void expireFrames(); const Url url_{}; const qreal scale_{1}; std::atomic_bool empty_{false}; mutable std::chrono::time_point lastUsed_; bool shouldLoad_{false}; // gui thread only std::unique_ptr frames_{}; friend class ImageExpirationPool; }; class ImageExpirationPool { private: friend class Image; ImageExpirationPool(); static ImageExpirationPool &instance(); void addImagePtr(ImagePtr imgPtr); void removeImagePtr(Image *rawPtr); /** * @brief Frees frame data for all images that ImagePool deems to have expired. * * Expiration is based on last accessed time of the Image, stored in Image::lastUsed_. * Must be ran in the GUI thread. */ void freeOld(); private: // Timer to periodically run freeOld() QTimer freeTimer_; std::map> allImages_; std::mutex mutex_; }; } // namespace chatterino