mirror of
https://github.com/roytam1/palemoon27.git
synced 2026-05-26 14:18:48 +00:00
7f8ba9c1d7
- Bug 1236786 - [WebGL2] pass getVertexAttrib in gl-object-get-calls.html, r=jgilbert (60a2c91a38)
- Bug 1233046 - Fix OES_texture_float on OSX. - r=jrmuizel (4bc0059f5f)
- Bug 1233557 - Allow RGB8 to be renderable again for web-compat. - r=jrmuizel (4c13bfd8e8)
- Bug 1233549. Disallow ES3 compressed texture formats. r=jgilbert (1073033161)
- Bug 1241702 - Allow unsized DEPTH_STENCIL for RBs in WebGL 2. - r=kamidphish (87d17d2cf9)
- Bug 1239126. Handle gl_InstanceID attribute with no location. r=jgilbert (4894997e98)
- Bug 1236782 - [WebGL2] pass getProgramParameter in gl-object-get-calls.html; r=jgilbert (2136fcce48)
- Bug 1232462. Only ask for a higher version of GLSL when using WebGL2. r=jgilbert (0317be4eb4)
- Bug 1242330 - "Four extensions were promoted to core in WebGL 2 and should no longer be available as extensions." r=jgilbert r=jmuizelaar (6df020b8d4)
- Bug 1233626 - Default MaxDrawingBuffers to 1 unless ext/webgl2. - r=jrmuizel (a7580d661c)
- Bug 1231657. Don't allow linking different versions shaders. r=jgilbert (e610f98066)
- Bug 1241777 - TexCompareFunc should be stored in ascending order. r=jgilbert (b6151a0076)
- Bug 1228885 - Implement WebGLTexture::MemoryUsage. - r=kamidphish (ea06815414)
- Bug 1239259 - Fix WebGL2 generateMipmap checking. r=jgilbert (39f587c421)
- Bug 1242347 - Allow unsized internal format when generate mipmap. r=jgilbert (b203a8898c)
- Bug 1232502. Use the correct internalFormat when calling CopyTexImage2D. r=jgilbert (eeaef3215e)
- Bug 1243663 - Max uniform and attribute location lengths in WebGL2 should be 1024. r=jgilbert (c4ec6de507)
- Bug 1239488 - Add int/uint to vertex attrib data type. r=jgilbert (11b4968025)
- Bug 1184242 - Remove aTabParent != sActiveTabParent warning from IMEStateManager::SetInputContextForChildProcess. r=masayuki (0fcda10e15)
- Bug 1178652 - Send NOTIFY_IME_OF_COMPOSITION_UPDATE to parent process correctly. r=masayuki (bce28e2c91)
- Bug 1107782 - Only accept certain mouse, gamepad events as user-active. r=smaug (00542c80b9)
- Bug 1247850 - Shrink NameTableKey in nsStaticCaseInsensitiveNameTable. r=froydnj,erahm. (ce3cb3edfb)
- Bug 1247359 - micro-optimize the common case of String{Begins,End}With; r=erahm (333e042b31)
- Bug 1239125. Add operator!=(char_type*) to nsTSubstring. r=froydnj (0cc047a9a1)
- Bug 1213862 - Align nsString whitespace handling with web specs; r=froydnj (db5b11ca52)
- Bug 1141884 - Trigger compositor smooth scrolling to snap points when APZ is enabled. r=mstange,kip (593af59f2a)
- Bug 1244582: Add back in a null check that was accidentally removed. r=smaug (76bff1b01f)
- Bug 1234176 - Introduce and use the WriteSysFile() helper function. r=dhylands (22a46fbe8b)
- missing bit of Bug 1198124 - Enable -Wshadow (f84535a7a2)
- Bug 1249171 - Simplify nsCOMArray::SizeOfExcludingThis(). r=erahm. (57efdce1c6)
- Bug 1156416 - Validate camera parameters supplied by the application. r=mikeh (f8b4b84ccf)
- Bug 1186808 - Replace nsBaseHashtable::EnumerateRead() calls in dom/camera/ with iterators. r=mikeh. (7b1db5f6a1)
- Bug 1158378 - Fix how a failed set configuration call would try to shutdown the camera after release. (9d5e323bca)
- Bug 1171374 - Permit software video codecs with the emulated camera. r=sotaro (c1ae26ea0d)
- Bug 1234458 P1 Allow the CacheChild to be "locked" into memory so it will delay destruction. r=ehsan a=ritu (9e46185779)
- Bug 1234458 P2 Lock the CacheChild actor while Cache DOM methods are running. r=ehsan a=ritu (038342a6e2)
- Bug 1244764 P1 Make Cache .add()/.addAll() fail if a Response.ok() is false. r=ehsan (ae26ca9ef1)
- Bug 1172562 - Clear QuotaManager storage when uninstalling an app. Test. r=bkelly (b07311a3b7)
- Bug 1172629 - Use the caches global property from an iframe loaded after setting the pref in order to make the tests pass with the pref disabled; r=bkelly a=RyanVM (e7c05d8b79)
- Bug 1244764 P2 Make dom/cache mochitests pass with new add()/addAll() behavior. r=ehsan (e1f667c1b4)
- Bug 1244764 P3 Make service worker tests pass with new Cache add()/addAll() behavior. r=ehsan (1518ae5225)
- Bug 1003860 - Simplify storage setup tasks in storage inspector tests. r=mratcliffe (249a8bdb2b)
- Bug 1003860 - Service worker cache for storage actor. r=mratcliffe (5c3d1ecd0c)
- Bug 1244764 P5 Fix devtools test to work with new Cache add()/addAll() behavior. r=ehsan (bf85405de8)
- Bug 1232901 - Use channel.asyncOpen2 within dom/browser-element/BrowserElementParent.js (r=sicking,aus) (2a228ed551)
- Bug 1180330 - http auth prompt shown when opening browser if prompt canceled/dismissed earlier. r=fabrice (ba3666f4bd)
- Bug 1234118 - Delete code for supporting 'do-command' and 'copypaste-docommand'. r=mtseng, r=smaug (b1b575d3c5)
- Bug 1238883 - [TV Browser] It shows "The page cannot be displayed" when user browse some webpages. r=roc (e6d7739dd6)
- Bug 1238440 - FileReader should throw an error when the blob changed size when reading, r=khuey (b006adba10)
- Bug 1230422 - FileReader should handle nested ReadAs*() calls. r=khuey (5a3ff84a31)
- Bug 1225202, part 3 - Create files in test_fileapi_slice.html using SpecialPowers.createFiles. r=baku (1137975548)
- Bug 1241171 - FormData should not force 'blob' as filename, r=smaug (748055f751)
- Bug 1246375 - Restore the previous spec version of FormData, r=smaug (3586af2b88)
- Bug 1237183 - Modify implementation of reading preference. r=seanlin (a132bc7246)
- Bug 801545 - Remove DocumentType.internalSubset, r=bz (ea30c9b5ee)
- Bug 1226440 - Expose a method to get a node's immediate dominator; r=bz,sfink (f77ae44037)
- Bug 825318 - Implement adoptDownload for mozDownloadManager, r=aus, r=sicking (e98cb05210)
- Bug 1237370 - Always log the reason for remote AppRep lookup failures. r=gcp (2c804e68fc)
- Bug 1167493 - Application Reputation: disable remote lookup of zip files on Mac/Linux, r=gcp (517459e064)
- Bug 1195519 - Use channel->ascynOpen2 toolkit/components/downloads/ApplicationReputation.cpp (r=sicking) (2856e5213a)
- Bug 1237856 - Add prefs to honor/ignore Application Reputation verdicts. r=gcp (54ee06264f)
- Bug 1243643 - Deprecate unsafe CPOW usage in contentAreaUtils' saveImage. r=jld (6ae790f1ef)
- Bug 1229224: Add an eslint plugin for importing all browser.js globals for browser-chrome tests. r=miker (9df52a7f3b)
- Bug 1245916: Add additional browser window scripts to eslint globals. r=felipe (92d316ca5e)
- Bug 1246244 - Allow non-CPOW documents to pass through saveImageURL properly. r=jaws,Margaret (c8d4ca241d)
- some missing bits after world fix (c0439eebb0)
- add some missing stuff (ddbd47dc03)
- bissing bit of 1229519 (4e255c3dae)
- Bug 1199662 - Crash ping environment block is broken when any string field contains a quotation mark. Unescape INI fields properly using the library that already exists for the purpose. r=ted (874a999edc)
- Bug 1216150 - Turn on the experimental Intl.DateTimeFormat.prototype.formatToParts in b2g certified apps. r=fabrice (40eeb1a4d4)
- Bug 1216150 - Mini-bustage fix for something I think I unintentionally qref'd into the final patch. r=bustage in a CLOSED TREE (36d9b21a67)
- Bug 1141311 - Add async mode support to GonkNativeWindow on Lollipop Gonk r=pchang (39d9d56326)
- Bug 1146671 - Ensure camera not already released when performing operations. r=dhylands (71b59caa1f)
- Bug 1248737. Improve documentation for WorkerRunnable and associated classes. r=khuey (4ff57790c5)
- Bug 1235629 - Remove dead code in WorkerFeature.h, r=smaug (75a51fcf03)
- Bug 1212333 - WorkerDebuggerManager should live on the main thread;r=khuey (11fdfbbae6)
- Bug 1226443 P3 Re-enable service worker update wpt tests. r=ehsan (605dac5f9e)
- Bug 1226443 P4 Cleanup ServiceWorkerScriptCache objects when initialization fails. r=ehsan (43de3429a2)
- Bug 1234127: Change |BluetoothAdapter.pairingReqs| as a nullable object; r=btian, r=mrbkap (45d2038f6a)
- Bug 1188487 - BrowserElement webidl changes for muting and setting volume. r=ehsan (21bea70a07)
- Bug 1238210 - Correct the Promise return types on two Clients methods; r=baku (fa41b25df0)
- Bug 1246784 - Expose Console to the WorkerDebuggerGlobalScope - part 2, r=khuey (0da9ce8ff6)
- Bug 1228702. Don't expose the 'location' property of Exception/DOMException on workers. r=bholley (0fe86ea586)
- Bug 1223825 - Change Directory.path to include the directory's name. r=baku (0cdae4c2f0)
- Bug 1238225 - Mark ExtendableMessageEvent.ports as SameObject; r=baku (45b9a9746f)
- Bug 1236933 - Return null from FetchEvent.clientId for non-subresource network requests; r=bkelly (4a9c4b40cb)
- Bug 1238213 - Make FetchEvent.request non-nullable; r=baku (751082c8ba)
- Bug 1193125 - Avoid corrupting image data in test_fetch_event.html. r=bkelly (9f6bff232f)
- Bug 1201664 - Avoid using Request's constructor when creating FetchEvent.request; r=bkelly (7a3401e345)
- Bug 1175944 - Packaged app's (app://) JS files are not loaded and do not trigger "onfetch" handler. r=jdm (62df139153)
- Bug 1233644 - use pattern matching when listening clear-origin-data. r=baku (ea2594f50e)
- Bug 1237363 - Part 1: Unregister all service workers registered in mochitests at the end of the test; r=jdm (5be97e5bb0)
- Bug 1237363 - Part 2: Fail mochitests which register a service worker without unregistering it; r=jdm (c4160ffd5f)
- Bug 1237363 - Part 3: Add a test for a mochitest finishing without unregistering its service worker; r=jdm (911d37291b)
- Bug 1174078 - Calling "fetch" inside Service Worker's "onfetch" handler in b2g causes "onfetch" again that leads to an infinite loop. Test. r=nsm (208451f346)
- Bug 1197379 - Remove support for intercepting app:// URIs using service workers; r=jdm (3cbdd725f1)
- Bug 1179399 - Part 1: Relax the ShouldIntercept checks when overriding JAR channel info; r=jdm (850bb2bdb8)
- Bug 1238213 follow-up: Mark the FetchEventInit dictionary argument to FetchEvent's constructor optional too; r=bzbarsky (356cbe6db7)
- Bug 1232732 - modify NS_WARNING in MOZ_WIN_MEM_TRY_CATCH; r=aklotz (e2be4d6919)
- Bug 1247658 - Expose a method to JS for find the shortest retaining paths of some nodes in a heap snapshot; r=bz r=jimb (2c82198808)
- Bug 1188115: Expose IDBCursorWithValue in workers. r=baku (e1c40aeb6e)
- Bug 1162680 - Notify Keyboard.jsm to send blur event when the message manager is closed first. r=timdream (53727ab300)
- Bug 1192986 Also mark Cache/CacheStorage as release interfaces on workers. r=ehsan a=bustage (25cf83c154)
- Bug 1159742. Get rid of the pref annotation from test_interfaces, since it basically corresponds to disabling the test. r=jst (c229e3f881)
- Bug 1203160 - Part 2: Fix the interfaces tests to allow SW interfaces for non-release Fennec; r=baku (072840db1f)
- Bug 1197700 - Correct mistakes in InputMethod.webidl. r=kanru, r=janjongboom, sr=smaug (4edb6f201f)
- Bug 1206970 - Stop expecting AnimationPlaybackEvent to be exposed on release branches, where it's disabled by pref, r=smaug (30ae2b13db)
- Bug 1177276 - Pref on canvas.captureStream by default. r=smaug,mt (0cfe0f72f2)
- Bug 1215147 - Enable VR API's on FF for Android by default. r=snorp, r=vlad, r=bz (5ff3725318)
- Bug 1218482 - Enable WebVR By Default,r=bz (f26111ed82)
- Bug 1159755. Stop forcing the media.eme.apiVisible preference to be true in our test harness. r=cpearce (09f7887917)
- Bug 1149312 - Obtain test coverage for the file-backed case of MediaRecorder. r=roc (bd2e7e40f0)
- Bug 1154559 - Remove flaky timeouts from manifest.js and register SimpleTest.registerCleanupFunction() to report unfinished tests. r=cpearce. (eb68db0fb2)
- Bug 1154564 - Add the ability to notify timeouts to MediaTestManager and remove flaky timeouts from test_playback.html. r=cpearce. (c89b4e58d9)
- Bug 1135170 - Fix up racey test_seek-1.html. rpending=mattwoodrow (b3a7d0dcd6)
- Bug 902686 - Change manifest.js to use SpecialPowers.pushPrefEnv. r=edwin (636b0edc1a)
- Bug 1183502 - give androidVersion a correct value in manifest.js. r=sotaro. (933e9ea712)
- Bug 1235588 - add null check to SimpleTest. r=bechen. (958ede68de)
- misspatch (c8922447ff)
- Bug 1151740 - pass the callback object as-is to SpecialPowers.exactGC(). r=edwin (99ca873bce)
- Bug 1197682 - InputMethodManager#setSupportsSwitchingTypes, r=janjongboom, sr=smaug (e7eb54e491)
- Bug 1201407 - Add input-manage-only events for InputMethod API. r=janjongboom, sr=smaug (776d064bd1)
- Bug 1234459 - Expose full text in the input box to InputMethod API, r=masayuki, sr=smaug (4fa0554356)
- Bug 1198163 - Workaround Mochitest app and assign frame proper permissions, r=kanru (c3bcf8ecc1)
- Bug 990250 - Fold nsIStyleSheet into CSSStyleSheet. r=dbaron (23579cb300)
683 lines
22 KiB
JavaScript
683 lines
22 KiB
JavaScript
/* 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/. */
|
|
/* vim: set ft=javascript : */
|
|
|
|
"use strict";
|
|
|
|
var Cu = Components.utils;
|
|
var Ci = Components.interfaces;
|
|
var Cc = Components.classes;
|
|
var Cr = Components.results;
|
|
var Cm = Components.manager.QueryInterface(Ci.nsIComponentRegistrar);
|
|
|
|
this.EXPORTED_SYMBOLS = ["BrowserElementPromptService"];
|
|
|
|
Cu.import("resource://gre/modules/XPCOMUtils.jsm");
|
|
Cu.import("resource://gre/modules/Services.jsm");
|
|
|
|
const NS_PREFBRANCH_PREFCHANGE_TOPIC_ID = "nsPref:changed";
|
|
const BROWSER_FRAMES_ENABLED_PREF = "dom.mozBrowserFramesEnabled";
|
|
|
|
function debug(msg) {
|
|
//dump("BrowserElementPromptService - " + msg + "\n");
|
|
}
|
|
|
|
function BrowserElementPrompt(win, browserElementChild) {
|
|
this._win = win;
|
|
this._browserElementChild = browserElementChild;
|
|
}
|
|
|
|
BrowserElementPrompt.prototype = {
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIPrompt]),
|
|
|
|
alert: function(title, text) {
|
|
this._browserElementChild.showModalPrompt(
|
|
this._win, {promptType: "alert", title: title, message: text, returnValue: undefined});
|
|
},
|
|
|
|
alertCheck: function(title, text, checkMsg, checkState) {
|
|
// Treat this like a normal alert() call, ignoring the checkState. The
|
|
// front-end can do its own suppression of the alert() if it wants.
|
|
this.alert(title, text);
|
|
},
|
|
|
|
confirm: function(title, text) {
|
|
return this._browserElementChild.showModalPrompt(
|
|
this._win, {promptType: "confirm", title: title, message: text, returnValue: undefined});
|
|
},
|
|
|
|
confirmCheck: function(title, text, checkMsg, checkState) {
|
|
return this.confirm(title, text);
|
|
},
|
|
|
|
// Each button is described by an object with the following schema
|
|
// {
|
|
// string messageType, // 'builtin' or 'custom'
|
|
// string message, // 'ok', 'cancel', 'yes', 'no', 'save', 'dontsave',
|
|
// // 'revert' or a string from caller if messageType was 'custom'.
|
|
// }
|
|
//
|
|
// Expected result from embedder:
|
|
// {
|
|
// int button, // Index of the button that user pressed.
|
|
// boolean checked, // True if the check box is checked.
|
|
// }
|
|
confirmEx: function(title, text, buttonFlags, button0Title, button1Title,
|
|
button2Title, checkMsg, checkState) {
|
|
let buttonProperties = this._buildConfirmExButtonProperties(buttonFlags,
|
|
button0Title,
|
|
button1Title,
|
|
button2Title);
|
|
let defaultReturnValue = { selectedButton: buttonProperties.defaultButton };
|
|
if (checkMsg) {
|
|
defaultReturnValue.checked = checkState.value;
|
|
}
|
|
let ret = this._browserElementChild.showModalPrompt(
|
|
this._win,
|
|
{
|
|
promptType: "custom-prompt",
|
|
title: title,
|
|
message: text,
|
|
defaultButton: buttonProperties.defaultButton,
|
|
buttons: buttonProperties.buttons,
|
|
showCheckbox: !!checkMsg,
|
|
checkboxMessage: checkMsg,
|
|
checkboxCheckedByDefault: !!checkState.value,
|
|
returnValue: defaultReturnValue
|
|
}
|
|
);
|
|
if (checkMsg) {
|
|
checkState.value = ret.checked;
|
|
}
|
|
return buttonProperties.indexToButtonNumberMap[ret.selectedButton];
|
|
},
|
|
|
|
prompt: function(title, text, value, checkMsg, checkState) {
|
|
let rv = this._browserElementChild.showModalPrompt(
|
|
this._win,
|
|
{ promptType: "prompt",
|
|
title: title,
|
|
message: text,
|
|
initialValue: value.value,
|
|
returnValue: null });
|
|
|
|
value.value = rv;
|
|
|
|
// nsIPrompt::Prompt returns true if the user pressed "OK" at the prompt,
|
|
// and false if the user pressed "Cancel".
|
|
//
|
|
// BrowserElementChild returns null for "Cancel" and returns the string the
|
|
// user entered otherwise.
|
|
return rv !== null;
|
|
},
|
|
|
|
promptUsernameAndPassword: function(title, text, username, password, checkMsg, checkState) {
|
|
throw Cr.NS_ERROR_NOT_IMPLEMENTED;
|
|
},
|
|
|
|
promptPassword: function(title, text, password, checkMsg, checkState) {
|
|
throw Cr.NS_ERROR_NOT_IMPLEMENTED;
|
|
},
|
|
|
|
select: function(title, text, aCount, aSelectList, aOutSelection) {
|
|
throw Cr.NS_ERROR_NOT_IMPLEMENTED;
|
|
},
|
|
|
|
_buildConfirmExButtonProperties: function(buttonFlags, button0Title,
|
|
button1Title, button2Title) {
|
|
let r = {
|
|
defaultButton: -1,
|
|
buttons: [],
|
|
// This map is for translating array index to the button number that
|
|
// is recognized by Gecko. This shouldn't be exposed to embedder.
|
|
indexToButtonNumberMap: []
|
|
};
|
|
|
|
let defaultButton = 0; // Default to Button 0.
|
|
if (buttonFlags & Ci.nsIPrompt.BUTTON_POS_1_DEFAULT) {
|
|
defaultButton = 1;
|
|
} else if (buttonFlags & Ci.nsIPrompt.BUTTON_POS_2_DEFAULT) {
|
|
defaultButton = 2;
|
|
}
|
|
|
|
// Properties of each button.
|
|
let buttonPositions = [
|
|
Ci.nsIPrompt.BUTTON_POS_0,
|
|
Ci.nsIPrompt.BUTTON_POS_1,
|
|
Ci.nsIPrompt.BUTTON_POS_2
|
|
];
|
|
|
|
function buildButton(buttonTitle, buttonNumber) {
|
|
let ret = {};
|
|
let buttonPosition = buttonPositions[buttonNumber];
|
|
let mask = 0xff * buttonPosition; // 8 bit mask
|
|
let titleType = (buttonFlags & mask) / buttonPosition;
|
|
|
|
ret.messageType = 'builtin';
|
|
switch(titleType) {
|
|
case Ci.nsIPrompt.BUTTON_TITLE_OK:
|
|
ret.message = 'ok';
|
|
break;
|
|
case Ci.nsIPrompt.BUTTON_TITLE_CANCEL:
|
|
ret.message = 'cancel';
|
|
break;
|
|
case Ci.nsIPrompt.BUTTON_TITLE_YES:
|
|
ret.message = 'yes';
|
|
break;
|
|
case Ci.nsIPrompt.BUTTON_TITLE_NO:
|
|
ret.message = 'no';
|
|
break;
|
|
case Ci.nsIPrompt.BUTTON_TITLE_SAVE:
|
|
ret.message = 'save';
|
|
break;
|
|
case Ci.nsIPrompt.BUTTON_TITLE_DONT_SAVE:
|
|
ret.message = 'dontsave';
|
|
break;
|
|
case Ci.nsIPrompt.BUTTON_TITLE_REVERT:
|
|
ret.message = 'revert';
|
|
break;
|
|
case Ci.nsIPrompt.BUTTON_TITLE_IS_STRING:
|
|
ret.message = buttonTitle;
|
|
ret.messageType = 'custom';
|
|
break;
|
|
default:
|
|
// This button is not shown.
|
|
return;
|
|
}
|
|
|
|
// If this is the default button, set r.defaultButton to
|
|
// the index of this button in the array. This value is going to be
|
|
// exposed to the embedder.
|
|
if (defaultButton === buttonNumber) {
|
|
r.defaultButton = r.buttons.length;
|
|
}
|
|
r.buttons.push(ret);
|
|
r.indexToButtonNumberMap.push(buttonNumber);
|
|
}
|
|
|
|
buildButton(button0Title, 0);
|
|
buildButton(button1Title, 1);
|
|
buildButton(button2Title, 2);
|
|
|
|
// If defaultButton is still -1 here, it means the default button won't
|
|
// be shown.
|
|
if (r.defaultButton === -1) {
|
|
throw new Components.Exception("Default button won't be shown",
|
|
Cr.NS_ERROR_FAILURE);
|
|
}
|
|
|
|
return r;
|
|
},
|
|
};
|
|
|
|
|
|
function BrowserElementAuthPrompt() {
|
|
}
|
|
|
|
BrowserElementAuthPrompt.prototype = {
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIAuthPrompt2]),
|
|
|
|
promptAuth: function promptAuth(channel, level, authInfo) {
|
|
throw Cr.NS_ERROR_NOT_IMPLEMENTED;
|
|
},
|
|
|
|
asyncPromptAuth: function asyncPromptAuth(channel, callback, context, level, authInfo) {
|
|
debug("asyncPromptAuth");
|
|
|
|
// The cases that we don't support now.
|
|
if ((authInfo.flags & Ci.nsIAuthInformation.AUTH_PROXY) &&
|
|
(authInfo.flags & Ci.nsIAuthInformation.ONLY_PASSWORD)) {
|
|
throw Cr.NS_ERROR_FAILURE;
|
|
}
|
|
|
|
let frame = this._getFrameFromChannel(channel);
|
|
if (!frame) {
|
|
debug("Cannot get frame, asyncPromptAuth fail");
|
|
throw Cr.NS_ERROR_FAILURE;
|
|
}
|
|
|
|
let browserElementParent =
|
|
BrowserElementPromptService.getBrowserElementParentForFrame(frame);
|
|
|
|
if (!browserElementParent) {
|
|
debug("Failed to load browser element parent.");
|
|
throw Cr.NS_ERROR_FAILURE;
|
|
}
|
|
|
|
let consumer = {
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsICancelable]),
|
|
callback: callback,
|
|
context: context,
|
|
cancel: function() {
|
|
this.callback.onAuthCancelled(this.context, false);
|
|
this.callback = null;
|
|
this.context = null;
|
|
}
|
|
};
|
|
|
|
let [hostname, httpRealm] = this._getAuthTarget(channel, authInfo);
|
|
let hashKey = level + "|" + hostname + "|" + httpRealm;
|
|
let asyncPrompt = this._asyncPrompts[hashKey];
|
|
if (asyncPrompt) {
|
|
asyncPrompt.consumers.push(consumer);
|
|
return consumer;
|
|
}
|
|
|
|
asyncPrompt = {
|
|
consumers: [consumer],
|
|
channel: channel,
|
|
authInfo: authInfo,
|
|
level: level,
|
|
inProgress: false,
|
|
browserElementParent: browserElementParent
|
|
};
|
|
|
|
this._asyncPrompts[hashKey] = asyncPrompt;
|
|
this._doAsyncPrompt();
|
|
return consumer;
|
|
},
|
|
|
|
// Utilities for nsIAuthPrompt2 ----------------
|
|
|
|
_asyncPrompts: {},
|
|
_asyncPromptInProgress: new WeakMap(),
|
|
_doAsyncPrompt: function() {
|
|
// Find the key of a prompt whose browser element parent does not have
|
|
// async prompt in progress.
|
|
let hashKey = null;
|
|
for (let key in this._asyncPrompts) {
|
|
let prompt = this._asyncPrompts[key];
|
|
if (!this._asyncPromptInProgress.get(prompt.browserElementParent)) {
|
|
hashKey = key;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Didn't find an available prompt, so just return.
|
|
if (!hashKey)
|
|
return;
|
|
|
|
let prompt = this._asyncPrompts[hashKey];
|
|
let [hostname, httpRealm] = this._getAuthTarget(prompt.channel,
|
|
prompt.authInfo);
|
|
|
|
this._asyncPromptInProgress.set(prompt.browserElementParent, true);
|
|
prompt.inProgress = true;
|
|
|
|
let self = this;
|
|
let callback = function(ok, username, password) {
|
|
debug("Async auth callback is called, ok = " +
|
|
ok + ", username = " + username);
|
|
|
|
// Here we got the username and password provided by embedder, or
|
|
// ok = false if the prompt was cancelled by embedder.
|
|
delete self._asyncPrompts[hashKey];
|
|
prompt.inProgress = false;
|
|
self._asyncPromptInProgress.delete(prompt.browserElementParent);
|
|
|
|
// Fill authentication information with username and password provided
|
|
// by user.
|
|
let flags = prompt.authInfo.flags;
|
|
if (username) {
|
|
if (flags & Ci.nsIAuthInformation.NEED_DOMAIN) {
|
|
// Domain is separated from username by a backslash
|
|
let idx = username.indexOf("\\");
|
|
if (idx == -1) {
|
|
prompt.authInfo.username = username;
|
|
} else {
|
|
prompt.authInfo.domain = username.substring(0, idx);
|
|
prompt.authInfo.username = username.substring(idx + 1);
|
|
}
|
|
} else {
|
|
prompt.authInfo.username = username;
|
|
}
|
|
}
|
|
|
|
if (password) {
|
|
prompt.authInfo.password = password;
|
|
}
|
|
|
|
for (let consumer of prompt.consumers) {
|
|
if (!consumer.callback) {
|
|
// Not having a callback means that consumer didn't provide it
|
|
// or canceled the notification.
|
|
continue;
|
|
}
|
|
|
|
try {
|
|
if (ok) {
|
|
debug("Ok, calling onAuthAvailable to finish auth");
|
|
consumer.callback.onAuthAvailable(consumer.context, prompt.authInfo);
|
|
} else {
|
|
debug("Cancelled, calling onAuthCancelled to finish auth.");
|
|
consumer.callback.onAuthCancelled(consumer.context, true);
|
|
}
|
|
} catch (e) { /* Throw away exceptions caused by callback */ }
|
|
}
|
|
|
|
// Process the next prompt, if one is pending.
|
|
self._doAsyncPrompt();
|
|
};
|
|
|
|
let runnable = {
|
|
run: function() {
|
|
// Call promptAuth of browserElementParent, to show the prompt.
|
|
prompt.browserElementParent.promptAuth(
|
|
self._createAuthDetail(prompt.channel, prompt.authInfo),
|
|
callback);
|
|
}
|
|
}
|
|
|
|
Services.tm.currentThread.dispatch(runnable, Ci.nsIThread.DISPATCH_NORMAL);
|
|
},
|
|
|
|
_getFrameFromChannel: function(channel) {
|
|
let loadContext = channel.notificationCallbacks.getInterface(Ci.nsILoadContext);
|
|
return loadContext.topFrameElement;
|
|
},
|
|
|
|
_createAuthDetail: function(channel, authInfo) {
|
|
let [hostname, httpRealm] = this._getAuthTarget(channel, authInfo);
|
|
return {
|
|
host: hostname,
|
|
path: channel.URI.path,
|
|
realm: httpRealm,
|
|
username: authInfo.username,
|
|
isProxy: !!(authInfo.flags & Ci.nsIAuthInformation.AUTH_PROXY),
|
|
isOnlyPassword: !!(authInfo.flags & Ci.nsIAuthInformation.ONLY_PASSWORD)
|
|
};
|
|
},
|
|
|
|
// The code is taken from nsLoginManagerPrompter.js, with slight
|
|
// modification for parameter name consistency here.
|
|
_getAuthTarget : function (channel, authInfo) {
|
|
let hostname, realm;
|
|
|
|
// If our proxy is demanding authentication, don't use the
|
|
// channel's actual destination.
|
|
if (authInfo.flags & Ci.nsIAuthInformation.AUTH_PROXY) {
|
|
if (!(channel instanceof Ci.nsIProxiedChannel))
|
|
throw new Error("proxy auth needs nsIProxiedChannel");
|
|
|
|
let info = channel.proxyInfo;
|
|
if (!info)
|
|
throw new Error("proxy auth needs nsIProxyInfo");
|
|
|
|
// Proxies don't have a scheme, but we'll use "moz-proxy://"
|
|
// so that it's more obvious what the login is for.
|
|
var idnService = Cc["@mozilla.org/network/idn-service;1"].
|
|
getService(Ci.nsIIDNService);
|
|
hostname = "moz-proxy://" +
|
|
idnService.convertUTF8toACE(info.host) +
|
|
":" + info.port;
|
|
realm = authInfo.realm;
|
|
if (!realm)
|
|
realm = hostname;
|
|
|
|
return [hostname, realm];
|
|
}
|
|
|
|
hostname = this._getFormattedHostname(channel.URI);
|
|
|
|
// If a HTTP WWW-Authenticate header specified a realm, that value
|
|
// will be available here. If it wasn't set or wasn't HTTP, we'll use
|
|
// the formatted hostname instead.
|
|
realm = authInfo.realm;
|
|
if (!realm)
|
|
realm = hostname;
|
|
|
|
return [hostname, realm];
|
|
},
|
|
|
|
/**
|
|
* Strip out things like userPass and path for display.
|
|
*/
|
|
_getFormattedHostname : function(uri) {
|
|
return uri.scheme + "://" + uri.hostPort;
|
|
},
|
|
};
|
|
|
|
|
|
function AuthPromptWrapper(oldImpl, browserElementImpl) {
|
|
this._oldImpl = oldImpl;
|
|
this._browserElementImpl = browserElementImpl;
|
|
}
|
|
|
|
AuthPromptWrapper.prototype = {
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIAuthPrompt2]),
|
|
promptAuth: function(channel, level, authInfo) {
|
|
if (this._canGetParentElement(channel)) {
|
|
return this._browserElementImpl.promptAuth(channel, level, authInfo);
|
|
} else {
|
|
return this._oldImpl.promptAuth(channel, level, authInfo);
|
|
}
|
|
},
|
|
|
|
asyncPromptAuth: function(channel, callback, context, level, authInfo) {
|
|
if (this._canGetParentElement(channel)) {
|
|
return this._browserElementImpl.asyncPromptAuth(channel, callback, context, level, authInfo);
|
|
} else {
|
|
return this._oldImpl.asyncPromptAuth(channel, callback, context, level, authInfo);
|
|
}
|
|
},
|
|
|
|
_canGetParentElement: function(channel) {
|
|
try {
|
|
let context = channel.notificationCallbacks.getInterface(Ci.nsILoadContext);
|
|
let frame = context.topFrameElement;
|
|
if (!frame) {
|
|
// This function returns a boolean value
|
|
return !!context.nestedFrameId;
|
|
}
|
|
|
|
if (!BrowserElementPromptService.getBrowserElementParentForFrame(frame))
|
|
return false;
|
|
|
|
return true;
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
}
|
|
};
|
|
|
|
function BrowserElementPromptFactory(toWrap) {
|
|
this._wrapped = toWrap;
|
|
}
|
|
|
|
BrowserElementPromptFactory.prototype = {
|
|
classID: Components.ID("{24f3d0cf-e417-4b85-9017-c9ecf8bb1299}"),
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIPromptFactory]),
|
|
|
|
_mayUseNativePrompt: function() {
|
|
try {
|
|
return Services.prefs.getBoolPref("browser.prompt.allowNative");
|
|
} catch (e) {
|
|
// This properity is default to true.
|
|
return true;
|
|
}
|
|
},
|
|
|
|
_getNativePromptIfAllowed: function(win, iid, err) {
|
|
if (this._mayUseNativePrompt())
|
|
return this._wrapped.getPrompt(win, iid);
|
|
else {
|
|
// Not allowed, throw an exception.
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
getPrompt: function(win, iid) {
|
|
// It is possible for some object to get a prompt without passing
|
|
// valid reference of window, like nsNSSComponent. In such case, we
|
|
// should just fall back to the native prompt service
|
|
if (!win)
|
|
return this._getNativePromptIfAllowed(win, iid, Cr.NS_ERROR_INVALID_ARG);
|
|
|
|
if (iid.number != Ci.nsIPrompt.number &&
|
|
iid.number != Ci.nsIAuthPrompt2.number) {
|
|
debug("We don't recognize the requested IID (" + iid + ", " +
|
|
"allowed IID: " +
|
|
"nsIPrompt=" + Ci.nsIPrompt + ", " +
|
|
"nsIAuthPrompt2=" + Ci.nsIAuthPrompt2 + ")");
|
|
return this._getNativePromptIfAllowed(win, iid, Cr.NS_ERROR_INVALID_ARG);
|
|
}
|
|
|
|
// Try to find a BrowserElementChild for the window.
|
|
let browserElementChild =
|
|
BrowserElementPromptService.getBrowserElementChildForWindow(win);
|
|
|
|
if (iid.number === Ci.nsIAuthPrompt2.number) {
|
|
debug("Caller requests an instance of nsIAuthPrompt2.");
|
|
|
|
if (browserElementChild) {
|
|
// If we are able to get a BrowserElementChild, it means that
|
|
// the auth prompt is for a mozbrowser. Therefore we don't need to
|
|
// fall back.
|
|
return new BrowserElementAuthPrompt().QueryInterface(iid);
|
|
}
|
|
|
|
// Because nsIAuthPrompt2 is called in parent process. If caller
|
|
// wants nsIAuthPrompt2 and we cannot get BrowserElementchild,
|
|
// it doesn't mean that we should fallback. It is possible that we can
|
|
// get the BrowserElementParent from nsIChannel that passed to
|
|
// functions of nsIAuthPrompt2.
|
|
if (this._mayUseNativePrompt()) {
|
|
return new AuthPromptWrapper(
|
|
this._wrapped.getPrompt(win, iid),
|
|
new BrowserElementAuthPrompt().QueryInterface(iid))
|
|
.QueryInterface(iid);
|
|
} else {
|
|
// Falling back is not allowed, so we don't need wrap the
|
|
// BrowserElementPrompt.
|
|
return new BrowserElementAuthPrompt().QueryInterface(iid);
|
|
}
|
|
}
|
|
|
|
if (!browserElementChild) {
|
|
debug("We can't find a browserElementChild for " +
|
|
win + ", " + win.location);
|
|
return this._getNativePromptIfAllowed(win, iid, Cr.NS_ERROR_FAILURE);
|
|
}
|
|
|
|
debug("Returning wrapped getPrompt for " + win);
|
|
return new BrowserElementPrompt(win, browserElementChild)
|
|
.QueryInterface(iid);
|
|
}
|
|
};
|
|
|
|
this.BrowserElementPromptService = {
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIObserver,
|
|
Ci.nsISupportsWeakReference]),
|
|
|
|
_initialized: false,
|
|
|
|
_init: function() {
|
|
if (this._initialized) {
|
|
return;
|
|
}
|
|
|
|
// If the pref is disabled, do nothing except wait for the pref to change.
|
|
if (!this._browserFramesPrefEnabled()) {
|
|
var prefs = Cc["@mozilla.org/preferences-service;1"].getService(Ci.nsIPrefBranch);
|
|
prefs.addObserver(BROWSER_FRAMES_ENABLED_PREF, this, /* ownsWeak = */ true);
|
|
return;
|
|
}
|
|
|
|
this._initialized = true;
|
|
this._browserElementParentMap = new WeakMap();
|
|
|
|
var os = Cc["@mozilla.org/observer-service;1"].getService(Ci.nsIObserverService);
|
|
os.addObserver(this, "outer-window-destroyed", /* ownsWeak = */ true);
|
|
|
|
// Wrap the existing @mozilla.org/prompter;1 implementation.
|
|
var contractID = "@mozilla.org/prompter;1";
|
|
var oldCID = Cm.contractIDToCID(contractID);
|
|
var newCID = BrowserElementPromptFactory.prototype.classID;
|
|
var oldFactory = Cm.getClassObject(Cc[contractID], Ci.nsIFactory);
|
|
|
|
if (oldCID == newCID) {
|
|
debug("WARNING: Wrapped prompt factory is already installed!");
|
|
return;
|
|
}
|
|
|
|
Cm.unregisterFactory(oldCID, oldFactory);
|
|
|
|
var oldInstance = oldFactory.createInstance(null, Ci.nsIPromptFactory);
|
|
var newInstance = new BrowserElementPromptFactory(oldInstance);
|
|
|
|
var newFactory = {
|
|
createInstance: function(outer, iid) {
|
|
if (outer != null) {
|
|
throw Cr.NS_ERROR_NO_AGGREGATION;
|
|
}
|
|
return newInstance.QueryInterface(iid);
|
|
}
|
|
};
|
|
Cm.registerFactory(newCID,
|
|
"BrowserElementPromptService's prompter;1 wrapper",
|
|
contractID, newFactory);
|
|
|
|
debug("Done installing new prompt factory.");
|
|
},
|
|
|
|
_getOuterWindowID: function(win) {
|
|
return win.QueryInterface(Ci.nsIInterfaceRequestor)
|
|
.getInterface(Ci.nsIDOMWindowUtils)
|
|
.outerWindowID;
|
|
},
|
|
|
|
_browserElementChildMap: {},
|
|
mapWindowToBrowserElementChild: function(win, browserElementChild) {
|
|
this._browserElementChildMap[this._getOuterWindowID(win)] = browserElementChild;
|
|
},
|
|
|
|
getBrowserElementChildForWindow: function(win) {
|
|
// We only have a mapping for <iframe mozbrowser>s, not their inner
|
|
// <iframes>, so we look up win.top below. window.top (when called from
|
|
// script) respects <iframe mozbrowser> boundaries.
|
|
return this._browserElementChildMap[this._getOuterWindowID(win.top)];
|
|
},
|
|
|
|
mapFrameToBrowserElementParent: function(frame, browserElementParent) {
|
|
this._browserElementParentMap.set(frame, browserElementParent);
|
|
},
|
|
|
|
getBrowserElementParentForFrame: function(frame) {
|
|
return this._browserElementParentMap.get(frame);
|
|
},
|
|
|
|
_observeOuterWindowDestroyed: function(outerWindowID) {
|
|
let id = outerWindowID.QueryInterface(Ci.nsISupportsPRUint64).data;
|
|
debug("observeOuterWindowDestroyed " + id);
|
|
delete this._browserElementChildMap[outerWindowID.data];
|
|
},
|
|
|
|
_browserFramesPrefEnabled: function() {
|
|
var prefs = Cc["@mozilla.org/preferences-service;1"].getService(Ci.nsIPrefBranch);
|
|
try {
|
|
return prefs.getBoolPref(BROWSER_FRAMES_ENABLED_PREF);
|
|
}
|
|
catch(e) {
|
|
return false;
|
|
}
|
|
},
|
|
|
|
observe: function(subject, topic, data) {
|
|
switch(topic) {
|
|
case NS_PREFBRANCH_PREFCHANGE_TOPIC_ID:
|
|
if (data == BROWSER_FRAMES_ENABLED_PREF) {
|
|
this._init();
|
|
}
|
|
break;
|
|
case "outer-window-destroyed":
|
|
this._observeOuterWindowDestroyed(subject);
|
|
break;
|
|
default:
|
|
debug("Observed unexpected topic " + topic);
|
|
}
|
|
}
|
|
};
|
|
|
|
BrowserElementPromptService._init();
|