Compare commits
12 Commits
7d3c24e6e1
...
v0.8.1
Author | SHA1 | Date | |
---|---|---|---|
20a1f8b2cd | |||
a47af257f3 | |||
9686ef30c7 | |||
abce4cfcd9 | |||
d55187a71c | |||
9e1bc98f38 | |||
496aefaa09 | |||
b4320f611b | |||
1b1ab2387e | |||
49a1a14009 | |||
48ca25abe3 | |||
42e9ac5f41 |
17
CHANGELOG.md
17
CHANGELOG.md
@ -1,4 +1,21 @@
|
|||||||
# looqs: Release notes
|
# looqs: Release notes
|
||||||
|
|
||||||
|
## 2022-11-19 - v0.8.1
|
||||||
|
|
||||||
|
CHANGES:
|
||||||
|
- Fix regression causing previews in second (and higher) result page to not render
|
||||||
|
- Minor improvements
|
||||||
|
|
||||||
|
## 2022-10-22 - v0.8
|
||||||
|
|
||||||
|
CHANGES:
|
||||||
|
- For new, not previously indexed files, start creating an additional index using sqlite's experimental trigram tokenizer. Thanks to that, we can now match substrings >= 3 of an unicode sequence. Results of the usual index are prioritized.
|
||||||
|
- GUI: Ensure order of previews matches ranking exactly. Previously, it depended simply on the time preview generators took, i. e. it was more or less a race.
|
||||||
|
- Report progress more often during indexing, so users don't get the impression that it's stuck when processing dirs with large documents.
|
||||||
|
- Fix a regression that caused phrase queries to be broken
|
||||||
|
- Minor improvements
|
||||||
|
- Add packages: Ubuntu 22.10.
|
||||||
|
|
||||||
## 2022-09-10 - v0.7
|
## 2022-09-10 - v0.7
|
||||||
|
|
||||||
CHANGES:
|
CHANGES:
|
||||||
|
@ -28,11 +28,12 @@ There is no need to write the long form of filters. There are also booleans avai
|
|||||||
The screenshots in this section may occasionally be slightly outdated, but they are usually recent enough to get an overall impression of the current state of the GUI.
|
The screenshots in this section may occasionally be slightly outdated, but they are usually recent enough to get an overall impression of the current state of the GUI.
|
||||||
|
|
||||||
## Current status
|
## Current status
|
||||||
Latest version: 2022-09-10, v0.7
|
Latest version: 2022-11-19, v0.8.1
|
||||||
|
|
||||||
Please keep in mind: looqs is still at an early stage and may exhibit some weirdness and contain bugs.
|
Please keep in mind: looqs is still at an early stage and may exhibit some weirdness and contain bugs.
|
||||||
|
|
||||||
Please see [Changelog](CHANGELOG.md) for a human readable list of changes.
|
Please see [Changelog](CHANGELOG.md) for a human readable list of changes. For download instructions, see
|
||||||
|
further down this document.
|
||||||
|
|
||||||
|
|
||||||
## Goals and principles
|
## Goals and principles
|
||||||
@ -96,7 +97,7 @@ The GUI is located in `gui/looqs-gui`, the binary for the CLI is in `cli/looqs`
|
|||||||
## Packages
|
## Packages
|
||||||
At this point, looqs is not in any official distro package repo, but I maintain some packages.
|
At this point, looqs is not in any official distro package repo, but I maintain some packages.
|
||||||
|
|
||||||
### Ubuntu 22.04
|
### Ubuntu 22.04, 22.10
|
||||||
Latest release can be installed using apt from the repo.
|
Latest release can be installed using apt from the repo.
|
||||||
```
|
```
|
||||||
# First, obtain key, assume it's trusted.
|
# First, obtain key, assume it's trusted.
|
||||||
|
@ -643,7 +643,6 @@ void MainWindow::previewReceived(QSharedPointer<PreviewResult> preview, unsigned
|
|||||||
{
|
{
|
||||||
QString docPath = preview->getDocumentPath();
|
QString docPath = preview->getDocumentPath();
|
||||||
auto previewPage = preview->getPage();
|
auto previewPage = preview->getPage();
|
||||||
|
|
||||||
ClickLabel *headerLabel = new ClickLabel();
|
ClickLabel *headerLabel = new ClickLabel();
|
||||||
headerLabel->setText(QString("Path: ") + preview->getDocumentPath());
|
headerLabel->setText(QString("Path: ") + preview->getDocumentPath());
|
||||||
|
|
||||||
@ -685,7 +684,24 @@ void MainWindow::previewReceived(QSharedPointer<PreviewResult> preview, unsigned
|
|||||||
|
|
||||||
previewWidget->setLayout(previewLayout);
|
previewWidget->setLayout(previewLayout);
|
||||||
|
|
||||||
ui->scrollAreaWidgetContents->layout()->addWidget(previewWidget);
|
QBoxLayout *layout = static_cast<QBoxLayout *>(ui->scrollAreaWidgetContents->layout());
|
||||||
|
int pos = previewOrder[docPath + QString::number(previewPage)];
|
||||||
|
if(pos <= layout->count())
|
||||||
|
{
|
||||||
|
layout->insertWidget(pos, previewWidget);
|
||||||
|
for(auto it = previewWidgetOrderCache.constKeyValueBegin();
|
||||||
|
it != previewWidgetOrderCache.constKeyValueEnd(); it++)
|
||||||
|
{
|
||||||
|
if(it->first <= layout->count())
|
||||||
|
{
|
||||||
|
layout->insertWidget(it->first, it->second);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
previewWidgetOrderCache[pos] = previewWidget;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -924,12 +940,12 @@ void MainWindow::makePreviews(int page)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
int end = previewsPerPage;
|
int length = previewsPerPage;
|
||||||
int begin = page * previewsPerPage - previewsPerPage;
|
int beginOffset = page * previewsPerPage - previewsPerPage;
|
||||||
if(begin < 0)
|
if(beginOffset < 0)
|
||||||
{
|
{
|
||||||
// Should not happen actually
|
// Should not happen actually
|
||||||
begin = 0;
|
beginOffset = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int currentScale = currentSelectedScale();
|
int currentScale = currentSelectedScale();
|
||||||
@ -938,6 +954,10 @@ void MainWindow::makePreviews(int page)
|
|||||||
renderConfig.scaleY = QGuiApplication::primaryScreen()->physicalDotsPerInchY() * (currentScale / 100.);
|
renderConfig.scaleY = QGuiApplication::primaryScreen()->physicalDotsPerInchY() * (currentScale / 100.);
|
||||||
renderConfig.wordsToHighlight = wordsToHighlight;
|
renderConfig.wordsToHighlight = wordsToHighlight;
|
||||||
|
|
||||||
|
this->previewOrder.clear();
|
||||||
|
this->previewWidgetOrderCache.clear();
|
||||||
|
|
||||||
|
int previewPos = 0;
|
||||||
QVector<RenderTarget> targets;
|
QVector<RenderTarget> targets;
|
||||||
for(SearchResult &sr : this->previewableSearchResults)
|
for(SearchResult &sr : this->previewableSearchResults)
|
||||||
{
|
{
|
||||||
@ -952,10 +972,14 @@ void MainWindow::makePreviews(int page)
|
|||||||
renderTarget.path = sr.fileData.absPath;
|
renderTarget.path = sr.fileData.absPath;
|
||||||
renderTarget.page = (int)sr.page;
|
renderTarget.page = (int)sr.page;
|
||||||
targets.append(renderTarget);
|
targets.append(renderTarget);
|
||||||
|
|
||||||
|
int pos = previewPos - beginOffset;
|
||||||
|
this->previewOrder[renderTarget.path + QString::number(renderTarget.page)] = pos;
|
||||||
|
++previewPos;
|
||||||
}
|
}
|
||||||
int numpages = ceil(static_cast<double>(targets.size()) / previewsPerPage);
|
int numpages = ceil(static_cast<double>(targets.size()) / previewsPerPage);
|
||||||
ui->spinPreviewPage->setMaximum(numpages);
|
ui->spinPreviewPage->setMaximum(numpages);
|
||||||
targets = targets.mid(begin, end);
|
targets = targets.mid(beginOffset, length);
|
||||||
|
|
||||||
ui->lblTotalPreviewPagesCount->setText(QString::number(numpages));
|
ui->lblTotalPreviewPagesCount->setText(QString::number(numpages));
|
||||||
ui->previewProcessBar->setMaximum(targets.count());
|
ui->previewProcessBar->setMaximum(targets.count());
|
||||||
|
@ -23,16 +23,6 @@ class MainWindow : public QMainWindow
|
|||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
|
||||||
explicit MainWindow(QWidget *parent, QString socketPath);
|
|
||||||
~MainWindow();
|
|
||||||
signals:
|
|
||||||
void beginSearch(const QString &query);
|
|
||||||
void startPdfPreviewGeneration(QVector<SearchResult> paths, double scalefactor);
|
|
||||||
|
|
||||||
protected:
|
|
||||||
void closeEvent(QCloseEvent *event) override;
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
DatabaseFactory *dbFactory;
|
DatabaseFactory *dbFactory;
|
||||||
SqliteDbService *dbService;
|
SqliteDbService *dbService;
|
||||||
@ -40,37 +30,40 @@ class MainWindow : public QMainWindow
|
|||||||
IPCPreviewClient ipcPreviewClient;
|
IPCPreviewClient ipcPreviewClient;
|
||||||
QThread ipcClientThread;
|
QThread ipcClientThread;
|
||||||
QThread syncerThread;
|
QThread syncerThread;
|
||||||
|
Indexer *indexer;
|
||||||
IndexSyncer *indexSyncer;
|
IndexSyncer *indexSyncer;
|
||||||
QProgressDialog progressDialog;
|
QProgressDialog progressDialog;
|
||||||
|
|
||||||
Indexer *indexer;
|
|
||||||
QFileIconProvider iconProvider;
|
QFileIconProvider iconProvider;
|
||||||
bool previewDirty;
|
|
||||||
QSqlDatabase db;
|
QSqlDatabase db;
|
||||||
QFutureWatcher<QVector<SearchResult>> searchWatcher;
|
QFutureWatcher<QVector<SearchResult>> searchWatcher;
|
||||||
void add(QString path, unsigned int page);
|
|
||||||
QVector<SearchResult> previewableSearchResults;
|
QVector<SearchResult> previewableSearchResults;
|
||||||
|
LooqsQuery contentSearchQuery;
|
||||||
|
QVector<QString> searchHistory;
|
||||||
|
int currentSearchHistoryIndex = 0;
|
||||||
|
QString currentSavedSearchText;
|
||||||
|
QHash<QString, int> previewOrder; /* Quick lookup for the order a preview should have */
|
||||||
|
QMap<int, QWidget *>
|
||||||
|
previewWidgetOrderCache /* Saves those that arrived out of order to be inserted later at the correct pos */;
|
||||||
|
bool previewDirty = false;
|
||||||
|
int previewsPerPage = 20;
|
||||||
|
unsigned int processedPdfPreviews = 0;
|
||||||
|
unsigned int currentPreviewGeneration = 1;
|
||||||
|
|
||||||
void connectSignals();
|
void connectSignals();
|
||||||
void makePreviews(int page);
|
void makePreviews(int page);
|
||||||
bool previewTabActive();
|
bool previewTabActive();
|
||||||
bool indexerTabActive();
|
bool indexerTabActive();
|
||||||
void keyPressEvent(QKeyEvent *event) override;
|
void keyPressEvent(QKeyEvent *event) override;
|
||||||
unsigned int processedPdfPreviews;
|
|
||||||
void handleSearchResults(const QVector<SearchResult> &results);
|
void handleSearchResults(const QVector<SearchResult> &results);
|
||||||
void handleSearchError(QString error);
|
void handleSearchError(QString error);
|
||||||
LooqsQuery contentSearchQuery;
|
|
||||||
int previewsPerPage;
|
|
||||||
void createSearchResutlMenu(QMenu &menu, const QFileInfo &fileInfo);
|
void createSearchResutlMenu(QMenu &menu, const QFileInfo &fileInfo);
|
||||||
void openDocument(QString path, int num);
|
void openDocument(QString path, int num);
|
||||||
void openFile(QString path);
|
void openFile(QString path);
|
||||||
unsigned int currentPreviewGeneration = 1;
|
|
||||||
void initSettingsTabs();
|
void initSettingsTabs();
|
||||||
int currentSelectedScale();
|
int currentSelectedScale();
|
||||||
void processShortcut(int key);
|
void processShortcut(int key);
|
||||||
bool eventFilter(QObject *object, QEvent *event);
|
bool eventFilter(QObject *object, QEvent *event);
|
||||||
QVector<QString> searchHistory;
|
|
||||||
int currentSearchHistoryIndex = 0;
|
|
||||||
QString currentSavedSearchText;
|
|
||||||
private slots:
|
private slots:
|
||||||
void lineEditReturnPressed();
|
void lineEditReturnPressed();
|
||||||
void treeSearchItemActivated(QTreeWidgetItem *item, int i);
|
void treeSearchItemActivated(QTreeWidgetItem *item, int i);
|
||||||
@ -90,6 +83,16 @@ class MainWindow : public QMainWindow
|
|||||||
void startIpcPreviews(RenderConfig config, const QVector<RenderTarget> &targets);
|
void startIpcPreviews(RenderConfig config, const QVector<RenderTarget> &targets);
|
||||||
void stopIpcPreviews();
|
void stopIpcPreviews();
|
||||||
void beginIndexSync();
|
void beginIndexSync();
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit MainWindow(QWidget *parent, QString socketPath);
|
||||||
|
~MainWindow();
|
||||||
|
signals:
|
||||||
|
void beginSearch(const QString &query);
|
||||||
|
void startPdfPreviewGeneration(QVector<SearchResult> paths, double scalefactor);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void closeEvent(QCloseEvent *event) override;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // MAINWINDOW_H
|
#endif // MAINWINDOW_H
|
||||||
|
@ -1,20 +1,24 @@
|
|||||||
#include "../shared/common.h"
|
#include "../shared/common.h"
|
||||||
#include "previewgenerator.h"
|
#include "previewgenerator.h"
|
||||||
|
#include <QMutexLocker>
|
||||||
#include "previewgeneratorpdf.h"
|
#include "previewgeneratorpdf.h"
|
||||||
#include "previewgeneratorplaintext.h"
|
#include "previewgeneratorplaintext.h"
|
||||||
#include "previewgeneratorodt.h"
|
#include "previewgeneratorodt.h"
|
||||||
|
|
||||||
static PreviewGenerator *plainTextGenerator = new PreviewGeneratorPlainText();
|
static PreviewGenerator *plainTextGenerator = new PreviewGeneratorPlainText();
|
||||||
|
|
||||||
static QMap<QString, PreviewGenerator *> generators{
|
static QHash<QString, PreviewGenerator *> generators{
|
||||||
{"pdf", new PreviewGeneratorPdf()}, {"txt", plainTextGenerator}, {"md", plainTextGenerator},
|
{"pdf", new PreviewGeneratorPdf()}, {"txt", plainTextGenerator}, {"md", plainTextGenerator},
|
||||||
{"py", plainTextGenerator}, {"java", plainTextGenerator}, {"js", plainTextGenerator},
|
{"py", plainTextGenerator}, {"java", plainTextGenerator}, {"js", plainTextGenerator},
|
||||||
{"cpp", plainTextGenerator}, {"c", plainTextGenerator}, {"sql", plainTextGenerator},
|
{"cpp", plainTextGenerator}, {"c", plainTextGenerator}, {"sql", plainTextGenerator},
|
||||||
{"odt", new PreviewGeneratorOdt()}};
|
{"odt", new PreviewGeneratorOdt()}};
|
||||||
|
|
||||||
|
static QMutex generatorsMutex;
|
||||||
PreviewGenerator *PreviewGenerator::get(QFileInfo &info)
|
PreviewGenerator *PreviewGenerator::get(QFileInfo &info)
|
||||||
{
|
{
|
||||||
|
QMutexLocker locker(&generatorsMutex);
|
||||||
PreviewGenerator *result = generators.value(info.suffix(), nullptr);
|
PreviewGenerator *result = generators.value(info.suffix(), nullptr);
|
||||||
|
locker.unlock();
|
||||||
if(result == nullptr)
|
if(result == nullptr)
|
||||||
{
|
{
|
||||||
if(Common::isTextFile(info))
|
if(Common::isTextFile(info))
|
||||||
|
@ -7,10 +7,12 @@ static QMutex cacheMutex;
|
|||||||
|
|
||||||
Poppler::Document *PreviewGeneratorPdf::document(QString path)
|
Poppler::Document *PreviewGeneratorPdf::document(QString path)
|
||||||
{
|
{
|
||||||
|
QMutexLocker locker(&cacheMutex);
|
||||||
if(documentcache.contains(path))
|
if(documentcache.contains(path))
|
||||||
{
|
{
|
||||||
return documentcache.value(path);
|
return documentcache.value(path);
|
||||||
}
|
}
|
||||||
|
locker.unlock();
|
||||||
Poppler::Document *result = Poppler::Document::load(path);
|
Poppler::Document *result = Poppler::Document::load(path);
|
||||||
if(result == nullptr)
|
if(result == nullptr)
|
||||||
{
|
{
|
||||||
@ -19,7 +21,7 @@ Poppler::Document *PreviewGeneratorPdf::document(QString path)
|
|||||||
}
|
}
|
||||||
result->setRenderHint(Poppler::Document::TextAntialiasing);
|
result->setRenderHint(Poppler::Document::TextAntialiasing);
|
||||||
|
|
||||||
QMutexLocker locker(&cacheMutex);
|
locker.relock();
|
||||||
documentcache.insert(path, result);
|
documentcache.insert(path, result);
|
||||||
locker.unlock();
|
locker.unlock();
|
||||||
return result;
|
return result;
|
||||||
|
@ -103,7 +103,7 @@ QString PreviewGeneratorPlainText::generateLineBasedPreviewText(QTextStream &in,
|
|||||||
int foundWordsCount = 0;
|
int foundWordsCount = 0;
|
||||||
for(QString &word : config.wordsToHighlight)
|
for(QString &word : config.wordsToHighlight)
|
||||||
{
|
{
|
||||||
QRegularExpression searchRegex("\\b" + word + "\\b");
|
QRegularExpression searchRegex("\\b" + QRegularExpression::escape(word) + "\\b");
|
||||||
bool containsRegex = line.contains(searchRegex);
|
bool containsRegex = line.contains(searchRegex);
|
||||||
bool contains = false;
|
bool contains = false;
|
||||||
if(!containsRegex)
|
if(!containsRegex)
|
||||||
|
@ -30,7 +30,7 @@ QByteArray PreviewResultPdf::serialize() const
|
|||||||
|
|
||||||
QSharedPointer<PreviewResultPdf> PreviewResultPdf::deserialize(QByteArray &ba)
|
QSharedPointer<PreviewResultPdf> PreviewResultPdf::deserialize(QByteArray &ba)
|
||||||
{
|
{
|
||||||
PreviewResultPdf *result = new PreviewResultPdf();
|
QSharedPointer<PreviewResultPdf> result(new PreviewResultPdf());
|
||||||
PreviewResultType type;
|
PreviewResultType type;
|
||||||
|
|
||||||
QDataStream stream{&ba, QIODevice::ReadOnly};
|
QDataStream stream{&ba, QIODevice::ReadOnly};
|
||||||
@ -40,5 +40,5 @@ QSharedPointer<PreviewResultPdf> PreviewResultPdf::deserialize(QByteArray &ba)
|
|||||||
throw std::runtime_error("Invalid byte array: Not a pdf preview");
|
throw std::runtime_error("Invalid byte array: Not a pdf preview");
|
||||||
}
|
}
|
||||||
stream >> result->documentPath >> result->page >> result->previewImage;
|
stream >> result->documentPath >> result->page >> result->previewImage;
|
||||||
return QSharedPointer<PreviewResultPdf>(result);
|
return result;
|
||||||
}
|
}
|
||||||
|
@ -40,7 +40,8 @@ QByteArray PreviewResultPlainText::serialize() const
|
|||||||
|
|
||||||
QSharedPointer<PreviewResultPlainText> PreviewResultPlainText::deserialize(QByteArray &ba)
|
QSharedPointer<PreviewResultPlainText> PreviewResultPlainText::deserialize(QByteArray &ba)
|
||||||
{
|
{
|
||||||
PreviewResultPlainText *result = new PreviewResultPlainText();
|
QSharedPointer<PreviewResultPlainText> result(new PreviewResultPlainText());
|
||||||
|
|
||||||
PreviewResultType type;
|
PreviewResultType type;
|
||||||
|
|
||||||
QDataStream stream{&ba, QIODevice::ReadOnly};
|
QDataStream stream{&ba, QIODevice::ReadOnly};
|
||||||
@ -50,5 +51,5 @@ QSharedPointer<PreviewResultPlainText> PreviewResultPlainText::deserialize(QByte
|
|||||||
throw std::runtime_error("Invalid byte array: Not a pdf preview");
|
throw std::runtime_error("Invalid byte array: Not a pdf preview");
|
||||||
}
|
}
|
||||||
stream >> result->documentPath >> result->page >> result->text;
|
stream >> result->documentPath >> result->page >> result->text;
|
||||||
return QSharedPointer<PreviewResultPlainText>(result);
|
return result;
|
||||||
}
|
}
|
||||||
|
@ -157,7 +157,6 @@ QSqlQuery SqliteSearch::makeSqlQuery(const LooqsQuery &query)
|
|||||||
throw LooqsGeneralException("Nothing to search for supplied");
|
throw LooqsGeneralException("Nothing to search for supplied");
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ftsAlreadyJoined = false;
|
|
||||||
auto tokens = query.getTokens();
|
auto tokens = query.getTokens();
|
||||||
for(const Token &token : tokens)
|
for(const Token &token : tokens)
|
||||||
{
|
{
|
||||||
|
Submodule submodules/exile.h updated: e711a1d53a...769f729dc5
Reference in New Issue
Block a user