blob: 315630232a83a1d1425b75a178cd0e25eb52292a [file] [log] [blame]
// Copyright 2014 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "ash/app_list/views/search_box_view.h"
#include <cctype>
#include <map>
#include <memory>
#include <string>
#include <utility>
#include "ash/app_list/test/app_list_test_view_delegate.h"
#include "ash/app_list/views/app_list_view.h"
#include "ash/app_list/views/contents_view.h"
#include "ash/public/cpp/app_list/app_list_features.h"
#include "ash/public/cpp/app_list/vector_icons/vector_icons.h"
#include "base/macros.h"
#include "base/strings/utf_string_conversions.h"
#include "base/test/scoped_feature_list.h"
#include "ui/base/ime/composition_text.h"
#include "ui/chromeos/search_box/search_box_constants.h"
#include "ui/chromeos/search_box/search_box_view_delegate.h"
#include "ui/events/base_event_utils.h"
#include "ui/gfx/image/image_skia.h"
#include "ui/gfx/image/image_unittest_util.h"
#include "ui/gfx/paint_vector_icon.h"
#include "ui/views/controls/image_view.h"
#include "ui/views/controls/textfield/textfield.h"
#include "ui/views/test/widget_test.h"
namespace app_list {
namespace test {
class KeyPressCounterView : public ContentsView {
public:
explicit KeyPressCounterView(AppListView* app_list_view)
: ContentsView(app_list_view), count_(0) {}
~KeyPressCounterView() override {}
int GetCountAndReset() {
int count = count_;
count_ = 0;
return count;
}
private:
// Overridden from views::View:
bool OnKeyPressed(const ui::KeyEvent& key_event) override {
if (!::isalnum(static_cast<int>(key_event.key_code()))) {
++count_;
return true;
}
return false;
}
int count_;
DISALLOW_COPY_AND_ASSIGN(KeyPressCounterView);
};
class SearchBoxViewTest : public views::test::WidgetTest,
public search_box::SearchBoxViewDelegate {
public:
SearchBoxViewTest() = default;
~SearchBoxViewTest() override = default;
// Overridden from testing::Test:
void SetUp() override {
views::test::WidgetTest::SetUp();
app_list_view_ = new AppListView(&view_delegate_);
app_list_view_->InitView(false /*is_tablet_mode*/, GetContext());
widget_ = CreateTopLevelPlatformWidget();
view_ =
std::make_unique<SearchBoxView>(this, &view_delegate_, app_list_view());
view_->Init(false /*is_tablet_mode*/);
widget_->SetBounds(gfx::Rect(0, 0, 300, 200));
counter_view_ = new KeyPressCounterView(app_list_view_);
widget_->GetContentsView()->AddChildView(view());
widget_->GetContentsView()->AddChildView(counter_view_);
counter_view_->Init(view_delegate_.GetModel());
view()->set_contents_view(counter_view_);
}
void TearDown() override {
view_.reset();
app_list_view_->GetWidget()->Close();
widget_->CloseNow();
views::test::WidgetTest::TearDown();
}
protected:
views::Widget* widget() { return widget_; }
SearchBoxView* view() { return view_.get(); }
AppListView* app_list_view() { return app_list_view_; }
AppListTestViewDelegate* view_delegate() { return &view_delegate_; }
void SetSearchEngineIsGoogle(bool is_google) {
view_delegate_.SetSearchEngineIsGoogle(is_google);
}
void SetSearchBoxActive(bool active, ui::EventType type) {
view()->SetSearchBoxActive(active, type);
}
int GetContentsViewKeyPressCountAndReset() {
return counter_view_->GetCountAndReset();
}
void KeyPress(ui::KeyboardCode key_code) {
ui::KeyEvent event(ui::ET_KEY_PRESSED, key_code, ui::EF_NONE);
view()->search_box()->OnKeyEvent(&event);
// Emulates the input method.
if (::isalnum(static_cast<int>(key_code))) {
base::char16 character = ::tolower(static_cast<int>(key_code));
view()->search_box()->InsertText(base::string16(1, character));
}
}
std::string GetLastQueryAndReset() {
base::string16 query = last_query_;
last_query_.clear();
return base::UTF16ToUTF8(query);
}
int GetQueryChangedCountAndReset() {
int result = query_changed_count_;
query_changed_count_ = 0;
return result;
}
SearchModel::SearchResults* results() {
return view_delegate_.GetSearchModel()->results();
}
private:
// Overridden from SearchBoxViewDelegate:
void QueryChanged(search_box::SearchBoxViewBase* sender) override {
++query_changed_count_;
last_query_ = sender->search_box()->text();
}
void AssistantButtonPressed() override {}
void BackButtonPressed() override {}
void ActiveChanged(search_box::SearchBoxViewBase* sender) override {}
void SearchBoxFocusChanged(search_box::SearchBoxViewBase* sender) override {}
AppListTestViewDelegate view_delegate_;
views::Widget* widget_;
AppListView* app_list_view_ = nullptr;
std::unique_ptr<SearchBoxView> view_;
KeyPressCounterView* counter_view_;
base::string16 last_query_;
int query_changed_count_ = 0;
DISALLOW_COPY_AND_ASSIGN(SearchBoxViewTest);
};
// Tests that the close button is invisible by default.
TEST_F(SearchBoxViewTest, CloseButtonInvisibleByDefault) {
EXPECT_FALSE(view()->close_button()->GetVisible());
}
// Tests that the close button becomes visible after typing in the search box.
TEST_F(SearchBoxViewTest, CloseButtonVisibleAfterTyping) {
KeyPress(ui::VKEY_A);
EXPECT_TRUE(view()->close_button()->GetVisible());
}
// Tests that the close button is still invisible after the search box is
// activated.
TEST_F(SearchBoxViewTest, CloseButtonInvisibleAfterSearchBoxActived) {
SetSearchBoxActive(true, ui::ET_MOUSE_PRESSED);
// UI behavior is different with Zero State enabled.
if (app_list_features::IsZeroStateSuggestionsEnabled())
EXPECT_TRUE(view()->close_button()->GetVisible());
else
EXPECT_FALSE(view()->close_button()->GetVisible());
}
// Tests that the close button becomes invisible after close button is clicked.
TEST_F(SearchBoxViewTest, CloseButtonInvisibleAfterCloseButtonClicked) {
KeyPress(ui::VKEY_A);
view()->ButtonPressed(
view()->close_button(),
ui::MouseEvent(ui::ET_MOUSE_PRESSED, gfx::Point(), gfx::Point(),
base::TimeTicks(), ui::EF_LEFT_MOUSE_BUTTON,
ui::EF_LEFT_MOUSE_BUTTON));
EXPECT_FALSE(view()->close_button()->GetVisible());
}
// Tests that the search box becomes empty after close button is clicked.
TEST_F(SearchBoxViewTest, SearchBoxEmptyAfterCloseButtonClicked) {
KeyPress(ui::VKEY_A);
view()->ButtonPressed(
view()->close_button(),
ui::MouseEvent(ui::ET_MOUSE_PRESSED, gfx::Point(), gfx::Point(),
base::TimeTicks(), ui::EF_LEFT_MOUSE_BUTTON,
ui::EF_LEFT_MOUSE_BUTTON));
EXPECT_TRUE(view()->search_box()->text().empty());
}
// Tests that the search box is no longer active after close button is clicked.
TEST_F(SearchBoxViewTest, SearchBoxActiveAfterCloseButtonClicked) {
KeyPress(ui::VKEY_A);
view()->ButtonPressed(
view()->close_button(),
ui::MouseEvent(ui::ET_MOUSE_PRESSED, gfx::Point(), gfx::Point(),
base::TimeTicks(), ui::EF_LEFT_MOUSE_BUTTON,
ui::EF_LEFT_MOUSE_BUTTON));
EXPECT_FALSE(view()->is_search_box_active());
}
// Tests that the search box is inactive by default.
TEST_F(SearchBoxViewTest, SearchBoxInactiveByDefault) {
ASSERT_FALSE(view()->is_search_box_active());
}
// Tests that the black Google icon is used for an inactive Google search.
TEST_F(SearchBoxViewTest, SearchBoxInactiveSearchBoxGoogle) {
SetSearchEngineIsGoogle(true);
SetSearchBoxActive(false, ui::ET_UNKNOWN);
const gfx::ImageSkia expected_icon =
gfx::CreateVectorIcon(kGoogleBlackIcon, search_box::kIconSize,
search_box::kDefaultSearchboxColor);
view()->ModelChanged();
const gfx::ImageSkia actual_icon =
view()->get_search_icon_for_test()->GetImage();
EXPECT_TRUE(gfx::test::AreBitmapsEqual(*expected_icon.bitmap(),
*actual_icon.bitmap()));
}
// Tests that the colored Google icon is used for an active Google search.
TEST_F(SearchBoxViewTest, SearchBoxActiveSearchEngineGoogle) {
SetSearchEngineIsGoogle(true);
SetSearchBoxActive(true, ui::ET_MOUSE_PRESSED);
const gfx::ImageSkia expected_icon =
gfx::CreateVectorIcon(kGoogleColorIcon, search_box::kIconSize,
search_box::kDefaultSearchboxColor);
view()->ModelChanged();
const gfx::ImageSkia actual_icon =
view()->get_search_icon_for_test()->GetImage();
EXPECT_TRUE(gfx::test::AreBitmapsEqual(*expected_icon.bitmap(),
*actual_icon.bitmap()));
}
// Tests that the non-Google icon is used for an inactive non-Google search.
TEST_F(SearchBoxViewTest, SearchBoxInactiveSearchEngineNotGoogle) {
SetSearchEngineIsGoogle(false);
SetSearchBoxActive(false, ui::ET_UNKNOWN);
const gfx::ImageSkia expected_icon =
gfx::CreateVectorIcon(kSearchEngineNotGoogleIcon, search_box::kIconSize,
search_box::kDefaultSearchboxColor);
view()->ModelChanged();
const gfx::ImageSkia actual_icon =
view()->get_search_icon_for_test()->GetImage();
EXPECT_TRUE(gfx::test::AreBitmapsEqual(*expected_icon.bitmap(),
*actual_icon.bitmap()));
}
// Tests that the non-Google icon is used for an active non-Google search.
TEST_F(SearchBoxViewTest, SearchBoxActiveSearchEngineNotGoogle) {
SetSearchEngineIsGoogle(false);
SetSearchBoxActive(true, ui::ET_UNKNOWN);
const gfx::ImageSkia expected_icon =
gfx::CreateVectorIcon(kSearchEngineNotGoogleIcon, search_box::kIconSize,
search_box::kDefaultSearchboxColor);
view()->ModelChanged();
const gfx::ImageSkia actual_icon =
view()->get_search_icon_for_test()->GetImage();
EXPECT_TRUE(gfx::test::AreBitmapsEqual(*expected_icon.bitmap(),
*actual_icon.bitmap()));
}
class SearchBoxViewAssistantButtonTest : public SearchBoxViewTest {
public:
SearchBoxViewAssistantButtonTest() = default;
~SearchBoxViewAssistantButtonTest() override = default;
// Overridden from testing::Test
void SetUp() override {
SearchBoxViewTest::SetUp();
view_delegate()->GetSearchModel()->search_box()->SetShowAssistantButton(
true);
}
private:
DISALLOW_COPY_AND_ASSIGN(SearchBoxViewAssistantButtonTest);
};
// Tests that the assistant button is visible by default.
TEST_F(SearchBoxViewAssistantButtonTest, AssistantButtonVisibleByDefault) {
EXPECT_TRUE(view()->assistant_button()->GetVisible());
}
// Tests that the assistant button is visible after the search box is activated.
TEST_F(SearchBoxViewAssistantButtonTest,
AssistantButtonVisibleAfterSearchBoxActived) {
// Assistant button is not showing up under zero state for now.
// TODO(jennyz): Make assistant button show up under zero state.
if (!app_list_features::IsZeroStateSuggestionsEnabled()) {
SetSearchBoxActive(true, ui::ET_MOUSE_PRESSED);
EXPECT_TRUE(view()->assistant_button()->GetVisible());
}
}
// Tests that the assistant button is invisible after typing in the search box,
// and comes back when search box is empty.
TEST_F(SearchBoxViewAssistantButtonTest,
AssistantButtonChangeVisibilityWithTyping) {
KeyPress(ui::VKEY_A);
EXPECT_FALSE(view()->assistant_button()->GetVisible());
// Assistant button is not showing up under zero state for now.
// TODO(crbug.com/925455): Make assistant button show up under zero state.
if (!app_list_features::IsZeroStateSuggestionsEnabled()) {
KeyPress(ui::VKEY_BACK);
EXPECT_TRUE(view()->assistant_button()->GetVisible());
}
}
class SearchBoxViewAutocompleteTest
: public SearchBoxViewTest,
public ::testing::WithParamInterface<ui::KeyboardCode> {
public:
SearchBoxViewAutocompleteTest() = default;
~SearchBoxViewAutocompleteTest() override = default;
// Overridden from testing::Test
void SetUp() override {
scoped_feature_list_.InitWithFeatures(
{app_list_features::kEnableAppListSearchAutocomplete}, {});
SearchBoxViewTest::SetUp();
}
// Creates a SearchResult with the given parameters.
void CreateSearchResult(ash::SearchResultDisplayType display_type,
double display_score,
const base::string16& title,
const base::string16& details) {
auto search_result = std::make_unique<SearchResult>();
search_result->set_display_type(display_type);
search_result->set_display_score(display_score);
search_result->set_title(title);
search_result->set_details(details);
results()->Add(std::move(search_result));
}
// Expect the entire autocomplete suggestion if |should_autocomplete| is true,
// expect only typed characters otherwise.
void ExpectAutocompleteSuggestion(bool should_autocomplete) {
if (should_autocomplete) {
// Search box autocomplete suggestion is accepted, but it should not
// trigger another query, thus it is not reflected in Search Model.
EXPECT_EQ(base::ASCIIToUTF16("hello world!"),
view()->search_box()->text());
EXPECT_EQ(base::ASCIIToUTF16("he"),
view_delegate()->GetSearchModel()->search_box()->text());
} else {
// Search box autocomplete suggestion is removed and is reflected in
// SearchModel.
EXPECT_EQ(view()->search_box()->text(),
view_delegate()->GetSearchModel()->search_box()->text());
EXPECT_EQ(base::ASCIIToUTF16("he"), view()->search_box()->text());
// ProcessAutocomplete should be a no-op.
view()->ProcessAutocomplete();
// The autocomplete suggestion should still not be present.
EXPECT_EQ(base::ASCIIToUTF16("he"), view()->search_box()->text());
}
}
// Sets up the test by creating a SearchResult and displaying an autocomplete
// suggestion.
void SetupAutocompleteBehaviorTest() {
// Add a search result with a non-empty title field.
CreateSearchResult(ash::SearchResultDisplayType::kList, 1.0,
base::ASCIIToUTF16("hello world!"), base::string16());
// Send H, E to the SearchBoxView textfield, then trigger an autocomplete.
KeyPress(ui::VKEY_H);
KeyPress(ui::VKEY_E);
view()->ProcessAutocomplete();
}
// Clears all existing text from search_box() and all existing SearchResults
// from results().
void ResetAutocompleteBehaviorTest() {
view()->search_box()->SetText(base::string16());
results()->RemoveAll();
}
// Test a GestureEvent's autocomplete behavior according to
// |should_autocomplete|. Expect the entire autocomplete suggestion if
// |should_autocomplete| is true, expect only typed characters otherwise.
void TestGestureEvent(const ui::GestureEvent& gesture_event,
bool should_autocomplete) {
SetupAutocompleteBehaviorTest();
// Forward |gesture_event| to HandleGestureEvent() directly because we
// cannot test GestureEvents properly due to not having ash dependencies.
// Static cast to TextfieldController because HandleGestureEvent() is
// private in SearchBoxView. TODO(crbug.com/878984): Derive
// SearchBoxViewTest from AshTestBase in order to test events using
// EventGenerator instead.
static_cast<views::TextfieldController*>(view())->HandleGestureEvent(
view()->search_box(), gesture_event);
ExpectAutocompleteSuggestion(should_autocomplete);
// Reset search box text and SearchResults for next test.
ResetAutocompleteBehaviorTest();
}
// Test a KeyEvent's autocomplete behavior according to |should_autocomplete|.
// Expect the entire autocomplete suggestion if |should_autocomplete| is true,
// expect only typed characters otherwise.
void TestKeyEvent(const ui::KeyEvent& key_event, bool should_autocomplete) {
SetupAutocompleteBehaviorTest();
// TODO(crbug.com/878984): Change KeyPress() to use EventGenerator::PressKey
// instead.
if (key_event.key_code() == ui::VKEY_BACK) {
// Use KeyPress() to mimic backspace. HandleKeyEvent() will not delete the
// text.
KeyPress(key_event.key_code());
} else {
// Forward |key_event| to HandleKeyEvent(). We use HandleKeyEvent()
// because KeyPress() will replace the existing highlighted text. Static
// cast to TextfieldController because HandleGestureEvent() is private in
// SearchBoxView.
static_cast<views::TextfieldController*>(view())->HandleKeyEvent(
view()->search_box(), key_event);
}
ExpectAutocompleteSuggestion(should_autocomplete);
// Reset search box text and SearchResults for next test.
ResetAutocompleteBehaviorTest();
}
// Test a MouseEvent's autocomplete behavior according to
// |should_autocomplete|. Expect the entire autocomplete suggestion if
// |should_autocomplete| is true, expect only typed characters otherwise.
void TestMouseEvent(const ui::MouseEvent& mouse_event,
bool should_autocomplete) {
SetupAutocompleteBehaviorTest();
// Forward |mouse_event| to HandleMouseEvent() directly because we cannot
// test MouseEvents properly due to not having ash dependencies. Static cast
// to TextfieldController because HandleGestureEvent() is a private method
// in SearchBoxView. TODO(crbug.com/878984): Derive SearchBoxViewTest from
// AshTestBase in order to test events using EventGenerator instead.
static_cast<views::TextfieldController*>(view())->HandleMouseEvent(
view()->search_box(), mouse_event);
ExpectAutocompleteSuggestion(should_autocomplete);
// Reset search box text and SearchResults for next test.
ResetAutocompleteBehaviorTest();
}
ui::KeyboardCode key_code() const { return GetParam(); }
private:
base::test::ScopedFeatureList scoped_feature_list_;
DISALLOW_COPY_AND_ASSIGN(SearchBoxViewAutocompleteTest);
};
INSTANTIATE_TEST_SUITE_P(,
SearchBoxViewAutocompleteTest,
::testing::Values(ui::VKEY_LEFT,
ui::VKEY_RIGHT,
ui::VKEY_UP,
ui::VKEY_DOWN,
ui::VKEY_BACK));
// Tests that autocomplete suggestions are consistent with top SearchResult list
// titles.
TEST_F(SearchBoxViewAutocompleteTest,
SearchBoxAutocompletesTopListResultTitle) {
// Add two SearchResults, one with higher ranking. Initialize their title
// field to a non-empty string.
CreateSearchResult(ash::SearchResultDisplayType::kList, 1.0,
base::ASCIIToUTF16("hello list"), base::string16());
CreateSearchResult(ash::SearchResultDisplayType::kTile, 0.5,
base::ASCIIToUTF16("hello tile"), base::string16());
// Send H, E to the SearchBoxView textfield, then trigger an autocomplete.
KeyPress(ui::VKEY_H);
KeyPress(ui::VKEY_E);
view()->ProcessAutocomplete();
EXPECT_EQ(view()->search_box()->text(), base::ASCIIToUTF16("hello list"));
EXPECT_EQ(view()->search_box()->GetSelectedText(),
base::ASCIIToUTF16("llo list"));
}
// Tests that autocomplete suggestions are consistent with top SearchResult tile
// titles.
TEST_F(SearchBoxViewAutocompleteTest,
SearchBoxAutocompletesTopTileResultTitle) {
// Add two SearchResults, one with higher ranking. Initialize their title
// field to a non-empty string.
CreateSearchResult(ash::SearchResultDisplayType::kTile, 1.0,
base::ASCIIToUTF16("hello tile"), base::string16());
CreateSearchResult(ash::SearchResultDisplayType::kList, 0.5,
base::ASCIIToUTF16("hello list"), base::string16());
// Send H, E to the SearchBoxView textfield, then trigger an autocomplete.
KeyPress(ui::VKEY_H);
KeyPress(ui::VKEY_E);
view()->ProcessAutocomplete();
EXPECT_EQ(view()->search_box()->text(), base::ASCIIToUTF16("hello tile"));
EXPECT_EQ(view()->search_box()->GetSelectedText(),
base::ASCIIToUTF16("llo tile"));
}
// Tests that autocomplete suggestions are consistent with top SearchResult list
// details.
TEST_F(SearchBoxViewAutocompleteTest,
SearchBoxAutocompletesTopListResultDetails) {
// Add two SearchResults, one with higher ranking. Initialize their details
// field to a non-empty string.
CreateSearchResult(ash::SearchResultDisplayType::kList, 1.0, base::string16(),
base::ASCIIToUTF16("hello list"));
CreateSearchResult(ash::SearchResultDisplayType::kTile, 0.5, base::string16(),
base::ASCIIToUTF16("hello tile"));
// Send H, E to the SearchBoxView textfield, then trigger an autocomplete.
KeyPress(ui::VKEY_H);
KeyPress(ui::VKEY_E);
view()->ProcessAutocomplete();
EXPECT_EQ(view()->search_box()->text(), base::ASCIIToUTF16("hello list"));
EXPECT_EQ(view()->search_box()->GetSelectedText(),
base::ASCIIToUTF16("llo list"));
}
// Tests that autocomplete suggestions are consistent with top SearchResult tile
// details.
TEST_F(SearchBoxViewAutocompleteTest,
SearchBoxAutocompletesTopTileResultDetails) {
// Add two SearchResults, one with higher ranking. Initialize their details
// field to a non-empty string.
CreateSearchResult(ash::SearchResultDisplayType::kTile, 1.0, base::string16(),
base::ASCIIToUTF16("hello tile"));
CreateSearchResult(ash::SearchResultDisplayType::kList, 0.5, base::string16(),
base::ASCIIToUTF16("hello list"));
// Send H, E to the SearchBoxView textfield, then trigger an autocomplete.
KeyPress(ui::VKEY_H);
KeyPress(ui::VKEY_E);
view()->ProcessAutocomplete();
EXPECT_EQ(view()->search_box()->text(), base::ASCIIToUTF16("hello tile"));
EXPECT_EQ(view()->search_box()->GetSelectedText(),
base::ASCIIToUTF16("llo tile"));
}
// Tests that SearchBoxView's textfield text does not autocomplete if the top
// result title or details do not have a matching prefix.
TEST_F(SearchBoxViewAutocompleteTest,
SearchBoxDoesNotAutocompleteWrongCharacter) {
// Add a search result with non-empty details and title fields.
CreateSearchResult(ash::SearchResultDisplayType::kList, 1.0,
base::ASCIIToUTF16("title"),
base::ASCIIToUTF16("details"));
// Send Z to the SearchBoxView textfield, then trigger an autocomplete.
KeyPress(ui::VKEY_Z);
view()->ProcessAutocomplete();
// The text should not be autocompleted.
EXPECT_EQ(view()->search_box()->text(), base::ASCIIToUTF16("z"));
}
// Tests that autocomplete suggestion will remain if next key in the suggestion
// is typed.
TEST_F(SearchBoxViewAutocompleteTest, SearchBoxAutocompletesAcceptsNextChar) {
// Add a search result with a non-empty title field.
CreateSearchResult(ash::SearchResultDisplayType::kList, 1.0,
base::ASCIIToUTF16("hello world!"), base::string16());
// Send H, E to the SearchBoxView textfield, then trigger an autocomplete.
KeyPress(ui::VKEY_H);
KeyPress(ui::VKEY_E);
view()->ProcessAutocomplete();
// After typing L, the highlighted text will be replaced by L.
KeyPress(ui::VKEY_L);
base::string16 selected_text = view()->search_box()->GetSelectedText();
EXPECT_EQ(view()->search_box()->text(), base::ASCIIToUTF16("hel"));
EXPECT_EQ(base::ASCIIToUTF16(""), selected_text);
// After handling autocomplete, the highlighted text will show again.
view()->ProcessAutocomplete();
selected_text = view()->search_box()->GetSelectedText();
EXPECT_EQ(view()->search_box()->text(), base::ASCIIToUTF16("hello world!"));
EXPECT_EQ(base::ASCIIToUTF16("lo world!"), selected_text);
}
// Tests that autocomplete suggestion is accepted and displayed in SearchModel
// after clicking or tapping on the search box.
TEST_F(SearchBoxViewAutocompleteTest, SearchBoxAcceptsAutocompleteForClickTap) {
TestMouseEvent(ui::MouseEvent(ui::ET_MOUSE_PRESSED, gfx::Point(),
gfx::Point(), ui::EventTimeForNow(), 0, 0),
true);
TestGestureEvent(
ui::GestureEvent(0, 0, 0, ui::EventTimeForNow(),
ui::GestureEventDetails(ui::ET_GESTURE_TAP)),
true);
}
// Tests that autocomplete is not handled if IME is using composition text.
TEST_F(SearchBoxViewAutocompleteTest, SearchBoxAutocompletesNotHandledForIME) {
// Add a search result with a non-empty title field.
CreateSearchResult(ash::SearchResultDisplayType::kList, 1.0,
base::ASCIIToUTF16("hello world!"), base::string16());
// Simulate uncomposited text. The autocomplete should be handled.
view()->search_box()->SetText(base::ASCIIToUTF16("he"));
view()->set_highlight_range_for_test(gfx::Range(2, 2));
view()->ProcessAutocomplete();
base::string16 selected_text = view()->search_box()->GetSelectedText();
EXPECT_EQ(view()->search_box()->text(), base::ASCIIToUTF16("hello world!"));
EXPECT_EQ(base::ASCIIToUTF16("llo world!"), selected_text);
view()->search_box()->SetText(base::string16());
// Simulate IME composition text. The autocomplete should not be handled.
ui::CompositionText composition_text;
composition_text.text = base::ASCIIToUTF16("he");
view()->search_box()->SetCompositionText(composition_text);
view()->set_highlight_range_for_test(gfx::Range(2, 2));
view()->ProcessAutocomplete();
selected_text = view()->search_box()->GetSelectedText();
EXPECT_EQ(view()->search_box()->text(), base::ASCIIToUTF16("he"));
EXPECT_EQ(base::ASCIIToUTF16(""), selected_text);
}
} // namespace test
} // namespace app_list