2013-04-10 18:14:10 +00:00
|
|
|
#include "scriptsubview.hpp"
|
|
|
|
|
|
|
|
#include <stdexcept>
|
|
|
|
|
2015-04-29 10:24:17 +00:00
|
|
|
#include <QStatusBar>
|
|
|
|
#include <QStackedLayout>
|
2015-07-16 10:36:20 +00:00
|
|
|
#include <QSplitter>
|
2015-07-18 11:09:17 +00:00
|
|
|
#include <QTimer>
|
2015-04-29 10:24:17 +00:00
|
|
|
|
2013-04-10 18:14:10 +00:00
|
|
|
#include "../../model/doc/document.hpp"
|
|
|
|
#include "../../model/world/universalid.hpp"
|
|
|
|
#include "../../model/world/data.hpp"
|
|
|
|
#include "../../model/world/columnbase.hpp"
|
|
|
|
#include "../../model/world/commands.hpp"
|
|
|
|
#include "../../model/world/idtable.hpp"
|
2015-04-29 20:32:03 +00:00
|
|
|
#include "../../model/settings/usersettings.hpp"
|
2013-04-10 18:14:10 +00:00
|
|
|
|
2014-02-15 18:52:40 +00:00
|
|
|
#include "scriptedit.hpp"
|
2015-07-11 14:09:13 +00:00
|
|
|
#include "recordbuttonbar.hpp"
|
2015-07-14 08:05:45 +00:00
|
|
|
#include "tablebottombox.hpp"
|
|
|
|
#include "genericcreator.hpp"
|
2015-07-16 10:36:20 +00:00
|
|
|
#include "scripterrortable.hpp"
|
2013-04-10 20:49:22 +00:00
|
|
|
|
2015-07-13 10:52:18 +00:00
|
|
|
void CSVWorld::ScriptSubView::addButtonBar()
|
|
|
|
{
|
|
|
|
if (mButtons)
|
|
|
|
return;
|
|
|
|
|
2015-07-14 11:52:48 +00:00
|
|
|
mButtons = new RecordButtonBar (getUniversalId(), *mModel, mBottom, &mCommandDispatcher, this);
|
2015-07-13 10:52:18 +00:00
|
|
|
|
|
|
|
mLayout.insertWidget (1, mButtons);
|
|
|
|
|
|
|
|
connect (mButtons, SIGNAL (switchToRow (int)), this, SLOT (switchToRow (int)));
|
|
|
|
|
|
|
|
connect (this, SIGNAL (universalIdChanged (const CSMWorld::UniversalId&)),
|
|
|
|
mButtons, SLOT (universalIdChanged (const CSMWorld::UniversalId&)));
|
|
|
|
}
|
|
|
|
|
2015-07-18 11:09:17 +00:00
|
|
|
void CSVWorld::ScriptSubView::recompile()
|
|
|
|
{
|
2015-07-18 15:00:00 +00:00
|
|
|
if (!mCompileDelay->isActive() && !isDeleted())
|
2015-07-18 13:32:38 +00:00
|
|
|
mCompileDelay->start (
|
|
|
|
CSMSettings::UserSettings::instance().setting ("script-editor/compile-delay").toInt());
|
2015-07-18 11:09:17 +00:00
|
|
|
}
|
|
|
|
|
2015-07-18 15:00:00 +00:00
|
|
|
bool CSVWorld::ScriptSubView::isDeleted() const
|
|
|
|
{
|
|
|
|
return mModel->data (mModel->getModelIndex (getUniversalId().getId(), mStateColumn)).toInt()
|
|
|
|
==CSMWorld::RecordBase::State_Deleted;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::ScriptSubView::updateDeletedState()
|
|
|
|
{
|
|
|
|
if (isDeleted())
|
|
|
|
{
|
|
|
|
mErrors->clear();
|
2015-11-21 11:45:11 +00:00
|
|
|
adjustSplitter();
|
2015-07-18 15:00:00 +00:00
|
|
|
mEditor->setEnabled (false);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mEditor->setEnabled (true);
|
|
|
|
recompile();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-21 11:45:11 +00:00
|
|
|
void CSVWorld::ScriptSubView::adjustSplitter()
|
|
|
|
{
|
|
|
|
QList<int> sizes;
|
|
|
|
|
|
|
|
if (mErrors->rowCount())
|
|
|
|
{
|
2015-11-21 11:52:32 +00:00
|
|
|
if (mErrors->height())
|
|
|
|
return; // keep old height if the error panel was already open
|
|
|
|
|
2015-11-21 11:45:11 +00:00
|
|
|
sizes << 1 << 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
sizes << 1 << 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
mMain->setSizes (sizes);
|
|
|
|
}
|
|
|
|
|
2013-04-10 18:14:10 +00:00
|
|
|
CSVWorld::ScriptSubView::ScriptSubView (const CSMWorld::UniversalId& id, CSMDoc::Document& document)
|
2015-07-14 08:05:45 +00:00
|
|
|
: SubView (id), mDocument (document), mColumn (-1), mBottom(0), mButtons (0),
|
2015-07-11 14:09:13 +00:00
|
|
|
mCommandDispatcher (document, CSMWorld::UniversalId::getParentType (id.getType()))
|
2013-04-10 18:14:10 +00:00
|
|
|
{
|
2015-07-11 14:09:13 +00:00
|
|
|
std::vector<std::string> selection (1, id.getId());
|
|
|
|
mCommandDispatcher.setSelection (selection);
|
2015-04-29 10:24:17 +00:00
|
|
|
|
2015-07-16 10:36:20 +00:00
|
|
|
mMain = new QSplitter (this);
|
|
|
|
mMain->setOrientation (Qt::Vertical);
|
|
|
|
mLayout.addWidget (mMain, 2);
|
|
|
|
|
|
|
|
mEditor = new ScriptEdit (mDocument, ScriptHighlighter::Mode_General, this);
|
|
|
|
mMain->addWidget (mEditor);
|
|
|
|
mMain->setCollapsible (0, false);
|
|
|
|
|
|
|
|
mErrors = new ScriptErrorTable (document, this);
|
|
|
|
mMain->addWidget (mErrors);
|
2015-04-29 10:24:17 +00:00
|
|
|
|
2015-07-11 14:09:13 +00:00
|
|
|
QWidget *widget = new QWidget (this);;
|
2015-07-13 10:52:18 +00:00
|
|
|
widget->setLayout (&mLayout);
|
2015-04-29 10:24:17 +00:00
|
|
|
setWidget (widget);
|
2013-04-10 18:14:10 +00:00
|
|
|
|
|
|
|
mModel = &dynamic_cast<CSMWorld::IdTable&> (
|
|
|
|
*document.getData().getTableModel (CSMWorld::UniversalId::Type_Scripts));
|
|
|
|
|
2015-07-18 15:00:00 +00:00
|
|
|
mColumn = mModel->findColumnIndex (CSMWorld::Columns::ColumnId_ScriptText);
|
2015-09-22 13:36:00 +00:00
|
|
|
mIdColumn = mModel->findColumnIndex (CSMWorld::Columns::ColumnId_Id);
|
2015-07-18 15:00:00 +00:00
|
|
|
mStateColumn = mModel->findColumnIndex (CSMWorld::Columns::ColumnId_Modification);
|
2013-04-10 18:14:10 +00:00
|
|
|
|
2015-07-16 10:36:20 +00:00
|
|
|
QString source = mModel->data (mModel->getModelIndex (id.getId(), mColumn)).toString();
|
|
|
|
|
|
|
|
mEditor->setPlainText (source);
|
2015-07-14 11:52:48 +00:00
|
|
|
// bottom box and buttons
|
|
|
|
mBottom = new TableBottomBox (CreatorFactory<GenericCreator>(), document, id, this);
|
2013-04-10 18:14:10 +00:00
|
|
|
|
2015-07-13 10:52:18 +00:00
|
|
|
if (CSMSettings::UserSettings::instance().setting ("script-editor/toolbar", QString("true")) == "true")
|
|
|
|
addButtonBar();
|
2015-07-11 14:09:13 +00:00
|
|
|
|
2015-07-14 08:05:45 +00:00
|
|
|
connect (mBottom, SIGNAL (requestFocus (const std::string&)),
|
2015-07-14 11:52:48 +00:00
|
|
|
this, SLOT (switchToId (const std::string&)));
|
2015-07-11 14:09:13 +00:00
|
|
|
|
2015-07-14 08:05:45 +00:00
|
|
|
mLayout.addWidget (mBottom);
|
2015-07-11 14:09:13 +00:00
|
|
|
|
|
|
|
// signals
|
2013-04-10 18:14:10 +00:00
|
|
|
connect (mEditor, SIGNAL (textChanged()), this, SLOT (textChanged()));
|
|
|
|
|
|
|
|
connect (mModel, SIGNAL (dataChanged (const QModelIndex&, const QModelIndex&)),
|
|
|
|
this, SLOT (dataChanged (const QModelIndex&, const QModelIndex&)));
|
|
|
|
|
2013-04-10 19:31:14 +00:00
|
|
|
connect (mModel, SIGNAL (rowsAboutToBeRemoved (const QModelIndex&, int, int)),
|
|
|
|
this, SLOT (rowsAboutToBeRemoved (const QModelIndex&, int, int)));
|
2015-04-29 10:24:17 +00:00
|
|
|
|
|
|
|
updateStatusBar();
|
|
|
|
connect(mEditor, SIGNAL(cursorPositionChanged()), this, SLOT(updateStatusBar()));
|
2015-07-16 10:36:20 +00:00
|
|
|
|
|
|
|
mErrors->update (source.toUtf8().constData());
|
2015-07-17 11:42:25 +00:00
|
|
|
|
2015-07-17 11:53:22 +00:00
|
|
|
connect (mErrors, SIGNAL (highlightError (int, int)),
|
|
|
|
this, SLOT (highlightError (int, int)));
|
2015-07-18 11:09:17 +00:00
|
|
|
|
|
|
|
mCompileDelay = new QTimer (this);
|
|
|
|
mCompileDelay->setSingleShot (true);
|
|
|
|
connect (mCompileDelay, SIGNAL (timeout()), this, SLOT (updateRequest()));
|
|
|
|
|
2015-07-18 15:00:00 +00:00
|
|
|
updateDeletedState();
|
2015-04-29 10:24:17 +00:00
|
|
|
}
|
|
|
|
|
2015-04-29 20:32:03 +00:00
|
|
|
void CSVWorld::ScriptSubView::updateUserSetting (const QString& name, const QStringList& value)
|
2015-04-29 10:24:17 +00:00
|
|
|
{
|
2015-04-30 20:08:04 +00:00
|
|
|
if (name == "script-editor/show-linenum")
|
|
|
|
{
|
2015-07-13 10:52:18 +00:00
|
|
|
std::string showLinenum = value.at(0).toUtf8().constData();
|
2015-04-30 20:08:04 +00:00
|
|
|
mEditor->showLineNum(showLinenum == "true");
|
|
|
|
mBottom->setVisible(showLinenum == "true");
|
|
|
|
}
|
|
|
|
else if (name == "script-editor/mono-font")
|
|
|
|
{
|
2015-07-13 10:52:18 +00:00
|
|
|
mEditor->setMonoFont (value.at(0)==QString ("true"));
|
|
|
|
}
|
|
|
|
else if (name=="script-editor/toolbar")
|
|
|
|
{
|
|
|
|
if (value.at(0)==QString ("true"))
|
|
|
|
{
|
|
|
|
addButtonBar();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (mButtons)
|
|
|
|
{
|
|
|
|
mLayout.removeWidget (mButtons);
|
|
|
|
delete mButtons;
|
|
|
|
mButtons = 0;
|
|
|
|
}
|
|
|
|
}
|
2015-04-30 20:08:04 +00:00
|
|
|
}
|
2015-07-18 13:32:38 +00:00
|
|
|
else if (name=="script-editor/compile-delay")
|
|
|
|
{
|
|
|
|
mCompileDelay->setInterval (value.at (0).toInt());
|
|
|
|
}
|
2015-07-11 14:09:13 +00:00
|
|
|
|
2015-07-13 10:52:18 +00:00
|
|
|
if (mButtons)
|
|
|
|
mButtons->updateUserSetting (name, value);
|
2015-07-16 10:36:20 +00:00
|
|
|
|
|
|
|
mErrors->updateUserSetting (name, value);
|
2015-07-18 11:09:17 +00:00
|
|
|
|
|
|
|
if (name=="script-editor/warnings")
|
|
|
|
recompile();
|
2015-04-29 10:24:17 +00:00
|
|
|
}
|
|
|
|
|
2015-07-14 08:05:45 +00:00
|
|
|
void CSVWorld::ScriptSubView::setStatusBar (bool show)
|
2015-04-29 10:24:17 +00:00
|
|
|
{
|
2015-07-14 08:05:45 +00:00
|
|
|
mBottom->setStatusBar (show);
|
|
|
|
}
|
2015-04-29 10:24:17 +00:00
|
|
|
|
2015-07-14 08:05:45 +00:00
|
|
|
void CSVWorld::ScriptSubView::updateStatusBar ()
|
|
|
|
{
|
|
|
|
mBottom->positionChanged (mEditor->textCursor().blockNumber() + 1,
|
|
|
|
mEditor->textCursor().columnNumber() + 1);
|
2013-04-10 18:14:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::ScriptSubView::setEditLock (bool locked)
|
|
|
|
{
|
|
|
|
mEditor->setReadOnly (locked);
|
2015-07-13 10:52:18 +00:00
|
|
|
|
|
|
|
if (mButtons)
|
|
|
|
mButtons->setEditLock (locked);
|
|
|
|
|
2015-07-11 14:09:13 +00:00
|
|
|
mCommandDispatcher.setEditLock (locked);
|
2013-04-10 18:14:10 +00:00
|
|
|
}
|
|
|
|
|
2014-12-08 11:29:23 +00:00
|
|
|
void CSVWorld::ScriptSubView::useHint (const std::string& hint)
|
|
|
|
{
|
|
|
|
if (hint.empty())
|
|
|
|
return;
|
|
|
|
|
2015-10-09 11:52:12 +00:00
|
|
|
unsigned line = 0, column = 0;
|
|
|
|
char c;
|
|
|
|
std::istringstream stream (hint.c_str()+1);
|
2015-10-07 02:39:19 +00:00
|
|
|
switch(hint[0]){
|
|
|
|
case 'R':
|
|
|
|
case 'r':
|
2014-12-08 11:29:23 +00:00
|
|
|
{
|
2015-10-07 02:39:19 +00:00
|
|
|
QModelIndex index = mModel->getModelIndex (getUniversalId().getId(), mColumn);
|
|
|
|
QString source = mModel->data (index).toString();
|
2015-10-09 11:52:12 +00:00
|
|
|
unsigned pos, dummy;
|
|
|
|
if (!(stream >> c >> dummy >> pos) )
|
|
|
|
return;
|
2015-10-07 02:39:19 +00:00
|
|
|
|
2015-10-09 11:52:12 +00:00
|
|
|
for (unsigned i = 0; i <= pos; ++i){
|
|
|
|
if (source[i] == '\n'){
|
2015-10-07 02:39:19 +00:00
|
|
|
++line;
|
2015-10-09 11:52:12 +00:00
|
|
|
column = i+1;
|
2015-10-07 02:39:19 +00:00
|
|
|
}
|
|
|
|
}
|
2015-10-09 11:52:12 +00:00
|
|
|
column = pos - column;
|
2015-10-07 02:39:19 +00:00
|
|
|
break;
|
2014-12-08 11:29:23 +00:00
|
|
|
}
|
2015-10-07 02:39:19 +00:00
|
|
|
case 'l':
|
2015-10-09 11:52:12 +00:00
|
|
|
if (!(stream >> c >> line >> column))
|
|
|
|
return;
|
2014-12-08 11:29:23 +00:00
|
|
|
}
|
2015-10-07 02:39:19 +00:00
|
|
|
|
|
|
|
QTextCursor cursor = mEditor->textCursor();
|
|
|
|
|
|
|
|
cursor.movePosition (QTextCursor::Start);
|
|
|
|
if (cursor.movePosition (QTextCursor::Down, QTextCursor::MoveAnchor, line))
|
|
|
|
cursor.movePosition (QTextCursor::Right, QTextCursor::MoveAnchor, column);
|
|
|
|
|
|
|
|
mEditor->setFocus();
|
|
|
|
mEditor->setTextCursor (cursor);
|
2014-12-08 11:29:23 +00:00
|
|
|
}
|
|
|
|
|
2013-04-10 18:14:10 +00:00
|
|
|
void CSVWorld::ScriptSubView::textChanged()
|
|
|
|
{
|
2014-08-21 12:50:13 +00:00
|
|
|
if (mEditor->isChangeLocked())
|
2013-09-19 12:26:09 +00:00
|
|
|
return;
|
|
|
|
|
2014-08-21 12:50:13 +00:00
|
|
|
ScriptEdit::ChangeLock lock (*mEditor);
|
2013-04-10 18:14:10 +00:00
|
|
|
|
2015-07-16 10:36:20 +00:00
|
|
|
QString source = mEditor->toPlainText();
|
|
|
|
|
2013-04-10 18:14:10 +00:00
|
|
|
mDocument.getUndoStack().push (new CSMWorld::ModifyCommand (*mModel,
|
2015-07-16 10:36:20 +00:00
|
|
|
mModel->getModelIndex (getUniversalId().getId(), mColumn), source));
|
|
|
|
|
2015-07-18 11:09:17 +00:00
|
|
|
recompile();
|
2013-04-10 18:14:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::ScriptSubView::dataChanged (const QModelIndex& topLeft, const QModelIndex& bottomRight)
|
|
|
|
{
|
2014-08-21 12:50:13 +00:00
|
|
|
if (mEditor->isChangeLocked())
|
2013-04-10 18:14:10 +00:00
|
|
|
return;
|
|
|
|
|
2014-08-21 12:50:13 +00:00
|
|
|
ScriptEdit::ChangeLock lock (*mEditor);
|
2013-09-19 12:26:09 +00:00
|
|
|
|
2015-09-22 13:36:00 +00:00
|
|
|
bool updateRequired = false;
|
|
|
|
|
|
|
|
for (int i=topLeft.row(); i<=bottomRight.row(); ++i)
|
|
|
|
{
|
|
|
|
std::string id = mModel->data (mModel->index (i, mIdColumn)).toString().toUtf8().constData();
|
|
|
|
if (mErrors->clearLocals (id))
|
|
|
|
updateRequired = true;
|
|
|
|
}
|
|
|
|
|
2013-04-10 18:14:10 +00:00
|
|
|
QModelIndex index = mModel->getModelIndex (getUniversalId().getId(), mColumn);
|
|
|
|
|
2015-07-18 15:00:00 +00:00
|
|
|
if (index.row()>=topLeft.row() && index.row()<=bottomRight.row())
|
2013-04-10 18:14:10 +00:00
|
|
|
{
|
2015-07-18 15:00:00 +00:00
|
|
|
if (mStateColumn>=topLeft.column() && mStateColumn<=bottomRight.column())
|
|
|
|
updateDeletedState();
|
2015-07-16 10:36:20 +00:00
|
|
|
|
2015-07-18 15:00:00 +00:00
|
|
|
if (mColumn>=topLeft.column() && mColumn<=bottomRight.column())
|
|
|
|
{
|
|
|
|
QString source = mModel->data (index).toString();
|
2015-07-16 10:36:20 +00:00
|
|
|
|
2015-07-18 15:00:00 +00:00
|
|
|
QTextCursor cursor = mEditor->textCursor();
|
|
|
|
mEditor->setPlainText (source);
|
|
|
|
mEditor->setTextCursor (cursor);
|
|
|
|
|
2015-09-22 13:36:00 +00:00
|
|
|
updateRequired = true;
|
2015-07-18 15:00:00 +00:00
|
|
|
}
|
2013-04-10 18:14:10 +00:00
|
|
|
}
|
2015-09-22 13:36:00 +00:00
|
|
|
|
|
|
|
if (updateRequired)
|
|
|
|
recompile();
|
2013-04-10 18:14:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::ScriptSubView::rowsAboutToBeRemoved (const QModelIndex& parent, int start, int end)
|
|
|
|
{
|
2015-09-22 13:36:00 +00:00
|
|
|
bool updateRequired = false;
|
|
|
|
|
|
|
|
for (int i=start; i<=end; ++i)
|
|
|
|
{
|
|
|
|
std::string id = mModel->data (mModel->index (i, mIdColumn)).toString().toUtf8().constData();
|
|
|
|
if (mErrors->clearLocals (id))
|
|
|
|
updateRequired = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (updateRequired)
|
|
|
|
recompile();
|
|
|
|
|
2013-04-10 19:31:14 +00:00
|
|
|
QModelIndex index = mModel->getModelIndex (getUniversalId().getId(), mColumn);
|
2013-04-10 18:14:10 +00:00
|
|
|
|
2013-04-10 19:31:14 +00:00
|
|
|
if (!parent.isValid() && index.row()>=start && index.row()<=end)
|
2014-10-25 16:13:56 +00:00
|
|
|
emit closeRequest();
|
2013-09-19 12:26:09 +00:00
|
|
|
}
|
|
|
|
|
2015-07-11 14:09:13 +00:00
|
|
|
void CSVWorld::ScriptSubView::switchToRow (int row)
|
|
|
|
{
|
|
|
|
int idColumn = mModel->findColumnIndex (CSMWorld::Columns::ColumnId_Id);
|
|
|
|
std::string id = mModel->data (mModel->index (row, idColumn)).toString().toUtf8().constData();
|
|
|
|
setUniversalId (CSMWorld::UniversalId (CSMWorld::UniversalId::Type_Script, id));
|
|
|
|
|
|
|
|
mEditor->setPlainText (mModel->data (mModel->index (row, mColumn)).toString());
|
|
|
|
|
|
|
|
std::vector<std::string> selection (1, id);
|
|
|
|
mCommandDispatcher.setSelection (selection);
|
2015-07-18 15:00:00 +00:00
|
|
|
|
|
|
|
updateDeletedState();
|
2015-07-11 14:09:13 +00:00
|
|
|
}
|
2015-07-14 11:52:48 +00:00
|
|
|
|
|
|
|
void CSVWorld::ScriptSubView::switchToId (const std::string& id)
|
|
|
|
{
|
|
|
|
switchToRow (mModel->getModelIndex (id, 0).row());
|
|
|
|
}
|
2015-07-17 11:42:25 +00:00
|
|
|
|
2015-07-17 11:53:22 +00:00
|
|
|
void CSVWorld::ScriptSubView::highlightError (int line, int column)
|
2015-07-17 11:42:25 +00:00
|
|
|
{
|
|
|
|
QTextCursor cursor = mEditor->textCursor();
|
|
|
|
|
|
|
|
cursor.movePosition (QTextCursor::Start);
|
|
|
|
if (cursor.movePosition (QTextCursor::Down, QTextCursor::MoveAnchor, line))
|
2015-07-17 11:53:22 +00:00
|
|
|
cursor.movePosition (QTextCursor::Right, QTextCursor::MoveAnchor, column);
|
2015-07-17 11:42:25 +00:00
|
|
|
|
|
|
|
mEditor->setFocus();
|
|
|
|
mEditor->setTextCursor (cursor);
|
|
|
|
}
|
2015-07-18 11:09:17 +00:00
|
|
|
|
|
|
|
void CSVWorld::ScriptSubView::updateRequest()
|
|
|
|
{
|
|
|
|
QModelIndex index = mModel->getModelIndex (getUniversalId().getId(), mColumn);
|
|
|
|
|
|
|
|
QString source = mModel->data (index).toString();
|
|
|
|
|
|
|
|
mErrors->update (source.toUtf8().constData());
|
2015-11-21 11:45:11 +00:00
|
|
|
|
|
|
|
adjustSplitter();
|
2015-07-18 11:09:17 +00:00
|
|
|
}
|