src/main.cpp

Sun, 09 Apr 2023 15:59:08 +0300

author
Teemu Piippo <teemu.s.piippo@gmail.com>
date
Sun, 09 Apr 2023 15:59:08 +0300
changeset 362
e1d646a4cbd8
parent 361
c5e8b68e34f8
child 363
16f5dcfa7c9f
permissions
-rw-r--r--

Extracted the state of the program into a MainState structure, and extracted local functions of main() into static functions.
I was planning to make the core logic and state of the program into a Main class, which would be a QObject that would
have lots of signals and slots, but it looks like this works even without it

#include <QApplication>
#include <QClipboard>
#include <QCloseEvent>
#include <QFileDialog>
#include <QMdiSubWindow>
#include <QMessageBox>
#include <QScrollBar>
#include <QStackedWidget>
#include <QTranslator>
#include "src/gl/partrenderer.h"
#include "src/layers/axeslayer.h"
#include "src/layers/edittools.h"
#include "src/layers/gridlayer.h"
#include "src/ldrawalgorithm.h"
#include "src/mainwindow.h"
#include "src/messagelog.h"
#include "src/modelsubwindow.h"
#include "src/settings.h"
#include "src/settingseditor/settingseditor.h"
#include "src/ui/circletooloptionswidget.h"
#include "src/version.h"
#include "src/widgets/colorselectdialog.h"
#include "src/parser.h"
#include "src/ldrawsyntaxhighlighter.h"
#include <GL/glew.h>
#include "src/main.h"

static const QDir LOCALE_DIR {":/locale"};

class ModelData : public QObject
{
	Q_OBJECT
public:
	ModelData(QObject* parent) : QObject {parent} {}
	std::unique_ptr<PartRenderer> canvas;
	std::unique_ptr<EditTools> tools;
	std::unique_ptr<AxesLayer> axesLayer;
	std::unique_ptr<GridLayer> gridLayer;
	std::unique_ptr<QTextCursor> textcursor;
	QTextDocument* model;
};

class Signal final : public QObject
{
	Q_OBJECT
public:
	Signal() : QObject{}{}
	virtual ~Signal(){}
	void emit()
	{
		Q_EMIT this->triggered();
	}
Q_SIGNALS:
	void triggered();
};
#include <main.moc>

static void doQtRegistrations()
{
	QCoreApplication::setApplicationName(QStringLiteral(CMAKE_PROJECT_NAME));
	QCoreApplication::setOrganizationName("hecknology.net");
	QCoreApplication::setOrganizationDomain("hecknology.net");
	qRegisterMetaType<Message>();
	qRegisterMetaType<Library>();
	qRegisterMetaType<QList<Library>>();
	qRegisterMetaType<QMdiArea::ViewMode>();
	qRegisterMetaType<gl::RenderStyle>();
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
	qRegisterMetaTypeStreamOperators<Library>("Library");
	qRegisterMetaTypeStreamOperators<Libraries>("Libraries");
	qRegisterMetaTypeStreamOperators<gl::RenderStyle>();
	qRegisterMetaTypeStreamOperators<QMdiArea::ViewMode>();
	qRegisterMetaTypeStreamOperators<Qt::ToolButtonStyle>();
#endif
}

static std::optional<ModelId> openModelFromPath(
	const QString& path,
	const LibrariesModel* libraries,
	DocumentManager* documents,
	QWidget* parent)
{
	QString errorString;
	QTextStream errorStream{&errorString};
	const std::optional<ModelId> modelIdOpt = documents->openModel(
		path,
		errorStream,
		OpenType::ManuallyOpened);
	if (modelIdOpt.has_value()) {
		const DocumentManager::MissingDependencies missing = documents->loadDependenciesForAllModels(*libraries);
		if (not missing.empty()) {
			QMessageBox::warning(
				parent,
				QObject::tr("Problem loading references"),
				errorStringFromMissingDependencies(missing));
		}
	}
	else {
		QMessageBox::critical(
			parent,
			QObject::tr("Problem opening file"),
			QObject::tr("Could not open %1: %2").arg(quoted(path), errorString)
		);
	}
	return modelIdOpt;
}

static QString getOpenModelPath(QWidget* parent)
{
	return QFileDialog::getOpenFileName(
		parent,
		QObject::tr("Open model"),
		"",
		QObject::tr("LDraw models (*.ldr *.dat)"));
}

static ModelData* findModelData(const DocumentManager* documents, ModelId modelId)
{
	return documents->findPayload<ModelData>(modelId);
}

static ModelSubWindow* currentModelSubWindow(MainWindow* ui)
{
	auto* w = ui->mdiArea->activeSubWindow();
	return qobject_cast<ModelSubWindow*>(w);
}

static ModelData* currentModelData(MainWindow* ui, const DocumentManager* documents)
{
	if (auto* const activeSubWindow = currentModelSubWindow(ui)) {
		return findModelData(documents, activeSubWindow->modelId);
	}
	else {
		return nullptr;
	}
}

static std::optional<ModelId> findCurrentModelId(MainWindow* ui)
{
	ModelSubWindow* activeSubWindow = qobject_cast<ModelSubWindow*>(ui->mdiArea->activeSubWindow());
	if (activeSubWindow != nullptr) {
		return activeSubWindow->modelId;
	}
	else {
		return {};
	}
}

static ColorTable loadColors(const LibrariesModel* libraries)
{
	QTextStream errors;
	return libraries->loadColorTable(errors);
}

static QString tabName(const QFileInfo& fileInfo)
{
	QString result = fileInfo.baseName();
	if (result.isEmpty()) {
		result = QObject::tr("<unnamed>");
	}
	return result;
}

template<typename Fn>
static void forEachModel(const DocumentManager* documents, Fn&& fn)
{
	forValueInMap(*documents, [&fn](const DocumentManager::ModelInfo& info)
	{
		ModelData* modelSpecificData = qobject_cast<ModelData*>(info.payload);
		if (modelSpecificData != nullptr) {
			fn(&info, modelSpecificData);
		}
	});
}

static void updateRenderPreferences(
	MainWindow* ui,
	const gl::RenderPreferences* renderPreferences,
	const DocumentManager* documents)
{
	forEachModel(documents, [&renderPreferences](const void*, const ModelData* data){
		if (data->canvas != nullptr) {
			data->canvas->setRenderPreferences(*renderPreferences);
			data->canvas->setLayerEnabled(data->axesLayer.get(), renderPreferences->drawAxes);
		}
	});
	ui->setRenderStyle(renderPreferences->style);
	ui->actionDrawAxes->setChecked(renderPreferences->drawAxes);
	ui->actionWireframe->setChecked(renderPreferences->wireframe);
}

static gl::RenderPreferences loadRenderPreferences()
{
	return gl::RenderPreferences{
		.style = setting<Setting::RenderStyle>(),
		.mainColor = setting<Setting::MainColor>(),
		.backgroundColor = setting<Setting::BackgroundColor>(),
		.selectedColor = setting<Setting::SelectedColor>(),
		.lineThickness = setting<Setting::LineThickness>(),
		.lineAntiAliasing = setting<Setting::LineAntiAliasing>(),
		.drawAxes = setting<Setting::DrawAxes>(),
		.wireframe = setting<Setting::Wireframe>(),
	};
}

struct ToolWidgets
{
	CircleToolOptionsWidget* circleToolOptions;
};

static void initializeTools(MainWindow* ui, ToolWidgets* toolWidgets, QWidget* parent)
{
	const struct
	{
		QString name, tooltip;
		QPixmap icon;
		QWidget* widget;
	} editingModesInfo[] = {
		{
			.name = QObject::tr("Select"),
			.tooltip = QObject::tr("Select elements from the model."),
			.icon = {":/icons/navigate-outline.png"},
			.widget = nullptr,
		},
		{
			.name = QObject::tr("Draw"),
			.tooltip = QObject::tr("Draw new elements into the model."),
			.icon = {":/icons/pencil-outline.png"},
			.widget = nullptr,
		},
		{
			.name = QObject::tr("Circle"),
			.tooltip = QObject::tr("Draw circular primitives."),
			.icon = {":/icons/linetype-circularprimitive.png"},
			.widget = toolWidgets->circleToolOptions,
		},
	};
	for (int i = 0; i < countof(editingModesInfo); ++i) {
		const auto& editingModeInfo = editingModesInfo[i];
		QAction* action = new QAction{editingModeInfo.name, parent};
		action->setCheckable(true);
		action->setEnabled(false);
		action->setData(static_cast<EditingMode>(i));
		action->setToolTip(editingModeInfo.tooltip);
		action->setIcon(QPixmap{editingModeInfo.icon});
		ui->editingModesToolBar->addAction(action);
		QWidget* widget = editingModeInfo.widget;
		if (widget == nullptr) {
			widget = new QWidget{parent};
		}
		ui->toolWidgetStack->addWidget(widget);
		QObject::connect(action, &QAction::triggered, [ui, i]{
			ui->toolWidgetStack->setCurrentIndex(i);
		});
	}
}

template<class SubWindow, class... Args>
SubWindow* createSubWindow(QMdiArea* mdiArea, Args&&... args)
{
	// Qt seems to have a bug where the first created sub window does not render
	// properly until it is minimized and maximized again. This only happens
	// if we give the mdi area as a parent argument. As a work-around, we create
	// the sub window with parent=nullptr, and add it manually.
	// c.f. https://bugreports.qt.io/browse/QTBUG-69495
	SubWindow* subWindow = new SubWindow{args..., nullptr};
	mdiArea->addSubWindow(subWindow);
	return subWindow;
}

static void executeAction(QTextDocument* model, const ModelAction& action)
{
	std::visit(overloaded{
		[model](const AppendToModel& action){
			QTextCursor cursor{model};
			cursor.movePosition(QTextCursor::End);
			const QString newText = modelElementToString(action.newElement);
			// Make sure we have an empty line
			if (not model->lastBlock().text().isEmpty()) {
				cursor.insertBlock();
			}
			cursor.insertText(newText);
		},
		[](const DeleteFromModel&){},
		[model](const ModifyModel& action){
			QTextBlock block = model->findBlockByLineNumber((int) action.position);
			if (block.isValid()) {
				QTextCursor cursor{block};
				cursor.select(QTextCursor::LineUnderCursor);
				cursor.insertText(modelElementToString(action.newElement));
			}
			//model->assignAt(action.position, action.newElement);
		},
	}, action);
}

QFont codeEditorFontFromSettings()
{
	QFont font{};
	if (setting<Setting::CodeEditorUseSystemFont>())
	{
		font.setStyleHint(QFont::Monospace);
	}
	else
	{
		font.setFamily(setting<Setting::CodeEditorFontFamily>());
		font.setPointSize(setting<Setting::CodeEditorFontSize>());
	}
	return font;
}

struct MainState
{
	MainWindow mainWindow;
	DocumentManager documents;
	QString currentLanguage = "en";
	QTranslator translator{&mainWindow};
	LibrariesModel libraries{&mainWindow};
	QStringList recentlyOpenedFiles;
	ColorTable colorTable;
	gl::RenderPreferences renderPreferences;
	MessageLog messageLog;
	ToolWidgets toolWidgets{
		.circleToolOptions = new CircleToolOptionsWidget{&mainWindow},
	};
};

static void openModelForEditing(MainState* state, const ModelId modelId)
{
	QTextDocument* model = state->documents.getModelById(modelId);
	if (model != nullptr) {
		ModelData* data = new ModelData(&state->documents);
		data->tools = std::make_unique<EditTools>();
		data->canvas = std::make_unique<PartRenderer>(model, &state->documents, state->colorTable);
		data->axesLayer = std::make_unique<AxesLayer>();
		data->gridLayer = std::make_unique<GridLayer>();
		data->gridLayer->setGridMatrix(DEFAULT_GRID_MATRIX);
		data->tools->setGridMatrix(DEFAULT_GRID_MATRIX);
		data->model = model;
		data->canvas->addRenderLayer(data->axesLayer.get());
		data->canvas->setLayerEnabled(data->axesLayer.get(), setting<Setting::DrawAxes>());
		data->canvas->addRenderLayer(data->gridLayer.get());
		data->canvas->addRenderLayer(data->tools.get());
		new LDrawSyntaxHighlighter{model};
		data->textcursor = std::make_unique<QTextCursor>(model);
		state->documents.setModelPayload(modelId, data);
		QObject::connect(
			data->tools.get(),
			&EditTools::modelAction,
			std::bind(executeAction, model, std::placeholders::_1));
		data->canvas->setRenderPreferences(state->renderPreferences);
		QObject::connect(
			data->tools.get(),
			&EditTools::newStatusText,
			[&state](const QString& newStatusText) {
				state->mainWindow.statusBar()->showMessage(newStatusText);
			});
#if 0
		QObject::connect(
			data->tools.get(),
			&EditTools::select,
			[modelId, &documents](const QSet<ElementId>& indices, bool retain) {
				ModelData* data = findModelData(&documents, modelId);
				if (data != nullptr) {
					if (not retain) {
						data->textcursor->clearSelection();
					}
					for (const ElementId id : indices) {
						opt<int> index = data->model->find(id);
						if (index.has_value()) {
							const QModelIndex qindex = data->model->index(*index);
							data->itemSelectionModel->select(qindex, QItemSelectionModel::Select);
						}
					}
				}
			});
#endif
#if 0
		QObject::connect(this, &Main::settingsChanged, [modelId, this]{
			ModelData* data = findModelData(&state.documents, modelId);
			if (data != nullptr) {
				data->gridLayer->settingsChanged();
			}
		});
#endif
		QObject::connect(data->canvas.get(), &PartRenderer::message, &state->messageLog, &MessageLog::addMessage);
		QObject::connect(
			data->tools.get(),
			&EditTools::suggestCursor,
			data->canvas.get(),
			&QWidget::setCursor);
		data->tools->setEditMode(SelectMode);
		const QFileInfo fileInfo{*state->documents.modelPath(modelId)};
		auto* const subWindow = createSubWindow<ModelSubWindow>(state->mainWindow.mdiArea, modelId);
		subWindow->setMinimumSize({96, 96});
		subWindow->resize({320, 200});
		subWindow->setWidget(data->canvas.get());
		subWindow->setWindowTitle(tabName(fileInfo));
		subWindow->show();
	}
}

static void updateRecentlyOpenedDocumentsMenu(MainState* state)
{
	state->mainWindow.rebuildRecentFilesMenu(state->recentlyOpenedFiles);
}

static void restoreSettings(MainState* state)
{
	state->recentlyOpenedFiles = setting<Setting::RecentFiles>();
	state->renderPreferences = loadRenderPreferences();
	state->libraries.restoreFromSettings();
	updateRecentlyOpenedDocumentsMenu(state);
	state->colorTable = loadColors(&state->libraries);
	updateRenderPreferences(&state->mainWindow, &state->renderPreferences, &state->documents);
	state->mainWindow.mdiArea->setViewMode(setting<Setting::ViewMode>());
	state->mainWindow.retranslateUi(&state->mainWindow);
	state->mainWindow.setToolButtonStyle(setting<Setting::ToolButtonStyle>());
}

static void saveSettings(MainState* state)
{
	setSetting<Setting::MainWindowGeometry>(state->mainWindow.saveGeometry());
	setSetting<Setting::MainWindowState>(state->mainWindow.saveState());
	setSetting<Setting::RecentFiles>(state->recentlyOpenedFiles);
	setSetting<Setting::RenderStyle>(state->renderPreferences.style);
	setSetting<Setting::DrawAxes>(state->renderPreferences.drawAxes);
	setSetting<Setting::Wireframe>(state->renderPreferences.wireframe);
	state->libraries.storeToSettings();
}

static void addRecentlyOpenedFile(MainState* state, const QString& path)
{
	constexpr int maxRecentlyOpenedFiles = 10;
	state->recentlyOpenedFiles.removeAll(path);
	state->recentlyOpenedFiles.insert(0, path);
	while (state->recentlyOpenedFiles.size() > maxRecentlyOpenedFiles)
	{
		state->recentlyOpenedFiles.removeLast();
	}
	saveSettings(state);
	updateRecentlyOpenedDocumentsMenu(state);
}

static void saveCurrentModel(MainState* state, ModelId modelId)
{
	QString error;
	QTextStream errorStream{&error};
	const bool succeeded = state->documents.saveModel(modelId, errorStream);
	if (not succeeded)
	{
		QMessageBox::critical(&state->mainWindow, QObject::tr("Save error"), error);
	}
	else
	{
		const QString* pathPtr = state->documents.modelPath(modelId);
		if (pathPtr != nullptr) {
			addRecentlyOpenedFile(state, *pathPtr);
		}
	}
}

static void saveCurrentModelAs(MainState* state)
{
	const std::optional<ModelId> modelId = findCurrentModelId(&state->mainWindow);
	if (modelId.has_value())
	{
		const QString* pathPtr = state->documents.modelPath(*modelId);
		QString defaultPath = (pathPtr != nullptr) ? *pathPtr : "";
		const QString newPath = QFileDialog::getSaveFileName(
			&state->mainWindow,
			QObject::tr("Save as…"),
			QFileInfo{defaultPath}.absoluteDir().path(), 
			QObject::tr("LDraw files (*.ldr *dat);;All files (*)")
		);
		if (not newPath.isEmpty()) {
			QString error;
			QTextStream errorStream{&error};
			state->documents.setModelPath(*modelId, newPath, state->libraries, errorStream);
			QMdiSubWindow* const subWindow = state->mainWindow.mdiArea->currentSubWindow();
			if (subWindow != nullptr) {
				subWindow->setWindowTitle(tabName(QFileInfo{newPath}));
			}
			saveCurrentModel(state, *modelId);
		}
	}
}

static void checkEditingModeAction(MainState* state, const EditingMode mode)
{
	const bool hasDocument = currentModelData(&state->mainWindow, &state->documents) != nullptr;
	for (QAction* action : state->mainWindow.editingModesToolBar->actions())
	{
		action->setEnabled(hasDocument);
		action->setChecked(hasDocument and action->data().value<EditingMode>() == mode);
	}
}

int main(int argc, char *argv[])
{
	doQtRegistrations();
	QApplication app{argc, argv};
	QApplication::setWindowIcon(QIcon{":/icons/appicon.png"});
	MainState state;
	QObject::connect(
		&state.mainWindow,
		&MainWindow::recentFileSelected,
		[&state](const QString& path) {
			const auto id = openModelFromPath(path, &state.libraries, &state.documents, &state.mainWindow);
			if (id.has_value())
			{
				openModelForEditing(&state, id.value());
				addRecentlyOpenedFile(&state, path);
			}
		}
	);
	QObject::connect(state.mainWindow.actionNew, &QAction::triggered,
		[&state]{
			openModelForEditing(&state, state.documents.newModel());
		}
	);
	QObject::connect(state.mainWindow.actionOpen, &QAction::triggered,
		[&state]
		{
			const QString path = getOpenModelPath(&state.mainWindow);
			if (not path.isEmpty())
			{
				const std::optional<ModelId> id = openModelFromPath(path, &state.libraries, &state.documents, &state.mainWindow);
				if (id.has_value()) {
					openModelForEditing(&state, id.value());
					addRecentlyOpenedFile(&state, path);
				}
			}
		}
	);
	QObject::connect(state.mainWindow.actionSettingsEditor, &QAction::triggered, [
		&state,
		defaultKeyboardShortcuts = uiutilities::makeKeySequenceMap(uiutilities::collectActions(&state.mainWindow))]
	{
		if (state.mainWindow.mdiArea->findChildren<SettingsEditor*>().isEmpty())
		{
			auto* const settingsEditor = createSubWindow<SettingsEditor>(state.mainWindow.mdiArea, defaultKeyboardShortcuts);
			QObject::connect(settingsEditor, &SettingsEditor::settingsChanged, [&]{
				restoreSettings(&state);
			});
			settingsEditor->setAttribute(Qt::WA_DeleteOnClose);
			settingsEditor->show();
		}
	});
	QObject::connect(state.mainWindow.actionQuit, &QAction::triggered, &state.mainWindow, &QMainWindow::close);
#if 0
	QObject::connect(ui.actionAdjustGridToView, &QAction::triggered, [&]{
		if (ModelData* data = currentModelData(&ui, &documents)) {
			adjustGridToView(data->canvas.get());
		}
	});
#endif
	QObject::connect(state.mainWindow.actionClose, &QAction::triggered, [&state]{
		if (ModelData* data = currentModelData(&state.mainWindow, &state.documents)) {
			// TODO
		}
	});
	QObject::connect(state.mainWindow.actionSaveAs, &QAction::triggered, [&state]{
		saveCurrentModelAs(&state);
	});
	QObject::connect(state.mainWindow.actionSave, &QAction::triggered, [&state]
	{
		const std::optional<ModelId> modelId = findCurrentModelId(&state.mainWindow);
		if (modelId.has_value()) {
			const QString* path = state.documents.modelPath(*modelId);
			if (path == nullptr or path->isEmpty()) {
				saveCurrentModelAs(&state);
			}
			else {
				saveCurrentModel(&state, *modelId);
			}
		}
	});
	QObject::connect(state.mainWindow.actionDrawAxes, &QAction::triggered, [&state]
		(bool drawAxes)
	{
		state.renderPreferences.drawAxes = drawAxes;
		saveSettings(&state);
		updateRenderPreferences(&state.mainWindow, &state.renderPreferences, &state.documents);
	});
	QObject::connect(state.mainWindow.actionWireframe, &QAction::triggered, [&state]
		(bool enabled)
	{
		state.renderPreferences.wireframe = enabled;
		saveSettings(&state);
		updateRenderPreferences(&state.mainWindow, &state.renderPreferences, &state.documents);
	});
	QObject::connect(&state.mainWindow, &MainWindow::renderStyleSelected, [&state]
		(gl::RenderStyle newStyle)
	{
		state.renderPreferences.style = newStyle;
		saveSettings(&state);
		updateRenderPreferences(&state.mainWindow, &state.renderPreferences, &state.documents);
	});
	initializeTools(&state.mainWindow, &state.toolWidgets, &state.mainWindow);
	for (QAction* action : state.mainWindow.editingModesToolBar->actions()) {
		QObject::connect(action, &QAction::triggered, [action, &state]
		{
			if (ModelData* data = currentModelData(&state.mainWindow, &state.documents))
			{
				const EditingMode mode = action->data().value<EditingMode>();
				data->tools->setEditMode(mode);
				checkEditingModeAction(&state, mode);
			}
		});
	}
	QObject::connect(state.mainWindow.mdiArea, &QMdiArea::subWindowActivated,
		[&state](QMdiSubWindow* subWindow)
	{
		ModelSubWindow* modelSubWindow = qobject_cast<ModelSubWindow*>(subWindow);
		if (modelSubWindow != nullptr)
		{
			if (ModelData* data = state.documents.findPayload<ModelData>(modelSubWindow->modelId))
			{
				checkEditingModeAction(&state, data->tools->currentEditingMode());
				state.mainWindow.modelEdit->setDocument(data->model);
				state.mainWindow.modelEdit->setTextCursor(*data->textcursor);
				state.mainWindow.modelEdit->setFont(codeEditorFontFromSettings());
			}
		}
		else
		{
			checkEditingModeAction(&state, EditingMode::SelectMode);
		}
		state.mainWindow.modelEdit->setEnabled(modelSubWindow != nullptr);
	});
	state.mainWindow.messageLog->setModel(&state.messageLog);
	QObject::connect(&state.documents, &DocumentManager::message, &state.messageLog, &MessageLog::addMessage);
	QObject::connect(&state.messageLog, &MessageLog::rowsAboutToBeInserted, [&state]{
		const auto bar = state.mainWindow.messageLog->verticalScrollBar();
		state.mainWindow.messageLog->setProperty("shouldAutoScroll", bar->value() == bar->maximum());
	});
	QObject::connect(&state.messageLog, &MessageLog::rowsInserted, [&state]{
		state.mainWindow.messageLog->resizeRowsToContents();
		if (state.mainWindow.messageLog->property("shouldAutoScroll").toBool()) {
			state.mainWindow.messageLog->scrollToBottom();
		}
	});
	QObject::connect(
		state.toolWidgets.circleToolOptions,
		&CircleToolOptionsWidget::optionsChanged,
		[&state](const CircleToolOptions& options) {
			if (ModelData* data = currentModelData(&state.mainWindow, &state.documents)) {
				data->tools->setCircleToolOptions(options);
			}
		});
	QObject::connect(
		state.mainWindow.actionMakeUnofficial,
		&QAction::triggered,
		[&state]{
			if (ModelData* data = currentModelData(&state.mainWindow, &state.documents)) {
				QTextDocument* const model = data->model;
				for (const ModelAction& action : ldraw::makeUnofficial(model)) {
					executeAction(model, action);
				}
			}
		});
	QObject::connect(state.mainWindow.actionAboutQt, &QAction::triggered, &QApplication::aboutQt);
	QObject::connect(
		state.mainWindow.modelEdit,
		&QPlainTextEdit::textChanged,
		[&state]{
			if (ModelData* data = currentModelData(&state.mainWindow, &state.documents)) {
				state.documents.loadDependenciesForAllModels(state.libraries);
				data->canvas->update();
			}
		});
	QObject::connect(
		state.mainWindow.gridMatrix,
		&MatrixEditor::valueChanged,
		[&](const glm::mat4& newGridMatrix)
		{
			forEachModel(&state.documents, [&](const void*, const ModelData* data)
			{
				if (data->gridLayer != nullptr and data->tools != nullptr and data->canvas != nullptr)
				{
					data->gridLayer->setGridMatrix(newGridMatrix);
					data->tools->setGridMatrix(newGridMatrix);
					data->canvas->setModelViewOrigin(newGridMatrix[3]);
					data->canvas->update();
				}
			});
		}
	);
	restoreSettings(&state);
	updateRenderPreferences(&state.mainWindow, &state.renderPreferences, &state.documents);
	const int result = app.exec();
	saveSettings(&state);
	return result;
}

mercurial