How to redirect stdout to a text box in a UI?

15 views (last 30 days)
I'm using R2016b and would like to create a UI using the new app developer tool. I'm using the UI to wrap a rather long execution process and would like the user to be able to follow the messages the software is generating as it progress. Is it possible to redirect the standard output from the screen to a text box inside the UI and have it refresh as it goes?
Thanks,
Rachel
  1 Comment
Joe Yeh
Joe Yeh on 5 Oct 2016
By standard output, do you mean command window ? What are the functions that generate these outputs ?

Sign in to comment.

Accepted Answer

Walter Roberson
Walter Roberson on 5 Oct 2016
No.
You might be able to diary() into a file and while it is running, read the file. The prospects are uncertain on ms windows due to the way file locks are done.
The two mechanisms for grabbing a copy of MATLAB output are diary and evalc(), but evalc cannot be queried while the code is running. The diary approach would require timers unless you had the cooperation of the code.
If you have the parallel computing toolbox then another method of getting completed output is parfeval
  1 Comment
Rachel Soreq
Rachel Soreq on 6 Oct 2016
Moved: Rik on 16 Jan 2024
Well, at least now I won't spend time looking for a solution...
Thanks

Sign in to comment.

More Answers (1)

Delprat Sebastien
Delprat Sebastien on 16 Jan 2024
Edited: Delprat Sebastien on 16 Jan 2024
Here is a Piece of Code that works using Events
clearvars
close all
clc
% Create a process
pProcess = System.Diagnostics.Process();
% Set filename and any options here
pProcess.StartInfo.FileName = "your program.exe";
pProcess.StartInfo.Arguments = "some options";
pProcess.StartInfo.WorkingDirectory = pwd;
% Other options
pProcess.StartInfo.UseShellExecute = false; % Use shell or not to start process
pProcess.StartInfo.RedirectStandardOutput = true;
pProcess.StartInfo.RedirectStandardError = true;
pProcess.StartInfo.CreateNoWindow = true; % Set this to false if you want the process to create a windows
% Create event listener (there is a third one,"Exited" triggered when the process is ended)
% You may use any callback function in the form of myEventListener(src,event)
% Here, I simply use a disp command for simplicity
listener1 = addlistener(pProcess, 'OutputDataReceived', @(src, event) disp(['Ok' char(event.Data)]));
listener2 = addlistener(pProcess, 'ErrorDataReceived', @(src, event) disp(['err ' char(event.Data)]));
% Start the process
pProcess.Start();
% Read stdout & stderr
pProcess.BeginOutputReadLine()
pProcess.BeginErrorReadLine()
% Wait for exit
pProcess.WaitForExit();
You may change the event listener by any function of you choice:
function myEventListener(src,event)
% do something with the event here
end
This works well in a script because the process remains in the workspace once the program is ended.
If now, the whole code is enclosed within a function, the function will terminate just after the WaitForExit. But due to Matlab processing time, it is highly probable that some events remains unprocessed by the event listner (Matlab event may not have a priority high enough, or is simply too slow. I do not really know why it happens, but I experienced it). As a result, the function will terminate and all the objects (process, event listeners) will be deleted, and possibly many event can remain unprocessed (especially if the program run by the process terminates very quick).
An uggly fix consists in adding a pause after pProcess.WaitForExit()... It is uggly because there is no really whay to know how long it will takes. So, it can be better to do a poll.
So one way to deal with this is to "poll" the redirected outputs explicitely. The programs looks like this:
% Create a process
pProcess = System.Diagnostics.Process();
% Set filename and any options here
pProcess.StartInfo.FileName = "your program.exe";
pProcess.StartInfo.Arguments = "some options";
pProcess.StartInfo.WorkingDirectory = pwd;
pProcess.StartInfo.UseShellExecute = false;
pProcess.StartInfo.RedirectStandardOutput = true;
pProcess.StartInfo.RedirectStandardError = true;
pProcess.StartInfo.CreateNoWindow = false;
Start(pProcess);
% Retrieve stdout & stderr stream
stdErrOutput='non empty';
outdErrOutput='non empty';
% Loop while process is running or there are some outputs
out='non empty';
while ~pProcess.HasExited || ~isempty(outdErrOutput) || ~isempty(stdErrOutput)
% Read stderr
stdErrOutput = string(stderr.ReadLine);
if ~isempty(stdErrOutput)
% Do something with the line
disp(stdErrOutput);
% Sometimes adding a small pause leaves enough time to Matlab to
% refresh. Within an APP, use redraw
pause(0.01);
end
% Read strdout
outdErrOutput = string(stdout.ReadLine);
if ~isempty(outdErrOutput)
disp(outdErrOutput);
% Sometimes adding a small pause leaves enough time to Matlab to
% refresh. Within an APP, use redraw
pause(0.01);
end
end
With this two code, you should be able to find a solution to your problem.
  1 Comment
Rik
Rik on 16 Jan 2024
While it does look like useful code, I don't see how this answers the original question. Can you explain that?

Sign in to comment.

Community Treasure Hunt

Find the treasures in MATLAB Central and discover how the community can help you!

Start Hunting!