cleaned up CompletionModel

This commit is contained in:
fourtf 2018-08-13 14:38:03 +02:00
parent f6e110b7fb
commit 2448f6f538
5 changed files with 35 additions and 147 deletions

View file

@ -22,15 +22,10 @@ namespace chatterino {
// Channel // Channel
// //
Channel::Channel(const QString &name, Type type) Channel::Channel(const QString &name, Type type)
: completionModel(name) : completionModel(*this)
, name_(name) , name_(name)
, type_(type) , type_(type)
{ {
QObject::connect(&this->clearCompletionModelTimer_, &QTimer::timeout,
[this]() {
this->completionModel.clearExpiredStrings(); //
});
this->clearCompletionModelTimer_.start(60 * 1000);
} }
Channel::~Channel() Channel::~Channel()

View file

@ -12,38 +12,20 @@
#include "singletons/Emotes.hpp" #include "singletons/Emotes.hpp"
#include <QtAlgorithms> #include <QtAlgorithms>
#include <utility> #include <utility>
namespace chatterino { namespace chatterino {
// -- TaggedString //
// TaggedString
//
CompletionModel::TaggedString::TaggedString(const QString &_str, Type _type) CompletionModel::TaggedString::TaggedString(const QString &_string, Type _type)
: str(_str) : string(_string)
, type(_type) , 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 bool CompletionModel::TaggedString::isEmote() const
{ {
return this->type > Type::EmoteStart && this->type < Type::EmoteEnd; return this->type > Type::EmoteStart && this->type < Type::EmoteEnd;
@ -51,35 +33,23 @@ bool CompletionModel::TaggedString::isEmote() const
bool CompletionModel::TaggedString::operator<(const TaggedString &that) const bool CompletionModel::TaggedString::operator<(const TaggedString &that) const
{ {
if (this->isEmote()) { if (this->isEmote() != that.isEmote()) {
if (that.isEmote()) { return this->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()) { // try comparing insensitively, if they are the same then senstively
return false; // (fixes order of LuL and LUL)
} int k = QString::compare(this->string, that.string, Qt::CaseInsensitive);
if (k == 0) return this->string > that.string;
int k = QString::compare(this->str, that.str, Qt::CaseInsensitive);
if (k == 0) {
return false;
}
return k < 0; return k < 0;
} }
// -- CompletionModel //
// CompletionModel
CompletionModel::CompletionModel(const QString &channelName) //
: channelName_(channelName) CompletionModel::CompletionModel(Channel &channel)
: channel_(channel)
{ {
} }
@ -90,39 +60,32 @@ int CompletionModel::columnCount(const QModelIndex &) const
QVariant CompletionModel::data(const QModelIndex &index, int) const QVariant CompletionModel::data(const QModelIndex &index, int) const
{ {
std::lock_guard<std::mutex> lock(this->emotesMutex_); std::lock_guard<std::mutex> lock(this->itemsMutex_);
// TODO: Implement more safely auto it = this->items_.begin();
auto it = this->emotes_.begin();
std::advance(it, index.row()); std::advance(it, index.row());
return QVariant(it->str); return QVariant(it->string);
} }
int CompletionModel::rowCount(const QModelIndex &) const int CompletionModel::rowCount(const QModelIndex &) const
{ {
std::lock_guard<std::mutex> lock(this->emotesMutex_); std::lock_guard<std::mutex> lock(this->itemsMutex_);
return this->emotes_.size(); return this->items_.size();
} }
void CompletionModel::refresh(const QString &prefix) void CompletionModel::refresh(const QString &prefix)
{ {
{ std::lock_guard<std::mutex> guard(this->itemsMutex_);
std::lock_guard<std::mutex> guard(this->emotesMutex_); this->items_.clear();
this->emotes_.clear();
}
if (prefix.length() < 2) return; if (prefix.length() < 2) return;
BenchmarkGuard guard("CompletionModel::refresh");
auto addString = [&](const QString &str, TaggedString::Type type) { auto addString = [&](const QString &str, TaggedString::Type type) {
if (str.startsWith(prefix)) this->emotes_.emplace(str + " ", type); if (str.startsWith(prefix)) this->items_.emplace(str + " ", type);
}; };
auto _channel = getApp()->twitch2->getChannelOrEmpty(this->channelName_); if (auto channel = dynamic_cast<TwitchChannel *>(&this->channel_)) {
if (auto channel = dynamic_cast<TwitchChannel *>(_channel.get())) {
// account emotes // account emotes
if (auto account = getApp()->accounts->twitch.getCurrent()) { if (auto account = getApp()->accounts->twitch.getCurrent()) {
for (const auto &emote : account->accessEmotes()->allEmoteNames) { for (const auto &emote : account->accessEmotes()->allEmoteNames) {
@ -182,60 +145,4 @@ void CompletionModel::refresh(const QString &prefix)
} }
} }
void CompletionModel::addString(const QString &str, TaggedString::Type type)
{
std::lock_guard<std::mutex> lock(this->emotesMutex_);
// Always add a space at the end of completions
this->emotes_.insert({str + " ", type});
}
void CompletionModel::addUser(const QString &username)
{
auto add = [this](const QString &str) {
auto ts = this->createUser(str + " ");
// Always add a space at the end of completions
auto p = this->emotes_.insert(ts);
if (!p.second) {
// No inseration was made, figure out if we need to replace the
// username.
if (p.first->str > ts.str) {
// Replace lowercase version of name with mixed-case version
this->emotes_.erase(p.first);
auto result2 = this->emotes_.insert(ts);
assert(result2.second);
} else {
p.first->timeAdded = std::chrono::steady_clock::now();
}
}
};
add(username);
add("@" + username);
}
void CompletionModel::clearExpiredStrings()
{
std::lock_guard<std::mutex> lock(this->emotesMutex_);
auto now = std::chrono::steady_clock::now();
for (auto it = this->emotes_.begin(); it != this->emotes_.end();) {
const auto &taggedString = *it;
if (taggedString.isExpired(now)) {
// Log("String {} expired", taggedString.str);
it = this->emotes_.erase(it);
} else {
++it;
}
}
}
CompletionModel::TaggedString CompletionModel::createUser(const QString &str)
{
return TaggedString{str, TaggedString::Type::Username};
}
} // namespace chatterino } // namespace chatterino

View file

@ -8,7 +8,7 @@
namespace chatterino { namespace chatterino {
class TwitchChannel; class Channel;
class CompletionModel : public QAbstractListModel class CompletionModel : public QAbstractListModel
{ {
@ -31,39 +31,30 @@ class CompletionModel : public QAbstractListModel
Command, Command,
}; };
TaggedString(const QString &_str, Type _type); TaggedString(const QString &string, Type type);
bool isExpired(const std::chrono::steady_clock::time_point &now) const;
bool isEmote() const; bool isEmote() const;
bool operator<(const TaggedString &that) const; bool operator<(const TaggedString &that) const;
QString str; QString string;
// Type will help decide the lifetime of the tagged strings
Type type; Type type;
mutable std::chrono::steady_clock::time_point timeAdded;
}; };
public: public:
CompletionModel(const QString &channelName); CompletionModel(Channel &channel);
virtual int columnCount(const QModelIndex &) const override; virtual int columnCount(const QModelIndex &) const override;
virtual QVariant data(const QModelIndex &index, int) const override; virtual QVariant data(const QModelIndex &index, int) const override;
virtual int rowCount(const QModelIndex &) const override; virtual int rowCount(const QModelIndex &) const override;
void refresh(const QString &prefix); void refresh(const QString &prefix);
void addString(const QString &str, TaggedString::Type type);
void addUser(const QString &str);
void clearExpiredStrings();
private: private:
TaggedString createUser(const QString &str); TaggedString createUser(const QString &str);
mutable std::mutex emotesMutex_; std::set<TaggedString> items_;
std::set<TaggedString> emotes_; mutable std::mutex itemsMutex_;
Channel &channel_;
QString channelName_;
}; };
} // namespace chatterino } // namespace chatterino

View file

@ -83,9 +83,6 @@ TwitchChannel::TwitchChannel(const QString &name, BttvEmotes &bttv,
{ {
log("[TwitchChannel:{}] Opened", name); log("[TwitchChannel:{}] Opened", name);
// this->refreshChannelEmotes();
// this->refreshViewerList();
this->managedConnect(getApp()->accounts->twitch.currentUserChanged, this->managedConnect(getApp()->accounts->twitch.currentUserChanged,
[=] { this->setMod(false); }); [=] { this->setMod(false); });
@ -220,9 +217,7 @@ bool TwitchChannel::isBroadcaster() const
void TwitchChannel::addRecentChatter(const MessagePtr &message) void TwitchChannel::addRecentChatter(const MessagePtr &message)
{ {
assert(!message->loginName.isEmpty()); this->chatters_.access()->insert(message->displayName);
this->completionModel.addUser(message->displayName);
} }
void TwitchChannel::addJoinedUser(const QString &user) void TwitchChannel::addJoinedUser(const QString &user)

View file

@ -16,8 +16,8 @@ namespace chatterino {
class Settings; class Settings;
class Paths; class Paths;
class PubSub; class PubSub;
class TwitchChannel;
class TwitchServer final : public AbstractIrcServer, public Singleton class TwitchServer final : public AbstractIrcServer, public Singleton
{ {