12#include <QCoreApplication>
14#include <QDomDocument>
17#include <QImageReader>
18#include <QReadWriteLock>
37#define LOC QString("MythUIImage(0x%1): ").arg((uint64_t)this,0,16)
123 LOG(VB_GUI | VB_FILE, LOG_DEBUG,
124 QString(
"ImageLoader::PreLoad(%1), this "
125 "file is already being loaded by this same MythUIImage "
126 "in another thread.").arg(cacheKey));
151 QString extension =
filename.section(
'.', -1);
152 return !
filename.startsWith(
"myth://") &&
153 (extension ==
"gif" ||
154 extension ==
"apng" ||
171 s_Attrib +=
"reflected";
174 s_Attrib +=
"greyscale";
178 s_Attrib +=
"orientation";
194 imagelabel = QString(
"%1-%2-%3x%4.png")
199 imagelabel.replace(
'/',
'-');
201 imagelabel.replace(
':',
'-');
219 if (!
PreLoad(cacheKey, parent))
228 bool bResize =
false;
229 bool bFoundInCache =
false;
257 LOG(VB_GUI | VB_FILE, LOG_INFO,
258 QString(
"ImageLoader::LoadImage(%1) Found in cache, "
260 .arg(cacheKey).arg(cnt));
269 bFoundInCache =
true;
273 LOG(VB_GUI | VB_FILE, LOG_INFO,
274 QString(
"ImageLoader::LoadImage(%1) NOT Found in cache. "
275 "Loading Directly").arg(cacheKey));
281 ok = image->
Load(imageReader);
292 if (image && image->isNull())
294 LOG(VB_GUI | VB_FILE, LOG_INFO,
295 QString(
"ImageLoader::LoadImage(%1) Image is NULL")
302 if (image && !bFoundInCache)
325 if (wmult != 1.0F || hmult != 1.0F)
327 w = image->size().width() * wmult;
328 h = image->size().height() * hmult;
344 if (wmult != 1.0F || hmult != 1.0F)
346 int width = newMaskImage->size().width() * wmult;
347 int height = newMaskImage->size().height() * hmult;
348 newMaskImage->
Resize(QSize(width, height));
359 QRect imageArea = image->rect();
366 if (maskArea.width() > imageArea.width())
367 x = (maskArea.width() - imageArea.width()) / 2;
369 if (maskArea.height() > imageArea.height())
370 y = (maskArea.height() - imageArea.height()) / 2;
373 imageArea.translate(x, y);
376 image->setAlphaChannel(mask.convertToFormat(QImage::Format_Alpha8));
402 QString frameFilename;
408 while (imageReader->canRead() && !aborted)
410 frameFilename =
filename.arg(imageCount);
416 aborted, imageReader);
421 images->append(
AnimationFrame(im, std::chrono::milliseconds(imageReader->nextImageDelay())));
444 int number,
bool aborted)
483 (QEvent::Type) QEvent::registerEventType();
501 bool aborted =
false;
514 if (frames && frames->count() > 1)
519 QCoreApplication::postEvent(
m_parent, le);
534 QCoreApplication::postEvent(
m_parent, le);
562 int low,
int high, std::chrono::milliseconds delay,
734 for (std::chrono::milliseconds delay : std::as_const(delays))
765 if (!forceSize.isNull())
767 int w = (forceSize.width() <= 0) ? img->width() : forceSize.width();
768 int h = (forceSize.height() <= 0) ? img->height() : forceSize.height();
819 for (
auto *im : std::as_const(*images))
832 if (!forceSize.isNull())
834 int w = (forceSize.width() <= 0) ? im->width() : forceSize.width();
835 int h = (forceSize.height() <= 0) ? im->height() : forceSize.height();
858 aSize = aSize.expandedTo(im->size());
878 QVector<std::chrono::milliseconds> delays;
879 QVector<MythImage *> images;
881 for (
const auto & frame : std::as_const(frames))
883 images.append(frame.first);
884 delays.append(frame.second);
891 if (
m_delay < 0ms && !delays.empty())
982 if (bFilename.isEmpty())
991 if (qEnvironmentVariableIsSet(
"DISABLETHREADEDMYTHUIIMAGE"))
992 allowLoadInBackground =
false;
1002 bool complete =
true;
1011 bFilename.contains(
"%1"))
1030 bool do_background_load =
false;
1031 if (allowLoadInBackground)
1039 do_background_load =
true;
1042 if (do_background_load)
1045 LOG(VB_GUI | VB_FILE, LOG_DEBUG,
LOC +
1046 QString(
"Load(), spawning thread to load '%1'").arg(
filename));
1050 imProps, bFilename, i,
1056 if (!isAnimation && !
GetMythUI()->IsImageInCache(imagelabel))
1060 LOG(VB_GUI | VB_FILE, LOG_DEBUG,
LOC +
1061 QString(
"Load(), loading '%1' in foreground").arg(
filename));
1062 bool aborted =
false;
1074 LOG(VB_GUI, LOG_DEBUG, QString(
"Aborted loading animated"
1075 "image %1 in foreground")
1095 LOG(VB_GUI, LOG_DEBUG, QString(
"Aborted loading animated"
1096 "image %1 in foreground")
1106 rect.setSize(image->size());
1132 complete &= !do_background_load;
1156 abs(
m_lastDisplay.msecsTo(QTime::currentTime())) > delay.count())
1210 int alphaMod, QRect clipRect)
1236 area.translate(xoffset, yoffset);
1253 QRect currentImageArea = currentImage->rect();
1256 area.setSize(area.size().expandedTo(currentImage->size()));
1263 if (area.width() > visibleImage.width())
1264 x = (area.width() / 2) + visibleImage.topLeft().x();
1266 if (area.height() > visibleImage.height())
1267 y = (area.height() / 2) + visibleImage.topLeft().y();
1269 if ((x > 0 || y > 0))
1270 area.translate(x, y);
1278 srcRect = currentImageArea;
1280 p->SetClipRect(clipRect);
1281 p->DrawImage(area, currentImage, srcRect, alpha);
1295 const QString &
filename, QDomElement &element,
bool showWarnings)
1299 if (element.tagName() ==
"filename")
1312 else if (element.tagName() ==
"filepattern")
1316 QString
tmp = element.attribute(
"low");
1321 tmp = element.attribute(
"high");
1326 tmp = element.attribute(
"cycle",
"start");
1328 if (
tmp ==
"reverse")
1331 else if (element.tagName() ==
"area")
1336 else if (element.tagName() ==
"preserveaspect")
1340 else if (element.tagName() ==
"crop")
1344 else if (element.tagName() ==
"delay")
1348 if (value.contains(
","))
1350 QVector<std::chrono::milliseconds> delays;
1351 QStringList tokens = value.split(
",");
1352 for (
const auto & token : std::as_const(tokens))
1354 if (token.isEmpty())
1356 if (!delays.empty())
1357 delays.append(delays[delays.size()-1]);
1363 delays.append(std::chrono::milliseconds(token.toInt()));
1367 if (!delays.empty())
1375 m_delay = std::chrono::milliseconds(value.toInt());
1378 else if (element.tagName() ==
"reflection")
1381 QString
tmp = element.attribute(
"axis");
1385 if (
tmp.toLower() ==
"horizontal")
1391 tmp = element.attribute(
"shear");
1396 tmp = element.attribute(
"scale");
1401 tmp = element.attribute(
"length");
1406 tmp = element.attribute(
"spacing");
1411 else if (element.tagName() ==
"mask")
1416 else if (element.tagName() ==
"grayscale" ||
1417 element.tagName() ==
"greyscale")
1443 LOG(VB_GENERAL, LOG_ERR,
1444 QString(
"'%1' (%2) ERROR, bad parsing '%3' (%4)")
1551 if (!le || le->GetParent() !=
this)
1555 int number = le->GetNumber();
1556 QString
filename = le->GetFilename();
1558 bool aborted = le->GetAbortState();
1571 if (aborted || (le->GetBasefile() != propFilename))
1574 LOG(VB_GUI, LOG_DEBUG, QString(
"Aborted loading image %1")
1580 if (animationFrames)
1582 for (
const auto & frame : std::as_const(*animationFrames))
1589 delete animationFrames;
1592 else if (animationFrames)
1596 delete animationFrames;
1612 rect.setSize(image->size());
1656 if (!imageDir.exists())
1662 QStringList imageTypes;
1664 QList< QByteArray > exts = QImageReader::supportedImageFormats();
1665 for (
const auto & ext : std::as_const(exts))
1667 imageTypes.append(QString(
"*.").append(ext));
1670 imageDir.setNameFilters(imageTypes);
1681 std::random_device rd;
1682 std::mt19937 g(rd());
1696 std::random_device rd;
1697 std::mt19937 g(rd());
ImageLoadEvent(const MythUIImage *parent, MythImage *image, QString basefile, QString filename, int number, bool aborted)
const MythUIImage * GetParent() const
QString GetFilename() const
ImageLoadEvent(const MythUIImage *parent, AnimationFrames *frames, QString basefile, QString filename, bool aborted)
bool GetAbortState() const
AnimationFrames * m_images
static const Type kEventType
AnimationFrames * GetAnimationFrames() const
const MythUIImage * m_parent
MythImage * GetImage() const
QString GetBasefile() const
ImageCacheMode m_cacheMode
ImageLoadThread(MythUIImage *parent, MythPainter *painter, const ImageProperties &imProps, QString basefile, int number, ImageCacheMode mode)
ImageProperties m_imageProperties
static QHash< QString, const MythUIImage * > m_loadingImages
static MythImage * LoadImage(MythPainter *painter, ImageProperties imProps, ImageCacheMode cacheMode, const MythUIImage *parent, bool &aborted, MythImageReader *imageReader=nullptr)
static QMutex m_loadingImagesLock
static QString GenImageLabel(const ImageProperties &imProps)
Generates a unique identifying string for this image which is used as a key in the image cache.
static void PostLoad(const QString &cacheKey)
static bool PreLoad(const QString &cacheKey, const MythUIImage *uitype)
static AnimationFrames * LoadAnimatedImage(MythPainter *painter, const ImageProperties &imProps, ImageCacheMode cacheMode, const MythUIImage *parent, bool &aborted)
static QWaitCondition m_loadingImagesCond
static bool SupportsAnimation(const QString &filename)
QRect GetMaskImageRect(void)
ImageProperties()=default
void SetMaskImageFilename(const QString &filename)
ReflectAxis m_reflectAxis
void SetMaskImage(MythImage *image)
void Copy(const ImageProperties &other)
ImageProperties & operator=(const ImageProperties &other)
QString GetMaskImageFilename()
QString m_maskImageFilename
QImage GetMaskImageSubset(QRect imageArea)
void start(QRunnable *runnable, const QString &debugName, int priority=0)
bool Load(MythImageReader *reader)
virtual void SetChanged(bool change=true)
void setIsOriented(bool oriented)
void Orientation(int orientation)
Changes the orientation angle of the image according to the exif rotation values.
void Reflect(ReflectAxis axis, int shear, int scale, int length, int spacing=0)
void Resize(QSize newSize, bool preserveAspect=false)
void setIsReflected(bool reflected)
int DecrRef(void) override
Decrements reference count and deletes on 0.
QString GetFileName(void) const
int IncrRef(void) override
Increments reference count.
MythImage * GetFormatImage()
Returns a blank reference counted image in the format required for the Draw functions for this painte...
Wrapper around QRect allowing us to handle percentage and other relative values for areas in mythui.
void CalculateArea(QRect parentArea)
QRect toQRect(void) const
~MythUIImagePrivate()=default
MythUIImagePrivate(MythUIImage *p)
QReadWriteLock m_updateLock
Image widget, displays a single image or multiple images in sequence.
void LoadNow(void) override
Cause images in this and child widgets to be loaded.
bool Load(bool allowLoadInBackground=true, bool forceStat=false)
Load the image(s), wraps ImageLoader::LoadImage()
MythUIImage(const QString &filepattern, int low, int high, std::chrono::milliseconds delay, MythUIType *parent, const QString &name)
void ForceSize(QSize size)
Force the dimensions of the widget and image to the given size.
QHash< int, MythImage * > m_images
void CopyFrom(MythUIType *base) override
Copy this widgets state from another.
void SetFilename(const QString &filename)
Must be followed by a call to Load() to load the image.
std::chrono::milliseconds m_delay
friend class ImageLoadThread
void SetFilepattern(const QString &filepattern, int low, int high)
Must be followed by a call to Load() to load the image.
void DrawSelf(MythPainter *p, int xoffset, int yoffset, int alphaMod, QRect clipRect) override
void CreateCopy(MythUIType *parent) override
Copy the state of this widget to the one given, it must be of the same type.
AnimationCycle m_animationCycle
void FindRandomImage(void)
void SetCropRect(int x, int y, int width, int height)
Crop the image using the given rectangle, useful for removing unsightly edges from imported images or...
void SetDelays(const QVector< std::chrono::milliseconds > &delays)
Sets the delays between each image in an animation.
void customEvent(QEvent *event) override
void SetImage(MythImage *img)
Should not be used unless absolutely necessary since it bypasses the image caching and threaded loade...
void Finalize(void) override
Perform any post-xml parsing initialisation tasks.
bool ParseElement(const QString &filename, QDomElement &element, bool showWarnings) override
Parse the xml definition of this widget setting the state of the object accordingly.
bool m_showingRandomImage
void Clear(void)
Remove all images from the widget.
void SetDelay(std::chrono::milliseconds delayms)
Set the delay between each image in an animation.
ImageProperties m_imageProperties
void SetImages(QVector< MythImage * > *images)
Should not be used unless absolutely necessary since it bypasses the image caching and threaded loade...
QHash< int, std::chrono::milliseconds > m_delays
void Pulse(void) override
Pulse is called 70 times a second to trigger a single frame of an animation.
void SetSize(int width, int height)
Set the size of the widget.
void Reset(void) override
Reset the image back to the default defined in the theme.
void SetAnimationFrames(const AnimationFrames &frames)
void SetOrientation(int orientation)
Saves the exif orientation value of the first image in the widget.
void SetImageCount(int low, int high)
Set the integer range for an animated image pattern.
void GetScalingFactors(float &Horizontal, float &Vertical) const
MThreadPool * GetImageThreadPool()
MythImage * CacheImage(const QString &URL, MythImage *Image, bool NoDisk=false)
MythImage * LoadCacheImage(QString File, const QString &Label, MythPainter *Painter, ImageCacheMode cacheMode=kCacheNormal)
The base class on which all widgets and screens are based.
virtual void SetSize(QSize size)
virtual MythPainter * GetPainter(void)
QString GetXMLLocation(void) const
virtual void SetArea(const MythRect &rect)
virtual void LoadNow(void)
Cause images in this and child widgets to be loaded.
virtual void CopyFrom(MythUIType *base)
Copy this widgets state from another.
virtual void Finalize(void)
Perform any post-xml parsing initialisation tasks.
virtual void SetMinArea(const MythRect &rect)
Set the minimum area based on the given size.
virtual void Pulse(void)
Pulse is called 70 times a second to trigger a single frame of an animation.
virtual MythRect GetArea(void) const
If the object has a minimum area defined, return it, other wise return the default area.
bool IsDeferredLoading(bool recurse=false) const
virtual MythRect GetFullArea(void) const
int CalcAlpha(int alphamod) const
void DependChanged(bool isDefault)
virtual void Reset(void)
Reset the widget to it's original state, should not reset changes made by the theme.
virtual bool ParseElement(const QString &filename, QDomElement &element, bool showWarnings)
Parse the xml definition of this widget setting the state of the object accordingly.
QRect GetExtent(QSize size) const
static MythRect parseRect(const QString &text, bool normalize=true)
static QString getFirstText(QDomElement &element)
static bool parseBool(const QString &text)
static const iso6937table * d
static bool VERBOSE_LEVEL_CHECK(uint64_t mask, LogLevel_t level)
#define LOG(_MASK_, _LEVEL_, _QSTRING_)
MythMainWindow * GetMythMainWindow(void)
MythUIHelper * GetMythUI()
QPair< MythImage *, std::chrono::milliseconds > AnimationFrame
QVector< AnimationFrame > AnimationFrames