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 "kitemlistgroupheader.h"
23 #include "kfileitemmodelrolesupdater.h"
24 #include "kfileitemlistwidget.h"
25 #include "kfileitemmodel.h"
27 #include <KStringHandler>
35 #define KFILEITEMLISTVIEW_DEBUG
38 const int ShortInterval
= 50;
39 const int LongInterval
= 300;
42 KFileItemListView::KFileItemListView(QGraphicsWidget
* parent
) :
43 KItemListView(parent
),
44 m_itemLayout(IconsLayout
),
45 m_modelRolesUpdater(0),
46 m_updateVisibleIndexRangeTimer(0),
47 m_updateIconSizeTimer(0),
48 m_minimumRolesWidths()
52 setScrollOrientation(Qt::Vertical
);
53 setWidgetCreator(new KItemListWidgetCreator
<KFileItemListWidget
>());
54 setGroupHeaderCreator(new KItemListGroupHeaderCreator
<KItemListGroupHeader
>());
56 m_updateVisibleIndexRangeTimer
= new QTimer(this);
57 m_updateVisibleIndexRangeTimer
->setSingleShot(true);
58 m_updateVisibleIndexRangeTimer
->setInterval(ShortInterval
);
59 connect(m_updateVisibleIndexRangeTimer
, SIGNAL(timeout()), this, SLOT(updateVisibleIndexRange()));
61 m_updateIconSizeTimer
= new QTimer(this);
62 m_updateIconSizeTimer
->setSingleShot(true);
63 m_updateIconSizeTimer
->setInterval(ShortInterval
);
64 connect(m_updateIconSizeTimer
, SIGNAL(timeout()), this, SLOT(updateIconSize()));
66 updateMinimumRolesWidths();
69 KFileItemListView::~KFileItemListView()
71 delete widgetCreator();
72 delete groupHeaderCreator();
74 delete m_modelRolesUpdater
;
75 m_modelRolesUpdater
= 0;
78 void KFileItemListView::setPreviewsShown(bool show
)
80 if (m_modelRolesUpdater
) {
81 m_modelRolesUpdater
->setPreviewShown(show
);
85 bool KFileItemListView::previewsShown() const
87 return m_modelRolesUpdater
->isPreviewShown();
90 void KFileItemListView::setItemLayout(Layout layout
)
92 if (m_itemLayout
!= layout
) {
93 m_itemLayout
= layout
;
94 updateLayoutOfVisibleItems();
98 KFileItemListView::Layout
KFileItemListView::itemLayout() const
103 QSizeF
KFileItemListView::itemSizeHint(int index
) const
105 const QHash
<QByteArray
, QVariant
> values
= model()->data(index
);
106 const KItemListStyleOption
& option
= styleOption();
107 const int additionalRolesCount
= qMax(visibleRoles().count() - 1, 0);
109 switch (m_itemLayout
) {
111 const QString text
= KStringHandler::preProcessWrap(values
["name"].toString());
113 const qreal maxWidth
= itemSize().width() - 2 * option
.margin
;
114 int textLinesCount
= 0;
117 // Calculate the number of lines required for wrapping the name
118 QTextOption
textOption(Qt::AlignHCenter
);
119 textOption
.setWrapMode(QTextOption::WrapAtWordBoundaryOrAnywhere
);
121 QTextLayout
layout(text
, option
.font
);
122 layout
.setTextOption(textOption
);
123 layout
.beginLayout();
124 while ((line
= layout
.createLine()).isValid()) {
125 line
.setLineWidth(maxWidth
);
126 line
.naturalTextWidth();
131 // Add one line for each additional information
132 textLinesCount
+= additionalRolesCount
;
134 const qreal height
= textLinesCount
* option
.fontMetrics
.height() +
137 return QSizeF(itemSize().width(), height
);
140 case CompactLayout
: {
141 // For each row exactly one role is shown. Calculate the maximum required width that is necessary
142 // to show all roles without horizontal clipping.
143 qreal maximumRequiredWidth
= 0.0;
145 foreach (const QByteArray
& role
, visibleRoles()) {
146 const QString text
= KFileItemListWidget::roleText(role
, values
);
147 const qreal requiredWidth
= option
.fontMetrics
.width(text
);
148 maximumRequiredWidth
= qMax(maximumRequiredWidth
, requiredWidth
);
151 const qreal width
= option
.margin
* 4 + option
.iconSize
+ maximumRequiredWidth
;
152 const qreal height
= option
.margin
* 2 + qMax(option
.iconSize
, (1 + additionalRolesCount
) * option
.fontMetrics
.height());
153 return QSizeF(width
, height
);
156 case DetailsLayout
: {
157 // The width will be determined dynamically by KFileItemListView::visibleRoleSizes()
158 const qreal height
= option
.margin
* 2 + qMax(option
.iconSize
, option
.fontMetrics
.height());
159 return QSizeF(-1, height
);
170 QHash
<QByteArray
, QSizeF
> KFileItemListView::visibleRolesSizes() const
175 QHash
<QByteArray
, QSizeF
> sizes
;
177 const int itemCount
= model()->count();
178 for (int i
= 0; i
< itemCount
; ++i
) {
179 foreach (const QByteArray
& visibleRole
, visibleRoles()) {
180 QSizeF maxSize
= sizes
.value(visibleRole
, QSizeF(0, 0));
181 const QSizeF itemSize
= visibleRoleSizeHint(i
, visibleRole
);
182 maxSize
= maxSize
.expandedTo(itemSize
);
183 sizes
.insert(visibleRole
, maxSize
);
186 if (i
> 100 && timer
.elapsed() > 200) {
187 // When having several thousands of items calculating the sizes can get
188 // very expensive. We accept a possibly too small role-size in favour
189 // of having no blocking user interface.
190 #ifdef KFILEITEMLISTVIEW_DEBUG
191 kDebug() << "Timer exceeded, stopped after" << i
<< "items";
197 // Stretch the width of the first role so that the full visible view-width
198 // is used to show all roles.
199 const qreal availableWidth
= size().width();
202 QHashIterator
<QByteArray
, QSizeF
> it(sizes
);
203 while (it
.hasNext()) {
205 usedWidth
+= it
.value().width();
208 if (usedWidth
< availableWidth
) {
209 const QByteArray role
= visibleRoles().first();
210 QSizeF firstRoleSize
= sizes
.value(role
);
211 firstRoleSize
.rwidth() += availableWidth
- usedWidth
;
212 sizes
.insert(role
, firstRoleSize
);
215 #ifdef KFILEITEMLISTVIEW_DEBUG
216 kDebug() << "[TIME] Calculated dynamic item size for " << itemCount
<< "items:" << timer
.elapsed();
221 QPixmap
KFileItemListView::createDragPixmap(const QSet
<int>& indexes
) const
226 QSetIterator
<int> it(indexes
);
227 while (it
.hasNext()) {
228 const int index
= it
.next();
229 // TODO: Only one item is considered currently
230 pixmap
= model()->data(index
).value("iconPixmap").value
<QPixmap
>();
231 if (pixmap
.isNull()) {
232 KIcon
icon(model()->data(index
).value("iconName").toString());
233 pixmap
= icon
.pixmap(itemSize().toSize());
241 void KFileItemListView::initializeItemListWidget(KItemListWidget
* item
)
243 KFileItemListWidget
* fileItemListWidget
= static_cast<KFileItemListWidget
*>(item
);
245 switch (m_itemLayout
) {
246 case IconsLayout
: fileItemListWidget
->setLayout(KFileItemListWidget::IconsLayout
); break;
247 case CompactLayout
: fileItemListWidget
->setLayout(KFileItemListWidget::CompactLayout
); break;
248 case DetailsLayout
: fileItemListWidget
->setLayout(KFileItemListWidget::DetailsLayout
); break;
249 default: Q_ASSERT(false); break;
253 bool KFileItemListView::itemSizeHintUpdateRequired(const QSet
<QByteArray
>& changedRoles
) const
255 // Even if the icons have a different size they are always aligned within
256 // the area defined by KItemStyleOption.iconSize and hence result in no
257 // change of the item-size.
258 const bool containsIconName
= changedRoles
.contains("iconName");
259 const bool containsIconPixmap
= changedRoles
.contains("iconPixmap");
260 const int count
= changedRoles
.count();
262 const bool iconChanged
= (containsIconName
&& containsIconPixmap
&& count
== 2) ||
263 (containsIconName
&& count
== 1) ||
264 (containsIconPixmap
&& count
== 1);
268 void KFileItemListView::onModelChanged(KItemModelBase
* current
, KItemModelBase
* previous
)
271 Q_ASSERT(qobject_cast
<KFileItemModel
*>(current
));
273 if (m_modelRolesUpdater
) {
274 delete m_modelRolesUpdater
;
277 m_modelRolesUpdater
= new KFileItemModelRolesUpdater(static_cast<KFileItemModel
*>(current
), this);
278 const int size
= styleOption().iconSize
;
279 m_modelRolesUpdater
->setIconSize(QSize(size
, size
));
282 void KFileItemListView::onScrollOrientationChanged(Qt::Orientation current
, Qt::Orientation previous
)
286 updateLayoutOfVisibleItems();
289 void KFileItemListView::onItemSizeChanged(const QSizeF
& current
, const QSizeF
& previous
)
293 triggerVisibleIndexRangeUpdate();
296 void KFileItemListView::onScrollOffsetChanged(qreal current
, qreal previous
)
300 triggerVisibleIndexRangeUpdate();
303 void KFileItemListView::onVisibleRolesChanged(const QList
<QByteArray
>& current
, const QList
<QByteArray
>& previous
)
307 Q_ASSERT(qobject_cast
<KFileItemModel
*>(model()));
308 KFileItemModel
* fileItemModel
= static_cast<KFileItemModel
*>(model());
310 // KFileItemModel does not distinct between "visible" and "invisible" roles.
311 // Add all roles that are mandatory for having a working KFileItemListView:
312 QSet
<QByteArray
> keys
= current
.toSet();
313 QSet
<QByteArray
> roles
= keys
;
314 roles
.insert("iconPixmap");
315 roles
.insert("iconName");
316 roles
.insert("name"); // TODO: just don't allow to disable it
317 roles
.insert("isDir");
318 if (m_itemLayout
== DetailsLayout
) {
319 roles
.insert("isExpanded");
320 roles
.insert("expansionLevel");
323 fileItemModel
->setRoles(roles
);
325 m_modelRolesUpdater
->setRoles(keys
);
328 void KFileItemListView::onStyleOptionChanged(const KItemListStyleOption
& current
, const KItemListStyleOption
& previous
)
332 triggerIconSizeUpdate();
335 void KFileItemListView::onTransactionBegin()
337 m_modelRolesUpdater
->setPaused(true);
340 void KFileItemListView::onTransactionEnd()
342 // Only unpause the model-roles-updater if no timer is active. If one
343 // timer is still active the model-roles-updater will be unpaused later as
344 // soon as the timer has been exceeded.
345 const bool timerActive
= m_updateVisibleIndexRangeTimer
->isActive() ||
346 m_updateIconSizeTimer
->isActive();
348 m_modelRolesUpdater
->setPaused(false);
352 void KFileItemListView::resizeEvent(QGraphicsSceneResizeEvent
* event
)
354 KItemListView::resizeEvent(event
);
355 triggerVisibleIndexRangeUpdate();
358 void KFileItemListView::slotItemsRemoved(const KItemRangeList
& itemRanges
)
360 KItemListView::slotItemsRemoved(itemRanges
);
361 updateTimersInterval();
364 void KFileItemListView::triggerVisibleIndexRangeUpdate()
366 m_modelRolesUpdater
->setPaused(true);
367 m_updateVisibleIndexRangeTimer
->start();
370 void KFileItemListView::updateVisibleIndexRange()
372 if (!m_modelRolesUpdater
) {
376 const int index
= firstVisibleIndex();
377 const int count
= lastVisibleIndex() - index
+ 1;
378 m_modelRolesUpdater
->setVisibleIndexRange(index
, count
);
380 if (m_updateIconSizeTimer
->isActive()) {
381 // If the icon-size update is pending do an immediate update
382 // of the icon-size before unpausing m_modelRolesUpdater. This prevents
383 // an unnecessary expensive recreation of all previews afterwards.
384 m_updateIconSizeTimer
->stop();
385 const KItemListStyleOption
& option
= styleOption();
386 m_modelRolesUpdater
->setIconSize(QSize(option
.iconSize
, option
.iconSize
));
389 m_modelRolesUpdater
->setPaused(isTransactionActive());
390 updateTimersInterval();
393 void KFileItemListView::triggerIconSizeUpdate()
395 m_modelRolesUpdater
->setPaused(true);
396 m_updateIconSizeTimer
->start();
399 void KFileItemListView::updateIconSize()
401 if (!m_modelRolesUpdater
) {
405 const KItemListStyleOption
& option
= styleOption();
406 m_modelRolesUpdater
->setIconSize(QSize(option
.iconSize
, option
.iconSize
));
408 if (m_updateVisibleIndexRangeTimer
->isActive()) {
409 // If the visibility-index-range update is pending do an immediate update
410 // of the range before unpausing m_modelRolesUpdater. This prevents
411 // an unnecessary expensive recreation of all previews afterwards.
412 m_updateVisibleIndexRangeTimer
->stop();
413 const int index
= firstVisibleIndex();
414 const int count
= lastVisibleIndex() - index
+ 1;
415 m_modelRolesUpdater
->setVisibleIndexRange(index
, count
);
418 m_modelRolesUpdater
->setPaused(isTransactionActive());
419 updateTimersInterval();
422 QSizeF
KFileItemListView::visibleRoleSizeHint(int index
, const QByteArray
& role
) const
424 const KItemListStyleOption
& option
= styleOption();
426 qreal width
= m_minimumRolesWidths
.value(role
, 0);
427 const qreal height
= option
.margin
* 2 + option
.fontMetrics
.height();
429 const QHash
<QByteArray
, QVariant
> values
= model()->data(index
);
430 const QString text
= KFileItemListWidget::roleText(role
, values
);
431 if (!text
.isEmpty()) {
432 const qreal columnMargin
= option
.margin
* 3;
433 width
= qMax(width
, qreal(2 * columnMargin
+ option
.fontMetrics
.width(text
)));
436 if (role
== "name") {
437 // Increase the width by the expansion-toggle and the current expansion level
438 const int expansionLevel
= values
.value("expansionLevel", 0).toInt();
439 width
+= option
.margin
+ expansionLevel
* itemSize().height() + KIconLoader::SizeSmall
;
441 // Increase the width by the required space for the icon
442 width
+= option
.margin
* 2 + option
.iconSize
;
445 return QSizeF(width
, height
);
448 void KFileItemListView::updateLayoutOfVisibleItems()
450 foreach (KItemListWidget
* widget
, visibleItemListWidgets()) {
451 initializeItemListWidget(widget
);
453 triggerVisibleIndexRangeUpdate();
456 void KFileItemListView::updateTimersInterval()
462 // The ShortInterval is used for cases like switching the directory: If the
463 // model is empty and filled later the creation of the previews should be done
464 // as soon as possible. The LongInterval is used when the model already contains
465 // items and assures that operations like zooming don't result in too many temporary
466 // recreations of the previews.
468 const int interval
= (model()->count() <= 0) ? ShortInterval
: LongInterval
;
469 m_updateVisibleIndexRangeTimer
->setInterval(interval
);
470 m_updateIconSizeTimer
->setInterval(interval
);
473 void KFileItemListView::updateMinimumRolesWidths()
475 m_minimumRolesWidths
.clear();
477 const KItemListStyleOption
& option
= styleOption();
478 const QString sizeText
= QLatin1String("888888") + i18nc("@item:intable", "items");
479 m_minimumRolesWidths
.insert("size", option
.fontMetrics
.width(sizeText
));
482 #include "kfileitemlistview.moc"