2014-07-19 04:49:19 +04:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
|
2019-01-17 21:18:31 +03:00
|
|
|
const { Services } = ChromeUtils.import("resource://gre/modules/Services.jsm");
|
2020-05-18 21:57:40 +03:00
|
|
|
const { XPCOMUtils } = ChromeUtils.import(
|
|
|
|
"resource://gre/modules/XPCOMUtils.jsm"
|
|
|
|
);
|
2019-01-17 21:18:31 +03:00
|
|
|
const { webrtcUI } = ChromeUtils.import("resource:///modules/webrtcUI.jsm");
|
2014-07-19 04:49:19 +04:00
|
|
|
|
2020-05-26 21:00:36 +03:00
|
|
|
ChromeUtils.defineModuleGetter(
|
|
|
|
this,
|
|
|
|
"AppConstants",
|
|
|
|
"resource://gre/modules/AppConstants.jsm"
|
|
|
|
);
|
|
|
|
|
|
|
|
ChromeUtils.defineModuleGetter(
|
|
|
|
this,
|
|
|
|
"MacOSWebRTCStatusbarIndicator",
|
|
|
|
"resource:///modules/webrtcUI.jsm"
|
|
|
|
);
|
|
|
|
|
2020-06-16 16:19:16 +03:00
|
|
|
ChromeUtils.defineModuleGetter(
|
|
|
|
this,
|
|
|
|
"BrowserWindowTracker",
|
|
|
|
"resource:///modules/BrowserWindowTracker.jsm"
|
|
|
|
);
|
|
|
|
|
2020-05-18 21:57:40 +03:00
|
|
|
XPCOMUtils.defineLazyServiceGetter(
|
|
|
|
this,
|
|
|
|
"gScreenManager",
|
|
|
|
"@mozilla.org/gfx/screenmanager;1",
|
|
|
|
"nsIScreenManager"
|
|
|
|
);
|
2014-07-19 04:49:19 +04:00
|
|
|
|
2020-05-18 21:57:40 +03:00
|
|
|
/**
|
|
|
|
* Public function called by webrtcUI to update the indicator
|
|
|
|
* display when the active streams change.
|
|
|
|
*/
|
2014-07-19 04:49:19 +04:00
|
|
|
function updateIndicatorState() {
|
2020-05-18 21:57:40 +03:00
|
|
|
WebRTCIndicator.updateIndicatorState();
|
2014-07-19 04:49:19 +04:00
|
|
|
}
|
|
|
|
|
2020-07-17 04:22:10 +03:00
|
|
|
/**
|
|
|
|
* Public function called by webrtcUI to indicate that webrtcUI
|
|
|
|
* is about to close the indicator. This is so that we can differentiate
|
|
|
|
* between closes that are caused by webrtcUI, and closes that are
|
|
|
|
* caused by other reasons (like the user closing the window via the
|
|
|
|
* OS window controls somehow).
|
|
|
|
*
|
|
|
|
* If the window is closed without having called this method first, the
|
|
|
|
* indicator will ask webrtcUI to shutdown any remaining streams and then
|
|
|
|
* select and focus the most recent browser tab that a stream was shared
|
|
|
|
* with.
|
|
|
|
*/
|
|
|
|
function closingInternally() {
|
|
|
|
WebRTCIndicator.closingInternally();
|
|
|
|
}
|
|
|
|
|
2020-05-18 21:57:40 +03:00
|
|
|
/**
|
|
|
|
* Main control object for the WebRTC global indicator
|
|
|
|
*/
|
|
|
|
const WebRTCIndicator = {
|
|
|
|
init(event) {
|
|
|
|
addEventListener("load", this);
|
2020-05-26 21:00:36 +03:00
|
|
|
addEventListener("unload", this);
|
2014-07-19 04:49:19 +04:00
|
|
|
|
2020-05-18 21:57:40 +03:00
|
|
|
// If the user customizes the position of the indicator, we will
|
|
|
|
// not try to re-center it on the primary display after indicator
|
|
|
|
// state updates.
|
|
|
|
this.positionCustomized = false;
|
2014-07-19 04:49:19 +04:00
|
|
|
|
2020-05-18 21:57:40 +03:00
|
|
|
this.updatingIndicatorState = false;
|
|
|
|
this.loaded = false;
|
2020-07-17 04:22:10 +03:00
|
|
|
this.isClosingInternally = false;
|
2020-05-26 21:00:36 +03:00
|
|
|
|
|
|
|
if (AppConstants.platform == "macosx") {
|
|
|
|
this.macOSIndicator = new MacOSWebRTCStatusbarIndicator();
|
|
|
|
}
|
2020-07-15 13:40:04 +03:00
|
|
|
|
|
|
|
if (
|
|
|
|
Services.prefs.getBoolPref("privacy.webrtc.hideGlobalIndicator", false)
|
|
|
|
) {
|
|
|
|
let baseWin = window.docShell.treeOwner.QueryInterface(Ci.nsIBaseWindow);
|
|
|
|
baseWin.visibility = false;
|
|
|
|
}
|
2020-05-18 21:57:40 +03:00
|
|
|
},
|
2014-07-19 04:49:19 +04:00
|
|
|
|
2020-05-18 21:57:40 +03:00
|
|
|
/**
|
|
|
|
* Exposed externally so that webrtcUI can alert the indicator to
|
|
|
|
* update itself when sharing states have changed.
|
|
|
|
*/
|
2020-08-11 13:03:07 +03:00
|
|
|
updateIndicatorState() {
|
2020-05-26 21:00:36 +03:00
|
|
|
if (this.macOSIndicator) {
|
|
|
|
this.macOSIndicator.updateIndicatorState();
|
|
|
|
}
|
2020-05-18 21:57:40 +03:00
|
|
|
// It's possible that we were called externally before the indicator
|
|
|
|
// finished loading. If so, then bail out - we're going to call
|
|
|
|
// updateIndicatorState ourselves automatically once the load
|
|
|
|
// event fires.
|
|
|
|
if (!this.loaded) {
|
|
|
|
return;
|
|
|
|
}
|
2014-07-19 04:49:19 +04:00
|
|
|
|
2020-05-18 21:57:40 +03:00
|
|
|
// We've started to update the indicator state. We set this flag so
|
|
|
|
// that the MozUpdateWindowPos event handler doesn't interpret indicator
|
|
|
|
// state updates as window movement caused by the user.
|
|
|
|
this.updatingIndicatorState = true;
|
2014-07-19 04:49:19 +04:00
|
|
|
|
2020-05-18 21:57:40 +03:00
|
|
|
this.updateWindowAttr("sharingvideo", webrtcUI.showCameraIndicator);
|
|
|
|
this.updateWindowAttr("sharingaudio", webrtcUI.showMicrophoneIndicator);
|
2020-06-16 14:20:57 +03:00
|
|
|
|
|
|
|
let sharingScreen = webrtcUI.showScreenSharingIndicator.startsWith(
|
|
|
|
"Screen"
|
2020-05-18 21:57:40 +03:00
|
|
|
);
|
2020-06-16 14:20:57 +03:00
|
|
|
this.updateWindowAttr("sharingscreen", sharingScreen);
|
2014-08-01 17:15:49 +04:00
|
|
|
|
2020-05-18 21:57:40 +03:00
|
|
|
// We don't currently support the browser-tab sharing case, so we don't
|
|
|
|
// check if the screen sharing indicator starts with "Browser".
|
|
|
|
|
|
|
|
// We special-case sharing a window, because we want to have a slightly
|
|
|
|
// different UI if we're sharing a browser window.
|
|
|
|
let sharingWindow = webrtcUI.showScreenSharingIndicator.startsWith(
|
|
|
|
"Window"
|
|
|
|
);
|
|
|
|
this.updateWindowAttr("sharingwindow", sharingWindow);
|
|
|
|
|
|
|
|
if (sharingWindow) {
|
|
|
|
// Get the active window streams and see if any of them are "scary".
|
|
|
|
// If so, then we're sharing a browser window.
|
|
|
|
let activeStreams = webrtcUI.getActiveStreams(
|
|
|
|
false /* camera */,
|
|
|
|
false /* microphone */,
|
|
|
|
false /* screen */,
|
|
|
|
true /* window */
|
2016-03-04 19:12:40 +03:00
|
|
|
);
|
2020-05-18 21:57:40 +03:00
|
|
|
let hasBrowserWindow = activeStreams.some(stream => {
|
|
|
|
return stream.devices.some(device => device.scary);
|
|
|
|
});
|
|
|
|
|
|
|
|
this.updateWindowAttr("sharingbrowserwindow", hasBrowserWindow);
|
|
|
|
this.sharingBrowserWindow = hasBrowserWindow;
|
2014-08-01 17:15:49 +04:00
|
|
|
} else {
|
2020-05-18 21:57:40 +03:00
|
|
|
this.updateWindowAttr("sharingbrowserwindow");
|
|
|
|
this.sharingBrowserWindow = false;
|
2014-08-01 17:15:49 +04:00
|
|
|
}
|
2020-05-18 21:57:40 +03:00
|
|
|
|
2020-06-16 14:20:57 +03:00
|
|
|
// The label that's displayed when sharing a display followed a priority.
|
|
|
|
// The more "risky" we deem the display is for sharing, the higher priority.
|
|
|
|
// This gives us the following priorities, from highest to lowest.
|
|
|
|
//
|
|
|
|
// 1. Screen
|
|
|
|
// 2. Browser window
|
|
|
|
// 3. Other application window
|
|
|
|
// 4. Browser tab (unimplemented)
|
|
|
|
//
|
|
|
|
// The CSS for the indicator does the work of showing or hiding these labels
|
|
|
|
// for us, but we need to update the aria-labelledby attribute on the container
|
|
|
|
// of those labels to make it clearer for screenreaders which one the user cares
|
|
|
|
// about.
|
|
|
|
let displayShare = document.getElementById("display-share");
|
|
|
|
let labelledBy;
|
|
|
|
if (sharingScreen) {
|
|
|
|
labelledBy = "screen-share-info";
|
|
|
|
} else if (this.sharingBrowserWindow) {
|
|
|
|
labelledBy = "browser-window-share-info";
|
|
|
|
} else if (sharingWindow) {
|
|
|
|
labelledBy = "window-share-info";
|
|
|
|
}
|
|
|
|
displayShare.setAttribute("aria-labelledby", labelledBy);
|
|
|
|
|
2020-07-08 12:34:39 +03:00
|
|
|
if (window.windowState != window.STATE_MINIMIZED) {
|
|
|
|
// Resize and ensure the window position is correct
|
|
|
|
// (sizeToContent messes with our position).
|
|
|
|
let docElStyle = document.documentElement.style;
|
|
|
|
docElStyle.minWidth = docElStyle.maxWidth = "unset";
|
|
|
|
docElStyle.minHeight = docElStyle.maxHeight = "unset";
|
|
|
|
window.sizeToContent();
|
|
|
|
|
|
|
|
// On Linux GTK, the style of window we're using by default is resizable. We
|
|
|
|
// workaround this by setting explicit limits on the height and width of the
|
|
|
|
// window.
|
|
|
|
if (AppConstants.platform == "linux") {
|
|
|
|
let { width, height } = window.windowUtils.getBoundsWithoutFlushing(
|
|
|
|
document.documentElement
|
|
|
|
);
|
|
|
|
|
|
|
|
docElStyle.minWidth = docElStyle.maxWidth = `${width}px`;
|
|
|
|
docElStyle.minHeight = docElStyle.maxHeight = `${height}px`;
|
|
|
|
}
|
2020-06-17 20:04:51 +03:00
|
|
|
|
2020-07-08 12:34:39 +03:00
|
|
|
this.ensureOnScreen();
|
2020-05-18 21:57:40 +03:00
|
|
|
|
2020-07-08 12:34:39 +03:00
|
|
|
if (!this.positionCustomized) {
|
2020-08-11 13:03:07 +03:00
|
|
|
this.centerOnLatestBrowser();
|
2020-07-08 12:34:39 +03:00
|
|
|
}
|
2020-05-18 21:57:40 +03:00
|
|
|
}
|
2020-07-08 12:34:39 +03:00
|
|
|
|
2020-05-18 21:57:40 +03:00
|
|
|
this.updatingIndicatorState = false;
|
2014-08-01 17:15:49 +04:00
|
|
|
},
|
2020-05-18 21:57:40 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
* After the indicator has been updated, checks to see if it has expanded
|
|
|
|
* such that part of the indicator is now outside of the screen. If so,
|
|
|
|
* it then adjusts the position to put the entire indicator on screen.
|
|
|
|
*/
|
|
|
|
ensureOnScreen() {
|
|
|
|
let desiredX = Math.max(window.screenX, screen.availLeft);
|
2014-08-01 17:15:49 +04:00
|
|
|
let maxX =
|
|
|
|
screen.availLeft +
|
|
|
|
screen.availWidth -
|
|
|
|
document.documentElement.clientWidth;
|
2020-05-18 21:57:40 +03:00
|
|
|
window.moveTo(Math.min(desiredX, maxX), window.screenY);
|
2014-08-01 17:15:49 +04:00
|
|
|
},
|
|
|
|
|
2020-05-18 21:57:40 +03:00
|
|
|
/**
|
2020-08-11 13:03:07 +03:00
|
|
|
* If the indicator is first being opened, we'll find the browser window
|
|
|
|
* associated with the most recent share, and pin the indicator to the
|
|
|
|
* very top of the content area.
|
2020-05-18 21:57:40 +03:00
|
|
|
*/
|
2020-08-11 13:03:07 +03:00
|
|
|
centerOnLatestBrowser() {
|
|
|
|
let activeStreams = webrtcUI.getActiveStreams(
|
|
|
|
true /* camera */,
|
|
|
|
true /* microphone */,
|
|
|
|
true /* screen */,
|
|
|
|
true /* window */
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!activeStreams.length) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
let browser = activeStreams[activeStreams.length - 1].browser;
|
|
|
|
let browserWindow = browser.ownerGlobal;
|
|
|
|
let browserRect = browserWindow.windowUtils.getBoundsWithoutFlushing(
|
|
|
|
browser
|
|
|
|
);
|
|
|
|
|
2020-05-18 21:57:40 +03:00
|
|
|
// This should be called in initialize right after we've just called
|
|
|
|
// updateIndicatorState. Since updateIndicatorState uses
|
|
|
|
// window.sizeToContent, the layout information should be up to date,
|
|
|
|
// and so the numbers that we get without flushing should be sufficient.
|
2020-08-11 13:03:07 +03:00
|
|
|
let { width: windowWidth } = window.windowUtils.getBoundsWithoutFlushing(
|
|
|
|
document.documentElement
|
|
|
|
);
|
2020-06-16 16:19:16 +03:00
|
|
|
|
2020-06-12 21:59:59 +03:00
|
|
|
window.moveTo(
|
2020-08-11 13:03:07 +03:00
|
|
|
browserWindow.mozInnerScreenX +
|
|
|
|
browserRect.left +
|
|
|
|
(browserRect.width - windowWidth) / 2,
|
|
|
|
browserWindow.mozInnerScreenY + browserRect.top
|
2020-06-12 21:59:59 +03:00
|
|
|
);
|
2020-05-18 21:57:40 +03:00
|
|
|
},
|
|
|
|
|
|
|
|
handleEvent(event) {
|
|
|
|
switch (event.type) {
|
|
|
|
case "load": {
|
2020-05-26 21:00:36 +03:00
|
|
|
this.onLoad();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case "unload": {
|
|
|
|
this.onUnload();
|
2014-08-01 17:15:49 +04:00
|
|
|
break;
|
2020-05-18 21:57:40 +03:00
|
|
|
}
|
|
|
|
case "click": {
|
|
|
|
this.onClick(event);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case "MozUpdateWindowPos": {
|
|
|
|
if (!this.updatingIndicatorState) {
|
|
|
|
// The window moved while not updating the indicator state,
|
|
|
|
// so the user probably moved it.
|
|
|
|
this.positionCustomized = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2020-07-08 12:34:39 +03:00
|
|
|
case "sizemodechange": {
|
|
|
|
if (window.windowState != window.STATE_MINIMIZED) {
|
|
|
|
this.updateIndicatorState();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2020-05-18 21:57:40 +03:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
onLoad() {
|
|
|
|
this.loaded = true;
|
2014-08-01 17:15:49 +04:00
|
|
|
|
2020-08-11 13:03:07 +03:00
|
|
|
this.updateIndicatorState();
|
2020-05-18 21:57:40 +03:00
|
|
|
|
|
|
|
window.addEventListener("click", this);
|
2020-07-08 12:34:39 +03:00
|
|
|
window.addEventListener("sizemodechange", this);
|
2020-05-18 21:57:40 +03:00
|
|
|
window.windowRoot.addEventListener("MozUpdateWindowPos", this);
|
|
|
|
|
|
|
|
// Alert accessibility implementations stuff just changed. We only need to do
|
|
|
|
// this initially, because changes after this will automatically fire alert
|
|
|
|
// events if things change materially.
|
|
|
|
let ev = new CustomEvent("AlertActive", {
|
|
|
|
bubbles: true,
|
|
|
|
cancelable: true,
|
|
|
|
});
|
|
|
|
document.documentElement.dispatchEvent(ev);
|
2020-05-26 21:00:36 +03:00
|
|
|
|
2020-05-18 21:57:40 +03:00
|
|
|
this.loaded = true;
|
|
|
|
},
|
|
|
|
|
2020-05-26 21:00:36 +03:00
|
|
|
onUnload() {
|
|
|
|
if (this.macOSIndicator) {
|
|
|
|
this.macOSIndicator.close();
|
2020-07-17 04:22:10 +03:00
|
|
|
this.macOSIndicator = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!this.isClosingInternally) {
|
|
|
|
// Something has closed the indicator, but it wasn't webrtcUI. This
|
|
|
|
// means we might still have some streams being shared. To protect
|
|
|
|
// the user from unknowingly sharing streams, we shut those streams
|
|
|
|
// down.
|
|
|
|
let activeStreams = webrtcUI.getActiveStreams(
|
|
|
|
true /* camera */,
|
|
|
|
true /* microphone */,
|
|
|
|
true /* screen */,
|
|
|
|
true /* window */
|
|
|
|
);
|
|
|
|
webrtcUI.stopSharingStreams(activeStreams);
|
2020-05-26 21:00:36 +03:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2020-05-18 21:57:40 +03:00
|
|
|
onClick(event) {
|
|
|
|
switch (event.target.id) {
|
2020-08-05 06:38:19 +03:00
|
|
|
case "stop-sharing": {
|
2020-05-18 21:57:40 +03:00
|
|
|
let activeStreams = webrtcUI.getActiveStreams(
|
|
|
|
false /* camera */,
|
|
|
|
false /* microphone */,
|
|
|
|
true /* screen */,
|
2020-08-05 01:00:42 +03:00
|
|
|
true /* window */
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!activeStreams.length) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// getActiveStreams is filtering for streams that have screen
|
|
|
|
// sharing, but those streams might _also_ be sharing other
|
|
|
|
// devices like camera or microphone. This is why we need to
|
|
|
|
// tell stopSharingStreams explicitly which device type we want
|
|
|
|
// to stop.
|
|
|
|
webrtcUI.stopSharingStreams(
|
2020-08-05 06:38:19 +03:00
|
|
|
activeStreams,
|
2020-08-05 01:00:42 +03:00
|
|
|
false /* camera */,
|
|
|
|
false /* microphone */,
|
2020-08-05 06:38:19 +03:00
|
|
|
true /* screen */,
|
2020-07-24 18:14:17 +03:00
|
|
|
true /* window */
|
|
|
|
);
|
2014-08-01 17:15:49 +04:00
|
|
|
break;
|
2020-05-18 21:57:40 +03:00
|
|
|
}
|
2020-08-27 18:39:49 +03:00
|
|
|
case "microphone-button":
|
|
|
|
// Intentional fall-through
|
|
|
|
case "camera-button": {
|
|
|
|
// Revoking the microphone also revokes the camera and vice-versa.
|
|
|
|
let activeStreams = webrtcUI.getActiveStreams(
|
|
|
|
true /* camera */,
|
|
|
|
true /* microphone */,
|
|
|
|
false /* screen */
|
|
|
|
);
|
|
|
|
this.showSharingDoorhanger(activeStreams);
|
2020-08-27 00:31:29 +03:00
|
|
|
break;
|
|
|
|
}
|
2020-08-27 18:39:49 +03:00
|
|
|
case "minimize": {
|
|
|
|
window.minimize();
|
2020-08-27 00:31:29 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2020-05-18 21:57:40 +03:00
|
|
|
/**
|
2020-08-27 18:39:49 +03:00
|
|
|
* Find the most recent share in the set of active streams passed,
|
|
|
|
* and opens up the Permissions Panel for the associated tab to
|
|
|
|
* let the user revoke the streaming permission.
|
2020-05-18 21:57:40 +03:00
|
|
|
*
|
2020-08-27 18:39:49 +03:00
|
|
|
* @param activeStreams (Array<Object>)
|
|
|
|
* An array of streams obtained via webrtcUI.getActiveStreams.
|
2020-05-18 21:57:40 +03:00
|
|
|
*/
|
2020-08-27 18:39:49 +03:00
|
|
|
showSharingDoorhanger(activeStreams) {
|
|
|
|
if (!activeStreams.length) {
|
|
|
|
return;
|
|
|
|
}
|
2020-05-18 21:57:40 +03:00
|
|
|
|
2020-08-27 18:39:49 +03:00
|
|
|
let index = activeStreams.length - 1;
|
|
|
|
webrtcUI.showSharingDoorhanger(activeStreams[index]);
|
2020-05-18 21:57:40 +03:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Updates an attribute on the <window> element.
|
|
|
|
*
|
|
|
|
* @param attr (String)
|
|
|
|
* The name of the attribute to update.
|
|
|
|
* @param value (String?)
|
|
|
|
* A string to set the attribute to. If the value is false-y,
|
|
|
|
* the attribute is removed.
|
|
|
|
*/
|
|
|
|
updateWindowAttr(attr, value) {
|
|
|
|
let docEl = document.documentElement;
|
|
|
|
if (value) {
|
|
|
|
docEl.setAttribute(attr, "true");
|
|
|
|
} else {
|
|
|
|
docEl.removeAttribute(attr);
|
2014-08-01 17:15:49 +04:00
|
|
|
}
|
|
|
|
},
|
2020-07-17 04:22:10 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
* See the documentation on the script global closingInternally() function.
|
|
|
|
*/
|
|
|
|
closingInternally() {
|
|
|
|
this.isClosingInternally = true;
|
|
|
|
},
|
2014-08-01 17:15:49 +04:00
|
|
|
};
|
2020-05-18 21:57:40 +03:00
|
|
|
|
|
|
|
WebRTCIndicator.init();
|