From b98120ebb877ac36a16fefa5e4d1f94b19ff5b90 Mon Sep 17 00:00:00 2001
From: dfighter1985 <dfighter1985@localhost>
Date: Mon, 22 Sep 2014 23:24:48 +0200
Subject: [PATCH] Studio should no longer crash when multiple plugins that use
 LIGO are loaded. LIGO classes are now guarded against multiple registrations.
 If it's tried log messages are generated. Mission Compiler and World Editor
 will now apply their own LIGO configs when the user switches to their tab.

---
 code/nel/src/ligo/primitive.cpp               | 11 ++++
 .../src/plugins/core/context_manager.cpp      |  4 +-
 code/studio/src/plugins/core/icontext.h       |  2 +
 .../mission_compiler_main_window.cpp          |  5 ++
 .../mission_compiler_main_window.h            |  2 +
 .../mission_compiler_plugin.h                 |  6 ++
 .../world_editor/world_editor_plugin.cpp      | 65 ++++++++++---------
 .../world_editor/world_editor_plugin.h        |  6 +-
 .../world_editor/world_editor_window.h        |  1 +
 9 files changed, 69 insertions(+), 33 deletions(-)

diff --git a/code/nel/src/ligo/primitive.cpp b/code/nel/src/ligo/primitive.cpp
index 9cf7df13f..ba9b69435 100644
--- a/code/nel/src/ligo/primitive.cpp
+++ b/code/nel/src/ligo/primitive.cpp
@@ -2738,8 +2738,17 @@ CPrimitiveContext::CPrimitiveContext():
 }
 
 
+static bool LIGORegistered = false;
+
+
 void Register ()
 {
+    if( LIGORegistered )
+    {
+        nlinfo( "LIGO classes have already been registered." );
+        return;
+    }
+
 	NLMISC_REGISTER_CLASS(CPropertyString);
 	NLMISC_REGISTER_CLASS(CPropertyStringArray);
 	NLMISC_REGISTER_CLASS(CPropertyColor);
@@ -2748,6 +2757,8 @@ void Register ()
 	NLMISC_REGISTER_CLASS(CPrimPath);
 	NLMISC_REGISTER_CLASS(CPrimZone);
 	NLMISC_REGISTER_CLASS(CPrimAlias);
+
+    LIGORegistered = true;
 }
 
 // ***************************************************************************
diff --git a/code/studio/src/plugins/core/context_manager.cpp b/code/studio/src/plugins/core/context_manager.cpp
index 3b02b411c..203738faf 100644
--- a/code/studio/src/plugins/core/context_manager.cpp
+++ b/code/studio/src/plugins/core/context_manager.cpp
@@ -143,6 +143,8 @@ void ContextManager::currentTabChanged(int index)
 	if (index >= 0)
 	{
 		IContext *context = d->m_contexts.at(index);
+        context->onActivated();
+
 		Q_EMIT currentContextChanged(context);
 	}
 }
@@ -158,4 +160,4 @@ int ContextManager::indexOf(const QString &id) const
 	return -1;
 }
 
-} /* namespace Core */
\ No newline at end of file
+} /* namespace Core */
diff --git a/code/studio/src/plugins/core/icontext.h b/code/studio/src/plugins/core/icontext.h
index d2cbb412c..616e0db14 100644
--- a/code/studio/src/plugins/core/icontext.h
+++ b/code/studio/src/plugins/core/icontext.h
@@ -69,6 +69,8 @@ public:
 	virtual void newDocument(){}
 
 	virtual void close(){}
+
+    virtual void onActivated(){}
 };
 
 } // namespace Core
diff --git a/code/studio/src/plugins/mission_compiler/mission_compiler_main_window.cpp b/code/studio/src/plugins/mission_compiler/mission_compiler_main_window.cpp
index 10985aa38..efadd3949 100644
--- a/code/studio/src/plugins/mission_compiler/mission_compiler_main_window.cpp
+++ b/code/studio/src/plugins/mission_compiler/mission_compiler_main_window.cpp
@@ -484,6 +484,11 @@ void MissionCompilerMainWindow::saveConfig() {
 	settings->sync();
 }
 
+void MissionCompilerMainWindow::onActivated()
+{
+    NLLIGO::CPrimitiveContext::instance().CurrentLigoConfig = &m_ligoConfig;
+}
+
 void MissionCompilerMainWindow::handleChangedSettings()
 {
 	QStringList servers;
diff --git a/code/studio/src/plugins/mission_compiler/mission_compiler_main_window.h b/code/studio/src/plugins/mission_compiler/mission_compiler_main_window.h
index dc19db1c6..3d59e206a 100644
--- a/code/studio/src/plugins/mission_compiler/mission_compiler_main_window.h
+++ b/code/studio/src/plugins/mission_compiler/mission_compiler_main_window.h
@@ -31,6 +31,8 @@ public:
 	void saveConfig();
 	QUndoStack *getUndoStack() { return m_undoStack; }
 
+    void onActivated();
+
 	typedef std::map<std::string, CMission> TMissionContainer;
 
 public Q_SLOTS:
diff --git a/code/studio/src/plugins/mission_compiler/mission_compiler_plugin.h b/code/studio/src/plugins/mission_compiler/mission_compiler_plugin.h
index 2ad92b40f..cc2cac47c 100644
--- a/code/studio/src/plugins/mission_compiler/mission_compiler_plugin.h
+++ b/code/studio/src/plugins/mission_compiler/mission_compiler_plugin.h
@@ -83,6 +83,12 @@ public:
     virtual void open() {}
 
 
+    void onActivated()
+    {
+        m_missionCompilerMainWindow->onActivated();
+    }
+
+
 	MissionCompilerMainWindow *m_missionCompilerMainWindow;
 };
 
diff --git a/code/studio/src/plugins/world_editor/world_editor_plugin.cpp b/code/studio/src/plugins/world_editor/world_editor_plugin.cpp
index 3170bce60..722c14733 100644
--- a/code/studio/src/plugins/world_editor/world_editor_plugin.cpp
+++ b/code/studio/src/plugins/world_editor/world_editor_plugin.cpp
@@ -54,36 +54,6 @@ bool WorldEditorPlugin::initialize(ExtensionSystem::IPluginManager *pluginManage
 
 	WorldEditorSettingsPage *weSettings = new WorldEditorSettingsPage(this);
 	addAutoReleasedObject(weSettings);
-
-	QSettings *settings = Core::ICore::instance()->settings();
-	settings->beginGroup(Constants::WORLD_EDITOR_SECTION);
-	m_ligoConfig.CellSize = settings->value(Constants::WORLD_EDITOR_CELL_SIZE, "160").toFloat();
-	m_ligoConfig.Snap = settings->value(Constants::WORLD_EDITOR_SNAP, "1").toFloat();
-	m_ligoConfig.ZoneSnapShotRes = settings->value(Constants::ZONE_SNAPSHOT_RES, "128").toUInt();
-	QString fileName = settings->value(Constants::PRIMITIVE_CLASS_FILENAME, "world_editor_classes.xml").toString();
-	settings->endGroup();
-	try
-	{
-		// Search path of file world_editor_classes.xml
-		std::string ligoPath = NLMISC::CPath::lookup(fileName.toUtf8().constData());
-		// Init LIGO
-		m_ligoConfig.readPrimitiveClass(ligoPath.c_str(), true);
-		NLLIGO::Register();
-		NLLIGO::CPrimitiveContext::instance().CurrentLigoConfig = &m_ligoConfig;
-	}
-	catch (NLMISC::Exception &e)
-	{
-		*errorString = tr("(%1)").arg(e.what());
-		return false;
-	}
-
-	// Reset
-	m_ligoConfig.resetPrimitiveConfiguration ();
-
-	// TODO: get file names! from settings
-	m_ligoConfig.readPrimitiveClass("world_editor_primitive_configuration.xml", true);
-
-
 	addAutoReleasedObject(new WorldEditorContext(this));
 	return true;
 }
@@ -117,6 +87,34 @@ WorldEditorContext::WorldEditorContext(QObject *parent)
 	  m_worldEditorWindow(0)
 {
 	m_worldEditorWindow = new WorldEditorWindow();
+
+    QSettings *settings = Core::ICore::instance()->settings();
+    settings->beginGroup(Constants::WORLD_EDITOR_SECTION);
+    m_ligoConfig.CellSize = settings->value(Constants::WORLD_EDITOR_CELL_SIZE, "160").toFloat();
+    m_ligoConfig.Snap = settings->value(Constants::WORLD_EDITOR_SNAP, "1").toFloat();
+    m_ligoConfig.ZoneSnapShotRes = settings->value(Constants::ZONE_SNAPSHOT_RES, "128").toUInt();
+    QString fileName = settings->value(Constants::PRIMITIVE_CLASS_FILENAME, "world_editor_classes.xml").toString();
+    settings->endGroup();
+    try
+    {
+        // Search path of file world_editor_classes.xml
+        std::string ligoPath = NLMISC::CPath::lookup(fileName.toUtf8().constData());
+        // Init LIGO
+        m_ligoConfig.readPrimitiveClass(ligoPath.c_str(), true);
+        NLLIGO::Register();
+        NLLIGO::CPrimitiveContext::instance().CurrentLigoConfig = &m_ligoConfig;
+    }
+    catch (NLMISC::Exception &e)
+    {
+        nlinfo( "Error starting LIGO." );
+    }
+
+    // Reset
+    m_ligoConfig.resetPrimitiveConfiguration ();
+
+    // TODO: get file names! from settings
+    m_ligoConfig.readPrimitiveClass("world_editor_primitive_configuration.xml", true);
+
 }
 
 QUndoStack *WorldEditorContext::undoStack()
@@ -124,6 +122,11 @@ QUndoStack *WorldEditorContext::undoStack()
 	return m_worldEditorWindow->undoStack();
 }
 
+void WorldEditorContext::onActivated()
+{
+    NLLIGO::CPrimitiveContext::instance().CurrentLigoConfig = &m_ligoConfig;
+}
+
 void WorldEditorContext::open()
 {
 	m_worldEditorWindow->open();
@@ -136,4 +139,4 @@ QWidget *WorldEditorContext::widget()
 
 }
 
-Q_EXPORT_PLUGIN(WorldEditor::WorldEditorPlugin)
\ No newline at end of file
+Q_EXPORT_PLUGIN(WorldEditor::WorldEditorPlugin)
diff --git a/code/studio/src/plugins/world_editor/world_editor_plugin.h b/code/studio/src/plugins/world_editor/world_editor_plugin.h
index 686b87e14..cfb5448e3 100644
--- a/code/studio/src/plugins/world_editor/world_editor_plugin.h
+++ b/code/studio/src/plugins/world_editor/world_editor_plugin.h
@@ -59,7 +59,6 @@ protected:
 	NLMISC::CLibraryContext *m_libContext;
 
 private:
-	NLLIGO::CLigoConfig m_ligoConfig;
 	ExtensionSystem::IPluginManager *m_plugMan;
 	QList<QObject *> m_autoReleaseObjects;
 };
@@ -88,9 +87,14 @@ public:
 
 	virtual QUndoStack *undoStack();
 
+    void onActivated();
+
 	virtual QWidget *widget();
 
 	WorldEditorWindow *m_worldEditorWindow;
+
+private:
+    NLLIGO::CLigoConfig m_ligoConfig;
 };
 
 } // namespace WorldEditor
diff --git a/code/studio/src/plugins/world_editor/world_editor_window.h b/code/studio/src/plugins/world_editor/world_editor_window.h
index 60a6a988a..9080187f9 100644
--- a/code/studio/src/plugins/world_editor/world_editor_window.h
+++ b/code/studio/src/plugins/world_editor/world_editor_window.h
@@ -46,6 +46,7 @@ public:
 	~WorldEditorWindow();
 
 	QUndoStack *undoStack() const;
+    void onActivated();
 	void maybeSave();
 
 Q_SIGNALS: