mirror of
https://github.com/Chatterino/chatterino2.git
synced 2024-11-21 22:24:07 +01:00
312 lines
8.6 KiB
C++
312 lines
8.6 KiB
C++
#include "Toasts.hpp"
|
|
|
|
#include "Application.hpp"
|
|
#include "common/Literals.hpp"
|
|
#include "common/QLogging.hpp"
|
|
#include "common/Version.hpp"
|
|
#include "controllers/notifications/NotificationController.hpp"
|
|
#include "providers/twitch/api/Helix.hpp"
|
|
#include "providers/twitch/TwitchIrcServer.hpp"
|
|
#include "singletons/Paths.hpp"
|
|
#include "singletons/Settings.hpp"
|
|
#include "singletons/StreamerMode.hpp"
|
|
#include "util/StreamLink.hpp"
|
|
#include "widgets/helper/CommonTexts.hpp"
|
|
|
|
#ifdef Q_OS_WIN
|
|
# include <wintoastlib.h>
|
|
#endif
|
|
|
|
#include <QDesktopServices>
|
|
#include <QFileInfo>
|
|
#include <QNetworkAccessManager>
|
|
#include <QNetworkReply>
|
|
#include <QStringBuilder>
|
|
#include <QUrl>
|
|
|
|
#include <utility>
|
|
|
|
namespace {
|
|
|
|
using namespace chatterino;
|
|
using namespace literals;
|
|
|
|
QString avatarFilePath(const QString &channelName)
|
|
{
|
|
// TODO: cleanup channel (to be used as a file) and use combinePath
|
|
return getIApp()->getPaths().twitchProfileAvatars % '/' % channelName %
|
|
u".png";
|
|
}
|
|
|
|
bool hasAvatarForChannel(const QString &channelName)
|
|
{
|
|
QFileInfo avatarFile(avatarFilePath(channelName));
|
|
return avatarFile.exists() && avatarFile.isFile();
|
|
}
|
|
|
|
/// A job that downlaods a twitch avatar and saves it to a file
|
|
class AvatarDownloader : public QObject
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
AvatarDownloader(const QString &avatarURL, const QString &channelName);
|
|
|
|
private:
|
|
QNetworkAccessManager manager_;
|
|
QFile file_;
|
|
QNetworkReply *reply_{};
|
|
|
|
signals:
|
|
void downloadComplete();
|
|
};
|
|
|
|
} // namespace
|
|
|
|
namespace chatterino {
|
|
|
|
#ifdef Q_OS_WIN
|
|
using WinToastLib::WinToast;
|
|
using WinToastLib::WinToastTemplate;
|
|
#endif
|
|
|
|
bool Toasts::isEnabled()
|
|
{
|
|
#ifdef Q_OS_WIN
|
|
return WinToast::isCompatible() && getSettings()->notificationToast &&
|
|
!(getIApp()->getStreamerMode()->isEnabled() &&
|
|
getSettings()->streamerModeSuppressLiveNotifications);
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
QString Toasts::findStringFromReaction(const ToastReaction &reaction)
|
|
{
|
|
// The constants are macros right now, but we want to avoid ASCII casts,
|
|
// so we're concatenating them with a QString literal - effectively making them part of it.
|
|
switch (reaction)
|
|
{
|
|
case ToastReaction::OpenInBrowser:
|
|
return OPEN_IN_BROWSER u""_s;
|
|
case ToastReaction::OpenInPlayer:
|
|
return OPEN_PLAYER_IN_BROWSER u""_s;
|
|
case ToastReaction::OpenInStreamlink:
|
|
return OPEN_IN_STREAMLINK u""_s;
|
|
case ToastReaction::DontOpen:
|
|
default:
|
|
return DONT_OPEN u""_s;
|
|
}
|
|
}
|
|
|
|
QString Toasts::findStringFromReaction(
|
|
const pajlada::Settings::Setting<int> &reaction)
|
|
{
|
|
static_assert(std::is_same_v<std::underlying_type_t<ToastReaction>, int>);
|
|
int value = reaction;
|
|
return Toasts::findStringFromReaction(static_cast<ToastReaction>(value));
|
|
}
|
|
|
|
// NOLINTNEXTLINE(readability-convert-member-functions-to-static)
|
|
void Toasts::sendChannelNotification(const QString &channelName,
|
|
const QString &channelTitle, Platform p)
|
|
{
|
|
#ifdef Q_OS_WIN
|
|
auto sendChannelNotification = [this, channelName, channelTitle, p] {
|
|
this->sendWindowsNotification(channelName, channelTitle, p);
|
|
};
|
|
#else
|
|
(void)channelTitle;
|
|
auto sendChannelNotification = [] {
|
|
// Unimplemented for macOS and Linux
|
|
};
|
|
#endif
|
|
// Fetch user profile avatar
|
|
if (p == Platform::Twitch)
|
|
{
|
|
if (hasAvatarForChannel(channelName))
|
|
{
|
|
sendChannelNotification();
|
|
}
|
|
else
|
|
{
|
|
getHelix()->getUserByName(
|
|
channelName,
|
|
[channelName, sendChannelNotification](const auto &user) {
|
|
// gets deleted when finished
|
|
auto *downloader =
|
|
new AvatarDownloader(user.profileImageUrl, channelName);
|
|
QObject::connect(downloader,
|
|
&AvatarDownloader::downloadComplete,
|
|
sendChannelNotification);
|
|
},
|
|
[] {
|
|
// on failure
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef Q_OS_WIN
|
|
|
|
class CustomHandler : public WinToastLib::IWinToastHandler
|
|
{
|
|
private:
|
|
QString channelName_;
|
|
Platform platform_;
|
|
|
|
public:
|
|
CustomHandler(QString channelName, Platform p)
|
|
: channelName_(std::move(channelName))
|
|
, platform_(p)
|
|
{
|
|
}
|
|
void toastActivated() const override
|
|
{
|
|
auto toastReaction =
|
|
static_cast<ToastReaction>(getSettings()->openFromToast.getValue());
|
|
|
|
switch (toastReaction)
|
|
{
|
|
case ToastReaction::OpenInBrowser:
|
|
if (platform_ == Platform::Twitch)
|
|
{
|
|
QDesktopServices::openUrl(
|
|
QUrl(u"https://www.twitch.tv/" % channelName_));
|
|
}
|
|
break;
|
|
case ToastReaction::OpenInPlayer:
|
|
if (platform_ == Platform::Twitch)
|
|
{
|
|
QDesktopServices::openUrl(QUrl(
|
|
u"https://player.twitch.tv/?parent=twitch.tv&channel=" %
|
|
channelName_));
|
|
}
|
|
break;
|
|
case ToastReaction::OpenInStreamlink: {
|
|
openStreamlinkForChannel(channelName_);
|
|
break;
|
|
}
|
|
case ToastReaction::DontOpen:
|
|
// nothing should happen
|
|
break;
|
|
}
|
|
}
|
|
|
|
void toastActivated(int actionIndex) const override
|
|
{
|
|
}
|
|
|
|
void toastFailed() const override
|
|
{
|
|
}
|
|
|
|
void toastDismissed(WinToastDismissalReason state) const override
|
|
{
|
|
}
|
|
};
|
|
|
|
void Toasts::ensureInitialized()
|
|
{
|
|
if (this->initialized_)
|
|
{
|
|
return;
|
|
}
|
|
this->initialized_ = true;
|
|
|
|
auto *instance = WinToast::instance();
|
|
instance->setAppName(L"Chatterino2");
|
|
instance->setAppUserModelId(
|
|
WinToast::configureAUMI(L"", L"Chatterino 2", L"",
|
|
Version::instance().version().toStdWString()));
|
|
instance->setShortcutPolicy(WinToast::SHORTCUT_POLICY_IGNORE);
|
|
WinToast::WinToastError error{};
|
|
instance->initialize(&error);
|
|
|
|
if (error != WinToast::NoError)
|
|
{
|
|
qCDebug(chatterinoNotification)
|
|
<< "Failed to initialize WinToast - error:" << error;
|
|
}
|
|
}
|
|
|
|
void Toasts::sendWindowsNotification(const QString &channelName,
|
|
const QString &channelTitle, Platform p)
|
|
{
|
|
this->ensureInitialized();
|
|
|
|
WinToastTemplate templ(WinToastTemplate::ImageAndText03);
|
|
QString str = channelName % u" is live!";
|
|
|
|
templ.setTextField(str.toStdWString(), WinToastTemplate::FirstLine);
|
|
if (static_cast<ToastReaction>(getSettings()->openFromToast.getValue()) !=
|
|
ToastReaction::DontOpen)
|
|
{
|
|
QString mode =
|
|
Toasts::findStringFromReaction(getSettings()->openFromToast);
|
|
mode = mode.toLower();
|
|
|
|
templ.setTextField(
|
|
u"%1 \nClick to %2"_s.arg(channelTitle).arg(mode).toStdWString(),
|
|
WinToastTemplate::SecondLine);
|
|
}
|
|
|
|
QString avatarPath;
|
|
if (p == Platform::Twitch)
|
|
{
|
|
avatarPath = avatarFilePath(channelName);
|
|
}
|
|
templ.setImagePath(avatarPath.toStdWString());
|
|
if (getSettings()->notificationPlaySound)
|
|
{
|
|
templ.setAudioOption(WinToastTemplate::AudioOption::Silent);
|
|
}
|
|
|
|
WinToast::WinToastError error = WinToast::NoError;
|
|
WinToast::instance()->showToast(templ, new CustomHandler(channelName, p),
|
|
&error);
|
|
if (error != WinToast::NoError)
|
|
{
|
|
qCWarning(chatterinoNotification) << "Failed to show toast:" << error;
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
} // namespace chatterino
|
|
|
|
namespace {
|
|
|
|
AvatarDownloader::AvatarDownloader(const QString &avatarURL,
|
|
const QString &channelName)
|
|
: file_(avatarFilePath(channelName))
|
|
{
|
|
if (!this->file_.open(QFile::WriteOnly | QFile::Truncate))
|
|
{
|
|
qCWarning(chatterinoNotification)
|
|
<< "Failed to open avatar file" << this->file_.errorString();
|
|
}
|
|
|
|
this->reply_ = this->manager_.get(QNetworkRequest(avatarURL));
|
|
|
|
connect(this->reply_, &QNetworkReply::readyRead, this, [this] {
|
|
this->file_.write(this->reply_->readAll());
|
|
});
|
|
connect(this->reply_, &QNetworkReply::finished, this, [this] {
|
|
if (this->reply_->error() != QNetworkReply::NoError)
|
|
{
|
|
qCWarning(chatterinoNotification)
|
|
<< "Failed to download avatar" << this->reply_->errorString();
|
|
}
|
|
|
|
if (this->file_.isOpen())
|
|
{
|
|
this->file_.close();
|
|
}
|
|
emit downloadComplete();
|
|
this->deleteLater();
|
|
});
|
|
}
|
|
|
|
#include "Toasts.moc"
|
|
|
|
} // namespace
|