javaswingalways-on-top

Frame always on top of my program only


I'm trying to create a kind of toolbar in an undecorated alwaysOnTop frame. Thus, I want my frame to be on top of my main frame, but not on top of frames from other programs. I tried this code :

public class Test {
    private static JFrame mainFrame;

    /**
     * @param args the command line arguments
     */
    public static void main(String[] args) {
        SwingUtilities.invokeLater(new Runnable() {
            @Override
            public void run() {
                mainFrame = new JFrame("test");
                mainFrame.setSize(800,600);
                mainFrame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
                mainFrame.setVisible(true);

                A a = new A();
            }
        });
    }

    public static class A extends JDialog {

        public A() {
            super(mainFrame);
            setAlwaysOnTop(true);
            setFocusable(false);
            setSize(80,60);
            setVisible(true);
        }
    }
}

But despite the use of JDialog and precising the owner, the frame stay on top of other applications (at least with Ubuntu. Maybe the result is different with other OS ?)

EDIT : Ok, I tried this code for my dialog :

public static class A extends JDialog {

    public A(String name) {
        super(mainFrame, name);
        setAlwaysOnTop(true);
        setFocusable(false);
        setSize(80, 60);
        setVisible(true);

        mainFrame.addWindowListener(new WindowAdapter() {
            @Override
            public void windowActivated(WindowEvent e) {
                A.this.setAlwaysOnTop(true);
            }

            @Override
            public void windowDeactivated(WindowEvent e) {
                // A.this.setAlwaysOnTop(false);
                A.this.toBack();
            }
        });
    }
}

The issue now is that when the main window loose focus, the dialog steals the focus back and I don't understand why. For instance, I run my app, I try to switch to Firefox, Firefox appears and covers the mainFrame, but the A dialog gets the focus and stays in the screen. Now, if I select Firefox again, the dialog will at last correctly disappear. Could you explain me why the dialog gets the focus?

Thanks


Solution

  • Ok, I found a solution (don't know if it is THE solution, but it's working, so...)

    I discovered setFocusableWindowState(), that is perfect for toolbars. By the way, I don't know if my previous setFocusable(false) had any effect.

    The next issue was that the focus gets very weird behaviour with this code : If I switch from MyApp to Firefox, here is what happens :

    focus : MyApp -> Firefox
    execution of MyDialog.toFront()
    focus : Firefox -> MyDialog
    MyDialog not focusable !
    focus : MyDialog -> MyApp !!!
    

    result : nothing changed !

    So I finally got the tricks : just after MyDialog.toFront(), you give back the focus to the previous owner. And the only way I found to do this with no error was : mainFrame.toBack()

    Final code :

    public class Test {
        private static JFrame mainFrame;
    
        public static void main(String[] args) {
            SwingUtilities.invokeLater(new Runnable() {
                @Override
                public void run() {
                    mainFrame = new JFrame("test");
                    mainFrame.setSize(800,600);
                    mainFrame.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
                    mainFrame.setVisible(true);
    
                    A a = new A();
                }
            });
        }
    
        public static class A extends JDialog {
    
            public A() {
                super(mainFrame);
                setAlwaysOnTop(true);
                setFocusableWindowState(false);
                setSize(80,60);
                setVisible(true);
    
                mainFrame.addWindowListener(new WindowAdapter() {
                    @Override
                    public void windowActivated(WindowEvent e) {
                        A.this.setAlwaysOnTop(true);
                        A.this.toFront();
                    }
                    @Override
                    public void windowDeactivated(WindowEvent e) {
                        A.this.setAlwaysOnTop(false);
                    }
                });
            }
        }
    }