summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--client/src/com/vaadin/client/ui/orderedlayout/Slot.java51
-rw-r--r--uitest/src/com/vaadin/tests/components/orderedlayout/VerticalLayoutFocusWithDOMChanges.java63
-rw-r--r--uitest/src/com/vaadin/tests/components/orderedlayout/VerticalLayoutFocusWithDOMChangesTest.java100
-rw-r--r--uitest/src/com/vaadin/tests/tb3/AbstractTB3Test.java16
4 files changed, 230 insertions, 0 deletions
diff --git a/client/src/com/vaadin/client/ui/orderedlayout/Slot.java b/client/src/com/vaadin/client/ui/orderedlayout/Slot.java
index 49b3661431..37a97f3399 100644
--- a/client/src/com/vaadin/client/ui/orderedlayout/Slot.java
+++ b/client/src/com/vaadin/client/ui/orderedlayout/Slot.java
@@ -19,9 +19,11 @@ package com.vaadin.client.ui.orderedlayout;
import java.util.List;
import com.google.gwt.aria.client.Roles;
+import com.google.gwt.dom.client.Document;
import com.google.gwt.user.client.DOM;
import com.google.gwt.user.client.Element;
import com.google.gwt.user.client.Event;
+import com.google.gwt.user.client.Timer;
import com.google.gwt.user.client.ui.SimplePanel;
import com.google.gwt.user.client.ui.UIObject;
import com.google.gwt.user.client.ui.Widget;
@@ -456,6 +458,9 @@ public final class Slot extends SimplePanel {
// Caption wrappers
Widget widget = getWidget();
+ final Element focusedElement = Util.getFocusedElement();
+ // By default focus will not be lost
+ boolean focusLost = false;
if (captionText != null || iconUrl != null || error != null || required) {
if (caption == null) {
caption = DOM.createDiv();
@@ -466,6 +471,10 @@ public final class Slot extends SimplePanel {
orphan(widget);
captionWrap.appendChild(widget.getElement());
adopt(widget);
+
+ // Made changes to DOM. Focus can be lost if it was in the
+ // widget.
+ focusLost = widget.getElement().isOrHasChild(focusedElement);
}
} else if (caption != null) {
orphan(widget);
@@ -474,6 +483,9 @@ public final class Slot extends SimplePanel {
captionWrap.removeFromParent();
caption = null;
captionWrap = null;
+
+ // Made changes to DOM. Focus can be lost if it was in the widget.
+ focusLost = widget.getElement().isOrHasChild(focusedElement);
}
// Caption text
@@ -560,6 +572,45 @@ public final class Slot extends SimplePanel {
setCaptionPosition(CaptionPosition.RIGHT);
}
}
+
+ if (focusLost) {
+ // Find out what element is currently focused.
+ Element currentFocus = Util.getFocusedElement();
+ if (currentFocus != null
+ && currentFocus.equals(Document.get().getBody())) {
+ // Focus has moved to BodyElement and should be moved back to
+ // original location. This happened because of adding or
+ // removing the captionWrap
+ focusedElement.focus();
+ } else if (currentFocus != focusedElement) {
+ // Focus is either moved somewhere else on purpose or IE has
+ // lost it. Investigate further.
+ Timer focusTimer = new Timer() {
+
+ @Override
+ public void run() {
+ if (Util.getFocusedElement() == null) {
+ // This should never become an infinite loop and
+ // even if it does it will be stopped once something
+ // is done with the browser.
+ schedule(25);
+ } else if (Util.getFocusedElement().equals(
+ Document.get().getBody())) {
+ // Focus found it's way to BodyElement. Now it can
+ // be restored
+ focusedElement.focus();
+ }
+ }
+ };
+ if (BrowserInfo.get().isIE8()) {
+ // IE8 can't fix the focus immediately. It will fail.
+ focusTimer.schedule(25);
+ } else {
+ // Newer IE versions can handle things immediately.
+ focusTimer.run();
+ }
+ }
+ }
}
/**
diff --git a/uitest/src/com/vaadin/tests/components/orderedlayout/VerticalLayoutFocusWithDOMChanges.java b/uitest/src/com/vaadin/tests/components/orderedlayout/VerticalLayoutFocusWithDOMChanges.java
new file mode 100644
index 0000000000..1e7d817094
--- /dev/null
+++ b/uitest/src/com/vaadin/tests/components/orderedlayout/VerticalLayoutFocusWithDOMChanges.java
@@ -0,0 +1,63 @@
+/*
+ * Copyright 2000-2013 Vaadin Ltd.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License"); you may not
+ * use this file except in compliance with the License. You may obtain a copy of
+ * the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
+ * License for the specific language governing permissions and limitations under
+ * the License.
+ */
+package com.vaadin.tests.components.orderedlayout;
+
+import com.vaadin.data.Property.ValueChangeEvent;
+import com.vaadin.data.Property.ValueChangeListener;
+import com.vaadin.server.VaadinRequest;
+import com.vaadin.tests.components.AbstractTestUI;
+import com.vaadin.ui.Button;
+import com.vaadin.ui.TextField;
+import com.vaadin.ui.VerticalLayout;
+
+public class VerticalLayoutFocusWithDOMChanges extends AbstractTestUI implements
+ ValueChangeListener {
+
+ Button dummyButton = new Button("Just a button");
+ TextField listenedTextField = new TextField();
+ TextField changingTextField = new TextField();
+
+ @Override
+ protected void setup(VaadinRequest request) {
+ VerticalLayout content = new VerticalLayout();
+ setSizeFull();
+ listenedTextField.addValueChangeListener(this);
+ listenedTextField.setImmediate(true);
+ changingTextField.setImmediate(true);
+ content.addComponent(dummyButton);
+ content.addComponent(listenedTextField);
+ content.addComponent(changingTextField);
+ content.setMargin(true);
+ content.setSpacing(true);
+ setContent(content);
+ }
+
+ @Override
+ protected String getTestDescription() {
+ return "Check that creating or removing caption wrap doesn't lose focus";
+ }
+
+ @Override
+ protected Integer getTicketNumber() {
+ return 12967;
+ }
+
+ @Override
+ public void valueChange(ValueChangeEvent event) {
+ changingTextField.setRequired(!changingTextField.isRequired());
+ }
+
+}
diff --git a/uitest/src/com/vaadin/tests/components/orderedlayout/VerticalLayoutFocusWithDOMChangesTest.java b/uitest/src/com/vaadin/tests/components/orderedlayout/VerticalLayoutFocusWithDOMChangesTest.java
new file mode 100644
index 0000000000..14c26a0e17
--- /dev/null
+++ b/uitest/src/com/vaadin/tests/components/orderedlayout/VerticalLayoutFocusWithDOMChangesTest.java
@@ -0,0 +1,100 @@
+/*
+ * Copyright 2000-2013 Vaadin Ltd.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License"); you may not
+ * use this file except in compliance with the License. You may obtain a copy of
+ * the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
+ * License for the specific language governing permissions and limitations under
+ * the License.
+ */
+package com.vaadin.tests.components.orderedlayout;
+
+import java.util.List;
+
+import org.junit.Assert;
+import org.junit.Test;
+import org.openqa.selenium.WebElement;
+import org.openqa.selenium.interactions.Actions;
+import org.openqa.selenium.remote.DesiredCapabilities;
+
+import com.vaadin.testbench.By;
+import com.vaadin.tests.tb3.MultiBrowserTest;
+
+public class VerticalLayoutFocusWithDOMChangesTest extends MultiBrowserTest {
+
+ private String initialText = "Some";
+ private String incrementalText = " text";
+
+ @Test
+ public void inputTextAndChangeFocus() throws InterruptedException {
+ openTestURL();
+ List<WebElement> textFields = getDriver().findElements(
+ By.tagName("input"));
+ WebElement tf1 = textFields.get(0);
+ WebElement tf2 = textFields.get(1);
+ tf1.sendKeys(initialText);
+ new Actions(getDriver()).moveToElement(tf2).click().build().perform();
+
+ WebElement activeElement = getFocusedElement();
+ Assert.assertEquals("input", activeElement.getTagName());
+ Assert.assertEquals("", activeElement.getAttribute("value"));
+
+ tf1.sendKeys(incrementalText);
+ new Actions(getDriver())
+ .moveToElement(
+ getDriver().findElement(By.className("v-button")))
+ .click().build().perform();
+ activeElement = getFocusedElement();
+ Assert.assertEquals("Just a button", activeElement.getText());
+
+ DesiredCapabilities capabilities = getDesiredCapabilities();
+ if (capabilities.equals(BrowserUtil.ie(8))
+ || capabilities.equals(BrowserUtil.ie(9))) {
+ // IE8 and IE9 insert cursor in the start of input instead of end.
+ Assert.assertEquals(incrementalText + initialText,
+ tf1.getAttribute("value"));
+ } else {
+ Assert.assertEquals(initialText + incrementalText,
+ tf1.getAttribute("value"));
+ }
+ }
+
+ @Test
+ public void moveFocusAndChangeFieldWithValue() {
+ openTestURL();
+ List<WebElement> textFields = getDriver().findElements(
+ By.tagName("input"));
+ WebElement tf1 = textFields.get(0);
+ WebElement tf2 = textFields.get(1);
+
+ String firstText = "This is";
+ String secondText = " default value";
+
+ tf2.sendKeys(firstText);
+ tf1.sendKeys(initialText);
+ new Actions(getDriver()).moveToElement(tf2).click().build().perform();
+
+ WebElement activeElement = getFocusedElement();
+ Assert.assertEquals("input", activeElement.getTagName());
+ Assert.assertEquals(firstText, activeElement.getAttribute("value"));
+
+ new Actions(getDriver()).sendKeys(secondText).build().perform();
+ DesiredCapabilities capabilities = getDesiredCapabilities();
+ if (capabilities.equals(BrowserUtil.ie(8))
+ || capabilities.equals(BrowserUtil.ie(9))) {
+ // IE8 and IE9 insert cursor in the start of input instead of end.
+ Assert.assertEquals(secondText + firstText,
+ tf2.getAttribute("value"));
+ } else {
+ Assert.assertEquals(firstText + secondText,
+ tf2.getAttribute("value"));
+ }
+ }
+
+}
diff --git a/uitest/src/com/vaadin/tests/tb3/AbstractTB3Test.java b/uitest/src/com/vaadin/tests/tb3/AbstractTB3Test.java
index 7a214bd60c..55a2b80918 100644
--- a/uitest/src/com/vaadin/tests/tb3/AbstractTB3Test.java
+++ b/uitest/src/com/vaadin/tests/tb3/AbstractTB3Test.java
@@ -28,6 +28,7 @@ import org.junit.After;
import org.junit.Before;
import org.junit.runner.RunWith;
import org.openqa.selenium.By;
+import org.openqa.selenium.JavascriptExecutor;
import org.openqa.selenium.Platform;
import org.openqa.selenium.WebDriver;
import org.openqa.selenium.WebElement;
@@ -278,6 +279,21 @@ public abstract class AbstractTB3Test extends TestBenchTestCase {
}
/**
+ * Uses JavaScript to determine the currently focused element.
+ *
+ * @return Focused element or null
+ */
+ protected WebElement getFocusedElement() {
+ Object focusedElement = ((JavascriptExecutor) getDriver())
+ .executeScript("return document.activeElement");
+ if (null != focusedElement) {
+ return (WebElement) focusedElement;
+ } else {
+ return null;
+ }
+ }
+
+ /**
* Find a Vaadin element based on its id given using Component.setId
*
* @param id