mirror of
https://github.com/HifiExperiments/overte.git
synced 2025-06-23 14:29:07 +02:00
255 lines
10 KiB
C++
255 lines
10 KiB
C++
//
|
|
// ScriptEditorWidget.cpp
|
|
// interface/src/ui
|
|
//
|
|
// Created by Thijs Wenker on 4/14/14.
|
|
// Copyright 2014 High Fidelity, Inc.
|
|
//
|
|
// Distributed under the Apache License, Version 2.0.
|
|
// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html
|
|
//
|
|
|
|
#include "ui_scriptEditorWidget.h"
|
|
#include "ScriptEditorWidget.h"
|
|
#include "ScriptEditorWindow.h"
|
|
|
|
#include <QGridLayout>
|
|
#include <QFileDialog>
|
|
#include <QFrame>
|
|
#include <QLayoutItem>
|
|
#include <QMainWindow>
|
|
#include <QMessageBox>
|
|
#include <QPalette>
|
|
#include <QScrollBar>
|
|
#include <QSizePolicy>
|
|
#include <QTimer>
|
|
#include <QWidget>
|
|
|
|
#include <ScriptEngines.h>
|
|
#include <NetworkAccessManager.h>
|
|
#include <OffscreenUi.h>
|
|
|
|
#include "Application.h"
|
|
#include "ScriptHighlighting.h"
|
|
|
|
ScriptEditorWidget::ScriptEditorWidget() :
|
|
_scriptEditorWidgetUI(new Ui::ScriptEditorWidget),
|
|
_scriptEngine(NULL),
|
|
_isRestarting(false),
|
|
_isReloading(false)
|
|
{
|
|
setAttribute(Qt::WA_DeleteOnClose);
|
|
|
|
_scriptEditorWidgetUI->setupUi(this);
|
|
|
|
connect(_scriptEditorWidgetUI->scriptEdit->document(), &QTextDocument::modificationChanged, this,
|
|
&ScriptEditorWidget::scriptModified);
|
|
connect(_scriptEditorWidgetUI->scriptEdit->document(), &QTextDocument::contentsChanged, this,
|
|
&ScriptEditorWidget::onScriptModified);
|
|
|
|
// remove the title bar (see the Qt docs on setTitleBarWidget)
|
|
setTitleBarWidget(new QWidget());
|
|
QFontMetrics fm(_scriptEditorWidgetUI->scriptEdit->font());
|
|
_scriptEditorWidgetUI->scriptEdit->setTabStopWidth(fm.width('0') * 4);
|
|
// We create a new ScriptHighligting QObject and provide it with a parent so this is NOT a memory leak.
|
|
new ScriptHighlighting(_scriptEditorWidgetUI->scriptEdit->document());
|
|
QTimer::singleShot(0, _scriptEditorWidgetUI->scriptEdit, SLOT(setFocus()));
|
|
|
|
_console = new JSConsole(this);
|
|
_console->setFixedHeight(CONSOLE_HEIGHT);
|
|
_scriptEditorWidgetUI->verticalLayout->addWidget(_console);
|
|
connect(_scriptEditorWidgetUI->clearButton, &QPushButton::clicked, _console, &JSConsole::clear);
|
|
}
|
|
|
|
ScriptEditorWidget::~ScriptEditorWidget() {
|
|
delete _scriptEditorWidgetUI;
|
|
delete _console;
|
|
}
|
|
|
|
void ScriptEditorWidget::onScriptModified() {
|
|
if(_scriptEditorWidgetUI->onTheFlyCheckBox->isChecked() && isModified() && isRunning() && !_isReloading) {
|
|
_isRestarting = true;
|
|
setRunning(false);
|
|
// Script is restarted once current script instance finishes.
|
|
}
|
|
}
|
|
|
|
void ScriptEditorWidget::onScriptFinished(const QString& scriptPath) {
|
|
_scriptEngine = NULL;
|
|
_console->setScriptEngine(NULL);
|
|
if (_isRestarting) {
|
|
_isRestarting = false;
|
|
setRunning(true);
|
|
}
|
|
}
|
|
|
|
bool ScriptEditorWidget::isModified() {
|
|
return _scriptEditorWidgetUI->scriptEdit->document()->isModified();
|
|
}
|
|
|
|
bool ScriptEditorWidget::isRunning() {
|
|
return (_scriptEngine != NULL) ? _scriptEngine->isRunning() : false;
|
|
}
|
|
|
|
bool ScriptEditorWidget::setRunning(bool run) {
|
|
if (run && isModified() && !save()) {
|
|
return false;
|
|
}
|
|
|
|
if (_scriptEngine != NULL) {
|
|
disconnect(_scriptEngine, &ScriptEngine::runningStateChanged, this, &ScriptEditorWidget::runningStateChanged);
|
|
disconnect(_scriptEngine, &ScriptEngine::update, this, &ScriptEditorWidget::onScriptModified);
|
|
disconnect(_scriptEngine, &ScriptEngine::finished, this, &ScriptEditorWidget::onScriptFinished);
|
|
}
|
|
|
|
auto scriptEngines = DependencyManager::get<ScriptEngines>();
|
|
if (run) {
|
|
const QString& scriptURLString = QUrl(_currentScript).toString();
|
|
// Reload script so that an out of date copy is not retrieved from the cache
|
|
_scriptEngine = scriptEngines->loadScript(scriptURLString, true, true, false, true);
|
|
connect(_scriptEngine, &ScriptEngine::runningStateChanged, this, &ScriptEditorWidget::runningStateChanged);
|
|
connect(_scriptEngine, &ScriptEngine::update, this, &ScriptEditorWidget::onScriptModified);
|
|
connect(_scriptEngine, &ScriptEngine::finished, this, &ScriptEditorWidget::onScriptFinished);
|
|
} else {
|
|
connect(_scriptEngine, &ScriptEngine::finished, this, &ScriptEditorWidget::onScriptFinished);
|
|
const QString& scriptURLString = QUrl(_currentScript).toString();
|
|
scriptEngines->stopScript(scriptURLString);
|
|
_scriptEngine = NULL;
|
|
}
|
|
_console->setScriptEngine(_scriptEngine);
|
|
return true;
|
|
}
|
|
|
|
bool ScriptEditorWidget::saveFile(const QString &scriptPath) {
|
|
QFile file(scriptPath);
|
|
if (!file.open(QFile::WriteOnly | QFile::Text)) {
|
|
OffscreenUi::warning(this, tr("Interface"), tr("Cannot write script %1:\n%2.").arg(scriptPath)
|
|
.arg(file.errorString()));
|
|
return false;
|
|
}
|
|
|
|
QTextStream out(&file);
|
|
out << _scriptEditorWidgetUI->scriptEdit->toPlainText();
|
|
file.close();
|
|
|
|
setScriptFile(scriptPath);
|
|
return true;
|
|
}
|
|
|
|
void ScriptEditorWidget::loadFile(const QString& scriptPath) {
|
|
QUrl url(scriptPath);
|
|
|
|
// if the scheme length is one or lower, maybe they typed in a file, let's try
|
|
const int WINDOWS_DRIVE_LETTER_SIZE = 1;
|
|
if (url.scheme().size() <= WINDOWS_DRIVE_LETTER_SIZE) {
|
|
QFile file(scriptPath);
|
|
if (!file.open(QFile::ReadOnly | QFile::Text)) {
|
|
OffscreenUi::warning(this, tr("Interface"), tr("Cannot read script %1:\n%2.").arg(scriptPath)
|
|
.arg(file.errorString()));
|
|
return;
|
|
}
|
|
QTextStream in(&file);
|
|
_scriptEditorWidgetUI->scriptEdit->setPlainText(in.readAll());
|
|
file.close();
|
|
setScriptFile(scriptPath);
|
|
|
|
if (_scriptEngine != NULL) {
|
|
disconnect(_scriptEngine, &ScriptEngine::runningStateChanged, this, &ScriptEditorWidget::runningStateChanged);
|
|
disconnect(_scriptEngine, &ScriptEngine::update, this, &ScriptEditorWidget::onScriptModified);
|
|
disconnect(_scriptEngine, &ScriptEngine::finished, this, &ScriptEditorWidget::onScriptFinished);
|
|
}
|
|
} else {
|
|
QNetworkAccessManager& networkAccessManager = NetworkAccessManager::getInstance();
|
|
QNetworkRequest networkRequest = QNetworkRequest(url);
|
|
networkRequest.setHeader(QNetworkRequest::UserAgentHeader, HIGH_FIDELITY_USER_AGENT);
|
|
QNetworkReply* reply = networkAccessManager.get(networkRequest);
|
|
qDebug() << "Downloading included script at" << scriptPath;
|
|
QEventLoop loop;
|
|
QObject::connect(reply, &QNetworkReply::finished, &loop, &QEventLoop::quit);
|
|
loop.exec();
|
|
_scriptEditorWidgetUI->scriptEdit->setPlainText(reply->readAll());
|
|
delete reply;
|
|
|
|
if (!saveAs()) {
|
|
static_cast<ScriptEditorWindow*>(this->parent()->parent()->parent())->terminateCurrentTab();
|
|
}
|
|
}
|
|
const QString& scriptURLString = QUrl(_currentScript).toString();
|
|
_scriptEngine = DependencyManager::get<ScriptEngines>()->getScriptEngine(scriptURLString);
|
|
if (_scriptEngine != NULL) {
|
|
connect(_scriptEngine, &ScriptEngine::runningStateChanged, this, &ScriptEditorWidget::runningStateChanged);
|
|
connect(_scriptEngine, &ScriptEngine::update, this, &ScriptEditorWidget::onScriptModified);
|
|
connect(_scriptEngine, &ScriptEngine::finished, this, &ScriptEditorWidget::onScriptFinished);
|
|
}
|
|
_console->setScriptEngine(_scriptEngine);
|
|
}
|
|
|
|
bool ScriptEditorWidget::save() {
|
|
return _currentScript.isEmpty() ? saveAs() : saveFile(_currentScript);
|
|
}
|
|
|
|
bool ScriptEditorWidget::saveAs() {
|
|
auto scriptEngines = DependencyManager::get<ScriptEngines>();
|
|
QString fileName = QFileDialog::getSaveFileName(this, tr("Save script"),
|
|
qApp->getPreviousScriptLocation(),
|
|
tr("JavaScript Files (*.js)"));
|
|
if (!fileName.isEmpty()) {
|
|
qApp->setPreviousScriptLocation(fileName);
|
|
return saveFile(fileName);
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void ScriptEditorWidget::setScriptFile(const QString& scriptPath) {
|
|
_currentScript = scriptPath;
|
|
_currentScriptModified = QFileInfo(_currentScript).lastModified();
|
|
_scriptEditorWidgetUI->scriptEdit->document()->setModified(false);
|
|
setWindowModified(false);
|
|
|
|
emit scriptnameChanged();
|
|
}
|
|
|
|
bool ScriptEditorWidget::questionSave() {
|
|
if (_scriptEditorWidgetUI->scriptEdit->document()->isModified()) {
|
|
QMessageBox::StandardButton button = OffscreenUi::warning(this, tr("Interface"),
|
|
tr("The script has been modified.\nDo you want to save your changes?"), QMessageBox::Save | QMessageBox::Discard |
|
|
QMessageBox::Cancel, QMessageBox::Save);
|
|
return button == QMessageBox::Save ? save() : (button == QMessageBox::Discard);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void ScriptEditorWidget::onWindowActivated() {
|
|
if (!_isReloading) {
|
|
_isReloading = true;
|
|
|
|
QDateTime fileStamp = QFileInfo(_currentScript).lastModified();
|
|
if (fileStamp > _currentScriptModified) {
|
|
bool doReload = false;
|
|
auto window = static_cast<ScriptEditorWindow*>(this->parent()->parent()->parent());
|
|
window->inModalDialog = true;
|
|
if (window->autoReloadScripts()
|
|
|| OffscreenUi::question(this, tr("Reload Script"),
|
|
tr("The following file has been modified outside of the Interface editor:") + "\n" + _currentScript + "\n"
|
|
+ (isModified()
|
|
? tr("Do you want to reload it and lose the changes you've made in the Interface editor?")
|
|
: tr("Do you want to reload it?")),
|
|
QMessageBox::Yes | QMessageBox::No) == QMessageBox::Yes) {
|
|
doReload = true;
|
|
}
|
|
window->inModalDialog = false;
|
|
if (doReload) {
|
|
loadFile(_currentScript);
|
|
if (_scriptEditorWidgetUI->onTheFlyCheckBox->isChecked() && isRunning()) {
|
|
_isRestarting = true;
|
|
setRunning(false);
|
|
// Script is restarted once current script instance finishes.
|
|
}
|
|
} else {
|
|
_currentScriptModified = fileStamp; // Asked and answered. Don't ask again until the external file is changed again.
|
|
}
|
|
}
|
|
_isReloading = false;
|
|
}
|
|
}
|