1 /***************************************************************************
2 * Copyright (C) 2011 by Peter Penz <peter.penz19@gmail.com> *
3 * Copyright (C) 2013 by Frank Reininghaus <frank78ac@googlemail.com> *
5 * This program is free software; you can redistribute it and/or modify *
6 * it under the terms of the GNU General Public License as published by *
7 * the Free Software Foundation; either version 2 of the License, or *
8 * (at your option) any later version. *
10 * This program is distributed in the hope that it will be useful, *
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
13 * GNU General Public License for more details. *
15 * You should have received a copy of the GNU General Public License *
16 * along with this program; if not, write to the *
17 * Free Software Foundation, Inc., *
18 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA *
19 ***************************************************************************/
21 #include "kdirectorycontentscounter.h"
22 #include "kitemviews/kfileitemmodel.h"
29 KDirectoryContentsCounter::KDirectoryContentsCounter(KFileItemModel
* model
, QObject
* parent
) :
34 m_workerIsBusy(false),
35 m_dirWatcher(nullptr),
38 connect(m_model
, &KFileItemModel::itemsRemoved
,
39 this, &KDirectoryContentsCounter::slotItemsRemoved
);
41 if (!m_workerThread
) {
42 m_workerThread
= new QThread();
43 m_workerThread
->start();
46 m_worker
= new KDirectoryContentsCounterWorker();
47 m_worker
->moveToThread(m_workerThread
);
50 connect(this, &KDirectoryContentsCounter::requestDirectoryContentsCount
,
51 m_worker
, &KDirectoryContentsCounterWorker::countDirectoryContents
);
52 connect(m_worker
, &KDirectoryContentsCounterWorker::result
,
53 this, &KDirectoryContentsCounter::slotResult
);
55 m_dirWatcher
= new KDirWatch(this);
56 connect(m_dirWatcher
, &KDirWatch::dirty
, this, &KDirectoryContentsCounter::slotDirWatchDirty
);
59 KDirectoryContentsCounter::~KDirectoryContentsCounter()
63 if (m_workersCount
> 0) {
64 // The worker thread will continue running. It could even be running
65 // a method of m_worker at the moment, so we delete it using
66 // deleteLater() to prevent a crash.
67 m_worker
->deleteLater();
69 // There are no remaining workers -> stop the worker thread.
70 m_workerThread
->quit();
71 m_workerThread
->wait();
72 delete m_workerThread
;
73 m_workerThread
= nullptr;
75 // The worker thread has finished running now, so it's safe to delete
76 // m_worker. deleteLater() would not work at all because the event loop
77 // which would deliver the event to m_worker is not running any more.
82 void KDirectoryContentsCounter::addDirectory(const QString
& path
)
87 int KDirectoryContentsCounter::countDirectoryContentsSynchronously(const QString
& path
)
89 const QString resolvedPath
= QFileInfo(path
).canonicalFilePath();
91 if (!m_dirWatcher
->contains(resolvedPath
)) {
92 m_dirWatcher
->addDir(resolvedPath
);
93 m_watchedDirs
.insert(resolvedPath
);
96 KDirectoryContentsCounterWorker::Options options
;
98 if (m_model
->showHiddenFiles()) {
99 options
|= KDirectoryContentsCounterWorker::CountHiddenFiles
;
102 if (m_model
->showDirectoriesOnly()) {
103 options
|= KDirectoryContentsCounterWorker::CountDirectoriesOnly
;
106 return KDirectoryContentsCounterWorker::subItemsCount(path
, options
);
109 void KDirectoryContentsCounter::slotResult(const QString
& path
, int count
)
111 m_workerIsBusy
= false;
113 const QString resolvedPath
= QFileInfo(path
).canonicalFilePath();
115 if (!m_dirWatcher
->contains(resolvedPath
)) {
116 m_dirWatcher
->addDir(resolvedPath
);
117 m_watchedDirs
.insert(resolvedPath
);
120 if (!m_queue
.isEmpty()) {
121 startWorker(m_queue
.dequeue());
124 emit
result(path
, count
);
127 void KDirectoryContentsCounter::slotDirWatchDirty(const QString
& path
)
129 const int index
= m_model
->index(QUrl::fromLocalFile(path
));
131 if (!m_model
->fileItem(index
).isDir()) {
132 // If INotify is used, KDirWatch issues the dirty() signal
133 // also for changed files inside the directory, even if we
134 // don't enable this behavior explicitly (see bug 309740).
142 void KDirectoryContentsCounter::slotItemsRemoved()
144 const bool allItemsRemoved
= (m_model
->count() == 0);
146 if (!m_watchedDirs
.isEmpty()) {
147 // Don't let KDirWatch watch for removed items
148 if (allItemsRemoved
) {
149 foreach (const QString
& path
, m_watchedDirs
) {
150 m_dirWatcher
->removeDir(path
);
152 m_watchedDirs
.clear();
155 QMutableSetIterator
<QString
> it(m_watchedDirs
);
156 while (it
.hasNext()) {
157 const QString
& path
= it
.next();
158 if (m_model
->index(QUrl::fromLocalFile(path
)) < 0) {
159 m_dirWatcher
->removeDir(path
);
167 void KDirectoryContentsCounter::startWorker(const QString
& path
)
169 if (m_workerIsBusy
) {
170 m_queue
.enqueue(path
);
172 KDirectoryContentsCounterWorker::Options options
;
174 if (m_model
->showHiddenFiles()) {
175 options
|= KDirectoryContentsCounterWorker::CountHiddenFiles
;
178 if (m_model
->showDirectoriesOnly()) {
179 options
|= KDirectoryContentsCounterWorker::CountDirectoriesOnly
;
182 emit
requestDirectoryContentsCount(path
, options
);
183 m_workerIsBusy
= true;
187 QThread
* KDirectoryContentsCounter::m_workerThread
= nullptr;
188 int KDirectoryContentsCounter::m_workersCount
= 0;