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 |