| // 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 "android_webview/browser/aw_autofill_client.h" |
| |
| #include <utility> |
| |
| #include "android_webview/browser/aw_browser_context.h" |
| #include "android_webview/browser/aw_content_browser_client.h" |
| #include "android_webview/browser/aw_contents.h" |
| #include "android_webview/browser/aw_form_database_service.h" |
| #include "android_webview/browser_jni_headers/AwAutofillClient_jni.h" |
| #include "base/android/jni_android.h" |
| #include "base/android/jni_string.h" |
| #include "base/android/scoped_java_ref.h" |
| #include "base/notreached.h" |
| #include "components/autofill/core/browser/payments/legal_message_line.h" |
| #include "components/autofill/core/browser/ui/autofill_popup_delegate.h" |
| #include "components/autofill/core/browser/ui/suggestion.h" |
| #include "components/autofill/core/browser/webdata/autofill_webdata_service.h" |
| #include "components/prefs/pref_registry_simple.h" |
| #include "components/prefs/pref_service.h" |
| #include "components/prefs/pref_service_factory.h" |
| #include "components/user_prefs/user_prefs.h" |
| #include "content/public/browser/navigation_entry.h" |
| #include "content/public/browser/ssl_status.h" |
| #include "content/public/browser/web_contents.h" |
| #include "ui/android/view_android.h" |
| #include "ui/gfx/geometry/rect_f.h" |
| |
| using base::android::AttachCurrentThread; |
| using base::android::ConvertUTF16ToJavaString; |
| using base::android::JavaParamRef; |
| using base::android::ScopedJavaLocalRef; |
| using content::WebContents; |
| |
| namespace android_webview { |
| |
| AwAutofillClient::~AwAutofillClient() { |
| HideAutofillPopup(autofill::PopupHidingReason::kTabGone); |
| } |
| |
| void AwAutofillClient::SetSaveFormData(bool enabled) { |
| save_form_data_ = enabled; |
| } |
| |
| bool AwAutofillClient::GetSaveFormData() { |
| return save_form_data_; |
| } |
| |
| autofill::PersonalDataManager* AwAutofillClient::GetPersonalDataManager() { |
| return nullptr; |
| } |
| |
| autofill::AutocompleteHistoryManager* |
| AwAutofillClient::GetAutocompleteHistoryManager() { |
| return AwBrowserContext::FromWebContents(web_contents_) |
| ->GetAutocompleteHistoryManager(); |
| } |
| |
| PrefService* AwAutofillClient::GetPrefs() { |
| return const_cast<PrefService*>(base::as_const(*this).GetPrefs()); |
| } |
| |
| const PrefService* AwAutofillClient::GetPrefs() const { |
| return user_prefs::UserPrefs::Get( |
| AwBrowserContext::FromWebContents(web_contents_)); |
| } |
| |
| syncer::SyncService* AwAutofillClient::GetSyncService() { |
| return nullptr; |
| } |
| |
| signin::IdentityManager* AwAutofillClient::GetIdentityManager() { |
| return nullptr; |
| } |
| |
| autofill::FormDataImporter* AwAutofillClient::GetFormDataImporter() { |
| return nullptr; |
| } |
| |
| autofill::payments::PaymentsClient* AwAutofillClient::GetPaymentsClient() { |
| return nullptr; |
| } |
| |
| autofill::StrikeDatabase* AwAutofillClient::GetStrikeDatabase() { |
| return nullptr; |
| } |
| |
| ukm::UkmRecorder* AwAutofillClient::GetUkmRecorder() { |
| return nullptr; |
| } |
| |
| ukm::SourceId AwAutofillClient::GetUkmSourceId() { |
| // UKM recording is not supported for WebViews. |
| return ukm::kInvalidSourceId; |
| } |
| |
| autofill::AddressNormalizer* AwAutofillClient::GetAddressNormalizer() { |
| return nullptr; |
| } |
| |
| const GURL& AwAutofillClient::GetLastCommittedURL() const { |
| return web_contents_->GetLastCommittedURL(); |
| } |
| |
| security_state::SecurityLevel |
| AwAutofillClient::GetSecurityLevelForUmaHistograms() { |
| // The metrics are not recorded for Android webview, so return the count value |
| // which will not be recorded. |
| return security_state::SecurityLevel::SECURITY_LEVEL_COUNT; |
| } |
| |
| const translate::LanguageState* AwAutofillClient::GetLanguageState() { |
| return nullptr; |
| } |
| |
| translate::TranslateDriver* AwAutofillClient::GetTranslateDriver() { |
| return nullptr; |
| } |
| |
| void AwAutofillClient::ShowAutofillSettings(bool show_credit_card_settings) { |
| NOTIMPLEMENTED(); |
| } |
| |
| void AwAutofillClient::ShowUnmaskPrompt( |
| const autofill::CreditCard& card, |
| UnmaskCardReason reason, |
| base::WeakPtr<autofill::CardUnmaskDelegate> delegate) { |
| NOTIMPLEMENTED(); |
| } |
| |
| void AwAutofillClient::OnUnmaskVerificationResult(PaymentsRpcResult result) { |
| NOTIMPLEMENTED(); |
| } |
| |
| void AwAutofillClient::ConfirmAccountNameFixFlow( |
| base::OnceCallback<void(const std::u16string&)> callback) { |
| NOTIMPLEMENTED(); |
| } |
| |
| void AwAutofillClient::ConfirmExpirationDateFixFlow( |
| const autofill::CreditCard& card, |
| base::OnceCallback<void(const std::u16string&, const std::u16string&)> |
| callback) { |
| NOTIMPLEMENTED(); |
| } |
| |
| void AwAutofillClient::ConfirmSaveCreditCardLocally( |
| const autofill::CreditCard& card, |
| SaveCreditCardOptions options, |
| LocalSaveCardPromptCallback callback) { |
| NOTIMPLEMENTED(); |
| } |
| |
| void AwAutofillClient::ConfirmSaveCreditCardToCloud( |
| const autofill::CreditCard& card, |
| const autofill::LegalMessageLines& legal_message_lines, |
| SaveCreditCardOptions options, |
| UploadSaveCardPromptCallback callback) { |
| NOTIMPLEMENTED(); |
| } |
| |
| void AwAutofillClient::CreditCardUploadCompleted(bool card_saved) { |
| NOTIMPLEMENTED(); |
| } |
| |
| void AwAutofillClient::ConfirmCreditCardFillAssist( |
| const autofill::CreditCard& card, |
| base::OnceClosure callback) { |
| NOTIMPLEMENTED(); |
| } |
| |
| void AwAutofillClient::ConfirmSaveAddressProfile( |
| const autofill::AutofillProfile& profile, |
| const autofill::AutofillProfile* original_profile, |
| SaveAddressProfilePromptOptions options, |
| AddressProfileSavePromptCallback callback) { |
| NOTIMPLEMENTED(); |
| } |
| |
| bool AwAutofillClient::HasCreditCardScanFeature() { |
| return false; |
| } |
| |
| void AwAutofillClient::ScanCreditCard(CreditCardScanCallback callback) { |
| NOTIMPLEMENTED(); |
| } |
| |
| void AwAutofillClient::ShowAutofillPopup( |
| const autofill::AutofillClient::PopupOpenArgs& open_args, |
| base::WeakPtr<autofill::AutofillPopupDelegate> delegate) { |
| suggestions_ = open_args.suggestions; |
| delegate_ = delegate; |
| |
| // Convert element_bounds to be in screen space. |
| gfx::Rect client_area = web_contents_->GetContainerBounds(); |
| gfx::RectF element_bounds_in_screen_space = |
| open_args.element_bounds + client_area.OffsetFromOrigin(); |
| |
| ShowAutofillPopupImpl(element_bounds_in_screen_space, |
| open_args.text_direction == base::i18n::RIGHT_TO_LEFT, |
| open_args.suggestions); |
| } |
| |
| void AwAutofillClient::UpdateAutofillPopupDataListValues( |
| const std::vector<std::u16string>& values, |
| const std::vector<std::u16string>& labels) { |
| // Leaving as an empty method since updating autofill popup window |
| // dynamically does not seem to be a useful feature for android webview. |
| // See crrev.com/18102002 if need to implement. |
| } |
| |
| base::span<const autofill::Suggestion> AwAutofillClient::GetPopupSuggestions() |
| const { |
| NOTIMPLEMENTED(); |
| return base::span<const autofill::Suggestion>(); |
| } |
| |
| void AwAutofillClient::PinPopupView() { |
| NOTIMPLEMENTED(); |
| } |
| |
| autofill::AutofillClient::PopupOpenArgs AwAutofillClient::GetReopenPopupArgs() |
| const { |
| NOTIMPLEMENTED(); |
| return {}; |
| } |
| |
| void AwAutofillClient::UpdatePopup( |
| const std::vector<autofill::Suggestion>& suggestions, |
| autofill::PopupType popup_type) { |
| NOTIMPLEMENTED(); |
| } |
| |
| void AwAutofillClient::HideAutofillPopup(autofill::PopupHidingReason reason) { |
| JNIEnv* env = AttachCurrentThread(); |
| ScopedJavaLocalRef<jobject> obj = java_ref_.get(env); |
| if (!obj) |
| return; |
| delegate_.reset(); |
| Java_AwAutofillClient_hideAutofillPopup(env, obj); |
| } |
| |
| bool AwAutofillClient::IsAutocompleteEnabled() { |
| return GetSaveFormData(); |
| } |
| |
| void AwAutofillClient::PropagateAutofillPredictions( |
| content::RenderFrameHost* rfh, |
| const std::vector<autofill::FormStructure*>& forms) {} |
| |
| void AwAutofillClient::DidFillOrPreviewField( |
| const std::u16string& autofilled_value, |
| const std::u16string& profile_full_name) {} |
| |
| bool AwAutofillClient::IsContextSecure() const { |
| content::SSLStatus ssl_status; |
| content::NavigationEntry* navigation_entry = |
| web_contents_->GetController().GetLastCommittedEntry(); |
| if (!navigation_entry) |
| return false; |
| |
| ssl_status = navigation_entry->GetSSL(); |
| // Note: As of crbug.com/701018, Chrome relies on SecurityStateTabHelper to |
| // determine whether the page is secure, but WebView has no equivalent class. |
| |
| return navigation_entry->GetURL().SchemeIsCryptographic() && |
| ssl_status.certificate && |
| !net::IsCertStatusError(ssl_status.cert_status) && |
| !(ssl_status.content_status & |
| content::SSLStatus::RAN_INSECURE_CONTENT); |
| } |
| |
| bool AwAutofillClient::ShouldShowSigninPromo() { |
| return false; |
| } |
| |
| bool AwAutofillClient::AreServerCardsSupported() const { |
| return true; |
| } |
| |
| void AwAutofillClient::ExecuteCommand(int id) { |
| NOTIMPLEMENTED(); |
| } |
| |
| void AwAutofillClient::LoadRiskData( |
| base::OnceCallback<void(const std::string&)> callback) { |
| NOTIMPLEMENTED(); |
| } |
| |
| void AwAutofillClient::Dismissed(JNIEnv* env, |
| const JavaParamRef<jobject>& obj) { |
| anchor_view_.Reset(); |
| } |
| |
| void AwAutofillClient::SuggestionSelected(JNIEnv* env, |
| const JavaParamRef<jobject>& object, |
| jint position) { |
| if (delegate_) { |
| delegate_->DidAcceptSuggestion(suggestions_[position].value, |
| suggestions_[position].frontend_id, |
| suggestions_[position].backend_id, position); |
| } |
| } |
| |
| // Ownership: The native object is created (if autofill enabled) and owned by |
| // AwContents. The native object creates the java peer which handles most |
| // autofill functionality at the java side. The java peer is owned by Java |
| // AwContents. The native object only maintains a weak ref to it. |
| AwAutofillClient::AwAutofillClient(WebContents* contents) |
| : web_contents_(contents) { |
| JNIEnv* env = AttachCurrentThread(); |
| ScopedJavaLocalRef<jobject> delegate; |
| delegate.Reset( |
| Java_AwAutofillClient_create(env, reinterpret_cast<intptr_t>(this))); |
| |
| AwContents* aw_contents = AwContents::FromWebContents(web_contents_); |
| aw_contents->SetAwAutofillClient(delegate); |
| java_ref_ = JavaObjectWeakGlobalRef(env, delegate); |
| } |
| |
| void AwAutofillClient::ShowAutofillPopupImpl( |
| const gfx::RectF& element_bounds, |
| bool is_rtl, |
| const std::vector<autofill::Suggestion>& suggestions) { |
| JNIEnv* env = AttachCurrentThread(); |
| ScopedJavaLocalRef<jobject> obj = java_ref_.get(env); |
| if (!obj) |
| return; |
| |
| // We need an array of AutofillSuggestion. |
| size_t count = suggestions.size(); |
| |
| ScopedJavaLocalRef<jobjectArray> data_array = |
| Java_AwAutofillClient_createAutofillSuggestionArray(env, count); |
| |
| for (size_t i = 0; i < count; ++i) { |
| ScopedJavaLocalRef<jstring> name = |
| ConvertUTF16ToJavaString(env, suggestions[i].value); |
| ScopedJavaLocalRef<jstring> label = |
| ConvertUTF16ToJavaString(env, suggestions[i].label); |
| Java_AwAutofillClient_addToAutofillSuggestionArray( |
| env, data_array, i, name, label, suggestions[i].frontend_id); |
| } |
| ui::ViewAndroid* view_android = web_contents_->GetNativeView(); |
| if (!view_android) |
| return; |
| |
| const ScopedJavaLocalRef<jobject> current_view = anchor_view_.view(); |
| if (!current_view) |
| anchor_view_ = view_android->AcquireAnchorView(); |
| |
| const ScopedJavaLocalRef<jobject> view = anchor_view_.view(); |
| if (!view) |
| return; |
| |
| view_android->SetAnchorRect(view, element_bounds); |
| Java_AwAutofillClient_showAutofillPopup(env, obj, view, is_rtl, data_array); |
| } |
| |
| WEB_CONTENTS_USER_DATA_KEY_IMPL(AwAutofillClient) |
| |
| } // namespace android_webview |