import com.google.gwt.core.client.JavaScriptObject;
import com.google.gwt.core.client.Scheduler;
import com.google.gwt.core.client.Scheduler.ScheduledCommand;
+import com.google.gwt.dom.client.Document;
import com.google.gwt.dom.client.Element;
import com.google.gwt.dom.client.NativeEvent;
import com.google.gwt.dom.client.Style;
debug("VFS.SP: setPosition(" + offsetWidth + ", " + offsetHeight
+ ")");
- int top = topPosition;
+ int top;
int left = getPopupLeft();
// reset menu size and retrieve its "natural" size
getContainerElement().getStyle().setWidth(rootWidth, Unit.PX);
}
- final int vfsHeight = VFilterSelect.this.getOffsetHeight();
- final int spaceAvailableAbove = top - vfsHeight;
- final int spaceAvailableBelow = Window.getClientHeight() - top;
- if (spaceAvailableBelow < offsetHeight
- && spaceAvailableBelow < spaceAvailableAbove) {
+ final int textInputHeight = VFilterSelect.this.getOffsetHeight();
+ final int textInputTopOnPage = tb.getAbsoluteTop();
+ final int viewportOffset = Document.get().getScrollTop();
+ final int textInputTopInViewport = textInputTopOnPage
+ - viewportOffset;
+ final int textInputBottomInViewport = textInputTopInViewport
+ + textInputHeight;
+
+ final int spaceAboveInViewport = textInputTopInViewport;
+ final int spaceBelowInViewport = Window.getClientHeight()
+ - textInputBottomInViewport;
+
+ if (spaceBelowInViewport < offsetHeight
+ && spaceBelowInViewport < spaceAboveInViewport) {
// popup on top of input instead
- top -= offsetHeight + vfsHeight;
- if (top < 0) {
- offsetHeight += top;
- top = 0;
+ if (offsetHeight > spaceAboveInViewport) {
+ // Shrink popup height to fit above
+ offsetHeight = spaceAboveInViewport;
}
+ top = textInputTopOnPage - offsetHeight;
} else {
- offsetHeight = Math.min(offsetHeight, spaceAvailableBelow);
+ // Show below, position calculated in showSuggestions for some
+ // strange reason
+ top = topPosition;
+ offsetHeight = Math.min(offsetHeight, spaceBelowInViewport);
}
// fetch real width (mac FF bugs here due GWT popups overflow:auto )
--- /dev/null
+/*
+ * Copyright 2000-2014 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.combobox;
+
+import com.vaadin.server.VaadinRequest;
+import com.vaadin.tests.components.AbstractTestUI;
+import com.vaadin.ui.ComboBox;
+import com.vaadin.ui.Label;
+
+public class ComboBoxPopupWhenBodyScrolls extends AbstractTestUI {
+
+ @Override
+ protected void setup(VaadinRequest request) {
+ getPage().getStyles().add(
+ "body.v-generated-body { overflow: auto;height:auto;}");
+ getPage()
+ .getStyles()
+ .add("body.v-generated-body .v-ui.v-scrollable{ overflow: visible;height:auto !important;}");
+ ComboBox cb = new ComboBox();
+ for (int i = 0; i < 10; i++) {
+ cb.addItem("Item " + i);
+ }
+
+ Label spacer = new Label("foo");
+ spacer.setHeight("2000px");
+ addComponent(spacer);
+ addComponent(cb);
+ spacer = new Label("foo");
+ spacer.setHeight("2000px");
+ addComponent(spacer);
+ // Chrome requires document.scrollTop (<body>)
+ // Firefox + IE wants document.documentElement.scrollTop (<html>)
+ getPage()
+ .getJavaScript()
+ .execute(
+ "document.body.scrollTop=1800;document.documentElement.scrollTop=1800;");
+ }
+}
--- /dev/null
+/*
+ * Copyright 2000-2014 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.combobox;
+
+import org.junit.Assert;
+import org.junit.Test;
+import org.openqa.selenium.WebElement;
+
+import com.vaadin.tests.tb3.MultiBrowserTest;
+import com.vaadin.tests.tb3.newelements.ComboBoxElement;
+
+public class ComboBoxPopupWhenBodyScrollsTest extends MultiBrowserTest {
+
+ @Test
+ public void popupBelow() {
+ openTestURL();
+ ComboBoxElement combobox = $(ComboBoxElement.class).first();
+ combobox.openPopup();
+ WebElement popup = $(ComboBoxElement.class).first()
+ .getSuggestionPopup();
+
+ int comboboxTop = combobox.getLocation().getY();
+ int popupTop = popup.getLocation().getY();
+ Assert.assertTrue("Popup should be below combobox",
+ popupTop > comboboxTop);
+ }
+}
public class ComboBoxElement extends
com.vaadin.testbench.elements.ComboBoxElement {
+ private static org.openqa.selenium.By bySuggestionPopup = By
+ .vaadin("#popup");
+
public WebElement getInputField() {
return findElement(By.vaadin("#textbox"));
}
getInputField().clear();
}
+ public WebElement getSuggestionPopup() {
+ return findElement(bySuggestionPopup);
+ }
+
@Override
public void sendKeys(CharSequence... keysToSend) {
sendKeys(50, keysToSend);