Elim cr-checkbox
[chromium-blink-merge.git] / extensions / components / native_app_window / native_app_window_views.cc
blob75188cea32a428a7b062b2adab48b61170eaf34d
1 // Copyright 2014 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "extensions/components/native_app_window/native_app_window_views.h"
7 #include "base/threading/sequenced_worker_pool.h"
8 #include "content/public/browser/render_view_host.h"
9 #include "content/public/browser/render_widget_host_view.h"
10 #include "content/public/browser/web_contents.h"
11 #include "extensions/browser/app_window/app_window.h"
12 #include "extensions/common/draggable_region.h"
13 #include "third_party/skia/include/core/SkRegion.h"
14 #include "ui/gfx/path.h"
15 #include "ui/views/controls/webview/webview.h"
16 #include "ui/views/widget/widget.h"
17 #include "ui/views/window/non_client_view.h"
19 #if defined(USE_AURA)
20 #include "ui/aura/window.h"
21 #endif
23 using extensions::AppWindow;
25 namespace native_app_window {
27 NativeAppWindowViews::NativeAppWindowViews()
28 : app_window_(NULL),
29 web_view_(NULL),
30 widget_(NULL),
31 frameless_(false),
32 resizable_(false) {
35 void NativeAppWindowViews::Init(AppWindow* app_window,
36 const AppWindow::CreateParams& create_params) {
37 app_window_ = app_window;
38 frameless_ = create_params.frame == AppWindow::FRAME_NONE;
39 resizable_ = create_params.resizable;
40 size_constraints_.set_minimum_size(
41 create_params.GetContentMinimumSize(gfx::Insets()));
42 size_constraints_.set_maximum_size(
43 create_params.GetContentMaximumSize(gfx::Insets()));
44 Observe(app_window_->web_contents());
46 widget_ = new views::Widget;
47 widget_->AddObserver(this);
48 InitializeWindow(app_window, create_params);
50 OnViewWasResized();
53 NativeAppWindowViews::~NativeAppWindowViews() {
54 web_view_->SetWebContents(NULL);
57 void NativeAppWindowViews::OnCanHaveAlphaEnabledChanged() {
58 app_window_->OnNativeWindowChanged();
61 void NativeAppWindowViews::InitializeWindow(
62 AppWindow* app_window,
63 const AppWindow::CreateParams& create_params) {
64 // Stub implementation. See also ChromeNativeAppWindowViews.
65 views::Widget::InitParams init_params(views::Widget::InitParams::TYPE_WINDOW);
66 init_params.delegate = this;
67 init_params.keep_on_top = create_params.always_on_top;
68 widget_->Init(init_params);
69 widget_->CenterWindow(
70 create_params.GetInitialWindowBounds(gfx::Insets()).size());
73 // ui::BaseWindow implementation.
75 bool NativeAppWindowViews::IsActive() const {
76 return widget_->IsActive();
79 bool NativeAppWindowViews::IsMaximized() const {
80 return widget_->IsMaximized();
83 bool NativeAppWindowViews::IsMinimized() const {
84 return widget_->IsMinimized();
87 bool NativeAppWindowViews::IsFullscreen() const {
88 return widget_->IsFullscreen();
91 gfx::NativeWindow NativeAppWindowViews::GetNativeWindow() const {
92 return widget_->GetNativeWindow();
95 gfx::Rect NativeAppWindowViews::GetRestoredBounds() const {
96 return widget_->GetRestoredBounds();
99 ui::WindowShowState NativeAppWindowViews::GetRestoredState() const {
100 // Stub implementation. See also ChromeNativeAppWindowViews.
101 if (IsMaximized())
102 return ui::SHOW_STATE_MAXIMIZED;
103 if (IsFullscreen())
104 return ui::SHOW_STATE_FULLSCREEN;
105 return ui::SHOW_STATE_NORMAL;
108 gfx::Rect NativeAppWindowViews::GetBounds() const {
109 return widget_->GetWindowBoundsInScreen();
112 void NativeAppWindowViews::Show() {
113 if (widget_->IsVisible()) {
114 widget_->Activate();
115 return;
117 widget_->Show();
120 void NativeAppWindowViews::ShowInactive() {
121 if (widget_->IsVisible())
122 return;
124 widget_->ShowInactive();
127 void NativeAppWindowViews::Hide() {
128 widget_->Hide();
131 void NativeAppWindowViews::Close() {
132 widget_->Close();
135 void NativeAppWindowViews::Activate() {
136 widget_->Activate();
139 void NativeAppWindowViews::Deactivate() {
140 widget_->Deactivate();
143 void NativeAppWindowViews::Maximize() {
144 widget_->Maximize();
147 void NativeAppWindowViews::Minimize() {
148 widget_->Minimize();
151 void NativeAppWindowViews::Restore() {
152 widget_->Restore();
155 void NativeAppWindowViews::SetBounds(const gfx::Rect& bounds) {
156 widget_->SetBounds(bounds);
159 void NativeAppWindowViews::FlashFrame(bool flash) {
160 widget_->FlashFrame(flash);
163 bool NativeAppWindowViews::IsAlwaysOnTop() const {
164 // Stub implementation. See also ChromeNativeAppWindowViews.
165 return widget_->IsAlwaysOnTop();
168 void NativeAppWindowViews::SetAlwaysOnTop(bool always_on_top) {
169 widget_->SetAlwaysOnTop(always_on_top);
172 gfx::NativeView NativeAppWindowViews::GetHostView() const {
173 return widget_->GetNativeView();
176 gfx::Point NativeAppWindowViews::GetDialogPosition(const gfx::Size& size) {
177 gfx::Size app_window_size = widget_->GetWindowBoundsInScreen().size();
178 return gfx::Point(app_window_size.width() / 2 - size.width() / 2,
179 app_window_size.height() / 2 - size.height() / 2);
182 gfx::Size NativeAppWindowViews::GetMaximumDialogSize() {
183 return widget_->GetWindowBoundsInScreen().size();
186 void NativeAppWindowViews::AddObserver(
187 web_modal::ModalDialogHostObserver* observer) {
188 observer_list_.AddObserver(observer);
190 void NativeAppWindowViews::RemoveObserver(
191 web_modal::ModalDialogHostObserver* observer) {
192 observer_list_.RemoveObserver(observer);
195 void NativeAppWindowViews::OnViewWasResized() {
196 FOR_EACH_OBSERVER(web_modal::ModalDialogHostObserver,
197 observer_list_,
198 OnPositionRequiresUpdate());
201 // WidgetDelegate implementation.
203 void NativeAppWindowViews::OnWidgetMove() {
204 app_window_->OnNativeWindowChanged();
207 views::View* NativeAppWindowViews::GetInitiallyFocusedView() {
208 return web_view_;
211 bool NativeAppWindowViews::CanResize() const {
212 return resizable_ && !size_constraints_.HasFixedSize() &&
213 !WidgetHasHitTestMask();
216 bool NativeAppWindowViews::CanMaximize() const {
217 return resizable_ && !size_constraints_.HasMaximumSize() &&
218 !app_window_->window_type_is_panel() && !WidgetHasHitTestMask();
221 bool NativeAppWindowViews::CanMinimize() const {
222 return true;
225 base::string16 NativeAppWindowViews::GetWindowTitle() const {
226 return app_window_->GetTitle();
229 bool NativeAppWindowViews::ShouldShowWindowTitle() const {
230 return app_window_->window_type() == AppWindow::WINDOW_TYPE_V1_PANEL;
233 bool NativeAppWindowViews::ShouldShowWindowIcon() const {
234 return app_window_->window_type() == AppWindow::WINDOW_TYPE_V1_PANEL;
237 void NativeAppWindowViews::SaveWindowPlacement(const gfx::Rect& bounds,
238 ui::WindowShowState show_state) {
239 views::WidgetDelegate::SaveWindowPlacement(bounds, show_state);
240 app_window_->OnNativeWindowChanged();
243 void NativeAppWindowViews::DeleteDelegate() {
244 widget_->RemoveObserver(this);
245 app_window_->OnNativeClose();
248 views::Widget* NativeAppWindowViews::GetWidget() {
249 return widget_;
252 const views::Widget* NativeAppWindowViews::GetWidget() const {
253 return widget_;
256 views::View* NativeAppWindowViews::GetContentsView() {
257 return this;
260 bool NativeAppWindowViews::ShouldDescendIntoChildForEventHandling(
261 gfx::NativeView child,
262 const gfx::Point& location) {
263 #if defined(USE_AURA)
264 if (child->Contains(web_view_->web_contents()->GetNativeView())) {
265 // App window should claim mouse events that fall within the draggable
266 // region.
267 return !draggable_region_.get() ||
268 !draggable_region_->contains(location.x(), location.y());
270 #endif
272 return true;
275 // WidgetObserver implementation.
277 void NativeAppWindowViews::OnWidgetVisibilityChanged(views::Widget* widget,
278 bool visible) {
279 app_window_->OnNativeWindowChanged();
282 void NativeAppWindowViews::OnWidgetActivationChanged(views::Widget* widget,
283 bool active) {
284 app_window_->OnNativeWindowChanged();
285 if (active)
286 app_window_->OnNativeWindowActivated();
289 // WebContentsObserver implementation.
291 void NativeAppWindowViews::RenderViewCreated(
292 content::RenderViewHost* render_view_host) {
293 if (app_window_->requested_alpha_enabled() && CanHaveAlphaEnabled()) {
294 content::RenderWidgetHostView* view = render_view_host->GetView();
295 DCHECK(view);
296 view->SetBackgroundColor(SK_ColorTRANSPARENT);
300 void NativeAppWindowViews::RenderViewHostChanged(
301 content::RenderViewHost* old_host,
302 content::RenderViewHost* new_host) {
303 OnViewWasResized();
306 // views::View implementation.
308 void NativeAppWindowViews::Layout() {
309 DCHECK(web_view_);
310 web_view_->SetBounds(0, 0, width(), height());
311 OnViewWasResized();
314 void NativeAppWindowViews::ViewHierarchyChanged(
315 const ViewHierarchyChangedDetails& details) {
316 if (details.is_add && details.child == this) {
317 web_view_ = new views::WebView(NULL);
318 AddChildView(web_view_);
319 web_view_->SetWebContents(app_window_->web_contents());
323 gfx::Size NativeAppWindowViews::GetMinimumSize() const {
324 return size_constraints_.GetMinimumSize();
327 gfx::Size NativeAppWindowViews::GetMaximumSize() const {
328 return size_constraints_.GetMaximumSize();
331 void NativeAppWindowViews::OnFocus() {
332 web_view_->RequestFocus();
335 // NativeAppWindow implementation.
337 void NativeAppWindowViews::SetFullscreen(int fullscreen_types) {
338 // Stub implementation. See also ChromeNativeAppWindowViews.
339 widget_->SetFullscreen(fullscreen_types != AppWindow::FULLSCREEN_TYPE_NONE);
342 bool NativeAppWindowViews::IsFullscreenOrPending() const {
343 // Stub implementation. See also ChromeNativeAppWindowViews.
344 return widget_->IsFullscreen();
347 void NativeAppWindowViews::UpdateWindowIcon() {
348 widget_->UpdateWindowIcon();
351 void NativeAppWindowViews::UpdateWindowTitle() {
352 widget_->UpdateWindowTitle();
355 void NativeAppWindowViews::UpdateDraggableRegions(
356 const std::vector<extensions::DraggableRegion>& regions) {
357 // Draggable region is not supported for non-frameless window.
358 if (!frameless_)
359 return;
361 draggable_region_.reset(AppWindow::RawDraggableRegionsToSkRegion(regions));
362 OnViewWasResized();
365 SkRegion* NativeAppWindowViews::GetDraggableRegion() {
366 return draggable_region_.get();
369 void NativeAppWindowViews::UpdateShape(scoped_ptr<SkRegion> region) {
370 // Stub implementation. See also ChromeNativeAppWindowViews.
373 void NativeAppWindowViews::HandleKeyboardEvent(
374 const content::NativeWebKeyboardEvent& event) {
375 unhandled_keyboard_event_handler_.HandleKeyboardEvent(event,
376 GetFocusManager());
379 bool NativeAppWindowViews::IsFrameless() const {
380 return frameless_;
383 bool NativeAppWindowViews::HasFrameColor() const {
384 return false;
387 SkColor NativeAppWindowViews::ActiveFrameColor() const {
388 return SK_ColorBLACK;
391 SkColor NativeAppWindowViews::InactiveFrameColor() const {
392 return SK_ColorBLACK;
395 gfx::Insets NativeAppWindowViews::GetFrameInsets() const {
396 if (frameless_)
397 return gfx::Insets();
399 // The pretend client_bounds passed in need to be large enough to ensure that
400 // GetWindowBoundsForClientBounds() doesn't decide that it needs more than
401 // the specified amount of space to fit the window controls in, and return a
402 // number larger than the real frame insets. Most window controls are smaller
403 // than 1000x1000px, so this should be big enough.
404 gfx::Rect client_bounds = gfx::Rect(1000, 1000);
405 gfx::Rect window_bounds =
406 widget_->non_client_view()->GetWindowBoundsForClientBounds(client_bounds);
407 return window_bounds.InsetsFrom(client_bounds);
410 void NativeAppWindowViews::HideWithApp() {
413 void NativeAppWindowViews::ShowWithApp() {
416 gfx::Size NativeAppWindowViews::GetContentMinimumSize() const {
417 return size_constraints_.GetMinimumSize();
420 gfx::Size NativeAppWindowViews::GetContentMaximumSize() const {
421 return size_constraints_.GetMaximumSize();
424 void NativeAppWindowViews::SetContentSizeConstraints(
425 const gfx::Size& min_size,
426 const gfx::Size& max_size) {
427 size_constraints_.set_minimum_size(min_size);
428 size_constraints_.set_maximum_size(max_size);
429 widget_->OnSizeConstraintsChanged();
432 bool NativeAppWindowViews::CanHaveAlphaEnabled() const {
433 return widget_->IsTranslucentWindowOpacitySupported();
436 void NativeAppWindowViews::SetVisibleOnAllWorkspaces(bool always_visible) {
437 widget_->SetVisibleOnAllWorkspaces(always_visible);
440 } // namespace native_app_window