UPDATE:
Kolejne testy pokazały, że brak odświeżania lampki LED w GUI następuje nie tylko wtedy, gdy kursor znajdzie się poza oknem aplikacji, ale także, gdy zostanie przesunięty w obszar aplikacji i np. 5 sekund jest nieruchomy. Wtedy np. poruszenie kursora, przykrycie okna aplikacji innym oknem lub przeciągnięcie okna aplikacji w inne miejsce powoduje wznowienie odświeżania.
W momencie zatrzymania odświeżania, lampka LED zawsze pozostaje w stanie wygaszenia. Problem występuje gdy aplikację uruchamiam na Linuxie (Mint 21), a na Windowsie nie ma tego problemu.
Wydaje mi się, że Linux może w inny sposób dzielić sobie obsługę wątków, a odświeżanie niektórych elementów w GUI staje się czasem jakby zamrożone do czasu np. poruszenia kursora w obszarze okna aplikacji. Pakiety odbieram w innym wątku co sekundę i w tym samym momencie powinna zapalić się lampka LED na 0,3 sekundy. Z kolei tabela, która wyświetla odebrane w tym samym wątku pakiety (przez aktualizacje jej modelu), zawsze poprawnie się aktualizuje.
UPDATE 2:
Napisałem prosty programik, który używa wyżej opisanego mechanizmu do zapalania i gaszenia LED przy użyciu timera. Program ten zmienia tylko cyklicznie kolor LED z czerwonego na zielony co sekundę, na okres 300ms (powstaje cykliczne miganie lampki). Poniżej zamieszczam kod tego programu.
Wniosek:
Okazuje się, że w przypadku umieszczenia kursora poza obszarem okna aplikacji jest ten sam problem. Zrobiłem też test, w którym nie używam timera ledDelayTimer
do wyświetlania zielonej lampki przez 300ms, ale tylko cyklicznie za pomocą eventTimer
co sekundę na przemian ustawiam LED czerwony i zielony, i wtedy problem z odświeżaniem nigdy nie występuje.
public class GuiTimerIcon {
/**
* @param args the command line arguments
*/
public static void main(String[] args) {
new MainWindow();
}
}
import java.awt.event.ActionEvent;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.imageio.ImageIO;
import javax.swing.ImageIcon;
import javax.swing.Timer;
public class MainWindow extends javax.swing.JFrame {
private static Timer eventTimer;
private static Timer ledDelayTimer;
private static ImageIcon ENABLED_LED;
private static ImageIcon DISABLED_LED;
/**
* Creates new form MainWindow
*/
public MainWindow() {
initComponents();
this.setVisible(true);
try {
BufferedImage buffEnabled = ImageIO.read(new File("/home/greenLed.png"));
BufferedImage buffDisabled = ImageIO.read(new File("/home/redLed.png"));
ENABLED_LED = new ImageIcon(buffEnabled);
DISABLED_LED = new ImageIcon(buffDisabled);
} catch (IOException ex) {
Logger.getLogger(MainWindow.class.getName()).log(Level.SEVERE, null, ex);
}
initLedDelayTimer();
eventTimer = new Timer(1000, (ActionEvent ae) -> {
jLabel1.setIcon(ENABLED_LED);
ledDelayTimer.restart();
});
eventTimer.start();
}
private void initLedDelayTimer() {
ledDelayTimer = new Timer(300, (ActionEvent ae) -> {
jLabel1.setIcon(DISABLED_LED);
});
ledDelayTimer.setRepeats(false);
}
/**
* 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() {
jPanel1 = new javax.swing.JPanel();
jLabel1 = new javax.swing.JLabel();
setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
setMinimumSize(new java.awt.Dimension(400, 300));
javax.swing.GroupLayout jPanel1Layout = new javax.swing.GroupLayout(jPanel1);
jPanel1.setLayout(jPanel1Layout);
jPanel1Layout.setHorizontalGroup(
jPanel1Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
.addGroup(jPanel1Layout.createSequentialGroup()
.addGap(82, 82, 82)
.addComponent(jLabel1, javax.swing.GroupLayout.PREFERRED_SIZE, 220, javax.swing.GroupLayout.PREFERRED_SIZE)
.addContainerGap(98, Short.MAX_VALUE))
);
jPanel1Layout.setVerticalGroup(
jPanel1Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
.addGroup(jPanel1Layout.createSequentialGroup()
.addGap(28, 28, 28)
.addComponent(jLabel1, javax.swing.GroupLayout.PREFERRED_SIZE, 220, javax.swing.GroupLayout.PREFERRED_SIZE)
.addContainerGap(29, 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(jPanel1, javax.swing.GroupLayout.DEFAULT_SIZE, javax.swing.GroupLayout.DEFAULT_SIZE, Short.MAX_VALUE)
);
layout.setVerticalGroup(
layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
.addGroup(layout.createSequentialGroup()
.addContainerGap()
.addComponent(jPanel1, javax.swing.GroupLayout.DEFAULT_SIZE, javax.swing.GroupLayout.DEFAULT_SIZE, Short.MAX_VALUE)
.addContainerGap())
);
pack();
}// </editor-fold>
// Variables declaration - do not modify
private javax.swing.JLabel jLabel1;
private javax.swing.JPanel jPanel1;
// End of variables declaration
}
UPDATE 3:
Problem z odświeżaniem rozwiązuje użycie polecenia -Dsun.java2d.opengl=True
, ale wtedy większość kontrolek w oknie ma kolor czarny (chwilowo pomaga resize okna).
Może ma ktoś inny sposób na implementację zapalania lampki LED na określony czas?