| // Copyright 2013 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "components/page_info/android/page_info_controller_android.h" |
| #include <string> |
| |
| #include "base/android/jni_android.h" |
| #include "base/android/jni_array.h" |
| #include "base/android/jni_string.h" |
| #include "base/command_line.h" |
| #include "base/containers/contains.h" |
| #include "components/content_settings/core/browser/host_content_settings_map.h" |
| #include "components/content_settings/core/common/content_settings.h" |
| #include "components/content_settings/core/common/content_settings_types.h" |
| #include "components/page_info/android/jni_headers/PageInfoController_jni.h" |
| #include "components/page_info/android/page_info_client.h" |
| #include "components/page_info/core/features.h" |
| #include "components/page_info/page_info.h" |
| #include "components/page_info/page_info_ui.h" |
| #include "components/security_state/core/security_state.h" |
| #include "content/public/browser/browser_context.h" |
| #include "content/public/browser/navigation_controller.h" |
| #include "content/public/browser/navigation_entry.h" |
| #include "content/public/browser/web_contents.h" |
| #include "content/public/common/content_features.h" |
| #include "content/public/common/content_switches.h" |
| #include "url/origin.h" |
| |
| using base::android::ConvertUTF16ToJavaString; |
| using base::android::ConvertUTF8ToJavaString; |
| using base::android::JavaParamRef; |
| |
| // static |
| static jlong JNI_PageInfoController_Init( |
| JNIEnv* env, |
| const JavaParamRef<jobject>& obj, |
| const JavaParamRef<jobject>& java_web_contents) { |
| content::WebContents* web_contents = |
| content::WebContents::FromJavaWebContents(java_web_contents); |
| |
| // Important to use GetVisibleEntry to match what's showing in the omnibox. |
| content::NavigationEntry* nav_entry = |
| web_contents->GetController().GetVisibleEntry(); |
| if (nav_entry->IsInitialEntry()) |
| return 0; |
| |
| return reinterpret_cast<intptr_t>( |
| new PageInfoControllerAndroid(env, obj, web_contents)); |
| } |
| |
| PageInfoControllerAndroid::PageInfoControllerAndroid( |
| JNIEnv* env, |
| jobject java_page_info_pop, |
| content::WebContents* web_contents) { |
| content::NavigationEntry* nav_entry = |
| web_contents->GetController().GetVisibleEntry(); |
| |
| url_ = nav_entry->GetURL(); |
| web_contents_ = web_contents; |
| |
| controller_jobject_.Reset(env, java_page_info_pop); |
| |
| page_info::PageInfoClient* page_info_client = page_info::GetPageInfoClient(); |
| DCHECK(page_info_client); |
| presenter_ = std::make_unique<PageInfo>( |
| page_info_client->CreatePageInfoDelegate(web_contents), web_contents, |
| nav_entry->GetURL()); |
| presenter_->InitializeUiState(this, base::DoNothing()); |
| } |
| |
| PageInfoControllerAndroid::~PageInfoControllerAndroid() = default; |
| |
| void PageInfoControllerAndroid::Destroy(JNIEnv* env, |
| const JavaParamRef<jobject>& obj) { |
| delete this; |
| } |
| |
| void PageInfoControllerAndroid::RecordPageInfoAction( |
| JNIEnv* env, |
| const JavaParamRef<jobject>& obj, |
| jint action) { |
| presenter_->RecordPageInfoAction( |
| static_cast<PageInfo::PageInfoAction>(action)); |
| } |
| |
| void PageInfoControllerAndroid::UpdatePermissions( |
| JNIEnv* env, |
| const JavaParamRef<jobject>& obj) { |
| presenter_->UpdatePermissions(); |
| } |
| |
| void PageInfoControllerAndroid::SetIdentityInfo( |
| const IdentityInfo& identity_info) { |
| JNIEnv* env = base::android::AttachCurrentThread(); |
| std::unique_ptr<PageInfoUI::SecurityDescription> security_description = |
| GetSecurityDescription(identity_info); |
| |
| Java_PageInfoController_setSecurityDescription( |
| env, controller_jobject_, |
| ConvertUTF16ToJavaString(env, security_description->summary), |
| ConvertUTF16ToJavaString(env, security_description->details)); |
| } |
| |
| void PageInfoControllerAndroid::SetCookieInfo( |
| const CookieInfoList& cookie_info_list) { |
| NOTIMPLEMENTED(); |
| } |
| |
| void PageInfoControllerAndroid::SetPageFeatureInfo( |
| const PageFeatureInfo& info) { |
| NOTIMPLEMENTED(); |
| } |
| |
| void PageInfoControllerAndroid::SetPermissionInfo( |
| const PermissionInfoList& permission_info_list, |
| ChosenObjectInfoList chosen_object_info_list) { |
| JNIEnv* env = base::android::AttachCurrentThread(); |
| |
| // Exit without permissions if it is an internal page. |
| if (PageInfo::IsFileOrInternalPage(url_)) { |
| Java_PageInfoController_updatePermissionDisplay(env, controller_jobject_); |
| return; |
| } |
| |
| // On Android, we only want to display a subset of the available options in |
| // a particular order, but only if their value is different from the |
| // default. This order comes from https://crbug.com/610358. |
| std::vector<ContentSettingsType> permissions_to_display; |
| permissions_to_display.push_back(ContentSettingsType::GEOLOCATION); |
| permissions_to_display.push_back(ContentSettingsType::MEDIASTREAM_CAMERA); |
| permissions_to_display.push_back(ContentSettingsType::MEDIASTREAM_MIC); |
| permissions_to_display.push_back(ContentSettingsType::NOTIFICATIONS); |
| permissions_to_display.push_back(ContentSettingsType::IDLE_DETECTION); |
| permissions_to_display.push_back(ContentSettingsType::IMAGES); |
| permissions_to_display.push_back(ContentSettingsType::JAVASCRIPT); |
| permissions_to_display.push_back(ContentSettingsType::POPUPS); |
| permissions_to_display.push_back(ContentSettingsType::ADS); |
| permissions_to_display.push_back( |
| ContentSettingsType::PROTECTED_MEDIA_IDENTIFIER); |
| permissions_to_display.push_back(ContentSettingsType::SOUND); |
| if (base::FeatureList::IsEnabled(features::kWebNfc)) |
| permissions_to_display.push_back(ContentSettingsType::NFC); |
| base::CommandLine* cmd = base::CommandLine::ForCurrentProcess(); |
| if (cmd->HasSwitch(switches::kEnableExperimentalWebPlatformFeatures)) |
| permissions_to_display.push_back(ContentSettingsType::BLUETOOTH_SCANNING); |
| permissions_to_display.push_back(ContentSettingsType::VR); |
| permissions_to_display.push_back(ContentSettingsType::AR); |
| if (base::FeatureList::IsEnabled(features::kFedCm)) { |
| permissions_to_display.push_back( |
| ContentSettingsType::FEDERATED_IDENTITY_API); |
| } |
| |
| std::map<ContentSettingsType, ContentSetting> |
| user_specified_settings_to_display; |
| |
| for (const auto& permission : permission_info_list) { |
| if (base::Contains(permissions_to_display, permission.type)) { |
| absl::optional<ContentSetting> setting_to_display = |
| GetSettingToDisplay(permission); |
| if (setting_to_display) { |
| user_specified_settings_to_display[permission.type] = |
| *setting_to_display; |
| } |
| } |
| } |
| |
| for (const auto& permission : permissions_to_display) { |
| if (base::Contains(user_specified_settings_to_display, permission)) { |
| std::u16string setting_title = |
| PageInfoUI::PermissionTypeToUIString(permission); |
| std::u16string setting_title_mid_sentence = |
| PageInfoUI::PermissionTypeToUIStringMidSentence(permission); |
| |
| Java_PageInfoController_addPermissionSection( |
| env, controller_jobject_, |
| ConvertUTF16ToJavaString(env, setting_title), |
| ConvertUTF16ToJavaString(env, setting_title_mid_sentence), |
| static_cast<jint>(permission), |
| static_cast<jint>(user_specified_settings_to_display[permission])); |
| } |
| } |
| |
| for (const auto& chosen_object : chosen_object_info_list) { |
| std::u16string object_title = |
| presenter_->GetChooserContextFromUIInfo(*chosen_object->ui_info) |
| ->GetObjectDisplayName(chosen_object->chooser_object->value); |
| |
| Java_PageInfoController_addPermissionSection( |
| env, controller_jobject_, ConvertUTF16ToJavaString(env, object_title), |
| ConvertUTF16ToJavaString(env, object_title), |
| static_cast<jint>(chosen_object->ui_info->content_settings_type), |
| static_cast<jint>(CONTENT_SETTING_ALLOW)); |
| } |
| |
| Java_PageInfoController_updatePermissionDisplay(env, controller_jobject_); |
| } |
| |
| absl::optional<ContentSetting> PageInfoControllerAndroid::GetSettingToDisplay( |
| const PageInfo::PermissionInfo& permission) { |
| // All permissions should be displayed if they are non-default. |
| if (permission.setting != CONTENT_SETTING_DEFAULT && |
| permission.setting != permission.default_setting) { |
| return permission.setting; |
| } |
| |
| // Handle exceptions for permissions which need to be displayed even if they |
| // are set to the default. |
| if (permission.type == ContentSettingsType::ADS) { |
| // The subresource filter permission should always display the default |
| // setting if it is showing up in Page Info. Logic for whether the |
| // setting should show up in Page Info is in ShouldShowPermission in |
| // page_info.cc. |
| return permission.default_setting; |
| } else if (permission.type == ContentSettingsType::JAVASCRIPT) { |
| // The javascript content setting should show up if it is blocked globally |
| // to give users an easy way to create exceptions. |
| return permission.default_setting; |
| } else if (permission.type == ContentSettingsType::SOUND) { |
| // The sound content setting should always show up when the tab has played |
| // audio since last navigation. |
| if (web_contents_->WasEverAudible()) |
| return permission.default_setting; |
| } |
| |
| // TODO(crbug.com/1077766): Also return permissions that are non |
| // factory-default after we add the functionality to populate the permissions |
| // subpage directly from the permissions returned from this controller. |
| |
| return absl::optional<ContentSetting>(); |
| } |
| |
| void PageInfoControllerAndroid::SetAdPersonalizationInfo( |
| const AdPersonalizationInfo& info) { |
| JNIEnv* env = base::android::AttachCurrentThread(); |
| std::vector<std::u16string> topic_names; |
| for (const auto& topic : info.accessed_topics) { |
| topic_names.push_back(topic.GetLocalizedRepresentation()); |
| } |
| Java_PageInfoController_setAdPersonalizationInfo( |
| env, controller_jobject_, info.has_joined_user_to_interest_group, |
| base::android::ToJavaArrayOfStrings(env, topic_names)); |
| } |