Differences between namespaces and configurations
The way configuration files are loaded has changed radically from version 4.x. Previously, configuration files were imported into a Pilsner virtual machine, using the SpaceAPI.Send() method. If the SpaceX control was used, this import was done indirectly by setting the ConfigurationFile property on the control. The import operation didn't return any created objects; instead it had the side effect that some of the created objects were registered in so called namespaces in the global Pilsner machine.
There were several problems with this solution. One was that there was always a risk of name clashes. An application might directly or indirectly load many configuration files, it could happen that two objects with the same name and namespace was created. Another problem was that it was virtually impossible to "unload" a configuration, since lots of objects were registered globally.
In Carmenta Engine 5, there is no global Pilsner virtual machine. Instead, a configuration file is loaded by creating a Configuration instance, passing the name of the file to the constructor. The created instance holds all the objects loaded from the configuration file, and the Configuration.GetPublicObject method can be used to retrieve them. An application typically loads the configuration this way, retrieves the view and passes this directly to the map control.
This new approach solves most of the previous problems:
There are no longer any implicit side effect of loading a file - if the Configuration object is released, or if an error occurs halfway through the load, all created objects will be destroyed again.
Chances of name collisions are reduced.
Created objects can easily be be retrieved, either by Configuration.GetPublicObject or using the View.FindChildObject method of a loaded view.
Another common problem was that cloning a view and add it to a second SpaceX control was awkward, to say the least. This is now much simpler:
// Clone a view
bool CloneView(SpaceX oldMapControl, SpaceX newMapControl)
{
String name = GenerateNewUniqueName();
View newView = v.CloneAs(name) as View;
newView.Register();
newMapControl.ConfigurationFile = oldMapControl.ConfigurationFile; // Does nothing, file already loaded
newMapControl.NameSpace = oldMapControl.NameSpace;
newMapControl.ViewName = name;
}
// Clone a view
bool CloneView(MapControl oldMapControl, MapControl newMapControl)
{
newMapControl.View = oldMapControl.View.Clone() as View;
}
As in version 4.x, the Configuration.GetPublicObject requires that the object is marked as Public in Carmenta Studio. This is not required if you already have a reference to a view, layer or operator and calls FindChildObject(). View and LoadableLayer instances are always marked as public.
If a reference to another configuration file is found during a load operation, that file is read in its entirety to create a "sub-configuration". If another reference is found to the same file, it will not be loaded again; the already loaded sub-configuration will be reused. This makes it possible to put e.g. all datasets in separate file, and reference them from several loadable layers. The object from the loaded sub-configurations will be merged into the main configuration object before return.
The above reuse of loaded files works only during one load operation. There is however a possibility to reuse configurations loaded by the application at different times: If a configuration is marked as Shared in Carmenta Studio, it will be added to a global list of shared configurations. If the same file is ever referenced again, the same configuration object will be returned.