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>
21 // #define KFILEITEMLISTVIEW_DEBUG
24 // If the visible index range changes, KFileItemModelRolesUpdater is not
25 // informed immediately, but with a short delay. This ensures that scrolling
26 // always feels smooth and is not interrupted by icon loading (which can be
27 // quite expensive if a disk access is required to determine the final icon).
28 const int ShortInterval
= 50;
30 // If the icon size changes, a longer delay is used. This prevents that
31 // the expensive re-generation of all previews is triggered repeatedly when
32 // changing the zoom level.
33 const int LongInterval
= 300;
36 KFileItemListView::KFileItemListView(QGraphicsWidget
* parent
) :
37 KStandardItemListView(parent
),
38 m_modelRolesUpdater(nullptr),
39 m_updateVisibleIndexRangeTimer(nullptr),
40 m_updateIconSizeTimer(nullptr)
44 setScrollOrientation(Qt::Vertical
);
46 m_updateVisibleIndexRangeTimer
= new QTimer(this);
47 m_updateVisibleIndexRangeTimer
->setSingleShot(true);
48 m_updateVisibleIndexRangeTimer
->setInterval(ShortInterval
);
49 connect(m_updateVisibleIndexRangeTimer
, &QTimer::timeout
, this, &KFileItemListView::updateVisibleIndexRange
);
51 m_updateIconSizeTimer
= new QTimer(this);
52 m_updateIconSizeTimer
->setSingleShot(true);
53 m_updateIconSizeTimer
->setInterval(LongInterval
);
54 connect(m_updateIconSizeTimer
, &QTimer::timeout
, this, &KFileItemListView::updateIconSize
);
56 setVisibleRoles({"text"});
59 KFileItemListView::~KFileItemListView()
63 void KFileItemListView::setPreviewsShown(bool show
)
65 if (!m_modelRolesUpdater
) {
69 if (m_modelRolesUpdater
->previewsShown() != show
) {
71 m_modelRolesUpdater
->setPreviewsShown(show
);
72 onPreviewsShownChanged(show
);
77 bool KFileItemListView::previewsShown() const
79 return m_modelRolesUpdater
? m_modelRolesUpdater
->previewsShown() : false;
82 void KFileItemListView::setEnlargeSmallPreviews(bool enlarge
)
84 if (m_modelRolesUpdater
) {
85 m_modelRolesUpdater
->setEnlargeSmallPreviews(enlarge
);
89 bool KFileItemListView::enlargeSmallPreviews() const
91 return m_modelRolesUpdater
? m_modelRolesUpdater
->enlargeSmallPreviews() : false;
94 void KFileItemListView::setEnabledPlugins(const QStringList
& list
)
96 if (m_modelRolesUpdater
) {
97 m_modelRolesUpdater
->setEnabledPlugins(list
);
101 QStringList
KFileItemListView::enabledPlugins() const
103 return m_modelRolesUpdater
? m_modelRolesUpdater
->enabledPlugins() : QStringList();
106 void KFileItemListView::setLocalFileSizePreviewLimit(const qlonglong size
)
108 if (m_modelRolesUpdater
) {
109 m_modelRolesUpdater
->setLocalFileSizePreviewLimit(size
);
113 qlonglong
KFileItemListView::localFileSizePreviewLimit() const
115 return m_modelRolesUpdater
? m_modelRolesUpdater
->localFileSizePreviewLimit() : 0;
118 QPixmap
KFileItemListView::createDragPixmap(const KItemSet
& indexes
) const
124 const int itemCount
= indexes
.count();
125 Q_ASSERT(itemCount
> 0);
126 if (itemCount
== 1) {
127 return KItemListView::createDragPixmap(indexes
);
130 // If more than one item is dragged, align the items inside a
131 // rectangular grid. The maximum grid size is limited to 5 x 5 items.
134 if (itemCount
> 16) {
136 size
= KIconLoader::SizeSmall
;
137 } else if (itemCount
> 9) {
139 size
= KIconLoader::SizeSmallMedium
;
142 size
= KIconLoader::SizeMedium
;
145 if (itemCount
< xCount
) {
149 int yCount
= itemCount
/ xCount
;
150 if (itemCount
% xCount
!= 0) {
153 if (yCount
> xCount
) {
157 const qreal dpr
= scene()->views()[0]->devicePixelRatio();
158 // Draw the selected items into the grid cells.
159 QPixmap
dragPixmap(QSize(xCount
* size
+ xCount
, yCount
* size
+ yCount
) * dpr
);
160 dragPixmap
.setDevicePixelRatio(dpr
);
161 dragPixmap
.fill(Qt::transparent
);
163 QPainter
painter(&dragPixmap
);
167 for (int index
: indexes
) {
168 QPixmap pixmap
= model()->data(index
).value("iconPixmap").value
<QPixmap
>();
169 if (pixmap
.isNull()) {
170 QIcon icon
= QIcon::fromTheme(model()->data(index
).value("iconName").toString());
171 pixmap
= icon
.pixmap(size
, size
);
173 KPixmapModifier::scale(pixmap
, QSize(size
, size
) * dpr
);
176 painter
.drawPixmap(x
, y
, pixmap
);
179 if (x
>= dragPixmap
.width()) {
184 if (y
>= dragPixmap
.height()) {
192 KItemListWidgetCreatorBase
* KFileItemListView::defaultWidgetCreator() const
194 return new KItemListWidgetCreator
<KFileItemListWidget
>();
197 void KFileItemListView::initializeItemListWidget(KItemListWidget
* item
)
199 KStandardItemListView::initializeItemListWidget(item
);
201 // Make sure that the item has an icon.
202 QHash
<QByteArray
, QVariant
> data
= item
->data();
203 if (!data
.contains("iconName") && data
["iconPixmap"].value
<QPixmap
>().isNull()) {
204 Q_ASSERT(qobject_cast
<KFileItemModel
*>(model()));
205 KFileItemModel
* fileItemModel
= static_cast<KFileItemModel
*>(model());
207 const KFileItem fileItem
= fileItemModel
->fileItem(item
->index());
208 data
.insert("iconName", fileItem
.iconName());
209 item
->setData(data
, {"iconName"});
213 void KFileItemListView::onPreviewsShownChanged(bool shown
)
218 void KFileItemListView::onItemLayoutChanged(ItemLayout current
, ItemLayout previous
)
220 KStandardItemListView::onItemLayoutChanged(current
, previous
);
221 triggerVisibleIndexRangeUpdate();
224 void KFileItemListView::onModelChanged(KItemModelBase
* current
, KItemModelBase
* previous
)
226 Q_ASSERT(qobject_cast
<KFileItemModel
*>(current
));
227 KStandardItemListView::onModelChanged(current
, previous
);
229 delete m_modelRolesUpdater
;
230 m_modelRolesUpdater
= nullptr;
233 m_modelRolesUpdater
= new KFileItemModelRolesUpdater(static_cast<KFileItemModel
*>(current
), this);
234 m_modelRolesUpdater
->setIconSize(availableIconSize());
240 void KFileItemListView::onScrollOrientationChanged(Qt::Orientation current
, Qt::Orientation previous
)
242 KStandardItemListView::onScrollOrientationChanged(current
, previous
);
243 triggerVisibleIndexRangeUpdate();
246 void KFileItemListView::onItemSizeChanged(const QSizeF
& current
, const QSizeF
& previous
)
250 triggerVisibleIndexRangeUpdate();
253 void KFileItemListView::onScrollOffsetChanged(qreal current
, qreal previous
)
255 KStandardItemListView::onScrollOffsetChanged(current
, previous
);
256 triggerVisibleIndexRangeUpdate();
259 void KFileItemListView::onVisibleRolesChanged(const QList
<QByteArray
>& current
, const QList
<QByteArray
>& previous
)
261 KStandardItemListView::onVisibleRolesChanged(current
, previous
);
265 void KFileItemListView::onStyleOptionChanged(const KItemListStyleOption
& current
, const KItemListStyleOption
& previous
)
267 KStandardItemListView::onStyleOptionChanged(current
, previous
);
268 triggerIconSizeUpdate();
271 void KFileItemListView::onSupportsItemExpandingChanged(bool supportsExpanding
)
274 KStandardItemListView::onSupportsItemExpandingChanged(supportsExpanding
);
275 triggerVisibleIndexRangeUpdate();
278 void KFileItemListView::onTransactionBegin()
280 if (m_modelRolesUpdater
) {
281 m_modelRolesUpdater
->setPaused(true);
285 void KFileItemListView::onTransactionEnd()
287 if (!m_modelRolesUpdater
) {
291 // Only unpause the model-roles-updater if no timer is active. If one
292 // timer is still active the model-roles-updater will be unpaused later as
293 // soon as the timer has been exceeded.
294 const bool timerActive
= m_updateVisibleIndexRangeTimer
->isActive() ||
295 m_updateIconSizeTimer
->isActive();
297 m_modelRolesUpdater
->setPaused(false);
301 void KFileItemListView::resizeEvent(QGraphicsSceneResizeEvent
* event
)
303 KStandardItemListView::resizeEvent(event
);
304 triggerVisibleIndexRangeUpdate();
307 void KFileItemListView::slotItemsRemoved(const KItemRangeList
& itemRanges
)
309 KStandardItemListView::slotItemsRemoved(itemRanges
);
312 void KFileItemListView::slotSortRoleChanged(const QByteArray
& current
, const QByteArray
& previous
)
314 const QByteArray sortRole
= model()->sortRole();
315 if (!visibleRoles().contains(sortRole
)) {
319 KStandardItemListView::slotSortRoleChanged(current
, previous
);
322 void KFileItemListView::triggerVisibleIndexRangeUpdate()
327 m_modelRolesUpdater
->setPaused(true);
329 // If the icon size has been changed recently, wait until
330 // m_updateIconSizeTimer expires.
331 if (!m_updateIconSizeTimer
->isActive()) {
332 m_updateVisibleIndexRangeTimer
->start();
336 void KFileItemListView::updateVisibleIndexRange()
338 if (!m_modelRolesUpdater
) {
342 const int index
= firstVisibleIndex();
343 const int count
= lastVisibleIndex() - index
+ 1;
344 m_modelRolesUpdater
->setMaximumVisibleItems(maximumVisibleItems());
345 m_modelRolesUpdater
->setVisibleIndexRange(index
, count
);
346 m_modelRolesUpdater
->setPaused(isTransactionActive());
349 void KFileItemListView::triggerIconSizeUpdate()
354 m_modelRolesUpdater
->setPaused(true);
355 m_updateIconSizeTimer
->start();
357 // The visible index range will be updated when m_updateIconSizeTimer expires.
358 // Stop m_updateVisibleIndexRangeTimer to prevent an expensive re-generation
359 // of all previews (note that the user might change the icon size again soon).
360 m_updateVisibleIndexRangeTimer
->stop();
363 void KFileItemListView::updateIconSize()
365 if (!m_modelRolesUpdater
) {
369 m_modelRolesUpdater
->setIconSize(availableIconSize());
371 // Update the visible index range (which has most likely changed after the
372 // icon size change) before unpausing m_modelRolesUpdater.
373 const int index
= firstVisibleIndex();
374 const int count
= lastVisibleIndex() - index
+ 1;
375 m_modelRolesUpdater
->setVisibleIndexRange(index
, count
);
377 m_modelRolesUpdater
->setPaused(isTransactionActive());
380 void KFileItemListView::applyRolesToModel()
386 Q_ASSERT(qobject_cast
<KFileItemModel
*>(model()));
387 KFileItemModel
* fileItemModel
= static_cast<KFileItemModel
*>(model());
389 // KFileItemModel does not distinct between "visible" and "invisible" roles.
390 // Add all roles that are mandatory for having a working KFileItemListView:
391 const auto visibleRoles
= this->visibleRoles();
392 auto roles
= QSet
<QByteArray
>(visibleRoles
.constBegin(), visibleRoles
.constEnd());
393 roles
.insert("iconPixmap");
394 roles
.insert("iconName");
395 roles
.insert("text");
396 roles
.insert("isDir");
397 roles
.insert("isLink");
398 roles
.insert("isHidden");
399 if (supportsItemExpanding()) {
400 roles
.insert("isExpanded");
401 roles
.insert("isExpandable");
402 roles
.insert("expandedParentsCount");
405 // Assure that the role that is used for sorting will be determined
406 roles
.insert(fileItemModel
->sortRole());
408 fileItemModel
->setRoles(roles
);
409 m_modelRolesUpdater
->setRoles(roles
);
412 QSize
KFileItemListView::availableIconSize() const
414 const KItemListStyleOption
& option
= styleOption();
415 const int iconSize
= option
.iconSize
;
416 if (itemLayout() == IconsLayout
) {
417 const int maxIconWidth
= itemSize().width() - 2 * option
.padding
;
418 return QSize(maxIconWidth
, iconSize
);
421 return QSize(iconSize
, iconSize
);