| // Copyright (c) 2012 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 "chrome/browser/ui/views/extensions/shell_window_views.h" |
| |
| #include "base/utf_string_conversions.h" |
| #include "chrome/browser/extensions/extension_host.h" |
| #include "chrome/browser/favicon/favicon_tab_helper.h" |
| #include "chrome/browser/ui/tab_contents/tab_contents.h" |
| #include "chrome/browser/ui/views/extensions/extension_keybinding_registry_views.h" |
| #include "chrome/common/extensions/draggable_region.h" |
| #include "chrome/common/extensions/extension.h" |
| #include "content/public/browser/render_view_host.h" |
| #include "content/public/browser/render_widget_host_view.h" |
| #include "content/public/browser/web_contents.h" |
| #include "content/public/browser/web_contents_view.h" |
| #include "grit/ui_resources.h" |
| #include "grit/ui_strings.h" // Accessibility names |
| #include "third_party/skia/include/core/SkPaint.h" |
| #include "ui/base/hit_test.h" |
| #include "ui/base/l10n/l10n_util.h" |
| #include "ui/base/resource/resource_bundle.h" |
| #include "ui/gfx/canvas.h" |
| #include "ui/gfx/image/image.h" |
| #include "ui/gfx/path.h" |
| #include "ui/views/controls/button/button.h" |
| #include "ui/views/controls/button/image_button.h" |
| #include "ui/views/controls/webview/webview.h" |
| #include "ui/views/layout/grid_layout.h" |
| #include "ui/views/views_delegate.h" |
| #include "ui/views/widget/widget.h" |
| #include "ui/views/window/non_client_view.h" |
| |
| #if defined(OS_WIN) && !defined(USE_AURA) |
| #include "chrome/browser/shell_integration.h" |
| #include "chrome/browser/web_applications/web_app.h" |
| #include "ui/base/win/shell.h" |
| #endif |
| |
| #if defined(USE_ASH) |
| #include "ash/ash_constants.h" |
| #include "ash/wm/custom_frame_view_ash.h" |
| #include "ui/aura/env.h" |
| #include "ui/aura/window.h" |
| #endif |
| |
| namespace { |
| #if !defined(USE_ASH) |
| const int kResizeInsideBoundsSize = 5; |
| const int kResizeAreaCornerSize = 16; |
| #endif |
| |
| // Height of the chrome-style caption, in pixels. |
| const int kCaptionHeight = 25; |
| } // namespace |
| |
| class ShellWindowFrameView : public views::NonClientFrameView, |
| public views::ButtonListener { |
| public: |
| static const char kViewClassName[]; |
| |
| explicit ShellWindowFrameView(ShellWindowViews* window); |
| virtual ~ShellWindowFrameView(); |
| |
| void Init(views::Widget* frame); |
| |
| // views::NonClientFrameView implementation. |
| virtual gfx::Rect GetBoundsForClientView() const OVERRIDE; |
| virtual gfx::Rect GetWindowBoundsForClientBounds( |
| const gfx::Rect& client_bounds) const OVERRIDE; |
| virtual int NonClientHitTest(const gfx::Point& point) OVERRIDE; |
| virtual void GetWindowMask(const gfx::Size& size, |
| gfx::Path* window_mask) OVERRIDE; |
| virtual void ResetWindowControls() OVERRIDE {} |
| virtual void UpdateWindowIcon() OVERRIDE {} |
| |
| // views::View implementation. |
| virtual gfx::Size GetPreferredSize() OVERRIDE; |
| virtual void Layout() OVERRIDE; |
| virtual std::string GetClassName() const OVERRIDE; |
| virtual void OnPaint(gfx::Canvas* canvas) OVERRIDE; |
| virtual gfx::Size GetMinimumSize() OVERRIDE; |
| virtual gfx::Size GetMaximumSize() OVERRIDE; |
| |
| private: |
| // views::ButtonListener implementation. |
| virtual void ButtonPressed(views::Button* sender, const ui::Event& event) |
| OVERRIDE; |
| |
| ShellWindowViews* window_; |
| views::Widget* frame_; |
| views::ImageButton* close_button_; |
| |
| DISALLOW_COPY_AND_ASSIGN(ShellWindowFrameView); |
| }; |
| |
| const char ShellWindowFrameView::kViewClassName[] = |
| "browser/ui/views/extensions/ShellWindowFrameView"; |
| |
| ShellWindowFrameView::ShellWindowFrameView(ShellWindowViews* window) |
| : window_(window), |
| frame_(NULL), |
| close_button_(NULL) { |
| } |
| |
| ShellWindowFrameView::~ShellWindowFrameView() { |
| } |
| |
| void ShellWindowFrameView::Init(views::Widget* frame) { |
| frame_ = frame; |
| |
| if (!window_->frameless()) { |
| ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance(); |
| close_button_ = new views::ImageButton(this); |
| close_button_->SetImage(views::CustomButton::BS_NORMAL, |
| rb.GetNativeImageNamed(IDR_CLOSE_BAR).ToImageSkia()); |
| close_button_->SetImage(views::CustomButton::BS_HOT, |
| rb.GetNativeImageNamed(IDR_CLOSE_BAR_H).ToImageSkia()); |
| close_button_->SetImage(views::CustomButton::BS_PUSHED, |
| rb.GetNativeImageNamed(IDR_CLOSE_BAR_P).ToImageSkia()); |
| close_button_->SetAccessibleName( |
| l10n_util::GetStringUTF16(IDS_APP_ACCNAME_CLOSE)); |
| AddChildView(close_button_); |
| } |
| |
| #if defined(USE_ASH) |
| aura::Window* window = frame->GetNativeWindow(); |
| // Ensure we get resize cursors for a few pixels outside our bounds. |
| window->SetHitTestBoundsOverrideOuter( |
| gfx::Insets(-ash::kResizeOutsideBoundsSize, |
| -ash::kResizeOutsideBoundsSize, |
| -ash::kResizeOutsideBoundsSize, |
| -ash::kResizeOutsideBoundsSize), |
| ash::kResizeOutsideBoundsScaleForTouch); |
| // Ensure we get resize cursors just inside our bounds as well. |
| // TODO(jeremya): do we need to update these when in fullscreen/maximized? |
| window->set_hit_test_bounds_override_inner( |
| gfx::Insets(ash::kResizeInsideBoundsSize, ash::kResizeInsideBoundsSize, |
| ash::kResizeInsideBoundsSize, ash::kResizeInsideBoundsSize)); |
| #endif |
| } |
| |
| gfx::Rect ShellWindowFrameView::GetBoundsForClientView() const { |
| if (window_->frameless() || frame_->IsFullscreen()) |
| return bounds(); |
| return gfx::Rect(0, kCaptionHeight, width(), |
| std::max(0, height() - kCaptionHeight)); |
| } |
| |
| gfx::Rect ShellWindowFrameView::GetWindowBoundsForClientBounds( |
| const gfx::Rect& client_bounds) const { |
| if (window_->frameless()) { |
| gfx::Rect window_bounds = client_bounds; |
| // Enforce minimum size (1, 1) in case that client_bounds is passed with |
| // empty size. This could occur when the frameless window is being |
| // initialized. |
| if (window_bounds.IsEmpty()) { |
| window_bounds.set_width(1); |
| window_bounds.set_height(1); |
| } |
| return window_bounds; |
| } |
| |
| int closeButtonOffsetX = |
| (kCaptionHeight - close_button_->height()) / 2; |
| int header_width = close_button_->width() + closeButtonOffsetX * 2; |
| return gfx::Rect(client_bounds.x(), |
| std::max(0, client_bounds.y() - kCaptionHeight), |
| std::max(header_width, client_bounds.width()), |
| client_bounds.height() + kCaptionHeight); |
| } |
| |
| int ShellWindowFrameView::NonClientHitTest(const gfx::Point& point) { |
| if (frame_->IsFullscreen()) |
| return HTCLIENT; |
| |
| #if defined(USE_ASH) |
| gfx::Rect expanded_bounds = bounds(); |
| int outside_bounds = ash::kResizeOutsideBoundsSize; |
| if (aura::Env::GetInstance()->is_touch_down()) |
| outside_bounds *= ash::kResizeOutsideBoundsScaleForTouch; |
| expanded_bounds.Inset(-outside_bounds, -outside_bounds); |
| if (!expanded_bounds.Contains(point)) |
| return HTNOWHERE; |
| |
| int kResizeInsideBoundsSize = ash::kResizeInsideBoundsSize; |
| int kResizeAreaCornerSize = ash::kResizeAreaCornerSize; |
| #endif |
| |
| // Check the frame first, as we allow a small area overlapping the contents |
| // to be used for resize handles. |
| bool can_ever_resize = frame_->widget_delegate() ? |
| frame_->widget_delegate()->CanResize() : |
| false; |
| // Don't allow overlapping resize handles when the window is maximized or |
| // fullscreen, as it can't be resized in those states. |
| int resize_border = |
| frame_->IsMaximized() || frame_->IsFullscreen() ? 0 : |
| kResizeInsideBoundsSize; |
| int frame_component = GetHTComponentForFrame(point, |
| resize_border, |
| resize_border, |
| kResizeAreaCornerSize, |
| kResizeAreaCornerSize, |
| can_ever_resize); |
| if (frame_component != HTNOWHERE) |
| return frame_component; |
| |
| // Check for possible draggable region in the client area for the frameless |
| // window. |
| if (window_->frameless() && |
| window_->draggable_region() && |
| window_->draggable_region()->contains(point.x(), point.y())) |
| return HTCAPTION; |
| |
| int client_component = frame_->client_view()->NonClientHitTest(point); |
| if (client_component != HTNOWHERE) |
| return client_component; |
| |
| // Then see if the point is within any of the window controls. |
| if (close_button_ && close_button_->visible() && |
| close_button_->GetMirroredBounds().Contains(point)) |
| return HTCLOSE; |
| |
| // Caption is a safe default. |
| return HTCAPTION; |
| } |
| |
| void ShellWindowFrameView::GetWindowMask(const gfx::Size& size, |
| gfx::Path* window_mask) { |
| // We got nothing to say about no window mask. |
| } |
| |
| gfx::Size ShellWindowFrameView::GetPreferredSize() { |
| gfx::Size pref = frame_->client_view()->GetPreferredSize(); |
| gfx::Rect bounds(0, 0, pref.width(), pref.height()); |
| return frame_->non_client_view()->GetWindowBoundsForClientBounds( |
| bounds).size(); |
| } |
| |
| void ShellWindowFrameView::Layout() { |
| if (window_->frameless()) |
| return; |
| gfx::Size close_size = close_button_->GetPreferredSize(); |
| int closeButtonOffsetY = |
| (kCaptionHeight - close_size.height()) / 2; |
| int closeButtonOffsetX = closeButtonOffsetY; |
| close_button_->SetBounds( |
| width() - closeButtonOffsetX - close_size.width(), |
| closeButtonOffsetY, |
| close_size.width(), |
| close_size.height()); |
| } |
| |
| void ShellWindowFrameView::OnPaint(gfx::Canvas* canvas) { |
| if (window_->frameless()) |
| return; |
| // TODO(jeremya): different look for inactive? |
| SkPaint paint; |
| paint.setAntiAlias(false); |
| paint.setStyle(SkPaint::kFill_Style); |
| paint.setColor(SK_ColorWHITE); |
| gfx::Path path; |
| const int radius = 1; |
| path.moveTo(0, radius); |
| path.lineTo(radius, 0); |
| path.lineTo(width() - radius - 1, 0); |
| path.lineTo(width(), radius + 1); |
| path.lineTo(width(), kCaptionHeight); |
| path.lineTo(0, kCaptionHeight); |
| path.close(); |
| canvas->DrawPath(path, paint); |
| } |
| |
| std::string ShellWindowFrameView::GetClassName() const { |
| return kViewClassName; |
| } |
| |
| gfx::Size ShellWindowFrameView::GetMinimumSize() { |
| gfx::Size min_size = frame_->client_view()->GetMinimumSize(); |
| if (window_->frameless()) |
| return min_size; |
| |
| // Ensure we can display the top of the caption area. |
| gfx::Rect client_bounds = GetBoundsForClientView(); |
| min_size.Enlarge(0, client_bounds.y()); |
| // Ensure we have enough space for the window icon and buttons. We allow |
| // the title string to collapse to zero width. |
| int closeButtonOffsetX = |
| (kCaptionHeight - close_button_->height()) / 2; |
| int header_width = close_button_->width() + closeButtonOffsetX * 2; |
| if (header_width > min_size.width()) |
| min_size.set_width(header_width); |
| return min_size; |
| } |
| |
| gfx::Size ShellWindowFrameView::GetMaximumSize() { |
| gfx::Size max_size = frame_->client_view()->GetMaximumSize(); |
| if (window_->frameless()) |
| return max_size; |
| |
| if (!max_size.IsEmpty()) { |
| gfx::Rect client_bounds = GetBoundsForClientView(); |
| max_size.Enlarge(0, client_bounds.y()); |
| } |
| return max_size; |
| } |
| |
| void ShellWindowFrameView::ButtonPressed(views::Button* sender, |
| const ui::Event& event) { |
| DCHECK(!window_->frameless()); |
| if (sender == close_button_) |
| frame_->Close(); |
| } |
| |
| ShellWindowViews::ShellWindowViews(ShellWindow* shell_window, |
| const ShellWindow::CreateParams& win_params) |
| : shell_window_(shell_window), |
| web_view_(NULL), |
| is_fullscreen_(false), |
| frameless_(win_params.frame == ShellWindow::CreateParams::FRAME_NONE) { |
| window_ = new views::Widget; |
| views::Widget::InitParams params(views::Widget::InitParams::TYPE_WINDOW); |
| params.delegate = this; |
| params.remove_standard_frame = true; |
| params.use_system_default_icon = true; |
| minimum_size_ = win_params.minimum_size; |
| maximum_size_ = win_params.maximum_size; |
| window_->Init(params); |
| gfx::Rect window_bounds = |
| window_->non_client_view()->GetWindowBoundsForClientBounds( |
| win_params.bounds); |
| window_->SetBounds(window_bounds); |
| // Center window if no position was specified. |
| if (win_params.bounds.x() < 0 || win_params.bounds.y() < 0) |
| window_->CenterWindow(window_bounds.size()); |
| #if defined(OS_WIN) && !defined(USE_AURA) |
| std::string app_name = web_app::GenerateApplicationNameFromExtensionId( |
| extension()->id()); |
| ui::win::SetAppIdForWindow( |
| ShellIntegration::GetAppModelIdForProfile( |
| UTF8ToWide(app_name), shell_window_->profile()->GetPath()), |
| GetWidget()->GetTopLevelWidget()->GetNativeWindow()); |
| #endif |
| |
| extension_keybinding_registry_.reset( |
| new ExtensionKeybindingRegistryViews(shell_window_->profile(), |
| window_->GetFocusManager(), |
| extensions::ExtensionKeybindingRegistry::PLATFORM_APPS_ONLY, |
| shell_window_)); |
| |
| OnViewWasResized(); |
| |
| window_->Show(); |
| } |
| |
| views::View* ShellWindowViews::GetInitiallyFocusedView() { |
| return web_view_; |
| } |
| |
| bool ShellWindowViews::ShouldDescendIntoChildForEventHandling( |
| gfx::NativeView child, |
| const gfx::Point& location) { |
| #if defined(USE_AURA) |
| DCHECK_EQ(child, web_view_->web_contents()->GetView()->GetNativeView()); |
| // Shell window should claim mouse events that fall within the draggable |
| // region. |
| return !draggable_region_.get() || |
| !draggable_region_->contains(location.x(), location.y()); |
| #else |
| return true; |
| #endif |
| } |
| |
| void ShellWindowViews::OnFocus() { |
| web_view_->RequestFocus(); |
| } |
| |
| void ShellWindowViews::ViewHierarchyChanged( |
| bool is_add, views::View *parent, views::View *child) { |
| if (is_add && child == this) { |
| web_view_ = new views::WebView(NULL); |
| AddChildView(web_view_); |
| web_view_->SetWebContents(web_contents()); |
| } |
| } |
| |
| gfx::Size ShellWindowViews::GetMinimumSize() { |
| return minimum_size_; |
| } |
| |
| gfx::Size ShellWindowViews::GetMaximumSize() { |
| return maximum_size_; |
| } |
| |
| void ShellWindowViews::SetFullscreen(bool fullscreen) { |
| is_fullscreen_ = fullscreen; |
| window_->SetFullscreen(fullscreen); |
| // TODO(jeremya) we need to call RenderViewHost::ExitFullscreen() if we |
| // ever drop the window out of fullscreen in response to something that |
| // wasn't the app calling webkitCancelFullScreen(). |
| } |
| |
| bool ShellWindowViews::IsFullscreenOrPending() const { |
| return is_fullscreen_; |
| } |
| |
| ShellWindowViews::~ShellWindowViews() { |
| web_view_->SetWebContents(NULL); |
| } |
| |
| bool ShellWindowViews::IsActive() const { |
| return window_->IsActive(); |
| } |
| |
| bool ShellWindowViews::IsMaximized() const { |
| return window_->IsMaximized(); |
| } |
| |
| bool ShellWindowViews::IsMinimized() const { |
| return window_->IsMinimized(); |
| } |
| |
| bool ShellWindowViews::IsFullscreen() const { |
| return window_->IsFullscreen(); |
| } |
| |
| gfx::NativeWindow ShellWindowViews::GetNativeWindow() { |
| return window_->GetNativeWindow(); |
| } |
| |
| gfx::Rect ShellWindowViews::GetRestoredBounds() const { |
| return window_->GetRestoredBounds(); |
| } |
| |
| gfx::Rect ShellWindowViews::GetBounds() const { |
| return window_->GetWindowBoundsInScreen(); |
| } |
| |
| void ShellWindowViews::Show() { |
| if (window_->IsVisible()) { |
| window_->Activate(); |
| return; |
| } |
| |
| window_->Show(); |
| } |
| |
| void ShellWindowViews::ShowInactive() { |
| if (window_->IsVisible()) |
| return; |
| window_->ShowInactive(); |
| } |
| |
| void ShellWindowViews::Close() { |
| window_->Close(); |
| } |
| |
| void ShellWindowViews::Activate() { |
| window_->Activate(); |
| } |
| |
| void ShellWindowViews::Deactivate() { |
| window_->Deactivate(); |
| } |
| |
| void ShellWindowViews::Maximize() { |
| window_->Maximize(); |
| } |
| |
| void ShellWindowViews::Minimize() { |
| window_->Minimize(); |
| } |
| |
| void ShellWindowViews::Restore() { |
| window_->Restore(); |
| } |
| |
| void ShellWindowViews::SetBounds(const gfx::Rect& bounds) { |
| GetWidget()->SetBounds(bounds); |
| } |
| |
| void ShellWindowViews::FlashFrame(bool flash) { |
| window_->FlashFrame(flash); |
| } |
| |
| bool ShellWindowViews::IsAlwaysOnTop() const { |
| return false; |
| } |
| |
| void ShellWindowViews::DeleteDelegate() { |
| shell_window_->OnNativeClose(); |
| } |
| |
| bool ShellWindowViews::CanResize() const { |
| return maximum_size_.IsEmpty() || minimum_size_ != maximum_size_; |
| } |
| |
| bool ShellWindowViews::CanMaximize() const { |
| return true; |
| } |
| |
| views::View* ShellWindowViews::GetContentsView() { |
| return this; |
| } |
| |
| views::NonClientFrameView* ShellWindowViews::CreateNonClientFrameView( |
| views::Widget* widget) { |
| #if defined(USE_ASH) |
| if (!frameless_) { |
| ash::CustomFrameViewAsh* frame = new ash::CustomFrameViewAsh(); |
| frame->Init(widget); |
| return frame; |
| } |
| #endif |
| ShellWindowFrameView* frame_view = new ShellWindowFrameView(this); |
| frame_view->Init(window_); |
| return frame_view; |
| } |
| |
| string16 ShellWindowViews::GetWindowTitle() const { |
| return shell_window_->GetTitle(); |
| } |
| |
| views::Widget* ShellWindowViews::GetWidget() { |
| return window_; |
| } |
| |
| const views::Widget* ShellWindowViews::GetWidget() const { |
| return window_; |
| } |
| |
| void ShellWindowViews::OnViewWasResized() { |
| // TODO(jeremya): this doesn't seem like a terribly elegant way to keep the |
| // window shape in sync. |
| #if defined(OS_WIN) && !defined(USE_AURA) |
| // Set the window shape of the RWHV. |
| DCHECK(window_); |
| DCHECK(web_view_); |
| gfx::Size sz = web_view_->size(); |
| int height = sz.height(), width = sz.width(); |
| int radius = 1; |
| gfx::Path path; |
| if (window_->IsMaximized() || window_->IsFullscreen()) { |
| // Don't round the corners when the window is maximized or fullscreen. |
| path.addRect(0, 0, width, height); |
| } else { |
| if (frameless_) { |
| path.moveTo(0, radius); |
| path.lineTo(radius, 0); |
| path.lineTo(width - radius, 0); |
| path.lineTo(width, radius); |
| } else { |
| // Don't round the top corners in chrome-style frame mode. |
| path.moveTo(0, 0); |
| path.lineTo(width, 0); |
| } |
| path.lineTo(width, height - radius - 1); |
| path.lineTo(width - radius - 1, height); |
| path.lineTo(radius + 1, height); |
| path.lineTo(0, height - radius - 1); |
| path.close(); |
| } |
| SetWindowRgn(web_contents()->GetNativeView(), path.CreateNativeRegion(), 1); |
| |
| SkRegion* rgn = new SkRegion; |
| if (!window_->IsFullscreen()) { |
| if (draggable_region()) |
| rgn->op(*draggable_region(), SkRegion::kUnion_Op); |
| if (!window_->IsMaximized()) { |
| if (frameless_) |
| rgn->op(0, 0, width, kResizeInsideBoundsSize, SkRegion::kUnion_Op); |
| rgn->op(0, 0, kResizeInsideBoundsSize, height, SkRegion::kUnion_Op); |
| rgn->op(width - kResizeInsideBoundsSize, 0, width, height, |
| SkRegion::kUnion_Op); |
| rgn->op(0, height - kResizeInsideBoundsSize, width, height, |
| SkRegion::kUnion_Op); |
| } |
| } |
| if (web_contents()->GetRenderViewHost()->GetView()) |
| web_contents()->GetRenderViewHost()->GetView()->SetClickthroughRegion(rgn); |
| #endif |
| } |
| |
| gfx::ImageSkia ShellWindowViews::GetWindowAppIcon() { |
| gfx::Image app_icon = shell_window_->app_icon(); |
| if (app_icon.IsEmpty()) |
| return GetWindowIcon(); |
| else |
| return *app_icon.ToImageSkia(); |
| } |
| |
| gfx::ImageSkia ShellWindowViews::GetWindowIcon() { |
| TabContents* contents = shell_window_->tab_contents(); |
| if (contents) { |
| gfx::Image app_icon = contents->favicon_tab_helper()->GetFavicon(); |
| if (!app_icon.IsEmpty()) |
| return *app_icon.ToImageSkia(); |
| } |
| return gfx::ImageSkia(); |
| } |
| |
| bool ShellWindowViews::ShouldShowWindowTitle() const { |
| return false; |
| } |
| |
| void ShellWindowViews::Layout() { |
| DCHECK(web_view_); |
| web_view_->SetBounds(0, 0, width(), height()); |
| OnViewWasResized(); |
| } |
| |
| void ShellWindowViews::UpdateWindowIcon() { |
| window_->UpdateWindowIcon(); |
| } |
| |
| void ShellWindowViews::UpdateWindowTitle() { |
| window_->UpdateWindowTitle(); |
| } |
| |
| void ShellWindowViews::UpdateDraggableRegions( |
| const std::vector<extensions::DraggableRegion>& regions) { |
| // Draggable region is not supported for non-frameless window. |
| if (!frameless_) |
| return; |
| |
| SkRegion* draggable_region = new SkRegion; |
| |
| // By default, the whole window is draggable. |
| gfx::Rect bounds = GetBounds(); |
| draggable_region->op(0, 0, bounds.right(), bounds.bottom(), |
| SkRegion::kUnion_Op); |
| |
| // Exclude those desinated as non-draggable. |
| for (std::vector<extensions::DraggableRegion>::const_iterator iter = |
| regions.begin(); |
| iter != regions.end(); ++iter) { |
| const extensions::DraggableRegion& region = *iter; |
| draggable_region->op(region.bounds.x(), |
| region.bounds.y(), |
| region.bounds.right(), |
| region.bounds.bottom(), |
| SkRegion::kDifference_Op); |
| } |
| |
| draggable_region_.reset(draggable_region); |
| OnViewWasResized(); |
| } |
| |
| void ShellWindowViews::HandleKeyboardEvent( |
| const content::NativeWebKeyboardEvent& event) { |
| unhandled_keyboard_event_handler_.HandleKeyboardEvent(event, |
| GetFocusManager()); |
| } |
| |
| void ShellWindowViews::SaveWindowPlacement(const gfx::Rect& bounds, |
| ui::WindowShowState show_state) { |
| views::WidgetDelegate::SaveWindowPlacement(bounds, show_state); |
| shell_window_->SaveWindowPosition(); |
| } |
| |
| // static |
| NativeShellWindow* NativeShellWindow::Create( |
| ShellWindow* shell_window, const ShellWindow::CreateParams& params) { |
| return new ShellWindowViews(shell_window, params); |
| } |