mirror of
https://github.com/Chatterino/chatterino2.git
synced 2024-11-13 19:49:51 +01:00
refactor CompletionModel
This commit is contained in:
parent
aa3df2e6d8
commit
477d4751f7
|
@ -12,14 +12,97 @@
|
||||||
|
|
||||||
namespace chatterino {
|
namespace chatterino {
|
||||||
|
|
||||||
CompletionModel::CompletionModel(const QString &_channelName)
|
// -- TaggedString
|
||||||
: channelName(_channelName)
|
|
||||||
|
CompletionModel::TaggedString::TaggedString(const QString &_str, Type _type)
|
||||||
|
: str(_str)
|
||||||
|
, type(_type)
|
||||||
|
, timeAdded(std::chrono::steady_clock::now())
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool CompletionModel::TaggedString::isExpired(
|
||||||
|
const std::chrono::steady_clock::time_point &now) const
|
||||||
|
{
|
||||||
|
switch (this->type) {
|
||||||
|
case Type::Username: {
|
||||||
|
static std::chrono::minutes expirationTimer(10);
|
||||||
|
|
||||||
|
return (this->timeAdded + expirationTimer < now);
|
||||||
|
} break;
|
||||||
|
|
||||||
|
default: {
|
||||||
|
return false;
|
||||||
|
} break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool CompletionModel::TaggedString::isEmote() const
|
||||||
|
{
|
||||||
|
return this->type > Type::EmoteStart && this->type < Type::EmoteEnd;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool CompletionModel::TaggedString::operator<(const TaggedString &that) const
|
||||||
|
{
|
||||||
|
if (this->isEmote()) {
|
||||||
|
if (that.isEmote()) {
|
||||||
|
int k = QString::compare(this->str, that.str, Qt::CaseInsensitive);
|
||||||
|
if (k == 0) {
|
||||||
|
return this->str > that.str;
|
||||||
|
}
|
||||||
|
|
||||||
|
return k < 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (that.isEmote()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
int k = QString::compare(this->str, that.str, Qt::CaseInsensitive);
|
||||||
|
if (k == 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return k < 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// -- CompletionModel
|
||||||
|
|
||||||
|
CompletionModel::CompletionModel(const QString &_channelName)
|
||||||
|
: channelName_(_channelName)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
int CompletionModel::columnCount(const QModelIndex &) const
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
QVariant CompletionModel::data(const QModelIndex &index, int) const
|
||||||
|
{
|
||||||
|
std::lock_guard<std::mutex> lock(this->emotesMutex_);
|
||||||
|
|
||||||
|
// TODO: Implement more safely
|
||||||
|
auto it = this->emotes_.begin();
|
||||||
|
std::advance(it, index.row());
|
||||||
|
return QVariant(it->str);
|
||||||
|
}
|
||||||
|
|
||||||
|
int CompletionModel::rowCount(const QModelIndex &) const
|
||||||
|
{
|
||||||
|
std::lock_guard<std::mutex> lock(this->emotesMutex_);
|
||||||
|
|
||||||
|
return this->emotes_.size();
|
||||||
|
}
|
||||||
|
|
||||||
void CompletionModel::refresh()
|
void CompletionModel::refresh()
|
||||||
{
|
{
|
||||||
Log("[CompletionModel:{}] Refreshing...]", this->channelName);
|
Log("[CompletionModel:{}] Refreshing...]", this->channelName_);
|
||||||
|
|
||||||
auto app = getApp();
|
auto app = getApp();
|
||||||
|
|
||||||
|
@ -46,14 +129,14 @@ void CompletionModel::refresh()
|
||||||
|
|
||||||
// Channel-specific: BTTV Channel Emotes
|
// Channel-specific: BTTV Channel Emotes
|
||||||
std::vector<QString> &bttvChannelEmoteCodes =
|
std::vector<QString> &bttvChannelEmoteCodes =
|
||||||
app->emotes->bttv.channelEmoteCodes[this->channelName];
|
app->emotes->bttv.channelEmoteCodes[this->channelName_];
|
||||||
for (const auto &m : bttvChannelEmoteCodes) {
|
for (const auto &m : bttvChannelEmoteCodes) {
|
||||||
this->addString(m, TaggedString::Type::BTTVChannelEmote);
|
this->addString(m, TaggedString::Type::BTTVChannelEmote);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Channel-specific: FFZ Channel Emotes
|
// Channel-specific: FFZ Channel Emotes
|
||||||
std::vector<QString> &ffzChannelEmoteCodes =
|
std::vector<QString> &ffzChannelEmoteCodes =
|
||||||
app->emotes->ffz.channelEmoteCodes[this->channelName];
|
app->emotes->ffz.channelEmoteCodes[this->channelName_];
|
||||||
for (const auto &m : ffzChannelEmoteCodes) {
|
for (const auto &m : ffzChannelEmoteCodes) {
|
||||||
this->addString(m, TaggedString::Type::FFZChannelEmote);
|
this->addString(m, TaggedString::Type::FFZChannelEmote);
|
||||||
}
|
}
|
||||||
|
@ -91,10 +174,10 @@ void CompletionModel::refresh()
|
||||||
|
|
||||||
void CompletionModel::addString(const QString &str, TaggedString::Type type)
|
void CompletionModel::addString(const QString &str, TaggedString::Type type)
|
||||||
{
|
{
|
||||||
std::lock_guard<std::mutex> lock(this->emotesMutex);
|
std::lock_guard<std::mutex> lock(this->emotesMutex_);
|
||||||
|
|
||||||
// Always add a space at the end of completions
|
// Always add a space at the end of completions
|
||||||
this->emotes.insert({str + " ", type});
|
this->emotes_.insert({str + " ", type});
|
||||||
}
|
}
|
||||||
|
|
||||||
void CompletionModel::addUser(const QString &username)
|
void CompletionModel::addUser(const QString &username)
|
||||||
|
@ -102,14 +185,14 @@ void CompletionModel::addUser(const QString &username)
|
||||||
auto add = [this](const QString &str) {
|
auto add = [this](const QString &str) {
|
||||||
auto ts = this->createUser(str + " ");
|
auto ts = this->createUser(str + " ");
|
||||||
// Always add a space at the end of completions
|
// Always add a space at the end of completions
|
||||||
std::pair<std::set<TaggedString>::iterator, bool> p = this->emotes.insert(ts);
|
std::pair<std::set<TaggedString>::iterator, bool> p = this->emotes_.insert(ts);
|
||||||
if (!p.second) {
|
if (!p.second) {
|
||||||
// No inseration was made, figure out if we need to replace the username.
|
// No inseration was made, figure out if we need to replace the username.
|
||||||
|
|
||||||
if (p.first->str > ts.str) {
|
if (p.first->str > ts.str) {
|
||||||
// Replace lowercase version of name with mixed-case version
|
// Replace lowercase version of name with mixed-case version
|
||||||
this->emotes.erase(p.first);
|
this->emotes_.erase(p.first);
|
||||||
auto result2 = this->emotes.insert(ts);
|
auto result2 = this->emotes_.insert(ts);
|
||||||
assert(result2.second);
|
assert(result2.second);
|
||||||
} else {
|
} else {
|
||||||
p.first->timeAdded = std::chrono::steady_clock::now();
|
p.first->timeAdded = std::chrono::steady_clock::now();
|
||||||
|
@ -121,22 +204,27 @@ void CompletionModel::addUser(const QString &username)
|
||||||
add("@" + username);
|
add("@" + username);
|
||||||
}
|
}
|
||||||
|
|
||||||
void CompletionModel::ClearExpiredStrings()
|
void CompletionModel::clearExpiredStrings()
|
||||||
{
|
{
|
||||||
std::lock_guard<std::mutex> lock(this->emotesMutex);
|
std::lock_guard<std::mutex> lock(this->emotesMutex_);
|
||||||
|
|
||||||
auto now = std::chrono::steady_clock::now();
|
auto now = std::chrono::steady_clock::now();
|
||||||
|
|
||||||
for (auto it = this->emotes.begin(); it != this->emotes.end();) {
|
for (auto it = this->emotes_.begin(); it != this->emotes_.end();) {
|
||||||
const auto &taggedString = *it;
|
const auto &taggedString = *it;
|
||||||
|
|
||||||
if (taggedString.HasExpired(now)) {
|
if (taggedString.HasExpired(now)) {
|
||||||
// Log("String {} expired", taggedString.str);
|
// Log("String {} expired", taggedString.str);
|
||||||
it = this->emotes.erase(it);
|
it = this->emotes_.erase(it);
|
||||||
} else {
|
} else {
|
||||||
++it;
|
++it;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
CompletionModel::TaggedString CompletionModel::createUser(const QString &str)
|
||||||
|
{
|
||||||
|
return TaggedString{str, TaggedString::Type::Username};
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace chatterino
|
} // namespace chatterino
|
||||||
|
|
|
@ -16,123 +16,54 @@ class CompletionModel : public QAbstractListModel
|
||||||
enum Type {
|
enum Type {
|
||||||
Username,
|
Username,
|
||||||
|
|
||||||
// Emotes
|
// emotes
|
||||||
FFZGlobalEmote = 20,
|
EmoteStart,
|
||||||
|
FFZGlobalEmote,
|
||||||
FFZChannelEmote,
|
FFZChannelEmote,
|
||||||
BTTVGlobalEmote,
|
BTTVGlobalEmote,
|
||||||
BTTVChannelEmote,
|
BTTVChannelEmote,
|
||||||
TwitchGlobalEmote,
|
TwitchGlobalEmote,
|
||||||
TwitchSubscriberEmote,
|
TwitchSubscriberEmote,
|
||||||
Emoji,
|
Emoji,
|
||||||
|
EmoteEnd,
|
||||||
|
// end emotes
|
||||||
|
|
||||||
Command,
|
Command,
|
||||||
};
|
};
|
||||||
|
|
||||||
TaggedString(const QString &_str, Type _type)
|
TaggedString(const QString &_str, Type _type);
|
||||||
: str(_str)
|
|
||||||
, type(_type)
|
bool isExpired(const std::chrono::steady_clock::time_point &now) const;
|
||||||
, timeAdded(std::chrono::steady_clock::now())
|
bool isEmote() const;
|
||||||
{
|
bool operator<(const TaggedString &that) const;
|
||||||
}
|
|
||||||
|
|
||||||
QString str;
|
QString str;
|
||||||
|
|
||||||
// Type will help decide the lifetime of the tagged strings
|
// Type will help decide the lifetime of the tagged strings
|
||||||
Type type;
|
Type type;
|
||||||
|
|
||||||
mutable std::chrono::steady_clock::time_point timeAdded;
|
mutable std::chrono::steady_clock::time_point timeAdded;
|
||||||
|
|
||||||
bool HasExpired(const std::chrono::steady_clock::time_point &now) const
|
|
||||||
{
|
|
||||||
switch (this->type) {
|
|
||||||
case Type::Username: {
|
|
||||||
static std::chrono::minutes expirationTimer(10);
|
|
||||||
|
|
||||||
return (this->timeAdded + expirationTimer < now);
|
|
||||||
} break;
|
|
||||||
|
|
||||||
default: {
|
|
||||||
return false;
|
|
||||||
} break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool IsEmote() const
|
|
||||||
{
|
|
||||||
return this->type >= 20;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool operator<(const TaggedString &that) const
|
|
||||||
{
|
|
||||||
if (this->IsEmote()) {
|
|
||||||
if (that.IsEmote()) {
|
|
||||||
int k = QString::compare(this->str, that.str, Qt::CaseInsensitive);
|
|
||||||
if (k == 0) {
|
|
||||||
return this->str > that.str;
|
|
||||||
}
|
|
||||||
|
|
||||||
return k < 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (that.IsEmote()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
int k = QString::compare(this->str, that.str, Qt::CaseInsensitive);
|
|
||||||
if (k == 0) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return k < 0;
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
public:
|
public:
|
||||||
CompletionModel(const QString &_channelName);
|
CompletionModel(const QString &_channelName);
|
||||||
|
|
||||||
int columnCount(const QModelIndex &) const override
|
virtual int columnCount(const QModelIndex &) const override;
|
||||||
{
|
virtual QVariant data(const QModelIndex &index, int) const override;
|
||||||
return 1;
|
virtual int rowCount(const QModelIndex &) const override;
|
||||||
}
|
|
||||||
|
|
||||||
QVariant data(const QModelIndex &index, int) const override
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(this->emotesMutex);
|
|
||||||
|
|
||||||
// TODO: Implement more safely
|
|
||||||
auto it = this->emotes.begin();
|
|
||||||
std::advance(it, index.row());
|
|
||||||
return QVariant(it->str);
|
|
||||||
}
|
|
||||||
|
|
||||||
int rowCount(const QModelIndex &) const override
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(this->emotesMutex);
|
|
||||||
|
|
||||||
return this->emotes.size();
|
|
||||||
}
|
|
||||||
|
|
||||||
void refresh();
|
void refresh();
|
||||||
void addString(const QString &str, TaggedString::Type type);
|
void addString(const QString &str, TaggedString::Type type);
|
||||||
|
|
||||||
void addUser(const QString &str);
|
void addUser(const QString &str);
|
||||||
|
|
||||||
void ClearExpiredStrings();
|
void clearExpiredStrings();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
TaggedString createUser(const QString &str)
|
TaggedString createUser(const QString &str);
|
||||||
{
|
|
||||||
return TaggedString{str, TaggedString::Type::Username};
|
|
||||||
}
|
|
||||||
|
|
||||||
mutable std::mutex emotesMutex;
|
mutable std::mutex emotesMutex_;
|
||||||
std::set<TaggedString> emotes;
|
std::set<TaggedString> emotes_;
|
||||||
|
|
||||||
QString channelName;
|
QString channelName_;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace chatterino
|
} // namespace chatterino
|
||||||
|
|
Loading…
Reference in a new issue