7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
144#if QT_CONFIG(regularexpression)
145#include <QtCore/qregularexpression.h>
148#include <QtCore/private/qfilesystemiterator_p.h>
149#include <QtCore/private/qfilesystementry_p.h>
150#include <QtCore/private/qfilesystemmetadata_p.h>
151#include <QtCore/private/qfilesystemengine_p.h>
152#include <QtCore/private/qfileinfo_p.h>
153#include <QtCore/private/qduplicatetracker_p.h>
161using namespace Qt::StringLiterals;
170 template <
typename T>
196#if QT_CONFIG(regularexpression)
209#ifndef QT_NO_FILESYSTEMITERATOR
218 bool matchesFilters(
const QFileInfo &fileInfo)
const;
219 bool matchesFilters(
QDirEntryInfo::Iterator &iterator)
const;
220 bool matchesFilters(
const QFileInfo &fileInfo,
const QString &fileName)
const;
226 if (nameFilters.contains(
"*"_L1))
229#if QT_CONFIG(regularexpression)
230 nameRegExps.reserve(size_t(nameFilters.size()));
232 const bool isCase = iteratorFlags.testAnyFlags(QDirListing::IteratorFlag::CaseSensitive);
233 const auto cs = isCase ? Qt::CaseSensitive : Qt::CaseInsensitive;
234 for (
const auto &filter : std::as_const(nameFilters))
235 nameRegExps.emplace_back(QRegularExpression::fromWildcard(filter, cs));
238 Q_ASSERT(std::holds_alternative<QDirEntryInfo::Native>(initialEntryInfo.content));
239 QDirEntryInfo::Native &native = *std::get_if<QDirEntryInfo::Native>(&initialEntryInfo.content);
240 engine = QFileSystemEngine::createLegacyEngine(native.entry, native.metaData);
244
245
246
247
248
251#ifndef QT_NO_FILESYSTEMITERATOR
252 nativeIterators.clear();
254 fileEngineIterators.clear();
255 visitedLinks.clear();
261 const QString path = [&entryInfo] {
263 if (entryInfo.isSymLink())
264 return entryInfo.canonicalFilePath();
266 return entryInfo.filePath();
270 if (iteratorFlags.testAnyFlags(QDirListing::IteratorFlag::FollowDirSymlinks)) {
272 if (visitedLinks.hasSeen(entryInfo.canonicalFilePath()))
277 engine->setFileName(path);
278 if (
auto it = engine->beginEntryList(path, iteratorFlags, nameFilters)) {
279 fileEngineIterators.push(std::move(it));
284#ifndef QT_NO_FILESYSTEMITERATOR
285 nativeIterators.push(std::make_unique<QFileSystemIterator>(
287 [](
const QDirEntryInfo::Native &native) {
return native.entry; },
288 [](
const QFileInfo &fileInfo) {
return fileInfo.d_ptr->fileEntry; }),
291 qWarning(
"Qt was built with -no-feature-filesystemiterator: no files/plugins will be found!");
303
304
305
306
307
308
309
310
311
312
313
314
315
319 while (!fileEngineIterators.empty()) {
322 while (fileEngineIterators.top()->advance()) {
325 currentEntryInfo = std::move(entryInfo);
330 fileEngineIterators.pop();
333#ifndef QT_NO_FILESYSTEMITERATOR
334 while (!nativeIterators.empty()) {
337 while (std::optional r = nativeIterators.top()->advance()) {
339 currentEntryInfo = std::move(*r);
344 nativeIterators.pop();
352 return fileName ==
"."_L1 || fileName ==
".."_L1;
357 using F = QDirListing::IteratorFlag;
359 if (!iteratorFlags.testAnyFlags(F::Recursive))
363 if (!iteratorFlags.testAnyFlags(F::FollowDirSymlinks) && entryInfo.isSymLink())
367 if (isDotOrDotDot(entryInfo.fileName()))
371 const bool includeHidden = iteratorFlags.testAnyFlags(QDirListing::IteratorFlag::IncludeHidden);
384 using F = QDirListing::IteratorFlag;
385 using M = QFileSystemMetaData;
387 const QString fileName = native.entry.fileName();
388 if (fileName.isEmpty())
392#if QT_CONFIG(regularexpression)
393 const bool skipNameFilters = iteratorFlags.testAnyFlags(F::NoNameFiltersForDirs)
394 && native.ensureFilled(M::DirectoryType).isDirectory();
395 if (!skipNameFilters) {
396 if (!regexMatchesName(fileName))
401 if (isDotOrDotDot(fileName))
402 return iteratorFlags.testFlags(F::IncludeDotAndDotDot);
404 if (!iteratorFlags.testAnyFlag(F::IncludeHidden)
405 && native.ensureFilled(M::HiddenAttribute).isHidden()) {
409 if (iteratorFlags.testAnyFlags(F::IncludeBrokenSymlinks)
410 && native.ensureFilled(M::LegacyLinkType).isLegacyLink()
411 && !native.ensureFilled(M::ExistsAttribute).exists()) {
415 if (iteratorFlags.testFlag(F::ResolveSymlinks)) {
416 if (native.ensureFilled(M::LegacyLinkType).isLegacyLink()
417 && !native.ensureFilled(M::ExistsAttribute).exists()) {
421 constexpr auto f = F::ExcludeFiles | F::ExcludeDirs | F::ExcludeOther;
422 const bool filterByTargetType = iteratorFlags.testAnyFlags(f);
423 if (filterByTargetType && native.ensureFilled(M::LegacyLinkType).isLegacyLink())
427 if (iteratorFlags.testAnyFlag(F::ExcludeOther)
428 && !native.ensureFilled(M::FileType).isFile()
429 && !native.ensureFilled(M::DirectoryType).isDirectory()
430 && !native.ensureFilled(M::LegacyLinkType).isLegacyLink()) {
434 if (iteratorFlags.testAnyFlags(F::ExcludeDirs)
435 && native.ensureFilled(M::DirectoryType).isDirectory()) {
439 if (iteratorFlags.testAnyFlags(F::ExcludeFiles) && native.ensureFilled(M::FileType).isFile())
445bool QDirListingPrivate::matchesFilters(
const QFileInfo &fileInfo,
const QString &fileName)
const
447 using F = QDirListing::IteratorFlag;
449 Q_ASSERT(!fileName.isEmpty());
452#if QT_CONFIG(regularexpression)
453 const bool skipNameFilters = iteratorFlags.testFlag(F::NoNameFiltersForDirs)
455 if (!skipNameFilters) {
456 if (!regexMatchesName(fileName))
461 if (isDotOrDotDot(fileName))
462 return iteratorFlags.testFlag(F::IncludeDotAndDotDot);
464 if (!iteratorFlags.testFlag(F::IncludeHidden) && fileInfo.isHidden())
467 const bool includeBrokenSymlinks = iteratorFlags.testAnyFlags(F::IncludeBrokenSymlinks);
468 if (includeBrokenSymlinks && fileInfo.isSymLink() && !fileInfo.exists())
471 if (iteratorFlags.testFlag(F::ResolveSymlinks)) {
472 if (fileInfo.isSymLink() && !fileInfo.exists())
475 constexpr auto f = F::ExcludeFiles | F::ExcludeDirs | F::ExcludeOther;
476 const bool filterByTargetType = iteratorFlags.testAnyFlags(f);
477 if (filterByTargetType && fileInfo.isSymLink())
481 if (iteratorFlags.testFlag(F::ExcludeOther)
482 && !fileInfo.isFile() && !fileInfo.isDir() && !fileInfo.isSymLink()) {
486 if (iteratorFlags.testFlag(F::ExcludeDirs) && fileInfo.isDir())
489 if (iteratorFlags.testFlag(F::ExcludeFiles) && fileInfo.isFile())
497 using F = QDirListing::IteratorFlag;
499 const QString fileName = iterator.iterator->currentFileName();
500 if (fileName.isEmpty())
504 if (iteratorFlags.testFlag(F::IncludeHidden)
505 && !iteratorFlags.testAnyFlags(
506 F::NoNameFiltersForDirs |
507 F::IncludeBrokenSymlinks |
513#if QT_CONFIG(regularexpression)
514 if (!regexMatchesName(fileName))
517 if (isDotOrDotDot(fileName))
518 return iteratorFlags.testFlags(F::IncludeDotAndDotDot);
524 return matchesFilters(iterator.ensureFileInfo(), fileName);
529 const QString fileName = fileInfo.fileName();
530 return !fileName.isEmpty() && matchesFilters(fileInfo, fileName);
534
535
536
537
538
539
542 return std::visit(QDirEntryInfoPrivate::overloaded {
543 [
this](QDirEntryInfo::Native &native) {
return matchesFilters(native); },
544 [
this](
const QFileInfo &fileInfo) {
return matchesFilters(fileInfo); },
545 [
this](QDirEntryInfo::Iterator &iterator) {
return matchesFilters(iterator); }
546 }, entryInfo.content);
552 return !fileEngineIterators.empty();
554#if !defined(QT_NO_FILESYSTEMITERATOR)
555 return !nativeIterators.empty();
562
563
564
565
566
567
568
569
570
574 d->initialEntryInfo.content = QDirEntryInfo::Native { QFileSystemEntry(path), {} };
575 d->iteratorFlags = flags;
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602QDirListing::QDirListing(
const QString &path,
const QStringList &nameFilters, IteratorFlags flags)
605 d->initialEntryInfo.content = QDirEntryInfo::Native { QFileSystemEntry(path), {} };
606 d->nameFilters = nameFilters;
607 d->iteratorFlags = flags;
612
613
614
615
616
617
618
619
620
621
624
625
626
627
628
629
632
633
640
641
644 return d->initialEntryInfo.filePath();
648
649
652 return d->iteratorFlags;
656
657
658
661 return d->nameFilters;
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
694
695
696
697
700
701
702
703
706
707
708
709
710
711
712
713
714
715
716
717
718
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
760
761
762
763
764
767
768
769
770
771
774
775
776
777
778
781
782
783
784
785
786
787
788
789
790
791
792
793
794
797
798
799
800
801
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
861 return dirListPtr->currentEntryInfo.fileInfo();
866 return dirListPtr->currentEntryInfo.fileName();
871 return dirListPtr->currentEntryInfo.baseName();
876 return dirListPtr->currentEntryInfo.completeBaseName();
881 return dirListPtr->currentEntryInfo.suffix();
886 return dirListPtr->currentEntryInfo.bundleName();
891 return dirListPtr->currentEntryInfo.completeSuffix();
896 return dirListPtr->currentEntryInfo.filePath();
901 return dirListPtr->currentEntryInfo.canonicalFilePath();
906 return dirListPtr->currentEntryInfo.absoluteFilePath();
911 return dirListPtr->currentEntryInfo.absolutePath();
916 return dirListPtr->currentEntryInfo.isDir();
921 return dirListPtr->currentEntryInfo.isFile();
926 return dirListPtr->currentEntryInfo.isSymLink();
931 return dirListPtr->currentEntryInfo.exists();
936 return dirListPtr->currentEntryInfo.isHidden();
941 return dirListPtr->currentEntryInfo.isReadable();
946 return dirListPtr->currentEntryInfo.isWritable();
951 return dirListPtr->currentEntryInfo.isExecutable();
956 return dirListPtr->currentEntryInfo.size();
961 return dirListPtr->currentEntryInfo.fileTime(type, tz);
bool entryMatches(QDirEntryInfo &info)
QDirListing::IteratorFlags iteratorFlags
void pushInitialDirectory()
void checkAndPushDirectory(QDirEntryInfo &info)
std::unique_ptr< QAbstractFileEngine > engine
QDuplicateTracker< QString > visitedLinks
bool hasIterators() const
vector_stack< FsIteratorPtr > nativeIterators
vector_stack< FEngineIteratorPtr > fileEngineIterators
bool matchesFilters(QDirEntryInfo &data) const
QDirEntryInfo initialEntryInfo
QDirEntryInfo currentEntryInfo
void pushDirectory(QDirEntryInfo &info)
Q_CORE_EXPORT bool isReadable() const
Q_CORE_EXPORT bool isHidden() const
Q_CORE_EXPORT bool isWritable() const
Q_CORE_EXPORT bool isExecutable() const
Q_CORE_EXPORT bool isFile() const
Q_CORE_EXPORT bool exists() const
Q_CORE_EXPORT bool isSymLink() const
Q_CORE_EXPORT bool isDir() const
const_iterator & operator++()
Pre-increment operator.
static bool isDotOrDotDot(QStringView fileName)