 jlokanis
		
			jlokanis
		
		
		 
		
		
		
		
		
	
			05-02-2013 05:13 PM
I was wondering if there was a better way to solve this problem.
When building a UI with AF, the examples I have seen appear to have an event loop send a message to self. That message is handled by the UI actor running in parallel with the event loop. The message ‘Do’ then sends other messages to other actors as needed.
The UI actor is the only thing that has access to state data for the UI so it makes sense to pass a message to the UI actor so it can update state data before passing the message up to the parent that performs most of the control logic. But this means that for every user action, there must be a minimum of two message classes created. One for the UI Actor and another to send from the UI actor to the parent controller.
I am wondering if there is a better way to do this so that we can skip the double messages. Has anyone come up with a cleaner solution?
 Daklu
		
			Daklu
		
		
		
		
		
		
		
		
	
			05-02-2013 10:27 PM
Without a definition of "cleaner solution," here are some things you could do:
The nature of command pattern messaging means every message requires a unique class. As soon as you diverge from that you're giving up some of the safety built into the AF. What are you willing to give up in exchange for fewer message classes?
 drjdpowell
		
			drjdpowell
		
		
		 
		
		
		
		
		
	
			05-03-2013 04:23 AM
If Daklu hadn't beaten me, I was going to suggest his third option. Move the message handling into the event loop. -- James
BTW added later: this is one of the reasons I suggested an alternate class structure for queues; so that you could have an alternate actor design based on a User Event as message receiving mechanism, with actors of different types still able to communicate with each other.
Message was edited by: drjdpowell
 rpodsim
		
			rpodsim
		
		
		
		
		
		
		
		
	
			05-03-2013 08:21 AM
How about using a Time-Delayed Repeating message that contains the event structure with a 0 timeout?
Or have the Event message send itself with a short timeout?
You just have to register all control events into dynamic events.
 D_Hooks
		
			D_Hooks
		
		
		 
		
		
		
		
		
	
			05-03-2013 05:46 PM
In my experience, I think that maintaining state data in the UI event loop makes a lot of sense. There could be numerous events that are handled within that loop which require knowledge of the state data, but do not involve any incoming our outgoing messaging. Outgoing class-based messages can be sent directly from the event loop to the parent Controller Actor.
So, at this point you have the UI Actor receiving incoming class-based messages in the actor loop, and forwarding them along to the event loop using user events. The event loop is maintaining its own state data and sending class-based messages directly to the Controller Actor as needed. If you are being thorough about the communication between the UI Actor's actor loop and event loop, maybe you've created an interface class which encapsulates the sending of user events.
At this point, why bother creating a UI Actor at all? You have already created an interface class to send event-based messages to the event loop, and the event loop can easily send class-based messages back to the Controller Actor as long as it has a copy of the correct enqueuer. Why not take out the middle man of the UI Actor actor loop, and simply let the Controller Actor send outgoing messages using the UI interface class? One additional method could be added to the UI interface class to dynamically launch the event loop initially.
 Todd_Lesher
		
			Todd_Lesher
		
		
		
		
		
		
		
		
	
			05-03-2013 06:15 PM
Make the UI Actor a child of the business actor and only override methods that will update the UI. Of course, swapping UIs is restricted to launch time - but maybe if you stop the actor, pass the final (stale) state to another launch ... 
05-06-2013 12:43 PM
I gave some thought to moving the UI control logic from the incoming message Do.vi to the event loop. My main concerns with this is loading the event loop up with extra code that can make it less responsive. But, since it currently messages the UI Actor when an event occurs, it is still technically blocked until the UI actor is free. I also am concerned about having to implement error handling outside the Do.vi in the event loop and all the issues that will bring up.
One benefit of having the event loop message the UI actor is once you create the message for the action, you can map many different controls to that action easily. In my came many user actions can be done via toolbar, menu selection and right-click menu. My event loop can capture each of those and simply send the same message to the UI Actor, who can then process them appropriately and send a message to the Parent as needed.
But the primary concern is several of my UIs will have 10 or more user actions.  This results in 20 or more message classes.  And I expect to have 5-6 UIs in the application which means I am likely to have at least 100 message classes just for user action input alone.  And that does not include the message classes for sending data to the UIs! 
Another issue I am dealing with is trying to build a test framework for each UI.  Currently, where the UI Actor would send a message to the Parent I am instead sending that message to a test harness to display the data from the UI Actor. This means that all the Do.vi code for these messages is currently customized to the test harness and not what the parent in the real application will do with the data.  This is fine for dev work as I can edit those message classes later when I implement the parent, but that will permanently break my test harness.  I would like to keep both functionalities in place for future development work.  The only idea I came up with was to subclass the parent and test harness to a common ancestor with abstract methods that the message classes would use and then have the test harness and the normal parent implement the concrete methods.  But the problem I run into there is the parent will need to deal with multiple different UI actors and I cannot have it inherit from multiple different ancestors to give it access to the all the possible methods for those UI messages. 
As Dave said, maybe this is all just the price of command pattern messaging systems.
 drjdpowell
		
			drjdpowell
		
		
		 
		
		
		
		
		
	
			05-06-2013 02:20 PM
jlokanis wrote:
I gave some thought to moving the UI control logic from the incoming message Do.vi to the event loop.
Don't know about the others, but I at least was suggesting you actually call "Do" on your incoming messages inside the event loop. Bypass the inner "Actor Core" parent completely. Don't know if you do that without modifying the framework, though. You'd need a helper loop to dequeue messages and forward them via user event.
One benefit of having the event loop message the UI actor is once you create the message for the action, you can map many different controls to that action easily. In my came many user actions can be done via toolbar, menu selection and right-click menu. My event loop can capture each of those and simply send the same message to the UI Actor, who can then process them appropriately and send a message to the Parent as needed.
With the "Actor" object in the event loop, you can just call a method on it directly.
05-06-2013 06:02 PM
drjdpowell wrote:
jlokanis wrote:
I gave some thought to moving the UI control logic from the incoming message Do.vi to the event loop.
Don't know about the others, but I at least was suggesting you actually call "Do" on your incoming messages inside the event loop. Bypass the inner "Actor Core" parent completely. Don't know if you do that without modifying the framework, though. You'd need a helper loop to dequeue messages and forward them via user event.
That's pretty much a no-go in the AF since the actor data is all up in the parent's Actor Core and the framework rules it out from being anywhere else -- no one else but that loop can dequeue messages from the queue.
05-06-2013 06:08 PM
jlokanis wrote:
But this means that for every user action, there must be a minimum of two message classes created. One for the UI Actor and another to send from the UI actor to the parent controller.
I am wondering if there is a better way to do this so that we can skip the double messages. Has anyone come up with a cleaner solution?
You just described the cleaner solution. 😉 The dirty ones involve variants and flattened strings. I think Daklu's list is pretty comprehensive, except for Todd_Lesher's solution, which actually does cut your number of message classes in half. It's my preferred solution, but it only really applies in applications where the UI and the model have the same lifetime.