Chromium Code Reviews| Index: views/controls/scrollbar/native_scroll_bar_views.cc |
| diff --git a/views/controls/scrollbar/native_scroll_bar_views.cc b/views/controls/scrollbar/native_scroll_bar_views.cc |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..be1fce50f2774c9bb1aa8b25ec5ef1d9300db618 |
| --- /dev/null |
| +++ b/views/controls/scrollbar/native_scroll_bar_views.cc |
| @@ -0,0 +1,571 @@ |
| +// Copyright (c) 2011 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 "views/controls/scrollbar/native_scroll_bar_views.h" |
| + |
| +#include "base/logging.h" |
| +#include "ui/base/keycodes/keyboard_codes.h" |
| +#include "ui/gfx/canvas.h" |
| +#include "ui/gfx/canvas_skia.h" |
| +#include "ui/gfx/path.h" |
| +#include "views/controls/button/custom_button.h" |
| +#include "views/controls/focusable_border.h" |
| +#include "views/controls/scrollbar/native_scroll_bar.h" |
| +#include "views/controls/scrollbar/scroll_bar.h" |
| + |
| +namespace views { |
| + |
| +namespace { |
| + |
| +// Wrapper for the scroll buttons. |
| +class ScrollBarButton : public CustomButton { |
|
Ben Goodger (Google)
2011/08/19 17:44:33
Rather than reinvent the wheel here I'd rather you
|
| + public: |
| + enum Type { |
| + UP, |
| + DOWN, |
| + LEFT, |
| + RIGHT, |
| + }; |
| + |
| + ScrollBarButton(ButtonListener* listener, Type type); |
| + virtual ~ScrollBarButton(); |
| + |
| + virtual gfx::Size GetPreferredSize() OVERRIDE; |
| + |
| + protected: |
| + virtual bool OnMousePressed(const MouseEvent& event) OVERRIDE; |
| + virtual void OnMouseReleased(const MouseEvent& event) OVERRIDE; |
| + virtual void OnMouseEntered(const MouseEvent& event) OVERRIDE; |
| + virtual void OnMouseExited(const MouseEvent& event) OVERRIDE; |
| + virtual void OnPaint(gfx::Canvas* canvas) OVERRIDE; |
| + |
| + private: |
| + gfx::NativeTheme::ExtraParams params_; |
| + gfx::NativeTheme::Part part_; |
| + gfx::NativeTheme::State state_; |
| +}; |
| + |
| +} // namespace |
| + |
| +// Wrapper for the scroll thumb |
| +class ScrollBarThumb : public View { |
| + public: |
| + enum Type { |
| + VERTICAL, |
| + HORIZONTAL, |
| + }; |
| + |
| + ScrollBarThumb(NativeScrollBarWrapper* wrapper, Type type); |
| + virtual ~ScrollBarThumb(); |
| + |
| + virtual gfx::Size GetPreferredSize() OVERRIDE; |
| + |
| + void Update(int position); |
| + void Update(int viewport_size, int content_size, int position); |
| + |
| + int current_position() const { return current_position_; } |
| + |
| + // This is called when the layout changes and the scrollbar changes |
| + // size. |
| + void set_track_bounds(const gfx::Rect& bounds) { track_bounds_ = bounds; } |
| + |
| + protected: |
| + virtual bool OnMousePressed(const MouseEvent& event) OVERRIDE; |
| + virtual bool OnMouseDragged(const MouseEvent& event) OVERRIDE; |
| + virtual void OnMouseEntered(const MouseEvent& event) OVERRIDE; |
| + virtual void OnMouseExited(const MouseEvent& event) OVERRIDE; |
| + virtual void OnMouseMoved(const MouseEvent& event) OVERRIDE; |
| + virtual void OnPaint(gfx::Canvas* canvas) OVERRIDE; |
| + |
| + private: |
| + // Converts the position of the thumb to the pixel size of the thumb |
| + // relative to the size of the drawing area. |
| + int GetThumbPosition(int drawing_size) const; |
| + |
| + // Returns the size of the thumb relative to the drawing area size. |
| + int GetThumbSize(int drawing_size) const; |
| + |
| + NativeScrollBarWrapper* wrapper_; |
| + |
| + Type type_; |
| + |
| + // The current position of the thumb. |
| + int current_position_; |
| + |
| + // The size of the content we want to display. |
| + int content_size_; |
| + // The size of the viewport (displayable area). |
| + int viewport_size_; |
| + |
| + // The track bounds. We need this to compute the size of the thumb relative |
| + // to the size of the track. |
| + gfx::Rect track_bounds_; |
| + |
| + // The distance from the tip of the thumb. Used when dragging to keep the |
| + // position of the thumb at the same relative distance from the mouse click |
| + // event. |
| + int delta_; |
| + |
| + // Parameters required for the native theme. We keep these here since |
| + // we may change them based on mouse events (hover, click, ...). |
| + gfx::NativeTheme::ExtraParams params_; |
| + gfx::NativeTheme::Part part_; |
| + gfx::NativeTheme::State state_; |
| +}; |
| + |
| +namespace { |
| + |
| +///////////////////////////////////////////////////////////////////////////// |
| +// ScrollBarButton |
| + |
| +ScrollBarButton::ScrollBarButton(ButtonListener* listener, Type type) |
| + : CustomButton(listener), |
| + params_(), |
| + state_(gfx::NativeTheme::kNormal) { |
| + switch (type) { |
| + case UP: |
| + part_ = gfx::NativeTheme::kScrollbarUpArrow; |
| + break; |
| + case DOWN: |
| + part_ = gfx::NativeTheme::kScrollbarDownArrow; |
| + break; |
| + case LEFT: |
| + part_ = gfx::NativeTheme::kScrollbarLeftArrow; |
| + break; |
| + case RIGHT: |
| + part_ = gfx::NativeTheme::kScrollbarRightArrow; |
| + break; |
| + } |
| + |
| + params_.scrollbar_arrow.is_hovering = false; |
| +} |
| + |
| +ScrollBarButton::~ScrollBarButton() { |
| +} |
| + |
| +gfx::Size ScrollBarButton::GetPreferredSize() { |
| + const gfx::NativeTheme* native_theme = gfx::NativeTheme::instance(); |
| + return native_theme->GetPartSize(part_, state_, params_); |
| +} |
| + |
| +bool ScrollBarButton::OnMousePressed(const MouseEvent& event) { |
| + if (event.IsOnlyLeftMouseButton()) |
| + Button::NotifyClick(event); |
| + return true; |
| +} |
| + |
| +void ScrollBarButton::OnMouseReleased(const MouseEvent& event) { |
| +} |
| + |
| +void ScrollBarButton::OnMouseEntered(const MouseEvent& event) { |
| + state_ = gfx::NativeTheme::kHovered; |
| + params_.scrollbar_arrow.is_hovering = true; |
| + SchedulePaint(); |
| +} |
| + |
| +void ScrollBarButton::OnMouseExited(const MouseEvent& event) { |
| + state_ = gfx::NativeTheme::kNormal; |
| + params_.scrollbar_arrow.is_hovering = false; |
| + SchedulePaint(); |
| +} |
| + |
| +void ScrollBarButton::OnPaint(gfx::Canvas* canvas) { |
| + const gfx::NativeTheme* native_theme = gfx::NativeTheme::instance(); |
| + gfx::Rect bounds; |
| + bounds.set_size(GetPreferredSize()); |
| + |
| + native_theme->Paint(canvas->AsCanvasSkia(), |
| + part_, |
| + state_, |
| + bounds, |
| + params_); |
| +} |
| + |
| +} // namespace |
| + |
| +///////////////////////////////////////////////////////////////////////////// |
| +// ScrollBarThumb |
| + |
| +ScrollBarThumb::ScrollBarThumb(NativeScrollBarWrapper* wrapper, Type type) |
| + : wrapper_(wrapper), |
| + type_(type), |
| + state_(gfx::NativeTheme::kNormal) { |
| + switch (type) { |
| + case VERTICAL: |
| + part_ = gfx::NativeTheme::kScrollbarVerticalThumb; |
| + break; |
| + case HORIZONTAL: |
| + part_ = gfx::NativeTheme::kScrollbarHorizontalThumb; |
| + break; |
| + } |
| + params_.scrollbar_thumb.is_hovering = false; |
| +} |
| + |
| +ScrollBarThumb::~ScrollBarThumb() { |
| +} |
| + |
| +gfx::Size ScrollBarThumb::GetPreferredSize() { |
| + const gfx::NativeTheme* native_theme = gfx::NativeTheme::instance(); |
| + return native_theme->GetPartSize(part_, state_, params_); |
| +} |
| + |
| +void ScrollBarThumb::Update(int position) { |
| + Update(viewport_size_, content_size_, position); |
| +} |
| + |
| +void ScrollBarThumb::Update(int viewport_size, |
| + int content_size, |
| + int position) { |
| + // First constrain the size and position to the actual values used by the |
| + // scrollbar. |
| + if (content_size < 0) |
| + content_size = 0; |
| + |
| + if (position < 0) |
| + position = 0; |
| + |
| + if (position > content_size - viewport_size) |
| + position = content_size - viewport_size; |
| + |
| + current_position_ = position; |
| + content_size_ = content_size; |
| + viewport_size_ = viewport_size; |
| + |
| + gfx::Rect bounds = GetLocalBounds(); |
| + |
| + if (type_ == HORIZONTAL) { |
| + bounds.set_x(track_bounds_.x() + |
| + GetThumbPosition(track_bounds_.width())); |
| + bounds.set_width(GetThumbSize(track_bounds_.width())); |
| + } else { |
| + bounds.set_y(track_bounds_.y() + |
| + GetThumbPosition(track_bounds_.height())); |
| + bounds.set_height(GetThumbSize(track_bounds_.height())); |
| + } |
| + |
| + // Update the position and size of the thumb. |
| + SetBoundsRect(bounds); |
| + |
| + SchedulePaint(); |
| +} |
| + |
| +bool ScrollBarThumb::OnMousePressed(const MouseEvent& event) { |
| + if (!event.IsOnlyLeftMouseButton()) |
| + return false; |
| + |
| + // Keep the distance from the tip of the thumb so that scrolling movement |
| + // looks nice. (We want events relative to the mouse click.) |
| + if (type_ == HORIZONTAL) { |
| + int track_size = track_bounds_.width(); |
| + delta_ = -event.x() * content_size_ / track_size; |
| + } else { |
| + int track_size = track_bounds_.height(); |
| + delta_ = -event.y() * content_size_ / track_size; |
| + } |
| + |
| + return true; |
| +} |
| + |
| + |
| +bool ScrollBarThumb::OnMouseDragged(const MouseEvent& event) { |
| + // Let the scrollbar wrapper know that we're changing the current position. |
| + // It should then all for UI updates. |
| + if (type_ == HORIZONTAL) { |
| + int track_size = track_bounds_.width(); |
| + wrapper_->Update(viewport_size_, |
| + content_size_, |
| + current_position_ + delta_ + |
| + event.x() * content_size_ / track_size); |
| + } else { |
| + int track_size = track_bounds_.height(); |
| + wrapper_->Update(viewport_size_, |
| + content_size_, |
| + current_position_ + delta_ + |
| + event.y() * content_size_ / track_size); |
| + } |
| + |
| + return true; |
| +} |
| + |
| +void ScrollBarThumb::OnMouseEntered(const MouseEvent& event) { |
| + if (state_ != gfx::NativeTheme::kHovered) { |
| + state_ = gfx::NativeTheme::kHovered; |
| + params_.scrollbar_thumb.is_hovering = true; |
| + SchedulePaint(); |
| + } |
| +} |
| + |
| +void ScrollBarThumb::OnMouseExited(const MouseEvent& event) { |
| + if (state_ != gfx::NativeTheme::kNormal) { |
| + state_ = gfx::NativeTheme::kNormal; |
| + params_.scrollbar_thumb.is_hovering = false; |
| + SchedulePaint(); |
| + } |
| +} |
| + |
| +void ScrollBarThumb::OnMouseMoved(const MouseEvent& event) { |
| + OnMouseEntered(event); |
| +} |
| + |
| +void ScrollBarThumb::OnPaint(gfx::Canvas* canvas) { |
| + const gfx::NativeTheme* native_theme = gfx::NativeTheme::instance(); |
| + |
| + native_theme->Paint(canvas->AsCanvasSkia(), |
| + part_, |
| + state_, |
| + GetLocalBounds(), |
| + params_); |
| +} |
| + |
| +int ScrollBarThumb::GetThumbSize(int drawing_size) const { |
| + return (viewport_size_ * drawing_size) / |
| + (content_size_ ? content_size_ : 1); |
| +} |
| + |
| +int ScrollBarThumb::GetThumbPosition(int drawing_size) const { |
| + return (current_position_ * drawing_size) / |
| + (content_size_ ? content_size_ : 1); |
| +} |
| + |
| +//////////////////////////////////////////////////////////////////////////////// |
| +// NativeScrollBarViews, public: |
| + |
| +NativeScrollBarViews::NativeScrollBarViews(NativeScrollBar* scroll_bar) |
| + : native_scroll_bar_(scroll_bar) { |
| + if (native_scroll_bar_->IsHorizontal()) { |
| + prev_button_ = new ScrollBarButton(this, ScrollBarButton::LEFT); |
| + next_button_ = new ScrollBarButton(this, ScrollBarButton::RIGHT); |
| + thumb_ = new ScrollBarThumb(this, ScrollBarThumb::HORIZONTAL); |
| + |
| + part_ = gfx::NativeTheme::kScrollbarHorizontalTrack; |
| + } else { |
| + prev_button_ = new ScrollBarButton(this, ScrollBarButton::UP); |
| + next_button_ = new ScrollBarButton(this, ScrollBarButton::DOWN); |
| + thumb_ = new ScrollBarThumb(this, ScrollBarThumb::VERTICAL); |
| + |
| + part_ = gfx::NativeTheme::kScrollbarVerticalTrack; |
| + } |
| + |
| + state_ = gfx::NativeTheme::kNormal; |
| + |
| + AddChildView(prev_button_); |
| + AddChildView(next_button_); |
| + AddChildView(thumb_); |
| +} |
| + |
| +NativeScrollBarViews::~NativeScrollBarViews() { |
| +} |
| + |
| +//////////////////////////////////////////////////////////////////////////////// |
| +// NativeScrollBarViews, View overrides: |
| + |
| +void NativeScrollBarViews::Layout() { |
| + SetBoundsRect(native_scroll_bar_->GetLocalBounds()); |
| + |
| + gfx::Size size = prev_button_->GetPreferredSize(); |
| + prev_button_->SetBounds(0, 0, size.width(), size.height()); |
| + |
| + if (native_scroll_bar_->IsHorizontal()) { |
| + next_button_->SetBounds(width() - size.width(), 0, |
| + size.width(), size.height()); |
| + } else { |
| + next_button_->SetBounds(0, height() - size.height(), |
| + size.width(), size.height()); |
| + } |
| + |
| + // Update the size of the thumb and tell it that the track size also changed. |
| + thumb_->set_track_bounds(GetTrackBounds()); |
| + thumb_->SetBoundsRect(GetTrackBounds()); |
| +} |
| + |
| +gfx::Size NativeScrollBarViews::GetPreferredSize() { |
| + if (native_scroll_bar_->IsHorizontal()) |
| + return gfx::Size(0, GetHorizontalScrollBarHeight()); |
| + return gfx::Size(GetVerticalScrollBarWidth(), 0); |
| +} |
| + |
| +// TODO(oshima|jcampan): key/mouse events are not delievered and |
| +// the following code is not tested. It requires the focus manager to be fully |
| +// implemented. |
| +bool NativeScrollBarViews::OnKeyPressed(const KeyEvent& event) { |
| + switch (event.key_code()) { |
| + case ui::VKEY_UP: |
| + if (!native_scroll_bar_->IsHorizontal()) |
| + MoveStep(false /* negative */); |
| + break; |
| + case ui::VKEY_DOWN: |
| + if (!native_scroll_bar_->IsHorizontal()) |
| + MoveStep(true /* positive */); |
| + break; |
| + case ui::VKEY_LEFT: |
| + if (native_scroll_bar_->IsHorizontal()) |
| + MoveStep(false /* negative */); |
| + break; |
| + case ui::VKEY_RIGHT: |
| + if (native_scroll_bar_->IsHorizontal()) |
| + MoveStep(true /* positive */); |
| + break; |
| + case ui::VKEY_PRIOR: |
| + MovePage(false /* negative */); |
| + break; |
| + case ui::VKEY_NEXT: |
| + MovePage(true /* positive */); |
| + break; |
| + case ui::VKEY_HOME: |
| + MoveTo(0); |
| + break; |
| + case ui::VKEY_END: |
| + MoveToBottom(); |
| + break; |
| + default: |
| + return false; |
| + } |
| + return true; |
| +} |
| + |
| +bool NativeScrollBarViews::OnMousePressed(const MouseEvent& event) { |
| + if (!event.IsOnlyLeftMouseButton()) |
| + return true; |
| + |
| + if (native_scroll_bar_->IsHorizontal()) { |
| + if (event.x() < thumb_->bounds().x()) |
| + MovePage(false); |
| + else |
| + MovePage(true); |
| + } else { |
| + if (event.y() < thumb_->bounds().y()) |
| + MovePage(false); |
| + else |
| + MovePage(true); |
| + } |
| + |
| + return true; |
| +} |
| + |
| +void NativeScrollBarViews::OnMouseEntered(const MouseEvent& event) { |
| + state_ = gfx::NativeTheme::kHovered; |
| + SchedulePaint(); |
| +} |
| + |
| +void NativeScrollBarViews::OnMouseExited(const MouseEvent& event) { |
| + state_ = gfx::NativeTheme::kNormal; |
| + SchedulePaint(); |
| +} |
| + |
| +bool NativeScrollBarViews::OnMouseWheel(const MouseWheelEvent& e) { |
| + MoveBy(-e.offset()); // e.GetOffset() > 0 means scroll up. |
| + return true; |
| +} |
| + |
| +void NativeScrollBarViews::OnPaint(gfx::Canvas* canvas) { |
| + const gfx::NativeTheme* native_theme = gfx::NativeTheme::instance(); |
| + gfx::Rect bounds = GetTrackBounds(); |
| + |
| + params_.scrollbar_track.track_x = bounds.x(); |
| + params_.scrollbar_track.track_y = bounds.y(); |
| + params_.scrollbar_track.track_width = bounds.width(); |
| + params_.scrollbar_track.track_height = bounds.height(); |
| + |
| + |
| + native_theme->Paint(canvas->AsCanvasSkia(), |
| + part_, |
| + state_, |
| + bounds, |
| + params_); |
| +} |
| + |
| +void NativeScrollBarViews::ButtonPressed(Button* sender, |
| + const views::Event& event) { |
| + if (sender == prev_button_) { |
| + MoveStep(false); |
| + } else if (sender == next_button_) { |
| + MoveStep(true); |
| + } |
| +} |
| + |
| +//////////////////////////////////////////////////////////////////////////////// |
| +// NativeScrollBarViews, NativeScrollBarWrapper overrides: |
| + |
| +int NativeScrollBarViews::GetPosition() const { |
| + return thumb_->current_position(); |
| +} |
| + |
| +View* NativeScrollBarViews::GetView() { |
| + return this; |
| +} |
| + |
| +void NativeScrollBarViews::Update(int viewport_size, |
| + int content_size, |
| + int current_pos) { |
| + thumb_->Update(viewport_size, content_size, current_pos); |
| + |
| + ValueChanged(); |
| +} |
| + |
| +//////////////////////////////////////////////////////////////////////////////// |
| +// NativeScrollBarViews, private: |
| + |
| +void NativeScrollBarViews::ValueChanged() { |
| + ScrollBarController* controller = native_scroll_bar_->GetController(); |
| + controller->ScrollToPosition(native_scroll_bar_, GetPosition()); |
| +} |
| + |
| +void NativeScrollBarViews::MoveBy(int o) { |
| + MoveTo(GetPosition() + o); |
| +} |
| + |
| +void NativeScrollBarViews::MovePage(bool positive) { |
| + ScrollBarController* controller = native_scroll_bar_->GetController(); |
| + int scroll_increment = controller->GetScrollIncrement(native_scroll_bar_, |
| + true, |
| + positive); |
| + if (!positive) |
| + scroll_increment *= -1; |
| + |
| + MoveBy(scroll_increment); |
| +} |
| + |
| +void NativeScrollBarViews::MoveStep(bool positive) { |
| + ScrollBarController* controller = native_scroll_bar_->GetController(); |
| + int scroll_increment = controller->GetScrollIncrement(native_scroll_bar_, |
| + false, |
| + positive); |
| + if (!positive) |
| + scroll_increment *= -1; |
| + |
| + MoveBy(scroll_increment); |
| +} |
| + |
| +void NativeScrollBarViews::MoveTo(int p) { |
| + if (p < native_scroll_bar_->GetMinPosition()) |
| + p = native_scroll_bar_->GetMinPosition(); |
| + if (p > native_scroll_bar_->GetMaxPosition()) |
| + p = native_scroll_bar_->GetMaxPosition(); |
| + |
| + thumb_->Update(p); |
| + ValueChanged(); |
| +} |
| + |
| +void NativeScrollBarViews::MoveToBottom() { |
| + MoveTo(native_scroll_bar_->GetMaxPosition()); |
| +} |
| + |
| +gfx::Rect NativeScrollBarViews::GetTrackBounds() const { |
| + gfx::Rect bounds = GetLocalBounds(); |
| + gfx::Size size = prev_button_->GetPreferredSize(); |
| + |
| + if (native_scroll_bar_->IsHorizontal()) { |
| + bounds.set_x(bounds.x() + size.width()); |
| + bounds.set_width(bounds.width() - 2 * size.width()); |
| + bounds.set_height(thumb_->GetPreferredSize().height()); |
| + } else { |
| + bounds.set_y(bounds.y() + size.height()); |
| + bounds.set_height(bounds.height() - 2 * size.height()); |
| + bounds.set_width(thumb_->GetPreferredSize().width()); |
| + } |
| + |
| + return bounds; |
| +} |
| + |
| +} // namespace views |