|  | // Copyright 2015 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 <utility> | 
|  | #include <vector> | 
|  |  | 
|  | #include "base/feature_list.h" | 
|  | #include "base/hash/hash.h" | 
|  | #include "base/logging.h" | 
|  | #include "base/notreached.h" | 
|  | #include "base/strings/string_util.h" | 
|  | #include "base/strings/utf_string_conversions.h" | 
|  | #include "build/build_config.h" | 
|  | #include "chrome/app/vector_icons/vector_icons.h" | 
|  | #include "chrome/browser/browser_process.h" | 
|  | #include "chrome/browser/profiles/profile_attributes_entry.h" | 
|  | #include "chrome/browser/profiles/profile_attributes_storage.h" | 
|  | #include "chrome/browser/profiles/profile_avatar_icon_util.h" | 
|  | #include "chrome/browser/profiles/profiles_state.h" | 
|  | #include "chrome/browser/signin/signin_util.h" | 
|  | #include "chrome/browser/ui/signin/profile_colors_util.h" | 
|  | #include "chrome/browser/ui/ui_features.h" | 
|  | #include "chrome/common/pref_names.h" | 
|  | #include "components/policy/core/browser/browser_policy_connector.h" | 
|  | #include "components/prefs/pref_registry_simple.h" | 
|  | #include "components/prefs/pref_service.h" | 
|  | #include "components/prefs/scoped_user_pref_update.h" | 
|  | #include "components/profile_metrics/state.h" | 
|  | #include "components/signin/public/base/signin_pref_names.h" | 
|  | #include "components/signin/public/identity_manager/account_info.h" | 
|  | #include "third_party/abseil-cpp/absl/types/optional.h" | 
|  | #include "ui/base/resource/resource_bundle.h" | 
|  | #include "ui/gfx/canvas.h" | 
|  | #include "ui/gfx/color_utils.h" | 
|  | #include "ui/gfx/image/canvas_image_source.h" | 
|  | #include "ui/gfx/paint_vector_icon.h" | 
|  | #include "ui/native_theme/native_theme.h" | 
|  |  | 
|  | #if BUILDFLAG(ENABLE_SUPERVISED_USERS) | 
|  | #include "chrome/browser/supervised_user/supervised_user_constants.h" | 
|  | #endif | 
|  |  | 
|  | #if !defined(OS_ANDROID) | 
|  | #include "chrome/browser/themes/theme_properties.h"  // nogncheck crbug.com/1125897 | 
|  | #endif | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | const char kGAIAGivenNameKey[] = "gaia_given_name"; | 
|  | const char kGAIANameKey[] = "gaia_name"; | 
|  | const char kShortcutNameKey[] = "shortcut_name"; | 
|  | const char kActiveTimeKey[] = "active_time"; | 
|  | const char kMetricsBucketIndex[] = "metrics_bucket_index"; | 
|  | const char kForceSigninProfileLockedKey[] = "force_signin_profile_locked"; | 
|  | const char kHostedDomain[] = "hosted_domain"; | 
|  |  | 
|  | // Avatar info. | 
|  | const char kLastDownloadedGAIAPictureUrlWithSizeKey[] = | 
|  | "last_downloaded_gaia_picture_url_with_size"; | 
|  | const char kGAIAPictureFileNameKey[] = "gaia_picture_file_name"; | 
|  |  | 
|  | // Profile colors info. | 
|  | const char kProfileHighlightColorKey[] = "profile_highlight_color"; | 
|  | const char kDefaultAvatarFillColorKey[] = "default_avatar_fill_color"; | 
|  | const char kDefaultAvatarStrokeColorKey[] = "default_avatar_stroke_color"; | 
|  |  | 
|  | // Low-entropy accounts info, for metrics only. | 
|  | const char kFirstAccountNameHash[] = "first_account_name_hash"; | 
|  | const char kHasMultipleAccountNames[] = "has_multiple_account_names"; | 
|  | const char kAccountCategories[] = "account_categories"; | 
|  |  | 
|  | // Local state pref to keep track of the next available profile bucket. | 
|  | const char kNextMetricsBucketIndex[] = "profile.metrics.next_bucket_index"; | 
|  |  | 
|  | // Deprecated 2/2021. | 
|  | const char kIsOmittedFromProfileListKey[] = "is_omitted_from_profile_list"; | 
|  |  | 
|  | // Deprecated 3/2021. | 
|  | const char kAuthCredentialsKey[] = "local_auth_credentials"; | 
|  | const char kPasswordTokenKey[] = "gaia_password_token"; | 
|  |  | 
|  | // Deprecated 6/2021. | 
|  | const char kSigninRequiredKey[] = "signin_required"; | 
|  | const char kIsAuthErrorKey[] = "is_auth_error"; | 
|  |  | 
|  | // Deprecated 7/2021. | 
|  | const char kProfileIsGuest[] = "is_guest"; | 
|  |  | 
|  | constexpr int kIntegerNotSet = -1; | 
|  |  | 
|  | // Persisted in prefs. | 
|  | constexpr int kAccountCategoriesConsumerOnly = 0; | 
|  | constexpr int kAccountCategoriesEnterpriseOnly = 1; | 
|  | constexpr int kAccountCategoriesBoth = 2; | 
|  |  | 
|  | // Number of distinct low-entropy hash values. Changing this value invalidates | 
|  | // existing persisted hashes. | 
|  | constexpr int kNumberOfLowEntropyHashValues = 1024; | 
|  |  | 
|  | // Returns the next available metrics bucket index and increases the index | 
|  | // counter. I.e. two consecutive calls will return two consecutive numbers. | 
|  | int NextAvailableMetricsBucketIndex() { | 
|  | PrefService* local_prefs = g_browser_process->local_state(); | 
|  | int next_index = local_prefs->GetInteger(kNextMetricsBucketIndex); | 
|  | DCHECK_GT(next_index, 0); | 
|  |  | 
|  | local_prefs->SetInteger(kNextMetricsBucketIndex, next_index + 1); | 
|  |  | 
|  | return next_index; | 
|  | } | 
|  |  | 
|  | int GetLowEntropyHashValue(const std::string& value) { | 
|  | return base::PersistentHash(value) % kNumberOfLowEntropyHashValues; | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | const char ProfileAttributesEntry::kSupervisedUserId[] = "managed_user_id"; | 
|  | const char ProfileAttributesEntry::kAvatarIconKey[] = "avatar_icon"; | 
|  | const char ProfileAttributesEntry::kBackgroundAppsKey[] = "background_apps"; | 
|  | const char ProfileAttributesEntry::kProfileIsEphemeral[] = "is_ephemeral"; | 
|  | const char ProfileAttributesEntry::kUserNameKey[] = "user_name"; | 
|  | const char ProfileAttributesEntry::kGAIAIdKey[] = "gaia_id"; | 
|  | const char ProfileAttributesEntry::kIsConsentedPrimaryAccountKey[] = | 
|  | "is_consented_primary_account"; | 
|  | const char ProfileAttributesEntry::kNameKey[] = "name"; | 
|  | const char ProfileAttributesEntry::kIsUsingDefaultNameKey[] = | 
|  | "is_using_default_name"; | 
|  | const char ProfileAttributesEntry::kIsUsingDefaultAvatarKey[] = | 
|  | "is_using_default_avatar"; | 
|  | const char ProfileAttributesEntry::kUseGAIAPictureKey[] = "use_gaia_picture"; | 
|  | const char ProfileAttributesEntry::kAccountIdKey[] = "account_id_key"; | 
|  |  | 
|  | // static | 
|  | void ProfileAttributesEntry::RegisterLocalStatePrefs( | 
|  | PrefRegistrySimple* registry) { | 
|  | // Bucket 0 is reserved for the guest profile, so start new bucket indices | 
|  | // at 1. | 
|  | registry->RegisterIntegerPref(kNextMetricsBucketIndex, 1); | 
|  | } | 
|  |  | 
|  | ProfileAttributesEntry::ProfileAttributesEntry() = default; | 
|  |  | 
|  | void ProfileAttributesEntry::Initialize(ProfileAttributesStorage* storage, | 
|  | const base::FilePath& path, | 
|  | PrefService* prefs) { | 
|  | DCHECK(!profile_attributes_storage_); | 
|  | DCHECK(storage); | 
|  | profile_attributes_storage_ = storage; | 
|  |  | 
|  | DCHECK(profile_path_.empty()); | 
|  | DCHECK(!path.empty()); | 
|  | profile_path_ = path; | 
|  |  | 
|  | DCHECK(!prefs_); | 
|  | DCHECK(prefs); | 
|  | prefs_ = prefs; | 
|  |  | 
|  | storage_key_ = | 
|  | profile_attributes_storage_->StorageKeyFromProfilePath(profile_path_); | 
|  |  | 
|  | MigrateObsoleteProfileAttributes(); | 
|  |  | 
|  | const base::Value* entry_data = GetEntryData(); | 
|  | if (entry_data) { | 
|  | if (!entry_data->FindKey(kIsConsentedPrimaryAccountKey)) { | 
|  | SetBool(kIsConsentedPrimaryAccountKey, | 
|  | !GetGAIAId().empty() || !GetUserName().empty()); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (signin_util::IsForceSigninEnabled()) { | 
|  | if (!IsAuthenticated()) | 
|  | SetBool(kForceSigninProfileLockedKey, true); | 
|  | } else { | 
|  | // Reset the locked state to avoid a profile being locked after the force | 
|  | // signin policy has been disabled. | 
|  | SetBool(kForceSigninProfileLockedKey, false); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::InitializeLastNameToDisplay() { | 
|  | DCHECK(last_name_to_display_.empty()); | 
|  | last_name_to_display_ = GetName(); | 
|  | } | 
|  |  | 
|  | std::u16string ProfileAttributesEntry::GetLocalProfileName() const { | 
|  | return GetString16(kNameKey); | 
|  | } | 
|  |  | 
|  | std::u16string ProfileAttributesEntry::GetGAIANameToDisplay() const { | 
|  | std::u16string gaia_given_name = GetGAIAGivenName(); | 
|  | return gaia_given_name.empty() ? GetGAIAName() : gaia_given_name; | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::ShouldShowProfileLocalName( | 
|  | const std::u16string& gaia_name_to_display) const { | 
|  | // Never show the profile name if it is equal to GAIA given name, | 
|  | // e.g. Matt (Matt), in that case we should only show the GAIA name. | 
|  | if (base::EqualsCaseInsensitiveASCII(gaia_name_to_display, | 
|  | GetLocalProfileName())) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | // Customized profile name that is not equal to Gaia name, e.g. Matt (Work). | 
|  | if (!IsUsingDefaultName()) | 
|  | return true; | 
|  |  | 
|  | // The profile local name is a default profile name : Person n. | 
|  | std::vector<ProfileAttributesEntry*> entries = | 
|  | profile_attributes_storage_->GetAllProfilesAttributes(); | 
|  |  | 
|  | for (ProfileAttributesEntry* entry : entries) { | 
|  | if (entry == this) | 
|  | continue; | 
|  |  | 
|  | std::u16string other_gaia_name_to_display = entry->GetGAIANameToDisplay(); | 
|  | if (other_gaia_name_to_display.empty() || | 
|  | other_gaia_name_to_display != gaia_name_to_display) | 
|  | continue; | 
|  |  | 
|  | // Another profile with the same GAIA name. | 
|  | bool other_profile_name_equal_GAIA_name = base::EqualsCaseInsensitiveASCII( | 
|  | other_gaia_name_to_display, entry->GetLocalProfileName()); | 
|  | // If for the other profile, the profile name is equal to GAIA name then it | 
|  | // will not be shown. For disambiguation, show for the current profile the | 
|  | // profile name even if it is Person n. | 
|  | if (other_profile_name_equal_GAIA_name) | 
|  | return true; | 
|  |  | 
|  | bool other_is_using_default_name = entry->IsUsingDefaultName(); | 
|  | // Both profiles have a default profile name, | 
|  | // e.g. Matt (Person 1), Matt (Person 2). | 
|  | if (other_is_using_default_name) { | 
|  | return true; | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::ShouldUpdateGAIAPicture( | 
|  | const std::string& image_url_with_size, | 
|  | bool image_is_empty) const { | 
|  | std::string old_file_name = GetString(kGAIAPictureFileNameKey); | 
|  | if (old_file_name.empty() && image_is_empty) { | 
|  | // On Windows, Taskbar and Desktop icons are refreshed every time | 
|  | // |OnProfileAvatarChanged| notification is fired. | 
|  | // Updating from an empty image to a null image is a no-op and it is | 
|  | // important to avoid firing |OnProfileAvatarChanged| in this case. | 
|  | // See http://crbug.com/900374 | 
|  | DCHECK(!IsGAIAPictureLoaded()); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | std::string current_gaia_image_url = | 
|  | GetLastDownloadedGAIAPictureUrlWithSize(); | 
|  | if (old_file_name.empty() || image_is_empty || | 
|  | current_gaia_image_url != image_url_with_size) { | 
|  | return true; | 
|  | } | 
|  | const gfx::Image* gaia_picture = GetGAIAPicture(); | 
|  | if (gaia_picture && !gaia_picture->IsEmpty()) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | // We either did not load the GAIA image or we failed to. In that case, only | 
|  | // update if the GAIA picture is used as the profile avatar. | 
|  | return IsUsingDefaultAvatar() || IsUsingGAIAPicture(); | 
|  | } | 
|  |  | 
|  | std::u16string ProfileAttributesEntry::GetLastNameToDisplay() const { | 
|  | return last_name_to_display_; | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::HasProfileNameChanged() { | 
|  | std::u16string name = GetName(); | 
|  | if (last_name_to_display_ == name) | 
|  | return false; | 
|  |  | 
|  | last_name_to_display_ = name; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | NameForm ProfileAttributesEntry::GetNameForm() const { | 
|  | std::u16string name_to_display = GetGAIANameToDisplay(); | 
|  | if (name_to_display.empty()) | 
|  | return NameForm::kLocalName; | 
|  | if (!ShouldShowProfileLocalName(name_to_display)) | 
|  | return NameForm::kGaiaName; | 
|  | return NameForm::kGaiaAndLocalName; | 
|  | } | 
|  |  | 
|  | std::u16string ProfileAttributesEntry::GetName() const { | 
|  | switch (GetNameForm()) { | 
|  | case NameForm::kGaiaName: | 
|  | return GetGAIANameToDisplay(); | 
|  | case NameForm::kLocalName: | 
|  | return GetLocalProfileName(); | 
|  | case NameForm::kGaiaAndLocalName: | 
|  | return GetGAIANameToDisplay() + u" (" + GetLocalProfileName() + u")"; | 
|  | } | 
|  | } | 
|  |  | 
|  | std::u16string ProfileAttributesEntry::GetShortcutName() const { | 
|  | return GetString16(kShortcutNameKey); | 
|  | } | 
|  |  | 
|  | base::FilePath ProfileAttributesEntry::GetPath() const { | 
|  | return profile_path_; | 
|  | } | 
|  |  | 
|  | base::Time ProfileAttributesEntry::GetActiveTime() const { | 
|  | if (IsDouble(kActiveTimeKey)) { | 
|  | return base::Time::FromDoubleT(GetDouble(kActiveTimeKey)); | 
|  | } else { | 
|  | return base::Time(); | 
|  | } | 
|  | } | 
|  |  | 
|  | std::u16string ProfileAttributesEntry::GetUserName() const { | 
|  | return GetString16(kUserNameKey); | 
|  | } | 
|  |  | 
|  | gfx::Image ProfileAttributesEntry::GetAvatarIcon( | 
|  | int size_for_placeholder_avatar, | 
|  | bool use_high_res_file) const { | 
|  | if (IsUsingGAIAPicture()) { | 
|  | const gfx::Image* image = GetGAIAPicture(); | 
|  | if (image) | 
|  | return *image; | 
|  | } | 
|  |  | 
|  | #if !defined(OS_ANDROID) && !BUILDFLAG(IS_CHROMEOS_ASH) | 
|  | // TODO(crbug.com/1100835): After launch, remove the treatment of placeholder | 
|  | // avatars from GetHighResAvatar() and from any other places. | 
|  | if (GetAvatarIconIndex() == profiles::GetPlaceholderAvatarIndex()) { | 
|  | return GetPlaceholderAvatarIcon(size_for_placeholder_avatar); | 
|  | } | 
|  | #endif  // !defined(OS_ANDROID) && !BUILDFLAG(IS_CHROMEOS_ASH) | 
|  |  | 
|  | #if !defined(OS_ANDROID) | 
|  | // Use the high resolution version of the avatar if it exists. Mobile doesn't | 
|  | // need the high resolution version so no need to fetch it. | 
|  | if (use_high_res_file) { | 
|  | const gfx::Image* image = GetHighResAvatar(); | 
|  | if (image) | 
|  | return *image; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | const int icon_index = GetAvatarIconIndex(); | 
|  | #if defined(OS_WIN) | 
|  | if (!profiles::IsModernAvatarIconIndex(icon_index)) { | 
|  | // Return the 2x version of the old avatar, defined specifically for | 
|  | // Windows. No special treatment is needed for modern avatars as they | 
|  | // already have high enough resolution. | 
|  | const int win_resource_id = | 
|  | profiles::GetOldDefaultAvatar2xIconResourceIDAtIndex(icon_index); | 
|  | return ui::ResourceBundle::GetSharedInstance().GetNativeImageNamed( | 
|  | win_resource_id); | 
|  | } | 
|  | #endif | 
|  | int resource_id = profiles::GetDefaultAvatarIconResourceIDAtIndex(icon_index); | 
|  | return ui::ResourceBundle::GetSharedInstance().GetNativeImageNamed( | 
|  | resource_id); | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::GetBackgroundStatus() const { | 
|  | return GetBool(kBackgroundAppsKey); | 
|  | } | 
|  |  | 
|  | std::u16string ProfileAttributesEntry::GetGAIAName() const { | 
|  | return GetString16(kGAIANameKey); | 
|  | } | 
|  |  | 
|  | std::u16string ProfileAttributesEntry::GetGAIAGivenName() const { | 
|  | return GetString16(kGAIAGivenNameKey); | 
|  | } | 
|  |  | 
|  | std::string ProfileAttributesEntry::GetGAIAId() const { | 
|  | return GetString(ProfileAttributesEntry::kGAIAIdKey); | 
|  | } | 
|  |  | 
|  | const gfx::Image* ProfileAttributesEntry::GetGAIAPicture() const { | 
|  | std::string file_name = GetString(kGAIAPictureFileNameKey); | 
|  |  | 
|  | // If the picture is not on disk then return nullptr. | 
|  | if (file_name.empty()) | 
|  | return nullptr; | 
|  |  | 
|  | base::FilePath image_path = profile_path_.AppendASCII(file_name); | 
|  | return profile_attributes_storage_->LoadAvatarPictureFromPath( | 
|  | profile_path_, storage_key_, image_path); | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::IsUsingGAIAPicture() const { | 
|  | bool result = GetBool(kUseGAIAPictureKey); | 
|  | if (!result) { | 
|  | // Prefer the GAIA avatar over a non-customized avatar. | 
|  | result = IsUsingDefaultAvatar() && GetGAIAPicture(); | 
|  | } | 
|  | return result; | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::IsGAIAPictureLoaded() const { | 
|  | return profile_attributes_storage_->IsGAIAPictureLoaded(storage_key_); | 
|  | } | 
|  |  | 
|  | std::string ProfileAttributesEntry::GetLastDownloadedGAIAPictureUrlWithSize() | 
|  | const { | 
|  | return GetString(kLastDownloadedGAIAPictureUrlWithSizeKey); | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::IsSupervised() const { | 
|  | return !GetSupervisedUserId().empty(); | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::IsChild() const { | 
|  | #if BUILDFLAG(ENABLE_SUPERVISED_USERS) | 
|  | return GetSupervisedUserId() == supervised_users::kChildAccountSUID; | 
|  | #else | 
|  | return false; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::IsOmitted() const { | 
|  | return is_omitted_; | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::IsSigninRequired() const { | 
|  | return GetBool(kForceSigninProfileLockedKey); | 
|  | } | 
|  |  | 
|  | std::string ProfileAttributesEntry::GetSupervisedUserId() const { | 
|  | return GetString(kSupervisedUserId); | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::IsEphemeral() const { | 
|  | return GetBool(kProfileIsEphemeral); | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::IsUsingDefaultName() const { | 
|  | return GetBool(kIsUsingDefaultNameKey); | 
|  | } | 
|  |  | 
|  | SigninState ProfileAttributesEntry::GetSigninState() const { | 
|  | bool is_consented_primary_account = GetBool(kIsConsentedPrimaryAccountKey); | 
|  | if (!GetGAIAId().empty() || !GetUserName().empty()) { | 
|  | return is_consented_primary_account | 
|  | ? SigninState::kSignedInWithConsentedPrimaryAccount | 
|  | : SigninState::kSignedInWithUnconsentedPrimaryAccount; | 
|  | } | 
|  | DCHECK(!is_consented_primary_account); | 
|  | return SigninState::kNotSignedIn; | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::IsAuthenticated() const { | 
|  | return GetBool(kIsConsentedPrimaryAccountKey); | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::IsUsingDefaultAvatar() const { | 
|  | return GetBool(kIsUsingDefaultAvatarKey); | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::IsSignedInWithCredentialProvider() const { | 
|  | return GetBool(prefs::kSignedInWithCredentialProvider); | 
|  | } | 
|  |  | 
|  | size_t ProfileAttributesEntry::GetAvatarIconIndex() const { | 
|  | std::string icon_url = GetString(kAvatarIconKey); | 
|  | size_t icon_index = 0; | 
|  | if (!profiles::IsDefaultAvatarIconUrl(icon_url, &icon_index)) | 
|  | DLOG(WARNING) << "Unknown avatar icon: " << icon_url; | 
|  |  | 
|  | return icon_index; | 
|  | } | 
|  |  | 
|  | absl::optional<ProfileThemeColors> | 
|  | ProfileAttributesEntry::GetProfileThemeColorsIfSet() const { | 
|  | absl::optional<SkColor> profile_highlight_color = | 
|  | GetProfileThemeColor(kProfileHighlightColorKey); | 
|  | absl::optional<SkColor> default_avatar_fill_color = | 
|  | GetProfileThemeColor(kDefaultAvatarFillColorKey); | 
|  | absl::optional<SkColor> default_avatar_stroke_color = | 
|  | GetProfileThemeColor(kDefaultAvatarStrokeColorKey); | 
|  |  | 
|  | DCHECK_EQ(profile_highlight_color.has_value(), | 
|  | default_avatar_stroke_color.has_value()); | 
|  | DCHECK_EQ(profile_highlight_color.has_value(), | 
|  | default_avatar_fill_color.has_value()); | 
|  |  | 
|  | if (!profile_highlight_color.has_value()) { | 
|  | return absl::nullopt; | 
|  | } | 
|  |  | 
|  | ProfileThemeColors colors; | 
|  | colors.profile_highlight_color = profile_highlight_color.value(); | 
|  | colors.default_avatar_fill_color = default_avatar_fill_color.value(); | 
|  | colors.default_avatar_stroke_color = default_avatar_stroke_color.value(); | 
|  | return colors; | 
|  | } | 
|  |  | 
|  | ProfileThemeColors ProfileAttributesEntry::GetProfileThemeColors() const { | 
|  | #if defined(OS_ANDROID) | 
|  | // Profile theme colors shouldn't be queried on Android. | 
|  | NOTREACHED(); | 
|  | return {SK_ColorRED, SK_ColorRED, SK_ColorRED}; | 
|  | #else | 
|  | absl::optional<ProfileThemeColors> theme_colors = | 
|  | GetProfileThemeColorsIfSet(); | 
|  | if (theme_colors) | 
|  | return *theme_colors; | 
|  |  | 
|  | return GetDefaultProfileThemeColors( | 
|  | ui::NativeTheme::GetInstanceForNativeUi()->ShouldUseDarkColors()); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | size_t ProfileAttributesEntry::GetMetricsBucketIndex() { | 
|  | int bucket_index = GetInteger(kMetricsBucketIndex); | 
|  | if (bucket_index == kIntegerNotSet) { | 
|  | bucket_index = NextAvailableMetricsBucketIndex(); | 
|  | SetInteger(kMetricsBucketIndex, bucket_index); | 
|  | } | 
|  | return bucket_index; | 
|  | } | 
|  |  | 
|  | std::string ProfileAttributesEntry::GetHostedDomain() const { | 
|  | return GetString(kHostedDomain); | 
|  | } | 
|  |  | 
|  | std::string ProfileAttributesEntry::GetAccountIdKey() const { | 
|  | return GetString(kAccountIdKey); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetLocalProfileName(const std::u16string& name, | 
|  | bool is_default_name) { | 
|  | bool changed = SetString16(kNameKey, name); | 
|  | changed |= SetBool(kIsUsingDefaultNameKey, is_default_name); | 
|  | if (changed) | 
|  | profile_attributes_storage_->NotifyIfProfileNamesHaveChanged(); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetShortcutName(const std::u16string& name) { | 
|  | SetString16(kShortcutNameKey, name); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetActiveTimeToNow() { | 
|  | if (IsDouble(kActiveTimeKey) && | 
|  | base::Time::Now() - GetActiveTime() < base::TimeDelta::FromHours(1)) { | 
|  | return; | 
|  | } | 
|  | SetDouble(kActiveTimeKey, base::Time::Now().ToDoubleT()); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetIsOmitted(bool is_omitted) { | 
|  | bool old_value = IsOmitted(); | 
|  | SetIsOmittedInternal(is_omitted); | 
|  |  | 
|  | // Send a notification only if the value has really changed. | 
|  | if (old_value != is_omitted_) | 
|  | profile_attributes_storage_->NotifyProfileIsOmittedChanged(GetPath()); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetSupervisedUserId(const std::string& id) { | 
|  | if (SetString(kSupervisedUserId, id)) | 
|  | profile_attributes_storage_->NotifyProfileSupervisedUserIdChanged( | 
|  | GetPath()); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetBackgroundStatus(bool running_background_apps) { | 
|  | SetBool(kBackgroundAppsKey, running_background_apps); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetGAIAName(const std::u16string& name) { | 
|  | if (SetString16(kGAIANameKey, name)) | 
|  | profile_attributes_storage_->NotifyIfProfileNamesHaveChanged(); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetGAIAGivenName(const std::u16string& name) { | 
|  | if (SetString16(kGAIAGivenNameKey, name)) | 
|  | profile_attributes_storage_->NotifyIfProfileNamesHaveChanged(); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetGAIAPicture( | 
|  | const std::string& image_url_with_size, | 
|  | gfx::Image image) { | 
|  | if (!ShouldUpdateGAIAPicture(image_url_with_size, image.IsEmpty())) | 
|  | return; | 
|  |  | 
|  | std::string old_file_name = GetString(kGAIAPictureFileNameKey); | 
|  | std::string new_file_name; | 
|  | if (image.IsEmpty()) { | 
|  | // Delete the old bitmap from disk. | 
|  | base::FilePath image_path = profile_path_.AppendASCII(old_file_name); | 
|  | profile_attributes_storage_->DeleteGAIAImageAtPath( | 
|  | profile_path_, storage_key_, image_path); | 
|  | } else { | 
|  | // Save the new bitmap to disk. | 
|  | new_file_name = | 
|  | old_file_name.empty() | 
|  | ? base::FilePath(profiles::kGAIAPictureFileName).MaybeAsASCII() | 
|  | : old_file_name; | 
|  | base::FilePath image_path = profile_path_.AppendASCII(new_file_name); | 
|  | profile_attributes_storage_->SaveGAIAImageAtPath( | 
|  | profile_path_, storage_key_, image, image_path, image_url_with_size); | 
|  | } | 
|  |  | 
|  | SetString(kGAIAPictureFileNameKey, new_file_name); | 
|  | profile_attributes_storage_->NotifyOnProfileAvatarChanged(profile_path_); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetIsUsingGAIAPicture(bool value) { | 
|  | SetBool(kUseGAIAPictureKey, value); | 
|  | // TODO(alexilin): send notification only if the value has changed. | 
|  | profile_attributes_storage_->NotifyOnProfileAvatarChanged(profile_path_); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetLastDownloadedGAIAPictureUrlWithSize( | 
|  | const std::string& image_url_with_size) { | 
|  | SetString(kLastDownloadedGAIAPictureUrlWithSizeKey, image_url_with_size); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetSignedInWithCredentialProvider(bool value) { | 
|  | if (value != GetBool(prefs::kSignedInWithCredentialProvider)) { | 
|  | SetBool(prefs::kSignedInWithCredentialProvider, value); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::LockForceSigninProfile(bool is_lock) { | 
|  | DCHECK(signin_util::IsForceSigninEnabled()); | 
|  | if (GetBool(kForceSigninProfileLockedKey) == is_lock) | 
|  | return; | 
|  | SetBool(kForceSigninProfileLockedKey, is_lock); | 
|  | profile_attributes_storage_->NotifyIsSigninRequiredChanged(GetPath()); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::RecordAccountMetrics() const { | 
|  | RecordAccountCategoriesMetric(); | 
|  | RecordAccountNamesMetric(); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetIsEphemeral(bool value) { | 
|  | if (!value) { | 
|  | DCHECK(!IsOmitted()) << "An omitted account should not be made " | 
|  | "non-ephemeral. Call SetIsOmitted(false) first."; | 
|  | } | 
|  |  | 
|  | SetBool(kProfileIsEphemeral, value); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetIsUsingDefaultName(bool value) { | 
|  | if (SetBool(kIsUsingDefaultNameKey, value)) | 
|  | profile_attributes_storage_->NotifyIfProfileNamesHaveChanged(); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetIsUsingDefaultAvatar(bool value) { | 
|  | SetBool(kIsUsingDefaultAvatarKey, value); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetAvatarIconIndex(size_t icon_index) { | 
|  | std::string default_avatar_icon_url = | 
|  | profiles::GetDefaultAvatarIconUrl(icon_index); | 
|  | if (default_avatar_icon_url == GetString(kAvatarIconKey)) { | 
|  | // On Windows, Taskbar and Desktop icons are refreshed every time | 
|  | // |OnProfileAvatarChanged| notification is fired. | 
|  | // As the current avatar icon is already set to |default_avatar_icon_url|, | 
|  | // it is important to avoid firing |OnProfileAvatarChanged| in this case. | 
|  | // See http://crbug.com/900374 | 
|  | return; | 
|  | } | 
|  |  | 
|  | SetString(kAvatarIconKey, default_avatar_icon_url); | 
|  |  | 
|  | base::FilePath profile_path = GetPath(); | 
|  | if (!profile_attributes_storage_->GetDisableAvatarDownloadForTesting()) { | 
|  | profile_attributes_storage_->DownloadHighResAvatarIfNeeded(icon_index, | 
|  | profile_path); | 
|  | } | 
|  |  | 
|  | profile_attributes_storage_->NotifyOnProfileAvatarChanged(profile_path); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetProfileThemeColors( | 
|  | const absl::optional<ProfileThemeColors>& colors) { | 
|  | bool changed = false; | 
|  | if (colors.has_value()) { | 
|  | changed |= | 
|  | SetInteger(kProfileHighlightColorKey, colors->profile_highlight_color); | 
|  | changed |= SetInteger(kDefaultAvatarFillColorKey, | 
|  | colors->default_avatar_fill_color); | 
|  | changed |= SetInteger(kDefaultAvatarStrokeColorKey, | 
|  | colors->default_avatar_stroke_color); | 
|  | } else { | 
|  | changed |= ClearValue(kProfileHighlightColorKey); | 
|  | changed |= ClearValue(kDefaultAvatarFillColorKey); | 
|  | changed |= ClearValue(kDefaultAvatarStrokeColorKey); | 
|  | } | 
|  |  | 
|  | if (changed) { | 
|  | profile_attributes_storage_->NotifyProfileThemeColorsChanged(GetPath()); | 
|  | if (GetAvatarIconIndex() == profiles::GetPlaceholderAvatarIndex()) | 
|  | profile_attributes_storage_->NotifyOnProfileAvatarChanged(GetPath()); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetHostedDomain(std::string hosted_domain) { | 
|  | if (SetString(kHostedDomain, hosted_domain)) | 
|  | profile_attributes_storage_->NotifyProfileHostedDomainChanged(GetPath()); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetAuthInfo(const std::string& gaia_id, | 
|  | const std::u16string& user_name, | 
|  | bool is_consented_primary_account) { | 
|  | // If gaia_id, username and consent state are unchanged, abort early. | 
|  | if (GetBool(kIsConsentedPrimaryAccountKey) == is_consented_primary_account && | 
|  | gaia_id == GetGAIAId() && user_name == GetUserName()) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | const base::Value* old_data = GetEntryData(); | 
|  | base::Value new_data = old_data ? GetEntryData()->Clone() | 
|  | : base::Value(base::Value::Type::DICTIONARY); | 
|  | new_data.SetStringKey(kGAIAIdKey, gaia_id); | 
|  | new_data.SetStringKey(kUserNameKey, user_name); | 
|  | DCHECK(!is_consented_primary_account || !gaia_id.empty() || | 
|  | !user_name.empty()); | 
|  | new_data.SetBoolKey(kIsConsentedPrimaryAccountKey, | 
|  | is_consented_primary_account); | 
|  | SetEntryData(std::move(new_data)); | 
|  | profile_attributes_storage_->NotifyProfileAuthInfoChanged(profile_path_); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::AddAccountName(const std::string& name) { | 
|  | int hash = GetLowEntropyHashValue(name); | 
|  | int first_hash = GetInteger(kFirstAccountNameHash); | 
|  | if (first_hash == kIntegerNotSet) { | 
|  | SetInteger(kFirstAccountNameHash, hash); | 
|  | return; | 
|  | } | 
|  |  | 
|  | if (first_hash != hash) { | 
|  | SetBool(kHasMultipleAccountNames, true); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::AddAccountCategory(AccountCategory category) { | 
|  | int current_categories = GetInteger(kAccountCategories); | 
|  | if (current_categories == kAccountCategoriesBoth) | 
|  | return; | 
|  |  | 
|  | int new_category = category == AccountCategory::kConsumer | 
|  | ? kAccountCategoriesConsumerOnly | 
|  | : kAccountCategoriesEnterpriseOnly; | 
|  | if (current_categories == kIntegerNotSet) { | 
|  | SetInteger(kAccountCategories, new_category); | 
|  | } else if (current_categories != new_category) { | 
|  | SetInteger(kAccountCategories, kAccountCategoriesBoth); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::ClearAccountNames() { | 
|  | ClearValue(kFirstAccountNameHash); | 
|  | ClearValue(kHasMultipleAccountNames); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::ClearAccountCategories() { | 
|  | ClearValue(kAccountCategories); | 
|  | } | 
|  |  | 
|  | const gfx::Image* ProfileAttributesEntry::GetHighResAvatar() const { | 
|  | const size_t avatar_index = GetAvatarIconIndex(); | 
|  |  | 
|  | // If this is the placeholder avatar, it is already included in the | 
|  | // resources, so it doesn't need to be downloaded. | 
|  | if (avatar_index == profiles::GetPlaceholderAvatarIndex()) { | 
|  | return &ui::ResourceBundle::GetSharedInstance().GetImageNamed( | 
|  | profiles::GetPlaceholderAvatarIconResourceID()); | 
|  | } | 
|  |  | 
|  | const std::string key = | 
|  | profiles::GetDefaultAvatarIconFileNameAtIndex(avatar_index); | 
|  | const base::FilePath image_path = | 
|  | profiles::GetPathOfHighResAvatarAtIndex(avatar_index); | 
|  | return profile_attributes_storage_->LoadAvatarPictureFromPath(GetPath(), key, | 
|  | image_path); | 
|  | } | 
|  |  | 
|  | gfx::Image ProfileAttributesEntry::GetPlaceholderAvatarIcon(int size) const { | 
|  | ProfileThemeColors colors = GetProfileThemeColors(); | 
|  | return profiles::GetPlaceholderAvatarIconWithColors( | 
|  | colors.default_avatar_fill_color, colors.default_avatar_stroke_color, | 
|  | size); | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::HasMultipleAccountNames() const { | 
|  | // If the value is not set, GetBool() returns false. | 
|  | return GetBool(kHasMultipleAccountNames); | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::HasBothAccountCategories() const { | 
|  | // If the value is not set, GetInteger returns kIntegerNotSet which does not | 
|  | // equal kAccountTypeBoth. | 
|  | return GetInteger(kAccountCategories) == kAccountCategoriesBoth; | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::RecordAccountCategoriesMetric() const { | 
|  | if (HasBothAccountCategories()) { | 
|  | if (IsAuthenticated()) { | 
|  | bool consumer_syncing = GetHostedDomain() == kNoHostedDomainFound; | 
|  | profile_metrics::LogProfileAllAccountsCategories( | 
|  | consumer_syncing ? profile_metrics::AllAccountsCategories:: | 
|  | kBothConsumerAndEnterpriseSyncingConsumer | 
|  | : profile_metrics::AllAccountsCategories:: | 
|  | kBothConsumerAndEnterpriseSyncingEnterprise); | 
|  | } else { | 
|  | profile_metrics::LogProfileAllAccountsCategories( | 
|  | profile_metrics::AllAccountsCategories:: | 
|  | kBothConsumerAndEnterpriseNoSync); | 
|  | } | 
|  | } else { | 
|  | profile_metrics::LogProfileAllAccountsCategories( | 
|  | profile_metrics::AllAccountsCategories::kSingleCategory); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::RecordAccountNamesMetric() const { | 
|  | if (HasMultipleAccountNames()) { | 
|  | profile_metrics::LogProfileAllAccountsNames( | 
|  | IsAuthenticated() | 
|  | ? profile_metrics::AllAccountsNames::kMultipleNamesWithSync | 
|  | : profile_metrics::AllAccountsNames::kMultipleNamesWithoutSync); | 
|  | } else { | 
|  | profile_metrics::LogProfileAllAccountsNames( | 
|  | profile_metrics::AllAccountsNames::kLikelySingleName); | 
|  | } | 
|  | } | 
|  |  | 
|  | const base::Value* ProfileAttributesEntry::GetEntryData() const { | 
|  | const base::DictionaryValue* attributes = | 
|  | prefs_->GetDictionary(prefs::kProfileAttributes); | 
|  | return attributes->FindKeyOfType(storage_key_, base::Value::Type::DICTIONARY); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetEntryData(base::Value data) { | 
|  | DCHECK(data.is_dict()); | 
|  |  | 
|  | DictionaryPrefUpdate update(prefs_, prefs::kProfileAttributes); | 
|  | base::DictionaryValue* attributes = update.Get(); | 
|  | attributes->SetKey(storage_key_, std::move(data)); | 
|  | } | 
|  |  | 
|  | const base::Value* ProfileAttributesEntry::GetValue(const char* key) const { | 
|  | const base::Value* entry_data = GetEntryData(); | 
|  | return entry_data ? entry_data->FindKey(key) : nullptr; | 
|  | } | 
|  |  | 
|  | std::string ProfileAttributesEntry::GetString(const char* key) const { | 
|  | const base::Value* value = GetValue(key); | 
|  | if (!value || !value->is_string()) | 
|  | return std::string(); | 
|  | return value->GetString(); | 
|  | } | 
|  |  | 
|  | std::u16string ProfileAttributesEntry::GetString16(const char* key) const { | 
|  | const base::Value* value = GetValue(key); | 
|  | if (!value || !value->is_string()) | 
|  | return std::u16string(); | 
|  | return base::UTF8ToUTF16(value->GetString()); | 
|  | } | 
|  |  | 
|  | double ProfileAttributesEntry::GetDouble(const char* key) const { | 
|  | const base::Value* value = GetValue(key); | 
|  | if (!value || !value->is_double()) | 
|  | return 0.0; | 
|  | return value->GetDouble(); | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::GetBool(const char* key) const { | 
|  | const base::Value* value = GetValue(key); | 
|  | return value && value->is_bool() && value->GetBool(); | 
|  | } | 
|  |  | 
|  | int ProfileAttributesEntry::GetInteger(const char* key) const { | 
|  | const base::Value* value = GetValue(key); | 
|  | if (!value || !value->is_int()) | 
|  | return kIntegerNotSet; | 
|  | return value->GetInt(); | 
|  | } | 
|  |  | 
|  | absl::optional<SkColor> ProfileAttributesEntry::GetProfileThemeColor( | 
|  | const char* key) const { | 
|  | const base::Value* value = GetValue(key); | 
|  | if (!value || !value->is_int()) | 
|  | return absl::nullopt; | 
|  | return value->GetInt(); | 
|  | } | 
|  |  | 
|  | // Type checking. Only IsDouble is implemented because others do not have | 
|  | // callsites. | 
|  | bool ProfileAttributesEntry::IsDouble(const char* key) const { | 
|  | const base::Value* value = GetValue(key); | 
|  | return value && value->is_double(); | 
|  | } | 
|  |  | 
|  | // Internal setters using keys; | 
|  | bool ProfileAttributesEntry::SetString(const char* key, std::string value) { | 
|  | const base::Value* old_data = GetEntryData(); | 
|  | if (old_data) { | 
|  | const base::Value* old_value = old_data->FindKey(key); | 
|  | if (old_value && old_value->is_string() && old_value->GetString() == value) | 
|  | return false; | 
|  | } | 
|  |  | 
|  | base::Value new_data = old_data ? GetEntryData()->Clone() | 
|  | : base::Value(base::Value::Type::DICTIONARY); | 
|  | new_data.SetKey(key, base::Value(value)); | 
|  | SetEntryData(std::move(new_data)); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::SetString16(const char* key, | 
|  | std::u16string value) { | 
|  | const base::Value* old_data = GetEntryData(); | 
|  | if (old_data) { | 
|  | const base::Value* old_value = old_data->FindKey(key); | 
|  | if (old_value && old_value->is_string() && | 
|  | base::UTF8ToUTF16(old_value->GetString()) == value) | 
|  | return false; | 
|  | } | 
|  |  | 
|  | base::Value new_data = old_data ? GetEntryData()->Clone() | 
|  | : base::Value(base::Value::Type::DICTIONARY); | 
|  | new_data.SetKey(key, base::Value(value)); | 
|  | SetEntryData(std::move(new_data)); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::SetDouble(const char* key, double value) { | 
|  | const base::Value* old_data = GetEntryData(); | 
|  | if (old_data) { | 
|  | const base::Value* old_value = old_data->FindKey(key); | 
|  | if (old_value && old_value->is_double() && old_value->GetDouble() == value) | 
|  | return false; | 
|  | } | 
|  |  | 
|  | base::Value new_data = old_data ? GetEntryData()->Clone() | 
|  | : base::Value(base::Value::Type::DICTIONARY); | 
|  | new_data.SetKey(key, base::Value(value)); | 
|  | SetEntryData(std::move(new_data)); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::SetBool(const char* key, bool value) { | 
|  | const base::Value* old_data = GetEntryData(); | 
|  | if (old_data) { | 
|  | const base::Value* old_value = old_data->FindKey(key); | 
|  | if (old_value && old_value->is_bool() && old_value->GetBool() == value) | 
|  | return false; | 
|  | } | 
|  |  | 
|  | base::Value new_data = old_data ? GetEntryData()->Clone() | 
|  | : base::Value(base::Value::Type::DICTIONARY); | 
|  | new_data.SetKey(key, base::Value(value)); | 
|  | SetEntryData(std::move(new_data)); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::SetInteger(const char* key, int value) { | 
|  | const base::Value* old_data = GetEntryData(); | 
|  | if (old_data) { | 
|  | const base::Value* old_value = old_data->FindKey(key); | 
|  | if (old_value && old_value->is_int() && old_value->GetInt() == value) | 
|  | return false; | 
|  | } | 
|  |  | 
|  | base::Value new_data = old_data ? GetEntryData()->Clone() | 
|  | : base::Value(base::Value::Type::DICTIONARY); | 
|  | new_data.SetKey(key, base::Value(value)); | 
|  | SetEntryData(std::move(new_data)); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool ProfileAttributesEntry::ClearValue(const char* key) { | 
|  | const base::Value* old_data = GetEntryData(); | 
|  | if (!old_data || !old_data->FindKey(key)) | 
|  | return false; | 
|  |  | 
|  | base::Value new_data = GetEntryData()->Clone(); | 
|  | new_data.RemoveKey(key); | 
|  | SetEntryData(std::move(new_data)); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | // This method should be periodically pruned of year+ old migrations. | 
|  | void ProfileAttributesEntry::MigrateObsoleteProfileAttributes() { | 
|  | // Added 2/2021. | 
|  | ClearValue(kIsOmittedFromProfileListKey); | 
|  |  | 
|  | // Added 3/2021. | 
|  | ClearValue(kAuthCredentialsKey); | 
|  | ClearValue(kPasswordTokenKey); | 
|  |  | 
|  | // Added 6/2021. | 
|  | ClearValue(kSigninRequiredKey); | 
|  | ClearValue(kIsAuthErrorKey); | 
|  |  | 
|  | // Added 7/2021. | 
|  | ClearValue(kProfileIsGuest); | 
|  | } | 
|  |  | 
|  | void ProfileAttributesEntry::SetIsOmittedInternal(bool is_omitted) { | 
|  | if (is_omitted) { | 
|  | DCHECK(IsEphemeral()) << "Only ephemeral profiles can be omitted."; | 
|  | } | 
|  |  | 
|  | is_omitted_ = is_omitted; | 
|  | } |