blob: c529e9c233c6e19a91e0325313991b38c09a4a97 [file] [log] [blame]
// 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;
}