OLD | NEW |
(Empty) | |
| 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 "views/controls/scrollbar/base_scroll_bar.h" |
| 6 |
| 7 #if defined(OS_LINUX) |
| 8 #include "ui/gfx/screen.h" |
| 9 #endif |
| 10 |
| 11 #include "base/callback.h" |
| 12 #include "base/compiler_specific.h" |
| 13 #include "base/message_loop.h" |
| 14 #include "base/string16.h" |
| 15 #include "base/utf_string_conversions.h" |
| 16 #include "grit/ui_strings.h" |
| 17 #include "ui/base/keycodes/keyboard_codes.h" |
| 18 #include "ui/base/l10n/l10n_util.h" |
| 19 #include "ui/gfx/canvas.h" |
| 20 #include "views/controls/menu/menu.h" |
| 21 #include "views/controls/scrollbar/base_scroll_bar_thumb.h" |
| 22 #include "views/controls/scroll_view.h" |
| 23 #include "views/widget/widget.h" |
| 24 |
| 25 #undef min |
| 26 #undef max |
| 27 |
| 28 namespace views { |
| 29 |
| 30 /////////////////////////////////////////////////////////////////////////////// |
| 31 // BaseScrollBar, public: |
| 32 |
| 33 BaseScrollBar::BaseScrollBar(bool horizontal, BaseScrollBarThumb* thumb) |
| 34 : ScrollBar(horizontal), |
| 35 thumb_(thumb), |
| 36 contents_size_(0), |
| 37 contents_scroll_offset_(0), |
| 38 thumb_track_state_(CustomButton::BS_NORMAL), |
| 39 last_scroll_amount_(SCROLL_NONE), |
| 40 ALLOW_THIS_IN_INITIALIZER_LIST(repeater_( |
| 41 NewCallback<BaseScrollBar>(this, |
| 42 &BaseScrollBar::TrackClicked))), |
| 43 context_menu_mouse_position_(0) { |
| 44 AddChildView(thumb_); |
| 45 |
| 46 set_context_menu_controller(this); |
| 47 thumb_->set_context_menu_controller(this); |
| 48 } |
| 49 |
| 50 void BaseScrollBar::ScrollByAmount(ScrollAmount amount) { |
| 51 int offset = contents_scroll_offset_; |
| 52 switch (amount) { |
| 53 case SCROLL_START: |
| 54 offset = GetMinPosition(); |
| 55 break; |
| 56 case SCROLL_END: |
| 57 offset = GetMaxPosition(); |
| 58 break; |
| 59 case SCROLL_PREV_LINE: |
| 60 offset -= GetScrollIncrement(false, false); |
| 61 offset = std::max(GetMinPosition(), offset); |
| 62 break; |
| 63 case SCROLL_NEXT_LINE: |
| 64 offset += GetScrollIncrement(false, true); |
| 65 offset = std::min(GetMaxPosition(), offset); |
| 66 break; |
| 67 case SCROLL_PREV_PAGE: |
| 68 offset -= GetScrollIncrement(true, false); |
| 69 offset = std::max(GetMinPosition(), offset); |
| 70 break; |
| 71 case SCROLL_NEXT_PAGE: |
| 72 offset += GetScrollIncrement(true, true); |
| 73 offset = std::min(GetMaxPosition(), offset); |
| 74 break; |
| 75 default: |
| 76 break; |
| 77 } |
| 78 contents_scroll_offset_ = offset; |
| 79 ScrollContentsToOffset(); |
| 80 } |
| 81 |
| 82 void BaseScrollBar::ScrollToThumbPosition(int thumb_position, |
| 83 bool scroll_to_middle) { |
| 84 contents_scroll_offset_ = |
| 85 CalculateContentsOffset(thumb_position, scroll_to_middle); |
| 86 if (contents_scroll_offset_ < GetMinPosition()) { |
| 87 contents_scroll_offset_ = GetMinPosition(); |
| 88 } else if (contents_scroll_offset_ > GetMaxPosition()) { |
| 89 contents_scroll_offset_ = GetMaxPosition(); |
| 90 } |
| 91 ScrollContentsToOffset(); |
| 92 SchedulePaint(); |
| 93 } |
| 94 |
| 95 void BaseScrollBar::ScrollByContentsOffset(int contents_offset) { |
| 96 contents_scroll_offset_ -= contents_offset; |
| 97 if (contents_scroll_offset_ < GetMinPosition()) { |
| 98 contents_scroll_offset_ = GetMinPosition(); |
| 99 } else if (contents_scroll_offset_ > GetMaxPosition()) { |
| 100 contents_scroll_offset_ = GetMaxPosition(); |
| 101 } |
| 102 ScrollContentsToOffset(); |
| 103 } |
| 104 |
| 105 /////////////////////////////////////////////////////////////////////////////// |
| 106 // BaseScrollBar, View implementation: |
| 107 |
| 108 bool BaseScrollBar::OnMousePressed(const MouseEvent& event) { |
| 109 if (event.IsOnlyLeftMouseButton()) { |
| 110 SetThumbTrackState(CustomButton::BS_PUSHED); |
| 111 gfx::Rect thumb_bounds = thumb_->bounds(); |
| 112 if (IsHorizontal()) { |
| 113 if (event.x() < thumb_bounds.x()) { |
| 114 last_scroll_amount_ = SCROLL_PREV_PAGE; |
| 115 } else if (event.x() > thumb_bounds.right()) { |
| 116 last_scroll_amount_ = SCROLL_NEXT_PAGE; |
| 117 } |
| 118 } else { |
| 119 if (event.y() < thumb_bounds.y()) { |
| 120 last_scroll_amount_ = SCROLL_PREV_PAGE; |
| 121 } else if (event.y() > thumb_bounds.bottom()) { |
| 122 last_scroll_amount_ = SCROLL_NEXT_PAGE; |
| 123 } |
| 124 } |
| 125 TrackClicked(); |
| 126 repeater_.Start(); |
| 127 } |
| 128 return true; |
| 129 } |
| 130 |
| 131 void BaseScrollBar::OnMouseReleased(const MouseEvent& event) { |
| 132 OnMouseCaptureLost(); |
| 133 } |
| 134 |
| 135 void BaseScrollBar::OnMouseCaptureLost() { |
| 136 SetThumbTrackState(CustomButton::BS_NORMAL); |
| 137 repeater_.Stop(); |
| 138 } |
| 139 |
| 140 bool BaseScrollBar::OnKeyPressed(const KeyEvent& event) { |
| 141 ScrollAmount amount = SCROLL_NONE; |
| 142 switch (event.key_code()) { |
| 143 case ui::VKEY_UP: |
| 144 if (!IsHorizontal()) |
| 145 amount = SCROLL_PREV_LINE; |
| 146 break; |
| 147 case ui::VKEY_DOWN: |
| 148 if (!IsHorizontal()) |
| 149 amount = SCROLL_NEXT_LINE; |
| 150 break; |
| 151 case ui::VKEY_LEFT: |
| 152 if (IsHorizontal()) |
| 153 amount = SCROLL_PREV_LINE; |
| 154 break; |
| 155 case ui::VKEY_RIGHT: |
| 156 if (IsHorizontal()) |
| 157 amount = SCROLL_NEXT_LINE; |
| 158 break; |
| 159 case ui::VKEY_PRIOR: |
| 160 amount = SCROLL_PREV_PAGE; |
| 161 break; |
| 162 case ui::VKEY_NEXT: |
| 163 amount = SCROLL_NEXT_PAGE; |
| 164 break; |
| 165 case ui::VKEY_HOME: |
| 166 amount = SCROLL_START; |
| 167 break; |
| 168 case ui::VKEY_END: |
| 169 amount = SCROLL_END; |
| 170 break; |
| 171 default: |
| 172 break; |
| 173 } |
| 174 if (amount != SCROLL_NONE) { |
| 175 ScrollByAmount(amount); |
| 176 return true; |
| 177 } |
| 178 return false; |
| 179 } |
| 180 |
| 181 bool BaseScrollBar::OnMouseWheel(const MouseWheelEvent& event) { |
| 182 ScrollByContentsOffset(event.offset()); |
| 183 return true; |
| 184 } |
| 185 |
| 186 /////////////////////////////////////////////////////////////////////////////// |
| 187 // BaseScrollBar, ContextMenuController implementation: |
| 188 |
| 189 enum ScrollBarContextMenuCommands { |
| 190 ScrollBarContextMenuCommand_ScrollHere = 1, |
| 191 ScrollBarContextMenuCommand_ScrollStart, |
| 192 ScrollBarContextMenuCommand_ScrollEnd, |
| 193 ScrollBarContextMenuCommand_ScrollPageUp, |
| 194 ScrollBarContextMenuCommand_ScrollPageDown, |
| 195 ScrollBarContextMenuCommand_ScrollPrev, |
| 196 ScrollBarContextMenuCommand_ScrollNext |
| 197 }; |
| 198 |
| 199 void BaseScrollBar::ShowContextMenuForView(View* source, |
| 200 const gfx::Point& p, |
| 201 bool is_mouse_gesture) { |
| 202 Widget* widget = GetWidget(); |
| 203 gfx::Rect widget_bounds = widget->GetWindowScreenBounds(); |
| 204 gfx::Point temp_pt(p.x() - widget_bounds.x(), p.y() - widget_bounds.y()); |
| 205 View::ConvertPointFromWidget(this, &temp_pt); |
| 206 context_menu_mouse_position_ = IsHorizontal() ? temp_pt.x() : temp_pt.y(); |
| 207 |
| 208 scoped_ptr<Menu> menu( |
| 209 Menu::Create(this, Menu::TOPLEFT, GetWidget()->GetNativeView())); |
| 210 menu->AppendDelegateMenuItem(ScrollBarContextMenuCommand_ScrollHere); |
| 211 menu->AppendSeparator(); |
| 212 menu->AppendDelegateMenuItem(ScrollBarContextMenuCommand_ScrollStart); |
| 213 menu->AppendDelegateMenuItem(ScrollBarContextMenuCommand_ScrollEnd); |
| 214 menu->AppendSeparator(); |
| 215 menu->AppendDelegateMenuItem(ScrollBarContextMenuCommand_ScrollPageUp); |
| 216 menu->AppendDelegateMenuItem(ScrollBarContextMenuCommand_ScrollPageDown); |
| 217 menu->AppendSeparator(); |
| 218 menu->AppendDelegateMenuItem(ScrollBarContextMenuCommand_ScrollPrev); |
| 219 menu->AppendDelegateMenuItem(ScrollBarContextMenuCommand_ScrollNext); |
| 220 menu->RunMenuAt(p.x(), p.y()); |
| 221 } |
| 222 |
| 223 /////////////////////////////////////////////////////////////////////////////// |
| 224 // BaseScrollBar, Menu::Delegate implementation: |
| 225 |
| 226 std::wstring BaseScrollBar::GetLabel(int id) const { |
| 227 int ids_value = 0; |
| 228 switch (id) { |
| 229 case ScrollBarContextMenuCommand_ScrollHere: |
| 230 ids_value = IDS_APP_SCROLLBAR_CXMENU_SCROLLHERE; |
| 231 break; |
| 232 case ScrollBarContextMenuCommand_ScrollStart: |
| 233 ids_value = IsHorizontal() ? IDS_APP_SCROLLBAR_CXMENU_SCROLLLEFTEDGE |
| 234 : IDS_APP_SCROLLBAR_CXMENU_SCROLLHOME; |
| 235 break; |
| 236 case ScrollBarContextMenuCommand_ScrollEnd: |
| 237 ids_value = IsHorizontal() ? IDS_APP_SCROLLBAR_CXMENU_SCROLLRIGHTEDGE |
| 238 : IDS_APP_SCROLLBAR_CXMENU_SCROLLEND; |
| 239 break; |
| 240 case ScrollBarContextMenuCommand_ScrollPageUp: |
| 241 ids_value = IDS_APP_SCROLLBAR_CXMENU_SCROLLPAGEUP; |
| 242 break; |
| 243 case ScrollBarContextMenuCommand_ScrollPageDown: |
| 244 ids_value = IDS_APP_SCROLLBAR_CXMENU_SCROLLPAGEDOWN; |
| 245 break; |
| 246 case ScrollBarContextMenuCommand_ScrollPrev: |
| 247 ids_value = IsHorizontal() ? IDS_APP_SCROLLBAR_CXMENU_SCROLLLEFT |
| 248 : IDS_APP_SCROLLBAR_CXMENU_SCROLLUP; |
| 249 break; |
| 250 case ScrollBarContextMenuCommand_ScrollNext: |
| 251 ids_value = IsHorizontal() ? IDS_APP_SCROLLBAR_CXMENU_SCROLLRIGHT |
| 252 : IDS_APP_SCROLLBAR_CXMENU_SCROLLDOWN; |
| 253 break; |
| 254 default: |
| 255 NOTREACHED() << "Invalid BaseScrollBar Context Menu command!"; |
| 256 } |
| 257 |
| 258 return ids_value ? UTF16ToWide(l10n_util::GetStringUTF16(ids_value)) : L""; |
| 259 } |
| 260 |
| 261 bool BaseScrollBar::IsCommandEnabled(int id) const { |
| 262 switch (id) { |
| 263 case ScrollBarContextMenuCommand_ScrollPageUp: |
| 264 case ScrollBarContextMenuCommand_ScrollPageDown: |
| 265 return !IsHorizontal(); |
| 266 } |
| 267 return true; |
| 268 } |
| 269 |
| 270 void BaseScrollBar::ExecuteCommand(int id) { |
| 271 switch (id) { |
| 272 case ScrollBarContextMenuCommand_ScrollHere: |
| 273 ScrollToThumbPosition(context_menu_mouse_position_, true); |
| 274 break; |
| 275 case ScrollBarContextMenuCommand_ScrollStart: |
| 276 ScrollByAmount(SCROLL_START); |
| 277 break; |
| 278 case ScrollBarContextMenuCommand_ScrollEnd: |
| 279 ScrollByAmount(SCROLL_END); |
| 280 break; |
| 281 case ScrollBarContextMenuCommand_ScrollPageUp: |
| 282 ScrollByAmount(SCROLL_PREV_PAGE); |
| 283 break; |
| 284 case ScrollBarContextMenuCommand_ScrollPageDown: |
| 285 ScrollByAmount(SCROLL_NEXT_PAGE); |
| 286 break; |
| 287 case ScrollBarContextMenuCommand_ScrollPrev: |
| 288 ScrollByAmount(SCROLL_PREV_LINE); |
| 289 break; |
| 290 case ScrollBarContextMenuCommand_ScrollNext: |
| 291 ScrollByAmount(SCROLL_NEXT_LINE); |
| 292 break; |
| 293 } |
| 294 } |
| 295 |
| 296 /////////////////////////////////////////////////////////////////////////////// |
| 297 // BaseScrollBar, ScrollBar implementation: |
| 298 |
| 299 void BaseScrollBar::Update(int viewport_size, int content_size, |
| 300 int contents_scroll_offset) { |
| 301 ScrollBar::Update(viewport_size, content_size, contents_scroll_offset); |
| 302 |
| 303 // Make sure contents_size is always > 0 to avoid divide by zero errors in |
| 304 // calculations throughout this code. |
| 305 contents_size_ = std::max(1, content_size); |
| 306 |
| 307 if (content_size < 0) |
| 308 content_size = 0; |
| 309 if (contents_scroll_offset < 0) |
| 310 contents_scroll_offset = 0; |
| 311 if (contents_scroll_offset > content_size) |
| 312 contents_scroll_offset = content_size; |
| 313 |
| 314 // Thumb Height and Thumb Pos. |
| 315 // The height of the thumb is the ratio of the Viewport height to the |
| 316 // content size multiplied by the height of the thumb track. |
| 317 double ratio = static_cast<double>(viewport_size) / contents_size_; |
| 318 int thumb_size = static_cast<int>(ratio * GetTrackSize()); |
| 319 thumb_->SetSize(thumb_size); |
| 320 |
| 321 int thumb_position = CalculateThumbPosition(contents_scroll_offset); |
| 322 thumb_->SetPosition(thumb_position); |
| 323 } |
| 324 |
| 325 int BaseScrollBar::GetPosition() const { |
| 326 return thumb_->GetPosition(); |
| 327 } |
| 328 |
| 329 /////////////////////////////////////////////////////////////////////////////// |
| 330 // BaseScrollBar, protected: |
| 331 |
| 332 BaseScrollBarThumb* BaseScrollBar::GetThumb() const { |
| 333 return thumb_; |
| 334 } |
| 335 |
| 336 CustomButton::ButtonState BaseScrollBar::GetThumbTrackState() const { |
| 337 return thumb_track_state_; |
| 338 } |
| 339 |
| 340 void BaseScrollBar::ScrollToPosition(int position) { |
| 341 GetController()->ScrollToPosition(this, position); |
| 342 } |
| 343 |
| 344 int BaseScrollBar::GetScrollIncrement(bool is_page, bool is_positive) { |
| 345 return GetController()->GetScrollIncrement(this, is_page, is_positive); |
| 346 } |
| 347 |
| 348 |
| 349 /////////////////////////////////////////////////////////////////////////////// |
| 350 // BaseScrollBar, private: |
| 351 |
| 352 void BaseScrollBar::TrackClicked() { |
| 353 if (last_scroll_amount_ != SCROLL_NONE) |
| 354 ScrollByAmount(last_scroll_amount_); |
| 355 } |
| 356 |
| 357 void BaseScrollBar::ScrollContentsToOffset() { |
| 358 ScrollToPosition(contents_scroll_offset_); |
| 359 thumb_->SetPosition(CalculateThumbPosition(contents_scroll_offset_)); |
| 360 } |
| 361 |
| 362 int BaseScrollBar::GetTrackSize() const { |
| 363 gfx::Rect track_bounds = GetTrackBounds(); |
| 364 return IsHorizontal() ? track_bounds.width() : track_bounds.height(); |
| 365 } |
| 366 |
| 367 int BaseScrollBar::CalculateThumbPosition(int contents_scroll_offset) const { |
| 368 return (contents_scroll_offset * GetTrackSize()) / contents_size_; |
| 369 } |
| 370 |
| 371 int BaseScrollBar::CalculateContentsOffset(int thumb_position, |
| 372 bool scroll_to_middle) const { |
| 373 if (scroll_to_middle) |
| 374 thumb_position = thumb_position - (thumb_->GetSize() / 2); |
| 375 return (thumb_position * contents_size_) / GetTrackSize(); |
| 376 } |
| 377 |
| 378 void BaseScrollBar::SetThumbTrackState(CustomButton::ButtonState state) { |
| 379 thumb_track_state_ = state; |
| 380 SchedulePaint(); |
| 381 } |
| 382 |
| 383 } // namespace views |
OLD | NEW |