// Copyright (c) 2010 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 #include "app/l10n_util_win.h" #include "app/resource_bundle.h" #include "base/logging.h" #include "base/stl_util-inl.h" #include "gfx/canvas_skia.h" #include "gfx/font.h" #include "gfx/native_theme_win.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->AsCanvasSkia()->beginPlatformPaint(); RECT r = {0, 0, view->width(), view->height()}; gfx::NativeTheme::instance()->PaintTabPanelBackground(dc, &r); canvas->AsCanvasSkia()->endPlatformPaint(); } private: DISALLOW_COPY_AND_ASSIGN(TabBackground); }; //////////////////////////////////////////////////////////////////////////////// // NativeTabbedPaneWin, public: NativeTabbedPaneWin::NativeTabbedPaneWin(TabbedPane* tabbed_pane) : NativeControlWin(), tabbed_pane_(tabbed_pane), content_window_(NULL), selected_index_(-1) { // 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(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(tab_views_.size())); contents->set_parent_owned(false); tab_views_.insert(tab_views_.begin() + index, contents); tab_titles_.insert(tab_titles_.begin() + index, title); 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. selected_index_ = 0; if (content_window_) content_window_->GetRootView()->AddChildView(contents); } // Add native tab only if the native control is alreay created. if (content_window_) { AddNativeTab(index, title, contents); // The newly added tab may have made the contents window smaller. ResizeContents(); } } void NativeTabbedPaneWin::AddNativeTab(int index, const std::wstring &title, views::View* 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 (base::i18n::IsRTL()) { tcitem.mask |= TCIF_RTLREADING; } tcitem.pszText = const_cast(title.c_str()); int result = TabCtrl_InsertItem(native_view(), index, &tcitem); DCHECK(result != -1); } View* NativeTabbedPaneWin::RemoveTabAtIndex(int index) { int tab_count = static_cast(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 if (content_window_) { // 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. if (content_window_) ResizeContents(); std::vector::iterator iter = tab_views_.begin() + index; View* removed_tab = *iter; tab_views_.erase(iter); tab_titles_.erase(tab_titles_.begin() + index); return removed_tab; } void NativeTabbedPaneWin::SelectTabAt(int index) { DCHECK((index >= 0) && (index < static_cast(tab_views_.size()))); if (native_view()) TabCtrl_SetCurSel(native_view(), index); DoSelectTabAt(index, true); } int NativeTabbedPaneWin::GetTabCount() { return TabCtrl_GetItemCount(native_view()); } int NativeTabbedPaneWin::GetSelectedTabIndex() { return TabCtrl_GetCurSel(native_view()); } View* NativeTabbedPaneWin::GetSelectedTab() { if (selected_index_ < 0) return NULL; return tab_views_[selected_index_]; } 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 WS_EX_LAYOUTRTL 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). DWORD style = WS_CHILD | WS_CLIPSIBLINGS | WS_VISIBLE | WS_CLIPCHILDREN; HWND tab_control = ::CreateWindowEx(0, WC_TABCONTROL, L"", style, 0, 0, width(), height(), GetWidget()->GetNativeView(), NULL, NULL, NULL); HFONT font = ResourceBundle::GetSharedInstance(). GetFont(ResourceBundle::BaseFont).hfont(); SendMessage(tab_control, WM_SETFONT, reinterpret_cast(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 why we waited until |content_window_| is created before we set this // property for the tabbed pane's HWND). if (base::i18n::IsRTL()) 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); NativeControlCreated(tab_control); // Add tabs that are already added if any. if (tab_views_.size() > 0) { InitializeTabs(); if (selected_index_ >= 0) DoSelectTabAt(selected_index_, false); } ResizeContents(); } bool NativeTabbedPaneWin::ProcessMessage(UINT message, WPARAM w_param, LPARAM l_param, LRESULT* result) { if (message == WM_NOTIFY && reinterpret_cast(l_param)->code == TCN_SELCHANGE) { int selected_tab = TabCtrl_GetCurSel(native_view()); DCHECK(selected_tab != -1); DoSelectTabAt(selected_tab, true); 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::InitializeTabs() { for (size_t i = 0; i < tab_views_.size(); ++i) { AddNativeTab(i, tab_titles_[i], tab_views_[i]); } } void NativeTabbedPaneWin::DoSelectTabAt(int index, boolean invoke_listener) { selected_index_ = index; if (content_window_) { 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 (invoke_listener && 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