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)
47 setScrollOrientation(Qt::Vertical
);
49 m_updateVisibleIndexRangeTimer
= new QTimer(this);
50 m_updateVisibleIndexRangeTimer
->setSingleShot(true);
51 m_updateVisibleIndexRangeTimer
->setInterval(ShortInterval
);
52 connect(m_updateVisibleIndexRangeTimer
, &QTimer::timeout
, this, &KFileItemListView::updateVisibleIndexRange
);
54 m_updateIconSizeTimer
= new QTimer(this);
55 m_updateIconSizeTimer
->setSingleShot(true);
56 m_updateIconSizeTimer
->setInterval(LongInterval
);
57 connect(m_updateIconSizeTimer
, &QTimer::timeout
, this, &KFileItemListView::updateIconSize
);
59 setVisibleRoles({"text"});
62 KFileItemListView::~KFileItemListView()
66 void KFileItemListView::setPreviewsShown(bool show
)
68 if (!m_modelRolesUpdater
) {
72 if (m_modelRolesUpdater
->previewsShown() != show
) {
74 m_modelRolesUpdater
->setPreviewsShown(show
);
75 onPreviewsShownChanged(show
);
80 bool KFileItemListView::previewsShown() const
82 return m_modelRolesUpdater
? m_modelRolesUpdater
->previewsShown() : false;
85 void KFileItemListView::setEnlargeSmallPreviews(bool enlarge
)
87 if (m_modelRolesUpdater
) {
88 m_modelRolesUpdater
->setEnlargeSmallPreviews(enlarge
);
92 bool KFileItemListView::enlargeSmallPreviews() const
94 return m_modelRolesUpdater
? m_modelRolesUpdater
->enlargeSmallPreviews() : false;
97 void KFileItemListView::setEnabledPlugins(const QStringList
&list
)
99 if (m_modelRolesUpdater
) {
100 m_modelRolesUpdater
->setEnabledPlugins(list
);
104 QStringList
KFileItemListView::enabledPlugins() const
106 return m_modelRolesUpdater
? m_modelRolesUpdater
->enabledPlugins() : QStringList();
109 void KFileItemListView::setLocalFileSizePreviewLimit(const qlonglong size
)
111 if (m_modelRolesUpdater
) {
112 m_modelRolesUpdater
->setLocalFileSizePreviewLimit(size
);
116 qlonglong
KFileItemListView::localFileSizePreviewLimit() const
118 return m_modelRolesUpdater
? m_modelRolesUpdater
->localFileSizePreviewLimit() : 0;
121 QPixmap
KFileItemListView::createDragPixmap(const KItemSet
&indexes
) const
127 const int itemCount
= indexes
.count();
128 Q_ASSERT(itemCount
> 0);
129 if (itemCount
== 1) {
130 return KItemListView::createDragPixmap(indexes
);
133 // If more than one item is dragged, align the items inside a
134 // rectangular grid. The maximum grid size is limited to 5 x 5 items.
137 if (itemCount
> 16) {
139 size
= KIconLoader::SizeSmall
;
140 } else if (itemCount
> 9) {
142 size
= KIconLoader::SizeSmallMedium
;
145 size
= KIconLoader::SizeMedium
;
148 if (itemCount
< xCount
) {
152 int yCount
= itemCount
/ xCount
;
153 if (itemCount
% xCount
!= 0) {
156 if (yCount
> xCount
) {
160 const qreal dpr
= scene()->views()[0]->devicePixelRatio();
161 // Draw the selected items into the grid cells.
162 QPixmap
dragPixmap(QSize(xCount
* size
+ xCount
, yCount
* size
+ yCount
) * dpr
);
163 dragPixmap
.setDevicePixelRatio(dpr
);
164 dragPixmap
.fill(Qt::transparent
);
166 QPainter
painter(&dragPixmap
);
170 for (int index
: indexes
) {
171 QPixmap pixmap
= model()->data(index
).value("iconPixmap").value
<QPixmap
>();
172 if (pixmap
.isNull()) {
173 QIcon icon
= QIcon::fromTheme(model()->data(index
).value("iconName").toString());
175 icon
= QIcon::fromTheme(QStringLiteral("unknown"));
177 if (!icon
.isNull()) {
178 pixmap
= icon
.pixmap(size
, size
);
180 pixmap
= QPixmap(size
, size
);
181 pixmap
.fill(Qt::transparent
);
184 KPixmapModifier::scale(pixmap
, QSize(size
, size
) * dpr
);
187 painter
.drawPixmap(x
, y
, pixmap
);
190 if (x
>= dragPixmap
.width()) {
195 if (y
>= dragPixmap
.height()) {
203 void KFileItemListView::setHoverSequenceState(const QUrl
&itemUrl
, int seqIdx
)
205 if (m_modelRolesUpdater
) {
206 m_modelRolesUpdater
->setHoverSequenceState(itemUrl
, seqIdx
);
210 KItemListWidgetCreatorBase
*KFileItemListView::defaultWidgetCreator() const
212 return new KItemListWidgetCreator
<KFileItemListWidget
>();
215 void KFileItemListView::initializeItemListWidget(KItemListWidget
*item
)
217 KStandardItemListView::initializeItemListWidget(item
);
219 // Make sure that the item has an icon.
220 QHash
<QByteArray
, QVariant
> data
= item
->data();
221 if (!data
.contains("iconName") && data
["iconPixmap"].value
<QPixmap
>().isNull()) {
222 Q_ASSERT(qobject_cast
<KFileItemModel
*>(model()));
223 KFileItemModel
*fileItemModel
= static_cast<KFileItemModel
*>(model());
225 const KFileItem fileItem
= fileItemModel
->fileItem(item
->index());
226 QString iconName
= fileItem
.iconName();
227 if (!QIcon::hasThemeIcon(iconName
)) {
228 QMimeDatabase mimeDb
;
229 iconName
= mimeDb
.mimeTypeForName(fileItem
.mimetype()).genericIconName();
231 data
.insert("iconName", iconName
);
232 item
->setData(data
, {"iconName"});
236 void KFileItemListView::onPreviewsShownChanged(bool shown
)
241 void KFileItemListView::onItemLayoutChanged(ItemLayout current
, ItemLayout previous
)
243 KStandardItemListView::onItemLayoutChanged(current
, previous
);
244 triggerVisibleIndexRangeUpdate();
247 void KFileItemListView::onModelChanged(KItemModelBase
*current
, KItemModelBase
*previous
)
249 Q_ASSERT(qobject_cast
<KFileItemModel
*>(current
));
250 KStandardItemListView::onModelChanged(current
, previous
);
252 delete m_modelRolesUpdater
;
253 m_modelRolesUpdater
= nullptr;
256 m_modelRolesUpdater
= new KFileItemModelRolesUpdater(static_cast<KFileItemModel
*>(current
), this);
257 m_modelRolesUpdater
->setIconSize(availableIconSize());
263 void KFileItemListView::onScrollOrientationChanged(Qt::Orientation current
, Qt::Orientation previous
)
265 KStandardItemListView::onScrollOrientationChanged(current
, previous
);
266 triggerVisibleIndexRangeUpdate();
269 void KFileItemListView::onItemSizeChanged(const QSizeF
¤t
, const QSizeF
&previous
)
273 triggerVisibleIndexRangeUpdate();
276 void KFileItemListView::onScrollOffsetChanged(qreal current
, qreal previous
)
278 KStandardItemListView::onScrollOffsetChanged(current
, previous
);
279 triggerVisibleIndexRangeUpdate();
282 void KFileItemListView::onVisibleRolesChanged(const QList
<QByteArray
> ¤t
, const QList
<QByteArray
> &previous
)
284 KStandardItemListView::onVisibleRolesChanged(current
, previous
);
288 void KFileItemListView::onStyleOptionChanged(const KItemListStyleOption
¤t
, const KItemListStyleOption
&previous
)
290 KStandardItemListView::onStyleOptionChanged(current
, previous
);
291 triggerIconSizeUpdate();
294 void KFileItemListView::onSupportsItemExpandingChanged(bool supportsExpanding
)
297 KStandardItemListView::onSupportsItemExpandingChanged(supportsExpanding
);
298 triggerVisibleIndexRangeUpdate();
301 void KFileItemListView::onTransactionBegin()
303 if (m_modelRolesUpdater
) {
304 m_modelRolesUpdater
->setPaused(true);
308 void KFileItemListView::onTransactionEnd()
310 if (!m_modelRolesUpdater
) {
314 // Only unpause the model-roles-updater if no timer is active. If one
315 // timer is still active the model-roles-updater will be unpaused later as
316 // soon as the timer has been exceeded.
317 const bool timerActive
= m_updateVisibleIndexRangeTimer
->isActive() || m_updateIconSizeTimer
->isActive();
319 m_modelRolesUpdater
->setPaused(false);
323 void KFileItemListView::resizeEvent(QGraphicsSceneResizeEvent
*event
)
325 KStandardItemListView::resizeEvent(event
);
326 triggerVisibleIndexRangeUpdate();
329 void KFileItemListView::focusInEvent(QFocusEvent
*event
)
332 updateSelectedWidgets();
335 void KFileItemListView::focusOutEvent(QFocusEvent
*event
)
338 updateSelectedWidgets();
341 void KFileItemListView::updateSelectedWidgets()
343 const auto visibleWidgets
= visibleItemListWidgets();
344 for (KItemListWidget
*widget
: visibleWidgets
) {
345 if (widget
->isSelected()) {
346 auto w
= qobject_cast
<KFileItemListWidget
*>(widget
);
354 void KFileItemListView::slotItemsRemoved(const KItemRangeList
&itemRanges
)
356 KStandardItemListView::slotItemsRemoved(itemRanges
);
359 void KFileItemListView::slotSortRoleChanged(const QByteArray
¤t
, const QByteArray
&previous
)
361 const QByteArray sortRole
= model()->sortRole();
362 if (!visibleRoles().contains(sortRole
)) {
366 KStandardItemListView::slotSortRoleChanged(current
, previous
);
369 void KFileItemListView::triggerVisibleIndexRangeUpdate()
374 m_modelRolesUpdater
->setPaused(true);
376 // If the icon size has been changed recently, wait until
377 // m_updateIconSizeTimer expires.
378 if (!m_updateIconSizeTimer
->isActive()) {
379 m_updateVisibleIndexRangeTimer
->start();
383 void KFileItemListView::updateVisibleIndexRange()
385 if (!m_modelRolesUpdater
) {
389 const int index
= firstVisibleIndex();
390 const int count
= lastVisibleIndex() - index
+ 1;
391 m_modelRolesUpdater
->setMaximumVisibleItems(maximumVisibleItems());
392 m_modelRolesUpdater
->setVisibleIndexRange(index
, count
);
393 m_modelRolesUpdater
->setPaused(isTransactionActive());
396 void KFileItemListView::triggerIconSizeUpdate()
401 m_modelRolesUpdater
->setPaused(true);
402 m_updateIconSizeTimer
->start();
404 // The visible index range will be updated when m_updateIconSizeTimer expires.
405 // Stop m_updateVisibleIndexRangeTimer to prevent an expensive re-generation
406 // of all previews (note that the user might change the icon size again soon).
407 m_updateVisibleIndexRangeTimer
->stop();
410 void KFileItemListView::updateIconSize()
412 if (!m_modelRolesUpdater
) {
416 m_modelRolesUpdater
->setIconSize(availableIconSize());
418 // Update the visible index range (which has most likely changed after the
419 // icon size change) before unpausing m_modelRolesUpdater.
420 const int index
= firstVisibleIndex();
421 const int count
= lastVisibleIndex() - index
+ 1;
422 m_modelRolesUpdater
->setVisibleIndexRange(index
, count
);
424 m_modelRolesUpdater
->setPaused(isTransactionActive());
427 void KFileItemListView::applyRolesToModel()
433 Q_ASSERT(qobject_cast
<KFileItemModel
*>(model()));
434 KFileItemModel
*fileItemModel
= static_cast<KFileItemModel
*>(model());
436 // KFileItemModel does not distinct between "visible" and "invisible" roles.
437 // Add all roles that are mandatory for having a working KFileItemListView:
438 const auto visibleRoles
= this->visibleRoles();
439 auto roles
= QSet
<QByteArray
>(visibleRoles
.constBegin(), visibleRoles
.constEnd());
440 roles
.insert("iconPixmap");
441 roles
.insert("iconName");
442 roles
.insert("text");
443 roles
.insert("isDir");
444 roles
.insert("isLink");
445 roles
.insert("isHidden");
446 if (supportsItemExpanding()) {
447 roles
.insert("isExpanded");
448 roles
.insert("isExpandable");
449 roles
.insert("expandedParentsCount");
452 // Assure that the role that is used for sorting will be determined
453 roles
.insert(fileItemModel
->sortRole());
455 fileItemModel
->setRoles(roles
);
456 m_modelRolesUpdater
->setRoles(roles
);
459 QSize
KFileItemListView::availableIconSize() const
461 const KItemListStyleOption
&option
= styleOption();
462 const int iconSize
= option
.iconSize
;
463 if (itemLayout() == IconsLayout
) {
464 const int maxIconWidth
= itemSize().width() - 2 * option
.padding
;
465 return QSize(maxIconWidth
, iconSize
);
468 return QSize(iconSize
, iconSize
);
471 #include "moc_kfileitemlistview.cpp"