// SPDX-License-Identifier: GPL-3.0-only /* * Prism Launcher - Minecraft Launcher * Copyright (c) 2022 Jamie Mansfield <jmansfield@cadixdev.org> * Copyright (C) 2022 TheKodeToad <TheKodeToad@proton.me> * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, version 3. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <https://www.gnu.org/licenses/>. * * This file incorporates work covered by the following copyright and * permission notice: * * Copyright 2013-2021 MultiMC Contributors * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "OtherLogsPage.h" #include "ui_OtherLogsPage.h" #include <QMessageBox> #include "ui/GuiUtil.h" #include "RecursiveFileSystemWatcher.h" #include <GZip.h> #include <FileSystem.h> #include <QShortcut> OtherLogsPage::OtherLogsPage(QString path, IPathMatcher::Ptr fileFilter, QWidget *parent) : QWidget(parent), ui(new Ui::OtherLogsPage), m_path(path), m_fileFilter(fileFilter), m_watcher(new RecursiveFileSystemWatcher(this)) { ui->setupUi(this); ui->tabWidget->tabBar()->hide(); m_watcher->setMatcher(fileFilter); m_watcher->setRootDir(QDir::current().absoluteFilePath(m_path)); connect(m_watcher, &RecursiveFileSystemWatcher::filesChanged, this, &OtherLogsPage::populateSelectLogBox); populateSelectLogBox(); auto findShortcut = new QShortcut(QKeySequence(QKeySequence::Find), this); connect(findShortcut, &QShortcut::activated, this, &OtherLogsPage::findActivated); auto findNextShortcut = new QShortcut(QKeySequence(QKeySequence::FindNext), this); connect(findNextShortcut, &QShortcut::activated, this, &OtherLogsPage::findNextActivated); auto findPreviousShortcut = new QShortcut(QKeySequence(QKeySequence::FindPrevious), this); connect(findPreviousShortcut, &QShortcut::activated, this, &OtherLogsPage::findPreviousActivated); connect(ui->searchBar, &QLineEdit::returnPressed, this, &OtherLogsPage::on_findButton_clicked); } OtherLogsPage::~OtherLogsPage() { delete ui; } void OtherLogsPage::retranslate() { ui->retranslateUi(this); } void OtherLogsPage::openedImpl() { m_watcher->enable(); } void OtherLogsPage::closedImpl() { m_watcher->disable(); } void OtherLogsPage::populateSelectLogBox() { ui->selectLogBox->clear(); ui->selectLogBox->addItems(m_watcher->files()); if (m_currentFile.isEmpty()) { setControlsEnabled(false); ui->selectLogBox->setCurrentIndex(-1); } else { const int index = ui->selectLogBox->findText(m_currentFile); if (index != -1) { ui->selectLogBox->setCurrentIndex(index); setControlsEnabled(true); } else { setControlsEnabled(false); } } } void OtherLogsPage::on_selectLogBox_currentIndexChanged(const int index) { QString file; if (index != -1) { file = ui->selectLogBox->itemText(index); } if (file.isEmpty() || !QFile::exists(FS::PathCombine(m_path, file))) { m_currentFile = QString(); ui->text->clear(); setControlsEnabled(false); } else { m_currentFile = file; on_btnReload_clicked(); setControlsEnabled(true); } } void OtherLogsPage::on_btnReload_clicked() { if(m_currentFile.isEmpty()) { setControlsEnabled(false); return; } QFile file(FS::PathCombine(m_path, m_currentFile)); if (!file.open(QFile::ReadOnly)) { setControlsEnabled(false); ui->btnReload->setEnabled(true); // allow reload m_currentFile = QString(); QMessageBox::critical(this, tr("Error"), tr("Unable to open %1 for reading: %2") .arg(m_currentFile, file.errorString())); } else { auto setPlainText = [&](const QString & text) { QString fontFamily = APPLICATION->settings()->get("ConsoleFont").toString(); bool conversionOk = false; int fontSize = APPLICATION->settings()->get("ConsoleFontSize").toInt(&conversionOk); if(!conversionOk) { fontSize = 11; } QTextDocument *doc = ui->text->document(); doc->setDefaultFont(QFont(fontFamily, fontSize)); ui->text->setPlainText(text); }; auto showTooBig = [&]() { setPlainText( tr("The file (%1) is too big. You may want to open it in a viewer optimized " "for large files.").arg(file.fileName())); }; if(file.size() > (1024ll * 1024ll * 12ll)) { showTooBig(); return; } QString content; if(file.fileName().endsWith(".gz")) { QByteArray temp; if(!GZip::unzip(file.readAll(), temp)) { setPlainText( tr("The file (%1) is not readable.").arg(file.fileName())); return; } content = QString::fromUtf8(temp); } else { content = QString::fromUtf8(file.readAll()); } if (content.size() >= 50000000ll) { showTooBig(); return; } setPlainText(content); } } void OtherLogsPage::on_btnPaste_clicked() { GuiUtil::uploadPaste(m_currentFile, ui->text->toPlainText(), this); } void OtherLogsPage::on_btnCopy_clicked() { GuiUtil::setClipboardText(ui->text->toPlainText()); } void OtherLogsPage::on_btnDelete_clicked() { if(m_currentFile.isEmpty()) { setControlsEnabled(false); return; } if (QMessageBox::question(this, tr("Confirm Deletion"), tr("You are about to delete \"%1\".\n" "This may be permanent and it will be gone from the logs folder.\n\n" "Are you sure?") .arg(m_currentFile), QMessageBox::Yes, QMessageBox::No) == QMessageBox::No) { return; } QFile file(FS::PathCombine(m_path, m_currentFile)); if (FS::trash(file.fileName())) { return; } if (!file.remove()) { QMessageBox::critical(this, tr("Error"), tr("Unable to delete %1: %2") .arg(m_currentFile, file.errorString())); } } void OtherLogsPage::on_btnClean_clicked() { auto toDelete = m_watcher->files(); if(toDelete.isEmpty()) { return; } QMessageBox *messageBox = new QMessageBox(this); messageBox->setWindowTitle(tr("Confirm Cleanup")); if(toDelete.size() > 5) { messageBox->setText(tr("Are you sure you want to delete all log files?")); messageBox->setDetailedText(toDelete.join('\n')); } else { messageBox->setText(tr("Are you sure you want to delete all these files?\n%1").arg(toDelete.join('\n'))); } messageBox->setStandardButtons(QMessageBox::Ok | QMessageBox::Cancel); messageBox->setDefaultButton(QMessageBox::Ok); messageBox->setTextInteractionFlags(Qt::TextSelectableByMouse); messageBox->setIcon(QMessageBox::Question); messageBox->setTextInteractionFlags(Qt::TextBrowserInteraction); if (messageBox->exec() != QMessageBox::Ok) { return; } QStringList failed; for(auto item: toDelete) { QFile file(FS::PathCombine(m_path, item)); if (FS::trash(file.fileName())) { continue; } if (!file.remove()) { failed.push_back(item); } } if(!failed.empty()) { QMessageBox *messageBox = new QMessageBox(this); messageBox->setWindowTitle(tr("Error")); if(failed.size() > 5) { messageBox->setText(tr("Couldn't delete some files!")); messageBox->setDetailedText(failed.join('\n')); } else { messageBox->setText(tr("Couldn't delete some files:\n%1").arg(failed.join('\n'))); } messageBox->setStandardButtons(QMessageBox::Ok); messageBox->setDefaultButton(QMessageBox::Ok); messageBox->setTextInteractionFlags(Qt::TextSelectableByMouse); messageBox->setIcon(QMessageBox::Critical); messageBox->setTextInteractionFlags(Qt::TextBrowserInteraction); messageBox->exec(); } } void OtherLogsPage::setControlsEnabled(const bool enabled) { ui->btnReload->setEnabled(enabled); ui->btnDelete->setEnabled(enabled); ui->btnCopy->setEnabled(enabled); ui->btnPaste->setEnabled(enabled); ui->text->setEnabled(enabled); ui->btnClean->setEnabled(enabled); } // FIXME: HACK, use LogView instead? static void findNext(QPlainTextEdit * _this, const QString& what, bool reverse) { _this->find(what, reverse ? QTextDocument::FindFlag::FindBackward : QTextDocument::FindFlag(0)); } void OtherLogsPage::on_findButton_clicked() { auto modifiers = QApplication::keyboardModifiers(); bool reverse = modifiers & Qt::ShiftModifier; findNext(ui->text, ui->searchBar->text(), reverse); } void OtherLogsPage::findNextActivated() { findNext(ui->text, ui->searchBar->text(), false); } void OtherLogsPage::findPreviousActivated() { findNext(ui->text, ui->searchBar->text(), true); } void OtherLogsPage::findActivated() { // focus the search bar if it doesn't have focus if (!ui->searchBar->hasFocus()) { ui->searchBar->setFocus(); ui->searchBar->selectAll(); } }