blob: 16d1de91b141eb4e28c16253ad62473155eab04c [file] [log] [blame]
// Copyright 2019 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/keyboard_accessory/android/address_accessory_controller_impl.h"
#include <algorithm>
#include <memory>
#include <ostream>
#include <string>
#include <vector>
#include "base/strings/utf_string_conversions.h"
#include "base/test/mock_callback.h"
#include "chrome/browser/autofill/personal_data_manager_factory.h"
#include "chrome/browser/keyboard_accessory/android/accessory_controller.h"
#include "chrome/browser/keyboard_accessory/test_utils/android/mock_manual_filling_controller.h"
#include "chrome/grit/generated_resources.h"
#include "chrome/test/base/chrome_render_view_host_test_harness.h"
#include "chrome/test/base/testing_profile.h"
#include "components/autofill/core/browser/address_data_manager.h"
#include "components/autofill/core/browser/autofill_test_utils.h"
#include "components/autofill/core/browser/test_personal_data_manager.h"
#include "components/strings/grit/components_strings.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/base/l10n/l10n_util.h"
namespace autofill {
namespace {
using autofill::UserInfo;
using base::ASCIIToUTF16;
using testing::_;
using testing::ByMove;
using testing::Mock;
using testing::Return;
using testing::SaveArg;
using testing::StrictMock;
using FillingSource = ManualFillingController::FillingSource;
using IsFillingSourceAvailable = AccessoryController::IsFillingSourceAvailable;
std::u16string addresses_empty_str() {
return l10n_util::GetStringUTF16(IDS_AUTOFILL_ADDRESS_SHEET_EMPTY_MESSAGE);
}
std::u16string manage_addresses_str() {
return l10n_util::GetStringUTF16(
IDS_AUTOFILL_ADDRESS_SHEET_ALL_ADDRESSES_LINK);
}
// Creates a AccessorySheetData::Builder with a "Manage Addresses" footer.
AccessorySheetData::Builder AddressAccessorySheetDataBuilder(
const std::u16string& title) {
return AccessorySheetData::Builder(AccessoryTabType::ADDRESSES, title)
.AppendFooterCommand(manage_addresses_str(),
AccessoryAction::MANAGE_ADDRESSES);
}
std::unique_ptr<KeyedService> BuildTestPersonalDataManager(
content::BrowserContext* context) {
auto personal_data_manager = std::make_unique<TestPersonalDataManager>();
personal_data_manager->SetAutofillProfileEnabled(true);
return personal_data_manager;
}
} // namespace
class AddressAccessoryControllerTest : public ChromeRenderViewHostTestHarness {
public:
AddressAccessoryControllerTest() {}
void SetUp() override {
ChromeRenderViewHostTestHarness::SetUp();
PersonalDataManagerFactory::GetInstance()->SetTestingFactory(
GetBrowserContext(),
base::BindRepeating(&BuildTestPersonalDataManager));
AddressAccessoryControllerImpl::CreateForWebContentsForTesting(
web_contents(), mock_manual_filling_controller_.AsWeakPtr());
}
void TearDown() override {
personal_data_manager()->ClearProfiles();
ChromeRenderViewHostTestHarness::TearDown();
}
AddressAccessoryController* controller() {
return AddressAccessoryControllerImpl::FromWebContents(web_contents());
}
TestPersonalDataManager* personal_data_manager() {
return static_cast<TestPersonalDataManager*>(
PersonalDataManagerFactory::GetForProfile(profile()));
}
protected:
StrictMock<MockManualFillingController> mock_manual_filling_controller_;
base::MockCallback<AccessoryController::FillingSourceObserver>
filling_source_observer_;
};
TEST_F(AddressAccessoryControllerTest, IsNotRecreatedForSameWebContents) {
AddressAccessoryControllerImpl* initial_controller =
AddressAccessoryControllerImpl::FromWebContents(web_contents());
EXPECT_NE(nullptr, initial_controller);
AddressAccessoryControllerImpl::CreateForWebContents(web_contents());
EXPECT_EQ(AddressAccessoryControllerImpl::FromWebContents(web_contents()),
initial_controller);
}
TEST_F(AddressAccessoryControllerTest, ProvidesNoSheetBeforeInitialRefresh) {
AutofillProfile canadian = test::GetFullValidProfileForCanada();
personal_data_manager()->address_data_manager().AddProfile(canadian);
controller()->RegisterFillingSourceObserver(filling_source_observer_.Get());
EXPECT_FALSE(controller()->GetSheetData().has_value());
EXPECT_CALL(filling_source_observer_,
Run(controller(), IsFillingSourceAvailable(true)));
controller()->RefreshSuggestions();
EXPECT_TRUE(controller()->GetSheetData().has_value());
}
TEST_F(AddressAccessoryControllerTest, RefreshSuggestionsCallsUI) {
AutofillProfile canadian = test::GetFullValidProfileForCanada();
personal_data_manager()->address_data_manager().AddProfile(canadian);
AccessorySheetData result(AccessoryTabType::PASSWORDS, std::u16string());
EXPECT_CALL(mock_manual_filling_controller_, RefreshSuggestions(_))
.WillOnce(SaveArg<0>(&result));
controller()->RefreshSuggestions();
EXPECT_EQ(result, controller()->GetSheetData());
EXPECT_EQ(
result,
AddressAccessorySheetDataBuilder(std::u16string())
.AddUserInfo()
.AppendSimpleField(canadian.GetRawInfo(FieldType::NAME_FULL))
.AppendSimpleField(canadian.GetRawInfo(FieldType::COMPANY_NAME))
.AppendSimpleField(canadian.GetRawInfo(FieldType::ADDRESS_HOME_LINE1))
.AppendSimpleField(canadian.GetRawInfo(FieldType::ADDRESS_HOME_LINE2))
.AppendSimpleField(canadian.GetRawInfo(FieldType::ADDRESS_HOME_ZIP))
.AppendSimpleField(canadian.GetRawInfo(FieldType::ADDRESS_HOME_CITY))
.AppendSimpleField(canadian.GetRawInfo(FieldType::ADDRESS_HOME_STATE))
.AppendSimpleField(
canadian.GetRawInfo(FieldType::ADDRESS_HOME_COUNTRY))
.AppendSimpleField(
canadian.GetRawInfo(FieldType::PHONE_HOME_WHOLE_NUMBER))
.AppendSimpleField(canadian.GetRawInfo(FieldType::EMAIL_ADDRESS))
.Build());
}
TEST_F(AddressAccessoryControllerTest, ProvidesEmptySuggestionsMessage) {
AccessorySheetData result(AccessoryTabType::PASSWORDS, std::u16string());
EXPECT_CALL(mock_manual_filling_controller_, RefreshSuggestions(_))
.WillOnce(SaveArg<0>(&result));
controller()->RefreshSuggestions();
EXPECT_EQ(result, controller()->GetSheetData());
EXPECT_EQ(result,
AddressAccessorySheetDataBuilder(addresses_empty_str()).Build());
}
TEST_F(AddressAccessoryControllerTest, TriggersRefreshWhenDataChanges) {
AccessorySheetData result(AccessoryTabType::PASSWORDS, std::u16string());
EXPECT_CALL(mock_manual_filling_controller_, RefreshSuggestions(_))
.WillRepeatedly(SaveArg<0>(&result));
// A refresh without data stores an empty sheet and registers an observer.
controller()->RefreshSuggestions();
EXPECT_EQ(result, controller()->GetSheetData());
EXPECT_EQ(result,
AddressAccessorySheetDataBuilder(addresses_empty_str()).Build());
// When new data is added, a refresh is automatically triggered.
AutofillProfile email = test::GetIncompleteProfile2();
personal_data_manager()->address_data_manager().AddProfile(email);
EXPECT_EQ(result, controller()->GetSheetData());
EXPECT_EQ(result,
AddressAccessorySheetDataBuilder(std::u16string())
.AddUserInfo()
/*name full:*/
.AppendSimpleField(std::u16string())
/*company name:*/
.AppendSimpleField(std::u16string())
/*address line1:*/
.AppendSimpleField(std::u16string())
/*address line2:*/
.AppendSimpleField(std::u16string())
/*address zip:*/
.AppendSimpleField(std::u16string())
/*address city:*/
.AppendSimpleField(std::u16string())
/*address state:*/
.AppendSimpleField(std::u16string())
/*address country:*/
.AppendSimpleField(std::u16string())
/*phone number:*/.AppendSimpleField(std::u16string())
.AppendSimpleField(email.GetRawInfo(FieldType::EMAIL_ADDRESS))
.Build());
}
} // namespace autofill