NavigatorOverride_3.3.x.zip
11 KB
This article shows how to override the standard project navigator in SDKs of version 3.3.x.
Notes
The solution presented in this article is for CODESYS V3.3.x. It will not work with older versions of CODESYS.
Idea
In the Automation Platform, there is a central implementation of a so-called NavigatorControl, accessible through the interface _3S.CODESYS.NavigatorControl.INavigatorControl. This control is able to display the contents of an arbitrary project in the Object Manager by specifying its project handle and the GUID of the structured view (The explanation of structured views goes beyond this article). The out-of-the-box functionality of NavigatorControl comprises amongst other things:
And it has got two interesting extension points:
So how is NavigatorControl used for the project navigators in CODESYS? Four plug-in classes utilize that control in the way we see it day-to-day:
The first idea would be to implement the factories all on ourselves, but this is not a good idea because the existing factories contain lots of code that is not part of the NavigatorControl and which we do not want to reimplement. Obviously the rendering and filter callbacks themselves, which contain lots of fine-tuned logic, but also other stuff like double-click handling, right-click handling, reaction to primary project changes, and so on.
So we must opt for another way. The magic word is delegation. Delegation is always a good design choice when somebody want to inherit from existing functionality but is not allowed to (as it is the case in Automation Platform: it is strictly forbidden to statically reference a plug-in in order to derive from a class implemented there). In our concrete case we can implement our own view factory implementation, but delegate most of the calls to the existing implementation in CODESYS. For the callbacks, we put our own implementations, but remember the existing ones so that we can delegate all aspects to the original implementation that we are not interested in. And one detail must also be considered: The existing menu commands to open or activate the navigator view must be overridden so that they redirect to our own view factory.
Solution
Let's take the "Devices" view for a detailed investigation of the idea. All things mentioned here are analogously transferable for the "POUs" view. First of all, we create our own implementation of the _3S.CODESYS.Core.Views.IViewFactory interface.
[TypeGuid("{33BE776C-385E-4550-9B14-464DC9BCAAA9}")] public class MyDeviceNavigatorFactory : IViewFactory {}
In the constructor, we create an instance of the original implementation. Therefore, we need the type GUID of the implementing class, but thanks to the type list file that is shipped with the Automation Platform SDK, this GUID can be quickly found.
_originalViewFactory = ComponentManager.Singleton.CreateInstance(GUID_DEVICENAVIGATORFACTORY) as IViewFactory;
With the member _originalViewFactory, most of the methods and properties of IViewFactory can be simply implemented by delegation, e.g. the Name property:
public string Name { get { return _originalViewFactory.Name; } }
The only method where we want to change the existing implementation is the Create method. First of all, we let the original implementation create the NavigatorControl, but then we assign our own callbacks to the control. Each of the two callbacks can have an additional transparent data object, which we use to transport the information of the original callbacks (as we do not want to reimplement the callbacks completely; rather we want the original implementations do the donkeywork and just change the things we additionally want). Just look at the code:
public IView Create()
{
INavigatorControl originalControl = OriginalNavigatorFactory.Create() as INavigatorControl;
if (originalControl != null)
{
originalControl.SetAppearance(
originalControl.SmallIcon,
originalControl.LargeIcon,
originalControl.Caption,
originalControl.DefaultDockingPosition,
originalControl.PossibleDockingPositions,
new NavigatorRenderingCallback(RenderingCallback),
new MyRenderingCallbackData(originalControl.RenderingCallback, originalControl.RenderingCallbackData));
originalControl.SetContent(
originalControl.ProjectHandle,
originalControl.StructuredViewGuid,
new NavigatorControlFilterCallback(FilterCallback),
new MyFilterCallbackData(originalControl.Callback, originalControl.CallbackData));
}
return originalControl;
}
...
class MyRenderingCallbackData
{
internal readonly NavigatorRenderingCallback OriginalCallback;
internal readonly object OriginalCallbackData;
internal MyRenderingCallbackData(NavigatorRenderingCallback originalCallback, object originalCallbackData)
{
OriginalCallback = originalCallback;
OriginalCallbackData = originalCallbackData;
}
}
class MyFilterCallbackData
{
internal readonly NavigatorControlFilterCallback OriginalCallback;
internal readonly object OriginalCallbackData;
internal MyFilterCallbackData(NavigatorControlFilterCallback originalCallback, object originalCallbackData)
{
OriginalCallback = originalCallback;
OriginalCallbackData = originalCallbackData;
}
}
You see, we inherit most of the original, but we put in our own callback methods. As callback data, we provide the original callback method and data object so that we can delegate in our delegate, like so:
internal static void RenderingCallback(
INavigatorControl control,
IMetaObjectStub mos,
object data,
ref string stText,
ref Color foreColor,
ref Color backColor,
ref FontStyle fontStyle,
ref Icon[] decoratingIcons)
{
MyRenderingCallbackData myData = data as MyRenderingCallbackData;
Debug.Assert(myData != null);
myData.OriginalCallback(control, mos, myData.OriginalCallbackData, ref stText, ref foreColor, ref backColor, ref fontStyle, ref decoratingIcons);
}
internal static bool FilterCallback(INavigatorControl control, IMetaObjectStub mos, object data)
{
MyFilterCallbackData myData = data as MyFilterCallbackData;
Debug.Assert(myData != null);
return myData.OriginalCallback(control, mos, myData.OriginalCallbackData);
}
So far, we have got an implementation of a navigator that does exactly the same things like the existing implementation in CoDeSys, but with two points in our very own code where we can influence the appearance of each node to any extend we like. (Okay, to be honest, the icon of an object cannot be changed, but it can be decorated with any number of additional icons, but anything else can be changed or supplemented at will.)
The second part of our solution deals with the necessity to override the existing view commands so that they are redirected to our implementation. The concept of command overriding is basically covered in another article (Basics: Overriding Standard Commands), so let's skip this stuff here. It is implemented in the sample solution, and it might be pretty self-explanatory.
The sample implementation appends the object GUID of all objects to the node text, highlights all POU objects with a yellow background, and hides all objects which name starts with "hide". Please make sure that you uninstall that plug-in again at a later time, our you'll never see objects that start with "hide" any more!
11 KB