Home | History | Annotate | Download | only in login
      1 // Copyright (c) 2011 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #include "chrome/browser/chromeos/login/username_view.h"
      6 
      7 #include "base/logging.h"
      8 #include "base/utf_string_conversions.h"
      9 #include "chrome/browser/chromeos/login/rounded_view.h"
     10 #include "grit/generated_resources.h"
     11 #include "third_party/skia/include/core/SkColorShader.h"
     12 #include "third_party/skia/include/core/SkComposeShader.h"
     13 #include "third_party/skia/include/effects/SkGradientShader.h"
     14 #include "ui/base/l10n/l10n_util.h"
     15 #include "ui/base/resource/resource_bundle.h"
     16 #include "ui/gfx/canvas.h"
     17 #include "ui/gfx/canvas_skia.h"
     18 #include "ui/gfx/gtk_util.h"
     19 #include "ui/gfx/rect.h"
     20 
     21 namespace chromeos {
     22 
     23 namespace {
     24 // Username label background color.
     25 const SkColor kLabelBackgoundColor = 0x55000000;
     26 // Holds margin to height ratio.
     27 const double kMarginRatio = 1.0 / 3.0;
     28 // Holds the frame width for the small shaped username view.
     29 const SkScalar kSmallShapeFrameWidth = SkIntToScalar(1);
     30 
     31 // Class that sets up half rounded rectangle (only the bottom corners are
     32 // rounded) as a clip region of the view.
     33 // For more info see the file "chrome/browser/chromeos/login/rounded_view.h".
     34 template<typename C>
     35 class HalfRoundedView : public RoundedView<C> {
     36  public:
     37   HalfRoundedView(const std::wstring &text, bool use_small_shape)
     38       : RoundedView<C>(text, use_small_shape) {
     39   }
     40 
     41  protected:
     42   // Overrides ViewFilter.
     43   virtual SkPath GetClipPath() const {
     44     if (!C::use_small_shape()) {
     45       return RoundedView<C>::GetClipPath();
     46     } else {
     47       SkPath path;
     48       gfx::Rect frame_bounds = this->bounds();
     49       frame_bounds.Inset(kSmallShapeFrameWidth, kSmallShapeFrameWidth,
     50                    kSmallShapeFrameWidth, kSmallShapeFrameWidth);
     51       path.addRect(SkIntToScalar(frame_bounds.x()),
     52                    SkIntToScalar(frame_bounds.y()),
     53                    SkIntToScalar(frame_bounds.x() + frame_bounds.width()),
     54                    SkIntToScalar(frame_bounds.y() + frame_bounds.height()));
     55       return path;
     56     }
     57   }
     58 
     59   virtual void DrawFrame(gfx::Canvas* canvas) {
     60     // No frame is needed.
     61   }
     62 
     63   virtual SkRect GetViewRect() const {
     64     SkRect view_rect;
     65     // The rectangle will be intersected with the bounds, so the correct half
     66     // of the round rectangle will be obtained.
     67     view_rect.iset(this->x(),
     68                    this->y() - this->height(),
     69                    this->x() + this->width(),
     70                    this->y() + this->height());
     71     return view_rect;
     72   }
     73 };
     74 
     75 }  // namespace
     76 
     77 UsernameView::UsernameView(const std::wstring& username, bool use_small_shape)
     78     : views::Label(username.empty()
     79           ? UTF16ToWide(l10n_util::GetStringUTF16(IDS_GUEST)) : username),
     80       use_small_shape_(use_small_shape),
     81       is_guest_(username.empty()) {
     82 }
     83 
     84 void UsernameView::OnPaint(gfx::Canvas* canvas) {
     85   gfx::Rect bounds = GetContentsBounds();
     86   if (text_image_ == NULL)
     87     PaintUsername(bounds);
     88   DCHECK(text_image_ != NULL);
     89   DCHECK(bounds.size() ==
     90          gfx::Size(text_image_->width(), text_image_->height()));
     91   canvas->DrawBitmapInt(*text_image_, bounds.x(), bounds.y());
     92 }
     93 
     94 // static
     95 UsernameView* UsernameView::CreateShapedUsernameView(
     96     const std::wstring& username, bool use_small_shape) {
     97   return new HalfRoundedView<UsernameView>(username, use_small_shape);
     98 }
     99 
    100 gfx::NativeCursor UsernameView::GetCursorForPoint(
    101     ui::EventType event_type,
    102     const gfx::Point& p) {
    103   return use_small_shape_ ? gfx::GetCursor(GDK_HAND2) : NULL;
    104 }
    105 
    106 void UsernameView::PaintUsername(const gfx::Rect& bounds) {
    107   margin_width_ = bounds.height() * kMarginRatio;
    108   gfx::CanvasSkia canvas(bounds.width(), bounds.height(), false);
    109   // Draw transparent background.
    110   canvas.drawColor(0);
    111 
    112   // Calculate needed space.
    113   int flags = gfx::Canvas::TEXT_ALIGN_LEFT |
    114       gfx::Canvas::TEXT_VALIGN_MIDDLE |
    115       gfx::Canvas::NO_ELLIPSIS;
    116   int text_height, text_width;
    117   gfx::CanvasSkia::SizeStringInt(WideToUTF16Hack(GetText()), font(),
    118                                  &text_width, &text_height,
    119                                  flags);
    120   text_width += margin_width_;
    121 
    122   // Also leave the right margin.
    123   bool use_fading_for_text = text_width + margin_width_ >= bounds.width();
    124 
    125   // Only alpha channel counts.
    126   SkColor gradient_colors[2];
    127   gradient_colors[0] = 0xFFFFFFFF;
    128   gradient_colors[1] = 0x00FFFFFF;
    129 
    130   int gradient_start = use_fading_for_text ?
    131                        bounds.width() - bounds.height() - margin_width_ :
    132                        text_width;
    133   int gradient_end = std::min(gradient_start + bounds.height(),
    134                               bounds.width() - margin_width_);
    135 
    136   SkPoint gradient_borders[2];
    137   gradient_borders[0].set(SkIntToScalar(gradient_start), SkIntToScalar(0));
    138   gradient_borders[1].set(SkIntToScalar(gradient_end), SkIntToScalar(0));
    139 
    140   SkShader* gradient_shader =
    141       SkGradientShader::CreateLinear(gradient_borders, gradient_colors, NULL, 2,
    142                                      SkShader::kClamp_TileMode, NULL);
    143 
    144   if (!use_fading_for_text) {
    145     // Draw the final background with the fading in the end.
    146     SkShader* solid_shader = new SkColorShader(kLabelBackgoundColor);
    147     SkXfermode* mode = SkXfermode::Create(SkXfermode::kSrcIn_Mode);
    148     SkShader* composite_shader = new SkComposeShader(gradient_shader,
    149                                                      solid_shader, mode);
    150     gradient_shader->unref();
    151     solid_shader->unref();
    152 
    153     SkPaint paint;
    154     paint.setShader(composite_shader)->unref();
    155     canvas.drawPaint(paint);
    156   }
    157 
    158   // Draw the text.
    159   // Note, direct call of the DrawStringInt method produces the green dots
    160   // along the text perimeter (when the label is place on the white background).
    161   SkColor kInvisibleHaloColor = 0x00000000;
    162   canvas.DrawStringWithHalo(WideToUTF16Hack(GetText()), font(), GetColor(),
    163                             kInvisibleHaloColor, bounds.x() + margin_width_,
    164                             bounds.y(), bounds.width() - 2 * margin_width_,
    165                             bounds.height(), flags);
    166 
    167   text_image_.reset(new SkBitmap(canvas.ExtractBitmap()));
    168 
    169   if (use_fading_for_text) {
    170     // Fade out only the text in the end. Use regular background.
    171     canvas.drawColor(kLabelBackgoundColor, SkXfermode::kSrc_Mode);
    172     SkShader* image_shader = SkShader::CreateBitmapShader(
    173         *text_image_,
    174         SkShader::kRepeat_TileMode,
    175         SkShader::kRepeat_TileMode);
    176     SkXfermode* mode = SkXfermode::Create(SkXfermode::kSrcIn_Mode);
    177     SkShader* composite_shader = new SkComposeShader(gradient_shader,
    178                                                      image_shader, mode);
    179     gradient_shader->unref();
    180     image_shader->unref();
    181 
    182     SkPaint paint;
    183     paint.setShader(composite_shader)->unref();
    184     canvas.drawPaint(paint);
    185     text_image_.reset(new SkBitmap(canvas.ExtractBitmap()));
    186   }
    187 }
    188 
    189 void UsernameView::OnLocaleChanged() {
    190   if (is_guest_) {
    191     SetText(UTF16ToWide(l10n_util::GetStringUTF16(IDS_GUEST)));
    192   }
    193   // Repaint because the font may have changed.
    194   text_image_.reset();
    195   SchedulePaint();
    196 }
    197 
    198 }  // namespace chromeos
    199