2019-09-08 22:27:57 +02:00
|
|
|
#include "HighlightingPage.hpp"
|
|
|
|
|
|
|
|
#include "Application.hpp"
|
|
|
|
#include "controllers/highlights/HighlightBlacklistModel.hpp"
|
|
|
|
#include "controllers/highlights/HighlightModel.hpp"
|
|
|
|
#include "controllers/highlights/UserHighlightModel.hpp"
|
|
|
|
#include "singletons/Settings.hpp"
|
2020-01-25 11:03:10 +01:00
|
|
|
#include "singletons/Theme.hpp"
|
2019-09-08 22:27:57 +02:00
|
|
|
#include "util/LayoutCreator.hpp"
|
|
|
|
#include "util/StandardItemHelper.hpp"
|
2020-01-25 11:03:10 +01:00
|
|
|
#include "widgets/dialogs/ColorPickerDialog.hpp"
|
2019-09-08 22:27:57 +02:00
|
|
|
|
|
|
|
#include <QFileDialog>
|
|
|
|
#include <QHeaderView>
|
|
|
|
#include <QListWidget>
|
|
|
|
#include <QPushButton>
|
|
|
|
#include <QStandardItemModel>
|
|
|
|
#include <QTabWidget>
|
|
|
|
#include <QTableView>
|
|
|
|
#include <QTextEdit>
|
|
|
|
|
|
|
|
#define ENABLE_HIGHLIGHTS "Enable Highlighting"
|
|
|
|
#define HIGHLIGHT_MSG "Highlight messages containing your name"
|
|
|
|
#define PLAY_SOUND "Play sound when your name is mentioned"
|
|
|
|
#define FLASH_TASKBAR "Flash taskbar when your name is mentioned"
|
|
|
|
#define ALWAYS_PLAY "Play highlight sound even when Chatterino is focused"
|
|
|
|
|
|
|
|
namespace chatterino {
|
|
|
|
|
|
|
|
HighlightingPage::HighlightingPage()
|
|
|
|
{
|
|
|
|
auto app = getApp();
|
|
|
|
LayoutCreator<HighlightingPage> layoutCreator(this);
|
|
|
|
|
|
|
|
auto layout = layoutCreator.emplace<QVBoxLayout>().withoutMargin();
|
|
|
|
{
|
|
|
|
// GENERAL
|
|
|
|
// layout.append(this->createCheckBox(ENABLE_HIGHLIGHTS,
|
|
|
|
// getSettings()->enableHighlights));
|
|
|
|
|
|
|
|
// TABS
|
|
|
|
auto tabs = layout.emplace<QTabWidget>();
|
|
|
|
{
|
|
|
|
// HIGHLIGHTS
|
|
|
|
auto highlights = tabs.appendTab(new QVBoxLayout, "Messages");
|
|
|
|
{
|
2020-01-25 11:03:10 +01:00
|
|
|
highlights.emplace<QLabel>(
|
2020-02-10 16:41:49 +01:00
|
|
|
"Play notification sounds and highlight messages based on "
|
|
|
|
"certain patterns.");
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2020-02-23 19:31:43 +01:00
|
|
|
auto view =
|
2019-09-08 22:27:57 +02:00
|
|
|
highlights
|
|
|
|
.emplace<EditableModelView>(
|
2020-02-23 19:31:43 +01:00
|
|
|
(new HighlightModel(nullptr))
|
2020-02-23 21:18:40 +01:00
|
|
|
->initialized(&app->highlightedMessages))
|
2019-09-08 22:27:57 +02:00
|
|
|
.getElement();
|
|
|
|
view->addRegexHelpLink();
|
|
|
|
view->setTitles({"Pattern", "Flash\ntaskbar", "Play\nsound",
|
2020-01-25 11:03:10 +01:00
|
|
|
"Enable\nregex", "Case-\nsensitive",
|
|
|
|
"Custom\nsound", "Color"});
|
2019-09-08 22:27:57 +02:00
|
|
|
view->getTableView()->horizontalHeader()->setSectionResizeMode(
|
|
|
|
QHeaderView::Fixed);
|
|
|
|
view->getTableView()->horizontalHeader()->setSectionResizeMode(
|
|
|
|
0, QHeaderView::Stretch);
|
|
|
|
|
|
|
|
// fourtf: make class extrend BaseWidget and add this to
|
|
|
|
// dpiChanged
|
|
|
|
QTimer::singleShot(1, [view] {
|
|
|
|
view->getTableView()->resizeColumnsToContents();
|
|
|
|
view->getTableView()->setColumnWidth(0, 200);
|
|
|
|
});
|
|
|
|
|
|
|
|
view->addButtonPressed.connect([] {
|
2020-02-23 21:18:40 +01:00
|
|
|
getApp()->highlightedMessages.append(HighlightPhrase{
|
2020-01-25 11:03:10 +01:00
|
|
|
"my phrase", true, false, false, false, "",
|
|
|
|
*ColorProvider::instance().color(
|
|
|
|
ColorType::SelfHighlight)});
|
2019-09-08 22:27:57 +02:00
|
|
|
});
|
2020-01-25 11:03:10 +01:00
|
|
|
|
|
|
|
QObject::connect(view->getTableView(), &QTableView::clicked,
|
|
|
|
[this, view](const QModelIndex &clicked) {
|
|
|
|
this->tableCellClicked(clicked, view);
|
|
|
|
});
|
2019-09-08 22:27:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
auto pingUsers = tabs.appendTab(new QVBoxLayout, "Users");
|
|
|
|
{
|
|
|
|
pingUsers.emplace<QLabel>(
|
2020-02-10 16:41:49 +01:00
|
|
|
"Play notification sounds and highlight messages from "
|
|
|
|
"certain users.\n"
|
|
|
|
"User highlights are prioritized over message "
|
|
|
|
"highlights.");
|
2019-09-08 22:27:57 +02:00
|
|
|
EditableModelView *view =
|
|
|
|
pingUsers
|
|
|
|
.emplace<EditableModelView>(
|
2020-02-23 19:31:43 +01:00
|
|
|
(new UserHighlightModel(nullptr))
|
2020-02-23 21:18:40 +01:00
|
|
|
->initialized(&app->highlightedUsers))
|
2019-09-08 22:27:57 +02:00
|
|
|
.getElement();
|
|
|
|
|
|
|
|
view->addRegexHelpLink();
|
|
|
|
view->getTableView()->horizontalHeader()->hideSection(4);
|
|
|
|
|
|
|
|
// Case-sensitivity doesn't make sense for user names so it is
|
2020-01-25 11:03:10 +01:00
|
|
|
// set to "false" by default & the column is hidden
|
2019-09-08 22:27:57 +02:00
|
|
|
view->setTitles({"Username", "Flash\ntaskbar", "Play\nsound",
|
2020-01-25 11:03:10 +01:00
|
|
|
"Enable\nregex", "Case-\nsensitive",
|
|
|
|
"Custom\nsound", "Color"});
|
2019-09-08 22:27:57 +02:00
|
|
|
view->getTableView()->horizontalHeader()->setSectionResizeMode(
|
|
|
|
QHeaderView::Fixed);
|
|
|
|
view->getTableView()->horizontalHeader()->setSectionResizeMode(
|
|
|
|
0, QHeaderView::Stretch);
|
|
|
|
|
|
|
|
// fourtf: make class extrend BaseWidget and add this to
|
|
|
|
// dpiChanged
|
|
|
|
QTimer::singleShot(1, [view] {
|
|
|
|
view->getTableView()->resizeColumnsToContents();
|
|
|
|
view->getTableView()->setColumnWidth(0, 200);
|
|
|
|
});
|
|
|
|
|
|
|
|
view->addButtonPressed.connect([] {
|
2020-02-23 21:18:40 +01:00
|
|
|
getApp()->highlightedUsers.append(HighlightPhrase{
|
|
|
|
"highlighted user", true, false, false, false, "",
|
|
|
|
*ColorProvider::instance().color(
|
|
|
|
ColorType::SelfHighlight)});
|
2019-09-08 22:27:57 +02:00
|
|
|
});
|
2020-01-25 11:03:10 +01:00
|
|
|
|
|
|
|
QObject::connect(view->getTableView(), &QTableView::clicked,
|
|
|
|
[this, view](const QModelIndex &clicked) {
|
|
|
|
this->tableCellClicked(clicked, view);
|
|
|
|
});
|
2019-09-08 22:27:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
auto disabledUsers =
|
2020-02-10 16:41:49 +01:00
|
|
|
tabs.appendTab(new QVBoxLayout, "Blacklisted Users");
|
2019-09-08 22:27:57 +02:00
|
|
|
{
|
|
|
|
disabledUsers.emplace<QLabel>(
|
2020-02-10 16:41:49 +01:00
|
|
|
"Disable notification sounds and highlights from certain "
|
|
|
|
"users (e.g. bots).");
|
2019-09-08 22:27:57 +02:00
|
|
|
EditableModelView *view =
|
|
|
|
disabledUsers
|
|
|
|
.emplace<EditableModelView>(
|
2020-02-23 19:31:43 +01:00
|
|
|
(new HighlightBlacklistModel(nullptr))
|
2020-02-23 21:18:40 +01:00
|
|
|
->initialized(&app->blacklistedUsers))
|
2019-09-08 22:27:57 +02:00
|
|
|
.getElement();
|
|
|
|
|
|
|
|
view->addRegexHelpLink();
|
2020-02-10 16:41:49 +01:00
|
|
|
view->setTitles({"Username", "Enable\nregex"});
|
2019-09-08 22:27:57 +02:00
|
|
|
view->getTableView()->horizontalHeader()->setSectionResizeMode(
|
|
|
|
QHeaderView::Fixed);
|
|
|
|
view->getTableView()->horizontalHeader()->setSectionResizeMode(
|
|
|
|
0, QHeaderView::Stretch);
|
|
|
|
|
|
|
|
// fourtf: make class extrend BaseWidget and add this to
|
|
|
|
// dpiChanged
|
|
|
|
QTimer::singleShot(1, [view] {
|
|
|
|
view->getTableView()->resizeColumnsToContents();
|
|
|
|
view->getTableView()->setColumnWidth(0, 200);
|
|
|
|
});
|
|
|
|
|
|
|
|
view->addButtonPressed.connect([] {
|
2020-02-23 21:18:40 +01:00
|
|
|
getApp()->blacklistedUsers.append(
|
2019-09-08 22:27:57 +02:00
|
|
|
HighlightBlacklistUser{"blacklisted user", false});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// MISC
|
|
|
|
auto customSound = layout.emplace<QHBoxLayout>().withoutMargin();
|
|
|
|
{
|
2020-01-25 11:03:10 +01:00
|
|
|
auto fallbackSound = customSound.append(this->createCheckBox(
|
|
|
|
"Fallback sound (played when no other sound is set)",
|
|
|
|
getSettings()->customHighlightSound));
|
|
|
|
|
|
|
|
auto getSelectFileText = [] {
|
|
|
|
const QString value = getSettings()->pathHighlightSound;
|
|
|
|
return value.isEmpty() ? "Select custom fallback sound"
|
|
|
|
: QUrl::fromLocalFile(value).fileName();
|
|
|
|
};
|
|
|
|
|
2019-09-08 22:27:57 +02:00
|
|
|
auto selectFile =
|
2020-01-25 11:03:10 +01:00
|
|
|
customSound.emplace<QPushButton>(getSelectFileText());
|
|
|
|
|
|
|
|
QObject::connect(
|
|
|
|
selectFile.getElement(), &QPushButton::clicked, this,
|
|
|
|
[=]() mutable {
|
|
|
|
auto fileName = QFileDialog::getOpenFileName(
|
|
|
|
this, tr("Open Sound"), "",
|
|
|
|
tr("Audio Files (*.mp3 *.wav)"));
|
|
|
|
|
|
|
|
getSettings()->pathHighlightSound = fileName;
|
|
|
|
selectFile.getElement()->setText(getSelectFileText());
|
|
|
|
|
|
|
|
// Set check box according to updated value
|
|
|
|
fallbackSound->setCheckState(
|
|
|
|
fileName.isEmpty() ? Qt::Unchecked : Qt::Checked);
|
|
|
|
});
|
2019-09-08 22:27:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
layout.append(createCheckBox(ALWAYS_PLAY,
|
|
|
|
getSettings()->highlightAlwaysPlaySound));
|
|
|
|
layout.append(createCheckBox(
|
|
|
|
"Flash taskbar only stops highlighting when chatterino is focused",
|
|
|
|
getSettings()->longAlerts));
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---- misc
|
|
|
|
this->disabledUsersChangedTimer_.setSingleShot(true);
|
2020-02-23 19:31:43 +01:00
|
|
|
} // namespace chatterino
|
2019-09-08 22:27:57 +02:00
|
|
|
|
2020-01-25 11:03:10 +01:00
|
|
|
void HighlightingPage::tableCellClicked(const QModelIndex &clicked,
|
|
|
|
EditableModelView *view)
|
|
|
|
{
|
|
|
|
using Column = HighlightModel::Column;
|
|
|
|
|
|
|
|
if (clicked.column() == Column::SoundPath)
|
|
|
|
{
|
|
|
|
auto fileUrl = QFileDialog::getOpenFileUrl(
|
|
|
|
this, tr("Open Sound"), QUrl(), tr("Audio Files (*.mp3 *.wav)"));
|
|
|
|
view->getModel()->setData(clicked, fileUrl, Qt::UserRole);
|
|
|
|
view->getModel()->setData(clicked, fileUrl.fileName(), Qt::DisplayRole);
|
|
|
|
|
|
|
|
// Enable custom sound check box if user set a sound
|
|
|
|
if (!fileUrl.isEmpty())
|
|
|
|
{
|
|
|
|
QModelIndex checkBox = clicked.siblingAtColumn(Column::PlaySound);
|
|
|
|
view->getModel()->setData(checkBox, Qt::Checked,
|
|
|
|
Qt::CheckStateRole);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (clicked.column() == Column::Color)
|
|
|
|
{
|
|
|
|
auto initial =
|
|
|
|
view->getModel()->data(clicked, Qt::DecorationRole).value<QColor>();
|
|
|
|
|
|
|
|
auto dialog = new ColorPickerDialog(initial, this);
|
|
|
|
dialog->setAttribute(Qt::WA_DeleteOnClose);
|
|
|
|
dialog->show();
|
|
|
|
dialog->closed.connect([=] {
|
|
|
|
QColor selected = dialog->selectedColor();
|
|
|
|
|
|
|
|
if (selected.isValid())
|
|
|
|
{
|
|
|
|
view->getModel()->setData(clicked, selected,
|
|
|
|
Qt::DecorationRole);
|
|
|
|
|
2020-01-27 00:16:09 +01:00
|
|
|
// Hacky (?) way to figure out what tab the cell was clicked in
|
|
|
|
const bool fromMessages = (dynamic_cast<HighlightModel *>(
|
|
|
|
view->getModel()) != nullptr);
|
|
|
|
|
|
|
|
if (fromMessages)
|
2020-01-25 11:03:10 +01:00
|
|
|
{
|
2020-01-27 00:16:09 +01:00
|
|
|
/*
|
|
|
|
* For preset highlights in the "Messages" tab, we need to
|
|
|
|
* manually update the color map.
|
|
|
|
*/
|
|
|
|
auto instance = ColorProvider::instance();
|
|
|
|
switch (clicked.row())
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
instance.updateColor(ColorType::SelfHighlight,
|
|
|
|
selected);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
instance.updateColor(ColorType::Whisper, selected);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
instance.updateColor(ColorType::Subscription,
|
|
|
|
selected);
|
|
|
|
break;
|
|
|
|
}
|
2020-01-25 11:03:10 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-08 22:27:57 +02:00
|
|
|
} // namespace chatterino
|