I\'m trying to use an action to control the visibility of a control. My code looks like this:
Pascal file
unit Unit1;
interface
us
Yes your diagnosis is correct as Rob already said and explained. A work around is not to use individual TAction's OnUpdate event handlers, but to use the TActionList's OnUpdate event handler.
procedure TForm1.ActionList1Update(Action: TBasicAction; var Handled: Boolean);
begin
Handled := True;
Action1.Visible := CheckBox1.Checked;
end;
You do need at least one visible control with a linked action for this to work, otherwise the ActionList's OnUpdate handler won't be called either.
Btw, when Actions and ActionLists were first introduced, Ray Konopka (Raize components) wrote a couple of articles on their implementation and gave very sound advice on how to use them. Ever since, I have adopted the practice of to use the OnExecute of each individual Action, but to use the OnUpdate of the ActionList. Also, the first thing I do in that handler is to set Handled to True so it won't be called more than necessary, and to only ever change an Action's Enabled property once in that handler so the GUI won't flicker as a result of turning it off and then on.
Article by Ray Konopka is "Effectively using Action Lists" : http://edn.embarcadero.com/article/27058 Ray used to have three articles on his own site, but on embarcadero there is but one, but it may well be the "combined" version (don't have the originals handy).
ActionUpdate event is not called when related controls are not visible. Try to explicity call ActionUpdate on Checkbox1's click event.
Your diagnosis is correct. Actions have worked that way since they were first introduced to Delphi.
I expect it's by design (an optimization to avoid excessive updating of the text and other visual aspects of invisible controls, probably), but that doesn't mean the design is good.
Any workaround I can imagine will involve your checkbox code directly manipulating the affected actions, which isn't very elegant since it shouldn't have to be aware of everything else it might affect — that's what the OnUpdate
event is supposed to do for you. When the checkbox gets checked, call Action1.Update
, or if that doesn't work, then Action1.Visible := True
.
You could also put your action-updating code in the TActionList.OnUpdate
event instead.