summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorgarykac <garykac@chromium.org>2015-01-06 17:11:44 -0800
committerCommit bot <commit-bot@chromium.org>2015-01-07 01:13:18 +0000
commitcf383c3bf373f34ce88f684d1601ca6a44e469f4 (patch)
treeba0b49d84cd85b66e72f67b266aee96fd9a714eb
parentb533d6533585377edd63ec6500469f6c4fba602a (diff)
downloadchromium_src-cf383c3bf373f34ce88f684d1601ca6a44e469f4.zip
chromium_src-cf383c3bf373f34ce88f684d1601ca6a44e469f4.tar.gz
chromium_src-cf383c3bf373f34ce88f684d1601ca6a44e469f4.tar.bz2
[Chromoting] Add AppRemoting webapp support.
This cl includes the new files necessary to build the app_remoting sample webapp and a few changes to make the core chromoting code aware of app_remoting (AR, as opposed to the standard CRD webapp) This does not add the app_remoting sample target to all.gyp, so to build it you will need to manually update your GYP config (for now). BUG= Review URL: https://codereview.chromium.org/789253009 Cr-Commit-Position: refs/heads/master@{#310202}
-rw-r--r--remoting/app_remoting_webapp.gyp18
-rw-r--r--remoting/app_remoting_webapp_build.gypi305
-rw-r--r--remoting/resources/remoting_strings.grd3
-rw-r--r--remoting/webapp/app_remoting/apps/sample_app/icon128.pngbin0 -> 2634 bytes
-rw-r--r--remoting/webapp/app_remoting/apps/sample_app/icon16.pngbin0 -> 458 bytes
-rw-r--r--remoting/webapp/app_remoting/apps/sample_app/icon48.pngbin0 -> 1190 bytes
-rw-r--r--remoting/webapp/app_remoting/apps/sample_app/manifest.json.jinja23
-rw-r--r--remoting/webapp/app_remoting/html/ar_dialog.css19
-rw-r--r--remoting/webapp/app_remoting/html/context_menu.css125
-rw-r--r--remoting/webapp/app_remoting/html/context_menu.html15
-rw-r--r--remoting/webapp/app_remoting/html/feedback_consent.css22
-rw-r--r--remoting/webapp/app_remoting/html/feedback_consent.html68
-rw-r--r--remoting/webapp/app_remoting/html/idle_dialog.html12
-rw-r--r--remoting/webapp/app_remoting/html/template_lg.html47
-rw-r--r--remoting/webapp/app_remoting/js/app_remoting.js360
-rw-r--r--remoting/webapp/app_remoting/js/application_context_menu.js113
-rw-r--r--remoting/webapp/app_remoting/js/ar_background.js73
-rw-r--r--remoting/webapp/app_remoting/js/ar_main.js19
-rw-r--r--remoting/webapp/app_remoting/js/context_menu_adapter.js56
-rw-r--r--remoting/webapp/app_remoting/js/context_menu_chrome.js92
-rw-r--r--remoting/webapp/app_remoting/js/context_menu_dom.js328
-rw-r--r--remoting/webapp/app_remoting/js/drag_and_drop.js142
-rw-r--r--remoting/webapp/app_remoting/js/feedback_consent.js229
-rw-r--r--remoting/webapp/app_remoting/js/idle_detector.js166
-rw-r--r--remoting/webapp/app_remoting/js/keyboard_layouts_menu.js185
-rw-r--r--remoting/webapp/app_remoting/js/loading_window.js71
-rw-r--r--remoting/webapp/app_remoting/js/submenu_manager.js109
-rw-r--r--remoting/webapp/app_remoting/js/window_activation_menu.js84
-rw-r--r--remoting/webapp/app_remoting/manifest_common.json.jinja246
-rw-r--r--remoting/webapp/base/html/main.css8
-rw-r--r--remoting/webapp/base/html/message_window.html2
-rw-r--r--remoting/webapp/base/js/application.js14
-rw-r--r--remoting/webapp/browser_test/mock_session_connector.js15
-rw-r--r--remoting/webapp/crd/js/client_session.js6
-rw-r--r--remoting/webapp/crd/js/connection_stats.js8
-rw-r--r--remoting/webapp/crd/js/error.js3
-rw-r--r--remoting/webapp/crd/js/plugin_settings.js16
-rw-r--r--remoting/webapp/crd/js/remoting.js31
-rw-r--r--remoting/webapp/crd/js/server_log_entry.js4
-rw-r--r--remoting/webapp/crd/js/session_connector.js13
-rw-r--r--remoting/webapp/crd/js/session_connector_impl.js19
-rw-r--r--remoting/webapp/js_proto/dom_proto.js21
42 files changed, 2844 insertions, 26 deletions
diff --git a/remoting/app_remoting_webapp.gyp b/remoting/app_remoting_webapp.gyp
new file mode 100644
index 0000000..639d84f
--- /dev/null
+++ b/remoting/app_remoting_webapp.gyp
@@ -0,0 +1,18 @@
+# Copyright 2014 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.
+
+{
+ 'includes': [
+ 'app_remoting_webapp_build.gypi',
+ ],
+
+ 'targets': [
+ {
+ 'target_name': 'ar_sample_app',
+ 'app_id': 'ljacajndfccfgnfohlgkdphmbnpkjflk',
+ 'app_name': 'App Remoting Client',
+ 'app_description': 'App Remoting client',
+ },
+ ], # end of targets
+}
diff --git a/remoting/app_remoting_webapp_build.gypi b/remoting/app_remoting_webapp_build.gypi
new file mode 100644
index 0000000..3ea39a9
--- /dev/null
+++ b/remoting/app_remoting_webapp_build.gypi
@@ -0,0 +1,305 @@
+# Copyright 2014 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.
+
+{
+ 'includes': [
+ 'remoting_version.gypi',
+ 'remoting_webapp_files.gypi',
+ ],
+
+ 'variables': {
+ 'chromium_code': 1,
+
+ 'run_jscompile%': 0,
+
+ # This variable is used to define the target environment for the app
+ # being built. The allowed values are dev, test, staging, and prod.
+ 'ar_service_environment%': 'dev',
+
+ # Identify internal vs. public build targets.
+ 'ar_internal%': 0,
+
+ 'remoting_localize_path': 'tools/build/remoting_localize.py',
+
+ # TODO(wez): Split into shared-stub and app-specific resources.
+ 'webapp_locale_dir': '<(SHARED_INTERMEDIATE_DIR)/remoting/webapp/_locales',
+ 'remoting_locales': [
+ 'en',
+ ],
+ 'remoting_webapp_locale_files': [
+ # Build the list of .json files generated from remoting_strings.grd.
+ '<!@pymod_do_main(remoting_localize --locale_output '
+ '"<(webapp_locale_dir)/@{json_suffix}/messages.json" '
+ '--print_only <(remoting_locales))',
+ ],
+
+ 'ar_shared_resource_files': [
+ 'webapp/app_remoting/html/ar_dialog.css',
+ 'webapp/app_remoting/html/feedback_consent.css',
+ 'webapp/app_remoting/html/feedback_consent.html',
+ 'webapp/app_remoting/html/context_menu.css',
+ 'resources/drag.webp',
+ '<@(remoting_webapp_resource_files)',
+ ],
+
+ # Variables for main.html.
+ # These template files are used to construct the webapp html files.
+ 'ar_main_template':
+ 'webapp/app_remoting/html/template_lg.html',
+ 'ar_main_template_files': [
+ 'webapp/base/html/client_plugin.html',
+ 'webapp/base/html/dialog_auth.html',
+ 'webapp/app_remoting/html/context_menu.html',
+ 'webapp/app_remoting/html/idle_dialog.html',
+ ],
+ 'ar_main_js_files': [
+ 'webapp/app_remoting/js/application_context_menu.js',
+ 'webapp/app_remoting/js/app_remoting.js',
+ 'webapp/app_remoting/js/ar_main.js',
+ 'webapp/app_remoting/js/context_menu_adapter.js',
+ 'webapp/app_remoting/js/context_menu_chrome.js',
+ 'webapp/app_remoting/js/context_menu_dom.js',
+ 'webapp/app_remoting/js/drag_and_drop.js',
+ 'webapp/app_remoting/js/idle_detector.js',
+ 'webapp/app_remoting/js/keyboard_layouts_menu.js',
+ 'webapp/app_remoting/js/loading_window.js',
+ 'webapp/app_remoting/js/submenu_manager.js',
+ 'webapp/app_remoting/js/window_activation_menu.js',
+ 'webapp/base/js/application.js',
+ 'webapp/base/js/auth_dialog.js',
+ 'webapp/base/js/base.js',
+ 'webapp/base/js/message_window_helper.js',
+ 'webapp/base/js/message_window_manager.js',
+ '<@(remoting_webapp_js_auth_client2host_files)',
+ '<@(remoting_webapp_js_auth_google_files)',
+ '<@(remoting_webapp_js_cast_extension_files)',
+ '<@(remoting_webapp_js_client_files)',
+ '<@(remoting_webapp_js_core_files)',
+ '<@(remoting_webapp_js_gnubby_auth_files)',
+ '<@(remoting_webapp_js_host_files)',
+ '<@(remoting_webapp_js_logging_files)',
+ '<@(remoting_webapp_js_signaling_files)',
+ '<@(remoting_webapp_js_ui_files)',
+ ],
+
+ 'ar_background_js_files': [
+ 'webapp/app_remoting/js/ar_background.js',
+ 'webapp/base/js/platform.js',
+ ],
+
+ 'ar_all_js_files': [
+ '<@(ar_main_js_files)',
+ # Referenced from wcs_sandbox.html.
+ '<@(remoting_webapp_js_wcs_sandbox_files)',
+ # Referenced from the manifest.
+ '<@(ar_background_js_files)',
+ # Referenced from feedback_consent.html.
+ 'webapp/app_remoting/js/feedback_consent.js',
+ # Referenced from message_window.html.
+ 'webapp/base/js/message_window.js',
+ ],
+ }, # end of variables
+
+ 'target_defaults': {
+ 'type': 'none',
+
+ 'dependencies': [
+ # TODO(wez): Create proper resources for shared-stub and app-specific
+ # stubs.
+ '../remoting/remoting.gyp:remoting_resources',
+ ],
+
+ 'locale_files': [
+ '<@(remoting_webapp_locale_files)',
+ ],
+
+ 'includes': [
+ '../chrome/js_unittest_vars.gypi',
+ ],
+
+ 'variables': {
+ 'ar_app_manifest_app':
+ '>(ar_app_path)/manifest.json.jinja2',
+ 'ar_app_manifest_common':
+ 'webapp/app_remoting/manifest_common.json.jinja2',
+ 'ar_app_specific_files': [
+ '>(ar_app_path)/icon16.png',
+ '>(ar_app_path)/icon48.png',
+ '>(ar_app_path)/icon128.png',
+ ],
+ 'ar_generated_html_files': [
+ '<(SHARED_INTERMEDIATE_DIR)/>(_target_name)/main.html',
+ '<(SHARED_INTERMEDIATE_DIR)/>(_target_name)/wcs_sandbox.html',
+ ],
+ 'ar_webapp_files': [
+ '<@(ar_app_specific_files)',
+ '<@(ar_shared_resource_files)',
+ '<@(ar_all_js_files)',
+ '<@(ar_generated_html_files)',
+ ],
+ 'output_dir': '<(PRODUCT_DIR)/app_streaming/<@(ar_service_environment)/>(_target_name)',
+ 'zip_path': '<(PRODUCT_DIR)/app_streaming/<@(ar_service_environment)/>(_target_name).zip',
+ 'remoting_app_id': [],
+ 'remoting_app_name': '>(_app_name)',
+ 'remoting_app_description': '>(_app_description)',
+
+ 'conditions': [
+ ['ar_internal != 1', {
+ 'ar_app_name': 'sample_app',
+ 'ar_app_path': 'webapp/app_remoting/apps/>(ar_app_name)',
+ }, {
+ # This takes target names of the form 'ar_vvv_xxx_xxx' and extracts
+ # the vendor ('vvv') and the app name ('xxx_xxx').
+ 'ar_app_vendor': '>!(python -c "import sys; print sys.argv[1].split(\'_\')[1]" >(_target_name))',
+ 'ar_app_name': '>!(python -c "import sys; print \'_\'.join(sys.argv[1].split(\'_\')[2:])" >(_target_name))',
+ 'ar_app_path': 'webapp/app_remoting/apps/internal/>(ar_app_vendor)/>(ar_app_name)',
+ }],
+ ], # conditions
+
+ }, # variables
+
+ 'actions': [
+ {
+ 'action_name': 'Build ">(ar_app_name)" application stub',
+ 'inputs': [
+ 'webapp/build-webapp.py',
+ '<(chrome_version_path)',
+ '<(remoting_version_path)',
+ '<@(ar_webapp_files)',
+ '<@(remoting_webapp_locale_files)',
+ '<@(ar_generated_html_files)',
+ '<(ar_app_manifest_app)',
+ '<(ar_app_manifest_common)',
+ ],
+ 'outputs': [
+ '<(output_dir)',
+ '<(zip_path)',
+ ],
+ 'action': [
+ 'python', 'webapp/build-webapp.py',
+ '<(buildtype)',
+ '<(version_full)',
+ '<(output_dir)',
+ '<(zip_path)',
+ '<(ar_app_manifest_app)', # Manifest template
+ 'app_remoting', # Web app type
+ '<@(ar_webapp_files)',
+ '<@(ar_generated_html_files)',
+ '--locales',
+ '<@(remoting_webapp_locale_files)',
+ '--jinja_paths',
+ 'webapp/app_remoting',
+ '<@(remoting_app_id)',
+ '--app_name',
+ '<(remoting_app_name)',
+ '--app_description',
+ '<(remoting_app_description)',
+ '--service_environment',
+ '<@(ar_service_environment)',
+ ],
+ },
+ {
+ 'action_name': 'Build ">(ar_app_name)" main.html',
+ 'inputs': [
+ 'webapp/build-html.py',
+ '<(ar_main_template)',
+ '<@(ar_main_template_files)',
+ ],
+ 'outputs': [
+ '<(SHARED_INTERMEDIATE_DIR)/>(_target_name)/main.html',
+ ],
+ 'action': [
+ 'python', 'webapp/build-html.py',
+ '<(SHARED_INTERMEDIATE_DIR)/>(_target_name)/main.html',
+ '<(ar_main_template)',
+ '--template',
+ '<@(ar_main_template_files)',
+ '--js',
+ '<@(ar_main_js_files)',
+ ],
+ },
+ {
+ 'action_name': 'Build ">(ar_app_name)" wcs_sandbox.html',
+ 'inputs': [
+ 'webapp/build-html.py',
+ '<(remoting_webapp_template_wcs_sandbox)',
+ ],
+ 'outputs': [
+ '<(SHARED_INTERMEDIATE_DIR)/>(_target_name)/wcs_sandbox.html',
+ ],
+ 'action': [
+ 'python', 'webapp/build-html.py',
+ '<(SHARED_INTERMEDIATE_DIR)/>(_target_name)/wcs_sandbox.html',
+ '<(remoting_webapp_template_wcs_sandbox)',
+ '--js',
+ '<@(remoting_webapp_wcs_sandbox_html_js_files)',
+ ],
+ },
+ ], # actions
+ 'conditions': [
+ ['buildtype == "Dev"', {
+ # Normally, the app-id for the orchestrator is automatically extracted
+ # from the webapp's extension id, but that approach doesn't work for
+ # dev webapp builds (since they all share the same dev extension id).
+ # The --appid arg will create a webapp that registers the given app-id
+ # rather than using the extension id.
+ # This is only done for Dev apps because the app-id for Release apps
+ # *must* match the extension id.
+ 'variables': {
+ 'remoting_app_id': ['--appid', '>(_app_id)'],
+ },
+ }],
+ ['run_jscompile != 0', {
+ 'actions': [
+ {
+ 'action_name': 'Verify >(ar_app_name) main.html',
+ 'variables': {
+ 'success_stamp': '<(PRODUCT_DIR)/>(_target_name)_main_jscompile.stamp',
+ },
+ 'inputs': [
+ 'tools/jscompile.py',
+ '<@(ar_main_js_files)',
+ '<@(remoting_webapp_js_proto_files)',
+ # Include zip as input so that this action is run after the build.
+ '<(zip_path)',
+ ],
+ 'outputs': [
+ '<(success_stamp)',
+ ],
+ 'action': [
+ 'python', 'tools/jscompile.py',
+ '<@(ar_main_js_files)',
+ '<@(remoting_webapp_js_proto_files)',
+ '--success-stamp',
+ '<(success_stamp)'
+ ],
+ },
+ {
+ 'action_name': 'Verify >(ar_app_name) background.js',
+ 'variables': {
+ 'success_stamp': '<(PRODUCT_DIR)/>(_target_name)_background_jscompile.stamp',
+ },
+ 'inputs': [
+ 'tools/jscompile.py',
+ '<@(ar_background_js_files)',
+ '<@(remoting_webapp_js_proto_files)',
+ # Include zip as input so that this action is run after the build.
+ '<(zip_path)',
+ ],
+ 'outputs': [
+ '<(success_stamp)',
+ ],
+ 'action': [
+ 'python', 'tools/jscompile.py',
+ '<@(ar_background_js_files)',
+ '<@(remoting_webapp_js_proto_files)',
+ '--success-stamp',
+ '<(success_stamp)'
+ ],
+ },
+ ], # actions
+ }],
+ ], # conditions
+ }, # target_defaults
+}
diff --git a/remoting/resources/remoting_strings.grd b/remoting/resources/remoting_strings.grd
index e2f95a3..868f158 100644
--- a/remoting/resources/remoting_strings.grd
+++ b/remoting/resources/remoting_strings.grd
@@ -298,6 +298,9 @@
<message desc="Error displayed when an operation is attempted that requires a signed-in user, but no-one is currently signed in." name="IDS_ERROR_NOT_AUTHENTICATED">
You are not signed in to Chrome Remote Desktop. Please sign in and try again.
</message>
+ <message desc="Error displayed when a user attempts to connect to an application that they have not been authorized to access." name="IDS_ERROR_NOT_AUTHORIZED">
+ You are not authorized to access this application.
+ </message>
<message desc="Error displayed if the client plugin fails to load." name="IDS_ERROR_MISSING_PLUGIN">
Some components required for Chrome Remote Desktop are missing. Please make sure you're running the latest version of Chrome and try again.
</message>
diff --git a/remoting/webapp/app_remoting/apps/sample_app/icon128.png b/remoting/webapp/app_remoting/apps/sample_app/icon128.png
new file mode 100644
index 0000000..9f705cb
--- /dev/null
+++ b/remoting/webapp/app_remoting/apps/sample_app/icon128.png
Binary files differ
diff --git a/remoting/webapp/app_remoting/apps/sample_app/icon16.png b/remoting/webapp/app_remoting/apps/sample_app/icon16.png
new file mode 100644
index 0000000..bc70a0c
--- /dev/null
+++ b/remoting/webapp/app_remoting/apps/sample_app/icon16.png
Binary files differ
diff --git a/remoting/webapp/app_remoting/apps/sample_app/icon48.png b/remoting/webapp/app_remoting/apps/sample_app/icon48.png
new file mode 100644
index 0000000..ad0bf76
--- /dev/null
+++ b/remoting/webapp/app_remoting/apps/sample_app/icon48.png
Binary files differ
diff --git a/remoting/webapp/app_remoting/apps/sample_app/manifest.json.jinja2 b/remoting/webapp/app_remoting/apps/sample_app/manifest.json.jinja2
new file mode 100644
index 0000000..bad2739
--- /dev/null
+++ b/remoting/webapp/app_remoting/apps/sample_app/manifest.json.jinja2
@@ -0,0 +1,3 @@
+{
+ {% include "manifest_common.json.jinja2" %}
+}
diff --git a/remoting/webapp/app_remoting/html/ar_dialog.css b/remoting/webapp/app_remoting/html/ar_dialog.css
new file mode 100644
index 0000000..9258578
--- /dev/null
+++ b/remoting/webapp/app_remoting/html/ar_dialog.css
@@ -0,0 +1,19 @@
+/* Copyright 2014 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.
+ */
+
+#idle-dialog {
+ position: absolute;
+ top: 100px;
+ width: 100%;
+}
+
+#idle-dialog .kd-modaldialog,
+#auth-dialog .kd-modaldialog {
+ /*
+ * kd-modaldialog uses outline, which doesn't affect the bounding box, and so
+ * doesn't work well when adjusting the window shape.
+ */
+ border: 1px solid gray;
+}
diff --git a/remoting/webapp/app_remoting/html/context_menu.css b/remoting/webapp/app_remoting/html/context_menu.css
new file mode 100644
index 0000000..ba3e0c1
--- /dev/null
+++ b/remoting/webapp/app_remoting/html/context_menu.css
@@ -0,0 +1,125 @@
+/* Copyright 2014 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.
+ */
+
+#context-menu {
+ position: fixed;
+ bottom: 0; /* The vertical position is controlled by context_menu_dom.js */
+ left: -48px;
+ background-color: #c4c4c4;
+ border: 1px solid #a6a6a6;
+ z-index: 101;
+ transition-property: left;
+ transition-duration: 0.3s;
+ box-shadow: 1px 1px 10px rgba(0, 0, 0, 0.5);
+ opacity: 0.8;
+}
+
+#context-menu:hover,
+#context-menu.opened,
+#context-menu.menu-opened {
+ left: 0;
+ opacity: 1;
+}
+
+.no-gaps {
+ font-size: 0;
+}
+
+.context-menu-icon {
+ margin-top: 2px;
+}
+
+.context-menu-icon:hover,
+.context-menu-stub:hover {
+ background-color: #d5d5d5;
+}
+
+#context-menu.opened .context-menu-stub,
+.context-menu-icon:active {
+ background-color: #a6a6a6;
+}
+
+.context-menu-stub {
+ display: inline-block;
+ width: 12px;
+ height: 50px;
+ background: url("drag.webp");
+ cursor: move;
+}
+
+.etched {
+ position: relative;
+}
+
+.etched:after {
+ content: "";
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ top: 0;
+ left: 0;
+ border-left: 1px solid rgba(255, 255, 255, 0.2);
+ border-top: 1px solid rgba(255, 255, 255, 0.2);
+}
+
+#context-menu ul {
+ visibility: hidden;
+ position: absolute;
+ left: 24px;
+ padding: 0;
+ margin: 0;
+ list-style-type: none;
+ background: white;
+ outline: 1px solid rgba(0, 0, 0, 0.2);
+ padding: 0 0 6px;
+ box-shadow: 0px 2px 4px rgba(0, 0, 0, 0.2);
+ z-index: 102;
+}
+
+#context-menu ul.menu-align-bottom {
+ bottom: 24px;
+}
+
+#context-menu ul:not(.menu-align-bottom) {
+ top: 24px;
+}
+
+#context-menu ul.opened {
+ visibility: visible;
+}
+
+#context-menu li {
+ padding: 6px 44px 6px 30px;
+ white-space: nowrap;
+}
+
+#context-menu li:hover {
+ background-color: #EEE;
+}
+
+#context-menu li.selected {
+ background-image: url('tick.webp');
+ background-position: left center;
+ background-repeat: no-repeat;
+}
+
+#context-menu li.menu-group-header {
+ pointer-events: none;
+ font-style: italic;
+ color: gray;
+}
+
+#context-menu li.menu-group-item {
+ margin-left: 16px;
+}
+
+.context-menu-screen {
+ width: 100%;
+ height: 100%;
+ position: fixed;
+ top: 0;
+ left: 0;
+ z-index: 100;
+}
diff --git a/remoting/webapp/app_remoting/html/context_menu.html b/remoting/webapp/app_remoting/html/context_menu.html
new file mode 100644
index 0000000..ebc056e
--- /dev/null
+++ b/remoting/webapp/app_remoting/html/context_menu.html
@@ -0,0 +1,15 @@
+<!--
+Copyright 2014 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.
+-->
+
+<div id="context-menu" draggable="true" hidden>
+ <div class="no-gaps">
+ <img src="icon48.png" class="context-menu-icon">
+ <span class="context-menu-stub etched"></span>
+ </div>
+ <ul>
+ </ul>
+ <div class="context-menu-screen" hidden></div>
+</div>
diff --git a/remoting/webapp/app_remoting/html/feedback_consent.css b/remoting/webapp/app_remoting/html/feedback_consent.css
new file mode 100644
index 0000000..3e02a05
--- /dev/null
+++ b/remoting/webapp/app_remoting/html/feedback_consent.css
@@ -0,0 +1,22 @@
+/* Copyright 2014 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.
+ */
+
+#privacy-info {
+ margin-top: 12px;
+}
+
+a.disabled {
+ color: #888;
+ pointer-events: none;
+}
+
+.information-box > p {
+ text-align: left;
+ margin-top: 8px;
+}
+
+.information-box > p:first-child {
+ margin-top: 0;
+} \ No newline at end of file
diff --git a/remoting/webapp/app_remoting/html/feedback_consent.html b/remoting/webapp/app_remoting/html/feedback_consent.html
new file mode 100644
index 0000000..5cd53a0
--- /dev/null
+++ b/remoting/webapp/app_remoting/html/feedback_consent.html
@@ -0,0 +1,68 @@
+<!doctype html>
+<!--
+Copyright 2014 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.
+-->
+
+<html>
+ <head>
+ <meta charset="utf-8">
+ <link rel="icon" type="image/png" href="icon16.png">
+ <link rel="stylesheet" href="open_sans.css">
+ <link rel="stylesheet" href="feedback_consent.css">
+ <link rel="stylesheet" href="main.css">
+ <link rel="stylesheet" href="message_window.css">
+ <script src="error.js"></script>
+ <script src="feedback_consent.js"></script>
+ <script src="oauth2_api.js"></script>
+ <script src="plugin_settings.js"></script>
+ <script src="l10n.js"></script>
+ <script src="xhr.js"></script>
+ <title i18n-content="FEEDBACK_CONSENT_TITLE"></title>
+ </head>
+ <body>
+ <h2 i18n-content="FEEDBACK_CONSENT_TITLE"></h2>
+ <p i18n-content="FEEDBACK_CONSENT"
+ class="message"></p>
+ <div id="form-body">
+ <label class="checkbox-label"
+ id="abandon-host-label">
+ <input id="abandon-host"
+ type="checkbox">
+ <span i18n-content="FEEDBACK_ABANDON_HOST"></span>
+ </label>
+ <label class="checkbox-label checkbox-indent disabled"
+ id="include-logs-label">
+ <input id="include-logs"
+ type="checkbox"
+ disabled>
+ <span i18n-content="FEEDBACK_INCLUDE_LOGS"></span>
+ <a id="learn-more"
+ i18n-content="LEARN_MORE"
+ class="disabled"></a>
+ </label>
+ <div id="privacy-info"
+ class="information-box"
+ hidden>
+ <p i18n-content="FEEDBACK_PRIVACY_INFORMATION1"></p>
+ <p i18n-content="FEEDBACK_PRIVACY_INFORMATION2"></p>
+ </div>
+ </div> <!-- form-body -->
+ <div id="abandon-failed"
+ class="message error-state multi-line-error-state"
+ i18n-content="FEEDBACK_ABANDON_FAILED"
+ hidden></div>
+ <div class="button-row">
+ <span id="working"
+ class="waiting"
+ i18n-content="WORKING"
+ hidden></span>
+ <button id="feedback-consent-ok"
+ i18n-content="OK"
+ autofocus="autofocus"></button>
+ <button id="feedback-consent-cancel"
+ i18n-content="CANCEL"></button>
+ </div>
+ </body>
+</html>
diff --git a/remoting/webapp/app_remoting/html/idle_dialog.html b/remoting/webapp/app_remoting/html/idle_dialog.html
new file mode 100644
index 0000000..bcca6bb
--- /dev/null
+++ b/remoting/webapp/app_remoting/html/idle_dialog.html
@@ -0,0 +1,12 @@
+<div id="idle-dialog" class="horizontally-centered" hidden>
+ <div class="kd-modaldialog">
+ <p class="idle-warning-message"
+ i18n-content="IDLE_TIMEOUT_WARNING"></p>
+ <div class="button-row">
+ <button class="idle-dialog-continue"
+ i18n-content="IDLE_CONTINUE"></button>
+ <button class="idle-dialog-disconnect"
+ i18n-content="IDLE_DISCONNECT"></button>
+ </div>
+ </div>
+</div>
diff --git a/remoting/webapp/app_remoting/html/template_lg.html b/remoting/webapp/app_remoting/html/template_lg.html
new file mode 100644
index 0000000..b6827e1
--- /dev/null
+++ b/remoting/webapp/app_remoting/html/template_lg.html
@@ -0,0 +1,47 @@
+<!doctype html>
+<!--
+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.
+-->
+
+<html class="full-height">
+ <head>
+ <meta charset="utf-8">
+ <link rel="icon" type="image/png" href="icon16.png">
+ <link rel="stylesheet" href="open_sans.css">
+ <link rel="stylesheet" href="connection_stats.css">
+ <link rel="stylesheet" href="context_menu.css">
+ <link rel="stylesheet" href="main.css">
+ <link rel="stylesheet" href="menu_button.css">
+
+ <meta-include type="javascript"/>
+
+ <title i18n-content="PRODUCT_NAME"></title>
+ </head>
+
+ <body class="full-height">
+
+ <div id="host-plugin-container"></div>
+
+ <iframe id="wcs-sandbox" src="wcs_sandbox.html" hidden></iframe>
+
+ <meta-include src="webapp/base/html/dialog_auth.html"/>
+
+ <div id="session-mode"
+ data-ui-mode="in-session home.client"
+ class="full-height"
+ hidden>
+
+ <meta-include src="webapp/base/html/client_plugin.html"/>
+
+ </div> <!-- session-mode -->
+
+ <meta-include src="webapp/app_remoting/html/context_menu.html"/>
+ <meta-include src="webapp/app_remoting/html/idle_dialog.html"/>
+
+ <div id="statistics" dir="ltr" class="selectable" hidden>
+ </div> <!-- statistics -->
+
+ </body>
+</html>
diff --git a/remoting/webapp/app_remoting/js/app_remoting.js b/remoting/webapp/app_remoting/js/app_remoting.js
new file mode 100644
index 0000000..319a2e0
--- /dev/null
+++ b/remoting/webapp/app_remoting/js/app_remoting.js
@@ -0,0 +1,360 @@
+// Copyright 2014 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.
+
+/**
+ * @fileoverview
+ * This class implements the functionality that is specific to application
+ * remoting ("AppRemoting" or AR).
+ */
+
+'use strict';
+
+/** @suppress {duplicate} */
+var remoting = remoting || {};
+
+/**
+ * @param {remoting.Application} app The main app that owns this delegate.
+ * @constructor
+ * @implements {remoting.Application.Delegate}
+ */
+remoting.AppRemoting = function(app) {
+ app.setDelegate(this);
+
+ /**
+ * @type {remoting.ApplicationContextMenu}
+ * @private
+ */
+ this.contextMenu_ = null;
+
+ /**
+ * @type {remoting.KeyboardLayoutsMenu}
+ * @private
+ */
+ this.keyboardLayoutsMenu_ = null;
+
+ /**
+ * @type {remoting.WindowActivationMenu}
+ * @private
+ */
+ this.windowActivationMenu_ = null;
+};
+
+/**
+ * Type definition for the RunApplicationResponse returned by the API.
+ *
+ * @constructor
+ * @private
+ */
+remoting.AppRemoting.AppHostResponse = function() {
+ /** @type {string} */
+ this.status = '';
+ /** @type {string} */
+ this.hostJid = '';
+ /** @type {string} */
+ this.authorizationCode = '';
+ /** @type {string} */
+ this.sharedSecret = '';
+
+ this.host = {
+ /** @type {string} */
+ applicationId: '',
+
+ /** @type {string} */
+ hostId: ''};
+};
+
+/**
+ * Callback for when the userinfo (email and user name) is available from
+ * the identity API.
+ *
+ * @param {string} email The user's email address.
+ * @param {string} fullName The user's full name.
+ * @return {void} Nothing.
+ */
+remoting.onUserInfoAvailable = function(email, fullName) {
+};
+
+/**
+ * Initialize the application and register all event handlers. After this
+ * is called, the app is running and waiting for user events.
+ *
+ * @param {remoting.SessionConnector} connector
+ * @return {void} Nothing.
+ */
+remoting.AppRemoting.prototype.init = function(connector) {
+ remoting.initGlobalObjects();
+ remoting.initIdentity(remoting.onUserInfoAvailable);
+
+ remoting.initGlobalEventHandlers();
+
+ var restoreHostWindows = function() {
+ if (remoting.clientSession) {
+ remoting.clientSession.sendClientMessage('restoreAllWindows', '');
+ }
+ };
+ chrome.app.window.current().onRestored.addListener(restoreHostWindows);
+
+ remoting.windowShape.updateClientWindowShape();
+
+ // Initialize the context menus.
+ if (remoting.platformIsChromeOS()) {
+ var adapter = new remoting.ContextMenuChrome();
+ } else {
+ var adapter = new remoting.ContextMenuDom(
+ document.getElementById('context-menu'));
+ }
+ this.contextMenu_ = new remoting.ApplicationContextMenu(adapter);
+ this.keyboardLayoutsMenu_ = new remoting.KeyboardLayoutsMenu(adapter);
+ this.windowActivationMenu_ = new remoting.WindowActivationMenu(adapter);
+
+ /** @type {remoting.AppRemoting} */
+ var that = this;
+
+ /** @param {XMLHttpRequest} xhr */
+ var parseAppHostResponse = function(xhr) {
+ if (xhr.status == 200) {
+ var response = /** @type {remoting.AppRemoting.AppHostResponse} */
+ (base.jsonParseSafe(xhr.responseText));
+ if (response &&
+ response.status &&
+ response.status == 'done' &&
+ response.hostJid &&
+ response.authorizationCode &&
+ response.sharedSecret &&
+ response.host &&
+ response.host.hostId) {
+ var hostJid = response.hostJid;
+ that.contextMenu_.setHostId(response.host.hostId);
+ var host = new remoting.Host;
+ host.hostId = response.host.hostId;
+ host.jabberId = hostJid;
+ host.authorizationCode = response.authorizationCode;
+ host.sharedSecret = response.sharedSecret;
+
+ remoting.setMode(remoting.AppMode.CLIENT_CONNECTING);
+
+ var idleDetector = new remoting.IdleDetector(
+ document.getElementById('idle-dialog'),
+ remoting.disconnect);
+
+ /**
+ * @param {string} tokenUrl Token-issue URL received from the host.
+ * @param {string} hostPublicKey Host public key (DER and Base64
+ * encoded).
+ * @param {string} scope OAuth scope to request the token for.
+ * @param {function(string, string):void} onThirdPartyTokenFetched
+ * Callback.
+ */
+ var fetchThirdPartyToken = function(
+ tokenUrl, hostPublicKey, scope, onThirdPartyTokenFetched) {
+ // Use the authentication tokens returned by the app-remoting server.
+ onThirdPartyTokenFetched(host['authorizationCode'],
+ host['sharedSecret']);
+ };
+
+ connector.connectMe2App(host, fetchThirdPartyToken);
+ } else if (response && response.status == 'pending') {
+ that.handleError(remoting.Error.SERVICE_UNAVAILABLE);
+ }
+ } else {
+ console.error('Invalid "runApplication" response from server.');
+ // TODO(garykac) Start using remoting.Error.fromHttpError once it has
+ // been updated to properly report 'unknown' errors (rather than
+ // reporting them as AUTHENTICATION_FAILED).
+ if (xhr.status == 0) {
+ that.handleError(remoting.Error.NETWORK_FAILURE);
+ } else if (xhr.status == 401) {
+ that.handleError(remoting.Error.AUTHENTICATION_FAILED);
+ } else if (xhr.status == 403) {
+ that.handleError(remoting.Error.NOT_AUTHORIZED);
+ } else if (xhr.status == 502 || xhr.status == 503) {
+ that.handleError(remoting.Error.SERVICE_UNAVAILABLE);
+ } else {
+ that.handleError(remoting.Error.UNEXPECTED);
+ }
+ }
+ };
+
+ /** @param {string} token */
+ var getAppHost = function(token) {
+ var headers = { 'Authorization': 'OAuth ' + token };
+ remoting.xhr.post(
+ that.runApplicationUrl(), parseAppHostResponse, '', headers);
+ };
+
+ /** @param {remoting.Error} error */
+ var onError = function(error) {
+ that.handleError(error);
+ };
+
+ remoting.LoadingWindow.show();
+
+ remoting.identity.callWithToken(getAppHost, onError);
+}
+
+/**
+ * @return {string} The default remap keys for the current platform.
+ */
+remoting.AppRemoting.prototype.getDefaultRemapKeys = function() {
+ // Map Cmd to Ctrl on Mac since hosts typically use Ctrl for keyboard
+ // shortcuts, but we want them to act as natively as possible.
+ if (remoting.platformIsMac()) {
+ return '0x0700e3>0x0700e0,0x0700e7>0x0700e4';
+ }
+ return '';
+};
+
+/**
+ * @return {Array.<string>} A list of |ClientSession.Capability|s required
+ * by this application.
+ */
+remoting.AppRemoting.prototype.getRequiredCapabilities = function() {
+ return [
+ remoting.ClientSession.Capability.SEND_INITIAL_RESOLUTION,
+ remoting.ClientSession.Capability.RATE_LIMIT_RESIZE_REQUESTS,
+ remoting.ClientSession.Capability.VIDEO_RECORDER,
+ remoting.ClientSession.Capability.GOOGLE_DRIVE
+ ];
+};
+
+/**
+ * Called when a new session has been connected.
+ *
+ * @param {remoting.ClientSession} clientSession
+ * @return {void} Nothing.
+ */
+remoting.AppRemoting.prototype.handleConnected = function(clientSession) {
+ remoting.clientSession.sendClientMessage(
+ 'setUserDisplayInfo',
+ JSON.stringify({fullName: remoting.identity.getCachedUserFullName()}));
+
+ // Set up a ping at 10-second intervals to test the connection speed.
+ function ping() {
+ var message = { timestamp: new Date().getTime() };
+ clientSession.sendClientMessage('pingRequest', JSON.stringify(message));
+ };
+ ping();
+ var timerId = window.setInterval(ping, 10 * 1000);
+
+ // Cancel the ping when the connection closes.
+ clientSession.addEventListener(
+ remoting.ClientSession.Events.stateChanged,
+ /** @param {remoting.ClientSession.StateEvent} state */
+ function(state) {
+ if (state.current === remoting.ClientSession.State.CLOSED ||
+ state.current === remoting.ClientSession.State.FAILED) {
+ window.clearInterval(timerId);
+ }
+ });
+};
+
+/**
+ * Called when the current session has been disconnected.
+ *
+ * @return {void} Nothing.
+ */
+remoting.AppRemoting.prototype.handleDisconnected = function() {
+ chrome.app.window.current().close();
+};
+
+/**
+ * Called when the current session's connection has failed.
+ *
+ * @param {remoting.SessionConnector} connector
+ * @param {remoting.Error} error
+ * @return {void} Nothing.
+ */
+remoting.AppRemoting.prototype.handleConnectionFailed = function(
+ connector, error) {
+ this.handleError(error);
+};
+
+/**
+ * Called when the current session has reached the point where the host has
+ * started streaming video frames to the client.
+ *
+ * @return {void} Nothing.
+ */
+remoting.AppRemoting.prototype.handleVideoStreamingStarted = function() {
+ remoting.LoadingWindow.close();
+};
+
+/**
+ * Called when an extension message needs to be handled.
+ *
+ * @param {string} type The type of the extension message.
+ * @param {string} data The payload of the extension message.
+ * @return {boolean} True if the extension message was recognized.
+ */
+remoting.AppRemoting.prototype.handleExtensionMessage = function(type, data) {
+ var request = /** @type {Object} */base.jsonParseSafe(data);
+ if (typeof request != 'object') {
+ return false;
+ }
+
+ switch (type) {
+
+ case 'openURL':
+ // URL requests from the hosted app are untrusted, so disallow anything
+ // other than HTTP or HTTPS.
+ var url = getStringAttr(request, 'url');
+ if (url.indexOf('http:') != 0 && url.indexOf('https:') != 0) {
+ console.error('Bad URL: ' + url);
+ } else {
+ window.open(url);
+ }
+ return true;
+
+ case 'onWindowRemoved':
+ var id = getNumberAttr(request, 'id');
+ this.windowActivationMenu_.remove(id);
+ return true;
+
+ case 'onWindowAdded':
+ var id = getNumberAttr(request, 'id');
+ var title = getStringAttr(request, 'title');
+ this.windowActivationMenu_.add(id, title);
+ return true;
+
+ case 'onAllWindowsMinimized':
+ chrome.app.window.current().minimize();
+ return true;
+
+ case 'setKeyboardLayouts':
+ var supportedLayouts = getArrayAttr(request, 'supportedLayouts');
+ var currentLayout = getStringAttr(request, 'currentLayout');
+ console.log('Current host keyboard layout: ' + currentLayout);
+ console.log('Supported host keyboard layouts: ' + supportedLayouts);
+ this.keyboardLayoutsMenu_.setLayouts(supportedLayouts, currentLayout);
+ return true;
+
+ case 'pingResponse':
+ var then = getNumberAttr(request, 'timestamp');
+ var now = new Date().getTime();
+ this.contextMenu_.updateConnectionRTT(now - then);
+ return true;
+ }
+
+ return false;
+};
+
+/**
+ * Called when an error needs to be displayed to the user.
+ *
+ * @param {remoting.Error} errorTag The error to be localized and displayed.
+ * @return {void} Nothing.
+ */
+remoting.AppRemoting.prototype.handleError = function(errorTag) {
+ console.error('Connection failed: ' + errorTag);
+ remoting.LoadingWindow.close();
+ remoting.MessageWindow.showErrorMessage(
+ chrome.i18n.getMessage(/**i18n-content*/'CONNECTION_FAILED'),
+ chrome.i18n.getMessage(/** @type {string} */ (errorTag)));
+};
+
+/** @return {string} */
+remoting.AppRemoting.prototype.runApplicationUrl = function() {
+ return remoting.settings.APP_REMOTING_API_BASE_URL + '/applications/' +
+ remoting.settings.getAppRemotingApplicationId() + '/run';
+};
diff --git a/remoting/webapp/app_remoting/js/application_context_menu.js b/remoting/webapp/app_remoting/js/application_context_menu.js
new file mode 100644
index 0000000..2e467b9
--- /dev/null
+++ b/remoting/webapp/app_remoting/js/application_context_menu.js
@@ -0,0 +1,113 @@
+// Copyright 2014 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.
+
+/**
+ * @fileoverview
+ * Class representing the application's context menu.
+ */
+
+'use strict';
+
+/** @suppress {duplicate} */
+var remoting = remoting || {};
+
+/**
+ * @param {remoting.ContextMenuAdapter} adapter
+ * @constructor
+ */
+remoting.ApplicationContextMenu = function(adapter) {
+ /**
+ * @type {remoting.ContextMenuAdapter}
+ * @private
+ */
+ this.adapter_ = adapter;
+
+ this.adapter_.create(
+ remoting.ApplicationContextMenu.kSendFeedbackId,
+ l10n.getTranslationOrError(/*i18n-content*/'SEND_FEEDBACK'),
+ false);
+ this.adapter_.create(
+ remoting.ApplicationContextMenu.kShowStatsId,
+ l10n.getTranslationOrError(/*i18n-content*/'SHOW_STATS'),
+ true);
+ this.adapter_.addListener(this.onClicked_.bind(this));
+
+ /**
+ * @type {string}
+ * @private
+ */
+ this.hostId_ = '';
+};
+
+/**
+ * @param {string} hostId
+ */
+remoting.ApplicationContextMenu.prototype.setHostId = function(hostId) {
+ this.hostId_ = hostId;
+}
+
+/**
+ * Add an indication of the connection RTT to the 'Show statistics' menu item.
+ *
+ * @param {number} rttMs The RTT of the connection, in ms.
+ */
+remoting.ApplicationContextMenu.prototype.updateConnectionRTT =
+ function(rttMs) {
+ var rttText =
+ rttMs < 50 ? /*i18n-content*/'CONNECTION_QUALITY_GOOD' :
+ rttMs < 100 ? /*i18n-content*/'CONNECTION_QUALITY_FAIR' :
+ /*i18n-content*/'CONNECTION_QUALITY_POOR';
+ rttText = l10n.getTranslationOrError(rttText);
+ this.adapter_.updateTitle(
+ remoting.ApplicationContextMenu.kShowStatsId,
+ l10n.getTranslationOrError(/*i18n-content*/'SHOW_STATS_WITH_RTT',
+ rttText));
+};
+
+/** @param {OnClickData} info */
+remoting.ApplicationContextMenu.prototype.onClicked_ = function(info) {
+ switch (info.menuItemId) {
+
+ case remoting.ApplicationContextMenu.kSendFeedbackId:
+ var windowAttributes = {
+ bounds: {
+ width: 400,
+ height: 100
+ },
+ resizable: false
+ };
+
+ /** @type {remoting.ApplicationContextMenu} */
+ var that = this;
+
+ /** @param {AppWindow} consentWindow */
+ var onCreate = function(consentWindow) {
+ var onLoad = function() {
+ var message = {
+ method: 'init',
+ hostId: that.hostId_,
+ connectionStats: JSON.stringify(remoting.stats.mostRecent())
+ };
+ consentWindow.contentWindow.postMessage(message, '*');
+ };
+ consentWindow.contentWindow.addEventListener('load', onLoad, false);
+ };
+ chrome.app.window.create(
+ 'feedback_consent.html', windowAttributes, onCreate);
+ break;
+
+ case remoting.ApplicationContextMenu.kShowStatsId:
+ if (remoting.stats) {
+ remoting.stats.show(info.checked);
+ }
+ break;
+ }
+};
+
+
+/** @type {string} */
+remoting.ApplicationContextMenu.kSendFeedbackId = 'send-feedback';
+
+/** @type {string} */
+remoting.ApplicationContextMenu.kShowStatsId = 'show-stats';
diff --git a/remoting/webapp/app_remoting/js/ar_background.js b/remoting/webapp/app_remoting/js/ar_background.js
new file mode 100644
index 0000000..8106517
--- /dev/null
+++ b/remoting/webapp/app_remoting/js/ar_background.js
@@ -0,0 +1,73 @@
+// Copyright 2014 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.
+
+/** @type {AppWindow} */
+var mainWindow = null;
+
+/**
+ * The main window cannot delete its context menu entries on close because it
+ * is being torn down at that point and doesn't have access to the necessary
+ * APIs. Instead, it notifies the background page of the entries it creates,
+ * and the background pages deletes them when the window is closed.
+ *
+ * @type {!Object}
+ */
+var contextMenuIds = {};
+
+/** @param {LaunchData=} opt_launchData */
+function createWindow(opt_launchData) {
+ // If there is already a window, give it focus.
+ if (mainWindow) {
+ mainWindow.focus();
+ return;
+ }
+
+ var typed_screen = /** @type {{width: number, height: number}} */ (screen);
+
+ var windowAttributes = {
+ resizable: false,
+ frame: 'none',
+ bounds: {
+ width: typed_screen.width,
+ height: typed_screen.height
+ }
+ };
+
+ function onClosed() {
+ mainWindow = null;
+ var ids = Object.keys(contextMenuIds);
+ for (var i = 0; i < ids.length; ++i) {
+ chrome.contextMenus.remove(ids[i]);
+ }
+ contextMenuIds = {};
+ };
+
+ /** @param {AppWindow} appWindow */
+ function onCreate(appWindow) {
+ // Set the global window.
+ mainWindow = appWindow;
+
+ // Clean up the windows sub-menu when the application quits.
+ appWindow.onClosed.addListener(onClosed);
+ };
+
+ chrome.app.window.create('main.html', windowAttributes, onCreate);
+};
+
+/** @param {Event} event */
+function onWindowMessage(event) {
+ var method = /** @type {string} */ (event.data['method']);
+ var id = /** @type {string} */ (event.data['id']);
+ switch (method) {
+ case 'addContextMenuId':
+ contextMenuIds[id] = true;
+ break;
+ case 'removeContextMenuId':
+ delete contextMenuIds[id];
+ break;
+ }
+};
+
+chrome.app.runtime.onLaunched.addListener(createWindow);
+window.addEventListener('message', onWindowMessage, false);
diff --git a/remoting/webapp/app_remoting/js/ar_main.js b/remoting/webapp/app_remoting/js/ar_main.js
new file mode 100644
index 0000000..b15cdff
--- /dev/null
+++ b/remoting/webapp/app_remoting/js/ar_main.js
@@ -0,0 +1,19 @@
+// Copyright 2014 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.
+
+'use strict';
+
+/** @suppress {duplicate} */
+var remoting = remoting || {};
+
+/**
+ * Entry point ('load' handler) for App Remoting webapp.
+ */
+remoting.startAppRemoting = function() {
+ remoting.app = new remoting.Application();
+ var app_remoting = new remoting.AppRemoting(remoting.app);
+ remoting.app.start();
+};
+
+window.addEventListener('load', remoting.startAppRemoting, false);
diff --git a/remoting/webapp/app_remoting/js/context_menu_adapter.js b/remoting/webapp/app_remoting/js/context_menu_adapter.js
new file mode 100644
index 0000000..4f77286
--- /dev/null
+++ b/remoting/webapp/app_remoting/js/context_menu_adapter.js
@@ -0,0 +1,56 @@
+// Copyright 2014 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.
+
+/**
+ * @fileoverview
+ * Wrapper interface for chrome.contextMenus.
+ */
+
+'use strict';
+
+/** @suppress {duplicate} */
+var remoting = remoting || {};
+
+/** @interface */
+remoting.ContextMenuAdapter = function() {
+};
+
+/**
+ * @param {string} id An identifier for the menu entry.
+ * @param {string} title The text to display in the menu.
+ * @param {boolean} isCheckable True if the state of this menu entry should
+ * have a check-box and manage its toggle state automatically. Note that
+ * checkable menu entries always start off unchecked; use updateCheckState
+ * to programmatically change the state.
+ * @param {string=} opt_parentId The id of the parent menu item for submenus.
+ */
+remoting.ContextMenuAdapter.prototype.create = function(
+ id, title, isCheckable, opt_parentId) {
+};
+
+/**
+ * @param {string} id
+ * @param {string} title
+ */
+remoting.ContextMenuAdapter.prototype.updateTitle = function(id, title) {
+};
+
+/**
+ * @param {string} id
+ * @param {boolean} checked
+ */
+remoting.ContextMenuAdapter.prototype.updateCheckState = function(id, checked) {
+};
+
+/**
+ * @param {string} id
+ */
+remoting.ContextMenuAdapter.prototype.remove = function(id) {
+};
+
+/**
+ * @param {function(OnClickData):void} listener
+ */
+remoting.ContextMenuAdapter.prototype.addListener = function(listener) {
+};
diff --git a/remoting/webapp/app_remoting/js/context_menu_chrome.js b/remoting/webapp/app_remoting/js/context_menu_chrome.js
new file mode 100644
index 0000000..c468386
--- /dev/null
+++ b/remoting/webapp/app_remoting/js/context_menu_chrome.js
@@ -0,0 +1,92 @@
+// Copyright 2014 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.
+
+/**
+ * @fileoverview
+ * remoting.ContextMenuAdapter implementation backed by chrome.contextMenus.
+ */
+
+'use strict';
+
+/** @suppress {duplicate} */
+var remoting = remoting || {};
+
+/**
+ * @constructor
+ * @implements {remoting.ContextMenuAdapter}
+ */
+remoting.ContextMenuChrome = function() {
+};
+
+/**
+ * @param {string} id An identifier for the menu entry.
+ * @param {string} title The text to display in the menu.
+ * @param {boolean} isCheckable True if the state of this menu entry should
+ * have a check-box and manage its toggle state automatically.
+ * @param {string=} opt_parentId The id of the parent menu item for submenus.
+ */
+remoting.ContextMenuChrome.prototype.create = function(
+ id, title, isCheckable, opt_parentId) {
+ if (!opt_parentId) {
+ var message = {
+ method: 'addContextMenuId',
+ id: id
+ };
+ chrome.runtime.getBackgroundPage(this.postMessage_.bind(this, message));
+ }
+ var params = {
+ id: id,
+ contexts: ['launcher'],
+ title: title,
+ parentId: opt_parentId
+ };
+ if (isCheckable) {
+ params.type = 'checkbox';
+ }
+ chrome.contextMenus.create(params);
+};
+
+/**
+ * @param {string} id
+ * @param {string} title
+ */
+remoting.ContextMenuChrome.prototype.updateTitle = function(id, title) {
+ chrome.contextMenus.update(id, {title: title});
+};
+
+/**
+ * @param {string} id
+ * @param {boolean} checked
+ */
+remoting.ContextMenuChrome.prototype.updateCheckState = function(id, checked) {
+ chrome.contextMenus.update(id, {checked: checked});
+};
+
+/**
+ * @param {string} id
+ */
+remoting.ContextMenuChrome.prototype.remove = function(id) {
+ chrome.contextMenus.remove(id);
+ var message = {
+ method: 'removeContextMenuId',
+ id: id
+ };
+ chrome.runtime.getBackgroundPage(this.postMessage_.bind(this, message));
+};
+
+/**
+ * @param {function(OnClickData):void} listener
+ */
+remoting.ContextMenuChrome.prototype.addListener = function(listener) {
+ chrome.contextMenus.onClicked.addListener(listener);
+};
+
+/**
+ * @param {*} message
+ * @param {Window} backgroundPage
+ */
+remoting.ContextMenuChrome.prototype.postMessage_ = function(
+ message, backgroundPage) {
+ backgroundPage.postMessage(message, '*');
+}; \ No newline at end of file
diff --git a/remoting/webapp/app_remoting/js/context_menu_dom.js b/remoting/webapp/app_remoting/js/context_menu_dom.js
new file mode 100644
index 0000000..c81482e
--- /dev/null
+++ b/remoting/webapp/app_remoting/js/context_menu_dom.js
@@ -0,0 +1,328 @@
+// Copyright 2014 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.
+
+/**
+ * @fileoverview
+ * Provide an alternative location for the application's context menu items
+ * on platforms that don't provide it.
+ *
+ * To mimic the behaviour of an OS-provided context menu, the menu is dismissed
+ * in three situations:
+ *
+ * 1. When the window loses focus (i.e, the user has clicked on another window
+ * or on the desktop).
+ * 2. When the user selects an option from the menu.
+ * 3. When the user clicks on another part of the same window; this is achieved
+ * using an invisible screen element behind the menu, but in front of all
+ * other DOM.
+ *
+ * TODO(jamiewalch): Fold this functionality into remoting.MenuButton.
+ */
+'use strict';
+
+/** @suppress {duplicate} */
+var remoting = remoting || {};
+
+/**
+ * @constructor
+ * @implements {remoting.WindowShape.ClientUI}
+ * @implements {remoting.ContextMenuAdapter}
+ * @param {HTMLElement} root The root of the context menu DOM.
+ */
+remoting.ContextMenuDom = function(root) {
+ /**
+ * @type {HTMLElement}
+ * @private
+ */
+ this.root_ = root;
+ /**
+ * @type {HTMLElement}
+ * @private
+ */
+ this.stub_ = /** @type {HTMLElement} */
+ this.root_.querySelector('.context-menu-stub');
+ /**
+ * @type {HTMLElement}
+ * @private
+ */
+ this.icon_ = /** @type {HTMLElement} */
+ this.root_.querySelector('.context-menu-icon');
+ /**
+ * @type {HTMLElement}
+ * @private
+ */
+ this.screen_ = /** @type {HTMLElement} */
+ this.root_.querySelector('.context-menu-screen');
+ /**
+ * @type {HTMLElement}
+ * @private
+ */
+ this.menu_ = /** @type {HTMLElement} */ this.root_.querySelector('ul');
+ /**
+ * @type {number}
+ * @private
+ */
+ this.bottom_ = 8;
+ /**
+ * @type {base.EventSource}
+ * @private
+ */
+ this.eventSource_ = new base.EventSource();
+ /**
+ * @type {string}
+ * @private
+ */
+ this.eventName_ = '_click';
+ /**
+ * Since the same element is used to lock the icon open and to drag it, we
+ * must keep track of drag events so that the corresponding click event can
+ * be ignored.
+ *
+ * @type {boolean}
+ * @private
+ */
+ this.stubDragged_ = false;
+
+ /*
+ * @private
+ */
+ this.dragAndDrop_ = new remoting.DragAndDrop(
+ this.stub_, this.onDragUpdate_.bind(this));
+
+ this.eventSource_.defineEvents([this.eventName_]);
+ this.root_.addEventListener(
+ 'transitionend', this.onTransitionEnd_.bind(this), false);
+ this.stub_.addEventListener('click', this.onStubClick_.bind(this), false);
+ this.icon_.addEventListener('click', this.onIconClick_.bind(this), false);
+ this.screen_.addEventListener('click', this.onIconClick_.bind(this), false);
+
+ this.root_.hidden = false;
+ this.root_.style.bottom = this.bottom_ + 'px';
+ remoting.windowShape.addCallback(this);
+};
+
+/**
+ * @param {Array.<{left: number, top: number, width: number, height: number}>}
+ * rects List of rectangles.
+ */
+remoting.ContextMenuDom.prototype.addToRegion = function(rects) {
+ var rect = /** @type {ClientRect} */ (this.root_.getBoundingClientRect());
+ // Clip the menu position to the main window in case the screen size has
+ // changed or a recent drag event tried to move it out of bounds.
+ if (rect.top < 0) {
+ this.bottom_ += rect.top;
+ this.root_.style.bottom = this.bottom_ + 'px';
+ rect = this.root_.getBoundingClientRect();
+ }
+
+ rects.push(rect);
+ if (this.root_.classList.contains('menu-opened')) {
+ var menuRect =
+ /** @type {ClientRect} */ (this.menu_.getBoundingClientRect());
+ rects.push(menuRect);
+ }
+};
+
+/**
+ * @param {string} id An identifier for the menu entry.
+ * @param {string} title The text to display in the menu.
+ * @param {boolean} isCheckable True if the state of this menu entry should
+ * have a check-box and manage its toggle state automatically. Note that
+ * checkable menu entries always start off unchecked.
+ * @param {string=} opt_parentId The id of the parent menu item for submenus.
+ */
+remoting.ContextMenuDom.prototype.create = function(
+ id, title, isCheckable, opt_parentId) {
+ var menuEntry = /** @type {HTMLElement} */ (document.createElement('li'));
+ menuEntry.innerText = title;
+ menuEntry.setAttribute('data-id', id);
+ if (isCheckable) {
+ menuEntry.setAttribute('data-checkable', true);
+ }
+ menuEntry.addEventListener('click', this.onClick_.bind(this), false);
+ /** @type {Node} */
+ var insertBefore = null;
+ if (opt_parentId) {
+ var parent = /** @type {HTMLElement} */
+ (this.menu_.querySelector('[data-id="' + opt_parentId + '"]'));
+ base.debug.assert(parent != null);
+ base.debug.assert(!parent.classList.contains('menu-group-item'));
+ parent.classList.add('menu-group-header');
+ menuEntry.classList.add('menu-group-item');
+ insertBefore = this.getInsertionPointForParent(
+ /** @type {string} */(opt_parentId));
+ }
+ this.menu_.insertBefore(menuEntry, insertBefore);
+};
+
+/**
+ * @param {string} id
+ * @param {string} title
+ */
+remoting.ContextMenuDom.prototype.updateTitle = function(id, title) {
+ var node = this.menu_.querySelector('[data-id="' + id + '"]');
+ if (node) {
+ node.innerText = title;
+ }
+};
+
+/**
+ * @param {string} id
+ * @param {boolean} checked
+ */
+remoting.ContextMenuDom.prototype.updateCheckState = function(id, checked) {
+ var node = /** @type {HTMLElement} */
+ (this.menu_.querySelector('[data-id="' + id + '"]'));
+ if (node) {
+ if (checked) {
+ node.classList.add('selected');
+ } else {
+ node.classList.remove('selected');
+ }
+ }
+};
+
+/**
+ * @param {string} id
+ */
+remoting.ContextMenuDom.prototype.remove = function(id) {
+ var node = this.menu_.querySelector('[data-id="' + id + '"]');
+ if (node) {
+ this.menu_.removeChild(node);
+ }
+};
+
+/**
+ * @param {function(OnClickData):void} listener
+ */
+remoting.ContextMenuDom.prototype.addListener = function(listener) {
+ this.eventSource_.addEventListener(this.eventName_, listener);
+};
+
+/**
+ * @param {Event} event
+ * @private
+ */
+remoting.ContextMenuDom.prototype.onClick_ = function(event) {
+ var element = /** @type {HTMLElement} */ (event.target);
+ if (element.getAttribute('data-checkable')) {
+ element.classList.toggle('selected')
+ }
+ var clickData = {
+ menuItemId: element.getAttribute('data-id'),
+ checked: element.classList.contains('selected')
+ };
+ this.eventSource_.raiseEvent(this.eventName_, clickData);
+ this.onIconClick_();
+};
+
+/**
+ * Get the insertion point for the specified sub-menu. This is the menu item
+ * immediately following the last child of that menu group, or null if there
+ * are no menu items after that group.
+ *
+ * @param {string} parentId
+ * @return {Node?}
+ */
+remoting.ContextMenuDom.prototype.getInsertionPointForParent = function(
+ parentId) {
+ var parentNode = this.menu_.querySelector('[data-id="' + parentId + '"]');
+ base.debug.assert(parentNode != null);
+ var childNode = /** @type {HTMLElement} */ (parentNode.nextSibling);
+ while (childNode != null && childNode.classList.contains('menu-group-item')) {
+ childNode = childNode.nextSibling;
+ }
+ return childNode;
+};
+
+/**
+ * Called when the CSS show/hide transition completes. Since this changes the
+ * visible dimensions of the context menu, the visible region of the window
+ * needs to be recomputed.
+ *
+ * @private
+ */
+remoting.ContextMenuDom.prototype.onTransitionEnd_ = function() {
+ remoting.windowShape.updateClientWindowShape();
+};
+
+/**
+ * Toggle the visibility of the context menu icon.
+ *
+ * @private
+ */
+remoting.ContextMenuDom.prototype.onStubClick_ = function() {
+ if (this.stubDragged_) {
+ this.stubDragged_ = false;
+ return;
+ }
+ this.root_.classList.toggle('opened');
+};
+
+/**
+ * Toggle the visibility of the context menu.
+ *
+ * @private
+ */
+remoting.ContextMenuDom.prototype.onIconClick_ = function() {
+ this.showMenu_(!this.menu_.classList.contains('opened'));
+};
+
+/**
+ * Explicitly show or hide the context menu.
+ *
+ * @param {boolean} show True to show the menu; false to hide it.
+ * @private
+ */
+remoting.ContextMenuDom.prototype.showMenu_ = function(show) {
+ if (show) {
+ // Ensure that the menu doesn't extend off the top or bottom of the
+ // screen by aligning it to the top or bottom of the icon, depending
+ // on the latter's vertical position.
+ var menuRect =
+ /** @type {ClientRect} */ (this.menu_.getBoundingClientRect());
+ if (menuRect.bottom > window.innerHeight) {
+ this.menu_.classList.add('menu-align-bottom');
+ } else {
+ this.menu_.classList.remove('menu-align-bottom');
+ }
+
+ /** @type {remoting.ContextMenuDom} */
+ var that = this;
+ var onBlur = function() {
+ that.showMenu_(false);
+ window.removeEventListener('blur', onBlur, false);
+ };
+ window.addEventListener('blur', onBlur, false);
+
+ // Show the menu and prevent the icon from auto-hiding on mouse-out.
+ this.menu_.classList.add('opened');
+ this.root_.classList.add('menu-opened');
+
+ } else { // if (!show)
+ this.menu_.classList.remove('opened');
+ this.root_.classList.remove('menu-opened');
+ }
+
+ this.screen_.hidden = !show;
+ remoting.windowShape.updateClientWindowShape();
+};
+
+/**
+ * @param {number} deltaX
+ * @param {number} deltaY
+ * @private
+ */
+remoting.ContextMenuDom.prototype.onDragUpdate_ = function(deltaX, deltaY) {
+ this.stubDragged_ = true;
+ this.bottom_ -= deltaY;
+ this.root_.style.bottom = this.bottom_ + 'px';
+ // Deferring the window shape update until the DOM update has completed
+ // helps keep the position of the context menu consistent with the window
+ // shape (though it's still not perfect).
+ window.requestAnimationFrame(
+ function() {
+ remoting.windowShape.updateClientWindowShape();
+ });
+};
diff --git a/remoting/webapp/app_remoting/js/drag_and_drop.js b/remoting/webapp/app_remoting/js/drag_and_drop.js
new file mode 100644
index 0000000..2f01db0
--- /dev/null
+++ b/remoting/webapp/app_remoting/js/drag_and_drop.js
@@ -0,0 +1,142 @@
+// Copyright 2014 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.
+
+/**
+ * @fileoverview
+ * Provide support for drag-and-drop operations in shaped windows. The
+ * standard API doesn't work because no "dragover" events are generated
+ * if the mouse moves outside the window region.
+ */
+'use strict';
+
+/** @suppress {duplicate} */
+var remoting = remoting || {};
+
+/**
+ * @constructor
+ * @param {Element} element The element to register for drag and drop.
+ * @param {function(number, number):void} dragUpdate Callback to receive the
+ * X and Y deltas as the element is dragged.
+ * @param {function():void=} opt_dragStart Initiation callback.
+ * @param {function():void=} opt_dragEnd Completion callback.
+ */
+remoting.DragAndDrop = function(element, dragUpdate,
+ opt_dragStart, opt_dragEnd) {
+ /**
+ * @private
+ */
+ this.element_ = element;
+
+ /**
+ * @private
+ */
+ this.dragUpdate_ = dragUpdate;
+
+ /**
+ * @private
+ */
+ this.dragStart_ = opt_dragStart;
+
+ /**
+ * @private
+ */
+ this.dragEnd_ = opt_dragEnd;
+
+ /**
+ * @type {number}
+ * @private
+ */
+ this.previousDeltaX_ = 0;
+
+ /**
+ * @type {number}
+ * @private
+ */
+ this.previousDeltaY_ = 0;
+
+ /**
+ * @type {boolean}
+ */
+ this.seenNonZeroDelta_ = false;
+
+ /**
+ * @type {function():void}
+ * @private
+ */
+ this.callOnMouseUp_ = this.onMouseUp_.bind(this);
+
+ /**
+ * @type {function():void}
+ * @private
+ */
+ this.callOnMouseMove_ = this.onMouseMove_.bind(this);
+
+ element.addEventListener('mousedown', this.onMouseDown_.bind(this), false);
+};
+
+/**
+ * @param {Event} event
+ */
+remoting.DragAndDrop.prototype.onMouseDown_ = function(event) {
+ if (event.button != 0) {
+ return;
+ }
+ this.previousDeltaX_ = 0;
+ this.previousDeltaY_ = 0;
+ this.seenNonZeroDelta_ = false;
+ this.element_.addEventListener('mousemove', this.callOnMouseMove_, false);
+ this.element_.addEventListener('mouseup', this.callOnMouseUp_, false);
+ this.element_.requestPointerLock();
+ if (this.dragStart_) {
+ this.dragStart_();
+ }
+};
+
+/**
+ * TODO(jamiewalch): Remove the workarounds in this method once the pointer-lock
+ * API is fixed (crbug.com/419562).
+ *
+ * @param {Event} event
+ */
+remoting.DragAndDrop.prototype.onMouseMove_ = function(event) {
+ // Ignore the first non-zero delta. A click event will generate a bogus
+ // mousemove event, even if the mouse doesn't move.
+ if (!this.seenNonZeroDelta_ &&
+ (event.movementX != 0 || event.movementY != 0)) {
+ this.seenNonZeroDelta_ = true;
+ }
+
+ /**
+ * The mouse lock API is buggy when used with shaped windows, and occasionally
+ * generates single, large deltas that must be filtered out.
+ *
+ * @param {number} previous
+ * @param {number} current
+ * @return {number}
+ */
+ var adjustDelta = function(previous, current) {
+ var THRESHOLD = 100; // Based on observed values.
+ if (Math.abs(previous < THRESHOLD) && Math.abs(current) >= THRESHOLD) {
+ return 0;
+ }
+ return current;
+ };
+ this.previousDeltaX_ = adjustDelta(this.previousDeltaX_, event.movementX);
+ this.previousDeltaY_ = adjustDelta(this.previousDeltaY_, event.movementY);
+ if (this.previousDeltaX_ != 0 || this.previousDeltaY_ != 0) {
+ this.dragUpdate_(this.previousDeltaX_, this.previousDeltaY_);
+ }
+};
+
+/**
+ * @param {Event} event
+ */
+remoting.DragAndDrop.prototype.onMouseUp_ = function(event) {
+ this.element_.removeEventListener('mousemove', this.callOnMouseMove_, false);
+ this.element_.removeEventListener('mouseup', this.callOnMouseUp_, false);
+ document.exitPointerLock();
+ if (this.dragEnd_) {
+ this.dragEnd_();
+ }
+}; \ No newline at end of file
diff --git a/remoting/webapp/app_remoting/js/feedback_consent.js b/remoting/webapp/app_remoting/js/feedback_consent.js
new file mode 100644
index 0000000..7bc7d3d
--- /dev/null
+++ b/remoting/webapp/app_remoting/js/feedback_consent.js
@@ -0,0 +1,229 @@
+// Copyright 2014 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.
+
+'use strict';
+
+/**
+ * @type {string} The host id corresponding to the user's VM. The @pending
+ * place-holder instructs the Orchestrator to abandon any pending host,
+ * and is used if no host id is provided by the main window.
+ */
+var hostId = '@pending';
+
+/**
+ * @type {string} The network stats at the time the feedback consent dialog
+ * was shown.
+ */
+var connectionStats = null;
+
+/**
+ * @type {string} "no" => user did not request a VM reset; "yes" => VM was
+ * successfully reset; "failed" => user requested a reset, but it failed.
+ */
+var abandonHost = 'no';
+
+/**
+ * @type {Window} The main application window.
+ */
+var applicationWindow = null;
+
+/**
+ * @type {string} An unique identifier that links the feedback post with the
+ * logs uploaded by the host.
+ */
+var crashServiceReportId = '';
+
+/**
+ * @param {string} email
+ * @param {string} realName
+ */
+function onUserInfo(email, realName) {
+ /** @type {number} Identifies this product to Google Feedback. **/
+ var productId = 93407;
+
+ /** @type {string} The base URL for Google Feedback. */
+ var url = 'https://www.google.com/tools/feedback/survey/xhtml';
+
+ /** @type {string} The feedback 'bucket', used for clustering. */
+ var bucket = 'feedback';
+
+ /** @type {string} The user's locale, used to localize the feedback page. */
+ var locale = chrome.i18n.getMessage('@@ui_locale');
+
+ window.open(url +
+ '?productId=' + productId +
+ '&bucket=' + escape(bucket) +
+ '&hl=' + escape(locale) +
+ '&psd_email=' + escape(email) +
+ '&psd_hostId=' + escape(hostId) +
+ '&psd_abandonHost=' + escape(abandonHost) +
+ '&psd_crashServiceReportId=' + escape(crashServiceReportId) +
+ '&psd_connectionStats=' + escape(connectionStats));
+ window.close();
+
+ // If the VM was successfully abandoned, close the application.
+ if (abandonHost == 'yes') {
+ applicationWindow.close();
+ }
+};
+
+/**
+ * @param {boolean} waiting
+ */
+function setWaiting(waiting) {
+ var ok = document.getElementById('feedback-consent-ok');
+ var cancel = document.getElementById('feedback-consent-cancel');
+ var abandon = document.getElementById('abandon-host');
+ var working = document.getElementById('working');
+ ok.disabled = waiting;
+ cancel.disabled = waiting;
+ abandon.disabled = waiting;
+ working.hidden = !waiting;
+}
+
+function showError() {
+ setWaiting(false);
+ var error = document.getElementById('abandon-failed');
+ var abandon = document.getElementById('abandon-host');
+ var logs = document.getElementById('include-logs');
+ var formBody = document.getElementById('form-body');
+ error.hidden = false;
+ abandon.checked = false;
+ logs.checked = false;
+ abandonHost = 'failed';
+ crashServiceReportId = '';
+ formBody.hidden = true;
+ resizeWindow();
+}
+
+/**
+ * @return {string} A random string ID.
+ */
+function generateId() {
+ var idArray = new Uint8Array(20);
+ crypto.getRandomValues(idArray);
+ return btoa(String.fromCharCode.apply(null, idArray));
+}
+
+/**
+ * @param {string} token
+ */
+function onToken(token) {
+ var getUserInfo = function() {
+ remoting.OAuth2Api.getUserInfo(
+ onUserInfo, onUserInfo.bind(null, 'unknown', 'unknown'), token);
+ };
+ if (!token) {
+ onUserInfo('unknown', 'unknown');
+ } else {
+ if (abandonHost == 'yes') {
+ var body = {
+ 'abandonHost': 'true',
+ 'crashServiceReportId': crashServiceReportId
+ };
+ var headers = {
+ 'Authorization': 'OAuth ' + token,
+ 'Content-type': 'application/json'
+ };
+ var uri = remoting.settings.APP_REMOTING_API_BASE_URL +
+ '/applications/' + remoting.settings.getAppRemotingApplicationId() +
+ '/hosts/' + hostId +
+ '/reportIssue';
+ /** @param {XMLHttpRequest} xhr */
+ var onDone = function(xhr) {
+ if (xhr.status >= 200 && xhr.status < 300) {
+ getUserInfo();
+ } else {
+ showError();
+ }
+ };
+ remoting.xhr.post(uri, onDone, JSON.stringify(body), headers);
+ } else {
+ getUserInfo();
+ }
+ }
+}
+
+function onOk() {
+ setWaiting(true);
+ var abandon = /** @type {HTMLInputElement} */
+ (document.getElementById('abandon-host'));
+ if (abandon.checked) {
+ abandonHost = 'yes';
+ }
+ chrome.identity.getAuthToken({ 'interactive': false }, onToken);
+}
+
+function onCancel() {
+ window.close();
+}
+
+function onToggleAbandon() {
+ var abandon = document.getElementById('abandon-host');
+ var includeLogs = document.getElementById('include-logs');
+ var includeLogsLabel = document.getElementById('include-logs-label');
+ var learnMoreLink = document.getElementById('learn-more');
+ includeLogs.disabled = !abandon.checked;
+ if (abandon.checked) {
+ includeLogsLabel.classList.remove('disabled');
+ learnMoreLink.classList.remove('disabled');
+ } else {
+ includeLogsLabel.classList.add('disabled');
+ learnMoreLink.classList.add('disabled');
+ }
+}
+
+function onToggleLogs() {
+ var includeLogs = document.getElementById('include-logs');
+ if (includeLogs.checked) {
+ crashServiceReportId = generateId();
+ } else {
+ crashServiceReportId = '';
+ }
+}
+
+function onLearnMore(event) {
+ event.preventDefault(); // Clicking the link should not tick the checkbox.
+ var learnMoreLink = document.getElementById('learn-more');
+ var learnMoreInfobox = document.getElementById('privacy-info');
+ learnMoreLink.hidden = true;
+ learnMoreInfobox.hidden = false;
+ resizeWindow();
+}
+
+function resizeWindow() {
+ var borderY = window.outerHeight - window.innerHeight;
+ window.resizeTo(window.outerWidth, document.body.clientHeight + borderY);
+}
+
+function onLoad() {
+ window.addEventListener('message', onWindowMessage, false);
+ remoting.settings = new remoting.Settings();
+ l10n.localize();
+ var ok = document.getElementById('feedback-consent-ok');
+ var cancel = document.getElementById('feedback-consent-cancel');
+ var abandon = document.getElementById('abandon-host-label');
+ var includeLogs = document.getElementById('include-logs-label');
+ var learnMoreLink = document.getElementById('learn-more');
+ ok.addEventListener('click', onOk, false);
+ cancel.addEventListener('click', onCancel, false);
+ abandon.addEventListener('click', onToggleAbandon, false);
+ includeLogs.addEventListener('click', onToggleLogs, false);
+ learnMoreLink.addEventListener('click', onLearnMore, false);
+ resizeWindow();
+}
+
+/** @param {Event} event */
+function onWindowMessage(event) {
+ applicationWindow = event.source;
+ var method = /** @type {string} */ (event.data['method']);
+ if (method == 'init') {
+ if (event.data['hostId']) {
+ hostId = /** @type {string} */ (event.data['hostId']);
+ }
+ connectionStats = /** @type {string} */ (event.data['connectionStats']);
+ }
+};
+
+window.addEventListener('load', onLoad, false);
diff --git a/remoting/webapp/app_remoting/js/idle_detector.js b/remoting/webapp/app_remoting/js/idle_detector.js
new file mode 100644
index 0000000..58b982d
--- /dev/null
+++ b/remoting/webapp/app_remoting/js/idle_detector.js
@@ -0,0 +1,166 @@
+// Copyright 2014 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.
+
+/**
+ * @fileoverview
+ * Class for detecting when the application is idle. Note that chrome.idle is
+ * not suitable for this purpose because it detects when the computer is idle,
+ * and we'd like to close the application and free up VM resources even if the
+ * user has been using another application for a long time.
+ *
+ * There are two idle timeouts. The first controls the visibility of the idle
+ * timeout warning dialog and is reset on mouse input; when it expires, the
+ * idle warning dialog is displayed. The second controls the length of time
+ * for which the idle warning dialog is displayed; when it expires, the ctor
+ * callback is invoked, which it is assumed will exit the application--no
+ * further idle detection is done.
+ */
+
+'use strict';
+
+/** @suppress {duplicate} */
+var remoting = remoting || {};
+
+/**
+ * @param {HTMLElement} idleWarning The idle warning dialog.
+ * @param {function():void} callback Called when the idle warning dialog has
+ * timed out or the user has explicitly indicated that they are no longer
+ * using the session.
+ * @constructor
+ * @implements {remoting.WindowShape.ClientUI}
+ */
+remoting.IdleDetector = function(idleWarning, callback) {
+ /** @private */
+ this.idleWarning_ = idleWarning;
+
+ /** @private */
+ this.callback_ = callback;
+
+ /**
+ * @type {number?} The id of the running timer, or null if no timer is
+ * running.
+ * @private
+ */
+ this.timerId_ = null;
+
+ /**
+ * @type {?function():void}
+ * @private
+ */
+ this.resetTimeoutRef_ = null;
+
+ var manifest = chrome.runtime.getManifest();
+ var message = this.idleWarning_.querySelector('.idle-warning-message');
+ l10n.localizeElement(message, manifest.name);
+
+ var cont = this.idleWarning_.querySelector('.idle-dialog-continue');
+ cont.addEventListener('click', this.onContinue_.bind(this), false);
+ var quit = this.idleWarning_.querySelector('.idle-dialog-disconnect');
+ quit.addEventListener('click', this.onDisconnect_.bind(this), false);
+
+ remoting.windowShape.addCallback(this);
+ this.resetTimeout_();
+};
+
+/**
+ * @param {boolean} register True to register the callbacks; false to remove
+ * them.
+ * @private
+ */
+remoting.IdleDetector.prototype.registerInputDetectionCallbacks_ =
+ function(register) {
+ var events = [ 'mousemove', 'mousedown', 'mouseup', 'click',
+ 'keyup', 'keydown', 'keypress' ];
+ if (register) {
+ base.debug.assert(this.resetTimeoutRef_ == null);
+ this.resetTimeoutRef_ = this.resetTimeout_.bind(this);
+ for (var i = 0; i < events.length; ++i) {
+ document.body.addEventListener(events[i], this.resetTimeoutRef_, true);
+ }
+ } else {
+ base.debug.assert(this.resetTimeoutRef_ != null);
+ for (var i = 0; i < events.length; ++i) {
+ document.body.removeEventListener(events[i], this.resetTimeoutRef_, true);
+ }
+ this.resetTimeoutRef_ = null;
+ }
+}
+
+/**
+ * @private
+ */
+remoting.IdleDetector.prototype.resetTimeout_ = function() {
+ if (this.timerId_ !== null) {
+ window.clearTimeout(this.timerId_);
+ }
+ if (this.resetTimeoutRef_ == null) {
+ this.registerInputDetectionCallbacks_(true);
+ }
+ this.timerId_ = window.setTimeout(this.onIdleTimeout_.bind(this),
+ remoting.IdleDetector.kIdleTimeoutMs);
+};
+
+/**
+ * @private
+ */
+remoting.IdleDetector.prototype.onIdleTimeout_ = function() {
+ this.registerInputDetectionCallbacks_(false);
+ this.showIdleWarning_(true);
+ this.timerId_ = window.setTimeout(this.onDialogTimeout_.bind(this),
+ remoting.IdleDetector.kDialogTimeoutMs);
+};
+
+/**
+ * @private
+ */
+remoting.IdleDetector.prototype.onDialogTimeout_ = function() {
+ this.timerId_ = null;
+ this.showIdleWarning_(false);
+ this.callback_();
+};
+
+/**
+ * @private
+ */
+remoting.IdleDetector.prototype.onContinue_ = function() {
+ this.showIdleWarning_(false);
+ this.resetTimeout_();
+};
+
+/**
+ * @private
+ */
+remoting.IdleDetector.prototype.onDisconnect_ = function() {
+ if (this.timerId_ !== null) {
+ window.clearTimeout(this.timerId_);
+ }
+ this.onDialogTimeout_();
+};
+
+/**
+ * @param {boolean} show True to show the warning dialog; false to hide it.
+ * @private
+ */
+remoting.IdleDetector.prototype.showIdleWarning_ = function(show) {
+ this.idleWarning_.hidden = !show;
+ remoting.windowShape.updateClientWindowShape();
+}
+
+/**
+ * @param {Array.<{left: number, top: number, width: number, height: number}>}
+ * rects List of rectangles.
+ */
+remoting.IdleDetector.prototype.addToRegion = function(rects) {
+ if (!this.idleWarning_.hidden) {
+ var dialog = this.idleWarning_.querySelector('.kd-modaldialog');
+ var rect = /** @type {ClientRect} */ (dialog.getBoundingClientRect());
+ rects.push(rect);
+ }
+};
+
+// Time-out after 1hr of no activity.
+remoting.IdleDetector.kIdleTimeoutMs = 60 * 60 * 1000;
+
+// Show the idle warning dialog for 2 minutes.
+remoting.IdleDetector.kDialogTimeoutMs = 2 * 60 * 1000;
diff --git a/remoting/webapp/app_remoting/js/keyboard_layouts_menu.js b/remoting/webapp/app_remoting/js/keyboard_layouts_menu.js
new file mode 100644
index 0000000..c16fd35
--- /dev/null
+++ b/remoting/webapp/app_remoting/js/keyboard_layouts_menu.js
@@ -0,0 +1,185 @@
+// Copyright 2014 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.
+
+/**
+ * @fileoverview
+ * Class managing the host's available keyboard layouts, allowing the user to
+ * select one that matches the local layout, or auto-selecting based on the
+ * current locale.
+ */
+
+'use strict';
+
+/** @suppress {duplicate} */
+var remoting = remoting || {};
+
+/**
+ * @param {remoting.ContextMenuAdapter} adapter
+ * @constructor
+ */
+remoting.KeyboardLayoutsMenu = function(adapter) {
+ /**
+ * @type {remoting.ContextMenuAdapter}
+ * @private
+ */
+ this.adapter_ = adapter;
+ /**
+ * @type {remoting.SubmenuManager}
+ * @private
+ */
+ this.submenuManager_ = new remoting.SubmenuManager(
+ adapter,
+ chrome.i18n.getMessage(/*i18n-content*/'KEYBOARD_LAYOUTS_SUBMENU_TITLE'),
+ true);
+ /**
+ * @type {string}
+ * @private
+ */
+ this.currentLayout_ = '';
+
+ adapter.addListener(this.onContextMenu_.bind(this));
+};
+
+/**
+ * @param {Array.<string>} layouts The keyboard layouts available on the host,
+ * for example en-US, de-DE
+ * @param {string} currentLayout The layout currently active on the host.
+ */
+remoting.KeyboardLayoutsMenu.prototype.setLayouts =
+ function(layouts, currentLayout) {
+ this.submenuManager_.removeAll();
+ this.currentLayout_ = '';
+ for (var i = 0; i < layouts.length; ++i) {
+ this.submenuManager_.add(this.makeMenuId_(layouts[i]), layouts[i]);
+ }
+ // Pick a suitable default layout.
+ this.getBestLayout_(layouts, currentLayout,
+ this.setLayout_.bind(this, false));
+};
+
+/**
+ * Notify the host that a new keyboard layout has been selected.
+ *
+ * @param {boolean} saveToLocalStorage If true, save the specified layout to
+ * local storage.
+ * @param {string} layout The new keyboard layout.
+ * @private
+ */
+remoting.KeyboardLayoutsMenu.prototype.setLayout_ =
+ function(saveToLocalStorage, layout) {
+ if (this.currentLayout_ != '') {
+ this.adapter_.updateCheckState(
+ this.makeMenuId_(this.currentLayout_), false);
+ }
+ this.adapter_.updateCheckState(this.makeMenuId_(layout), true);
+ this.currentLayout_ = layout;
+
+ console.log("Setting the keyboard layout to '" + layout + "'");
+ remoting.clientSession.sendClientMessage(
+ 'setKeyboardLayout',
+ JSON.stringify({layout: layout}));
+ if (saveToLocalStorage) {
+ var params = {};
+ params[remoting.KeyboardLayoutsMenu.KEY_] = layout;
+ chrome.storage.local.set(params);
+ }
+};
+
+/**
+ * Choose the best keyboard from the alternatives, based on the following
+ * algorithm:
+ * - Search local storage by for a preferred keyboard layout for the app;
+ * if it is found, prefer it over the current locale, falling back on the
+ * latter only if no match is found.
+ * - If the candidate layout matches one of the supported layouts, use it.
+ * - Otherwise, if the language portion of the candidate matches that of
+ * any of the supported layouts, use the first such layout (e.g, en-AU
+ * will match either en-US or en-GB, whichever appears first).
+ * - Otherwise, use the host's current layout.
+ *
+ * @param {Array.<string>} layouts
+ * @param {string} currentHostLayout
+ * @param {function(string):void} onDone
+ * @private
+ */
+remoting.KeyboardLayoutsMenu.prototype.getBestLayout_ =
+ function(layouts, currentHostLayout, onDone) {
+ /**
+ * Extract the language id from a string that is either "language" (e.g.
+ * "de") or "language-region" (e.g. "en-US").
+ *
+ * @param {string} layout
+ * @return {string}
+ */
+ var getLanguage = function(layout) {
+ var languageAndRegion = layout.split('-');
+ switch (languageAndRegion.length) {
+ case 1:
+ case 2:
+ return languageAndRegion[0];
+ default:
+ return '';
+ }
+ };
+
+ /** @param {Object.<string>} storage */
+ var chooseLayout = function(storage) {
+ var configuredLayout = storage[remoting.KeyboardLayoutsMenu.KEY_];
+ var tryLayouts = [ chrome.i18n.getUILanguage() ];
+ if (configuredLayout && typeof(configuredLayout) == 'string') {
+ tryLayouts.unshift(configuredLayout);
+ }
+ for (var i = 0; i < tryLayouts.length; ++i) {
+ if (layouts.indexOf(tryLayouts[i]) != -1) {
+ onDone(tryLayouts[i]);
+ return;
+ }
+ var language = getLanguage(tryLayouts[i]);
+ if (language) {
+ for (var j = 0; j < layouts.length; ++j) {
+ if (language == getLanguage(layouts[j])) {
+ onDone(layouts[j]);
+ return;
+ }
+ }
+ }
+ }
+ // Neither the stored layout nor UI locale was suitable.
+ onDone(currentHostLayout);
+ };
+
+ chrome.storage.local.get(remoting.KeyboardLayoutsMenu.KEY_, chooseLayout);
+};
+
+/**
+ * Create a menu id from the given keyboard layout.
+ *
+ * @param {string} layout Keyboard layout
+ * @return {string}
+ * @private
+ */
+remoting.KeyboardLayoutsMenu.prototype.makeMenuId_ = function(layout) {
+ return 'layout@' + layout;
+};
+
+/**
+ * Handle a click on the application's context menu.
+ *
+ * @param {OnClickData} info
+ * @private
+ */
+remoting.KeyboardLayoutsMenu.prototype.onContextMenu_ = function(info) {
+ /** @type {Array.<string>} */
+ var components = info.menuItemId.split('@');
+ if (components.length == 2 &&
+ this.makeMenuId_(components[1]) == info.menuItemId) {
+ this.setLayout_(true, components[1]);
+ }
+};
+
+/**
+ * @type {string}
+ * @private
+ */
+remoting.KeyboardLayoutsMenu.KEY_ = 'preferred-keyboard-layout';
diff --git a/remoting/webapp/app_remoting/js/loading_window.js b/remoting/webapp/app_remoting/js/loading_window.js
new file mode 100644
index 0000000..e7f8d55
--- /dev/null
+++ b/remoting/webapp/app_remoting/js/loading_window.js
@@ -0,0 +1,71 @@
+// Copyright 2014 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.
+
+/**
+ * @fileoverview
+ * Loading dialog for AppRemoting apps.
+ */
+
+'use strict';
+
+/** @suppress {duplicate} */
+var remoting = remoting || {};
+
+/**
+ * Namespace for loading window functions.
+ * @type {Object}
+ */
+remoting.LoadingWindow = function() {};
+
+/**
+ * When the loading window times out, replace it with a generic
+ * "Service Unavailable" message.
+ * @private
+ */
+remoting.LoadingWindow.onTimeout_ = function() {
+ remoting.MessageWindow.showErrorMessage(
+ chrome.i18n.getMessage(/**i18n-content*/'PRODUCT_NAME'),
+ chrome.i18n.getMessage(remoting.Error.SERVICE_UNAVAILABLE));
+};
+
+/**
+ * Show the loading dialog and start a timer When the timer expires, an error
+ * message will be displayed and the application will quit.
+ */
+remoting.LoadingWindow.show = function() {
+ if (remoting.loadingWindow_) {
+ return;
+ }
+
+ // TODO(garykac): Choose better default timeout.
+ // Timeout is currently 15min to handle when we need to spin up a new VM.
+ var kConnectionTimeout = 15 * 60 * 1000;
+
+ var transparencyWarning = '';
+ if (navigator.platform.indexOf('Mac') != -1) {
+ transparencyWarning =
+ chrome.i18n.getMessage(/**i18n-content*/'NO_TRANSPARENCY_WARNING');
+ }
+ remoting.loadingWindow_ = remoting.MessageWindow.showTimedMessageWindow(
+ chrome.i18n.getMessage(/**i18n-content*/'PRODUCT_NAME'),
+ chrome.i18n.getMessage(/**i18n-content*/'FOOTER_CONNECTING'),
+ transparencyWarning,
+ chrome.i18n.getMessage(/**i18n-content*/'CANCEL'),
+ remoting.MessageWindow.quitApp,
+ kConnectionTimeout,
+ remoting.LoadingWindow.onTimeout_);
+};
+
+/**
+ * Close the loading window.
+ */
+remoting.LoadingWindow.close = function() {
+ if (remoting.loadingWindow_) {
+ remoting.loadingWindow_.close();
+ }
+ remoting.loadingWindow_ = null;
+};
+
+/** @type {remoting.MessageWindow} */
+remoting.loadingWindow_ = null;
diff --git a/remoting/webapp/app_remoting/js/submenu_manager.js b/remoting/webapp/app_remoting/js/submenu_manager.js
new file mode 100644
index 0000000..d5f4a8f
--- /dev/null
+++ b/remoting/webapp/app_remoting/js/submenu_manager.js
@@ -0,0 +1,109 @@
+// Copyright 2014 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.
+
+/**
+ * @fileoverview
+ * Helper class for submenus that should add or remove the parent menu entry
+ * depending on whether or not any child items exist.
+ */
+
+'use strict';
+
+/** @suppress {duplicate} */
+var remoting = remoting || {};
+
+/**
+ * @param {remoting.ContextMenuAdapter} adapter
+ * @param {string} title The title of the parent menu item.
+ * @param {boolean} checkable True if the child menus should be checkable.
+ * @constructor
+ */
+remoting.SubmenuManager = function(adapter, title, checkable) {
+ /**
+ * @type {remoting.ContextMenuAdapter}
+ * @private
+ */
+ this.adapter_ = adapter;
+ /**
+ * @type {string}
+ * @private
+ */
+ this.title_ = title;
+ /**
+ * @type {boolean}
+ * @private
+ */
+ this.checkable_ = checkable;
+ /**
+ * @type {!Object}
+ * @private
+ */
+ this.childIds_ = {};
+ /**
+ * @type {string}
+ * @private
+ */
+ this.parentId_ = '';
+};
+
+/**
+ * Add a submenu item, or update the title of an existing submenu item.
+ *
+ * @param {string} id The child id.
+ * @param {string} title The window title.
+ * @return {boolean} True if a new menu item was created, false if an existing
+ * menu item was renamed.
+ */
+remoting.SubmenuManager.prototype.add = function(id, title) {
+ if (id in this.childIds_) {
+ this.adapter_.updateTitle(id, title);
+ return false;
+ } else {
+ this.childIds_[id] = true;
+ this.addOrRemoveParent_();
+ this.adapter_.create(id, title, this.checkable_, this.parentId_);
+ return true;
+ }
+};
+
+/**
+ * Remove a submenu item.
+ *
+ * @param {string} id The child id.
+ */
+remoting.SubmenuManager.prototype.remove = function(id) {
+ this.adapter_.remove(id);
+ delete this.childIds_[id];
+ this.addOrRemoveParent_();
+};
+
+/**
+ * Remove all submenu items.
+ */
+remoting.SubmenuManager.prototype.removeAll = function() {
+ var submenus = Object.keys(this.childIds_);
+ for (var i = 0; i < submenus.length; ++i) {
+ this.remove(submenus[i]);
+ }
+};
+
+/**
+ * Add the parent menu item if it doesn't exist but there are submenus items,
+ * or remove it if it exists but there are no submenus.
+ *
+ * @private
+ */
+remoting.SubmenuManager.prototype.addOrRemoveParent_ = function() {
+ if (Object.getOwnPropertyNames(this.childIds_).length != 0) {
+ if (!this.parentId_) {
+ this.parentId_ = base.generateXsrfToken(); // Use a random id
+ this.adapter_.create(this.parentId_, this.title_, false);
+ }
+ } else {
+ if (this.parentId_) {
+ this.adapter_.remove(this.parentId_);
+ this.parentId_ = '';
+ }
+ }
+};
diff --git a/remoting/webapp/app_remoting/js/window_activation_menu.js b/remoting/webapp/app_remoting/js/window_activation_menu.js
new file mode 100644
index 0000000..dfef499
--- /dev/null
+++ b/remoting/webapp/app_remoting/js/window_activation_menu.js
@@ -0,0 +1,84 @@
+// Copyright 2014 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.
+
+/**
+ * @fileoverview
+ * Class to update the application's context menu to include host-side windows
+ * and to notify the host when one of these menu items is selected.
+ */
+
+'use strict';
+
+/** @suppress {duplicate} */
+var remoting = remoting || {};
+
+/**
+ * @param {remoting.ContextMenuAdapter} adapter
+ * @constructor
+ */
+remoting.WindowActivationMenu = function(adapter) {
+ /**
+ * @type {remoting.SubmenuManager}
+ * @private
+ */
+ this.submenuManager_ = new remoting.SubmenuManager(
+ adapter,
+ chrome.i18n.getMessage(/*i18n-content*/'WINDOWS_SUBMENU_TITLE'),
+ false);
+
+ adapter.addListener(this.onContextMenu_.bind(this));
+};
+
+/**
+ * Add a window to the application's context menu, or update the title of an
+ * existing window.
+ *
+ * @param {number} id The window id.
+ * @param {string} title The window title.
+ */
+remoting.WindowActivationMenu.prototype.add = function(id, title) {
+ this.submenuManager_.add(this.makeMenuId_(id), title);
+ // TODO(jamiewalch): Once crbug.com/426283 is fixed, call drawAttention()
+ // here if the window does not have focus.
+};
+
+/**
+ * Remove a window from the application's context menu.
+ *
+ * @param {number} id The window id.
+ */
+remoting.WindowActivationMenu.prototype.remove = function(id) {
+ this.submenuManager_.remove(this.makeMenuId_(id));
+};
+
+/**
+ * Create a menu id from the given window id.
+ *
+ * @param {number} windowId
+ * @return {string}
+ * @private
+ */
+remoting.WindowActivationMenu.prototype.makeMenuId_ = function(windowId) {
+ return 'window-' + windowId;
+};
+
+/**
+ * Handle a click on the application's context menu.
+ *
+ * @param {OnClickData} info
+ * @private
+ */
+remoting.WindowActivationMenu.prototype.onContextMenu_ = function(info) {
+ /** @type {Array.<string>} */
+ var components = info.menuItemId.split('-');
+ if (components.length == 2 &&
+ this.makeMenuId_(parseInt(components[1], 10)) == info.menuItemId) {
+ remoting.clientSession.sendClientMessage(
+ 'activateWindow',
+ JSON.stringify({ id: parseInt(components[1], 0) }));
+ if (chrome.app.window.current().isMinimized()) {
+ chrome.app.window.current().restore();
+ }
+ }
+};
diff --git a/remoting/webapp/app_remoting/manifest_common.json.jinja2 b/remoting/webapp/app_remoting/manifest_common.json.jinja2
new file mode 100644
index 0000000..aca325c
--- /dev/null
+++ b/remoting/webapp/app_remoting/manifest_common.json.jinja2
@@ -0,0 +1,46 @@
+ {{ MANIFEST_KEY_FOR_UNOFFICIAL_BUILD }}
+ "name": "{{APP_NAME}}",
+ "description": "{{APP_DESCRIPTION}}",
+ "version": "{{ FULL_APP_VERSION }}",
+ "manifest_version": 2,
+ "default_locale": "en",
+ "app": {
+ "background": {
+ "scripts": ["ar_background.js", "platform.js"]
+ }
+ },
+ "icons": {
+ "128": "icon128.png",
+ "48": "icon48.png",
+ "16": "icon16.png"
+ },
+ "optional_permissions": [
+ "<all_urls>"
+ ],
+ "permissions": [
+ "{{ APP_REMOTING_API_BASE_URL }}/*",
+ "{{ DIRECTORY_API_BASE_URL }}/*",
+ "{{ OAUTH2_ACCOUNTS_HOST }}/*",
+ "{{ OAUTH2_API_BASE_URL }}/*",
+ "{{ TALK_GADGET_HOST }}/talkgadget/*",
+ "app.window.shape",
+ "clipboardRead",
+ "clipboardWrite",
+ "contextMenus",
+ "experimental",
+ "fileSystem",
+ "fullscreen",
+ "https://relay.google.com/*",
+ "identity",
+ "pointerLock",
+ "storage"
+ ],
+ "oauth2": {
+ "client_id": "{{ REMOTING_IDENTITY_API_CLIENT_ID }}",
+ "scopes": [
+ "https://www.googleapis.com/auth/appremoting.runapplication https://www.googleapis.com/auth/googletalk https://www.googleapis.com/auth/userinfo#email https://www.googleapis.com/auth/userinfo.profile https://docs.google.com/feeds/ https://www.googleapis.com/auth/drive"
+ ]
+ },
+ "sandbox": {
+ "pages": [ "wcs_sandbox.html" ]
+ }
diff --git a/remoting/webapp/base/html/main.css b/remoting/webapp/base/html/main.css
index 150ce45..5a6cd7e 100644
--- a/remoting/webapp/base/html/main.css
+++ b/remoting/webapp/base/html/main.css
@@ -561,6 +561,10 @@ button {
margin-top: 12px;
}
+.checkbox-label.disabled {
+ color: #888;
+}
+
.checkbox-label input[type=checkbox] {
float: __MSG_@@bidi_start_edge__;
margin-__MSG_@@bidi_start_edge__: -20px;
@@ -569,6 +573,10 @@ button {
margin-top: 2px;
}
+.checkbox-indent {
+ margin-left: 18px;
+}
+
#current-email {
color: rgba(0, 0, 0, 0.5);
}
diff --git a/remoting/webapp/base/html/message_window.html b/remoting/webapp/base/html/message_window.html
index 2e7074b..4b34c0a 100644
--- a/remoting/webapp/base/html/message_window.html
+++ b/remoting/webapp/base/html/message_window.html
@@ -20,7 +20,7 @@ found in the LICENSE file.
<p id="infobox" class="information-box"></p>
<p id="message"></p>
<div class="button-row">
- <button id="button-primary"></button>
+ <button id="button-primary" autofocus="autofocus"></button>
<button id="button-secondary"></button>
</div>
</body>
diff --git a/remoting/webapp/base/js/application.js b/remoting/webapp/base/js/application.js
index 82bef0f..29ffa88 100644
--- a/remoting/webapp/base/js/application.js
+++ b/remoting/webapp/base/js/application.js
@@ -44,7 +44,16 @@ remoting.Application.prototype.setDelegate = function(appDelegate) {
* @return {void} Nothing.
*/
remoting.Application.prototype.start = function() {
- this.delegate_.init();
+ // Create global objects.
+ remoting.ClientPlugin.factory = new remoting.DefaultClientPluginFactory();
+ remoting.SessionConnector.factory =
+ new remoting.DefaultSessionConnectorFactory();
+
+ // TODO(garykac): This should be owned properly rather than living in the
+ // global 'remoting' namespace.
+ remoting.settings = new remoting.Settings();
+
+ this.delegate_.init(this.getSessionConnector());
};
/**
@@ -155,9 +164,10 @@ remoting.Application.Delegate = function() {};
* Initialize the application and register all event handlers. After this
* is called, the app is running and waiting for user events.
*
+ * @param {remoting.SessionConnector} connector
* @return {void} Nothing.
*/
-remoting.Application.Delegate.prototype.init = function() {};
+remoting.Application.Delegate.prototype.init = function(connector) {};
/**
* @return {string} The default remap keys for the current platform.
diff --git a/remoting/webapp/browser_test/mock_session_connector.js b/remoting/webapp/browser_test/mock_session_connector.js
index 7ef5b50..f8cd341 100644
--- a/remoting/webapp/browser_test/mock_session_connector.js
+++ b/remoting/webapp/browser_test/mock_session_connector.js
@@ -37,6 +37,19 @@ remoting.MockSessionConnector.prototype.connectMe2Me =
this.connect_();
};
+remoting.MockSessionConnector.prototype.retryConnectMe2Me =
+ function(host, fetchPin, fetchThirdPartyToken,
+ clientPairingId, clientPairedSecret) {
+ this.mode_ = remoting.ClientSession.Mode.ME2ME;
+ this.connect_();
+};
+
+remoting.MockSessionConnector.prototype.connectMe2App =
+ function(host, fetchThirdPartyToken) {
+ this.mode_ = remoting.ClientSession.Mode.ME2APP;
+ this.connect_();
+};
+
remoting.MockSessionConnector.prototype.updatePairingInfo =
function(clientId, sharedSecret) {
};
@@ -117,4 +130,4 @@ remoting.MockSessionConnectorFactory.prototype.createConnector =
function(clientContainer, onConnected, onError, onExtensionMessage) {
return new remoting.MockSessionConnector(
clientContainer, onConnected, onError, onExtensionMessage);
-}; \ No newline at end of file
+};
diff --git a/remoting/webapp/crd/js/client_session.js b/remoting/webapp/crd/js/client_session.js
index f3ae7ce..bdbca7f 100644
--- a/remoting/webapp/crd/js/client_session.js
+++ b/remoting/webapp/crd/js/client_session.js
@@ -340,7 +340,8 @@ remoting.ClientSession.ConnectionError.fromString = function(error) {
/** @enum {number} */
remoting.ClientSession.Mode = {
IT2ME: 0,
- ME2ME: 1
+ ME2ME: 1,
+ APP_REMOTING: 2
};
/**
@@ -552,7 +553,8 @@ remoting.ClientSession.prototype.onPluginInitialized_ = function(initialized) {
remoting.ClientPlugin.Feature.INJECT_KEY_EVENT)) {
var sendKeysElement = document.getElementById('send-keys-menu');
sendKeysElement.hidden = true;
- } else if (this.mode_ != remoting.ClientSession.Mode.ME2ME) {
+ } else if (this.mode_ != remoting.ClientSession.Mode.ME2ME &&
+ this.mode_ != remoting.ClientSession.Mode.APP_REMOTING) {
var sendCadElement = document.getElementById('send-ctrl-alt-del');
sendCadElement.hidden = true;
}
diff --git a/remoting/webapp/crd/js/connection_stats.js b/remoting/webapp/crd/js/connection_stats.js
index 8b144c4..3da3993 100644
--- a/remoting/webapp/crd/js/connection_stats.js
+++ b/remoting/webapp/crd/js/connection_stats.js
@@ -48,6 +48,14 @@ remoting.ConnectionStats.prototype.toggle = function() {
};
/**
+ * Show or hide the connection stats div.
+ * @param {boolean} show
+ */
+remoting.ConnectionStats.prototype.show = function(show) {
+ this.statsElement_.hidden = !show;
+};
+
+/**
* If the stats panel is visible, add its bounding rectangle to the specified
* region.
* @param {Array.<{left: number, top: number, width: number, height: number}>}
diff --git a/remoting/webapp/crd/js/error.js b/remoting/webapp/crd/js/error.js
index 8c4a0e1..6da496d 100644
--- a/remoting/webapp/crd/js/error.js
+++ b/remoting/webapp/crd/js/error.js
@@ -31,7 +31,8 @@ remoting.Error = {
NOT_AUTHENTICATED: /*i18n-content*/'ERROR_NOT_AUTHENTICATED',
INVALID_HOST_DOMAIN: /*i18n-content*/'ERROR_INVALID_HOST_DOMAIN',
P2P_FAILURE: /*i18n-content*/'ERROR_P2P_FAILURE',
- REGISTRATION_FAILED: /*i18n-content*/'ERROR_HOST_REGISTRATION_FAILED'
+ REGISTRATION_FAILED: /*i18n-content*/'ERROR_HOST_REGISTRATION_FAILED',
+ NOT_AUTHORIZED: /*i18n-content*/'ERROR_NOT_AUTHORIZED'
};
/**
diff --git a/remoting/webapp/crd/js/plugin_settings.js b/remoting/webapp/crd/js/plugin_settings.js
index bb571bc..9aff0e6 100644
--- a/remoting/webapp/crd/js/plugin_settings.js
+++ b/remoting/webapp/crd/js/plugin_settings.js
@@ -43,6 +43,22 @@ remoting.Settings.prototype.OAUTH2_REDIRECT_URL = function() {
return 'OAUTH2_REDIRECT_URL';
}
+/** @type {string} Base URL for the App Remoting API. */
+remoting.Settings.prototype.APP_REMOTING_API_BASE_URL =
+ 'APP_REMOTING_API_BASE_URL';
+
+/**
+ * Return this app's Application ID.
+ *
+ * This is a function rather than a constant because the build script may
+ * replace this string with code to calculate the app id dynamically.
+ *
+ * @return {string} The Application ID.
+ */
+remoting.Settings.prototype.getAppRemotingApplicationId = function() {
+ return 'APP_REMOTING_APPLICATION_ID';
+};
+
/** @type {string} XMPP JID for the remoting directory server bot. */
remoting.Settings.prototype.DIRECTORY_BOT_JID = 'DIRECTORY_BOT_JID';
diff --git a/remoting/webapp/crd/js/remoting.js b/remoting/webapp/crd/js/remoting.js
index 96c39bb..d08287e 100644
--- a/remoting/webapp/crd/js/remoting.js
+++ b/remoting/webapp/crd/js/remoting.js
@@ -28,12 +28,6 @@ remoting.initGlobalObjects = function() {
console.log(remoting.getExtensionInfo());
l10n.localize();
- // Create global objects.
- remoting.ClientPlugin.factory = new remoting.DefaultClientPluginFactory();
- remoting.SessionConnector.factory =
- new remoting.DefaultSessionConnectorFactory();
- remoting.settings = new remoting.Settings();
-
if (base.isAppsV2()) {
remoting.fullscreen = new remoting.FullscreenAppsV2();
} else {
@@ -101,19 +95,18 @@ remoting.getExtensionInfo = function() {
* the more intuitive way to end a Me2Me session, and re-connecting is easy.
*/
remoting.promptClose = function() {
- if (!remoting.clientSession ||
- remoting.clientSession.getMode() == remoting.ClientSession.Mode.ME2ME) {
- return null;
- }
- switch (remoting.currentMode) {
- case remoting.AppMode.CLIENT_CONNECTING:
- case remoting.AppMode.HOST_WAITING_FOR_CODE:
- case remoting.AppMode.HOST_WAITING_FOR_CONNECTION:
- case remoting.AppMode.HOST_SHARED:
- case remoting.AppMode.IN_SESSION:
- return chrome.i18n.getMessage(/*i18n-content*/'CLOSE_PROMPT');
- default:
- return null;
+ if (remoting.clientSession &&
+ remoting.clientSession.getMode() == remoting.ClientSession.Mode.IT2ME) {
+ switch (remoting.currentMode) {
+ case remoting.AppMode.CLIENT_CONNECTING:
+ case remoting.AppMode.HOST_WAITING_FOR_CODE:
+ case remoting.AppMode.HOST_WAITING_FOR_CONNECTION:
+ case remoting.AppMode.HOST_SHARED:
+ case remoting.AppMode.IN_SESSION:
+ return chrome.i18n.getMessage(/*i18n-content*/'CLOSE_PROMPT');
+ default:
+ return null;
+ }
}
};
diff --git a/remoting/webapp/crd/js/server_log_entry.js b/remoting/webapp/crd/js/server_log_entry.js
index 93f1661..6813206 100644
--- a/remoting/webapp/crd/js/server_log_entry.js
+++ b/remoting/webapp/crd/js/server_log_entry.js
@@ -164,6 +164,8 @@ remoting.ServerLogEntry.VALUE_MODE_IT2ME_ = 'it2me';
/** @private */
remoting.ServerLogEntry.VALUE_MODE_ME2ME_ = 'me2me';
/** @private */
+remoting.ServerLogEntry.VALUE_MODE_APP_REMOTING_ = 'lgapp';
+/** @private */
remoting.ServerLogEntry.VALUE_MODE_UNKNOWN_ = 'unknown';
/**
@@ -471,6 +473,8 @@ remoting.ServerLogEntry.getModeField = function(mode) {
return remoting.ServerLogEntry.VALUE_MODE_IT2ME_;
case remoting.ClientSession.Mode.ME2ME:
return remoting.ServerLogEntry.VALUE_MODE_ME2ME_;
+ case remoting.ClientSession.Mode.APP_REMOTING:
+ return remoting.ServerLogEntry.VALUE_MODE_APP_REMOTING_;
default:
return remoting.ServerLogEntry.VALUE_MODE_UNKNOWN_;
}
diff --git a/remoting/webapp/crd/js/session_connector.js b/remoting/webapp/crd/js/session_connector.js
index 84b8d3b..529e75a 100644
--- a/remoting/webapp/crd/js/session_connector.js
+++ b/remoting/webapp/crd/js/session_connector.js
@@ -52,6 +52,19 @@ remoting.SessionConnector.prototype.connectMe2Me =
remoting.SessionConnector.prototype.retryConnectMe2Me = function(host) {};
/**
+ * Initiate a Me2App connection.
+ *
+ * @param {remoting.Host} host The Me2Me host to which to connect.
+ * @param {function(string, string, string,
+ * function(string, string): void): void}
+ * fetchThirdPartyToken Function to obtain a token from a third party
+ * authentication server.
+ * @return {void} Nothing.
+ */
+remoting.SessionConnector.prototype.connectMe2App =
+ function(host, fetchThirdPartyToken) {};
+
+/**
* Update the pairing info so that the reconnect function will work correctly.
*
* @param {string} clientId The paired client id.
diff --git a/remoting/webapp/crd/js/session_connector_impl.js b/remoting/webapp/crd/js/session_connector_impl.js
index c6f5bb1..0056554 100644
--- a/remoting/webapp/crd/js/session_connector_impl.js
+++ b/remoting/webapp/crd/js/session_connector_impl.js
@@ -256,6 +256,25 @@ remoting.SessionConnectorImpl.prototype.retryConnectMe2Me = function(host) {
};
/**
+ * Initiate a Me2App connection.
+ *
+ * @param {remoting.Host} host The Me2Me host to which to connect.
+ * @param {function(string, string, string,
+ * function(string, string): void): void}
+ * fetchThirdPartyToken Function to obtain a token from a third party
+ * authenticaiton server.
+ * @return {void} Nothing.
+ */
+remoting.SessionConnectorImpl.prototype.connectMe2App =
+ function(host, fetchThirdPartyToken) {
+ this.connectionMode_ = remoting.ClientSession.Mode.APP_REMOTING;
+ this.logHostOfflineErrors_ = true;
+ this.connectMe2MeInternal_(
+ host.hostId, host.jabberId, host.publicKey, host.hostName,
+ function() {}, fetchThirdPartyToken, '', '');
+};
+
+/**
* Update the pairing info so that the reconnect function will work correctly.
*
* @param {string} clientId The paired client id.
diff --git a/remoting/webapp/js_proto/dom_proto.js b/remoting/webapp/js_proto/dom_proto.js
index 668b88a..19d5c6a 100644
--- a/remoting/webapp/js_proto/dom_proto.js
+++ b/remoting/webapp/js_proto/dom_proto.js
@@ -21,6 +21,9 @@ Document.prototype.execCommand = function(command) {};
/** @return {void} Nothing. */
Document.prototype.webkitCancelFullScreen = function() {};
+/** @return {void} Nothing. */
+Document.prototype.exitPointerLock = function() {};
+
/** @type {boolean} */
Document.prototype.webkitIsFullScreen;
@@ -37,6 +40,9 @@ Element.ALLOW_KEYBOARD_INPUT;
/** @return {void} Nothing. */
Element.prototype.webkitRequestFullScreen = function(flags) {};
+/** @return {void} Nothing. */
+Element.prototype.requestPointerLock = function() {};
+
/** @type {boolean} */
Element.prototype.hidden;
@@ -122,6 +128,11 @@ MutationRecord.prototype.type;
/** @type {{getRandomValues: function((Uint16Array|Uint8Array)):void}} */
Window.prototype.crypto;
+/**
+ * @param {function():void} callback
+ */
+Window.prototype.requestAnimationFrame = function(callback) {};
+
/**
* @constructor
@@ -234,6 +245,16 @@ Promise.resolve = function (value) {};
Event.prototype.dataTransfer = null;
/**
+ * @type {number}
+ */
+Event.prototype.movementX = 0;
+
+/**
+ * @type {number}
+ */
+Event.prototype.movementY = 0;
+
+/**
* @param {string} type
* @param {boolean} canBubble
* @param {boolean} cancelable