Reputation: 245
I use SwingWorker to make a progress bar with Java Swing APIs.
I have a class that extends SwingWorker
class Swinger extends SwingWorker {
private ClassAnalyzer classAnalyzer;
public Swinger(ClassAnalyzer classAnalyzer){
this.classAnalyzer = classAnalyzer;
}
@Override
public Void doInBackground() throws InterruptedException {
try
{
int progress = 0;
while (progress < 100) {
// at this point I make certain elaboration on classAnalyzer
progress++;
//Call the process method to update the GUI
publish(progress);
}
}
catch(InterruptedException e)
{
}
return null;
}
@Override
protected void process(List chunks) {
for (Integer chunk : chunks) {
progressBar.setValue(chunk);
//if the switchtype checkbox is selected then
//change the progressbar to a determined type
//once the progress has reached 50
if (chunk > 49)
{
if (switchType.isEnabled() && switchType.isSelected())
{
progressBar.setStringPainted(true);
}
}
}
}
}
and a second class (I'm writing a piece of this)
public Tester()
{
JFrame guiFrame = new JFrame();
//make sure the program exits when the frame closes
guiFrame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
guiFrame.setTitle("Creating a Table Example");
guiFrame.setSize(700,200);
//This will center the JFrame in the middle of the screen
guiFrame.setLocationRelativeTo(null);
goButton = new JButton("Go");
goButton.setActionCommand("Go");
goButton.addActionListener(new ActionListener()
{
//When the button is clicked the SwingWorker class is executed and
//the button is disabled
@Override
public void actionPerformed(ActionEvent event)
{
progressBar.setStringPainted(progressType.isSelected());
ClassAnalyzer c = new ClassAnalyzer();
Swinger task = new Swinger(c);
task.execute();
int methods = c.getNumberOfMethods();
if(methods == 0){
JOptionPane.showMessageDialogo(null, "methods not found");
}
goButton.setEnabled(false);
}
});
}
When I launch in a Tester the second class, the message "methods not found" is displayed before progress bar appears, while I would like message appears in case after. What to do?
Upvotes: 0
Views: 3490
Reputation: 347204
task.execute()
will launch a background (in which the doInBackground
method will be called from) and the program will continue executing.
task.execute()
is not a blocking method, this is the reason for using it, so you don't block the Event Dispatching Thread
You can monitor the state of the SwingWorker
with a PropertyChangeListener
final ClassAnalyzer c = new ClassAnalyzer();
Swinger task = new Swinger(c);
task.addPropertyChangeListener(new PropertyChangeListener() {
@Override
public void propertyChange(PropertyChangeEvent evt) {
if (evt.getPropertyName().equals("state") && evt.getNewValue().equals(SwingWorker.StateValue.DONE)) {
int methods = c.getNumberOfMethods();
if(methods == 0){
JOptionPane.showMessageDialogo(null, "methods not found");
}
}
}
});
task.execute();
Upvotes: 3
Reputation: 3123
The SwingWorker
class also defines a done
method where you can define what to do once your task finishes.
class Swinger extends SwingWorker {
// all the rest of your code ^^
@Override
protected void done() {
if(this.classAnalyzer.getNumberOfMethods() == 0)
JOptionPane.showMessageDialog(null, "methods not found");
}
}
You don't have to worry about threading issues since this method is called on the EDT too.
Upvotes: 3