| OLD | NEW |
| 1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "chrome/browser/ui/views/profiles/new_avatar_button.h" | 5 #include "chrome/browser/ui/views/profiles/new_avatar_button.h" |
| 6 | 6 |
| 7 #include "base/win/windows_version.h" | 7 #include "base/win/windows_version.h" |
| 8 #include "chrome/browser/browser_process.h" | 8 #include "chrome/browser/browser_process.h" |
| 9 #include "chrome/browser/profiles/profile_manager.h" | 9 #include "chrome/browser/profiles/profile_manager.h" |
| 10 #include "chrome/browser/profiles/profiles_state.h" | 10 #include "chrome/browser/profiles/profiles_state.h" |
| (...skipping 13 matching lines...) Expand all Loading... |
| 24 const int pushed_image_set[]) { | 24 const int pushed_image_set[]) { |
| 25 scoped_ptr<views::LabelButtonBorder> border( | 25 scoped_ptr<views::LabelButtonBorder> border( |
| 26 new views::LabelButtonBorder(views::Button::STYLE_TEXTBUTTON)); | 26 new views::LabelButtonBorder(views::Button::STYLE_TEXTBUTTON)); |
| 27 border->SetPainter(false, views::Button::STATE_NORMAL, | 27 border->SetPainter(false, views::Button::STATE_NORMAL, |
| 28 views::Painter::CreateImageGridPainter(normal_image_set)); | 28 views::Painter::CreateImageGridPainter(normal_image_set)); |
| 29 border->SetPainter(false, views::Button::STATE_HOVERED, | 29 border->SetPainter(false, views::Button::STATE_HOVERED, |
| 30 views::Painter::CreateImageGridPainter(hot_image_set)); | 30 views::Painter::CreateImageGridPainter(hot_image_set)); |
| 31 border->SetPainter(false, views::Button::STATE_PRESSED, | 31 border->SetPainter(false, views::Button::STATE_PRESSED, |
| 32 views::Painter::CreateImageGridPainter(pushed_image_set)); | 32 views::Painter::CreateImageGridPainter(pushed_image_set)); |
| 33 | 33 |
| 34 const int kLeftRightInset = 10; | 34 const int kLeftRightInset = 8; |
| 35 const int kTopInset = 0; | 35 const int kTopInset = 2; |
| 36 const int kBottomInset = 4; | 36 const int kBottomInset = 4; |
| 37 border->set_insets(gfx::Insets(kTopInset, kLeftRightInset, | 37 border->set_insets(gfx::Insets(kTopInset, kLeftRightInset, |
| 38 kBottomInset, kLeftRightInset)); | 38 kBottomInset, kLeftRightInset)); |
| 39 | 39 |
| 40 return border.PassAs<views::Border>(); | 40 return border.PassAs<views::Border>(); |
| 41 } | 41 } |
| 42 | 42 |
| 43 } // namespace | 43 } // namespace |
| 44 | 44 |
| 45 NewAvatarButton::NewAvatarButton( | 45 NewAvatarButton::NewAvatarButton(views::ButtonListener* listener, |
| 46 views::ButtonListener* listener, | 46 AvatarButtonStyle button_style, |
| 47 const base::string16& profile_name, | 47 Browser* browser) |
| 48 AvatarButtonStyle button_style, | 48 : LabelButton(listener, base::string16()), |
| 49 Browser* browser) | |
| 50 : MenuButton(listener, | |
| 51 profiles::GetAvatarButtonTextForProfile(browser->profile()), | |
| 52 NULL, | |
| 53 true), | |
| 54 browser_(browser), | 49 browser_(browser), |
| 50 has_auth_error_(false), |
| 55 suppress_mouse_released_action_(false) { | 51 suppress_mouse_released_action_(false) { |
| 56 set_animate_on_state_change(false); | 52 set_animate_on_state_change(false); |
| 57 SetTextColor(views::Button::STATE_NORMAL, SK_ColorWHITE); | 53 SetTextColor(views::Button::STATE_NORMAL, SK_ColorWHITE); |
| 58 SetTextColor(views::Button::STATE_HOVERED, SK_ColorWHITE); | 54 SetTextColor(views::Button::STATE_HOVERED, SK_ColorWHITE); |
| 59 SetTextColor(views::Button::STATE_PRESSED, SK_ColorWHITE); | 55 SetTextColor(views::Button::STATE_PRESSED, SK_ColorWHITE); |
| 60 SetTextShadows(gfx::ShadowValues(10, | 56 SetTextShadows(gfx::ShadowValues(10, |
| 61 gfx::ShadowValue(gfx::Point(), 1.0f, SK_ColorDKGRAY))); | 57 gfx::ShadowValue(gfx::Point(), 1.0f, SK_ColorDKGRAY))); |
| 62 SetTextSubpixelRenderingEnabled(false); | 58 SetTextSubpixelRenderingEnabled(false); |
| 59 SetHorizontalAlignment(gfx::ALIGN_CENTER); |
| 63 | 60 |
| 64 // The largest text height that fits in the button. If the font list height | 61 // The largest text height that fits in the button. If the font list height |
| 65 // is larger than this, it will be shrunk to match it. | 62 // is larger than this, it will be shrunk to match it. |
| 66 // TODO(noms): Calculate this constant algorithmically. | 63 // TODO(noms): Calculate this constant algorithmically. |
| 67 const int kDisplayFontHeight = 15; | 64 const int kDisplayFontHeight = 15; |
| 68 SetFontList(GetFontList().DeriveWithHeightUpperBound(kDisplayFontHeight)); | 65 SetFontList(GetFontList().DeriveWithHeightUpperBound(kDisplayFontHeight)); |
| 69 | 66 |
| 70 ui::ResourceBundle* rb = &ui::ResourceBundle::GetSharedInstance(); | 67 ui::ResourceBundle* rb = &ui::ResourceBundle::GetSharedInstance(); |
| 71 if (button_style == THEMED_BUTTON) { | 68 if (button_style == THEMED_BUTTON) { |
| 72 const int kNormalImageSet[] = IMAGE_GRID(IDR_AVATAR_THEMED_BUTTON_NORMAL); | 69 const int kNormalImageSet[] = IMAGE_GRID(IDR_AVATAR_THEMED_BUTTON_NORMAL); |
| 73 const int kHotImageSet[] = IMAGE_GRID(IDR_AVATAR_THEMED_BUTTON_HOVER); | 70 const int kHotImageSet[] = IMAGE_GRID(IDR_AVATAR_THEMED_BUTTON_HOVER); |
| 74 const int kPushedImageSet[] = IMAGE_GRID(IDR_AVATAR_THEMED_BUTTON_PRESSED); | 71 const int kPushedImageSet[] = IMAGE_GRID(IDR_AVATAR_THEMED_BUTTON_PRESSED); |
| 75 | 72 |
| 76 SetBorder(CreateBorder(kNormalImageSet, kHotImageSet, kPushedImageSet)); | 73 SetBorder(CreateBorder(kNormalImageSet, kHotImageSet, kPushedImageSet)); |
| 77 set_menu_marker( | 74 generic_avatar_ = |
| 78 rb->GetImageNamed(IDR_AVATAR_THEMED_BUTTON_DROPARROW).ToImageSkia()); | 75 *rb->GetImageNamed(IDR_AVATAR_THEMED_BUTTON_AVATAR).ToImageSkia(); |
| 79 #if defined(OS_WIN) | 76 #if defined(OS_WIN) |
| 80 } else if (base::win::GetVersion() >= base::win::VERSION_WIN8) { | 77 } else if (base::win::GetVersion() >= base::win::VERSION_WIN8) { |
| 81 const int kNormalImageSet[] = IMAGE_GRID(IDR_AVATAR_METRO_BUTTON_NORMAL); | 78 const int kNormalImageSet[] = IMAGE_GRID(IDR_AVATAR_METRO_BUTTON_NORMAL); |
| 82 const int kHotImageSet[] = IMAGE_GRID(IDR_AVATAR_METRO_BUTTON_HOVER); | 79 const int kHotImageSet[] = IMAGE_GRID(IDR_AVATAR_METRO_BUTTON_HOVER); |
| 83 const int kPushedImageSet[] = IMAGE_GRID(IDR_AVATAR_METRO_BUTTON_PRESSED); | 80 const int kPushedImageSet[] = IMAGE_GRID(IDR_AVATAR_METRO_BUTTON_PRESSED); |
| 84 | 81 |
| 85 SetBorder(CreateBorder(kNormalImageSet, kHotImageSet, kPushedImageSet)); | 82 SetBorder(CreateBorder(kNormalImageSet, kHotImageSet, kPushedImageSet)); |
| 86 set_menu_marker( | 83 generic_avatar_ = |
| 87 rb->GetImageNamed(IDR_AVATAR_METRO_BUTTON_DROPARROW).ToImageSkia()); | 84 *rb->GetImageNamed(IDR_AVATAR_METRO_BUTTON_AVATAR).ToImageSkia(); |
| 88 #endif | 85 #endif |
| 89 } else { | 86 } else { |
| 90 const int kNormalImageSet[] = IMAGE_GRID(IDR_AVATAR_GLASS_BUTTON_NORMAL); | 87 const int kNormalImageSet[] = IMAGE_GRID(IDR_AVATAR_GLASS_BUTTON_NORMAL); |
| 91 const int kHotImageSet[] = IMAGE_GRID(IDR_AVATAR_GLASS_BUTTON_HOVER); | 88 const int kHotImageSet[] = IMAGE_GRID(IDR_AVATAR_GLASS_BUTTON_HOVER); |
| 92 const int kPushedImageSet[] = IMAGE_GRID(IDR_AVATAR_GLASS_BUTTON_PRESSED); | 89 const int kPushedImageSet[] = IMAGE_GRID(IDR_AVATAR_GLASS_BUTTON_PRESSED); |
| 93 | 90 |
| 94 SetBorder(CreateBorder(kNormalImageSet, kHotImageSet, kPushedImageSet)); | 91 SetBorder(CreateBorder(kNormalImageSet, kHotImageSet, kPushedImageSet)); |
| 95 set_menu_marker( | 92 generic_avatar_ = |
| 96 rb->GetImageNamed(IDR_AVATAR_GLASS_BUTTON_DROPARROW).ToImageSkia()); | 93 *rb->GetImageNamed(IDR_AVATAR_GLASS_BUTTON_AVATAR).ToImageSkia(); |
| 97 } | 94 } |
| 98 | 95 |
| 99 g_browser_process->profile_manager()->GetProfileInfoCache().AddObserver(this); | 96 g_browser_process->profile_manager()->GetProfileInfoCache().AddObserver(this); |
| 100 | 97 |
| 101 // Subscribe to authentication error changes so that the avatar button can | 98 // Subscribe to authentication error changes so that the avatar button can |
| 102 // update itself. Note that guest mode profiles won't have a token service. | 99 // update itself. Note that guest mode profiles won't have a token service. |
| 103 SigninErrorController* error = | 100 SigninErrorController* error = |
| 104 profiles::GetSigninErrorController(browser_->profile()); | 101 profiles::GetSigninErrorController(browser_->profile()); |
| 105 if (error) { | 102 if (error) { |
| 106 error->AddObserver(this); | 103 error->AddObserver(this); |
| 104 // This calls UpdateAvatarButtonAndRelayoutParent(). |
| 107 OnErrorChanged(); | 105 OnErrorChanged(); |
| 106 } else { |
| 107 UpdateAvatarButtonAndRelayoutParent(); |
| 108 } | 108 } |
| 109 | |
| 110 SchedulePaint(); | 109 SchedulePaint(); |
| 111 } | 110 } |
| 112 | 111 |
| 113 NewAvatarButton::~NewAvatarButton() { | 112 NewAvatarButton::~NewAvatarButton() { |
| 114 g_browser_process->profile_manager()-> | 113 g_browser_process->profile_manager()-> |
| 115 GetProfileInfoCache().RemoveObserver(this); | 114 GetProfileInfoCache().RemoveObserver(this); |
| 116 SigninErrorController* error = | 115 SigninErrorController* error = |
| 117 profiles::GetSigninErrorController(browser_->profile()); | 116 profiles::GetSigninErrorController(browser_->profile()); |
| 118 if (error) | 117 if (error) |
| 119 error->RemoveObserver(this); | 118 error->RemoveObserver(this); |
| 120 } | 119 } |
| 121 | 120 |
| 122 bool NewAvatarButton::OnMousePressed(const ui::MouseEvent& event) { | 121 bool NewAvatarButton::OnMousePressed(const ui::MouseEvent& event) { |
| 123 // Prevent the bubble from being re-shown if it's already showing. | 122 // Prevent the bubble from being re-shown if it's already showing. |
| 124 suppress_mouse_released_action_ = ProfileChooserView::IsShowing(); | 123 suppress_mouse_released_action_ = ProfileChooserView::IsShowing(); |
| 125 return MenuButton::OnMousePressed(event); | 124 return LabelButton::OnMousePressed(event); |
| 126 } | 125 } |
| 127 | 126 |
| 128 void NewAvatarButton::OnMouseReleased(const ui::MouseEvent& event) { | 127 void NewAvatarButton::OnMouseReleased(const ui::MouseEvent& event) { |
| 129 if (suppress_mouse_released_action_) | 128 if (suppress_mouse_released_action_) |
| 130 suppress_mouse_released_action_ = false; | 129 suppress_mouse_released_action_ = false; |
| 131 else | 130 else |
| 132 MenuButton::OnMouseReleased(event); | 131 LabelButton::OnMouseReleased(event); |
| 133 } | 132 } |
| 134 | 133 |
| 135 void NewAvatarButton::OnProfileAdded(const base::FilePath& profile_path) { | 134 void NewAvatarButton::OnProfileAdded(const base::FilePath& profile_path) { |
| 136 UpdateAvatarButtonAndRelayoutParent(); | 135 UpdateAvatarButtonAndRelayoutParent(); |
| 137 } | 136 } |
| 138 | 137 |
| 139 void NewAvatarButton::OnProfileWasRemoved( | 138 void NewAvatarButton::OnProfileWasRemoved( |
| 140 const base::FilePath& profile_path, | 139 const base::FilePath& profile_path, |
| 141 const base::string16& profile_name) { | 140 const base::string16& profile_name) { |
| 142 UpdateAvatarButtonAndRelayoutParent(); | 141 UpdateAvatarButtonAndRelayoutParent(); |
| 143 } | 142 } |
| 144 | 143 |
| 145 void NewAvatarButton::OnProfileNameChanged( | 144 void NewAvatarButton::OnProfileNameChanged( |
| 146 const base::FilePath& profile_path, | 145 const base::FilePath& profile_path, |
| 147 const base::string16& old_profile_name) { | 146 const base::string16& old_profile_name) { |
| 148 UpdateAvatarButtonAndRelayoutParent(); | 147 UpdateAvatarButtonAndRelayoutParent(); |
| 149 } | 148 } |
| 150 | 149 |
| 151 void NewAvatarButton::OnProfileAvatarChanged( | 150 void NewAvatarButton::OnProfileAvatarChanged( |
| 152 const base::FilePath& profile_path) { | 151 const base::FilePath& profile_path) { |
| 153 UpdateAvatarButtonAndRelayoutParent(); | 152 UpdateAvatarButtonAndRelayoutParent(); |
| 154 } | 153 } |
| 155 | 154 |
| 156 void NewAvatarButton::OnProfileSupervisedUserIdChanged( | 155 void NewAvatarButton::OnProfileSupervisedUserIdChanged( |
| 157 const base::FilePath& profile_path) { | 156 const base::FilePath& profile_path) { |
| 158 UpdateAvatarButtonAndRelayoutParent(); | 157 UpdateAvatarButtonAndRelayoutParent(); |
| 159 } | 158 } |
| 160 | 159 |
| 161 void NewAvatarButton::OnErrorChanged() { | 160 void NewAvatarButton::OnErrorChanged() { |
| 162 gfx::ImageSkia icon; | |
| 163 | |
| 164 // If there is an error, show an warning icon. | 161 // If there is an error, show an warning icon. |
| 165 const SigninErrorController* error = | 162 const SigninErrorController* error = |
| 166 profiles::GetSigninErrorController(browser_->profile()); | 163 profiles::GetSigninErrorController(browser_->profile()); |
| 167 if (error && error->HasError()) { | 164 has_auth_error_ = error && error->HasError(); |
| 168 icon = *ui::ResourceBundle::GetSharedInstance().GetImageNamed( | |
| 169 IDR_ICON_PROFILES_AVATAR_BUTTON_ERROR).ToImageSkia(); | |
| 170 } | |
| 171 | 165 |
| 172 SetImage(views::Button::STATE_NORMAL, icon); | |
| 173 UpdateAvatarButtonAndRelayoutParent(); | 166 UpdateAvatarButtonAndRelayoutParent(); |
| 174 } | 167 } |
| 175 | 168 |
| 176 void NewAvatarButton::UpdateAvatarButtonAndRelayoutParent() { | 169 void NewAvatarButton::UpdateAvatarButtonAndRelayoutParent() { |
| 170 const ProfileInfoCache& cache = |
| 171 g_browser_process->profile_manager()->GetProfileInfoCache(); |
| 172 |
| 173 // If we have a single local profile, then use the generic avatar |
| 174 // button instead of the profile name. Never use the generic button if |
| 175 // the active profile is Guest. |
| 176 bool use_generic_button = (!browser_->profile()->IsGuestSession() && |
| 177 cache.GetNumberOfProfiles() == 1 && |
| 178 cache.GetUserNameOfProfileAtIndex(0).empty()); |
| 179 |
| 180 SetText(use_generic_button ? base::string16() : |
| 181 profiles::GetAvatarButtonTextForProfile(browser_->profile())); |
| 177 // We want the button to resize if the new text is shorter. | 182 // We want the button to resize if the new text is shorter. |
| 178 SetText(profiles::GetAvatarButtonTextForProfile(browser_->profile())); | |
| 179 SetMinSize(gfx::Size()); | 183 SetMinSize(gfx::Size()); |
| 184 |
| 185 if (use_generic_button) { |
| 186 SetImage(views::Button::STATE_NORMAL, generic_avatar_); |
| 187 } else if (has_auth_error_) { |
| 188 SetImage(views::Button::STATE_NORMAL, |
| 189 *ui::ResourceBundle::GetSharedInstance().GetImageNamed( |
| 190 IDR_ICON_PROFILES_AVATAR_BUTTON_ERROR).ToImageSkia()); |
| 191 } else { |
| 192 SetImage(views::Button::STATE_NORMAL, gfx::ImageSkia()); |
| 193 } |
| 194 |
| 195 // If we are not using the generic button, then reset the spacing between |
| 196 // the text and the possible authentication error icon. |
| 197 const int kDefaultImageTextSpacing = 5; |
| 198 SetImageLabelSpacing(use_generic_button ? 0 : kDefaultImageTextSpacing); |
| 199 |
| 180 InvalidateLayout(); | 200 InvalidateLayout(); |
| 181 | 201 |
| 182 // Because the width of the button might have changed, the parent browser | 202 // Because the width of the button might have changed, the parent browser |
| 183 // frame needs to recalculate the button bounds and redraw it. | 203 // frame needs to recalculate the button bounds and redraw it. |
| 184 if (parent()) | 204 if (parent()) |
| 185 parent()->Layout(); | 205 parent()->Layout(); |
| 186 } | 206 } |
| OLD | NEW |