1 // Copyright 2015 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
7 * Class handling user-facing aspects of the client session.
12 /** @suppress {duplicate} */
13 var remoting = remoting || {};
16 * @param {HTMLElement} container
17 * @param {remoting.ConnectionInfo} connectionInfo
19 * @extends {base.EventSourceImpl}
20 * @implements {base.Disposable}
22 remoting.DesktopConnectedView = function(container, connectionInfo) {
24 /** @private {HTMLElement} */
25 this.container_ = container;
27 /** @private {remoting.ClientPlugin} */
28 this.plugin_ = connectionInfo.plugin();
30 /** @private {remoting.ClientSession} */
31 this.session_ = connectionInfo.session();
34 this.host_ = connectionInfo.host();
36 /** @private {remoting.DesktopViewport} */
37 this.viewport_ = null;
39 /** private {remoting.ConnectedView} */
42 /** @private {remoting.VideoFrameRecorder} */
43 this.videoFrameRecorder_ = null;
45 /** private {base.Disposable} */
46 this.eventHooks_ = null;
49 this.stats_ = new remoting.ConnectionStats(
50 document.getElementById('statistics'), connectionInfo.plugin());
56 /** @return {void} Nothing. */
57 remoting.DesktopConnectedView.prototype.dispose = function() {
58 if (remoting.windowFrame) {
59 remoting.windowFrame.setDesktopConnectedView(null);
61 if (remoting.toolbar) {
62 remoting.toolbar.setDesktopConnectedView(null);
64 if (remoting.optionsMenu) {
65 remoting.optionsMenu.setDesktopConnectedView(null);
68 document.body.classList.remove('connected');
70 base.dispose(this.eventHooks_);
71 this.eventHooks_ = null;
73 base.dispose(this.viewport_);
74 this.viewport_ = null;
76 base.dispose(this.stats_);
81 * Get host display name.
85 remoting.DesktopConnectedView.prototype.getHostDisplayName = function() {
86 return this.host_.hostName;
90 * @return {boolean} True if shrink-to-fit is enabled; false otherwise.
92 remoting.DesktopConnectedView.prototype.getShrinkToFit = function() {
94 return this.viewport_.getShrinkToFit();
100 * @return {boolean} True if resize-to-client is enabled; false otherwise.
102 remoting.DesktopConnectedView.prototype.getResizeToClient = function() {
103 if (this.viewport_) {
104 return this.viewport_.getResizeToClient();
110 * @return {boolean} True if the right-hand Ctrl key is mapped to the Meta
111 * (Windows, Command) key.
113 remoting.DesktopConnectedView.prototype.getMapRightCtrl = function() {
114 return this.host_.options.remapKeys[0x0700e4] === 0x0700e7;
117 remoting.DesktopConnectedView.prototype.toggleStats = function() {
118 this.stats_.toggle();
122 * @return {boolean} True if the connection stats is visible; false otherwise.
124 remoting.DesktopConnectedView.prototype.isStatsVisible = function() {
125 return this.stats_.isVisible();
129 * @return {Element} The element that should host the plugin.
132 remoting.DesktopConnectedView.prototype.getPluginContainer_ = function() {
133 return this.container_.querySelector('.client-plugin-container');
136 /** @return {remoting.DesktopViewport} */
137 remoting.DesktopConnectedView.prototype.getViewportForTesting = function() {
138 return this.viewport_;
141 /** @return {remoting.ConnectedView} */
142 remoting.DesktopConnectedView.prototype.getConnectedViewForTesting =
148 remoting.DesktopConnectedView.prototype.initPlugin_ = function() {
149 console.assert(remoting.app instanceof remoting.DesktopRemoting,
150 '|remoting.app| is not an instance of DesktopRemoting.');
151 var drApp = /** @type {remoting.DesktopRemoting} */ (remoting.app);
152 var mode = drApp.getConnectionMode();
154 // Show the Send Keys menu only if the plugin has the injectKeyEvent feature,
155 // and the Ctrl-Alt-Del button only in Me2Me mode.
156 if (!this.plugin_.hasFeature(
157 remoting.ClientPlugin.Feature.INJECT_KEY_EVENT)) {
158 var sendKeysElement = document.getElementById('send-keys-menu');
159 sendKeysElement.hidden = true;
160 } else if (mode == remoting.DesktopRemoting.Mode.IT2ME) {
161 var sendCadElement = document.getElementById('send-ctrl-alt-del');
162 sendCadElement.hidden = true;
167 * This is a callback that gets called when the window is resized.
169 * @return {void} Nothing.
172 remoting.DesktopConnectedView.prototype.onResize_ = function() {
173 if (this.viewport_) {
174 this.viewport_.onResize();
179 remoting.DesktopConnectedView.prototype.initUI_ = function() {
180 document.body.classList.add('connected');
182 this.view_ = new remoting.ConnectedView(
183 this.plugin_, this.container_,
184 this.container_.querySelector('.mouse-cursor-overlay'));
186 var scrollerElement = document.getElementById('scroller');
187 this.viewport_ = new remoting.DesktopViewport(
188 scrollerElement || document.body,
189 this.plugin_.hostDesktop(),
192 if (remoting.windowFrame) {
193 remoting.windowFrame.setDesktopConnectedView(this);
195 if (remoting.toolbar) {
196 remoting.toolbar.setDesktopConnectedView(this);
198 if (remoting.optionsMenu) {
199 remoting.optionsMenu.setDesktopConnectedView(this);
202 // Activate full-screen related UX.
203 this.eventHooks_ = new base.Disposables(
205 new base.EventHook(this.session_,
206 remoting.ClientSession.Events.videoChannelStateChanged,
207 this.view_.onConnectionReady.bind(this.view_)),
208 new base.DomEventHook(window, 'resize', this.onResize_.bind(this), false),
209 new remoting.Fullscreen.EventHook(this.onFullScreenChanged_.bind(this)));
210 this.onFullScreenChanged_(remoting.fullscreen.isActive());
214 * Set the shrink-to-fit and resize-to-client flags and save them if this is
215 * a Me2Me connection.
217 * @param {boolean} shrinkToFit True if the remote desktop should be scaled
218 * down if it is larger than the client window; false if scroll-bars
219 * should be added in this case.
220 * @param {boolean} resizeToClient True if window resizes should cause the
221 * host to attempt to resize its desktop to match the client window size;
222 * false to disable this behaviour for subsequent window resizes--the
223 * current host desktop size is not restored in this case.
224 * @return {void} Nothing.
226 remoting.DesktopConnectedView.prototype.setScreenMode =
227 function(shrinkToFit, resizeToClient) {
228 this.viewport_.setScreenMode(shrinkToFit, resizeToClient);
232 * Called when the full-screen status has changed, either via the
233 * remoting.Fullscreen class, or via a system event such as the Escape key
235 * @param {boolean=} fullscreen True if the app is entering full-screen mode;
236 * false if it is leaving it.
239 remoting.DesktopConnectedView.prototype.onFullScreenChanged_ = function (
241 if (this.viewport_) {
242 // When a window goes full-screen, a resize event is triggered, but the
243 // Fullscreen.isActive call is not guaranteed to return true until the
244 // full-screen event is triggered. In apps v2, the size of the window's
245 // client area is calculated differently in full-screen mode, so register
247 this.viewport_.onResize();
248 this.viewport_.enableBumpScroll(Boolean(fullscreen));
253 * Set whether or not the right-hand Ctrl key should send the Meta (Windows,
256 * @param {boolean} enable True to enable the mapping; false to disable.
258 remoting.DesktopConnectedView.prototype.setMapRightCtrl = function(enable) {
259 if (enable === this.getMapRightCtrl()) {
260 return; // In case right Ctrl is mapped, but not to right Meta.
264 this.host_.options.remapKeys[0x0700e4] = 0x0700e7;
266 delete this.host_.options.remapKeys[0x0700e4]
268 this.setRemapKeys(this.host_.options.remapKeys);
272 * Sends a Ctrl-Alt-Del sequence to the remoting client.
274 * @return {void} Nothing.
276 remoting.DesktopConnectedView.prototype.sendCtrlAltDel = function() {
277 console.log('Sending Ctrl-Alt-Del.');
278 this.plugin_.injectKeyCombination([0x0700e0, 0x0700e2, 0x07004c]);
282 * Sends a Print Screen keypress to the remoting client.
284 * @return {void} Nothing.
286 remoting.DesktopConnectedView.prototype.sendPrintScreen = function() {
287 console.log('Sending Print Screen.');
288 this.plugin_.injectKeyCombination([0x070046]);
292 * Sets and stores the key remapping setting for the current host. If set,
293 * these mappings override the defaults for the client platform.
295 * @param {!Object} remappings
297 remoting.DesktopConnectedView.prototype.setRemapKeys = function(remappings) {
298 this.plugin_.setRemapKeys(remappings);
299 // Save the new remapping setting.
300 this.host_.options.remapKeys =
301 /** @type {!Object} */ (base.deepCopy(remappings));
302 this.host_.options.save();
305 /** @param {remoting.VideoFrameRecorder} recorder */
306 remoting.DesktopConnectedView.prototype.setVideoFrameRecorder =
308 this.videoFrameRecorder_ = recorder;
312 * Returns true if the ClientSession can record video frames to a file.
315 remoting.DesktopConnectedView.prototype.canRecordVideo = function() {
316 return !!this.videoFrameRecorder_;
320 * Returns true if the ClientSession is currently recording video frames.
323 remoting.DesktopConnectedView.prototype.isRecordingVideo = function() {
324 if (!this.videoFrameRecorder_) {
327 return this.videoFrameRecorder_.isRecording();
331 * Starts or stops recording of video frames.
333 remoting.DesktopConnectedView.prototype.startStopRecording = function() {
334 if (this.videoFrameRecorder_) {
335 this.videoFrameRecorder_.startStopRecording();