| Index: views/controls/tabbed_pane/native_tabbed_pane_win.cc
|
| ===================================================================
|
| --- views/controls/tabbed_pane/native_tabbed_pane_win.cc (revision 0)
|
| +++ views/controls/tabbed_pane/native_tabbed_pane_win.cc (revision 0)
|
| @@ -0,0 +1,300 @@
|
| +// Copyright (c) 2009 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/tabbed_pane/native_tabbed_pane_win.h"
|
| +
|
| +#include <vssym32.h>
|
| +
|
| +#include "app/gfx/canvas.h"
|
| +#include "app/gfx/font.h"
|
| +#include "app/l10n_util_win.h"
|
| +#include "app/resource_bundle.h"
|
| +#include "base/gfx/native_theme.h"
|
| +#include "base/logging.h"
|
| +#include "base/stl_util-inl.h"
|
| +#include "views/controls/tabbed_pane/tabbed_pane.h"
|
| +#include "views/fill_layout.h"
|
| +#include "views/widget/root_view.h"
|
| +#include "views/widget/widget_win.h"
|
| +
|
| +namespace views {
|
| +
|
| +// A background object that paints the tab panel background which may be
|
| +// rendered by the system visual styles system.
|
| +class TabBackground : public Background {
|
| + public:
|
| + explicit TabBackground() {
|
| + // TMT_FILLCOLORHINT returns a color value that supposedly
|
| + // approximates the texture drawn by PaintTabPanelBackground.
|
| + SkColor tab_page_color =
|
| + gfx::NativeTheme::instance()->GetThemeColorWithDefault(
|
| + gfx::NativeTheme::TAB, TABP_BODY, 0, TMT_FILLCOLORHINT,
|
| + COLOR_3DFACE);
|
| + SetNativeControlColor(tab_page_color);
|
| + }
|
| + virtual ~TabBackground() {}
|
| +
|
| + virtual void Paint(gfx::Canvas* canvas, View* view) const {
|
| + HDC dc = canvas->beginPlatformPaint();
|
| + RECT r = {0, 0, view->width(), view->height()};
|
| + gfx::NativeTheme::instance()->PaintTabPanelBackground(dc, &r);
|
| + canvas->endPlatformPaint();
|
| + }
|
| +
|
| + private:
|
| + DISALLOW_COPY_AND_ASSIGN(TabBackground);
|
| +};
|
| +
|
| +////////////////////////////////////////////////////////////////////////////////
|
| +// NativeTabbedPaneWin, public:
|
| +
|
| +NativeTabbedPaneWin::NativeTabbedPaneWin(TabbedPane* tabbed_pane)
|
| + : NativeControlWin(),
|
| + tabbed_pane_(tabbed_pane),
|
| + content_window_(NULL) {
|
| + // Associates the actual HWND with the tabbed-pane so the tabbed-pane is
|
| + // the one considered as having the focus (not the wrapper) when the HWND is
|
| + // focused directly (with a click for example).
|
| + set_focus_view(tabbed_pane);
|
| +}
|
| +
|
| +NativeTabbedPaneWin::~NativeTabbedPaneWin() {
|
| + // We own the tab views, let's delete them.
|
| + STLDeleteContainerPointers(tab_views_.begin(), tab_views_.end());
|
| +}
|
| +
|
| +////////////////////////////////////////////////////////////////////////////////
|
| +// NativeTabbedPaneWin, NativeTabbedPaneWrapper implementation:
|
| +
|
| +void NativeTabbedPaneWin::AddTab(const std::wstring& title, View* contents) {
|
| + AddTabAtIndex(static_cast<int>(tab_views_.size()), title, contents, true);
|
| +}
|
| +
|
| +void NativeTabbedPaneWin::AddTabAtIndex(int index, const std::wstring& title,
|
| + View* contents,
|
| + bool select_if_first_tab) {
|
| + DCHECK(index <= static_cast<int>(tab_views_.size()));
|
| + contents->SetParentOwned(false);
|
| + tab_views_.insert(tab_views_.begin() + index, contents);
|
| +
|
| + TCITEM tcitem;
|
| + tcitem.mask = TCIF_TEXT;
|
| +
|
| + // If the locale is RTL, we set the TCIF_RTLREADING so that BiDi text is
|
| + // rendered properly on the tabs.
|
| + if (UILayoutIsRightToLeft()) {
|
| + tcitem.mask |= TCIF_RTLREADING;
|
| + }
|
| +
|
| + tcitem.pszText = const_cast<wchar_t*>(title.c_str());
|
| + int result = TabCtrl_InsertItem(native_view(), index, &tcitem);
|
| + DCHECK(result != -1);
|
| +
|
| + if (!contents->background())
|
| + contents->set_background(new TabBackground);
|
| +
|
| + if (tab_views_.size() == 1 && select_if_first_tab) {
|
| + // If this is the only tab displayed, make sure the contents is set.
|
| + content_window_->GetRootView()->AddChildView(contents);
|
| + }
|
| +
|
| + // The newly added tab may have made the contents window smaller.
|
| + ResizeContents();
|
| +}
|
| +
|
| +View* NativeTabbedPaneWin::RemoveTabAtIndex(int index) {
|
| + int tab_count = static_cast<int>(tab_views_.size());
|
| + DCHECK(index >= 0 && index < tab_count);
|
| +
|
| + if (index < (tab_count - 1)) {
|
| + // Select the next tab.
|
| + SelectTabAt(index + 1);
|
| + } else {
|
| + // We are the last tab, select the previous one.
|
| + if (index > 0) {
|
| + SelectTabAt(index - 1);
|
| + } else {
|
| + // That was the last tab. Remove the contents.
|
| + content_window_->GetRootView()->RemoveAllChildViews(false);
|
| + }
|
| + }
|
| + TabCtrl_DeleteItem(native_view(), index);
|
| +
|
| + // The removed tab may have made the contents window bigger.
|
| + ResizeContents();
|
| +
|
| + std::vector<View*>::iterator iter = tab_views_.begin() + index;
|
| + View* removed_tab = *iter;
|
| + tab_views_.erase(iter);
|
| +
|
| + return removed_tab;
|
| +}
|
| +
|
| +void NativeTabbedPaneWin::SelectTabAt(int index) {
|
| + DCHECK((index >= 0) && (index < static_cast<int>(tab_views_.size())));
|
| + TabCtrl_SetCurSel(native_view(), index);
|
| + DoSelectTabAt(index);
|
| +}
|
| +
|
| +int NativeTabbedPaneWin::GetTabCount() {
|
| + return TabCtrl_GetItemCount(native_view());
|
| +}
|
| +
|
| +int NativeTabbedPaneWin::GetSelectedTabIndex() {
|
| + return TabCtrl_GetCurSel(native_view());
|
| +}
|
| +
|
| +View* NativeTabbedPaneWin::GetSelectedTab() {
|
| + return content_window_->GetRootView()->GetChildViewAt(0);
|
| +}
|
| +
|
| +View* NativeTabbedPaneWin::GetView() {
|
| + return this;
|
| +}
|
| +
|
| +void NativeTabbedPaneWin::SetFocus() {
|
| + // Focus the associated HWND.
|
| + Focus();
|
| +}
|
| +
|
| +gfx::NativeView NativeTabbedPaneWin::GetTestingHandle() const {
|
| + return native_view();
|
| +}
|
| +
|
| +////////////////////////////////////////////////////////////////////////////////
|
| +// NativeTabbedPaneWin, NativeControlWin override:
|
| +
|
| +void NativeTabbedPaneWin::CreateNativeControl() {
|
| + // Create the tab control.
|
| + //
|
| + // Note that we don't follow the common convention for NativeControl
|
| + // subclasses and we don't pass the value returned from
|
| + // NativeControl::GetAdditionalExStyle() as the dwExStyle parameter. Here is
|
| + // why: on RTL locales, if we pass NativeControl::GetAdditionalExStyle() when
|
| + // we basically tell Windows to create our HWND with the WS_EX_LAYOUTRTL. If
|
| + // we do that, then the HWND we create for |content_window_| below will
|
| + // inherit the WS_EX_LAYOUTRTL property and this will result in the contents
|
| + // being flipped, which is not what we want (because we handle mirroring in
|
| + // views without the use of Windows' support for mirroring). Therefore,
|
| + // we initially create our HWND without the aforementioned property and we
|
| + // explicitly set this property our child is created. This way, on RTL
|
| + // locales, our tabs will be nicely rendered from right to left (by virtue of
|
| + // Windows doing the right thing with the TabbedPane HWND) and each tab
|
| + // contents will use an RTL layout correctly (by virtue of the mirroring
|
| + // infrastructure in views doing the right thing with each View we put
|
| + // in the tab).
|
| + HWND tab_control = ::CreateWindowEx(0,
|
| + WC_TABCONTROL,
|
| + L"",
|
| + WS_CHILD | WS_CLIPSIBLINGS | WS_VISIBLE,
|
| + 0, 0, width(), height(),
|
| + GetWidget()->GetNativeView(), NULL, NULL,
|
| + NULL);
|
| +
|
| + HFONT font = ResourceBundle::GetSharedInstance().
|
| + GetFont(ResourceBundle::BaseFont).hfont();
|
| + SendMessage(tab_control, WM_SETFONT, reinterpret_cast<WPARAM>(font), FALSE);
|
| +
|
| + // Create the view container which is a child of the TabControl.
|
| + content_window_ = new WidgetWin();
|
| + content_window_->Init(tab_control, gfx::Rect());
|
| +
|
| + // Explicitly setting the WS_EX_LAYOUTRTL property for the HWND (see above
|
| + // for a thorough explanation regarding why we waited until |content_window_|
|
| + // if created before we set this property for the tabbed pane's HWND).
|
| + if (UILayoutIsRightToLeft())
|
| + l10n_util::HWNDSetRTLLayout(tab_control);
|
| +
|
| + RootView* root_view = content_window_->GetRootView();
|
| + root_view->SetLayoutManager(new FillLayout());
|
| + DWORD sys_color = ::GetSysColor(COLOR_3DHILIGHT);
|
| + SkColor color = SkColorSetRGB(GetRValue(sys_color), GetGValue(sys_color),
|
| + GetBValue(sys_color));
|
| + root_view->set_background(Background::CreateSolidBackground(color));
|
| +
|
| + content_window_->SetFocusTraversableParentView(this);
|
| + ResizeContents();
|
| +
|
| + NativeControlCreated(tab_control);
|
| +}
|
| +
|
| +bool NativeTabbedPaneWin::ProcessMessage(UINT message,
|
| + WPARAM w_param,
|
| + LPARAM l_param,
|
| + LRESULT* result) {
|
| + if (message == WM_NOTIFY &&
|
| + reinterpret_cast<LPNMHDR>(l_param)->code == TCN_SELCHANGE) {
|
| + int selected_tab = TabCtrl_GetCurSel(native_view());
|
| + DCHECK(selected_tab != -1);
|
| + DoSelectTabAt(selected_tab);
|
| + return TRUE;
|
| + }
|
| + return NativeControlWin::ProcessMessage(message, w_param, l_param, result);
|
| +}
|
| +
|
| +////////////////////////////////////////////////////////////////////////////////
|
| +// View override:
|
| +
|
| +void NativeTabbedPaneWin::Layout() {
|
| + NativeControlWin::Layout();
|
| + ResizeContents();
|
| +}
|
| +
|
| +FocusTraversable* NativeTabbedPaneWin::GetFocusTraversable() {
|
| + return content_window_;
|
| +}
|
| +
|
| +void NativeTabbedPaneWin::ViewHierarchyChanged(bool is_add,
|
| + View *parent,
|
| + View *child) {
|
| + NativeControlWin::ViewHierarchyChanged(is_add, parent, child);
|
| +
|
| + if (is_add && (child == this) && content_window_) {
|
| + // We have been added to a view hierarchy, update the FocusTraversable
|
| + // parent.
|
| + content_window_->SetFocusTraversableParent(GetRootView());
|
| + }
|
| +}
|
| +
|
| +////////////////////////////////////////////////////////////////////////////////
|
| +// NativeTabbedPaneWin, private:
|
| +
|
| +void NativeTabbedPaneWin::DoSelectTabAt(int index) {
|
| + RootView* content_root = content_window_->GetRootView();
|
| +
|
| + // Clear the focus if the focused view was on the tab.
|
| + FocusManager* focus_manager = GetFocusManager();
|
| + DCHECK(focus_manager);
|
| + View* focused_view = focus_manager->GetFocusedView();
|
| + if (focused_view && content_root->IsParentOf(focused_view))
|
| + focus_manager->ClearFocus();
|
| +
|
| + content_root->RemoveAllChildViews(false);
|
| + content_root->AddChildView(tab_views_[index]);
|
| + content_root->Layout();
|
| +
|
| + if (tabbed_pane_->listener())
|
| + tabbed_pane_->listener()->TabSelectedAt(index);
|
| +}
|
| +
|
| +void NativeTabbedPaneWin::ResizeContents() {
|
| + CRect content_bounds;
|
| + if (!GetClientRect(native_view(), &content_bounds))
|
| + return;
|
| + TabCtrl_AdjustRect(native_view(), FALSE, &content_bounds);
|
| + content_window_->MoveWindow(content_bounds.left, content_bounds.top,
|
| + content_bounds.Width(), content_bounds.Height(),
|
| + TRUE);
|
| +}
|
| +
|
| +////////////////////////////////////////////////////////////////////////////////
|
| +// NativeTabbedPaneWrapper, public:
|
| +
|
| +// static
|
| +NativeTabbedPaneWrapper* NativeTabbedPaneWrapper::CreateNativeWrapper(
|
| + TabbedPane* tabbed_pane) {
|
| + return new NativeTabbedPaneWin(tabbed_pane);
|
| +}
|
| +
|
| +} // namespace views
|
|
|
| Property changes on: views\controls\tabbed_pane\native_tabbed_pane_win.cc
|
| ___________________________________________________________________
|
| Added: svn:eol-style
|
| + LF
|
|
|
|
|