mirror-chatterino2/src/Application.cpp

701 lines
23 KiB
C++
Raw Normal View History

2018-06-26 14:09:39 +02:00
#include "Application.hpp"
#include "common/Args.hpp"
#include "common/QLogging.hpp"
#include "common/Version.hpp"
2018-06-26 14:09:39 +02:00
#include "controllers/accounts/AccountController.hpp"
#include "controllers/commands/Command.hpp"
2018-06-26 14:09:39 +02:00
#include "controllers/commands/CommandController.hpp"
#include "controllers/highlights/HighlightController.hpp"
#include "controllers/hotkeys/HotkeyController.hpp"
2018-06-26 14:09:39 +02:00
#include "controllers/ignores/IgnoreController.hpp"
#include "controllers/notifications/NotificationController.hpp"
2023-02-02 13:06:29 +01:00
#ifdef CHATTERINO_HAVE_PLUGINS
# include "controllers/plugins/PluginController.hpp"
#endif
#include "controllers/sound/SoundController.hpp"
#include "controllers/userdata/UserDataController.hpp"
#include "debug/AssertInGuiThread.hpp"
#include "messages/Message.hpp"
2018-08-07 01:35:24 +02:00
#include "messages/MessageBuilder.hpp"
#include "providers/bttv/BttvLiveUpdates.hpp"
#include "providers/chatterino/ChatterinoBadges.hpp"
#include "providers/ffz/FfzBadges.hpp"
#include "providers/irc/Irc2.hpp"
#include "providers/seventv/eventapi/Dispatch.hpp"
#include "providers/seventv/eventapi/Subscription.hpp"
#include "providers/seventv/SeventvBadges.hpp"
#include "providers/seventv/SeventvEventAPI.hpp"
#include "providers/twitch/ChannelPointReward.hpp"
#include "providers/twitch/PubSubActions.hpp"
#include "providers/twitch/PubSubManager.hpp"
#include "providers/twitch/PubSubMessages.hpp"
#include "providers/twitch/TwitchChannel.hpp"
#include "providers/twitch/TwitchIrcServer.hpp"
#include "providers/twitch/TwitchMessageBuilder.hpp"
#include "singletons/Emotes.hpp"
2018-06-28 19:46:45 +02:00
#include "singletons/Fonts.hpp"
#include "singletons/helper/LoggingChannel.hpp"
2018-06-28 19:46:45 +02:00
#include "singletons/Logging.hpp"
#include "singletons/Paths.hpp"
#include "singletons/Settings.hpp"
2018-06-28 20:03:04 +02:00
#include "singletons/Theme.hpp"
2018-08-11 12:47:03 +02:00
#include "singletons/Toasts.hpp"
2019-09-02 10:52:01 +02:00
#include "singletons/Updates.hpp"
2018-06-26 14:09:39 +02:00
#include "singletons/WindowManager.hpp"
#include "util/Helpers.hpp"
2018-06-26 14:09:39 +02:00
#include "util/PostToThread.hpp"
2019-09-22 15:32:36 +02:00
#include "widgets/Notebook.hpp"
#include "widgets/splits/Split.hpp"
#include "widgets/Window.hpp"
#include <miniaudio.h>
#include <QDesktopServices>
#include <atomic>
namespace chatterino {
2023-02-08 00:31:56 +01:00
static std::atomic<bool> isAppInitialized{false};
2018-08-02 14:23:27 +02:00
Application *Application::instance = nullptr;
2022-05-22 15:00:18 +02:00
IApplication *IApplication::instance = nullptr;
IApplication::IApplication()
{
IApplication::instance = this;
}
// this class is responsible for handling the workflow of Chatterino
2018-08-06 21:17:03 +02:00
// It will create the instances of the major classes, and connect their signals
// to each other
2018-08-02 14:23:27 +02:00
Application::Application(Settings &_settings, Paths &_paths)
: themes(&this->emplace<Theme>())
2018-08-02 14:23:27 +02:00
, fonts(&this->emplace<Fonts>())
, emotes(&this->emplace<Emotes>())
, accounts(&this->emplace<AccountController>())
, hotkeys(&this->emplace<HotkeyController>())
2018-08-02 14:23:27 +02:00
, windows(&this->emplace<WindowManager>())
2018-08-19 15:09:00 +02:00
, toasts(&this->emplace<Toasts>())
2018-08-19 19:02:49 +02:00
2018-08-02 14:23:27 +02:00
, commands(&this->emplace<CommandController>())
, notifications(&this->emplace<NotificationController>())
, highlights(&this->emplace<HighlightController>())
, twitch(&this->emplace<TwitchIrcServer>())
, chatterinoBadges(&this->emplace<ChatterinoBadges>())
, ffzBadges(&this->emplace<FfzBadges>())
, seventvBadges(&this->emplace<SeventvBadges>())
, userData(&this->emplace<UserDataController>())
, sound(&this->emplace<SoundController>())
2023-02-02 13:06:29 +01:00
#ifdef CHATTERINO_HAVE_PLUGINS
, plugins(&this->emplace<PluginController>())
2023-02-02 13:06:29 +01:00
#endif
2018-08-02 14:23:27 +02:00
, logging(&this->emplace<Logging>())
{
2018-08-02 14:23:27 +02:00
this->instance = this;
this->fonts->fontChanged.connect([this]() {
this->windows->layoutChannelViews();
});
}
2018-08-02 14:23:27 +02:00
void Application::initialize(Settings &settings, Paths &paths)
{
assert(isAppInitialized == false);
isAppInitialized = true;
// Show changelog
if (!getArgs().isFramelessEmbed &&
getSettings()->currentVersion.getValue() != "" &&
getSettings()->currentVersion.getValue() != CHATTERINO_VERSION)
{
auto box = new QMessageBox(QMessageBox::Information, "Chatterino 2",
"Show changelog?",
QMessageBox::Yes | QMessageBox::No);
box->setAttribute(Qt::WA_DeleteOnClose);
if (box->exec() == QMessageBox::Yes)
{
QDesktopServices::openUrl(
QUrl("https://www.chatterino.com/changelog"));
}
}
if (!getArgs().isFramelessEmbed)
{
getSettings()->currentVersion.setValue(CHATTERINO_VERSION);
if (getSettings()->enableExperimentalIrc)
{
Irc::instance().load();
}
}
2019-09-14 20:45:01 +02:00
2018-10-21 13:43:02 +02:00
for (auto &singleton : this->singletons_)
{
2018-08-02 14:23:27 +02:00
singleton->initialize(settings, paths);
2018-07-07 11:41:01 +02:00
}
2019-09-22 15:32:36 +02:00
// add crash message
if (!getArgs().isFramelessEmbed && getArgs().crashRecovery)
2019-09-22 15:32:36 +02:00
{
if (auto selected =
this->windows->getMainWindow().getNotebook().getSelectedPage())
{
if (auto container = dynamic_cast<SplitContainer *>(selected))
{
for (auto &&split : container->getSplits())
{
if (auto channel = split->getChannel(); !channel->isEmpty())
{
channel->addMessage(makeSystemMessage(
2023-02-04 01:55:13 +01:00
"Chatterino unexpectedly crashed and restarted. "
2019-09-22 15:32:36 +02:00
"You can disable automatic restarts in the "
"settings."));
}
}
}
}
}
this->windows->updateWordTypeMask();
if (!getArgs().isFramelessEmbed)
{
this->initNm(paths);
}
this->initPubSub();
this->initBttvLiveUpdates();
this->initSeventvEventAPI();
2018-08-02 14:23:27 +02:00
}
int Application::run(QApplication &qtApp)
{
assert(isAppInitialized);
this->twitch->connect();
2018-08-02 14:23:27 +02:00
if (!getArgs().isFramelessEmbed)
{
this->windows->getMainWindow().show();
}
2018-08-02 14:23:27 +02:00
getSettings()->betaUpdates.connect(
[] {
Updates::instance().checkForUpdates();
},
false);
getSettings()->moderationActions.delayedItemsChanged.connect([this] {
this->windows->forceLayoutChannelViews();
});
2019-09-02 10:52:01 +02:00
getSettings()->highlightedMessages.delayedItemsChanged.connect([this] {
this->windows->forceLayoutChannelViews();
});
getSettings()->highlightedUsers.delayedItemsChanged.connect([this] {
this->windows->forceLayoutChannelViews();
});
getSettings()->removeSpacesBetweenEmotes.connect([this] {
this->windows->forceLayoutChannelViews();
});
getSettings()->enableBTTVGlobalEmotes.connect(
[this] {
this->twitch->reloadBTTVGlobalEmotes();
},
false);
getSettings()->enableBTTVChannelEmotes.connect(
[this] {
this->twitch->reloadAllBTTVChannelEmotes();
},
false);
getSettings()->enableFFZGlobalEmotes.connect(
[this] {
this->twitch->reloadFFZGlobalEmotes();
},
false);
getSettings()->enableFFZChannelEmotes.connect(
[this] {
this->twitch->reloadAllFFZChannelEmotes();
},
false);
getSettings()->enableSevenTVGlobalEmotes.connect(
[this] {
this->twitch->reloadSevenTVGlobalEmotes();
},
false);
getSettings()->enableSevenTVChannelEmotes.connect(
[this] {
this->twitch->reloadAllSevenTVChannelEmotes();
},
false);
2018-08-02 14:23:27 +02:00
return qtApp.exec();
}
2022-11-05 11:04:35 +01:00
IEmotes *Application::getEmotes()
{
return this->emotes;
}
IUserDataController *Application::getUserData()
{
return this->userData;
}
2018-08-02 14:23:27 +02:00
void Application::save()
{
2018-10-21 13:43:02 +02:00
for (auto &singleton : this->singletons_)
{
2018-08-02 14:23:27 +02:00
singleton->save();
}
}
2018-09-17 12:51:16 +02:00
void Application::initNm(Paths &paths)
2018-08-02 14:23:27 +02:00
{
2019-09-08 18:06:43 +02:00
(void)paths;
#ifdef Q_OS_WIN
2019-08-19 23:13:20 +02:00
# if defined QT_NO_DEBUG || defined C_DEBUG_NM
2018-09-17 12:51:16 +02:00
registerNmHost(paths);
this->nmServer.start();
2018-08-15 22:46:20 +02:00
# endif
#endif
2018-08-02 14:23:27 +02:00
}
void Application::initPubSub()
2018-08-02 14:23:27 +02:00
{
this->twitch->pubsub->signals_.moderation.chatCleared.connect(
2018-08-06 21:17:03 +02:00
[this](const auto &action) {
auto chan = this->twitch->getChannelOrEmptyByID(action.roomID);
2018-10-21 13:43:02 +02:00
if (chan->isEmpty())
{
2018-08-06 21:17:03 +02:00
return;
}
2018-04-29 13:24:37 +02:00
2018-08-06 21:17:03 +02:00
QString text =
QString("%1 cleared the chat").arg(action.source.login);
2018-04-29 13:24:37 +02:00
2018-08-07 01:35:24 +02:00
auto msg = makeSystemMessage(text);
postToThread([chan, msg] {
chan->addMessage(msg);
});
2018-08-06 21:17:03 +02:00
});
this->twitch->pubsub->signals_.moderation.modeChanged.connect(
2018-08-06 21:17:03 +02:00
[this](const auto &action) {
auto chan = this->twitch->getChannelOrEmptyByID(action.roomID);
2018-10-21 13:43:02 +02:00
if (chan->isEmpty())
{
2018-08-06 21:17:03 +02:00
return;
}
2018-04-29 13:24:37 +02:00
2018-08-06 21:17:03 +02:00
QString text =
QString("%1 turned %2 %3 mode")
.arg(action.source.login)
2018-08-06 21:17:03 +02:00
.arg(action.state == ModeChangedAction::State::On ? "on"
: "off")
.arg(action.getModeName());
2018-04-29 13:24:37 +02:00
2018-10-21 13:43:02 +02:00
if (action.duration > 0)
{
text += QString(" (%1 seconds)").arg(action.duration);
2018-08-06 21:17:03 +02:00
}
2018-04-29 13:24:37 +02:00
2018-08-07 01:35:24 +02:00
auto msg = makeSystemMessage(text);
postToThread([chan, msg] {
chan->addMessage(msg);
});
2018-08-06 21:17:03 +02:00
});
this->twitch->pubsub->signals_.moderation.moderationStateChanged.connect(
2018-06-26 17:42:35 +02:00
[this](const auto &action) {
auto chan = this->twitch->getChannelOrEmptyByID(action.roomID);
2018-10-21 13:43:02 +02:00
if (chan->isEmpty())
{
2018-06-26 17:42:35 +02:00
return;
}
2018-04-29 13:24:37 +02:00
2018-06-26 17:42:35 +02:00
QString text;
2018-04-29 13:24:37 +02:00
text = QString("%1 %2 %3")
.arg(action.source.login,
(action.modded ? "modded" : "unmodded"),
action.target.login);
2018-04-29 13:24:37 +02:00
2018-08-07 01:35:24 +02:00
auto msg = makeSystemMessage(text);
postToThread([chan, msg] {
chan->addMessage(msg);
});
2018-06-26 17:42:35 +02:00
});
this->twitch->pubsub->signals_.moderation.userBanned.connect(
2018-08-06 21:17:03 +02:00
[&](const auto &action) {
auto chan = this->twitch->getChannelOrEmptyByID(action.roomID);
2018-10-21 13:43:02 +02:00
if (chan->isEmpty())
{
2018-08-06 21:17:03 +02:00
return;
}
postToThread([chan, action] {
MessageBuilder msg(action);
msg->flags.set(MessageFlag::PubSub);
chan->addOrReplaceTimeout(msg.release());
2018-08-07 01:35:24 +02:00
});
2018-08-06 21:17:03 +02:00
});
this->twitch->pubsub->signals_.moderation.messageDeleted.connect(
[&](const auto &action) {
auto chan = this->twitch->getChannelOrEmptyByID(action.roomID);
if (chan->isEmpty() || getSettings()->hideDeletionActions)
{
return;
}
MessageBuilder msg;
TwitchMessageBuilder::deletionMessage(action, &msg);
msg->flags.set(MessageFlag::PubSub);
postToThread([chan, msg = msg.release()] {
auto replaced = false;
LimitedQueueSnapshot<MessagePtr> snapshot =
chan->getMessageSnapshot();
int snapshotLength = snapshot.size();
// without parens it doesn't build on windows
int end = (std::max)(0, snapshotLength - 200);
for (int i = snapshotLength - 1; i >= end; --i)
{
auto &s = snapshot[i];
if (!s->flags.has(MessageFlag::PubSub) &&
s->timeoutUser == msg->timeoutUser)
{
chan->replaceMessage(s, msg);
replaced = true;
break;
}
}
if (!replaced)
{
chan->addMessage(msg);
}
});
});
this->twitch->pubsub->signals_.moderation.userUnbanned.connect(
2018-08-06 21:17:03 +02:00
[&](const auto &action) {
auto chan = this->twitch->getChannelOrEmptyByID(action.roomID);
2018-10-21 13:43:02 +02:00
if (chan->isEmpty())
{
2018-08-06 21:17:03 +02:00
return;
}
2018-08-07 01:35:24 +02:00
auto msg = MessageBuilder(action).release();
postToThread([chan, msg] {
chan->addMessage(msg);
});
2018-08-06 21:17:03 +02:00
});
this->twitch->pubsub->signals_.moderation.autoModMessageCaught.connect(
[&](const auto &msg, const QString &channelID) {
2022-05-07 20:48:10 +02:00
auto chan = this->twitch->getChannelOrEmptyByID(channelID);
if (chan->isEmpty())
{
return;
}
switch (msg.type)
{
case PubSubAutoModQueueMessage::Type::AutoModCaughtMessage: {
if (msg.status == "PENDING")
{
AutomodAction action(msg.data, channelID);
action.reason = QString("%1 level %2")
.arg(msg.contentCategory)
.arg(msg.contentLevel);
action.msgID = msg.messageID;
action.message = msg.messageText;
// this message also contains per-word automod data, which could be implemented
// extract sender data manually because Twitch loves not being consistent
QString senderDisplayName =
msg.senderUserDisplayName; // Might be transformed later
bool hasLocalizedName = false;
if (!msg.senderUserDisplayName.isEmpty())
{
// check for non-ascii display names
if (QString::compare(msg.senderUserDisplayName,
msg.senderUserLogin,
Qt::CaseInsensitive) != 0)
{
hasLocalizedName = true;
}
}
QColor senderColor = msg.senderUserChatColor;
QString senderColor_;
if (!senderColor.isValid() &&
getSettings()->colorizeNicknames)
{
// color may be not present if user is a grey-name
senderColor = getRandomColor(msg.senderUserID);
}
// handle username style based on prefered setting
switch (getSettings()->usernameDisplayMode.getValue())
{
case UsernameDisplayMode::Username: {
if (hasLocalizedName)
{
senderDisplayName = msg.senderUserLogin;
}
break;
}
case UsernameDisplayMode::LocalizedName: {
break;
}
case UsernameDisplayMode::
UsernameAndLocalizedName: {
if (hasLocalizedName)
{
senderDisplayName = QString("%1(%2)").arg(
msg.senderUserLogin,
msg.senderUserDisplayName);
}
break;
}
}
action.target =
ActionUser{msg.senderUserID, msg.senderUserLogin,
senderDisplayName, senderColor};
2022-05-07 20:48:10 +02:00
postToThread([chan, action] {
const auto p = makeAutomodMessage(action);
chan->addMessage(p.first);
chan->addMessage(p.second);
});
}
// "ALLOWED" and "DENIED" statuses remain unimplemented
// They are versions of automod_message_(denied|approved) but for mods.
}
break;
case PubSubAutoModQueueMessage::Type::INVALID:
default: {
}
break;
}
});
this->twitch->pubsub->signals_.moderation.autoModMessageBlocked.connect(
2022-05-07 20:48:10 +02:00
[&](const auto &action) {
auto chan = this->twitch->getChannelOrEmptyByID(action.roomID);
if (chan->isEmpty())
{
return;
}
postToThread([chan, action] {
const auto p = makeAutomodMessage(action);
chan->addMessage(p.first);
chan->addMessage(p.second);
});
});
this->twitch->pubsub->signals_.moderation.automodUserMessage.connect(
[&](const auto &action) {
auto chan = this->twitch->getChannelOrEmptyByID(action.roomID);
if (chan->isEmpty())
{
return;
}
auto msg = MessageBuilder(action).release();
postToThread([chan, msg] {
chan->addMessage(msg);
});
chan->deleteMessage(msg->id);
});
this->twitch->pubsub->signals_.moderation.automodInfoMessage.connect(
[&](const auto &action) {
auto chan = this->twitch->getChannelOrEmptyByID(action.roomID);
if (chan->isEmpty())
{
return;
}
postToThread([chan, action] {
const auto p = makeAutomodInfoMessage(action);
chan->addMessage(p);
});
});
this->twitch->pubsub->signals_.pointReward.redeemed.connect(
[&](auto &data) {
QString channelId = data.value("channel_id").toString();
if (channelId.isEmpty())
{
qCDebug(chatterinoApp)
<< "Couldn't find channel id of point reward";
return;
}
auto chan = this->twitch->getChannelOrEmptyByID(channelId);
auto reward = ChannelPointReward(data);
postToThread([chan, reward] {
if (auto channel = dynamic_cast<TwitchChannel *>(chan.get()))
{
channel->addChannelPointReward(reward);
}
});
});
this->twitch->pubsub->start();
auto RequestModerationActions = [this]() {
this->twitch->pubsub->setAccount(
getApp()->accounts->twitch.getCurrent());
2018-08-06 21:17:03 +02:00
// TODO(pajlada): Unlisten to all authed topics instead of only
// moderation topics this->twitch->pubsub->UnlistenAllAuthedTopics();
this->twitch->pubsub->listenToWhispers();
};
this->accounts->twitch.currentUserChanged.connect(
[this] {
this->twitch->pubsub->unlistenAllModerationActions();
this->twitch->pubsub->unlistenAutomod();
this->twitch->pubsub->unlistenWhispers();
},
boost::signals2::at_front);
2018-05-26 20:26:25 +02:00
this->accounts->twitch.currentUserChanged.connect(RequestModerationActions);
RequestModerationActions();
}
void Application::initBttvLiveUpdates()
{
if (!this->twitch->bttvLiveUpdates)
{
qCDebug(chatterinoBttv)
<< "Skipping initialization of Live Updates as it's disabled";
return;
}
this->twitch->bttvLiveUpdates->signals_.emoteAdded.connect(
[&](const auto &data) {
auto chan = this->twitch->getChannelOrEmptyByID(data.channelID);
postToThread([chan, data] {
if (auto *channel = dynamic_cast<TwitchChannel *>(chan.get()))
{
channel->addBttvEmote(data);
}
});
});
this->twitch->bttvLiveUpdates->signals_.emoteUpdated.connect(
[&](const auto &data) {
auto chan = this->twitch->getChannelOrEmptyByID(data.channelID);
postToThread([chan, data] {
if (auto *channel = dynamic_cast<TwitchChannel *>(chan.get()))
{
channel->updateBttvEmote(data);
}
});
});
this->twitch->bttvLiveUpdates->signals_.emoteRemoved.connect(
[&](const auto &data) {
auto chan = this->twitch->getChannelOrEmptyByID(data.channelID);
postToThread([chan, data] {
if (auto *channel = dynamic_cast<TwitchChannel *>(chan.get()))
{
channel->removeBttvEmote(data);
}
});
});
this->twitch->bttvLiveUpdates->start();
}
void Application::initSeventvEventAPI()
{
if (!this->twitch->seventvEventAPI)
{
qCDebug(chatterinoSeventvEventAPI)
<< "Skipping initialization as the EventAPI is disabled";
return;
}
this->twitch->seventvEventAPI->signals_.emoteAdded.connect(
[&](const auto &data) {
postToThread([this, data] {
this->twitch->forEachSeventvEmoteSet(
data.emoteSetID, [data](TwitchChannel &chan) {
chan.addSeventvEmote(data);
});
});
});
this->twitch->seventvEventAPI->signals_.emoteUpdated.connect(
[&](const auto &data) {
postToThread([this, data] {
this->twitch->forEachSeventvEmoteSet(
data.emoteSetID, [data](TwitchChannel &chan) {
chan.updateSeventvEmote(data);
});
});
});
this->twitch->seventvEventAPI->signals_.emoteRemoved.connect(
[&](const auto &data) {
postToThread([this, data] {
this->twitch->forEachSeventvEmoteSet(
data.emoteSetID, [data](TwitchChannel &chan) {
chan.removeSeventvEmote(data);
});
});
});
this->twitch->seventvEventAPI->signals_.userUpdated.connect(
[&](const auto &data) {
this->twitch->forEachSeventvUser(data.userID,
[data](TwitchChannel &chan) {
chan.updateSeventvUser(data);
});
});
this->twitch->seventvEventAPI->start();
}
Application *getApp()
{
2018-08-02 14:23:27 +02:00
assert(Application::instance != nullptr);
assertInGuiThread();
2018-08-02 14:23:27 +02:00
return Application::instance;
}
2022-05-22 15:00:18 +02:00
IApplication *getIApp()
{
assert(IApplication::instance != nullptr);
assertInGuiThread();
return IApplication::instance;
}
} // namespace chatterino