File indexing completed on 2024-09-15 03:38:43
0001 /* 0002 This file is part of the KDE project 0003 SPDX-FileCopyrightText: 2001 Martin R. Jones <mjones@kde.org> 0004 SPDX-FileCopyrightText: 2001 Carsten Pfeiffer <pfeiffer@kde.org> 0005 SPDX-FileCopyrightText: 2008 Rafael Fernández López <ereslibre@kde.org> 0006 0007 SPDX-License-Identifier: LGPL-2.0-only 0008 */ 0009 0010 #include "kimagefilepreview.h" 0011 0012 #include <QCheckBox> 0013 #include <QLabel> 0014 #include <QPainter> 0015 #include <QResizeEvent> 0016 #include <QStyle> 0017 #include <QTimeLine> 0018 #include <QVBoxLayout> 0019 0020 #include <KConfig> 0021 #include <KConfigGroup> 0022 #include <KIconLoader> 0023 #include <KLocalizedString> 0024 #include <kfileitem.h> 0025 #include <kio/previewjob.h> 0026 0027 /**** KImageFilePreview ****/ 0028 0029 class KImageFilePreviewPrivate 0030 { 0031 public: 0032 KImageFilePreviewPrivate(KImageFilePreview *qq) 0033 : q(qq) 0034 { 0035 if (q->style()->styleHint(QStyle::SH_Widget_Animate, nullptr, q)) { 0036 m_timeLine = new QTimeLine(150, q); 0037 m_timeLine->setEasingCurve(QEasingCurve::InCurve); 0038 m_timeLine->setDirection(QTimeLine::Forward); 0039 m_timeLine->setFrameRange(0, 100); 0040 } 0041 } 0042 0043 void slotResult(KJob *); 0044 void slotFailed(const KFileItem &); 0045 void slotStepAnimation(); 0046 void slotFinished(); 0047 void slotActuallyClear(); 0048 0049 KImageFilePreview *q = nullptr; 0050 QUrl currentURL; 0051 QUrl lastShownURL; 0052 QLabel *imageLabel; 0053 KIO::PreviewJob *m_job = nullptr; 0054 QTimeLine *m_timeLine = nullptr; 0055 QPixmap m_pmCurrent; 0056 QPixmap m_pmTransition; 0057 float m_pmCurrentOpacity = 1; 0058 float m_pmTransitionOpacity = 0; 0059 bool clear = true; 0060 }; 0061 0062 KImageFilePreview::KImageFilePreview(QWidget *parent) 0063 : KPreviewWidgetBase(parent) 0064 , d(new KImageFilePreviewPrivate(this)) 0065 { 0066 QVBoxLayout *vb = new QVBoxLayout(this); 0067 vb->setContentsMargins(0, 0, 0, 0); 0068 0069 d->imageLabel = new QLabel(this); 0070 d->imageLabel->setAlignment(Qt::AlignHCenter | Qt::AlignVCenter); 0071 d->imageLabel->setSizePolicy(QSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding)); 0072 vb->addWidget(d->imageLabel); 0073 0074 setSupportedMimeTypes(KIO::PreviewJob::supportedMimeTypes()); 0075 setMinimumWidth(50); 0076 0077 if (d->m_timeLine) { 0078 connect(d->m_timeLine, &QTimeLine::frameChanged, this, [this]() { 0079 d->slotStepAnimation(); 0080 }); 0081 connect(d->m_timeLine, &QTimeLine::finished, this, [this]() { 0082 d->slotFinished(); 0083 }); 0084 } 0085 } 0086 0087 KImageFilePreview::~KImageFilePreview() 0088 { 0089 if (d->m_job) { 0090 d->m_job->kill(); 0091 } 0092 } 0093 0094 void KImageFilePreview::showPreview() 0095 { 0096 // Pass a copy since clearPreview() will clear currentURL 0097 QUrl url = d->currentURL; 0098 showPreview(url, true); 0099 } 0100 0101 // called via KPreviewWidgetBase interface 0102 void KImageFilePreview::showPreview(const QUrl &url) 0103 { 0104 showPreview(url, false); 0105 } 0106 0107 void KImageFilePreview::showPreview(const QUrl &url, bool force) 0108 { 0109 /* clang-format off */ 0110 if (!url.isValid() 0111 || (d->lastShownURL.isValid() 0112 && url.matches(d->lastShownURL, QUrl::StripTrailingSlash) 0113 && d->currentURL.isValid())) { 0114 return; 0115 } 0116 /* clang-format on*/ 0117 0118 d->clear = false; 0119 d->currentURL = url; 0120 d->lastShownURL = url; 0121 0122 int w = d->imageLabel->contentsRect().width() - 4; 0123 int h = d->imageLabel->contentsRect().height() - 4; 0124 0125 if (d->m_job) { 0126 disconnect(d->m_job, nullptr, this, nullptr); 0127 0128 d->m_job->kill(); 0129 } 0130 0131 d->m_job = createJob(url, w, h); 0132 if (force) { // explicitly requested previews shall always be generated! 0133 d->m_job->setIgnoreMaximumSize(true); 0134 } 0135 0136 connect(d->m_job, &KJob::result, this, [this](KJob *job) { 0137 d->slotResult(job); 0138 }); 0139 connect(d->m_job, &KIO::PreviewJob::gotPreview, this, &KImageFilePreview::gotPreview); 0140 connect(d->m_job, &KIO::PreviewJob::failed, this, [this](const KFileItem &item) { 0141 d->slotFailed(item); 0142 }); 0143 } 0144 0145 void KImageFilePreview::resizeEvent(QResizeEvent *) 0146 { 0147 // Nothing to do, if no current preview 0148 if (d->imageLabel->pixmap().isNull()) { 0149 return; 0150 } 0151 0152 clearPreview(); 0153 d->currentURL = QUrl(); // force this to actually happen 0154 showPreview(d->lastShownURL); 0155 } 0156 0157 QSize KImageFilePreview::sizeHint() const 0158 { 0159 return QSize(100, 200); 0160 } 0161 0162 KIO::PreviewJob *KImageFilePreview::createJob(const QUrl &url, int w, int h) 0163 { 0164 if (!url.isValid()) { 0165 return nullptr; 0166 } 0167 0168 KFileItemList items; 0169 items.append(KFileItem(url)); 0170 QStringList plugins = KIO::PreviewJob::availablePlugins(); 0171 0172 KIO::PreviewJob *previewJob = KIO::filePreview(items, QSize(w, h), &plugins); 0173 previewJob->setScaleType(KIO::PreviewJob::Scaled); 0174 return previewJob; 0175 } 0176 0177 void KImageFilePreview::gotPreview(const KFileItem &item, const QPixmap &pm) 0178 { 0179 if (item.url() != d->currentURL) { // Shouldn't happen 0180 return; 0181 } 0182 0183 if (d->m_timeLine) { 0184 if (d->m_timeLine->state() == QTimeLine::Running) { 0185 d->m_timeLine->setCurrentTime(0); 0186 } 0187 0188 d->m_pmTransition = pm; 0189 d->m_pmTransitionOpacity = 0; 0190 d->m_pmCurrentOpacity = 1; 0191 d->m_timeLine->setDirection(QTimeLine::Forward); 0192 d->m_timeLine->start(); 0193 } else { 0194 d->imageLabel->setPixmap(pm); 0195 } 0196 } 0197 0198 void KImageFilePreviewPrivate::slotFailed(const KFileItem &item) 0199 { 0200 if (item.isDir()) { 0201 imageLabel->clear(); 0202 } else if (item.url() == currentURL) { // should always be the case 0203 imageLabel->setPixmap(QIcon::fromTheme(QStringLiteral("image-missing")).pixmap(KIconLoader::SizeLarge, QIcon::Disabled)); 0204 } 0205 } 0206 0207 void KImageFilePreviewPrivate::slotResult(KJob *job) 0208 { 0209 if (job == m_job) { 0210 m_job = nullptr; 0211 } 0212 } 0213 0214 void KImageFilePreviewPrivate::slotStepAnimation() 0215 { 0216 const QSize currSize = m_pmCurrent.size(); 0217 const QSize transitionSize = m_pmTransition.size(); 0218 const int width = std::max(currSize.width(), transitionSize.width()); 0219 const int height = std::max(currSize.height(), transitionSize.height()); 0220 QPixmap pm(QSize(width, height)); 0221 pm.fill(Qt::transparent); 0222 0223 QPainter p(&pm); 0224 p.setOpacity(m_pmCurrentOpacity); 0225 0226 // If we have a current pixmap 0227 if (!m_pmCurrent.isNull()) { 0228 p.drawPixmap(QPoint(((float)pm.size().width() - m_pmCurrent.size().width()) / 2.0, ((float)pm.size().height() - m_pmCurrent.size().height()) / 2.0), 0229 m_pmCurrent); 0230 } 0231 if (!m_pmTransition.isNull()) { 0232 p.setOpacity(m_pmTransitionOpacity); 0233 p.drawPixmap( 0234 QPoint(((float)pm.size().width() - m_pmTransition.size().width()) / 2.0, ((float)pm.size().height() - m_pmTransition.size().height()) / 2.0), 0235 m_pmTransition); 0236 } 0237 p.end(); 0238 0239 imageLabel->setPixmap(pm); 0240 0241 m_pmCurrentOpacity = qMax(m_pmCurrentOpacity - 0.4, 0.0); // krazy:exclude=qminmax 0242 m_pmTransitionOpacity = qMin(m_pmTransitionOpacity + 0.4, 1.0); // krazy:exclude=qminmax 0243 } 0244 0245 void KImageFilePreviewPrivate::slotFinished() 0246 { 0247 m_pmCurrent = m_pmTransition; 0248 m_pmTransitionOpacity = 0; 0249 m_pmCurrentOpacity = 1; 0250 m_pmTransition = QPixmap(); 0251 // The animation might have lost some frames. Be sure that if the last one 0252 // was dropped, the last image shown is the opaque one. 0253 imageLabel->setPixmap(m_pmCurrent); 0254 clear = false; 0255 } 0256 0257 void KImageFilePreview::clearPreview() 0258 { 0259 if (d->m_job) { 0260 d->m_job->kill(); 0261 d->m_job = nullptr; 0262 } 0263 0264 if (d->clear || (d->m_timeLine && d->m_timeLine->state() == QTimeLine::Running)) { 0265 return; 0266 } 0267 0268 if (d->m_timeLine) { 0269 d->m_pmTransition = QPixmap(); 0270 // If we add a previous preview then we run the animation 0271 if (!d->m_pmCurrent.isNull()) { 0272 d->m_timeLine->setCurrentTime(0); 0273 d->m_timeLine->setDirection(QTimeLine::Backward); 0274 d->m_timeLine->start(); 0275 } 0276 d->currentURL.clear(); 0277 d->clear = true; 0278 } else { 0279 d->imageLabel->clear(); 0280 } 0281 } 0282 0283 #include "moc_kimagefilepreview.cpp"