From 9dddf67c18b450bbc954bd2a2c6d912dbb9d8ed0 Mon Sep 17 00:00:00 2001 From: Artur Signell Date: Wed, 25 Nov 2009 16:13:15 +0000 Subject: [PATCH] Test case and fix for #3605 - Concurrent modification exceptions in EventRouter svn changeset:10031/svn branch:6.2 --- src/com/vaadin/event/EventRouter.java | 12 ++++-- .../vaadin/tests/server/TestEventRouter.java | 37 +++++++++++++++++++ 2 files changed, 45 insertions(+), 4 deletions(-) create mode 100644 tests/src/com/vaadin/tests/server/TestEventRouter.java diff --git a/src/com/vaadin/event/EventRouter.java b/src/com/vaadin/event/EventRouter.java index ca98c5e799..76673d65f0 100644 --- a/src/com/vaadin/event/EventRouter.java +++ b/src/com/vaadin/event/EventRouter.java @@ -138,13 +138,17 @@ public class EventRouter implements MethodEventSource { public void fireEvent(EventObject event) { // It is not necessary to send any events if there are no listeners if (listenerList != null) { + + // Make a copy of the listener list to allow listeners to be added + // inside listener methods. Fixes #3605. + // Send the event to all listeners. The listeners themselves // will filter out unwanted events. - - final Iterator i = listenerList.iterator(); - while (i.hasNext()) { - ((ListenerMethod) i.next()).receiveEvent(event); + final Object[] listeners = listenerList.toArray(); + for (int i = 0; i < listeners.length; i++) { + ((ListenerMethod) listeners[i]).receiveEvent(event); } + } } } diff --git a/tests/src/com/vaadin/tests/server/TestEventRouter.java b/tests/src/com/vaadin/tests/server/TestEventRouter.java new file mode 100644 index 0000000000..dc388a66ac --- /dev/null +++ b/tests/src/com/vaadin/tests/server/TestEventRouter.java @@ -0,0 +1,37 @@ +package com.vaadin.tests.server; + +import junit.framework.TestCase; + +import com.vaadin.data.Property.ValueChangeEvent; +import com.vaadin.data.Property.ValueChangeListener; +import com.vaadin.ui.TextField; + +public class TestEventRouter extends TestCase { + + int innerListenerCalls = 0; + + public void testAddInEventListener() { + final TextField tf = new TextField(); + + final ValueChangeListener outer = new ValueChangeListener() { + + public void valueChange(ValueChangeEvent event) { + ValueChangeListener inner = new ValueChangeListener() { + + public void valueChange(ValueChangeEvent event) { + innerListenerCalls++; + System.out.println("The inner listener was called"); + } + }; + + tf.addListener(inner); + } + }; + + tf.addListener(outer); + tf.setValue("abc"); // No inner listener calls, adds one inner + tf.setValue("def"); // One inner listener call, adds one inner + tf.setValue("ghi"); // Two inner listener calls, adds one inner + assert (innerListenerCalls == 3); + } +} -- 2.39.5