Java AWT / Swing Обновление JPanel постоянно не работает - PullRequest
0 голосов
/ 17 апреля 2020

Я пытаюсь создать программу, которая заполняет JPanel с помощью GridLayout содержимым HashMap, содержащим ключи String для значений JButton. Поскольку размер HashMap может измениться, я не могу просто использовать setText() для каждой кнопки. До сих пор я звонил .removeAll(), чтобы удалить JPanel всех кнопок, а затем я oop через HashMap, чтобы заполнить JPanel. Затем я вызываю revalidate() на JPanel и repaint() на JFrame.

Текущий код:

public class GUI implements Runnable, ActionListener
{
    private ToDo td;

    JFrame frame;
    Thread t=null;
    int fontsize = 18;
    private Container contentPane;
    private JPanel topPane;
    private JButton main;
    private JButton add;
    private JButton settings;
    private JPanel centerPane;
    private JScrollPane centerScroll;
    private JPanel scrollable;

    private HashMap<String, JButton> items;

    public static void main(String[] args) {  
        new GUI();  
    }  

    public GUI(){
        td = new ToDo();

        frame = new JFrame();

        t = new Thread(this);  
        t.start();

        frame.setMinimumSize(new Dimension(480, 640));
        frame.setLayout(null);  
        frame.setVisible(true);

        contentPane = frame.getContentPane();
        contentPane.setLayout(new BorderLayout());

        topPane = new JPanel();
        topPane.setLayout(new GridLayout(1, 3));
        topPane.setPreferredSize(new Dimension(480, 40));

        main = new JButton("View Tasks");
        main.setFont(new Font("Sans Serif", Font.PLAIN, fontsize));
        add = new JButton("Add Task");
        add.setFont(new Font("Sans Serif", Font.PLAIN, fontsize));
        settings = new JButton("Settings");
        settings.setFont(new Font("Sans Serif", Font.PLAIN, fontsize));

        topPane.add(main);
        topPane.add(add);
        topPane.add(settings);

        contentPane.add(topPane, BorderLayout.NORTH);

        centerPane = new JPanel();
        centerPane.setPreferredSize(new Dimension(480, 600));

        items = new HashMap<>();

        HashMap<String, Assignment> assignments = td.getAssignments();

        scrollable = new JPanel();
        scrollable.setLayout(new GridLayout(assignments.size(), 1));
        centerScroll = new JScrollPane(scrollable);

        for(String key: assignments.keySet()){
            Assignment a = assignments.get(key);
            JButton button = new JButton(a.getTitle() + " | " + a.getDetails() + " | " + a.getClassification().getCls() + " | " + a.getStatus().getStatus());
            button.addActionListener(this);
            items.put(key, button);
            scrollable.add(button);
        }
        centerPane.add(centerScroll);        
        contentPane.add(centerPane, BorderLayout.CENTER);

        frame.pack();
        frame.setVisible(true);
    }
    public void update(int i){
        HashMap<String, Assignment> assignments = td.getAssignments();
        scrollable.removeAll();
        scrollable.setLayout(new GridLayout(assignments.size(), 1));
        for(String key: assignments.keySet()){
            Assignment a = assignments.get(key);
            JButton button = new JButton(Integer.toString(i));
            button.addActionListener(this);
            items.put(key, button);
            scrollable.add(button);
        }
        scrollable.revalidate();
        frame.repaint();
    }
    @Override
    public void run(){
        int counter = 0;
        try {

            while (true) {
                update(counter);
                t.sleep( 1000 );  // interval given in milliseconds 
                counter++;
            }  
        }  
        catch (Exception e) {
            System.out.println();
        }
    }
    @Override
    public void actionPerformed(ActionEvent e){
        for(String s: items.keySet()){
            if(items.get(s) == e.getSource()){
                EventMenu em = new EventMenu(td, s);
            }
        }
    }
}

Проблема в том, что кнопки не обновляются. Я ожидаю, что JPanel должен постоянно заполняться обновленными JButtons с другим текстом, но кажется, что программа зависает и не обновляется. Я попытался сделать более простой пример, который я изменил с здесь , с другими результатами:

public class DigitalWatch implements Runnable{  
    JFrame f;  
    JPanel p;
    Thread t=null;  
    int hours=0, minutes=0, seconds=0;  
    String timeString = "";  
    JButton b;  

    DigitalWatch(){  
        f=new JFrame();  
        p = new JPanel();
        t = new Thread(this);  
            t.start();  

        b=new JButton();  
            b.setBounds(100,100,100,50);  

        p.add(b);
        f.add(p);
        f.setSize(300,400);  
        f.setLayout(null);  
        f.setVisible(true);  
    }  

    public void run() {  
        try {  
            while (true) {  

                Calendar cal = Calendar.getInstance();  
                hours = cal.get( Calendar.HOUR_OF_DAY );  
                if ( hours > 12 ) hours -= 12;  
                minutes = cal.get( Calendar.MINUTE );  
                seconds = cal.get( Calendar.SECOND );  

                SimpleDateFormat formatter = new SimpleDateFormat("hh:mm:ss");  
                Date date = cal.getTime();  
                timeString = formatter.format( date );  

                p.removeAll();
                b=new JButton(timeString);  
                b.setBounds(100,100,100,50); 
                p.add(b);
                f.add(p);
                p.revalidate();
                f.repaint();

                //printTime();  

                t.sleep( 1000 );  // interval given in milliseconds  
            }  
        }  
          catch (Exception e) {
          System.out.println(e);
        }  
    }

    public static void main(String[] args) {  
        new DigitalWatch();  

    }  
} 

Этот фрагмент не может ничего рисовать, в отличие от первого, который по крайней мере dr aws объектов создано в конструкторе.

Как сделать обновление списка или сетки JPanel процедурно и в режиме реального времени и заполнить кнопки? Я знаю, что могу изменить текст каждой кнопки каждый раз, но количество кнопок может измениться в любое время.

Полный код здесь .

1 Ответ

2 голосов
/ 17 апреля 2020

вы нарушаете правило единственного потока Swing - вы не должны делать какие-либо вещи, связанные с пользовательским интерфейсом, вне потока отправки событий Swing.

Читайте об этом здесь и здесь .

Ниже приведен рабочий пример. Не уверен, почему они решили использовать кнопку, чтобы показать время, хотя. : -)

import java.awt.Dimension;
import java.awt.FlowLayout;
import java.time.LocalTime;
import java.time.format.DateTimeFormatter;
import java.time.format.FormatStyle;

import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.Timer;

public class DigitalWatch extends JFrame {

    private DateTimeFormatter formatter = DateTimeFormatter.ofLocalizedTime(FormatStyle.MEDIUM);

    public DigitalWatch() {
        JButton btn = new JButton(getCurrentTime());
        this.getContentPane().setLayout(new FlowLayout());
        this.getContentPane().add(btn);
        this.setPreferredSize(new Dimension(200, 150));
        this.pack();
        this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        this.setLocationRelativeTo(null); // center it on the screen
        new Timer(500, e -> btn.setText(getCurrentTime())).start();
    }

    private String getCurrentTime() {
        return formatter.format(LocalTime.now());
    }

    public static void main(String[] args) {
        new DigitalWatch().setVisible(true);
    }

}
...