Scenario:
- a TActionManager, a TAction, and a TButton (associated with this action)
- ActionManager constantly enables the Action in its OnUpdate event handler
- the code in the action event handler launches an external program using a ShellExecAndWait method (using Jedi Code Library JCL)
- requirement: the application should not allow to launch the application twice by clicking the button quickly another time
Problem:
- ShellExecAndWait does not block the application message loop, so the user can click while the external application is still open
- if he Action handler method disable the Action before the ShellExecAndWait call, the Update method will immediately re-enable it
So I could write like this
procedure TMyForm.OnMyAction(Sender: TObject);
begin
try
// notify Action Manager that the Action is temporarily disabled
SomeGlobalFlag := True;
// disable the action
(Sender as TAction).Enabled := False;
// do the call
ShellExecAndWait( ... );
finally
// enable the action
(Sender as TAction).Enabled := True;
// allow ActionManager to control the action again
SomeGlobalFlag := False;
end;
end;
Is there an easier way? As the title of this question says - could I block input for the execution of the external application?
It depends on how friendly you want your program to be to its users.
The approach shown in the question will suffice, but it could leave the user wondering why the button appears disabled. Your program could be more helpful if it left the button enabled, but changed its behavior when clicked. Instead of starting another copy of the program, it could notify the user that the previous program is still running, and perhaps even offer to set focus to that program.
The question title asks how to disable all the controls on the form. (Whether the form is modal is irrelevant; modality deals with disabling the parent form(s), not the modal form itself.) Mjn's answer sort of does that by suspending an action list. That won't disable controls that aren't associated with actions, and it won't disable controls that are associated with a different action list. It could also disable controls on other forms that are associated with the same action list.
Marck's answer implicitly disables all the controls, but will probably confuse the user since none of the controls will look disabled. That seems similar to the idea Sertac apparently mentioned in a comment, to disable the entire form.
To disable all controls on a form, and have them appear disabled, you can use a recursive function like this:
Use it like this:
Since we're directly modifying the
Enabled
properties of the controls, those properties will be severed from theEnabled
properties of any associated actions. That solves the immediate need, but has the unwanted side effect that further modifications to an action'sEnabled
property will not affect the controls on this form.Actions can be associated with multiple controls, and controls can reside on multiple forms. Since it's the action that's being updated, and not the controls directly, there isn't really a way to use actions to disable controls on just one form.
Now we come to the matter of whether disabling all the controls on a form is really the right solution to the problem that motivated this question. The question is a bit scattershot with regard to the goal and the proposed solution. The proposed solution is heavy-handed (disable everything on the form) for something that really only needs to prevent a single command from being invoked. And the command that shouldn't be invoked has nothing to do with the form; no matter how many controls are associated with the action on any number of forms, none should invoke the command. So, we should either disable the action, and implicitly disable whatever controls are associated with it, or we should modify the
OnExecute
event handler to detect re-entrance.The solution shown in the question is the way to disable the action. Set a flag to indicate that the action is executing, and clear it when execution completes. Check that flag in the
OnUpdate
event handler. There's no need to manually disable the action in theOnExecute
handler, though; actions already update themselves in theirExecute
methods. So we have this code:That requires multiple sections of code to cooperate regarding how to handle this action's execution. The action-updating code needs to know that the action needs to be able to temporarily disable itself.
For a more self-contained solution, we can leave the
OnUpdate
event alone and just keep the action enabled all the time. Instead, we'll keep track of re-entrance locally and notify the user:Mjn's answer calls the five lines of code for setting the state and managing the try-finally block "too much boilerplate." You can reduce it to one line with an interfaced object and a helper function:
Use it like this:
The function returns an interface reference, which the compiler stores in an implicitly declared temporary variable. At the end of the code, that variable is destroyed, and the stored interfaced object gets deallocated, returning the flag to its previous value.