tags:

views:

28

answers:

2

**Hello, I'm trying to create an archiver in java. This means that I am constantly reading and writing from streams. I want to be able to update a JProgressBar to show how much I have written. My code currently updates the progress bar after each entry in the archive. My variables and everything say that the code is updated, except its not showing on the screen. After each entry into the archive I call the revalidate() method on the JPanel that contains the progress bar and text, but nothing happens on the screen. The screen doesn't update until the output stream that is writing the archive has completely finished writing. Then, I see the full progress bar with "24 out of 24 files written" in the progress text.

Can anyone tell me how I can get the graphics to update before my code finished executing? Thanx

Here is my code:

WritingBox.java:

package org.apache.commons.compress;
import javax.swing.JDialog;
public class WritingBox extends javax.swing.JDialog {

/** Creates new form WritingBox */
public WritingBox(java.awt.Frame parent, boolean modal) {
    super(parent, modal);
    initComponents();
    theprogressbar.setMaximum(1);
    theprogressbar.setMinimum(0);
    this.setLocation(parent.getLocationOnScreen().x+(parent.getWidth()-this.getWidth())/2, parent.getLocationOnScreen().y+(parent.getHeight()-this.getHeight())/2);
    this.setVisible(true);
}
public void progressListener(Progress p) {
    writtenxoutofx.setText("Written "+Integer.toString(p.doneSoFar)+" out of "+Integer.toString(p.total)+" files to the Archive");
    theprogressbar.setValue(p.doneSoFar/p.total);
    if (p.doneSoFar == p.total) this.setDefaultCloseOperation(JDialog.DISPOSE_ON_CLOSE);
    thecontainer.revalidate();
};
/** This method is called from within the constructor to
 * initialize the form.
 * WARNING: Do NOT modify this code. The content of this method is
 * always regenerated by the Form Editor.
 */
@SuppressWarnings("unchecked")
// <editor-fold defaultstate="collapsed" desc="Generated Code">
private void initComponents() {

    thecontainer = new javax.swing.JPanel();
    jLabel1 = new javax.swing.JLabel();
    writtenxoutofx = new javax.swing.JLabel();
    theprogressbar = new javax.swing.JProgressBar();

    setDefaultCloseOperation(javax.swing.WindowConstants.DO_NOTHING_ON_CLOSE);
    setName("Form"); // NOI18N

    thecontainer.setName("thecontainer"); // NOI18N

    org.jdesktop.application.ResourceMap resourceMap = org.jdesktop.application.Application.getInstance(targzipmanager.TarGzipManagerApp.class).getContext().getResourceMap(WritingBox.class);
    jLabel1.setText(resourceMap.getString("jLabel1.text")); // NOI18N
    jLabel1.setName("jLabel1"); // NOI18N

    writtenxoutofx.setText(resourceMap.getString("writtenxoutofx.text")); // NOI18N
    writtenxoutofx.setName("writtenxoutofx"); // NOI18N

    theprogressbar.setName("theprogressbar"); // NOI18N

    javax.swing.GroupLayout thecontainerLayout = new javax.swing.GroupLayout(thecontainer);
    thecontainer.setLayout(thecontainerLayout);
    thecontainerLayout.setHorizontalGroup(
        thecontainerLayout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
        .addGroup(thecontainerLayout.createSequentialGroup()
            .addContainerGap()
            .addGroup(thecontainerLayout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
                .addComponent(theprogressbar, javax.swing.GroupLayout.DEFAULT_SIZE, 307, Short.MAX_VALUE)
                .addComponent(jLabel1)
                .addComponent(writtenxoutofx))
            .addContainerGap())
    );
    thecontainerLayout.setVerticalGroup(
        thecontainerLayout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
        .addGroup(thecontainerLayout.createSequentialGroup()
            .addContainerGap()
            .addComponent(jLabel1)
            .addPreferredGap(javax.swing.LayoutStyle.ComponentPlacement.RELATED)
            .addComponent(writtenxoutofx)
            .addPreferredGap(javax.swing.LayoutStyle.ComponentPlacement.RELATED)
            .addComponent(theprogressbar, javax.swing.GroupLayout.PREFERRED_SIZE, javax.swing.GroupLayout.DEFAULT_SIZE, javax.swing.GroupLayout.PREFERRED_SIZE)
            .addContainerGap(javax.swing.GroupLayout.DEFAULT_SIZE, Short.MAX_VALUE))
    );

    javax.swing.GroupLayout layout = new javax.swing.GroupLayout(getContentPane());
    getContentPane().setLayout(layout);
    layout.setHorizontalGroup(
        layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
        .addComponent(thecontainer, javax.swing.GroupLayout.DEFAULT_SIZE, javax.swing.GroupLayout.DEFAULT_SIZE, Short.MAX_VALUE)
    );
    layout.setVerticalGroup(
        layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
        .addComponent(thecontainer, javax.swing.GroupLayout.DEFAULT_SIZE, javax.swing.GroupLayout.DEFAULT_SIZE, Short.MAX_VALUE)
    );

    pack();
}// </editor-fold>

// Variables declaration - do not modify
private javax.swing.JLabel jLabel1;
private javax.swing.JPanel thecontainer;
private javax.swing.JProgressBar theprogressbar;
private javax.swing.JLabel writtenxoutofx;
// End of variables declaration

}

Progress.java

package org.apache.commons.compress;

public class Progress { //This is basically a Struct
public int total;
public int doneSoFar;
private Progress (){

}
public Progress(int d, int t){
    total = t;
    doneSoFar = t;
}
}

TarArchive.java: doSave method

public void doSave(OutputStream output) throws ArchiveException {
        // Stream initializing
        //BufferedInputStream origin = null;

        //out.setMethod(ZipOutputStream.DEFLATED);
        //byte data[] = new byte[BUFFER];

        // get a list of files from current directory
        // less than one file leads to an exception
        Iterator iterator = this.getEntryIterator();
        if(!iterator.hasNext()) {
              throw new ArchiveException("There must be at least one file to be pack.");
        }
        JFrame jf = targzipmanager.TarGzipManagerApp.getApplication().getMainFrame();
        WritingBox wb = new WritingBox(jf, false);
        TarOutputStream out = null;
        int x = 0;
        try {
              out = new TarOutputStream(new BufferedOutputStream(output));
              while(iterator.hasNext()) {
                    x++;
                    ArchiveEntry archiveEntry = (ArchiveEntry)iterator.next();
                    InputStream fInputStream = archiveEntry.getStream();

                    TarEntry entry = new TarEntry(archiveEntry.getName());
                    entry.setModTime( 0 );
                entry.setSize( fInputStream.available() );
                entry.setUserID( 0 );
                entry.setGroupID( 0 );
                entry.setUserName( "avalon" );
                entry.setGroupName( "excalibur" );
                entry.setMode( 0100000 );
                out.putNextEntry( entry );
                out.copyEntryContents( fInputStream );
                out.closeEntry();
                wb.progressListener(new Progress(x, entries.size()));
              }
        } catch (IOException e) {
              throw new ArchiveException("Creation of this archive failed cause of IOExceptions.", e);
        } finally {
              try {
                    out.close();
              } catch (IOException e1) {
                    throw new ArchiveException("Creation of this archive failed cause of IOExceptions.", e1);
              }
        }
  }

Can anyone help me? Even though I'm calling revalidate() after each entry, it only updates when the code stops and it starts waiting. **

+1  A: 

Sounds like you are doing the file I/O on the AWT Event Dispatch Thread (EDT). This blocks any repaints or event handling. Run the I/O on a different thread, and update the GUI through java.awt.EventQueue.invokeLater.

Tom Hawtin - tackline
How would I do this?
Leo Izen
@user411316: For an example, see "Continuations as objects": http://en.wikipedia.org/wiki/Continuation_passing_style#Continuations_as_objects
trashgod
+1  A: 

Read the section from the Swing tutorial on Concurrency for more information, including a solution using a Swing Worker.

Also, look at the section on "How to Use Progress Bars" for another working example.

camickr