diff options
Diffstat (limited to 'ui/views/controls/single_split_view.cc')
-rw-r--r-- | ui/views/controls/single_split_view.cc | 263 |
1 files changed, 263 insertions, 0 deletions
diff --git a/ui/views/controls/single_split_view.cc b/ui/views/controls/single_split_view.cc new file mode 100644 index 0000000..333c388 --- /dev/null +++ b/ui/views/controls/single_split_view.cc @@ -0,0 +1,263 @@ +// 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 "ui/views/controls/single_split_view.h" + +#if defined(TOOLKIT_USES_GTK) +#include <gdk/gdk.h> +#endif + +#include "skia/ext/skia_utils_win.h" +#include "ui/base/accessibility/accessible_view_state.h" +#include "ui/gfx/canvas.h" +#include "ui/views/controls/single_split_view_listener.h" +#include "views/background.h" + +#if defined(TOOLKIT_USES_GTK) +#include "ui/gfx/gtk_util.h" +#endif + +#if defined(USE_AURA) +#include "ui/aura/cursor.h" +#endif + +namespace views { + +// static +const char SingleSplitView::kViewClassName[] = + "ui/views/controls/SingleSplitView"; + +// Size of the divider in pixels. +static const int kDividerSize = 4; + +SingleSplitView::SingleSplitView(View* leading, + View* trailing, + Orientation orientation, + SingleSplitViewListener* listener) + : is_horizontal_(orientation == HORIZONTAL_SPLIT), + divider_offset_(-1), + resize_leading_on_bounds_change_(true), + listener_(listener) { + AddChildView(leading); + AddChildView(trailing); +#if defined(OS_WIN) + set_background( + views::Background::CreateSolidBackground( + skia::COLORREFToSkColor(GetSysColor(COLOR_3DFACE)))); +#endif +} + +void SingleSplitView::Layout() { + gfx::Rect leading_bounds; + gfx::Rect trailing_bounds; + CalculateChildrenBounds(bounds(), &leading_bounds, &trailing_bounds); + + if (has_children()) { + if (child_at(0)->IsVisible()) + child_at(0)->SetBoundsRect(leading_bounds); + if (child_count() > 1) { + if (child_at(1)->IsVisible()) + child_at(1)->SetBoundsRect(trailing_bounds); + } + } + + SchedulePaint(); + + // Invoke super's implementation so that the children are layed out. + View::Layout(); +} + +std::string SingleSplitView::GetClassName() const { + return kViewClassName; +} + +void SingleSplitView::GetAccessibleState(ui::AccessibleViewState* state) { + state->role = ui::AccessibilityTypes::ROLE_GROUPING; + state->name = accessible_name_; +} + +gfx::Size SingleSplitView::GetPreferredSize() { + int width = 0; + int height = 0; + for (int i = 0; i < 2 && i < child_count(); ++i) { + View* view = child_at(i); + gfx::Size pref = view->GetPreferredSize(); + if (is_horizontal_) { + width += pref.width(); + height = std::max(height, pref.height()); + } else { + width = std::max(width, pref.width()); + height += pref.height(); + } + } + if (is_horizontal_) + width += kDividerSize; + else + height += kDividerSize; + return gfx::Size(width, height); +} + +gfx::NativeCursor SingleSplitView::GetCursor(const MouseEvent& event) { + if (!IsPointInDivider(event.location())) + return gfx::kNullCursor; +#if defined(USE_AURA) + return is_horizontal_ ? + aura::kCursorEastWestResize : aura::kCursorNorthSouthResize; +#elif defined(OS_WIN) + static HCURSOR we_resize_cursor = LoadCursor(NULL, IDC_SIZEWE); + static HCURSOR ns_resize_cursor = LoadCursor(NULL, IDC_SIZENS); + return is_horizontal_ ? we_resize_cursor : ns_resize_cursor; +#elif defined(TOOLKIT_USES_GTK) + return gfx::GetCursor(is_horizontal_ ? GDK_SB_H_DOUBLE_ARROW : + GDK_SB_V_DOUBLE_ARROW); +#endif +} + +void SingleSplitView::CalculateChildrenBounds( + const gfx::Rect& bounds, + gfx::Rect* leading_bounds, + gfx::Rect* trailing_bounds) const { + bool is_leading_visible = has_children() && child_at(0)->IsVisible(); + bool is_trailing_visible = child_count() > 1 && child_at(1)->IsVisible(); + + if (!is_leading_visible && !is_trailing_visible) { + *leading_bounds = gfx::Rect(); + *trailing_bounds = gfx::Rect(); + return; + } + + int divider_at; + + if (!is_trailing_visible) { + divider_at = GetPrimaryAxisSize(bounds.width(), bounds.height()); + } else if (!is_leading_visible) { + divider_at = 0; + } else { + divider_at = + CalculateDividerOffset(divider_offset_, this->bounds(), bounds); + divider_at = NormalizeDividerOffset(divider_at, bounds); + } + + int divider_size = + !is_leading_visible || !is_trailing_visible ? 0 : kDividerSize; + + if (is_horizontal_) { + *leading_bounds = gfx::Rect(0, 0, divider_at, bounds.height()); + *trailing_bounds = + gfx::Rect(divider_at + divider_size, 0, + std::max(0, bounds.width() - divider_at - divider_size), + bounds.height()); + } else { + *leading_bounds = gfx::Rect(0, 0, bounds.width(), divider_at); + *trailing_bounds = + gfx::Rect(0, divider_at + divider_size, bounds.width(), + std::max(0, bounds.height() - divider_at - divider_size)); + } +} + +void SingleSplitView::SetAccessibleName(const string16& name) { + accessible_name_ = name; +} + +bool SingleSplitView::OnMousePressed(const MouseEvent& event) { + if (!IsPointInDivider(event.location())) + return false; + drag_info_.initial_mouse_offset = GetPrimaryAxisSize(event.x(), event.y()); + drag_info_.initial_divider_offset = + NormalizeDividerOffset(divider_offset_, bounds()); + return true; +} + +bool SingleSplitView::OnMouseDragged(const MouseEvent& event) { + if (child_count() < 2) + return false; + + int delta_offset = GetPrimaryAxisSize(event.x(), event.y()) - + drag_info_.initial_mouse_offset; + if (is_horizontal_ && base::i18n::IsRTL()) + delta_offset *= -1; + // Honor the minimum size when resizing. + gfx::Size min = child_at(0)->GetMinimumSize(); + int new_size = std::max(GetPrimaryAxisSize(min.width(), min.height()), + drag_info_.initial_divider_offset + delta_offset); + + // And don't let the view get bigger than our width. + new_size = std::min(GetPrimaryAxisSize() - kDividerSize, new_size); + + if (new_size != divider_offset_) { + set_divider_offset(new_size); + if (!listener_ || listener_->SplitHandleMoved(this)) + Layout(); + } + return true; +} + +void SingleSplitView::OnMouseCaptureLost() { + if (child_count() < 2) + return; + + if (drag_info_.initial_divider_offset != divider_offset_) { + set_divider_offset(drag_info_.initial_divider_offset); + if (!listener_ || listener_->SplitHandleMoved(this)) + Layout(); + } +} + +void SingleSplitView::OnBoundsChanged(const gfx::Rect& previous_bounds) { + divider_offset_ = CalculateDividerOffset(divider_offset_, previous_bounds, + bounds()); +} + +bool SingleSplitView::IsPointInDivider(const gfx::Point& p) { + if (child_count() < 2) + return false; + + if (!child_at(0)->IsVisible() || !child_at(1)->IsVisible()) + return false; + + int divider_relative_offset; + if (is_horizontal_) { + divider_relative_offset = + p.x() - child_at(base::i18n::IsRTL() ? 1 : 0)->width(); + } else { + divider_relative_offset = p.y() - child_at(0)->height(); + } + return (divider_relative_offset >= 0 && + divider_relative_offset < kDividerSize); +} + +int SingleSplitView::CalculateDividerOffset( + int divider_offset, + const gfx::Rect& previous_bounds, + const gfx::Rect& new_bounds) const { + if (resize_leading_on_bounds_change_ && divider_offset != -1) { + // We do not update divider_offset on minimize (to zero) and on restore + // (to largest value). As a result we get back to the original value upon + // window restore. + bool is_minimize_or_restore = + previous_bounds.height() == 0 || new_bounds.height() == 0; + if (!is_minimize_or_restore) { + if (is_horizontal_) + divider_offset += new_bounds.width() - previous_bounds.width(); + else + divider_offset += new_bounds.height() - previous_bounds.height(); + + if (divider_offset < 0) + divider_offset = kDividerSize; + } + } + return divider_offset; +} + +int SingleSplitView::NormalizeDividerOffset(int divider_offset, + const gfx::Rect& bounds) const { + int primary_axis_size = GetPrimaryAxisSize(bounds.width(), bounds.height()); + if (divider_offset < 0) + // primary_axis_size may < kDividerSize during initial layout. + return std::max(0, (primary_axis_size - kDividerSize) / 2); + return std::min(divider_offset, + std::max(primary_axis_size - kDividerSize, 0)); +} + +} // namespace views |