mirror of
https://github.com/Chatterino/chatterino2.git
synced 2024-11-21 22:24:07 +01:00
cleaned up CompletionModel
This commit is contained in:
parent
f6e110b7fb
commit
2448f6f538
5 changed files with 35 additions and 147 deletions
|
@ -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()
|
||||||
|
|
|
@ -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;
|
// try comparing insensitively, if they are the same then senstively
|
||||||
}
|
// (fixes order of LuL and LUL)
|
||||||
|
int k = QString::compare(this->string, that.string, Qt::CaseInsensitive);
|
||||||
return true;
|
if (k == 0) return this->string > that.string;
|
||||||
}
|
|
||||||
|
|
||||||
if (that.isEmote()) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
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
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in a new issue