mirror of https://github.com/procxx/kepka.git
Add caching for emoji keywords.
This commit is contained in:
parent
4b3a9fac67
commit
ce2204e120
|
@ -46,6 +46,10 @@ struct LangPackData {
|
||||||
return (code == 0x2122U) || (code == 0xA9U) || (code == 0xAEU);
|
return (code == 0x2122U) || (code == 0xA9U) || (code == 0xAEU);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void CreateCacheFilePath() {
|
||||||
|
QDir().mkpath(internal::CacheFileFolder() + qstr("/keywords"));
|
||||||
|
}
|
||||||
|
|
||||||
[[nodiscard]] QString CacheFilePath(QString id) {
|
[[nodiscard]] QString CacheFilePath(QString id) {
|
||||||
static const auto BadSymbols = QRegularExpression("[^a-zA-Z0-9_\\.\\-]");
|
static const auto BadSymbols = QRegularExpression("[^a-zA-Z0-9_\\.\\-]");
|
||||||
id.replace(BadSymbols, QString());
|
id.replace(BadSymbols, QString());
|
||||||
|
@ -60,21 +64,78 @@ struct LangPackData {
|
||||||
if (!file.open(QIODevice::ReadOnly)) {
|
if (!file.open(QIODevice::ReadOnly)) {
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
// #TODO emoji
|
|
||||||
auto result = LangPackData();
|
auto result = LangPackData();
|
||||||
|
auto stream = QDataStream(&file);
|
||||||
|
stream.setVersion(QDataStream::Qt_5_1);
|
||||||
|
auto version = qint32();
|
||||||
|
auto count = qint32();
|
||||||
|
stream
|
||||||
|
>> version
|
||||||
|
>> count;
|
||||||
|
if (version < 0 || count < 0 || stream.status() != QDataStream::Ok) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
for (auto i = 0; i != count; ++i) {
|
||||||
|
auto key = QString();
|
||||||
|
auto size = qint32();
|
||||||
|
stream
|
||||||
|
>> key
|
||||||
|
>> size;
|
||||||
|
if (size < 0 || stream.status() != QDataStream::Ok) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
auto &list = result.emoji[key];
|
||||||
|
for (auto j = 0; j != size; ++j) {
|
||||||
|
auto text = QString();
|
||||||
|
stream >> text;
|
||||||
|
if (stream.status() != QDataStream::Ok) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
const auto emoji = MustAddPostfix(text)
|
||||||
|
? (text + QChar(Ui::Emoji::kPostfix))
|
||||||
|
: text;
|
||||||
|
const auto entry = LangPackEmoji{ Find(emoji), text };
|
||||||
|
if (!entry.emoji) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
list.push_back(entry);
|
||||||
|
}
|
||||||
|
result.maxKeyLength = std::max(result.maxKeyLength, key.size());
|
||||||
|
}
|
||||||
|
result.version = version;
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
void WriteLocalCache(const QString &id, const LangPackData &data) {
|
void WriteLocalCache(const QString &id, const LangPackData &data) {
|
||||||
|
if (!data.version && data.emoji.empty()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
CreateCacheFilePath();
|
||||||
auto file = QFile(CacheFilePath(id));
|
auto file = QFile(CacheFilePath(id));
|
||||||
if (!file.open(QIODevice::WriteOnly)) {
|
if (!file.open(QIODevice::WriteOnly)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
auto stream = QDataStream(&file);
|
||||||
|
stream.setVersion(QDataStream::Qt_5_1);
|
||||||
|
stream
|
||||||
|
<< qint32(data.version)
|
||||||
|
<< qint32(data.emoji.size());
|
||||||
|
for (const auto &[key, list] : data.emoji) {
|
||||||
|
stream
|
||||||
|
<< key
|
||||||
|
<< qint32(list.size());
|
||||||
|
for (const auto &emoji : list) {
|
||||||
|
stream << emoji.text;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
[[nodiscard]] QString NormalizeQuery(const QString &query) {
|
[[nodiscard]] QString NormalizeQuery(const QString &query) {
|
||||||
return query.toLower().trimmed();
|
return query.toLower();
|
||||||
|
}
|
||||||
|
|
||||||
|
[[nodiscard]] QString NormalizeKey(const QString &key) {
|
||||||
|
return key.toLower().trimmed();
|
||||||
}
|
}
|
||||||
|
|
||||||
void AppendFoundEmoji(
|
void AppendFoundEmoji(
|
||||||
|
@ -133,7 +194,7 @@ void ApplyDifference(
|
||||||
data.version = version;
|
data.version = version;
|
||||||
for (const auto &keyword : keywords) {
|
for (const auto &keyword : keywords) {
|
||||||
keyword.match([&](const MTPDemojiKeyword &keyword) {
|
keyword.match([&](const MTPDemojiKeyword &keyword) {
|
||||||
const auto word = NormalizeQuery(qs(keyword.vkeyword));
|
const auto word = NormalizeKey(qs(keyword.vkeyword));
|
||||||
if (word.isEmpty()) {
|
if (word.isEmpty()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -156,7 +217,7 @@ void ApplyDifference(
|
||||||
});
|
});
|
||||||
list.insert(end(list), emoji.begin(), emoji.end());
|
list.insert(end(list), emoji.begin(), emoji.end());
|
||||||
}, [&](const MTPDemojiKeywordDeleted &keyword) {
|
}, [&](const MTPDemojiKeywordDeleted &keyword) {
|
||||||
const auto word = NormalizeQuery(qs(keyword.vkeyword));
|
const auto word = NormalizeKey(qs(keyword.vkeyword));
|
||||||
if (word.isEmpty()) {
|
if (word.isEmpty()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -313,23 +374,19 @@ void EmojiKeywords::LangPack::applyDifference(
|
||||||
_data.version = 0;
|
_data.version = 0;
|
||||||
_state = State::Refreshed;
|
_state = State::Refreshed;
|
||||||
return;
|
return;
|
||||||
} else if (keywords.isEmpty()) {
|
} else if (keywords.isEmpty() && _data.version >= version) {
|
||||||
if (_data.version < version) {
|
_state = State::Refreshed;
|
||||||
auto moved = std::move(_data);
|
|
||||||
moved.version = version;
|
|
||||||
applyData(std::move(moved));
|
|
||||||
} else {
|
|
||||||
_state = State::Refreshed;
|
|
||||||
}
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
const auto id = _id;
|
||||||
|
auto copy = _data;
|
||||||
auto callback = crl::guard(_guard.make_guard(), [=](
|
auto callback = crl::guard(_guard.make_guard(), [=](
|
||||||
LangPackData &&result) {
|
LangPackData &&result) {
|
||||||
applyData(std::move(result));
|
applyData(std::move(result));
|
||||||
});
|
});
|
||||||
auto copy = _data;
|
|
||||||
crl::async([=, callback = std::move(callback)]() mutable {
|
crl::async([=, callback = std::move(callback)]() mutable {
|
||||||
ApplyDifference(copy, keywords, version);
|
ApplyDifference(copy, keywords, version);
|
||||||
|
WriteLocalCache(id, copy);
|
||||||
crl::on_main([
|
crl::on_main([
|
||||||
result = std::move(copy),
|
result = std::move(copy),
|
||||||
callback = std::move(callback)
|
callback = std::move(callback)
|
||||||
|
|
|
@ -522,7 +522,10 @@ QString SuggestionsController::getEmojiQuery() {
|
||||||
const auto length = position - _queryStartPosition;
|
const auto length = position - _queryStartPosition;
|
||||||
for (auto i = length; i != 0;) {
|
for (auto i = length; i != 0;) {
|
||||||
if (text[--i] == ':') {
|
if (text[--i] == ':') {
|
||||||
if (i + 1 == length) {
|
const auto previous = (i > 0) ? text[i - 1] : QChar(0);
|
||||||
|
if (i > 0 && (previous.isLetter() || previous.isDigit())) {
|
||||||
|
return QString();
|
||||||
|
} else if (i + 1 == length || text[i + 1].isSpace()) {
|
||||||
return QString();
|
return QString();
|
||||||
}
|
}
|
||||||
_queryStartPosition += i + 2;
|
_queryStartPosition += i + 2;
|
||||||
|
@ -539,7 +542,9 @@ QString SuggestionsController::getEmojiQuery() {
|
||||||
cursor.movePosition(QTextCursor::End);
|
cursor.movePosition(QTextCursor::End);
|
||||||
return cursor.position();
|
return cursor.position();
|
||||||
}();
|
}();
|
||||||
if ((length > modernLimit)
|
if (!length
|
||||||
|
|| text[0].isSpace()
|
||||||
|
|| (length > modernLimit)
|
||||||
|| (_queryStartPosition != 0)
|
|| (_queryStartPosition != 0)
|
||||||
|| (position != end)) {
|
|| (position != end)) {
|
||||||
return QString();
|
return QString();
|
||||||
|
|
Loading…
Reference in New Issue