|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162 |
- /*
- ==============================================================================
-
- This file is part of the JUCE 6 technical preview.
- Copyright (c) 2020 - Raw Material Software Limited
-
- You may use this code under the terms of the GPL v3
- (see www.gnu.org/licenses).
-
- For this technical preview, this file is not subject to commercial licensing.
-
- JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
- EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
- DISCLAIMED.
-
- ==============================================================================
- */
-
- namespace juce
- {
-
- struct ImageCache::Pimpl : private Timer,
- private DeletedAtShutdown
- {
- Pimpl() {}
- ~Pimpl() override { clearSingletonInstance(); }
-
- JUCE_DECLARE_SINGLETON_SINGLETHREADED_MINIMAL (ImageCache::Pimpl)
-
- Image getFromHashCode (const int64 hashCode) noexcept
- {
- const ScopedLock sl (lock);
-
- for (auto& item : images)
- {
- if (item.hashCode == hashCode)
- {
- item.lastUseTime = Time::getApproximateMillisecondCounter();
- return item.image;
- }
- }
-
- return {};
- }
-
- void addImageToCache (const Image& image, const int64 hashCode)
- {
- if (image.isValid())
- {
- if (! isTimerRunning())
- startTimer (2000);
-
- const ScopedLock sl (lock);
- images.add ({ image, hashCode, Time::getApproximateMillisecondCounter() });
- }
- }
-
- void timerCallback() override
- {
- auto now = Time::getApproximateMillisecondCounter();
-
- const ScopedLock sl (lock);
-
- for (int i = images.size(); --i >= 0;)
- {
- auto& item = images.getReference(i);
-
- if (item.image.getReferenceCount() <= 1)
- {
- if (now > item.lastUseTime + cacheTimeout || now < item.lastUseTime - 1000)
- images.remove (i);
- }
- else
- {
- item.lastUseTime = now; // multiply-referenced, so this image is still in use.
- }
- }
-
- if (images.isEmpty())
- stopTimer();
- }
-
- void releaseUnusedImages()
- {
- const ScopedLock sl (lock);
-
- for (int i = images.size(); --i >= 0;)
- if (images.getReference(i).image.getReferenceCount() <= 1)
- images.remove (i);
- }
-
- struct Item
- {
- Image image;
- int64 hashCode;
- uint32 lastUseTime;
- };
-
- Array<Item> images;
- CriticalSection lock;
- unsigned int cacheTimeout = 5000;
-
- JUCE_DECLARE_NON_COPYABLE (Pimpl)
- };
-
- JUCE_IMPLEMENT_SINGLETON (ImageCache::Pimpl)
-
-
- //==============================================================================
- Image ImageCache::getFromHashCode (const int64 hashCode)
- {
- if (Pimpl::getInstanceWithoutCreating() != nullptr)
- return Pimpl::getInstanceWithoutCreating()->getFromHashCode (hashCode);
-
- return {};
- }
-
- void ImageCache::addImageToCache (const Image& image, const int64 hashCode)
- {
- Pimpl::getInstance()->addImageToCache (image, hashCode);
- }
-
- Image ImageCache::getFromFile (const File& file)
- {
- auto hashCode = file.hashCode64();
- auto image = getFromHashCode (hashCode);
-
- if (image.isNull())
- {
- image = ImageFileFormat::loadFrom (file);
- addImageToCache (image, hashCode);
- }
-
- return image;
- }
-
- Image ImageCache::getFromMemory (const void* imageData, const int dataSize)
- {
- auto hashCode = (int64) (pointer_sized_int) imageData;
- auto image = getFromHashCode (hashCode);
-
- if (image.isNull())
- {
- image = ImageFileFormat::loadFrom (imageData, (size_t) dataSize);
- addImageToCache (image, hashCode);
- }
-
- return image;
- }
-
- void ImageCache::setCacheTimeout (const int millisecs)
- {
- jassert (millisecs >= 0);
- Pimpl::getInstance()->cacheTimeout = (unsigned int) millisecs;
- }
-
- void ImageCache::releaseUnusedImages()
- {
- Pimpl::getInstance()->releaseUnusedImages();
- }
-
- } // namespace juce
|