2013-11-04 01:53:05 +00:00
|
|
|
/* Copyright 2013 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 "ConsoleWindow.h"
|
|
|
|
#include "ui_ConsoleWindow.h"
|
2013-11-23 00:41:28 +00:00
|
|
|
#include "MultiMC.h"
|
2013-02-22 17:18:23 +00:00
|
|
|
|
|
|
|
#include <QScrollBar>
|
2013-09-06 21:40:50 +01:00
|
|
|
#include <QMessageBox>
|
2014-01-06 01:52:51 +00:00
|
|
|
#include <QSystemTrayIcon>
|
2013-02-22 17:18:23 +00:00
|
|
|
|
2013-11-04 01:53:05 +00:00
|
|
|
#include <gui/Platform.h>
|
|
|
|
#include <gui/dialogs/CustomMessageBox.h>
|
2013-12-06 21:24:55 +00:00
|
|
|
#include <gui/dialogs/ProgressDialog.h>
|
2014-05-18 18:07:01 +01:00
|
|
|
#include "dialogs/ScreenshotDialog.h"
|
2013-12-06 21:24:55 +00:00
|
|
|
|
|
|
|
#include "logic/net/PasteUpload.h"
|
2014-01-06 01:52:51 +00:00
|
|
|
#include "logic/icons/IconList.h"
|
2014-05-18 18:07:01 +01:00
|
|
|
#include <logic/screenshots/ScreenshotList.h>
|
2013-10-18 17:42:41 +01:00
|
|
|
|
2013-11-12 23:24:49 +00:00
|
|
|
ConsoleWindow::ConsoleWindow(MinecraftProcess *mcproc, QWidget *parent)
|
2013-11-23 00:41:28 +00:00
|
|
|
: QMainWindow(parent), ui(new Ui::ConsoleWindow), proc(mcproc)
|
2013-02-22 17:18:23 +00:00
|
|
|
{
|
2013-10-29 18:38:11 +00:00
|
|
|
MultiMCPlatform::fixWM_CLASS(this);
|
2013-03-22 13:40:55 +00:00
|
|
|
ui->setupUi(this);
|
2013-11-23 00:41:28 +00:00
|
|
|
connect(mcproc, SIGNAL(log(QString, MessageLevel::Enum)), this,
|
|
|
|
SLOT(write(QString, MessageLevel::Enum)));
|
2014-05-18 18:07:01 +01:00
|
|
|
connect(mcproc, SIGNAL(ended(InstancePtr, int, QProcess::ExitStatus)), this,
|
|
|
|
SLOT(onEnded(InstancePtr, int, QProcess::ExitStatus)));
|
|
|
|
connect(mcproc, SIGNAL(prelaunch_failed(InstancePtr, int, QProcess::ExitStatus)), this,
|
|
|
|
SLOT(onEnded(InstancePtr, int, QProcess::ExitStatus)));
|
|
|
|
connect(mcproc, SIGNAL(launch_failed(InstancePtr)), this,
|
|
|
|
SLOT(onLaunchFailed(InstancePtr)));
|
2014-02-24 08:34:21 +00:00
|
|
|
|
2014-01-06 01:52:51 +00:00
|
|
|
restoreState(
|
|
|
|
QByteArray::fromBase64(MMC->settings()->get("ConsoleWindowState").toByteArray()));
|
|
|
|
restoreGeometry(
|
|
|
|
QByteArray::fromBase64(MMC->settings()->get("ConsoleWindowGeometry").toByteArray()));
|
2013-11-23 00:41:28 +00:00
|
|
|
|
2014-01-06 01:52:51 +00:00
|
|
|
QString iconKey = proc->instance()->iconKey();
|
|
|
|
QString name = proc->instance()->name();
|
|
|
|
auto icon = MMC->icons()->getIcon(iconKey);
|
|
|
|
setWindowIcon(icon);
|
|
|
|
m_trayIcon = new QSystemTrayIcon(icon, this);
|
2014-02-24 08:34:21 +00:00
|
|
|
// TODO add screenshot upload as a menu item in the tray icon
|
2014-01-06 01:52:51 +00:00
|
|
|
QString consoleTitle = tr("Console window for ") + name;
|
|
|
|
m_trayIcon->setToolTip(consoleTitle);
|
|
|
|
setWindowTitle(consoleTitle);
|
|
|
|
|
|
|
|
connect(m_trayIcon, SIGNAL(activated(QSystemTrayIcon::ActivationReason)),
|
|
|
|
SLOT(iconActivated(QSystemTrayIcon::ActivationReason)));
|
|
|
|
m_trayIcon->show();
|
2013-11-23 00:41:28 +00:00
|
|
|
if (mcproc->instance()->settings().get("ShowConsole").toBool())
|
|
|
|
{
|
|
|
|
show();
|
|
|
|
}
|
2013-12-10 22:48:20 +00:00
|
|
|
setMayClose(false);
|
2013-02-22 17:18:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ConsoleWindow::~ConsoleWindow()
|
|
|
|
{
|
2013-03-22 13:40:55 +00:00
|
|
|
delete ui;
|
2013-02-22 17:18:23 +00:00
|
|
|
}
|
|
|
|
|
2014-01-06 01:52:51 +00:00
|
|
|
void ConsoleWindow::iconActivated(QSystemTrayIcon::ActivationReason reason)
|
|
|
|
{
|
|
|
|
switch (reason)
|
|
|
|
{
|
|
|
|
case QSystemTrayIcon::Trigger:
|
|
|
|
{
|
|
|
|
toggleConsole();
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-06 08:32:44 +00:00
|
|
|
void ConsoleWindow::writeColor(QString text, const char *color, const char * background)
|
2013-02-22 17:18:23 +00:00
|
|
|
{
|
2013-03-22 13:40:55 +00:00
|
|
|
// append a paragraph
|
2013-12-16 01:19:07 +00:00
|
|
|
QString newtext;
|
|
|
|
newtext += "<span style=\"";
|
|
|
|
{
|
2014-01-06 01:52:51 +00:00
|
|
|
if (color)
|
2013-12-16 01:19:07 +00:00
|
|
|
newtext += QString("color:") + color + ";";
|
2014-02-06 08:32:44 +00:00
|
|
|
if (background)
|
|
|
|
newtext += QString("background-color:") + background + ";";
|
2013-12-16 01:19:07 +00:00
|
|
|
newtext += "font-family: monospace;";
|
|
|
|
}
|
|
|
|
newtext += "\">";
|
|
|
|
newtext += text.toHtmlEscaped();
|
|
|
|
newtext += "</span>";
|
|
|
|
ui->text->appendHtml(newtext);
|
2013-02-22 17:18:23 +00:00
|
|
|
}
|
|
|
|
|
2013-03-22 13:40:55 +00:00
|
|
|
void ConsoleWindow::write(QString data, MessageLevel::Enum mode)
|
2013-02-22 17:18:23 +00:00
|
|
|
{
|
2013-12-01 01:00:42 +00:00
|
|
|
QScrollBar *bar = ui->text->verticalScrollBar();
|
|
|
|
int max_bar = bar->maximum();
|
|
|
|
int val_bar = bar->value();
|
2014-01-06 01:52:51 +00:00
|
|
|
if(isVisible())
|
2013-12-01 01:00:42 +00:00
|
|
|
{
|
2014-01-06 01:52:51 +00:00
|
|
|
if (m_scroll_active)
|
|
|
|
{
|
|
|
|
m_scroll_active = (max_bar - val_bar) <= 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_scroll_active = val_bar == max_bar;
|
|
|
|
}
|
2013-12-01 01:00:42 +00:00
|
|
|
}
|
2013-03-22 13:40:55 +00:00
|
|
|
if (data.endsWith('\n'))
|
2013-11-12 23:24:49 +00:00
|
|
|
data = data.left(data.length() - 1);
|
2013-03-22 13:40:55 +00:00
|
|
|
QStringList paragraphs = data.split('\n');
|
2013-11-12 23:24:49 +00:00
|
|
|
for (QString ¶graph : paragraphs)
|
2013-10-16 02:46:57 +01:00
|
|
|
{
|
|
|
|
paragraph = paragraph.trimmed();
|
|
|
|
}
|
|
|
|
|
2013-03-22 13:40:55 +00:00
|
|
|
QListIterator<QString> iter(paragraphs);
|
|
|
|
if (mode == MessageLevel::MultiMC)
|
2013-11-12 23:24:49 +00:00
|
|
|
while (iter.hasNext())
|
2014-02-06 08:32:44 +00:00
|
|
|
writeColor(iter.next(), "blue", 0);
|
2013-03-22 13:40:55 +00:00
|
|
|
else if (mode == MessageLevel::Error)
|
2013-11-12 23:24:49 +00:00
|
|
|
while (iter.hasNext())
|
2014-02-06 08:32:44 +00:00
|
|
|
writeColor(iter.next(), "red", 0);
|
2013-09-06 21:40:50 +01:00
|
|
|
else if (mode == MessageLevel::Warning)
|
2013-11-12 23:24:49 +00:00
|
|
|
while (iter.hasNext())
|
2014-02-06 08:32:44 +00:00
|
|
|
writeColor(iter.next(), "orange", 0);
|
2013-09-08 14:02:52 +01:00
|
|
|
else if (mode == MessageLevel::Fatal)
|
2013-11-12 23:24:49 +00:00
|
|
|
while (iter.hasNext())
|
2014-02-06 08:32:44 +00:00
|
|
|
writeColor(iter.next(), "red", "black");
|
2013-09-08 14:02:52 +01:00
|
|
|
else if (mode == MessageLevel::Debug)
|
2013-11-12 23:24:49 +00:00
|
|
|
while (iter.hasNext())
|
2014-02-06 08:32:44 +00:00
|
|
|
writeColor(iter.next(), "green", 0);
|
2014-01-17 21:55:10 +00:00
|
|
|
else if (mode == MessageLevel::PrePost)
|
|
|
|
while (iter.hasNext())
|
2014-02-06 08:32:44 +00:00
|
|
|
writeColor(iter.next(), "grey", 0);
|
2013-03-22 13:40:55 +00:00
|
|
|
// TODO: implement other MessageLevels
|
|
|
|
else
|
2013-11-12 23:24:49 +00:00
|
|
|
while (iter.hasNext())
|
2014-02-06 08:32:44 +00:00
|
|
|
writeColor(iter.next(), 0, 0);
|
2014-01-06 01:52:51 +00:00
|
|
|
if(isVisible())
|
2013-12-01 01:00:42 +00:00
|
|
|
{
|
2014-01-06 01:52:51 +00:00
|
|
|
if (m_scroll_active)
|
|
|
|
{
|
|
|
|
bar->setValue(bar->maximum());
|
|
|
|
}
|
|
|
|
m_last_scroll_value = bar->value();
|
2013-12-01 01:00:42 +00:00
|
|
|
}
|
2013-02-22 17:18:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ConsoleWindow::clear()
|
|
|
|
{
|
2013-03-22 13:40:55 +00:00
|
|
|
ui->text->clear();
|
2013-02-22 17:18:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ConsoleWindow::on_closeButton_clicked()
|
|
|
|
{
|
2013-03-22 13:40:55 +00:00
|
|
|
close();
|
2013-02-22 17:18:23 +00:00
|
|
|
}
|
|
|
|
|
2014-05-18 18:07:01 +01:00
|
|
|
void ConsoleWindow::on_btnScreenshots_clicked()
|
|
|
|
{
|
|
|
|
ScreenshotList *list = new ScreenshotList(proc->instance());
|
|
|
|
Task *task = list->load();
|
|
|
|
ProgressDialog prog(this);
|
|
|
|
prog.exec(task);
|
|
|
|
if (!task->successful())
|
|
|
|
{
|
|
|
|
CustomMessageBox::selectable(this, tr("Failed to load screenshots!"),
|
|
|
|
task->failReason(), QMessageBox::Warning)->exec();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ScreenshotDialog dialog(list, this);
|
|
|
|
if (dialog.exec() == ScreenshotDialog::Accepted)
|
|
|
|
{
|
|
|
|
CustomMessageBox::selectable(this, tr("Done uploading!"), dialog.message(),
|
|
|
|
QMessageBox::Information)->exec();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-22 17:18:23 +00:00
|
|
|
void ConsoleWindow::setMayClose(bool mayclose)
|
|
|
|
{
|
2014-01-09 00:22:34 +00:00
|
|
|
if(mayclose)
|
|
|
|
ui->closeButton->setText(tr("Close"));
|
|
|
|
else
|
|
|
|
ui->closeButton->setText(tr("Hide"));
|
2013-03-22 13:40:55 +00:00
|
|
|
m_mayclose = mayclose;
|
2014-01-06 01:52:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ConsoleWindow::toggleConsole()
|
|
|
|
{
|
|
|
|
QScrollBar *bar = ui->text->verticalScrollBar();
|
|
|
|
if (isVisible())
|
|
|
|
{
|
2014-03-24 00:54:18 +00:00
|
|
|
if(!isActiveWindow())
|
|
|
|
{
|
|
|
|
activateWindow();
|
|
|
|
return;
|
|
|
|
}
|
2014-01-06 01:52:51 +00:00
|
|
|
int max_bar = bar->maximum();
|
|
|
|
int val_bar = m_last_scroll_value = bar->value();
|
|
|
|
m_scroll_active = (max_bar - val_bar) <= 1;
|
|
|
|
hide();
|
|
|
|
}
|
2013-03-22 13:40:55 +00:00
|
|
|
else
|
2014-01-06 01:52:51 +00:00
|
|
|
{
|
|
|
|
show();
|
2014-03-24 00:54:18 +00:00
|
|
|
isTopLevel();
|
2014-01-06 01:52:51 +00:00
|
|
|
if (m_scroll_active)
|
|
|
|
{
|
|
|
|
bar->setValue(bar->maximum());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
bar->setValue(m_last_scroll_value);
|
|
|
|
}
|
|
|
|
}
|
2013-02-22 17:18:23 +00:00
|
|
|
}
|
|
|
|
|
2013-11-12 23:24:49 +00:00
|
|
|
void ConsoleWindow::closeEvent(QCloseEvent *event)
|
2013-02-22 17:18:23 +00:00
|
|
|
{
|
2013-11-12 23:24:49 +00:00
|
|
|
if (!m_mayclose)
|
2014-01-06 01:52:51 +00:00
|
|
|
{
|
|
|
|
toggleConsole();
|
|
|
|
}
|
2013-03-22 13:40:55 +00:00
|
|
|
else
|
2013-11-23 00:41:28 +00:00
|
|
|
{
|
|
|
|
MMC->settings()->set("ConsoleWindowState", saveState().toBase64());
|
|
|
|
MMC->settings()->set("ConsoleWindowGeometry", saveGeometry().toBase64());
|
|
|
|
|
|
|
|
emit isClosing();
|
2014-01-06 01:52:51 +00:00
|
|
|
m_trayIcon->hide();
|
2013-11-23 00:41:28 +00:00
|
|
|
QMainWindow::closeEvent(event);
|
|
|
|
}
|
2013-02-22 17:18:23 +00:00
|
|
|
}
|
2013-09-06 21:40:50 +01:00
|
|
|
|
|
|
|
void ConsoleWindow::on_btnKillMinecraft_clicked()
|
|
|
|
{
|
|
|
|
ui->btnKillMinecraft->setEnabled(false);
|
2013-11-12 23:24:49 +00:00
|
|
|
auto response = CustomMessageBox::selectable(
|
|
|
|
this, tr("Kill Minecraft?"),
|
|
|
|
tr("This can cause the instance to get corrupted and should only be used if Minecraft "
|
|
|
|
"is frozen for some reason"),
|
|
|
|
QMessageBox::Question, QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes)->exec();
|
2013-10-29 12:40:09 +00:00
|
|
|
if (response == QMessageBox::Yes)
|
2013-09-06 21:40:50 +01:00
|
|
|
proc->killMinecraft();
|
|
|
|
else
|
|
|
|
ui->btnKillMinecraft->setEnabled(true);
|
2013-09-06 22:52:17 +01:00
|
|
|
}
|
|
|
|
|
2014-05-18 18:07:01 +01:00
|
|
|
void ConsoleWindow::onEnded(InstancePtr instance, int code, QProcess::ExitStatus status)
|
2013-09-06 22:52:17 +01:00
|
|
|
{
|
2014-01-06 01:52:51 +00:00
|
|
|
bool peacefulExit = code == 0 && status != QProcess::CrashExit;
|
2013-09-06 22:52:17 +01:00
|
|
|
ui->btnKillMinecraft->setEnabled(false);
|
2013-10-22 18:25:10 +01:00
|
|
|
|
2013-12-11 12:26:23 +00:00
|
|
|
setMayClose(true);
|
|
|
|
|
2013-11-12 23:24:49 +00:00
|
|
|
if (instance->settings().get("AutoCloseConsole").toBool())
|
2013-10-22 18:25:10 +01:00
|
|
|
{
|
2014-01-06 01:52:51 +00:00
|
|
|
if (peacefulExit)
|
2013-11-12 08:23:39 +00:00
|
|
|
{
|
|
|
|
this->close();
|
2013-11-23 00:41:28 +00:00
|
|
|
return;
|
2013-11-12 08:23:39 +00:00
|
|
|
}
|
2013-10-22 18:25:10 +01:00
|
|
|
}
|
2014-01-06 01:52:51 +00:00
|
|
|
/*
|
|
|
|
if(!peacefulExit)
|
|
|
|
{
|
|
|
|
m_trayIcon->showMessage(tr("Oh no!"), tr("Minecraft crashed!"), QSystemTrayIcon::Critical);
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
if (!isVisible())
|
2013-11-23 00:41:28 +00:00
|
|
|
show();
|
2014-04-17 13:13:16 +01:00
|
|
|
|
|
|
|
// Raise Window
|
|
|
|
if (MMC->settings()->get("RaiseConsole").toBool())
|
|
|
|
{
|
|
|
|
raise();
|
|
|
|
activateWindow();
|
|
|
|
}
|
2013-11-23 00:41:28 +00:00
|
|
|
}
|
|
|
|
|
2014-05-18 18:07:01 +01:00
|
|
|
void ConsoleWindow::onLaunchFailed(InstancePtr instance)
|
2013-11-23 00:41:28 +00:00
|
|
|
{
|
|
|
|
ui->btnKillMinecraft->setEnabled(false);
|
2013-12-11 12:26:23 +00:00
|
|
|
|
|
|
|
setMayClose(true);
|
|
|
|
|
2014-01-06 01:52:51 +00:00
|
|
|
if (!isVisible())
|
2013-11-23 00:41:28 +00:00
|
|
|
show();
|
2013-09-06 22:52:17 +01:00
|
|
|
}
|
2013-12-06 21:24:55 +00:00
|
|
|
|
|
|
|
void ConsoleWindow::on_btnPaste_clicked()
|
|
|
|
{
|
|
|
|
auto text = ui->text->toPlainText();
|
|
|
|
ProgressDialog dialog(this);
|
2014-01-06 01:52:51 +00:00
|
|
|
PasteUpload *paste = new PasteUpload(this, text);
|
2013-12-06 21:24:55 +00:00
|
|
|
dialog.exec(paste);
|
2014-01-06 01:52:51 +00:00
|
|
|
if (!paste->successful())
|
2013-12-06 21:24:55 +00:00
|
|
|
{
|
2014-01-06 01:52:51 +00:00
|
|
|
CustomMessageBox::selectable(this, "Upload failed", paste->failReason(),
|
|
|
|
QMessageBox::Critical)->exec();
|
2013-12-06 21:24:55 +00:00
|
|
|
}
|
|
|
|
}
|