/*
* Copyright (C) 2010 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include <gtest/gtest.h>
#include "Color.h"
#include "KeyboardCodes.h"
#include "PopupMenu.h"
#include "PopupMenuClient.h"
#include "PopupMenuChromium.h"
#include "WebFrameClient.h"
#include "WebFrameImpl.h"
#include "WebInputEvent.h"
#include "WebPopupMenuImpl.h"
#include "WebScreenInfo.h"
#include "WebViewClient.h"
#include "WebViewImpl.h"
using namespace WebCore;
using namespace WebKit;
namespace {
class TestPopupMenuClient : public PopupMenuClient {
public:
// Item at index 0 is selected by default.
TestPopupMenuClient() : m_selectIndex(0) { }
virtual ~TestPopupMenuClient() {}
virtual void valueChanged(unsigned listIndex, bool fireEvents = true)
{
m_selectIndex = listIndex;
}
virtual void selectionChanged(unsigned, bool) {}
virtual void selectionCleared() {}
virtual String itemText(unsigned listIndex) const
{
String str("Item ");
str.append(String::number(listIndex));
return str;
}
virtual String itemLabel(unsigned) const { return String(); }
virtual String itemIcon(unsigned) const { return String(); }
virtual String itemToolTip(unsigned listIndex) const { return itemText(listIndex); }
virtual String itemAccessibilityText(unsigned listIndex) const { return itemText(listIndex); }
virtual bool itemIsEnabled(unsigned listIndex) const { return true; }
virtual PopupMenuStyle itemStyle(unsigned listIndex) const
{
Font font(FontPlatformData(12.0, false, false), false);
return PopupMenuStyle(Color::black, Color::white, font, true, false, Length(), TextDirection(), false /* has text direction override */);
}
virtual PopupMenuStyle menuStyle() const { return itemStyle(0); }
virtual int clientInsetLeft() const { return 0; }
virtual int clientInsetRight() const { return 0; }
virtual int clientPaddingLeft() const { return 0; }
virtual int clientPaddingRight() const { return 0; }
virtual int listSize() const { return 10; }
virtual int selectedIndex() const { return m_selectIndex; }
virtual void popupDidHide() { }
virtual bool itemIsSeparator(unsigned listIndex) const { return false; }
virtual bool itemIsLabel(unsigned listIndex) const { return false; }
virtual bool itemIsSelected(unsigned listIndex) const { return listIndex == m_selectIndex; }
virtual bool shouldPopOver() const { return false; }
virtual bool valueShouldChangeOnHotTrack() const { return false; }
virtual void setTextFromItem(unsigned listIndex) { }
virtual FontSelector* fontSelector() const { return 0; }
virtual HostWindow* hostWindow() const { return 0; }
virtual PassRefPtr<Scrollbar> createScrollbar(ScrollableArea*, ScrollbarOrientation, ScrollbarControlSize) { return 0; }
private:
unsigned m_selectIndex;
};
class TestWebWidgetClient : public WebWidgetClient {
public:
~TestWebWidgetClient() { }
};
class TestWebPopupMenuImpl : public WebPopupMenuImpl {
public:
static PassRefPtr<TestWebPopupMenuImpl> create(WebWidgetClient* client)
{
return adoptRef(new TestWebPopupMenuImpl(client));
}
~TestWebPopupMenuImpl() { }
private:
TestWebPopupMenuImpl(WebWidgetClient* client) : WebPopupMenuImpl(client) { }
};
class TestWebWidget : public WebWidget {
public:
virtual ~TestWebWidget() { }
virtual void close() { }
virtual WebSize size() { return WebSize(100, 100); }
virtual void resize(const WebSize&) { }
virtual void layout() { }
virtual void paint(WebCanvas*, const WebRect&) { }
virtual void themeChanged() { }
virtual void composite(bool finish) { }
virtual bool handleInputEvent(const WebInputEvent&) { return true; }
virtual void mouseCaptureLost() { }
virtual void setFocus(bool) { }
virtual bool setComposition(
const WebString& text,
const WebVector<WebCompositionUnderline>& underlines,
int selectionStart,
int selectionEnd) { return true; }
virtual bool confirmComposition() { return true; }
virtual bool confirmComposition(const WebString& text) { return true; }
virtual WebTextInputType textInputType() { return WebKit::WebTextInputTypeNone; }
virtual WebRect caretOrSelectionBounds() { return WebRect(); }
virtual bool selectionRange(WebPoint& start, WebPoint& end) const { return false; }
virtual void setTextDirection(WebTextDirection) { }
};
class TestWebViewClient : public WebViewClient {
public:
TestWebViewClient() : m_webPopupMenu(TestWebPopupMenuImpl::create(&m_webWidgetClient)) { }
~TestWebViewClient() { }
virtual WebWidget* createPopupMenu(WebPopupType) { return m_webPopupMenu.get(); }
// We need to override this so that the popup menu size is not 0
// (the layout code checks to see if the popup fits on the screen).
virtual WebScreenInfo screenInfo()
{
WebScreenInfo screenInfo;
screenInfo.availableRect.height = 2000;
screenInfo.availableRect.width = 2000;
return screenInfo;
}
private:
TestWebWidgetClient m_webWidgetClient;
RefPtr<TestWebPopupMenuImpl> m_webPopupMenu;
};
class TestWebFrameClient : public WebFrameClient {
public:
~TestWebFrameClient() { }
};
class SelectPopupMenuTest : public testing::Test {
public:
SelectPopupMenuTest()
{
}
protected:
virtual void SetUp()
{
m_webView = static_cast<WebViewImpl*>(WebView::create(&m_webviewClient));
m_webView->initializeMainFrame(&m_webFrameClient);
m_popupMenu = adoptRef(new PopupMenuChromium(&m_popupMenuClient));
}
virtual void TearDown()
{
m_popupMenu = 0;
m_webView->close();
}
// Returns true if there currently is a select popup in the WebView.
bool popupOpen() const { return m_webView->selectPopup(); }
int selectedIndex() const { return m_popupMenuClient.selectedIndex(); }
void showPopup()
{
m_popupMenu->show(IntRect(0, 0, 100, 100),
static_cast<WebFrameImpl*>(m_webView->mainFrame())->frameView(), 0);
ASSERT_TRUE(popupOpen());
EXPECT_TRUE(m_webView->selectPopup()->popupType() == PopupContainer::Select);
}
void hidePopup()
{
m_popupMenu->hide();
EXPECT_FALSE(popupOpen());
}
void simulateKeyDownEvent(int keyCode)
{
simulateKeyEvent(WebInputEvent::RawKeyDown, keyCode);
}
void simulateKeyUpEvent(int keyCode)
{
simulateKeyEvent(WebInputEvent::KeyUp, keyCode);
}
// Simulates a key event on the WebView.
// The WebView forwards the event to the select popup if one is open.
void simulateKeyEvent(WebInputEvent::Type eventType, int keyCode)
{
WebKeyboardEvent keyEvent;
keyEvent.windowsKeyCode = keyCode;
keyEvent.type = eventType;
m_webView->handleInputEvent(keyEvent);
}
// Simulates a mouse event on the select popup.
void simulateLeftMouseDownEvent(const IntPoint& point)
{
PlatformMouseEvent mouseEvent(point, point, LeftButton, MouseEventPressed,
1, false, false, false, false, 0);
m_webView->selectPopup()->handleMouseDownEvent(mouseEvent);
}
void simulateLeftMouseUpEvent(const IntPoint& point)
{
PlatformMouseEvent mouseEvent(point, point, LeftButton, MouseEventReleased,
1, false, false, false, false, 0);
m_webView->selectPopup()->handleMouseReleaseEvent(mouseEvent);
}
protected:
TestWebViewClient m_webviewClient;
WebViewImpl* m_webView;
TestWebFrameClient m_webFrameClient;
TestPopupMenuClient m_popupMenuClient;
RefPtr<PopupMenu> m_popupMenu;
};
// Tests that show/hide and repeats. Select popups are reused in web pages when
// they are reopened, that what this is testing.
TEST_F(SelectPopupMenuTest, ShowThenHide)
{
for (int i = 0; i < 3; i++) {
showPopup();
hidePopup();
}
}
// Tests that showing a select popup and deleting it does not cause problem.
// This happens in real-life if a page navigates while a select popup is showing.
TEST_F(SelectPopupMenuTest, ShowThenDelete)
{
showPopup();
// Nothing else to do, TearDown() deletes the popup.
}
// Tests that losing focus closes the select popup.
TEST_F(SelectPopupMenuTest, ShowThenLoseFocus)
{
showPopup();
// Simulate losing focus.
m_webView->setFocus(false);
// Popup should have closed.
EXPECT_FALSE(popupOpen());
}
// Tests that pressing ESC closes the popup.
TEST_F(SelectPopupMenuTest, ShowThenPressESC)
{
showPopup();
simulateKeyDownEvent(VKEY_ESCAPE);
// Popup should have closed.
EXPECT_FALSE(popupOpen());
}
// Tests selecting an item with the arrows and enter/esc/tab.
TEST_F(SelectPopupMenuTest, SelectWithKeys)
{
showPopup();
// Simulate selecting the 2nd item by pressing Down, Down, enter.
simulateKeyDownEvent(VKEY_DOWN);
simulateKeyDownEvent(VKEY_DOWN);
simulateKeyDownEvent(VKEY_RETURN);
// Popup should have closed.
EXPECT_TRUE(!popupOpen());
EXPECT_EQ(2, selectedIndex());
// It should work as well with ESC.
showPopup();
simulateKeyDownEvent(VKEY_DOWN);
simulateKeyDownEvent(VKEY_ESCAPE);
EXPECT_FALSE(popupOpen());
EXPECT_EQ(3, selectedIndex());
// It should work as well with TAB.
showPopup();
simulateKeyDownEvent(VKEY_DOWN);
simulateKeyDownEvent(VKEY_TAB);
EXPECT_FALSE(popupOpen());
EXPECT_EQ(4, selectedIndex());
}
// Tests that selecting an item with the mouse does select the item and close
// the popup.
TEST_F(SelectPopupMenuTest, ClickItem)
{
showPopup();
// Y of 18 to be on the item at index 1 (12 font plus border and more to be safe).
IntPoint row1Point(2, 18);
// Simulate a click down/up on the first item.
simulateLeftMouseDownEvent(row1Point);
simulateLeftMouseUpEvent(row1Point);
// Popup should have closed and the item at index 1 selected.
EXPECT_FALSE(popupOpen());
EXPECT_EQ(1, selectedIndex());
}
// Tests that moving the mouse over an item and then clicking outside the select popup
// leaves the seleted item unchanged.
TEST_F(SelectPopupMenuTest, MouseOverItemClickOutside)
{
showPopup();
// Y of 18 to be on the item at index 1 (12 font plus border and more to be safe).
IntPoint row1Point(2, 18);
// Simulate the mouse moving over the first item.
PlatformMouseEvent mouseEvent(row1Point, row1Point, NoButton, MouseEventMoved,
1, false, false, false, false, 0);
m_webView->selectPopup()->handleMouseMoveEvent(mouseEvent);
// Click outside the popup.
simulateLeftMouseDownEvent(IntPoint(1000, 1000));
// Popup should have closed and item 0 should still be selected.
EXPECT_FALSE(popupOpen());
EXPECT_EQ(0, selectedIndex());
}
// Tests that selecting an item with the keyboard and then clicking outside the select
// popup does select that item.
TEST_F(SelectPopupMenuTest, SelectItemWithKeyboardItemClickOutside)
{
showPopup();
// Simulate selecting the 2nd item by pressing Down, Down.
simulateKeyDownEvent(VKEY_DOWN);
simulateKeyDownEvent(VKEY_DOWN);
// Click outside the popup.
simulateLeftMouseDownEvent(IntPoint(1000, 1000));
// Popup should have closed and the item should have been selected.
EXPECT_FALSE(popupOpen());
EXPECT_EQ(2, selectedIndex());
}
} // namespace