]> cloud.milkyroute.net Git - dolphin.git/blob - src/kitemviews/kfileitemlistview.cpp
Merge remote-tracking branch 'origin/master' into frameworks
[dolphin.git] / src / kitemviews / kfileitemlistview.cpp
1 /***************************************************************************
2 * Copyright (C) 2011 by Peter Penz <peter.penz19@gmail.com> *
3 * *
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. *
8 * *
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. *
13 * *
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 ***************************************************************************/
19
20 #include "kfileitemlistview.h"
21
22 #include "kfileitemmodelrolesupdater.h"
23 #include "kfileitemlistwidget.h"
24 #include "kfileitemmodel.h"
25 #include <KLocale>
26 #include <KStringHandler>
27 #include "private/kpixmapmodifier.h"
28
29 #include <KDebug>
30 #include <KIcon>
31 #include <KTextEdit>
32 #include <KIconLoader>
33 #include <KDateTime>
34
35 #include <QPainter>
36 #include <QTextLine>
37 #include <QTimer>
38
39 // #define KFILEITEMLISTVIEW_DEBUG
40
41 namespace {
42 // If the visible index range changes, KFileItemModelRolesUpdater is not
43 // informed immediatetly, but with a short delay. This ensures that scrolling
44 // always feels smooth and is not interrupted by icon loading (which can be
45 // quite expensive if a disk access is required to determine the final icon).
46 const int ShortInterval = 50;
47
48 // If the icon size changes, a longer delay is used. This prevents that
49 // the expensive re-generation of all previews is triggered repeatedly when
50 // chaning the zoom level.
51 const int LongInterval = 300;
52 }
53
54 KFileItemListView::KFileItemListView(QGraphicsWidget* parent) :
55 KStandardItemListView(parent),
56 m_modelRolesUpdater(0),
57 m_updateVisibleIndexRangeTimer(0),
58 m_updateIconSizeTimer(0)
59 {
60 setAcceptDrops(true);
61
62 setScrollOrientation(Qt::Vertical);
63
64 m_updateVisibleIndexRangeTimer = new QTimer(this);
65 m_updateVisibleIndexRangeTimer->setSingleShot(true);
66 m_updateVisibleIndexRangeTimer->setInterval(ShortInterval);
67 connect(m_updateVisibleIndexRangeTimer, &QTimer::timeout, this, &KFileItemListView::updateVisibleIndexRange);
68
69 m_updateIconSizeTimer = new QTimer(this);
70 m_updateIconSizeTimer->setSingleShot(true);
71 m_updateIconSizeTimer->setInterval(LongInterval);
72 connect(m_updateIconSizeTimer, &QTimer::timeout, this, &KFileItemListView::updateIconSize);
73
74 setVisibleRoles(QList<QByteArray>() << "text");
75 }
76
77 KFileItemListView::~KFileItemListView()
78 {
79 }
80
81 void KFileItemListView::setPreviewsShown(bool show)
82 {
83 if (!m_modelRolesUpdater) {
84 return;
85 }
86
87 if (m_modelRolesUpdater->previewsShown() != show) {
88 beginTransaction();
89 m_modelRolesUpdater->setPreviewsShown(show);
90 onPreviewsShownChanged(show);
91 endTransaction();
92 }
93 }
94
95 bool KFileItemListView::previewsShown() const
96 {
97 return m_modelRolesUpdater ? m_modelRolesUpdater->previewsShown() : false;
98 }
99
100 void KFileItemListView::setEnlargeSmallPreviews(bool enlarge)
101 {
102 if (m_modelRolesUpdater) {
103 m_modelRolesUpdater->setEnlargeSmallPreviews(enlarge);
104 }
105 }
106
107 bool KFileItemListView::enlargeSmallPreviews() const
108 {
109 return m_modelRolesUpdater ? m_modelRolesUpdater->enlargeSmallPreviews() : false;
110 }
111
112 void KFileItemListView::setEnabledPlugins(const QStringList& list)
113 {
114 if (m_modelRolesUpdater) {
115 m_modelRolesUpdater->setEnabledPlugins(list);
116 }
117 }
118
119 QStringList KFileItemListView::enabledPlugins() const
120 {
121 return m_modelRolesUpdater ? m_modelRolesUpdater->enabledPlugins() : QStringList();
122 }
123
124 QPixmap KFileItemListView::createDragPixmap(const KItemSet& indexes) const
125 {
126 if (!model()) {
127 return QPixmap();
128 }
129
130 const int itemCount = indexes.count();
131 Q_ASSERT(itemCount > 0);
132 if (itemCount == 1) {
133 return KItemListView::createDragPixmap(indexes);
134 }
135
136 // If more than one item is dragged, align the items inside a
137 // rectangular grid. The maximum grid size is limited to 5 x 5 items.
138 int xCount;
139 int size;
140 if (itemCount > 16) {
141 xCount = 5;
142 size = KIconLoader::SizeSmall;
143 } else if (itemCount > 9) {
144 xCount = 4;
145 size = KIconLoader::SizeSmallMedium;
146 } else {
147 xCount = 3;
148 size = KIconLoader::SizeMedium;
149 }
150
151 if (itemCount < xCount) {
152 xCount = itemCount;
153 }
154
155 int yCount = itemCount / xCount;
156 if (itemCount % xCount != 0) {
157 ++yCount;
158 }
159 if (yCount > xCount) {
160 yCount = xCount;
161 }
162
163 // Draw the selected items into the grid cells.
164 QPixmap dragPixmap(xCount * size + xCount, yCount * size + yCount);
165 dragPixmap.fill(Qt::transparent);
166
167 QPainter painter(&dragPixmap);
168 int x = 0;
169 int y = 0;
170
171 foreach (int index, indexes) {
172 QPixmap pixmap = model()->data(index).value("iconPixmap").value<QPixmap>();
173 if (pixmap.isNull()) {
174 KIcon icon(model()->data(index).value("iconName").toString());
175 pixmap = icon.pixmap(size, size);
176 } else {
177 KPixmapModifier::scale(pixmap, QSize(size, size));
178 }
179
180 painter.drawPixmap(x, y, pixmap);
181
182 x += size + 1;
183 if (x >= dragPixmap.width()) {
184 x = 0;
185 y += size + 1;
186 }
187
188 if (y >= dragPixmap.height()) {
189 break;
190 }
191 }
192
193 return dragPixmap;
194 }
195
196 KItemListWidgetCreatorBase* KFileItemListView::defaultWidgetCreator() const
197 {
198 return new KItemListWidgetCreator<KFileItemListWidget>();
199 }
200
201 void KFileItemListView::initializeItemListWidget(KItemListWidget* item)
202 {
203 KStandardItemListView::initializeItemListWidget(item);
204
205 // Make sure that the item has an icon.
206 QHash<QByteArray, QVariant> data = item->data();
207 if (!data.contains("iconName") && data["iconPixmap"].value<QPixmap>().isNull()) {
208 Q_ASSERT(qobject_cast<KFileItemModel*>(model()));
209 KFileItemModel* fileItemModel = static_cast<KFileItemModel*>(model());
210
211 const KFileItem fileItem = fileItemModel->fileItem(item->index());
212 data.insert("iconName", fileItem.iconName());
213 item->setData(data, QSet<QByteArray>() << "iconName");
214 }
215 }
216
217 void KFileItemListView::onPreviewsShownChanged(bool shown)
218 {
219 Q_UNUSED(shown);
220 }
221
222 void KFileItemListView::onItemLayoutChanged(ItemLayout current, ItemLayout previous)
223 {
224 KStandardItemListView::onItemLayoutChanged(current, previous);
225 triggerVisibleIndexRangeUpdate();
226 }
227
228 void KFileItemListView::onModelChanged(KItemModelBase* current, KItemModelBase* previous)
229 {
230 Q_ASSERT(qobject_cast<KFileItemModel*>(current));
231 KStandardItemListView::onModelChanged(current, previous);
232
233 delete m_modelRolesUpdater;
234 m_modelRolesUpdater = 0;
235
236 if (current) {
237 m_modelRolesUpdater = new KFileItemModelRolesUpdater(static_cast<KFileItemModel*>(current), this);
238 m_modelRolesUpdater->setIconSize(availableIconSize());
239
240 applyRolesToModel();
241 }
242 }
243
244 void KFileItemListView::onScrollOrientationChanged(Qt::Orientation current, Qt::Orientation previous)
245 {
246 KStandardItemListView::onScrollOrientationChanged(current, previous);
247 triggerVisibleIndexRangeUpdate();
248 }
249
250 void KFileItemListView::onItemSizeChanged(const QSizeF& current, const QSizeF& previous)
251 {
252 Q_UNUSED(current);
253 Q_UNUSED(previous);
254 triggerVisibleIndexRangeUpdate();
255 }
256
257 void KFileItemListView::onScrollOffsetChanged(qreal current, qreal previous)
258 {
259 KStandardItemListView::onScrollOffsetChanged(current, previous);
260 triggerVisibleIndexRangeUpdate();
261 }
262
263 void KFileItemListView::onVisibleRolesChanged(const QList<QByteArray>& current, const QList<QByteArray>& previous)
264 {
265 KStandardItemListView::onVisibleRolesChanged(current, previous);
266 applyRolesToModel();
267 }
268
269 void KFileItemListView::onStyleOptionChanged(const KItemListStyleOption& current, const KItemListStyleOption& previous)
270 {
271 KStandardItemListView::onStyleOptionChanged(current, previous);
272 triggerIconSizeUpdate();
273 }
274
275 void KFileItemListView::onSupportsItemExpandingChanged(bool supportsExpanding)
276 {
277 applyRolesToModel();
278 KStandardItemListView::onSupportsItemExpandingChanged(supportsExpanding);
279 triggerVisibleIndexRangeUpdate();
280 }
281
282 void KFileItemListView::onTransactionBegin()
283 {
284 if (m_modelRolesUpdater) {
285 m_modelRolesUpdater->setPaused(true);
286 }
287 }
288
289 void KFileItemListView::onTransactionEnd()
290 {
291 if (!m_modelRolesUpdater) {
292 return;
293 }
294
295 // Only unpause the model-roles-updater if no timer is active. If one
296 // timer is still active the model-roles-updater will be unpaused later as
297 // soon as the timer has been exceeded.
298 const bool timerActive = m_updateVisibleIndexRangeTimer->isActive() ||
299 m_updateIconSizeTimer->isActive();
300 if (!timerActive) {
301 m_modelRolesUpdater->setPaused(false);
302 }
303 }
304
305 void KFileItemListView::resizeEvent(QGraphicsSceneResizeEvent* event)
306 {
307 KStandardItemListView::resizeEvent(event);
308 triggerVisibleIndexRangeUpdate();
309 }
310
311 void KFileItemListView::slotItemsRemoved(const KItemRangeList& itemRanges)
312 {
313 KStandardItemListView::slotItemsRemoved(itemRanges);
314 }
315
316 void KFileItemListView::slotSortRoleChanged(const QByteArray& current, const QByteArray& previous)
317 {
318 const QByteArray sortRole = model()->sortRole();
319 if (!visibleRoles().contains(sortRole)) {
320 applyRolesToModel();
321 }
322
323 KStandardItemListView::slotSortRoleChanged(current, previous);
324 }
325
326 void KFileItemListView::triggerVisibleIndexRangeUpdate()
327 {
328 if (!model()) {
329 return;
330 }
331 m_modelRolesUpdater->setPaused(true);
332
333 // If the icon size has been changed recently, wait until
334 // m_updateIconSizeTimer expires.
335 if (!m_updateIconSizeTimer->isActive()) {
336 m_updateVisibleIndexRangeTimer->start();
337 }
338 }
339
340 void KFileItemListView::updateVisibleIndexRange()
341 {
342 if (!m_modelRolesUpdater) {
343 return;
344 }
345
346 const int index = firstVisibleIndex();
347 const int count = lastVisibleIndex() - index + 1;
348 m_modelRolesUpdater->setMaximumVisibleItems(maximumVisibleItems());
349 m_modelRolesUpdater->setVisibleIndexRange(index, count);
350 m_modelRolesUpdater->setPaused(isTransactionActive());
351 }
352
353 void KFileItemListView::triggerIconSizeUpdate()
354 {
355 if (!model()) {
356 return;
357 }
358 m_modelRolesUpdater->setPaused(true);
359 m_updateIconSizeTimer->start();
360
361 // The visible index range will be updated when m_updateIconSizeTimer expires.
362 // Stop m_updateVisibleIndexRangeTimer to prevent an expensive re-generation
363 // of all previews (note that the user might change the icon size again soon).
364 m_updateVisibleIndexRangeTimer->stop();
365 }
366
367 void KFileItemListView::updateIconSize()
368 {
369 if (!m_modelRolesUpdater) {
370 return;
371 }
372
373 m_modelRolesUpdater->setIconSize(availableIconSize());
374
375 // Update the visible index range (which has most likely changed after the
376 // icon size change) before unpausing m_modelRolesUpdater.
377 const int index = firstVisibleIndex();
378 const int count = lastVisibleIndex() - index + 1;
379 m_modelRolesUpdater->setVisibleIndexRange(index, count);
380
381 m_modelRolesUpdater->setPaused(isTransactionActive());
382 }
383
384 void KFileItemListView::applyRolesToModel()
385 {
386 if (!model()) {
387 return;
388 }
389
390 Q_ASSERT(qobject_cast<KFileItemModel*>(model()));
391 KFileItemModel* fileItemModel = static_cast<KFileItemModel*>(model());
392
393 // KFileItemModel does not distinct between "visible" and "invisible" roles.
394 // Add all roles that are mandatory for having a working KFileItemListView:
395 QSet<QByteArray> roles = visibleRoles().toSet();
396 roles.insert("iconPixmap");
397 roles.insert("iconName");
398 roles.insert("text");
399 roles.insert("isDir");
400 roles.insert("isLink");
401 if (supportsItemExpanding()) {
402 roles.insert("isExpanded");
403 roles.insert("isExpandable");
404 roles.insert("expandedParentsCount");
405 }
406
407 // Assure that the role that is used for sorting will be determined
408 roles.insert(fileItemModel->sortRole());
409
410 fileItemModel->setRoles(roles);
411 m_modelRolesUpdater->setRoles(roles);
412 }
413
414 QSize KFileItemListView::availableIconSize() const
415 {
416 const KItemListStyleOption& option = styleOption();
417 const int iconSize = option.iconSize;
418 if (itemLayout() == IconsLayout) {
419 const int maxIconWidth = itemSize().width() - 2 * option.padding;
420 return QSize(maxIconWidth, iconSize);
421 }
422
423 return QSize(iconSize, iconSize);
424 }
425
426 #include "kfileitemlistview.moc"