2 * SPDX-FileCopyrightText: 2011 Peter Penz <peter.penz19@gmail.com>
4 * SPDX-License-Identifier: GPL-2.0-or-later
7 #include "kfileitemlistview.h"
9 #include "kfileitemlistwidget.h"
10 #include "kfileitemmodel.h"
11 #include "kfileitemmodelrolesupdater.h"
12 #include "private/kpixmapmodifier.h"
14 #include <KIconLoader>
16 #include <QGraphicsScene>
17 #include <QGraphicsView>
19 #include <QMimeDatabase>
23 // #define KFILEITEMLISTVIEW_DEBUG
27 // If the visible index range changes, KFileItemModelRolesUpdater is not
28 // informed immediately, but with a short delay. This ensures that scrolling
29 // always feels smooth and is not interrupted by icon loading (which can be
30 // quite expensive if a disk access is required to determine the final icon).
31 const int ShortInterval
= 50;
33 // If the icon size changes, a longer delay is used. This prevents that
34 // the expensive re-generation of all previews is triggered repeatedly when
35 // changing the zoom level.
36 const int LongInterval
= 300;
39 KFileItemListView::KFileItemListView(QGraphicsWidget
*parent
)
40 : KStandardItemListView(parent
)
41 , m_modelRolesUpdater(nullptr)
42 , m_updateVisibleIndexRangeTimer(nullptr)
43 , m_updateIconSizeTimer(nullptr)
44 , m_scanDirectories(true)
48 setScrollOrientation(Qt::Vertical
);
50 m_updateVisibleIndexRangeTimer
= new QTimer(this);
51 m_updateVisibleIndexRangeTimer
->setSingleShot(true);
52 m_updateVisibleIndexRangeTimer
->setInterval(ShortInterval
);
53 connect(m_updateVisibleIndexRangeTimer
, &QTimer::timeout
, this, &KFileItemListView::updateVisibleIndexRange
);
55 m_updateIconSizeTimer
= new QTimer(this);
56 m_updateIconSizeTimer
->setSingleShot(true);
57 m_updateIconSizeTimer
->setInterval(LongInterval
);
58 connect(m_updateIconSizeTimer
, &QTimer::timeout
, this, &KFileItemListView::updateIconSize
);
60 setVisibleRoles({"text"});
63 KFileItemListView::~KFileItemListView()
67 void KFileItemListView::setPreviewsShown(bool show
)
69 if (!m_modelRolesUpdater
) {
73 if (m_modelRolesUpdater
->previewsShown() != show
) {
75 m_modelRolesUpdater
->setPreviewsShown(show
);
76 onPreviewsShownChanged(show
);
81 bool KFileItemListView::previewsShown() const
83 return m_modelRolesUpdater
? m_modelRolesUpdater
->previewsShown() : false;
86 void KFileItemListView::setEnlargeSmallPreviews(bool enlarge
)
88 if (m_modelRolesUpdater
) {
89 m_modelRolesUpdater
->setEnlargeSmallPreviews(enlarge
);
93 bool KFileItemListView::enlargeSmallPreviews() const
95 return m_modelRolesUpdater
? m_modelRolesUpdater
->enlargeSmallPreviews() : false;
98 void KFileItemListView::setEnabledPlugins(const QStringList
&list
)
100 if (m_modelRolesUpdater
) {
101 m_modelRolesUpdater
->setEnabledPlugins(list
);
105 QStringList
KFileItemListView::enabledPlugins() const
107 return m_modelRolesUpdater
? m_modelRolesUpdater
->enabledPlugins() : QStringList();
110 void KFileItemListView::setLocalFileSizePreviewLimit(const qlonglong size
)
112 if (m_modelRolesUpdater
) {
113 m_modelRolesUpdater
->setLocalFileSizePreviewLimit(size
);
117 qlonglong
KFileItemListView::localFileSizePreviewLimit() const
119 return m_modelRolesUpdater
? m_modelRolesUpdater
->localFileSizePreviewLimit() : 0;
122 void KFileItemListView::setScanDirectories(bool enabled
)
124 m_scanDirectories
= enabled
;
125 if (m_modelRolesUpdater
) {
126 m_modelRolesUpdater
->setScanDirectories(m_scanDirectories
);
130 bool KFileItemListView::scanDirectories()
132 return m_scanDirectories
;
135 QPixmap
KFileItemListView::createDragPixmap(const KItemSet
&indexes
) const
141 const int itemCount
= indexes
.count();
142 Q_ASSERT(itemCount
> 0);
143 if (itemCount
== 1) {
144 return KItemListView::createDragPixmap(indexes
);
147 // If more than one item is dragged, align the items inside a
148 // rectangular grid. The maximum grid size is limited to 5 x 5 items.
151 if (itemCount
> 16) {
153 size
= KIconLoader::SizeSmall
;
154 } else if (itemCount
> 9) {
156 size
= KIconLoader::SizeSmallMedium
;
159 size
= KIconLoader::SizeMedium
;
162 if (itemCount
< xCount
) {
166 int yCount
= itemCount
/ xCount
;
167 if (itemCount
% xCount
!= 0) {
170 if (yCount
> xCount
) {
174 const qreal dpr
= scene()->views()[0]->devicePixelRatio();
175 // Draw the selected items into the grid cells.
176 QPixmap
dragPixmap(QSize(xCount
* size
+ xCount
, yCount
* size
+ yCount
) * dpr
);
177 dragPixmap
.setDevicePixelRatio(dpr
);
178 dragPixmap
.fill(Qt::transparent
);
180 QPainter
painter(&dragPixmap
);
184 for (int index
: indexes
) {
185 QPixmap pixmap
= model()->data(index
).value("iconPixmap").value
<QPixmap
>();
186 if (pixmap
.isNull()) {
187 QIcon icon
= QIcon::fromTheme(model()->data(index
).value("iconName").toString());
189 icon
= QIcon::fromTheme(QStringLiteral("unknown"));
191 if (!icon
.isNull()) {
192 pixmap
= icon
.pixmap(size
, size
);
194 pixmap
= QPixmap(size
, size
);
195 pixmap
.fill(Qt::transparent
);
198 KPixmapModifier::scale(pixmap
, QSize(size
, size
) * dpr
);
201 painter
.drawPixmap(x
, y
, pixmap
);
204 if (x
>= dragPixmap
.width()) {
209 if (y
>= dragPixmap
.height()) {
217 void KFileItemListView::setHoverSequenceState(const QUrl
&itemUrl
, int seqIdx
)
219 if (m_modelRolesUpdater
) {
220 m_modelRolesUpdater
->setHoverSequenceState(itemUrl
, seqIdx
);
224 KItemListWidgetCreatorBase
*KFileItemListView::defaultWidgetCreator() const
226 return new KItemListWidgetCreator
<KFileItemListWidget
>();
229 void KFileItemListView::initializeItemListWidget(KItemListWidget
*item
)
231 KStandardItemListView::initializeItemListWidget(item
);
233 // Make sure that the item has an icon.
234 QHash
<QByteArray
, QVariant
> data
= item
->data();
235 if (!data
.contains("iconName") && data
["iconPixmap"].value
<QPixmap
>().isNull()) {
236 Q_ASSERT(qobject_cast
<KFileItemModel
*>(model()));
237 KFileItemModel
*fileItemModel
= static_cast<KFileItemModel
*>(model());
239 const KFileItem fileItem
= fileItemModel
->fileItem(item
->index());
240 QString iconName
= fileItem
.iconName();
241 if (!QIcon::hasThemeIcon(iconName
)) {
242 QMimeDatabase mimeDb
;
243 iconName
= mimeDb
.mimeTypeForName(fileItem
.mimetype()).genericIconName();
245 data
.insert("iconName", iconName
);
246 item
->setData(data
, {"iconName"});
250 void KFileItemListView::onPreviewsShownChanged(bool shown
)
255 void KFileItemListView::onItemLayoutChanged(ItemLayout current
, ItemLayout previous
)
257 KStandardItemListView::onItemLayoutChanged(current
, previous
);
258 triggerVisibleIndexRangeUpdate();
261 void KFileItemListView::onModelChanged(KItemModelBase
*current
, KItemModelBase
*previous
)
263 Q_ASSERT(qobject_cast
<KFileItemModel
*>(current
));
264 KStandardItemListView::onModelChanged(current
, previous
);
266 delete m_modelRolesUpdater
;
267 m_modelRolesUpdater
= nullptr;
270 m_modelRolesUpdater
= new KFileItemModelRolesUpdater(static_cast<KFileItemModel
*>(current
), this);
271 m_modelRolesUpdater
->setIconSize(availableIconSize());
272 m_modelRolesUpdater
->setScanDirectories(scanDirectories());
278 void KFileItemListView::onScrollOrientationChanged(Qt::Orientation current
, Qt::Orientation previous
)
280 KStandardItemListView::onScrollOrientationChanged(current
, previous
);
281 triggerVisibleIndexRangeUpdate();
284 void KFileItemListView::onItemSizeChanged(const QSizeF
¤t
, const QSizeF
&previous
)
288 triggerVisibleIndexRangeUpdate();
291 void KFileItemListView::onScrollOffsetChanged(qreal current
, qreal previous
)
293 KStandardItemListView::onScrollOffsetChanged(current
, previous
);
294 triggerVisibleIndexRangeUpdate();
297 void KFileItemListView::onVisibleRolesChanged(const QList
<QByteArray
> ¤t
, const QList
<QByteArray
> &previous
)
299 KStandardItemListView::onVisibleRolesChanged(current
, previous
);
303 void KFileItemListView::onStyleOptionChanged(const KItemListStyleOption
¤t
, const KItemListStyleOption
&previous
)
305 KStandardItemListView::onStyleOptionChanged(current
, previous
);
306 triggerIconSizeUpdate();
309 void KFileItemListView::onSupportsItemExpandingChanged(bool supportsExpanding
)
312 KStandardItemListView::onSupportsItemExpandingChanged(supportsExpanding
);
313 triggerVisibleIndexRangeUpdate();
316 void KFileItemListView::onTransactionBegin()
318 if (m_modelRolesUpdater
) {
319 m_modelRolesUpdater
->setPaused(true);
323 void KFileItemListView::onTransactionEnd()
325 if (!m_modelRolesUpdater
) {
329 // Only unpause the model-roles-updater if no timer is active. If one
330 // timer is still active the model-roles-updater will be unpaused later as
331 // soon as the timer has been exceeded.
332 const bool timerActive
= m_updateVisibleIndexRangeTimer
->isActive() || m_updateIconSizeTimer
->isActive();
334 m_modelRolesUpdater
->setPaused(false);
338 void KFileItemListView::resizeEvent(QGraphicsSceneResizeEvent
*event
)
340 KStandardItemListView::resizeEvent(event
);
341 triggerVisibleIndexRangeUpdate();
344 void KFileItemListView::slotItemsRemoved(const KItemRangeList
&itemRanges
)
346 KStandardItemListView::slotItemsRemoved(itemRanges
);
349 void KFileItemListView::slotSortRoleChanged(const QByteArray
¤t
, const QByteArray
&previous
)
351 const QByteArray sortRole
= model()->sortRole();
352 if (!visibleRoles().contains(sortRole
)) {
356 KStandardItemListView::slotSortRoleChanged(current
, previous
);
359 void KFileItemListView::triggerVisibleIndexRangeUpdate()
364 m_modelRolesUpdater
->setPaused(true);
366 // If the icon size has been changed recently, wait until
367 // m_updateIconSizeTimer expires.
368 if (!m_updateIconSizeTimer
->isActive()) {
369 m_updateVisibleIndexRangeTimer
->start();
373 void KFileItemListView::updateVisibleIndexRange()
375 if (!m_modelRolesUpdater
) {
379 const int index
= firstVisibleIndex();
380 const int count
= lastVisibleIndex() - index
+ 1;
381 m_modelRolesUpdater
->setMaximumVisibleItems(maximumVisibleItems());
382 m_modelRolesUpdater
->setVisibleIndexRange(index
, count
);
383 m_modelRolesUpdater
->setPaused(isTransactionActive());
386 void KFileItemListView::triggerIconSizeUpdate()
391 m_modelRolesUpdater
->setPaused(true);
392 m_updateIconSizeTimer
->start();
394 // The visible index range will be updated when m_updateIconSizeTimer expires.
395 // Stop m_updateVisibleIndexRangeTimer to prevent an expensive re-generation
396 // of all previews (note that the user might change the icon size again soon).
397 m_updateVisibleIndexRangeTimer
->stop();
400 void KFileItemListView::updateIconSize()
402 if (!m_modelRolesUpdater
) {
406 m_modelRolesUpdater
->setIconSize(availableIconSize());
408 // Update the visible index range (which has most likely changed after the
409 // icon size change) before unpausing m_modelRolesUpdater.
410 const int index
= firstVisibleIndex();
411 const int count
= lastVisibleIndex() - index
+ 1;
412 m_modelRolesUpdater
->setVisibleIndexRange(index
, count
);
414 m_modelRolesUpdater
->setPaused(isTransactionActive());
417 void KFileItemListView::applyRolesToModel()
423 Q_ASSERT(qobject_cast
<KFileItemModel
*>(model()));
424 KFileItemModel
*fileItemModel
= static_cast<KFileItemModel
*>(model());
426 // KFileItemModel does not distinct between "visible" and "invisible" roles.
427 // Add all roles that are mandatory for having a working KFileItemListView:
428 const auto visibleRoles
= this->visibleRoles();
429 auto roles
= QSet
<QByteArray
>(visibleRoles
.constBegin(), visibleRoles
.constEnd());
430 roles
.insert("iconPixmap");
431 roles
.insert("iconName");
432 roles
.insert("text");
433 roles
.insert("isDir");
434 roles
.insert("isLink");
435 roles
.insert("isHidden");
436 if (supportsItemExpanding()) {
437 roles
.insert("isExpanded");
438 roles
.insert("isExpandable");
439 roles
.insert("expandedParentsCount");
442 // Assure that the role that is used for sorting will be determined
443 roles
.insert(fileItemModel
->sortRole());
445 fileItemModel
->setRoles(roles
);
446 m_modelRolesUpdater
->setRoles(roles
);
449 QSize
KFileItemListView::availableIconSize() const
451 const KItemListStyleOption
&option
= styleOption();
452 const int iconSize
= option
.iconSize
;
453 if (itemLayout() == IconsLayout
) {
454 const int maxIconWidth
= itemSize().width() - 2 * option
.padding
;
455 return QSize(maxIconWidth
, iconSize
);
458 return QSize(iconSize
, iconSize
);