Welcome to OGeek Q&A Community for programmer and developer-Open, Learning and Share
Welcome To Ask or Share your Answers For Others

Categories

0 votes
542 views
in Technique[技术] by (71.8m points)

swing - Weird white space in a java JFrame

Here is my problem. When i use the following code:

package xyz.lexium.giapb.ui;

 import java.io.*;
import java.awt.*;
import java.awt.event.*;
import javax.swing.*;

public class ConsoleWindow extends WindowAdapter implements WindowListener, ActionListener, Runnable {

private JFrame frame;
private JTextArea textArea;
private Thread reader;
private Thread reader2;
private boolean quit;

private final PipedInputStream pin = new PipedInputStream();
private final PipedInputStream pin2 = new PipedInputStream();

public ConsoleWindow() {
    frame = new JFrame("GIAPB - Console");
    Dimension dimension = Toolkit.getDefaultToolkit().getScreenSize();
    int x = (int) ((dimension.getWidth() - frame.getWidth()) / 3);
    int y = (int) ((dimension.getHeight() - frame.getHeight()) / 3);
    frame.setSize(x, y);
    textArea = new JTextArea();
    textArea.setEditable(false);
    JButton button = new JButton("clear");
    button.setBorder(null);
    button.setBackground(Color.BLACK);
    button.setForeground(Color.white);
    frame.getContentPane().setBackground(Color.BLACK);
    frame.getContentPane().setLayout(new BorderLayout());
    frame.getContentPane().add(new JScrollPane(textArea), BorderLayout.CENTER);
    frame.getContentPane().add(button, BorderLayout.SOUTH);
    frame.addWindowListener(this);
    textArea.setBackground(Color.BLACK);
    textArea.setForeground(Color.white);
    textArea.setBorder(null);
    frame.setVisible(true);
    button.addActionListener(this);

    try {
        PipedOutputStream pout = new PipedOutputStream(this.pin);
        System.setOut(new PrintStream(pout, true));
    } catch (java.io.IOException io) {
        textArea.append("Couldn't redirect STDOUT to this console
" + io.getMessage());
    } catch (SecurityException se) {
        textArea.append("Couldn't redirect STDOUT to this console
" + se.getMessage());
    }

    try {
        PipedOutputStream pout2 = new PipedOutputStream(this.pin2);
        System.setErr(new PrintStream(pout2, true));
    } catch (java.io.IOException io) {
        textArea.append("Couldn't redirect STDERR to this console
" + io.getMessage());
    } catch (SecurityException se) {
        textArea.append("Couldn't redirect STDERR to this console
" + se.getMessage());
    }

    quit = false; // signals the Threads that they should exit

    // Starting two seperate threads to read from the PipedInputStreams
    //
    reader = new Thread(this);
    reader.setDaemon(true);
    reader.start();
    //
    reader2 = new Thread(this);
    reader2.setDaemon(true);
    reader2.start();
}

public synchronized void windowClosed(WindowEvent evt) {
    quit = true;
    this.notifyAll(); // stop all threads
    try {
        reader.join(1000);
        pin.close();
    } catch (Exception e) {
    }
    try {
        reader2.join(1000);
        pin2.close();
    } catch (Exception e) {
    }
    System.exit(0);
}

public synchronized void windowClosing(WindowEvent evt) {
    frame.setVisible(false); // default behaviour of JFrame
    frame.dispose();
}

public synchronized void actionPerformed(ActionEvent evt) {
    textArea.setText("");
}

public synchronized void run() {
    try {
        while (Thread.currentThread() == reader) {
            try {
                this.wait(100);
            } catch (InterruptedException ie) {
            }
            if (pin.available() != 0) {
                String input = this.readLine(pin);
                textArea.append(input);
            }
            if (quit)
                return;
        }

        while (Thread.currentThread() == reader2) {
            try {
                this.wait(100);
            } catch (InterruptedException ie) {
            }
            if (pin2.available() != 0) {
                String input = this.readLine(pin2);
                textArea.append(input);
            }
            if (quit)
                return;
        }
    } catch (Exception e) {
        textArea.append("
Console reports an Internal error.");
        textArea.append("The error is: " + e);
    }

}

public synchronized String readLine(PipedInputStream in) throws IOException {
    String input = "";
    do {
        int available = in.available();
        if (available == 0)
            break;
        byte b[] = new byte[available];
        in.read(b);
        input = input + new String(b, 0, b.length);
    } while (!input.endsWith("
") && !input.endsWith("
") && !quit);
    return input;
}
}

It makes my console, but adds a white line between the text area and the border. How do i remove this

See Question&Answers more detail:os

与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…
Welcome To Ask or Share your Answers For Others

1 Reply

0 votes
by (71.8m points)

You're forgetting that the JScrollPane has a border.

frame.getContentPane().add(new JScrollPane(textArea), BorderLayout.CENTER);

Set it to null.

JScrollPane scrollPane = new JScrollPane(textArea);
scrollPane.setBorder(null);
frame.getContentPane().add(scrollPane, BorderLayout.CENTER);

Another issue: this isn't Swing thread-safe:

public synchronized void run() {
    try {
        while (Thread.currentThread() == reader) {
            try {
                this.wait(100);
            } catch (InterruptedException ie) {
            }
            if (pin.available() != 0) {
                String input = this.readLine(pin);
                textArea.append(input);  // **************
            }
            if (quit)
                return;
        }

        while (Thread.currentThread() == reader2) {
            try {
                this.wait(100);
            } catch (InterruptedException ie) {
            }
            if (pin2.available() != 0) {
                String input = this.readLine(pin2);
                textArea.append(input);  // **************
            }
            if (quit)
                return;
        }
    } catch (Exception e) {
        textArea.append("
Console reports an Internal error.");  // **************
        textArea.append("The error is: " + e);  // **************
    }

}

You're making textArea.append(...) calls off of the Swing event thread, and this can cause hard to debug intermittent exceptions to be thrown. Be sure to only append to this text component on the event dispatch thread.


与恶龙缠斗过久,自身亦成为恶龙;凝视深渊过久,深渊将回以凝视…
OGeek|极客中国-欢迎来到极客的世界,一个免费开放的程序员编程交流平台!开放,进步,分享!让技术改变生活,让极客改变未来! Welcome to OGeek Q&A Community for programmer and developer-Open, Learning and Share
Click Here to Ask a Question

...