aboutsummaryrefslogtreecommitdiffstats
path: root/core/qthelper.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'core/qthelper.cpp')
-rw-r--r--core/qthelper.cpp96
1 files changed, 77 insertions, 19 deletions
diff --git a/core/qthelper.cpp b/core/qthelper.cpp
index 7ca80a96e..1d77dbad6 100644
--- a/core/qthelper.cpp
+++ b/core/qthelper.cpp
@@ -1277,37 +1277,95 @@ QStringList imageExtensionFilters() {
return filters;
}
-// This works on a copy of the string, because it runs in asynchronous context
-static void learnImage(QString filename)
+// Compare two full paths and return the number of matching levels, starting from the filename.
+// String comparison is case-insensitive.
+static int matchFilename(const QString &path1, const QString &path2)
+{
+ QFileInfo f1(path1);
+ QFileInfo f2(path2);
+
+ int score = 0;
+ for (;;) {
+ QString fn1 = f1.fileName();
+ QString fn2 = f2.fileName();
+ if (fn1.isEmpty() || fn2.isEmpty())
+ break;
+ if (fn1 == ".") {
+ f1 = QFileInfo(f1.path());
+ continue;
+ }
+ if (fn2 == ".") {
+ f2 = QFileInfo(f2.path());
+ continue;
+ }
+ if (QString::compare(fn1, fn2, Qt::CaseInsensitive) != 0)
+ break;
+ f1 = QFileInfo(f1.path());
+ f2 = QFileInfo(f2.path());
+ ++score;
+ }
+ return score;
+}
+
+struct ImageMatch {
+ QString localFilename;
+ int score;
+};
+
+static void learnImage(const QString &filename, QMap<QString, ImageMatch> &matches)
{
+ // Find the original filenames with the highest match-score
+ QStringList newMatches;
QByteArray hash = hashFile(filename);
- // TODO: This is inefficient: we search the hash map by value. But firstly,
- // this is running in asynchronously, so it doesn't block the UI. Secondly,
- // we might not want to learn pictures by hash anyway (the user might have
- // edited the picture, which changes the hash.
+ int bestScore = 1;
for (auto it = hashOf.cbegin(); it != hashOf.cend(); ++it) {
- if (it.value() == hash)
- learnPictureFilename(it.key(), filename);
+ int score = matchFilename(filename, it.key());
+ if (score < bestScore)
+ continue;
+ if (score > bestScore)
+ newMatches.clear();
+ newMatches.append(it.key());
+ bestScore = score;
+ }
+
+ // Add the new original filenames to the list of matches, if the score is higher than previously
+ for (const QString &originalFilename: newMatches) {
+ auto it = matches.find(originalFilename);
+ if (it == matches.end())
+ matches.insert(originalFilename, { filename, bestScore });
+ else if (it->score < bestScore)
+ *it = { filename, bestScore };
}
}
-void learnImages(const QDir dir, int max_recursions)
+void learnImages(const QStringList &dirNames, int max_recursions)
{
- QStringList files;
QStringList filters = imageExtensionFilters();
-
- if (max_recursions) {
- foreach (QString dirname, dir.entryList(QStringList(), QDir::NoDotAndDotDot | QDir::Dirs)) {
- learnImages(QDir(dir.filePath(dirname)), max_recursions - 1);
+ QMap<QString, ImageMatch> matches;
+
+ QVector<QStringList> stack; // Use a stack to recurse into directories
+ stack.reserve(max_recursions + 1);
+ stack.append(dirNames);
+ while (!stack.isEmpty()) {
+ if (stack.last().isEmpty()) {
+ stack.removeLast();
+ continue;
+ }
+ QDir dir(stack.last().takeLast());
+
+ for (const QString &file: dir.entryList(filters, QDir::Files))
+ learnImage(dir.absoluteFilePath(file), matches);
+ if (stack.size() <= max_recursions) {
+ stack.append(QStringList());
+ for (const QString &dirname: dir.entryList(QStringList(), QDir::NoDotAndDotDot | QDir::Dirs))
+ stack.last().append(dir.filePath(dirname));
}
}
+ for (auto it = matches.begin(); it != matches.end(); ++it)
+ learnPictureFilename(it.key(), it->localFilename);
- foreach (QString file, dir.entryList(filters, QDir::Files)) {
- files.append(dir.absoluteFilePath(file));
- }
-
- QtConcurrent::blockingMap(files, learnImage);
+ write_hashes();
}
extern "C" const char *local_file_path(struct picture *picture)