Index: chrome/browser/chromeos/accessibility/chromevox_panel.cc |
diff --git a/chrome/browser/chromeos/accessibility/chromevox_panel.cc b/chrome/browser/chromeos/accessibility/chromevox_panel.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..e8ae000eebbb7aa579afb2b4a071a4b16f68c541 |
--- /dev/null |
+++ b/chrome/browser/chromeos/accessibility/chromevox_panel.cc |
@@ -0,0 +1,160 @@ |
+// Copyright 2015 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 "ash/shell.h" |
+#include "chrome/browser/chromeos/accessibility/chromevox_panel.h" |
+#include "chrome/browser/profiles/profile.h" |
+#include "chrome/common/extensions/extension_constants.h" |
+#include "content/public/browser/web_contents.h" |
+#include "ui/views/controls/webview/webview.h" |
+#include "ui/views/layout/fill_layout.h" |
+#include "ui/views/widget/widget.h" |
+#include "ui/wm/core/shadow_types.h" |
+#include "ui/wm/core/window_animations.h" |
+ |
+namespace { |
+ |
+const int kPanelHeight = 35; |
+const char kChromeVoxPanelRelativeUrl[] = "/cvox2/background/panel.html"; |
+const char kFullscreenURLFragment[] = "fullscreen"; |
+ |
+class ChromeVoxPanelWebView : public views::WebView { |
Jun Mukai
2015/11/05 17:38:02
As I wrote below, OnDeviceScaleFactorChanged() met
dmazzoni
2015/11/05 20:12:58
Done.
|
+ public: |
+ ChromeVoxPanelWebView(Profile* profile, ChromeVoxPanel* panel) |
+ : views::WebView(profile), panel_(panel) {} |
+ |
+ ~ChromeVoxPanelWebView() override {} |
+ |
+ void OnDeviceScaleFactorChanged(float device_scale_factor) override { |
+ panel_->OnDeviceScaleFactorChanged(device_scale_factor); |
+ } |
+ |
+ private: |
+ ChromeVoxPanel* panel_; |
+ bool fullscreen_; |
Jun Mukai
2015/11/05 17:38:02
not used?
dmazzoni
2015/11/05 20:12:58
Done.
|
+ |
+ DISALLOW_COPY_AND_ASSIGN(ChromeVoxPanelWebView); |
+}; |
+ |
+} // namespace |
+ |
+class ChromeVoxPanelWebContentsObserver : public content::WebContentsObserver { |
+ public: |
+ ChromeVoxPanelWebContentsObserver(content::WebContents* web_contents, |
+ ChromeVoxPanel* panel) |
+ : content::WebContentsObserver(web_contents), panel_(panel) {} |
+ ~ChromeVoxPanelWebContentsObserver() override {} |
+ |
+ void DidFirstVisuallyNonEmptyPaint() override { |
+ panel_->DidFirstVisuallyNonEmptyPaint(); |
+ } |
+ |
+ void DidFinishNavigation( |
+ content::NavigationHandle* navigation_handle) override { |
+ // The ChromeVox panel uses the URL fragment to communicate state |
+ // to this panel host. |
+ std::string fragment = web_contents()->GetLastCommittedURL().ref(); |
+ if (fragment == kFullscreenURLFragment) |
+ panel_->EnterFullscreen(); |
+ else |
+ panel_->ExitFullscreen(); |
+ } |
+ |
+ private: |
+ ChromeVoxPanel* panel_; |
+ |
+ DISALLOW_COPY_AND_ASSIGN(ChromeVoxPanelWebContentsObserver); |
+}; |
+ |
+ChromeVoxPanel::ChromeVoxPanel(Profile* profile) : profile_(profile) { |
+ widget_ = new views::Widget(); |
+ views::Widget::InitParams params( |
+ views::Widget::InitParams::TYPE_WINDOW_FRAMELESS); |
+ root_window_ = ash::Shell::GetPrimaryRootWindow(); |
+ params.parent = root_window_; |
Jun Mukai
2015/11/05 17:38:01
Please do not create a window directly under the r
dmazzoni
2015/11/05 20:12:58
Done.
|
+ params.delegate = this; |
+ widget_->Init(params); |
+ UpdateWidgetBounds(); |
Jun Mukai
2015/11/05 17:38:01
IIRC, the initial bounds (set in params) and setti
dmazzoni
2015/11/05 20:12:58
Done.
|
+ |
+ std::string url("chrome-extension://"); |
+ url += extension_misc::kChromeVoxExtensionId; |
+ url += kChromeVoxPanelRelativeUrl; |
+ |
+ ChromeVoxPanelWebView* web_view = new ChromeVoxPanelWebView(profile_, this); |
+ widget_->SetContentsView(web_view); |
Jun Mukai
2015/11/05 17:38:01
It's better to implement GetContentsView() rather
dmazzoni
2015/11/05 20:12:58
Done.
|
+ SetShadowType(widget_->GetNativeWindow(), wm::SHADOW_TYPE_RECTANGULAR); |
+ web_contents_observer_.reset( |
+ new ChromeVoxPanelWebContentsObserver(web_view->GetWebContents(), this)); |
+ web_view->LoadInitialURL(GURL(url)); |
+ |
+ ash::Shell::GetScreen()->AddObserver(this); |
+} |
+ |
+ChromeVoxPanel::~ChromeVoxPanel() { |
+ ash::Shell::GetScreen()->RemoveObserver(this); |
+} |
+ |
+int ChromeVoxPanel::GetHeight() { |
+ return kPanelHeight; |
+} |
+ |
+aura::Window* ChromeVoxPanel::GetRootWindow() { |
+ return root_window_; |
Jun Mukai
2015/11/05 17:38:01
You don't have to remember the root window.
It's j
dmazzoni
2015/11/05 20:12:58
Done.
|
+} |
+ |
+void ChromeVoxPanel::Close() { |
+ widget_->Close(); |
+} |
+ |
+void ChromeVoxPanel::OnDeviceScaleFactorChanged(float device_scale_factor) { |
+ UpdateWidgetBounds(); |
Jun Mukai
2015/11/05 17:38:01
I think you don't have to handle the device scale
dmazzoni
2015/11/05 20:12:58
Done.
|
+} |
+ |
+void ChromeVoxPanel::DidFirstVisuallyNonEmptyPaint() { |
+ widget_->Show(); |
+ ash::Shell::GetInstance()->UpdateDisplayWorkAreaInsets(root_window_); |
+} |
+ |
+void ChromeVoxPanel::EnterFullscreen() { |
+ fullscreen_ = true; |
+ UpdateWidgetBounds(); |
+} |
+ |
+void ChromeVoxPanel::ExitFullscreen() { |
+ fullscreen_ = false; |
+ UpdateWidgetBounds(); |
+} |
+ |
+const views::Widget* ChromeVoxPanel::GetWidget() const { |
+ return widget_; |
+} |
+ |
+views::Widget* ChromeVoxPanel::GetWidget() { |
+ return widget_; |
+} |
+ |
+void ChromeVoxPanel::DeleteDelegate() { |
+ ash::Shell::GetInstance()->UpdateDisplayWorkAreaInsets(root_window_); |
Jun Mukai
2015/11/05 17:38:02
This may not work quite well. At this point, the w
dmazzoni
2015/11/05 20:12:58
Thanks.
I went with the first approach, observing
|
+ delete this; |
+} |
+ |
+void ChromeVoxPanel::OnDisplayMetricsChanged(const gfx::Display& display, |
+ uint32_t changed_metrics) { |
+ // This is called when the primary display changes. Move the panel to the |
+ // new primary display. |
+ if (root_window_ != ash::Shell::GetPrimaryRootWindow()) { |
+ UpdateWidgetBounds(); |
+ ash::Shell::GetInstance()->UpdateDisplayWorkAreaInsets(root_window_); |
+ root_window_ = ash::Shell::GetPrimaryRootWindow(); |
+ ash::Shell::GetInstance()->UpdateDisplayWorkAreaInsets(root_window_); |
+ } |
+} |
+ |
+void ChromeVoxPanel::UpdateWidgetBounds() { |
+ gfx::Rect bounds(0, 0, root_window_->bounds().width(), |
+ root_window_->bounds().height()); |
+ if (!fullscreen_) |
+ bounds.set_height(kPanelHeight); |
+ widget_->SetBounds(bounds); |
+} |