#include "widgets/splits/Split.hpp" #include "common/Common.hpp" #include "common/NetworkRequest.hpp" #include "debug/Log.hpp" #include "providers/twitch/EmoteValue.hpp" #include "providers/twitch/TwitchChannel.hpp" #include "providers/twitch/TwitchMessageBuilder.hpp" #include "providers/twitch/TwitchServer.hpp" #include "singletons/Settings.hpp" #include "singletons/Theme.hpp" #include "singletons/WindowManager.hpp" #include "util/StreamLink.hpp" #include "widgets/Notebook.hpp" #include "widgets/Window.hpp" #include "widgets/dialogs/QualityPopup.hpp" #include "widgets/dialogs/SelectChannelDialog.hpp" #include "widgets/dialogs/TextInputDialog.hpp" #include "widgets/dialogs/UserInfoPopup.hpp" #include "widgets/helper/ChannelView.hpp" #include "widgets/helper/DebugPopup.hpp" #include "widgets/helper/ResizingTextEdit.hpp" #include "widgets/helper/SearchPopup.hpp" #include "widgets/helper/Shortcut.hpp" #include "widgets/splits/SplitContainer.hpp" #include "widgets/splits/SplitHeader.hpp" #include "widgets/splits/SplitInput.hpp" #include "widgets/splits/SplitOverlay.hpp" #include #include #include #include #include #include #include #include #include #include #include #include namespace chatterino { pajlada::Signals::Signal Split::modifierStatusChanged; Qt::KeyboardModifiers Split::modifierStatus = Qt::NoModifier; Split::Split(SplitContainer *parent) : Split(static_cast(parent)) { this->container_ = parent; } Split::Split(QWidget *parent) : BaseWidget(parent) , container_(nullptr) , channel_(Channel::getEmpty()) , vbox_(new QVBoxLayout(this)) , header_(new SplitHeader(this)) , view_(new ChannelView(this)) , input_(new SplitInput(this)) , overlay_(new SplitOverlay(this)) { this->setMouseTracking(true); this->vbox_->setSpacing(0); this->vbox_->setMargin(1); this->vbox_->addWidget(this->header_); this->vbox_->addWidget(this->view_, 1); this->vbox_->addWidget(this->input_); // Initialize chat widget-wide hotkeys // CTRL+W: Close Split createShortcut(this, "CTRL+W", &Split::deleteFromContainer); // CTRL+R: Change Channel createShortcut(this, "CTRL+R", &Split::changeChannel); // CTRL+F: Search createShortcut(this, "CTRL+F", &Split::showSearch); // F12 createShortcut(this, "F10", [] { auto *popup = new DebugPopup; popup->setAttribute(Qt::WA_DeleteOnClose); popup->show(); }); // xd // CreateShortcut(this, "ALT+SHIFT+RIGHT", &Split::doIncFlexX); // CreateShortcut(this, "ALT+SHIFT+LEFT", &Split::doDecFlexX); // CreateShortcut(this, "ALT+SHIFT+UP", &Split::doIncFlexY); // CreateShortcut(this, "ALT+SHIFT+DOWN", &Split::doDecFlexY); this->input_->ui_.textEdit->installEventFilter(parent); this->view_->mouseDown.connect([this](QMouseEvent *) { // this->giveFocus(Qt::MouseFocusReason); }); this->view_->selectionChanged.connect([this]() { if (view_->hasSelection()) { this->input_->clearSelection(); } }); this->input_->textChanged.connect([=](const QString &newText) { if (getSettings()->showEmptyInput) { return; } if (newText.length() == 0) { this->input_->hide(); } else if (this->input_->isHidden()) { this->input_->show(); } }); getSettings()->showEmptyInput.connect( [this](const bool &showEmptyInput, auto) { if (!showEmptyInput && this->input_->getInputText().length() == 0) { this->input_->hide(); } else { this->input_->show(); } }, this->managedConnections_); this->header_->updateModerationModeIcon(); this->overlay_->hide(); this->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding); this->managedConnect(modifierStatusChanged, [this](Qt::KeyboardModifiers status) { if ((status == showSplitOverlayModifiers /*|| status == showAddSplitRegions*/) && this->isMouseOver_) { this->overlay_->show(); } else { this->overlay_->hide(); } }); this->input_->ui_.textEdit->focused.connect( [this] { this->focused.invoke(); }); this->input_->ui_.textEdit->focusLost.connect( [this] { this->focusLost.invoke(); }); } Split::~Split() { this->usermodeChangedConnection_.disconnect(); this->roomModeChangedConnection_.disconnect(); this->channelIDChangedConnection_.disconnect(); this->indirectChannelChangedConnection_.disconnect(); } ChannelView &Split::getChannelView() { return *this->view_; } SplitContainer *Split::getContainer() { return this->container_; } bool Split::isInContainer() const { return this->container_ != nullptr; } void Split::setContainer(SplitContainer *container) { this->container_ = container; } IndirectChannel Split::getIndirectChannel() { return this->channel_; } ChannelPtr Split::getChannel() { return this->channel_.get(); } void Split::setChannel(IndirectChannel newChannel) { this->channel_ = newChannel; this->view_->setChannel(newChannel.get()); this->usermodeChangedConnection_.disconnect(); this->roomModeChangedConnection_.disconnect(); this->indirectChannelChangedConnection_.disconnect(); TwitchChannel *tc = dynamic_cast(newChannel.get().get()); if (tc != nullptr) { this->usermodeChangedConnection_ = tc->userStateChanged.connect([this] { this->header_->updateModerationModeIcon(); this->header_->updateRoomModes(); }); this->roomModeChangedConnection_ = tc->roomModesChanged.connect( [this] { this->header_->updateRoomModes(); }); } this->indirectChannelChangedConnection_ = newChannel.getChannelChanged().connect([this] { // QTimer::singleShot(0, [this] { this->setChannel(this->channel_); }); }); this->header_->updateModerationModeIcon(); this->header_->updateChannelText(); this->header_->updateRoomModes(); this->channelChanged.invoke(); } void Split::setModerationMode(bool value) { if (value != this->moderationMode_) { this->moderationMode_ = value; this->header_->updateModerationModeIcon(); this->view_->layoutMessages(); } } bool Split::getModerationMode() const { return this->moderationMode_; } void Split::insertTextToInput(const QString &text) { this->input_->insertText(text); } void Split::showChangeChannelPopup(const char *dialogTitle, bool empty, std::function callback) { if (this->selectChannelDialog_.hasElement()) { this->selectChannelDialog_->raise(); return; } SelectChannelDialog *dialog = new SelectChannelDialog(this); if (!empty) { dialog->setSelectedChannel(this->getIndirectChannel()); } dialog->setAttribute(Qt::WA_DeleteOnClose); dialog->show(); dialog->closed.connect([=] { if (dialog->hasSeletedChannel()) { this->setChannel(dialog->getSelectedChannel()); if (this->isInContainer()) { this->container_->refreshTabTitle(); } } callback(dialog->hasSeletedChannel()); this->selectChannelDialog_ = nullptr; }); this->selectChannelDialog_ = dialog; } void Split::layoutMessages() { this->view_->layoutMessages(); } void Split::updateGifEmotes() { this->view_->queueUpdate(); } void Split::updateLastReadMessage() { this->view_->updateLastReadMessage(); } void Split::giveFocus(Qt::FocusReason reason) { this->input_->ui_.textEdit->setFocus(reason); } bool Split::hasFocus() const { return this->input_->ui_.textEdit->hasFocus(); } void Split::paintEvent(QPaintEvent *) { // color the background of the chat QPainter painter(this); painter.fillRect(this->rect(), this->theme->splits.background); } void Split::mouseMoveEvent(QMouseEvent *event) { this->handleModifiers(QGuiApplication::queryKeyboardModifiers()); } void Split::keyPressEvent(QKeyEvent *event) { this->view_->unsetCursor(); this->handleModifiers(QGuiApplication::queryKeyboardModifiers()); } void Split::keyReleaseEvent(QKeyEvent *event) { this->view_->unsetCursor(); this->handleModifiers(QGuiApplication::queryKeyboardModifiers()); } void Split::resizeEvent(QResizeEvent *event) { BaseWidget::resizeEvent(event); this->overlay_->setGeometry(this->rect()); } void Split::enterEvent(QEvent *event) { this->isMouseOver_ = true; this->handleModifiers(QGuiApplication::queryKeyboardModifiers()); if (modifierStatus == showSplitOverlayModifiers /*|| modifierStatus == showAddSplitRegions*/) { this->overlay_->show(); } if (this->container_ != nullptr) { this->container_->resetMouseStatus(); } } void Split::leaveEvent(QEvent *event) { this->isMouseOver_ = false; this->overlay_->hide(); this->handleModifiers(QGuiApplication::queryKeyboardModifiers()); } void Split::focusInEvent(QFocusEvent *event) { this->giveFocus(event->reason()); } void Split::handleModifiers(Qt::KeyboardModifiers modifiers) { if (modifierStatus != modifiers) { modifierStatus = modifiers; modifierStatusChanged.invoke(modifiers); } } void Split::setIsTopRightSplit(bool value) { this->isTopRightSplit_ = value; this->header_->setAddButtonVisible(value); } /// Slots void Split::addSibling() { if (this->container_) { this->container_->appendNewSplit(true); } } void Split::deleteFromContainer() { if (this->container_) { this->container_->deleteSplit(this); } } void Split::changeChannel() { this->showChangeChannelPopup("Change channel", false, [](bool) {}); auto popup = this->findChildren(); if (popup.size() && popup.at(0)->isVisible() && !popup.at(0)->isFloating()) { popup.at(0)->hide(); showViewerList(); } } void Split::popup() { auto app = getApp(); Window &window = app->windows->createWindow(WindowType::Popup); Split *split = new Split(static_cast( window.getNotebook().getOrAddSelectedPage())); split->setChannel(this->getIndirectChannel()); window.getNotebook().getOrAddSelectedPage()->appendSplit(split); window.show(); } void Split::clear() { this->view_->clearMessages(); } void Split::openInBrowser() { auto channel = this->getChannel(); if (auto twitchChannel = dynamic_cast(channel.get())) { QDesktopServices::openUrl("https://twitch.tv/" + twitchChannel->getName()); } } void Split::openBrowserPlayer() { ChannelPtr channel = this->getChannel(); if (auto twitchChannel = dynamic_cast(channel.get())) { QDesktopServices::openUrl("https://player.twitch.tv/?channel=" + twitchChannel->getName()); } } void Split::openInStreamlink() { try { openStreamlinkForChannel(this->getChannel()->getName()); } catch (const Exception &ex) { log("Error in doOpenStreamlink: {}", ex.what()); } } void Split::showViewerList() { auto viewerDock = new QDockWidget("Viewer List", this); viewerDock->setAllowedAreas(Qt::LeftDockWidgetArea); viewerDock->setFeatures(QDockWidget::DockWidgetVerticalTitleBar | QDockWidget::DockWidgetClosable | QDockWidget::DockWidgetFloatable); viewerDock->resize( 0.5 * this->width(), this->height() - this->header_->height() - this->input_->height()); viewerDock->move(0, this->header_->height()); auto multiWidget = new QWidget(viewerDock); auto dockVbox = new QVBoxLayout(viewerDock); auto searchBar = new QLineEdit(viewerDock); auto chattersList = new QListWidget(); auto resultList = new QListWidget(); static QStringList labels = {"Moderators", "Staff", "Admins", "Global Moderators", "Viewers"}; static QStringList jsonLabels = {"moderators", "staff", "admins", "global_mods", "viewers"}; QList labelList; for (auto &x : labels) { auto label = new QListWidgetItem(x); label->setBackgroundColor(this->theme->splits.header.background); labelList.append(label); } auto loadingLabel = new QLabel("Loading..."); auto request = NetworkRequest::twitchRequest( "https://tmi.twitch.tv/group/user/" + this->getChannel()->getName() + "/chatters"); request.setCaller(this); request.onSuccess([=](auto result) -> Outcome { auto obj = result.parseJson(); QJsonObject chattersObj = obj.value("chatters").toObject(); loadingLabel->hide(); for (int i = 0; i < jsonLabels.size(); i++) { auto currentCategory = chattersObj.value(jsonLabels.at(i)).toArray(); // If current category of chatters is empty, dont show this category. if (currentCategory.size() == 0) continue; chattersList->addItem(labelList.at(i)); foreach (const QJsonValue &v, currentCategory) chattersList->addItem(v.toString()); } return Success; }); request.execute(); searchBar->setPlaceholderText("Search User..."); QObject::connect(searchBar, &QLineEdit::textEdited, this, [=]() { auto query = searchBar->text(); if (!query.isEmpty()) { auto results = chattersList->findItems(query, Qt::MatchStartsWith); chattersList->hide(); resultList->clear(); for (auto &item : results) { if (!labels.contains(item->text())) resultList->addItem(item->text()); } resultList->show(); } else { resultList->hide(); chattersList->show(); } }); QObject::connect(viewerDock, &QDockWidget::topLevelChanged, this, [=]() { viewerDock->setMinimumWidth(300); }); QObject::connect(chattersList, &QListWidget::doubleClicked, this, [=]() { if (!labels.contains(chattersList->currentItem()->text())) { showUserInfoPopup(chattersList->currentItem()->text()); } }); QObject::connect(resultList, &QListWidget::doubleClicked, this, [=]() { if (!labels.contains(resultList->currentItem()->text())) { showUserInfoPopup(resultList->currentItem()->text()); } }); dockVbox->addWidget(searchBar); dockVbox->addWidget(loadingLabel); dockVbox->addWidget(chattersList); dockVbox->addWidget(resultList); resultList->hide(); multiWidget->setStyleSheet(this->theme->splits.input.styleSheet); multiWidget->setLayout(dockVbox); viewerDock->setWidget(multiWidget); viewerDock->show(); } void Split::showUserInfoPopup(const UserName &user) { auto *userPopup = new UserInfoPopup; userPopup->setData(user.string, this->getChannel()); userPopup->setAttribute(Qt::WA_DeleteOnClose); userPopup->move(QCursor::pos() - QPoint(int(150 * this->getScale()), int(70 * this->getScale()))); userPopup->show(); } void Split::copyToClipboard() { QApplication::clipboard()->setText(this->view_->getSelectedText()); } void Split::showSearch() { SearchPopup *popup = new SearchPopup(); popup->setChannel(this->getChannel()); popup->show(); } template static Iter select_randomly(Iter start, Iter end, RandomGenerator &g) { std::uniform_int_distribution<> dis(0, std::distance(start, end) - 1); std::advance(start, dis(g)); return start; } template static Iter select_randomly(Iter start, Iter end) { static std::random_device rd; static std::mt19937 gen(rd()); return select_randomly(start, end, gen); } void Split::drag() { if (auto container = dynamic_cast(this->parentWidget())) { SplitContainer::isDraggingSplit = true; SplitContainer::draggingSplit = this; auto originalLocation = container->releaseSplit(this); auto drag = new QDrag(this); auto mimeData = new QMimeData; mimeData->setData("chatterino/split", "xD"); drag->setMimeData(mimeData); if (drag->exec(Qt::MoveAction) == Qt::IgnoreAction) { container->insertSplit(this, originalLocation); } SplitContainer::isDraggingSplit = false; } } } // namespace chatterino