// // 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #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(); 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(this->parent()->parent()->parent())->terminateCurrentTab(); } } const QString& scriptURLString = QUrl(_currentScript).toString(); _scriptEngine = DependencyManager::get()->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(); 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(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; } }