1 /***************************************************************************
2 * Copyright (C) 2011 by Peter Penz <peter.penz19@gmail.com> *
4 * This program is free software; you can redistribute it and/or modify *
5 * it under the terms of the GNU General Public License as published by *
6 * the Free Software Foundation; either version 2 of the License, or *
7 * (at your option) any later version. *
9 * This program is distributed in the hope that it will be useful, *
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
12 * GNU General Public License for more details. *
14 * You should have received a copy of the GNU General Public License *
15 * along with this program; if not, write to the *
16 * Free Software Foundation, Inc., *
17 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA *
18 ***************************************************************************/
20 #include "kfileitemlistview.h"
22 #include "kfileitemlistwidget.h"
23 #include "kfileitemmodel.h"
24 #include "kfileitemmodelrolesupdater.h"
25 #include "private/kpixmapmodifier.h"
27 #include <KIconLoader>
29 #include <QGraphicsScene>
30 #include <QGraphicsView>
34 // #define KFILEITEMLISTVIEW_DEBUG
37 // If the visible index range changes, KFileItemModelRolesUpdater is not
38 // informed immediately, but with a short delay. This ensures that scrolling
39 // always feels smooth and is not interrupted by icon loading (which can be
40 // quite expensive if a disk access is required to determine the final icon).
41 const int ShortInterval
= 50;
43 // If the icon size changes, a longer delay is used. This prevents that
44 // the expensive re-generation of all previews is triggered repeatedly when
45 // changing the zoom level.
46 const int LongInterval
= 300;
49 KFileItemListView::KFileItemListView(QGraphicsWidget
* parent
) :
50 KStandardItemListView(parent
),
51 m_modelRolesUpdater(nullptr),
52 m_updateVisibleIndexRangeTimer(nullptr),
53 m_updateIconSizeTimer(nullptr)
57 setScrollOrientation(Qt::Vertical
);
59 m_updateVisibleIndexRangeTimer
= new QTimer(this);
60 m_updateVisibleIndexRangeTimer
->setSingleShot(true);
61 m_updateVisibleIndexRangeTimer
->setInterval(ShortInterval
);
62 connect(m_updateVisibleIndexRangeTimer
, &QTimer::timeout
, this, &KFileItemListView::updateVisibleIndexRange
);
64 m_updateIconSizeTimer
= new QTimer(this);
65 m_updateIconSizeTimer
->setSingleShot(true);
66 m_updateIconSizeTimer
->setInterval(LongInterval
);
67 connect(m_updateIconSizeTimer
, &QTimer::timeout
, this, &KFileItemListView::updateIconSize
);
69 setVisibleRoles({"text"});
72 KFileItemListView::~KFileItemListView()
76 void KFileItemListView::setPreviewsShown(bool show
)
78 if (!m_modelRolesUpdater
) {
82 if (m_modelRolesUpdater
->previewsShown() != show
) {
84 m_modelRolesUpdater
->setPreviewsShown(show
);
85 onPreviewsShownChanged(show
);
90 bool KFileItemListView::previewsShown() const
92 return m_modelRolesUpdater
? m_modelRolesUpdater
->previewsShown() : false;
95 void KFileItemListView::setEnlargeSmallPreviews(bool enlarge
)
97 if (m_modelRolesUpdater
) {
98 m_modelRolesUpdater
->setEnlargeSmallPreviews(enlarge
);
102 bool KFileItemListView::enlargeSmallPreviews() const
104 return m_modelRolesUpdater
? m_modelRolesUpdater
->enlargeSmallPreviews() : false;
107 void KFileItemListView::setEnabledPlugins(const QStringList
& list
)
109 if (m_modelRolesUpdater
) {
110 m_modelRolesUpdater
->setEnabledPlugins(list
);
114 QStringList
KFileItemListView::enabledPlugins() const
116 return m_modelRolesUpdater
? m_modelRolesUpdater
->enabledPlugins() : QStringList();
119 void KFileItemListView::setLocalFileSizePreviewLimit(const qlonglong size
)
121 if (m_modelRolesUpdater
) {
122 m_modelRolesUpdater
->setLocalFileSizePreviewLimit(size
);
126 qlonglong
KFileItemListView::localFileSizePreviewLimit() const
128 return m_modelRolesUpdater
? m_modelRolesUpdater
->localFileSizePreviewLimit() : 0;
131 QPixmap
KFileItemListView::createDragPixmap(const KItemSet
& indexes
) const
137 const int itemCount
= indexes
.count();
138 Q_ASSERT(itemCount
> 0);
139 if (itemCount
== 1) {
140 return KItemListView::createDragPixmap(indexes
);
143 // If more than one item is dragged, align the items inside a
144 // rectangular grid. The maximum grid size is limited to 5 x 5 items.
147 if (itemCount
> 16) {
149 size
= KIconLoader::SizeSmall
;
150 } else if (itemCount
> 9) {
152 size
= KIconLoader::SizeSmallMedium
;
155 size
= KIconLoader::SizeMedium
;
158 if (itemCount
< xCount
) {
162 int yCount
= itemCount
/ xCount
;
163 if (itemCount
% xCount
!= 0) {
166 if (yCount
> xCount
) {
170 const qreal dpr
= scene()->views()[0]->devicePixelRatio();
171 // Draw the selected items into the grid cells.
172 QPixmap
dragPixmap(QSize(xCount
* size
+ xCount
, yCount
* size
+ yCount
) * dpr
);
173 dragPixmap
.setDevicePixelRatio(dpr
);
174 dragPixmap
.fill(Qt::transparent
);
176 QPainter
painter(&dragPixmap
);
180 for (int index
: indexes
) {
181 QPixmap pixmap
= model()->data(index
).value("iconPixmap").value
<QPixmap
>();
182 if (pixmap
.isNull()) {
183 QIcon icon
= QIcon::fromTheme(model()->data(index
).value("iconName").toString());
184 pixmap
= icon
.pixmap(size
, size
);
186 KPixmapModifier::scale(pixmap
, QSize(size
, size
) * dpr
);
189 painter
.drawPixmap(x
, y
, pixmap
);
192 if (x
>= dragPixmap
.width()) {
197 if (y
>= dragPixmap
.height()) {
205 KItemListWidgetCreatorBase
* KFileItemListView::defaultWidgetCreator() const
207 return new KItemListWidgetCreator
<KFileItemListWidget
>();
210 void KFileItemListView::initializeItemListWidget(KItemListWidget
* item
)
212 KStandardItemListView::initializeItemListWidget(item
);
214 // Make sure that the item has an icon.
215 QHash
<QByteArray
, QVariant
> data
= item
->data();
216 if (!data
.contains("iconName") && data
["iconPixmap"].value
<QPixmap
>().isNull()) {
217 Q_ASSERT(qobject_cast
<KFileItemModel
*>(model()));
218 KFileItemModel
* fileItemModel
= static_cast<KFileItemModel
*>(model());
220 const KFileItem fileItem
= fileItemModel
->fileItem(item
->index());
221 data
.insert("iconName", fileItem
.iconName());
222 item
->setData(data
, {"iconName"});
226 void KFileItemListView::onPreviewsShownChanged(bool shown
)
231 void KFileItemListView::onItemLayoutChanged(ItemLayout current
, ItemLayout previous
)
233 KStandardItemListView::onItemLayoutChanged(current
, previous
);
234 triggerVisibleIndexRangeUpdate();
237 void KFileItemListView::onModelChanged(KItemModelBase
* current
, KItemModelBase
* previous
)
239 Q_ASSERT(qobject_cast
<KFileItemModel
*>(current
));
240 KStandardItemListView::onModelChanged(current
, previous
);
242 delete m_modelRolesUpdater
;
243 m_modelRolesUpdater
= nullptr;
246 m_modelRolesUpdater
= new KFileItemModelRolesUpdater(static_cast<KFileItemModel
*>(current
), this);
247 m_modelRolesUpdater
->setIconSize(availableIconSize());
253 void KFileItemListView::onScrollOrientationChanged(Qt::Orientation current
, Qt::Orientation previous
)
255 KStandardItemListView::onScrollOrientationChanged(current
, previous
);
256 triggerVisibleIndexRangeUpdate();
259 void KFileItemListView::onItemSizeChanged(const QSizeF
& current
, const QSizeF
& previous
)
263 triggerVisibleIndexRangeUpdate();
266 void KFileItemListView::onScrollOffsetChanged(qreal current
, qreal previous
)
268 KStandardItemListView::onScrollOffsetChanged(current
, previous
);
269 triggerVisibleIndexRangeUpdate();
272 void KFileItemListView::onVisibleRolesChanged(const QList
<QByteArray
>& current
, const QList
<QByteArray
>& previous
)
274 KStandardItemListView::onVisibleRolesChanged(current
, previous
);
278 void KFileItemListView::onStyleOptionChanged(const KItemListStyleOption
& current
, const KItemListStyleOption
& previous
)
280 KStandardItemListView::onStyleOptionChanged(current
, previous
);
281 triggerIconSizeUpdate();
284 void KFileItemListView::onSupportsItemExpandingChanged(bool supportsExpanding
)
287 KStandardItemListView::onSupportsItemExpandingChanged(supportsExpanding
);
288 triggerVisibleIndexRangeUpdate();
291 void KFileItemListView::onTransactionBegin()
293 if (m_modelRolesUpdater
) {
294 m_modelRolesUpdater
->setPaused(true);
298 void KFileItemListView::onTransactionEnd()
300 if (!m_modelRolesUpdater
) {
304 // Only unpause the model-roles-updater if no timer is active. If one
305 // timer is still active the model-roles-updater will be unpaused later as
306 // soon as the timer has been exceeded.
307 const bool timerActive
= m_updateVisibleIndexRangeTimer
->isActive() ||
308 m_updateIconSizeTimer
->isActive();
310 m_modelRolesUpdater
->setPaused(false);
314 void KFileItemListView::resizeEvent(QGraphicsSceneResizeEvent
* event
)
316 KStandardItemListView::resizeEvent(event
);
317 triggerVisibleIndexRangeUpdate();
320 void KFileItemListView::slotItemsRemoved(const KItemRangeList
& itemRanges
)
322 KStandardItemListView::slotItemsRemoved(itemRanges
);
325 void KFileItemListView::slotSortRoleChanged(const QByteArray
& current
, const QByteArray
& previous
)
327 const QByteArray sortRole
= model()->sortRole();
328 if (!visibleRoles().contains(sortRole
)) {
332 KStandardItemListView::slotSortRoleChanged(current
, previous
);
335 void KFileItemListView::triggerVisibleIndexRangeUpdate()
340 m_modelRolesUpdater
->setPaused(true);
342 // If the icon size has been changed recently, wait until
343 // m_updateIconSizeTimer expires.
344 if (!m_updateIconSizeTimer
->isActive()) {
345 m_updateVisibleIndexRangeTimer
->start();
349 void KFileItemListView::updateVisibleIndexRange()
351 if (!m_modelRolesUpdater
) {
355 const int index
= firstVisibleIndex();
356 const int count
= lastVisibleIndex() - index
+ 1;
357 m_modelRolesUpdater
->setMaximumVisibleItems(maximumVisibleItems());
358 m_modelRolesUpdater
->setVisibleIndexRange(index
, count
);
359 m_modelRolesUpdater
->setPaused(isTransactionActive());
362 void KFileItemListView::triggerIconSizeUpdate()
367 m_modelRolesUpdater
->setPaused(true);
368 m_updateIconSizeTimer
->start();
370 // The visible index range will be updated when m_updateIconSizeTimer expires.
371 // Stop m_updateVisibleIndexRangeTimer to prevent an expensive re-generation
372 // of all previews (note that the user might change the icon size again soon).
373 m_updateVisibleIndexRangeTimer
->stop();
376 void KFileItemListView::updateIconSize()
378 if (!m_modelRolesUpdater
) {
382 m_modelRolesUpdater
->setIconSize(availableIconSize());
384 // Update the visible index range (which has most likely changed after the
385 // icon size change) before unpausing m_modelRolesUpdater.
386 const int index
= firstVisibleIndex();
387 const int count
= lastVisibleIndex() - index
+ 1;
388 m_modelRolesUpdater
->setVisibleIndexRange(index
, count
);
390 m_modelRolesUpdater
->setPaused(isTransactionActive());
393 void KFileItemListView::applyRolesToModel()
399 Q_ASSERT(qobject_cast
<KFileItemModel
*>(model()));
400 KFileItemModel
* fileItemModel
= static_cast<KFileItemModel
*>(model());
402 // KFileItemModel does not distinct between "visible" and "invisible" roles.
403 // Add all roles that are mandatory for having a working KFileItemListView:
404 QSet
<QByteArray
> roles
= visibleRoles().toSet();
405 roles
.insert("iconPixmap");
406 roles
.insert("iconName");
407 roles
.insert("text");
408 roles
.insert("isDir");
409 roles
.insert("isLink");
410 roles
.insert("isHidden");
411 if (supportsItemExpanding()) {
412 roles
.insert("isExpanded");
413 roles
.insert("isExpandable");
414 roles
.insert("expandedParentsCount");
417 // Assure that the role that is used for sorting will be determined
418 roles
.insert(fileItemModel
->sortRole());
420 fileItemModel
->setRoles(roles
);
421 m_modelRolesUpdater
->setRoles(roles
);
424 QSize
KFileItemListView::availableIconSize() const
426 const KItemListStyleOption
& option
= styleOption();
427 const int iconSize
= option
.iconSize
;
428 if (itemLayout() == IconsLayout
) {
429 const int maxIconWidth
= itemSize().width() - 2 * option
.padding
;
430 return QSize(maxIconWidth
, iconSize
);
433 return QSize(iconSize
, iconSize
);