mirror of
https://github.com/Chatterino/chatterino2.git
synced 2024-11-21 22:24:07 +01:00
This change enforces strict include grouping using IncludeCategories In addition to adding this to the .clang-format file and applying it in the tests/src and src directories, I also did the following small changes: In ChatterSet.hpp, I changed lrucache to a <>include In Irc2.hpp, I change common/SignalVector.hpp to a "project-include" In AttachedWindow.cpp, NativeMessaging.cpp, WindowsHelper.hpp, BaseWindow.cpp, and StreamerMode.cpp, I disabled clang-format for the windows-includes In WindowDescriptors.hpp, I added the missing vector include. It was previously not needed because the include was handled by another file that was previously included first. clang-format minimum version has been bumped, so Ubuntu version used in the check-formatting job has been bumped to 22.04 (which is the latest LTS)
254 lines
6.7 KiB
C++
254 lines
6.7 KiB
C++
#include "common/CompletionModel.hpp"
|
|
|
|
#include "Application.hpp"
|
|
#include "common/ChatterSet.hpp"
|
|
#include "common/Common.hpp"
|
|
#include "controllers/accounts/AccountController.hpp"
|
|
#include "controllers/commands/CommandController.hpp"
|
|
#include "debug/Benchmark.hpp"
|
|
#include "providers/twitch/TwitchChannel.hpp"
|
|
#include "providers/twitch/TwitchCommon.hpp"
|
|
#include "providers/twitch/TwitchIrcServer.hpp"
|
|
#include "singletons/Emotes.hpp"
|
|
#include "singletons/Settings.hpp"
|
|
#include "util/Helpers.hpp"
|
|
#include "util/QStringHash.hpp"
|
|
|
|
#include <QtAlgorithms>
|
|
|
|
#include <utility>
|
|
|
|
namespace chatterino {
|
|
|
|
//
|
|
// TaggedString
|
|
//
|
|
|
|
CompletionModel::TaggedString::TaggedString(const QString &_string, Type _type)
|
|
: string(_string)
|
|
, type(_type)
|
|
{
|
|
}
|
|
|
|
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() != that.isEmote())
|
|
{
|
|
return this->isEmote();
|
|
}
|
|
|
|
return CompletionModel::compareStrings(this->string, that.string);
|
|
}
|
|
|
|
//
|
|
// CompletionModel
|
|
//
|
|
CompletionModel::CompletionModel(Channel &channel)
|
|
: channel_(channel)
|
|
{
|
|
}
|
|
|
|
int CompletionModel::columnCount(const QModelIndex &) const
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
QVariant CompletionModel::data(const QModelIndex &index, int) const
|
|
{
|
|
std::lock_guard<std::mutex> lock(this->itemsMutex_);
|
|
|
|
auto it = this->items_.begin();
|
|
std::advance(it, index.row());
|
|
return QVariant(it->string);
|
|
}
|
|
|
|
int CompletionModel::rowCount(const QModelIndex &) const
|
|
{
|
|
std::lock_guard<std::mutex> lock(this->itemsMutex_);
|
|
|
|
return this->items_.size();
|
|
}
|
|
|
|
void CompletionModel::refresh(const QString &prefix, bool isFirstWord)
|
|
{
|
|
std::lock_guard<std::mutex> guard(this->itemsMutex_);
|
|
this->items_.clear();
|
|
|
|
if (prefix.length() < 2 || !this->channel_.isTwitchChannel())
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Twitch channel
|
|
auto tc = dynamic_cast<TwitchChannel *>(&this->channel_);
|
|
|
|
auto addString = [=](const QString &str, TaggedString::Type type) {
|
|
// Special case for handling default Twitch commands
|
|
if (type == TaggedString::TwitchCommand)
|
|
{
|
|
if (prefix.size() < 2)
|
|
{
|
|
return;
|
|
}
|
|
|
|
auto prefixChar = prefix.at(0);
|
|
|
|
static std::set<QChar> validPrefixChars{'/', '.'};
|
|
|
|
if (validPrefixChars.find(prefixChar) == validPrefixChars.end())
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (startsWithOrContains((prefixChar + str), prefix,
|
|
Qt::CaseInsensitive,
|
|
getSettings()->prefixOnlyEmoteCompletion))
|
|
{
|
|
this->items_.emplace((prefixChar + str + " "), type);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
if (startsWithOrContains(str, prefix, Qt::CaseInsensitive,
|
|
getSettings()->prefixOnlyEmoteCompletion))
|
|
{
|
|
this->items_.emplace(str + " ", type);
|
|
}
|
|
};
|
|
|
|
if (auto account = getApp()->accounts->twitch.getCurrent())
|
|
{
|
|
// Twitch Emotes available globally
|
|
for (const auto &emote : account->accessEmotes()->emotes)
|
|
{
|
|
addString(emote.first.string, TaggedString::TwitchGlobalEmote);
|
|
}
|
|
|
|
// Twitch Emotes available locally
|
|
auto localEmoteData = account->accessLocalEmotes();
|
|
if (tc && localEmoteData->find(tc->roomId()) != localEmoteData->end())
|
|
{
|
|
for (const auto &emote : localEmoteData->at(tc->roomId()))
|
|
{
|
|
addString(emote.first.string,
|
|
TaggedString::Type::TwitchLocalEmote);
|
|
}
|
|
}
|
|
}
|
|
|
|
// 7TV Global
|
|
for (auto &emote : *getApp()->twitch->getSeventvEmotes().globalEmotes())
|
|
{
|
|
addString(emote.first.string, TaggedString::Type::SeventvGlobalEmote);
|
|
}
|
|
// Bttv Global
|
|
for (auto &emote : *getApp()->twitch->getBttvEmotes().emotes())
|
|
{
|
|
addString(emote.first.string, TaggedString::Type::BTTVChannelEmote);
|
|
}
|
|
|
|
// Ffz Global
|
|
for (auto &emote : *getApp()->twitch->getFfzEmotes().emotes())
|
|
{
|
|
addString(emote.first.string, TaggedString::Type::FFZChannelEmote);
|
|
}
|
|
|
|
// Emojis
|
|
if (prefix.startsWith(":"))
|
|
{
|
|
const auto &emojiShortCodes = getApp()->emotes->emojis.shortCodes;
|
|
for (auto &m : emojiShortCodes)
|
|
{
|
|
addString(QString(":%1:").arg(m), TaggedString::Type::Emoji);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Stuff below is available only in regular Twitch channels
|
|
if (!tc)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Usernames
|
|
if (prefix.startsWith("@"))
|
|
{
|
|
QString usernamePrefix = prefix;
|
|
usernamePrefix.remove(0, 1);
|
|
|
|
auto chatters = tc->accessChatters()->filterByPrefix(usernamePrefix);
|
|
|
|
for (const auto &name : chatters)
|
|
{
|
|
addString(
|
|
"@" + formatUserMention(name, isFirstWord,
|
|
getSettings()->mentionUsersWithComma),
|
|
TaggedString::Type::Username);
|
|
}
|
|
}
|
|
else if (!getSettings()->userCompletionOnlyWithAt)
|
|
{
|
|
auto chatters = tc->accessChatters()->filterByPrefix(prefix);
|
|
|
|
for (const auto &name : chatters)
|
|
{
|
|
addString(formatUserMention(name, isFirstWord,
|
|
getSettings()->mentionUsersWithComma),
|
|
TaggedString::Type::Username);
|
|
}
|
|
}
|
|
|
|
// 7TV Channel
|
|
for (auto &emote : *tc->seventvEmotes())
|
|
{
|
|
addString(emote.first.string, TaggedString::Type::SeventvChannelEmote);
|
|
}
|
|
// Bttv Channel
|
|
for (auto &emote : *tc->bttvEmotes())
|
|
{
|
|
addString(emote.first.string, TaggedString::Type::BTTVGlobalEmote);
|
|
}
|
|
|
|
// Ffz Channel
|
|
for (auto &emote : *tc->ffzEmotes())
|
|
{
|
|
addString(emote.first.string, TaggedString::Type::BTTVGlobalEmote);
|
|
}
|
|
|
|
// Custom Chatterino commands
|
|
for (auto &command : getApp()->commands->items)
|
|
{
|
|
addString(command.name, TaggedString::CustomCommand);
|
|
}
|
|
|
|
// Default Chatterino commands
|
|
for (auto &command : getApp()->commands->getDefaultChatterinoCommandList())
|
|
{
|
|
addString(command, TaggedString::ChatterinoCommand);
|
|
}
|
|
|
|
// Default Twitch commands
|
|
for (auto &command : TWITCH_DEFAULT_COMMANDS)
|
|
{
|
|
addString(command, TaggedString::TwitchCommand);
|
|
}
|
|
}
|
|
|
|
bool CompletionModel::compareStrings(const QString &a, const QString &b)
|
|
{
|
|
// try comparing insensitively, if they are the same then senstively
|
|
// (fixes order of LuL and LUL)
|
|
int k = QString::compare(a, b, Qt::CaseInsensitive);
|
|
if (k == 0)
|
|
return a > b;
|
|
|
|
return k < 0;
|
|
}
|
|
|
|
} // namespace chatterino
|