зеркало из https://github.com/mozilla/gecko-dev.git
Bug 1297336 - Preserve device sharing indicator when tearing off a tab, r=johannh.
This commit is contained in:
Родитель
7b4ec0f6cf
Коммит
1a53278e41
|
@ -2663,6 +2663,8 @@
|
||||||
if (aOtherTab.hasAttribute("sharing")) {
|
if (aOtherTab.hasAttribute("sharing")) {
|
||||||
aOurTab.setAttribute("sharing", aOtherTab.getAttribute("sharing"));
|
aOurTab.setAttribute("sharing", aOtherTab.getAttribute("sharing"));
|
||||||
modifiedAttrs.push("sharing");
|
modifiedAttrs.push("sharing");
|
||||||
|
aOurTab._sharingState = aOtherTab._sharingState;
|
||||||
|
webrtcUI.swapBrowserForNotification(otherBrowser, ourBrowser);
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the other tab is pending (i.e. has not been restored, yet)
|
// If the other tab is pending (i.e. has not been restored, yet)
|
||||||
|
|
|
@ -8,3 +8,4 @@ support-files =
|
||||||
skip-if = buildapp == 'mulet' || (os == "linux" && debug) # linux: bug 976544
|
skip-if = buildapp == 'mulet' || (os == "linux" && debug) # linux: bug 976544
|
||||||
[browser_devices_get_user_media_anim.js]
|
[browser_devices_get_user_media_anim.js]
|
||||||
[browser_devices_get_user_media_in_frame.js]
|
[browser_devices_get_user_media_in_frame.js]
|
||||||
|
[browser_devices_get_user_media_tear_off_tab.js]
|
||||||
|
|
|
@ -0,0 +1,106 @@
|
||||||
|
/* 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/. */
|
||||||
|
|
||||||
|
registerCleanupFunction(function() {
|
||||||
|
gBrowser.removeCurrentTab();
|
||||||
|
});
|
||||||
|
|
||||||
|
var gTests = [
|
||||||
|
|
||||||
|
{
|
||||||
|
desc: "getUserMedia: tearing-off a tab keeps sharing indicators",
|
||||||
|
run: function* checkTearingOff() {
|
||||||
|
let promise = promisePopupNotificationShown("webRTC-shareDevices");
|
||||||
|
yield promiseRequestDevice(true, true);
|
||||||
|
yield promise;
|
||||||
|
yield expectObserverCalled("getUserMedia:request");
|
||||||
|
checkDeviceSelectors(true, true);
|
||||||
|
|
||||||
|
let indicator = promiseIndicatorWindow();
|
||||||
|
yield promiseMessage("ok", () => {
|
||||||
|
PopupNotifications.panel.firstChild.button.click();
|
||||||
|
});
|
||||||
|
yield expectObserverCalled("getUserMedia:response:allow");
|
||||||
|
yield expectObserverCalled("recording-device-events");
|
||||||
|
is((yield getMediaCaptureState()), "CameraAndMicrophone",
|
||||||
|
"expected camera and microphone to be shared");
|
||||||
|
|
||||||
|
yield indicator;
|
||||||
|
yield checkSharingUI({video: true, audio: true});
|
||||||
|
|
||||||
|
info("tearing off the tab");
|
||||||
|
let win = gBrowser.replaceTabWithWindow(gBrowser.selectedTab);
|
||||||
|
yield whenDelayedStartupFinished(win);
|
||||||
|
yield checkSharingUI({audio: true, video: true}, win);
|
||||||
|
|
||||||
|
// Clicking the global sharing indicator should open the control center in
|
||||||
|
// the second window.
|
||||||
|
ok(win.gIdentityHandler._identityPopup.hidden, "control center should be hidden");
|
||||||
|
let activeStreams = webrtcUI.getActiveStreams(true, false, false);
|
||||||
|
webrtcUI.showSharingDoorhanger(activeStreams[0], "Devices");
|
||||||
|
ok(!win.gIdentityHandler._identityPopup.hidden,
|
||||||
|
"control center should be open in the second window");
|
||||||
|
ok(gIdentityHandler._identityPopup.hidden,
|
||||||
|
"control center should be hidden in the first window");
|
||||||
|
win.gIdentityHandler._identityPopup.hidden = true;
|
||||||
|
|
||||||
|
// Closing the new window should remove all sharing indicators.
|
||||||
|
// We need to load the content script in the first window so that we can
|
||||||
|
// catch the notifications fired globally when closing the second window.
|
||||||
|
gBrowser.selectedBrowser.messageManager.loadFrameScript(CONTENT_SCRIPT_HELPER, true);
|
||||||
|
yield BrowserTestUtils.closeWindow(win);
|
||||||
|
|
||||||
|
if ((yield promiseTodoObserverNotCalled("recording-device-events")) == 1) {
|
||||||
|
todo(false, "Got the 'recording-device-events' notification twice, likely because of bug 962719");
|
||||||
|
}
|
||||||
|
|
||||||
|
yield expectObserverCalled("recording-window-ended");
|
||||||
|
yield expectNoObserverCalled();
|
||||||
|
yield checkNotSharing();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
];
|
||||||
|
|
||||||
|
function test() {
|
||||||
|
waitForExplicitFinish();
|
||||||
|
|
||||||
|
// An empty tab where we can load the content script without leaving it
|
||||||
|
// behind at the end of the test.
|
||||||
|
gBrowser.addTab();
|
||||||
|
|
||||||
|
let tab = gBrowser.addTab();
|
||||||
|
gBrowser.selectedTab = tab;
|
||||||
|
let browser = tab.linkedBrowser;
|
||||||
|
|
||||||
|
browser.messageManager.loadFrameScript(CONTENT_SCRIPT_HELPER, true);
|
||||||
|
|
||||||
|
browser.addEventListener("load", function onload() {
|
||||||
|
browser.removeEventListener("load", onload, true);
|
||||||
|
|
||||||
|
is(PopupNotifications._currentNotifications.length, 0,
|
||||||
|
"should start the test without any prior popup notification");
|
||||||
|
ok(gIdentityHandler._identityPopup.hidden,
|
||||||
|
"should start the test with the control center hidden");
|
||||||
|
|
||||||
|
Task.spawn(function* () {
|
||||||
|
yield SpecialPowers.pushPrefEnv({"set": [[PREF_PERMISSION_FAKE, true]]});
|
||||||
|
|
||||||
|
for (let test of gTests) {
|
||||||
|
info(test.desc);
|
||||||
|
yield test.run();
|
||||||
|
|
||||||
|
// Cleanup before the next test
|
||||||
|
yield expectNoObserverCalled();
|
||||||
|
}
|
||||||
|
}).then(finish, ex => {
|
||||||
|
ok(false, "Unexpected Exception: " + ex);
|
||||||
|
finish();
|
||||||
|
});
|
||||||
|
}, true);
|
||||||
|
let rootDir = getRootDirectory(gTestPath);
|
||||||
|
rootDir = rootDir.replace("chrome://mochitests/content/",
|
||||||
|
"https://example.com/");
|
||||||
|
content.location = rootDir + "get_user_media.html";
|
||||||
|
}
|
|
@ -63,6 +63,18 @@ function promiseWindow(url) {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function whenDelayedStartupFinished(aWindow) {
|
||||||
|
return new Promise(resolve => {
|
||||||
|
info("Waiting for delayed startup to finish");
|
||||||
|
Services.obs.addObserver(function observer(aSubject, aTopic) {
|
||||||
|
if (aWindow == aSubject) {
|
||||||
|
Services.obs.removeObserver(observer, aTopic);
|
||||||
|
resolve();
|
||||||
|
}
|
||||||
|
}, "browser-delayed-startup-finished", false);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
function promiseIndicatorWindow() {
|
function promiseIndicatorWindow() {
|
||||||
// We don't show the indicator window on Mac.
|
// We don't show the indicator window on Mac.
|
||||||
if ("nsISystemStatusBar" in Ci)
|
if ("nsISystemStatusBar" in Ci)
|
||||||
|
@ -407,9 +419,10 @@ function checkDeviceSelectors(aAudio, aVideo) {
|
||||||
ok(cameraSelector.hidden, "camera selector hidden");
|
ok(cameraSelector.hidden, "camera selector hidden");
|
||||||
}
|
}
|
||||||
|
|
||||||
function* checkSharingUI(aExpected) {
|
function* checkSharingUI(aExpected, aWin = window) {
|
||||||
|
let doc = aWin.document;
|
||||||
// First check the icon above the control center (i) icon.
|
// First check the icon above the control center (i) icon.
|
||||||
let identityBox = document.getElementById("identity-box");
|
let identityBox = doc.getElementById("identity-box");
|
||||||
ok(identityBox.hasAttribute("sharing"), "sharing attribute is set");
|
ok(identityBox.hasAttribute("sharing"), "sharing attribute is set");
|
||||||
let sharing = identityBox.getAttribute("sharing");
|
let sharing = identityBox.getAttribute("sharing");
|
||||||
if (aExpected.video)
|
if (aExpected.video)
|
||||||
|
@ -419,7 +432,7 @@ function* checkSharingUI(aExpected) {
|
||||||
|
|
||||||
// Then check the sharing indicators inside the control center panel.
|
// Then check the sharing indicators inside the control center panel.
|
||||||
identityBox.click();
|
identityBox.click();
|
||||||
let permissions = document.getElementById("identity-popup-permission-list");
|
let permissions = doc.getElementById("identity-popup-permission-list");
|
||||||
for (let id of ["microphone", "camera", "screen"]) {
|
for (let id of ["microphone", "camera", "screen"]) {
|
||||||
let convertId = id => {
|
let convertId = id => {
|
||||||
if (id == "camera")
|
if (id == "camera")
|
||||||
|
@ -429,7 +442,7 @@ function* checkSharingUI(aExpected) {
|
||||||
return id;
|
return id;
|
||||||
};
|
};
|
||||||
let expected = aExpected[convertId(id)];
|
let expected = aExpected[convertId(id)];
|
||||||
is(!!gIdentityHandler._sharingState[id], !!expected,
|
is(!!aWin.gIdentityHandler._sharingState[id], !!expected,
|
||||||
"sharing state for " + id + " as expected");
|
"sharing state for " + id + " as expected");
|
||||||
let icon = permissions.querySelectorAll(
|
let icon = permissions.querySelectorAll(
|
||||||
".identity-popup-permission-icon." + id + "-icon");
|
".identity-popup-permission-icon." + id + "-icon");
|
||||||
|
@ -445,7 +458,7 @@ function* checkSharingUI(aExpected) {
|
||||||
is(icon.length, 1, "should not show more than 1 " + id + " icon");
|
is(icon.length, 1, "should not show more than 1 " + id + " icon");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
gIdentityHandler._identityPopup.hidden = true;
|
aWin.gIdentityHandler._identityPopup.hidden = true;
|
||||||
|
|
||||||
// Check the global indicators.
|
// Check the global indicators.
|
||||||
yield* assertWebRTCIndicatorStatus(aExpected);
|
yield* assertWebRTCIndicatorStatus(aExpected);
|
||||||
|
|
Загрузка…
Ссылка в новой задаче