/*
* 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 "AutoFillPopupMenuClient.h"
#include "CSSStyleSelector.h"
#include "CSSValueKeywords.h"
#include "Chrome.h"
#include "FrameView.h"
#include "HTMLInputElement.h"
#include "RenderTheme.h"
#include "WebAutoFillClient.h"
#include "WebNode.h"
#include "WebString.h"
#include "WebVector.h"
#include "WebViewClient.h"
#include "WebViewImpl.h"
using namespace WebCore;
namespace WebKit {
AutoFillPopupMenuClient::AutoFillPopupMenuClient()
: m_separatorIndex(-1)
, m_selectedIndex(-1)
, m_textField(0)
{
}
AutoFillPopupMenuClient::~AutoFillPopupMenuClient()
{
}
unsigned AutoFillPopupMenuClient::getSuggestionsCount() const
{
return m_names.size() + ((m_separatorIndex == -1) ? 0 : 1);
}
WebString AutoFillPopupMenuClient::getSuggestion(unsigned listIndex) const
{
int index = convertListIndexToInternalIndex(listIndex);
if (index == -1)
return WebString();
ASSERT(index >= 0 && static_cast<size_t>(index) < m_names.size());
return m_names[index];
}
WebString AutoFillPopupMenuClient::getLabel(unsigned listIndex) const
{
int index = convertListIndexToInternalIndex(listIndex);
if (index == -1)
return WebString();
ASSERT(index >= 0 && static_cast<size_t>(index) < m_labels.size());
return m_labels[index];
}
WebString AutoFillPopupMenuClient::getIcon(unsigned listIndex) const
{
int index = convertListIndexToInternalIndex(listIndex);
if (index == -1)
return WebString();
ASSERT(index >= 0 && static_cast<size_t>(index) < m_icons.size());
return m_icons[index];
}
void AutoFillPopupMenuClient::removeSuggestionAtIndex(unsigned listIndex)
{
if (!canRemoveSuggestionAtIndex(listIndex))
return;
int index = convertListIndexToInternalIndex(listIndex);
ASSERT(static_cast<unsigned>(index) < m_names.size());
m_names.remove(index);
m_labels.remove(index);
m_icons.remove(index);
m_uniqueIDs.remove(index);
// Shift the separator index if necessary.
if (m_separatorIndex != -1)
m_separatorIndex--;
}
bool AutoFillPopupMenuClient::canRemoveSuggestionAtIndex(unsigned listIndex)
{
// Only allow deletion of items before the separator that have unique id 0
// (i.e. are autocomplete rather than autofill items).
int index = convertListIndexToInternalIndex(listIndex);
return !m_uniqueIDs[index] && (m_separatorIndex == -1 || listIndex < static_cast<unsigned>(m_separatorIndex));
}
void AutoFillPopupMenuClient::valueChanged(unsigned listIndex, bool fireEvents)
{
WebViewImpl* webView = getWebView();
if (!webView)
return;
if (m_separatorIndex != -1 && listIndex > static_cast<unsigned>(m_separatorIndex))
--listIndex;
ASSERT(listIndex < m_names.size());
webView->autoFillClient()->didAcceptAutoFillSuggestion(WebNode(getTextField()),
m_names[listIndex],
m_labels[listIndex],
m_uniqueIDs[listIndex],
listIndex);
}
void AutoFillPopupMenuClient::selectionChanged(unsigned listIndex, bool fireEvents)
{
WebViewImpl* webView = getWebView();
if (!webView)
return;
if (m_separatorIndex != -1 && listIndex > static_cast<unsigned>(m_separatorIndex))
--listIndex;
ASSERT(listIndex < m_names.size());
webView->autoFillClient()->didSelectAutoFillSuggestion(WebNode(getTextField()),
m_names[listIndex],
m_labels[listIndex],
m_uniqueIDs[listIndex]);
}
void AutoFillPopupMenuClient::selectionCleared()
{
WebViewImpl* webView = getWebView();
if (webView)
webView->autoFillClient()->didClearAutoFillSelection(WebNode(getTextField()));
}
String AutoFillPopupMenuClient::itemText(unsigned listIndex) const
{
return getSuggestion(listIndex);
}
String AutoFillPopupMenuClient::itemLabel(unsigned listIndex) const
{
return getLabel(listIndex);
}
String AutoFillPopupMenuClient::itemIcon(unsigned listIndex) const
{
return getIcon(listIndex);
}
bool AutoFillPopupMenuClient::itemIsEnabled(unsigned listIndex) const
{
return !itemIsWarning(listIndex);
}
PopupMenuStyle AutoFillPopupMenuClient::itemStyle(unsigned listIndex) const
{
return itemIsWarning(listIndex) ? *m_warningStyle : *m_regularStyle;
}
PopupMenuStyle AutoFillPopupMenuClient::menuStyle() const
{
return *m_regularStyle;
}
int AutoFillPopupMenuClient::clientPaddingLeft() const
{
// Bug http://crbug.com/7708 seems to indicate the style can be 0.
RenderStyle* style = textFieldStyle();
if (!style)
return 0;
return RenderTheme::defaultTheme()->popupInternalPaddingLeft(style);
}
int AutoFillPopupMenuClient::clientPaddingRight() const
{
// Bug http://crbug.com/7708 seems to indicate the style can be 0.
RenderStyle* style = textFieldStyle();
if (!style)
return 0;
return RenderTheme::defaultTheme()->popupInternalPaddingRight(style);
}
void AutoFillPopupMenuClient::popupDidHide()
{
WebViewImpl* webView = getWebView();
if (!webView)
return;
webView->autoFillPopupDidHide();
webView->autoFillClient()->didClearAutoFillSelection(WebNode(getTextField()));
}
bool AutoFillPopupMenuClient::itemIsSeparator(unsigned listIndex) const
{
return (m_separatorIndex != -1 && static_cast<unsigned>(m_separatorIndex) == listIndex);
}
bool AutoFillPopupMenuClient::itemIsWarning(unsigned listIndex) const
{
int index = convertListIndexToInternalIndex(listIndex);
if (index == -1)
return false;
ASSERT(index >= 0 && static_cast<size_t>(index) < m_uniqueIDs.size());
return m_uniqueIDs[index] < 0;
}
void AutoFillPopupMenuClient::setTextFromItem(unsigned listIndex)
{
m_textField->setValue(getSuggestion(listIndex));
}
FontSelector* AutoFillPopupMenuClient::fontSelector() const
{
return m_textField->document()->styleSelector()->fontSelector();
}
HostWindow* AutoFillPopupMenuClient::hostWindow() const
{
return m_textField->document()->view()->hostWindow();
}
PassRefPtr<Scrollbar> AutoFillPopupMenuClient::createScrollbar(
ScrollableArea* scrollableArea,
ScrollbarOrientation orientation,
ScrollbarControlSize size)
{
return Scrollbar::createNativeScrollbar(scrollableArea, orientation, size);
}
void AutoFillPopupMenuClient::initialize(
HTMLInputElement* textField,
const WebVector<WebString>& names,
const WebVector<WebString>& labels,
const WebVector<WebString>& icons,
const WebVector<int>& uniqueIDs,
int separatorIndex)
{
ASSERT(names.size() == labels.size());
ASSERT(names.size() == icons.size());
ASSERT(names.size() == uniqueIDs.size());
ASSERT(separatorIndex < static_cast<int>(names.size()));
m_selectedIndex = -1;
m_textField = textField;
// The suggestions must be set before initializing the
// AutoFillPopupMenuClient.
setSuggestions(names, labels, icons, uniqueIDs, separatorIndex);
FontDescription regularFontDescription;
RenderTheme::defaultTheme()->systemFont(CSSValueWebkitControl,
regularFontDescription);
RenderStyle* style = m_textField->computedStyle();
regularFontDescription.setComputedSize(style->fontDescription().computedSize());
Font regularFont(regularFontDescription, 0, 0);
regularFont.update(textField->document()->styleSelector()->fontSelector());
// The direction of text in popup menu is set the same as the direction of
// the input element: textField.
m_regularStyle.set(new PopupMenuStyle(Color::black, Color::white, regularFont,
true, false, Length(WebCore::Fixed),
textField->renderer()->style()->direction(), textField->renderer()->style()->unicodeBidi() == Override));
FontDescription warningFontDescription = regularFont.fontDescription();
warningFontDescription.setItalic(true);
Font warningFont(warningFontDescription, regularFont.letterSpacing(), regularFont.wordSpacing());
warningFont.update(regularFont.fontSelector());
m_warningStyle.set(new PopupMenuStyle(Color::darkGray,
m_regularStyle->backgroundColor(),
warningFont,
m_regularStyle->isVisible(),
m_regularStyle->isDisplayNone(),
m_regularStyle->textIndent(),
m_regularStyle->textDirection(),
m_regularStyle->hasTextDirectionOverride()));
}
void AutoFillPopupMenuClient::setSuggestions(const WebVector<WebString>& names,
const WebVector<WebString>& labels,
const WebVector<WebString>& icons,
const WebVector<int>& uniqueIDs,
int separatorIndex)
{
ASSERT(names.size() == labels.size());
ASSERT(names.size() == icons.size());
ASSERT(names.size() == uniqueIDs.size());
ASSERT(separatorIndex < static_cast<int>(names.size()));
m_names.clear();
m_labels.clear();
m_icons.clear();
m_uniqueIDs.clear();
for (size_t i = 0; i < names.size(); ++i) {
m_names.append(names[i]);
m_labels.append(labels[i]);
m_icons.append(icons[i]);
m_uniqueIDs.append(uniqueIDs[i]);
}
m_separatorIndex = separatorIndex;
// Try to preserve selection if possible.
if (getSelectedIndex() >= static_cast<int>(names.size()))
setSelectedIndex(-1);
}
int AutoFillPopupMenuClient::convertListIndexToInternalIndex(unsigned listIndex) const
{
if (listIndex == static_cast<unsigned>(m_separatorIndex))
return -1;
if (m_separatorIndex == -1 || listIndex < static_cast<unsigned>(m_separatorIndex))
return listIndex;
return listIndex - 1;
}
WebViewImpl* AutoFillPopupMenuClient::getWebView() const
{
Frame* frame = m_textField->document()->frame();
if (!frame)
return 0;
Page* page = frame->page();
if (!page)
return 0;
return static_cast<WebViewImpl*>(page->chrome()->client()->webView());
}
RenderStyle* AutoFillPopupMenuClient::textFieldStyle() const
{
RenderStyle* style = m_textField->computedStyle();
if (!style) {
// It seems we can only have a 0 style in a TextField if the
// node is detached, in which case we the popup should not be
// showing. Please report this in http://crbug.com/7708 and
// include the page you were visiting.
ASSERT_NOT_REACHED();
}
return style;
}
} // namespace WebKit