Hi Anthony, 

Please, review the patch and test.

Best Regards, 

-- Jose Luis Martin.




El mié, 22-05-2013 a las 13:52 +0400, Anthony Petrov escribió:
> 
> 
> If you and other AWT folks agree with that, could you prepare a patch 
> for this change and a jtreg-based regression test (it should be put 
> somewhere in test/java/awt/Window/, I think)?
> 
> --
> best regards,
> Anthony
> 
> 

diff --git a/src/share/classes/java/awt/Window.java b/src/share/classes/java/awt/Window.java
--- a/src/share/classes/java/awt/Window.java
+++ b/src/share/classes/java/awt/Window.java
@@ -1200,6 +1200,7 @@
             }
         }
     }
+    	boolean fireWindowClosedEvent = isDisplayable();
         DisposeAction action = new DisposeAction();
         if (EventQueue.isDispatchThread()) {
             action.run();
@@ -1220,7 +1221,9 @@
         // Execute outside the Runnable because postWindowEvent is
         // synchronized on (this). We don't need to synchronize the call
         // on the EventQueue anyways.
-        postWindowEvent(WindowEvent.WINDOW_CLOSED);
+        if (fireWindowClosedEvent) {
+        	postWindowEvent(WindowEvent.WINDOW_CLOSED);
+        }
     }
 
     /*
diff --git a/test/java/awt/Window/WindowClosedEvents/WindowClosedEventOnDispose.java b/test/java/awt/Window/WindowClosedEvents/WindowClosedEventOnDispose.java
new file mode 100644
--- /dev/null
+++ b/test/java/awt/Window/WindowClosedEvents/WindowClosedEventOnDispose.java
@@ -0,0 +1,202 @@
+/*
+ * Copyright (c) 2006, Oracle and/or its affiliates. All rights reserved.
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
+ *
+ * This code is free software; you can redistribute it and/or modify it
+ * under the terms of the GNU General Public License version 2 only, as
+ * published by the Free Software Foundation.
+ *
+ * This code is distributed in the hope that it will be useful, but WITHOUT
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+ * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
+ * version 2 for more details (a copy is included in the LICENSE file that
+ * accompanied this code).
+ *
+ * You should have received a copy of the GNU General Public License version
+ * 2 along with this work; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
+ * or visit www.oracle.com if you need additional information or have any
+ * questions.
+ */
+
+/*
+  @test
+  @summary DisposeAction multiplies the WINDOW_CLOSED event.
+  @author [email protected]
+  @run main WindowClosedEventOnDispose
+*/
+
+
+import java.awt.Toolkit;
+import java.awt.event.WindowAdapter;
+import java.awt.event.WindowEvent;
+
+import javax.swing.JDialog;
+import javax.swing.JFrame;
+import javax.swing.SwingUtilities;
+
+/**
+ * WindowClosedEventOnDispose.java
+ * Summary: tests that Window don't multiplies the WINDOW_CLOSED event
+ * on dispose.
+ * Test fails if fire more events that expected;
+ */
+public class WindowClosedEventOnDispose {
+	
+	private static int N_LOOPS = 5;
+	private static int N_DIALOGS = 2;
+	
+    public static void main(String args[]) throws Exception {
+    	tesWithFrame();
+    	testWithoutFrame();
+    	testHidenChildDispose();
+    	testHidenWindowDispose();
+    }
+    
+	/**
+	 * Test WINDOW_CLOSED event received by a dialog
+	 * that have a owner window.
+	 * @throws Exception
+	 */
+	public static void tesWithFrame() throws Exception {
+	    doTest(true);
+	}
+	
+	/**
+	 * Test WINDOW_CLOSED event received by a dialog
+	 * that don't have a owner window.
+	 * @throws Exception
+	 */
+	public static void testWithoutFrame() throws Exception  {
+		System.out.println("Run without owner Frame");
+		doTest(false);
+	}
+	
+	/**
+	 * Test if a dialog that has never been shown fire 
+	 * the WINDOW_CLOSED event on parent dispose().
+	 * @throws Exception
+	 */
+	public static void testHidenChildDispose() throws Exception {
+		JFrame f = new JFrame();
+		JDialog dlg = new JDialog(f);
+		Listener l = new Listener();
+		dlg.addWindowListener(l);
+		f.dispose();
+		waitEvents();
+		
+		assertEquals(0, l.getCount());
+	}
+	
+	/**
+	 * Test if a dialog fire the WINDOW_CLOSED event 
+	 * on parent dispose().
+	 * @throws Exception
+	 */
+	public static void testVisibleChildParentDispose() throws Exception {
+		JFrame f = new JFrame();
+		JDialog dlg = new JDialog(f);
+		Listener l = new Listener();
+		dlg.addWindowListener(l);
+		dlg.setVisible(true);
+		f.dispose();
+		waitEvents();
+		
+		assertEquals(1, l.getCount());
+	}
+	
+	/**
+	 * Test if a Window that has never been shown fire the
+	 * WINDOW_CLOSED event on dispose()
+	 */
+	public static void testHidenWindowDispose() throws Exception {
+		JFrame f = new JFrame();
+		Listener l = new Listener();
+		f.addWindowListener(l);
+		f.dispose();
+		waitEvents();
+		
+		assertEquals(0, l.getCount());
+	}
+	
+	/**
+	 * Test if a JDialog receive the correct number 
+	 * of WINDOW_CLOSED_EVENT
+	 * @param useFrame true if use a owner frame
+	 * @throws Exception
+	 */
+	private static void doTest(final boolean useFrame) throws Exception {
+		final Listener l  = new Listener();
+		final JFrame f = new JFrame();
+
+		for (int i = 0; i < N_LOOPS; i++) {
+
+			SwingUtilities.invokeLater(new Runnable() {
+
+				public void run() {
+					JDialog[] dialogs = new JDialog[N_DIALOGS];
+					for (int i = 0; i < N_DIALOGS; i++) {
+						if (useFrame) {
+							dialogs[i]= new JDialog(f);
+						}
+						else {
+							dialogs[i] = new JDialog();
+						}
+						
+						dialogs[i].addWindowListener(l);
+						dialogs[i].setVisible(true);
+					}
+					
+					// Dipose all
+					for (JDialog d : dialogs)
+						d.dispose();
+					
+					f.dispose();
+				}
+			});
+		}
+
+		waitEvents();
+
+		assertEquals(N_DIALOGS * N_LOOPS, l.getCount());
+	}
+
+	private static void waitEvents() throws InterruptedException {
+		// Wait until events are dispatched
+		while (Toolkit.getDefaultToolkit().getSystemEventQueue().peekEvent() != null)
+			Thread.sleep(100);
+	}
+	
+	/**
+	 * @param expected the expected value
+	 * @param real the real value
+	 */
+	private static void assertEquals(int expected, int real) throws Exception {
+		if (expected != real) {
+			throw new Exception("Expected events: " + expected + " Received Events: " + real);
+		}
+	}
+	
+}
+
+/**
+ * Listener to count events
+ */
+class Listener extends WindowAdapter {
+
+	private volatile int count = 0;
+
+	public void windowClosed(WindowEvent e) {
+		count++;
+	}
+
+	public int getCount() {
+		return count;
+	}
+
+	public void setCount(int count) {
+		this.count = count;
+	}
+}

Reply via email to