PageRenderTime 58ms CodeModel.GetById 20ms RepoModel.GetById 0ms app.codeStats 1ms

/Client/MicrosoftAjax/Extensions/Sys/WebForms/PageRequestManager.js

#
JavaScript | 1261 lines | 836 code | 122 blank | 303 comment | 204 complexity | 7d45dffe4d6be8c33533d81cbd324317 MD5 | raw file
Possible License(s): BSD-3-Clause

Large files files are truncated, but you can click here to view the full file

  1. $type = Sys.WebForms.PageRequestManager = function PageRequestManager() {
  2. this._form = null;
  3. this._activeDefaultButton = null;
  4. this._activeDefaultButtonClicked = false;
  5. this._updatePanelIDs = null;
  6. this._updatePanelClientIDs = null;
  7. this._updatePanelHasChildrenAsTriggers = null;
  8. this._asyncPostBackControlIDs = null;
  9. this._asyncPostBackControlClientIDs = null;
  10. this._postBackControlIDs = null;
  11. this._postBackControlClientIDs = null;
  12. this._scriptManagerID = null;
  13. this._pageLoadedHandler = null;
  14. this._additionalInput = null;
  15. this._onsubmit = null;
  16. this._onSubmitStatements = [];
  17. this._originalDoPostBack = null;
  18. this._originalDoPostBackWithOptions = null;
  19. this._originalFireDefaultButton = null;
  20. this._originalDoCallback = null;
  21. this._isCrossPost = false;
  22. this._postBackSettings = null;
  23. this._request = null;
  24. this._onFormSubmitHandler = null;
  25. this._onFormElementClickHandler = null;
  26. this._onWindowUnloadHandler = null;
  27. this._asyncPostBackTimeout = null;
  28. this._controlIDToFocus = null;
  29. this._scrollPosition = null;
  30. this._processingRequest = false;
  31. this._scriptDisposes = {};
  32. // DevDiv Bugs 161922, 138251:
  33. // List of hidden fields that should be removed if an async update does not
  34. // explictly define it.
  35. this._transientFields = ["__VIEWSTATEENCRYPTED", "__VIEWSTATEFIELDCOUNT"];
  36. }
  37. $type.prototype = {
  38. get_isInAsyncPostBack: function PageRequestManager$get_isInAsyncPostBack() {
  39. /// <value type="Boolean" locid="P:J#Sys.WebForms.PageRequestManager.isInAsyncPostBack"></value>
  40. //#if DEBUG
  41. if (arguments.length !== 0) throw Error.parameterCount();
  42. //#endif
  43. return this._request !== null;
  44. },
  45. // Events
  46. add_beginRequest: function PageRequestManager$add_beginRequest(handler) {
  47. /// <summary locid="E:J#Sys.WebForms.PageRequestManager.beginRequest">Adds a beginRequest event handler.</summary>
  48. //#if DEBUG
  49. var e = Function._validateParams(arguments, [{name: "handler", type: Function}]);
  50. if (e) throw e;
  51. //#endif
  52. Sys.Observer.addEventHandler(this, "beginRequest", handler);
  53. },
  54. remove_beginRequest: function PageRequestManager$remove_beginRequest(handler) {
  55. //#if DEBUG
  56. var e = Function._validateParams(arguments, [{name: "handler", type: Function}]);
  57. if (e) throw e;
  58. //#endif
  59. Sys.Observer.removeEventHandler(this, "beginRequest", handler);
  60. },
  61. add_endRequest: function PageRequestManager$add_endRequest(handler) {
  62. /// <summary locid="E:J#Sys.WebForms.PageRequestManager.endRequest">Adds a endRequest event handler.</summary>
  63. //#if DEBUG
  64. var e = Function._validateParams(arguments, [{name: "handler", type: Function}]);
  65. if (e) throw e;
  66. //#endif
  67. Sys.Observer.addEventHandler(this, "endRequest", handler);
  68. },
  69. remove_endRequest: function PageRequestManager$remove_endRequest(handler) {
  70. //#if DEBUG
  71. var e = Function._validateParams(arguments, [{name: "handler", type: Function}]);
  72. if (e) throw e;
  73. //#endif
  74. Sys.Observer.removeEventHandler(this, "endRequest", handler);
  75. },
  76. add_initializeRequest: function PageRequestManager$add_initializeRequest(handler) {
  77. /// <summary locid="E:J#Sys.WebForms.PageRequestManager.initializeRequest">Adds a initializeRequest event handler.</summary>
  78. //#if DEBUG
  79. var e = Function._validateParams(arguments, [{name: "handler", type: Function}]);
  80. if (e) throw e;
  81. //#endif
  82. Sys.Observer.addEventHandler(this, "initializeRequest", handler);
  83. },
  84. remove_initializeRequest: function PageRequestManager$remove_initializeRequest(handler) {
  85. //#if DEBUG
  86. var e = Function._validateParams(arguments, [{name: "handler", type: Function}]);
  87. if (e) throw e;
  88. //#endif
  89. Sys.Observer.removeEventHandler(this, "initializeRequest", handler);
  90. },
  91. add_pageLoaded: function PageRequestManager$add_pageLoaded(handler) {
  92. /// <summary locid="E:J#Sys.WebForms.PageRequestManager.pageLoaded">Adds a pageLoaded event handler.</summary>
  93. //#if DEBUG
  94. var e = Function._validateParams(arguments, [{name: "handler", type: Function}]);
  95. if (e) throw e;
  96. //#endif
  97. Sys.Observer.addEventHandler(this, "pageLoaded", handler);
  98. },
  99. remove_pageLoaded: function PageRequestManager$remove_pageLoaded(handler) {
  100. //#if DEBUG
  101. var e = Function._validateParams(arguments, [{name: "handler", type: Function}]);
  102. if (e) throw e;
  103. //#endif
  104. Sys.Observer.removeEventHandler(this, "pageLoaded", handler);
  105. },
  106. add_pageLoading: function PageRequestManager$add_pageLoading(handler) {
  107. /// <summary locid="E:J#Sys.WebForms.PageRequestManager.pageLoading">Adds a pageLoading event handler.</summary>
  108. //#if DEBUG
  109. var e = Function._validateParams(arguments, [{name: "handler", type: Function}]);
  110. if (e) throw e;
  111. //#endif
  112. Sys.Observer.addEventHandler(this, "pageLoading", handler);
  113. },
  114. remove_pageLoading: function PageRequestManager$remove_pageLoading(handler) {
  115. //#if DEBUG
  116. var e = Function._validateParams(arguments, [{name: "handler", type: Function}]);
  117. if (e) throw e;
  118. //#endif
  119. Sys.Observer.removeEventHandler(this, "pageLoading", handler);
  120. },
  121. abortPostBack: function PageRequestManager$abortPostBack() {
  122. if (!this._processingRequest && this._request) {
  123. // cancel the request if a response hasn't already been received
  124. this._request.get_executor().abort();
  125. this._request = null;
  126. }
  127. },
  128. beginAsyncPostBack: function PageRequestManager$beginAsyncPostBack(updatePanelsToUpdate, eventTarget, eventArgument, causesValidation, validationGroup) {
  129. /// <summary locid="M:J#Sys.WebForms.PageRequestManager.beginAsyncPostBack">Begins an asynchronous postback.</summary>
  130. /// <param name="updatePanelsToUpdate" type="Array" elementType="String" mayBeNull="true" optional="true">A list of UniqueIDs or ClientIDs of UpdatePanel controls that should have their rendering updated.</param>
  131. /// <param name="eventTarget" type="String" mayBeNull="true" optional="true"></param>
  132. /// <param name="eventArgument" type="String" mayBeNull="true" optional="true"></param>
  133. /// <param name="causesValidation" type="Boolean" mayBeNull="true" optional="true"></param>
  134. /// <param name="validationGroup" type="String" mayBeNull="true" optional="true"></param>
  135. //#if DEBUG
  136. var e = Function._validateParams(arguments, [
  137. {name: "updatePanelsToUpdate", type: Array, mayBeNull: true, optional: true, elementType: String},
  138. {name: "eventTarget", type: String, mayBeNull: true, optional: true},
  139. {name: "eventArgument", type: String, mayBeNull: true, optional: true},
  140. {name: "causesValidation", type: Boolean, mayBeNull: true, optional: true},
  141. {name: "validationGroup", type: String, mayBeNull: true, optional: true}
  142. ]);
  143. if (e) throw e;
  144. //#endif
  145. if (causesValidation && (typeof(Page_ClientValidate) === 'function') && !Page_ClientValidate(validationGroup || null)) {
  146. return;
  147. }
  148. this._postBackSettings = this._createPostBackSettings(true, updatePanelsToUpdate, eventTarget);
  149. var form = this._form;
  150. form.__EVENTTARGET.value = (eventTarget || "");
  151. form.__EVENTARGUMENT.value = (eventArgument || "");
  152. this._isCrossPost = false;
  153. this._additionalInput = null;
  154. this._onFormSubmit();
  155. },
  156. _cancelPendingCallbacks: function PageRequestManager$_cancelPendingCallbacks() {
  157. // DevDiv Bugs 125825: To avoid EVENTVALIDATION corruption we must cancel pending callbacks when an async postback begins
  158. // to cancel callbacks, we run logic similar to WebForm_CallbackComplete,
  159. // except we do not run WebForm_ExecuteCallback for them. This code is exactly
  160. // WebForm_CallbackComplete except without the call to WebForm_ExecuteCallback.
  161. // We are basically treating each callback as completed, ignoring the response if any.
  162. for (var i = 0, l = window.__pendingCallbacks.length; i < l; i++) {
  163. var callback = window.__pendingCallbacks[i];
  164. if (callback) {
  165. if (!callback.async) {
  166. // we just cancelled the single allowed instance of a synchronous callback
  167. window.__synchronousCallBackIndex = -1;
  168. }
  169. window.__pendingCallbacks[i] = null;
  170. var callbackFrameID = "__CALLBACKFRAME" + i;
  171. var xmlRequestFrame = document.getElementById(callbackFrameID);
  172. if (xmlRequestFrame) {
  173. xmlRequestFrame.parentNode.removeChild(xmlRequestFrame);
  174. }
  175. }
  176. }
  177. },
  178. _commitControls: function PageRequestManager$_commitControls(updatePanelData, asyncPostBackTimeout) {
  179. // DevDiv Bugs 154403:
  180. // commits context data parsed from a delta. This is called after an async response
  181. // has proceeded past the script include loading phase, which when it can no longer
  182. // be cancelled and the HTML DOM will be updated.
  183. // _processUpdatePanelArrays is the method that creates these arrays..
  184. // DevDiv Bugs 188564: Update may not have an async postback timeout node and/or updatepanel nodes.
  185. if (updatePanelData) {
  186. this._updatePanelIDs = updatePanelData.updatePanelIDs;
  187. this._updatePanelClientIDs = updatePanelData.updatePanelClientIDs;
  188. this._updatePanelHasChildrenAsTriggers = updatePanelData.updatePanelHasChildrenAsTriggers;
  189. this._asyncPostBackControlIDs = updatePanelData.asyncPostBackControlIDs;
  190. this._asyncPostBackControlClientIDs = updatePanelData.asyncPostBackControlClientIDs;
  191. this._postBackControlIDs = updatePanelData.postBackControlIDs;
  192. this._postBackControlClientIDs = updatePanelData.postBackControlClientIDs;
  193. }
  194. if (typeof(asyncPostBackTimeout) !== 'undefined' && asyncPostBackTimeout !== null) {
  195. this._asyncPostBackTimeout = asyncPostBackTimeout * 1000;
  196. }
  197. },
  198. _createHiddenField: function PageRequestManager$_createHiddenField(id, value) {
  199. // DevDiv Bugs 27075: Creates a hidden field via innerHTML to workaround a caching issue.
  200. var container, field = document.getElementById(id);
  201. if (field) {
  202. // the field already exists
  203. if (!field._isContained) {
  204. // but it is not contained within a SPAN container, so we must create one
  205. // in order to recreate it with innerHTML.
  206. field.parentNode.removeChild(field);
  207. }
  208. else {
  209. // and it already has a container, we'll just set the container innerHTML to replace it,
  210. // which is much faster than removing the element, recreating it, and then setting innerHTML
  211. container = field.parentNode;
  212. }
  213. }
  214. if (!container) {
  215. container = document.createElement('span');
  216. // set display none in case this SPAN would be styled
  217. container.style.cssText = "display:none !important";
  218. this._form.appendChild(container);
  219. }
  220. // now create/replace the input by setting innerHTML
  221. container.innerHTML = "<input type='hidden' />";
  222. field = container.childNodes[0];
  223. // flag it as contained, so if it needs to be removed we know to remove the container, too.
  224. field._isContained = true;
  225. field.id = field.name = id;
  226. field.value = value;
  227. },
  228. _createPageRequestManagerTimeoutError: function PageRequestManager$_createPageRequestManagerTimeoutError() {
  229. // Creates a PageRequestManagerTimeoutException representing a request that timed out.
  230. var displayMessage = "Sys.WebForms.PageRequestManagerTimeoutException: " + Sys.WebForms.Res.PRM_TimeoutError;
  231. var e = Error.create(displayMessage, {name: 'Sys.WebForms.PageRequestManagerTimeoutException'});
  232. e.popStackFrame();
  233. return e;
  234. },
  235. _createPageRequestManagerServerError: function PageRequestManager$_createPageRequestManagerServerError(httpStatusCode, message) {
  236. // Creates a PageRequestManagerServerErrorException representing an error that occurred on the server.
  237. var displayMessage = "Sys.WebForms.PageRequestManagerServerErrorException: " +
  238. (message || String.format(Sys.WebForms.Res.PRM_ServerError, httpStatusCode));
  239. var e = Error.create(displayMessage, {
  240. name: 'Sys.WebForms.PageRequestManagerServerErrorException',
  241. httpStatusCode: httpStatusCode
  242. });
  243. e.popStackFrame();
  244. return e;
  245. },
  246. _createPageRequestManagerParserError: function PageRequestManager$_createPageRequestManagerParserError(parserErrorMessage) {
  247. // Creates a PageRequestManagerParserErrorException representing a parser error that occurred while processing a response from the server.
  248. var displayMessage = "Sys.WebForms.PageRequestManagerParserErrorException: " + String.format(Sys.WebForms.Res.PRM_ParserError, parserErrorMessage);
  249. var e = Error.create(displayMessage, {name: 'Sys.WebForms.PageRequestManagerParserErrorException'});
  250. e.popStackFrame();
  251. return e;
  252. },
  253. _createPanelID: function PageRequestManager$_createPanelID(panelsToUpdate, postBackSettings) {
  254. var asyncTarget = postBackSettings.asyncTarget,
  255. toUpdate = this._ensureUniqueIds(panelsToUpdate || postBackSettings.panelsToUpdate),
  256. panelArg = (toUpdate instanceof Array)
  257. ? toUpdate.join(',')
  258. : (toUpdate || this._scriptManagerID);
  259. if (asyncTarget) {
  260. panelArg += "|" + asyncTarget;
  261. }
  262. return encodeURIComponent(this._scriptManagerID) + '=' + encodeURIComponent(panelArg) + '&';
  263. },
  264. _createPostBackSettings: function PageRequestManager$_createPostBackSettings(async, panelsToUpdate, asyncTarget, sourceElement) {
  265. return { async:async, asyncTarget: asyncTarget, panelsToUpdate: panelsToUpdate, sourceElement: sourceElement };
  266. },
  267. _convertToClientIDs: function PageRequestManager$_convertToClientIDs(source, destinationIDs, destinationClientIDs, version4) {
  268. if (source) {
  269. for (var i = 0, l = source.length; i < l; i += (version4 ? 2 : 1)) {
  270. var uniqueID = source[i],
  271. clientID = (version4 ? source[i+1] : "") || this._uniqueIDToClientID(uniqueID);
  272. Array.add(destinationIDs, uniqueID);
  273. Array.add(destinationClientIDs, clientID);
  274. }
  275. }
  276. },
  277. dispose: function PageRequestManager$dispose() {
  278. Sys.Observer.clearEventHandlers(this);
  279. if (this._form) {
  280. Sys.UI.DomEvent.removeHandler(this._form, 'submit', this._onFormSubmitHandler);
  281. Sys.UI.DomEvent.removeHandler(this._form, 'click', this._onFormElementClickHandler);
  282. Sys.UI.DomEvent.removeHandler(window, 'unload', this._onWindowUnloadHandler);
  283. Sys.UI.DomEvent.removeHandler(window, 'load', this._pageLoadedHandler);
  284. }
  285. if (this._originalDoPostBack) {
  286. window.__doPostBack = this._originalDoPostBack;
  287. this._originalDoPostBack = null;
  288. }
  289. if (this._originalDoPostBackWithOptions) {
  290. window.WebForm_DoPostBackWithOptions = this._originalDoPostBackWithOptions;
  291. this._originalDoPostBackWithOptions = null;
  292. }
  293. if (this._originalFireDefaultButton) {
  294. window.WebForm_FireDefaultButton = this._originalFireDefaultButton;
  295. this._originalFireDefaultButton = null;
  296. }
  297. if (this._originalDoCallback) {
  298. window.WebForm_DoCallback = this._originalDoCallback;
  299. this._originalDoCallback = null;
  300. }
  301. this._form = null;
  302. this._updatePanelIDs = null;
  303. this._updatePanelClientIDs = null;
  304. this._asyncPostBackControlIDs = null;
  305. this._asyncPostBackControlClientIDs = null;
  306. this._postBackControlIDs = null;
  307. this._postBackControlClientIDs = null;
  308. this._asyncPostBackTimeout = null;
  309. this._scrollPosition = null;
  310. },
  311. _doCallback: function PageRequestManager$_doCallback(eventTarget, eventArgument, eventCallback, context, errorCallback, useAsync) {
  312. // DevDiv Bugs 125825: Do not allow callbacks to begin while an async postback is in progress to prevent EVENTVALIDATION corruption
  313. if (!this.get_isInAsyncPostBack()) {
  314. this._originalDoCallback(eventTarget, eventArgument, eventCallback, context, errorCallback, useAsync);
  315. }
  316. },
  317. // New implementation of __doPostBack
  318. _doPostBack: function PageRequestManager$_doPostBack(eventTarget, eventArgument) {
  319. this._additionalInput = null;
  320. var form = this._form;
  321. if ((eventTarget === null) || (typeof(eventTarget) === "undefined") || (this._isCrossPost)) {
  322. // Allow the default form submit to take place. Since it's a cross-page postback.
  323. // DevDiv 80942: we should fall to a full postback if event target is null or undefined
  324. this._postBackSettings = this._createPostBackSettings(false);
  325. // set to false so subsequent posts that don't go through DPWO aren't considered cross post
  326. this._isCrossPost = false;
  327. }
  328. else {
  329. var mpUniqueID = this._masterPageUniqueID;
  330. // If it's not a cross-page post, see if we can find the DOM element that caused the postback
  331. var clientID = this._uniqueIDToClientID(eventTarget);
  332. var postBackElement = document.getElementById(clientID);
  333. if (!postBackElement && mpUniqueID) {
  334. if (clientID.indexOf(mpUniqueID + "$") === 0) {
  335. // With ClientIDMode=Predictable, the MasterPageID is missing from the beginning
  336. // of the client ID.
  337. postBackElement = document.getElementById(clientID.substr(mpUniqueID.length + 1));
  338. }
  339. }
  340. if (!postBackElement) {
  341. // If the control has no matching DOM element we look for an exact
  342. // match from RegisterAsyncPostBackControl or RegisterPostBackControl.
  343. // If we can't find anything about it then we do a search based on
  344. // naming containers to still try and find a match.
  345. if (Array.contains(this._asyncPostBackControlIDs, eventTarget)) {
  346. // Exact match for async postback
  347. this._postBackSettings = this._createPostBackSettings(true, null, eventTarget);
  348. }
  349. else {
  350. if (Array.contains(this._postBackControlIDs, eventTarget)) {
  351. // Exact match for regular postback
  352. this._postBackSettings = this._createPostBackSettings(false);
  353. }
  354. else {
  355. // Find nearest element based on UniqueID in case the element calling
  356. // __doPostBack doesn't have an ID. GridView does this for its Update
  357. // button and without this we can't do async postbacks.
  358. var nearestUniqueIDMatch = this._findNearestElement(eventTarget);
  359. if (nearestUniqueIDMatch) {
  360. // We found a related parent element, so walk up the DOM to find out what kind
  361. // of postback we should do.
  362. this._postBackSettings = this._getPostBackSettings(nearestUniqueIDMatch, eventTarget);
  363. }
  364. else {
  365. // the control may have rendered without the master page id prefix due to ClientIDMode = Predictable
  366. // For example "ctl00$ctl00$foo$Button1" may have rendered an id of "foo_button1" if "ctl00$ctl00"
  367. // is the ID for a 2-level nested master page.
  368. // try stripping it from the beginning of the ID and trying again
  369. if (mpUniqueID) {
  370. mpUniqueID += "$";
  371. if (eventTarget.indexOf(mpUniqueID) === 0) {
  372. nearestUniqueIDMatch = this._findNearestElement(eventTarget.substr(mpUniqueID.length));
  373. }
  374. }
  375. if (nearestUniqueIDMatch) {
  376. // We found a related parent element, so walk up the DOM to find out what kind
  377. // of postback we should do.
  378. this._postBackSettings = this._getPostBackSettings(nearestUniqueIDMatch, eventTarget);
  379. }
  380. else {
  381. // Can't find any DOM element at all related to the eventTarget,
  382. // so we just give up and do a regular postback.
  383. this._postBackSettings = this._createPostBackSettings(false);
  384. }
  385. }
  386. }
  387. }
  388. }
  389. else {
  390. // The element was found, so walk up the DOM to find out what kind
  391. // of postback we should do.
  392. this._postBackSettings = this._getPostBackSettings(postBackElement, eventTarget);
  393. }
  394. }
  395. if (!this._postBackSettings.async) {
  396. // Temporarily restore the form's onsubmit handler expando while calling
  397. // the original ASP.NET 2.0 __doPostBack() function.
  398. form.onsubmit = this._onsubmit;
  399. this._originalDoPostBack(eventTarget, eventArgument);
  400. form.onsubmit = null;
  401. return;
  402. }
  403. form.__EVENTTARGET.value = eventTarget;
  404. form.__EVENTARGUMENT.value = eventArgument;
  405. this._onFormSubmit();
  406. },
  407. _doPostBackWithOptions: function PageRequestManager$_doPostBackWithOptions(options) {
  408. this._isCrossPost = options && options.actionUrl;
  409. // note that when DoPostBackWithOptions is used, _doPostBack or _onFormSubmit, one of the two,
  410. // are guaranteed to be called next.
  411. // In both of those methods it is important to clear the isCrossPost flag so subsequent posts that
  412. // don't use DoPostBackWithOptions are not considered cross page posts.
  413. this._originalDoPostBackWithOptions(options);
  414. },
  415. _elementContains: function PageRequestManager$_elementContains(container, element) {
  416. while (element) {
  417. if (element === container) {
  418. return true;
  419. }
  420. element = element.parentNode;
  421. }
  422. return false;
  423. },
  424. _endPostBack: function PageRequestManager$_endPostBack(error, executor, data) {
  425. if (this._request === executor.get_webRequest()) {
  426. // the postback being ended is the one being processed
  427. this._processingRequest = false;
  428. this._additionalInput = null;
  429. this._request = null;
  430. }
  431. var eventArgs = new Sys.WebForms.EndRequestEventArgs(error, data ? data.dataItems : {}, executor);
  432. Sys.Observer.raiseEvent(this, "endRequest", eventArgs);
  433. if (error && !eventArgs.get_errorHandled()) {
  434. // DevDiv 89485: throw, don't alert()
  435. throw error;
  436. }
  437. },
  438. _ensureUniqueIds: function PageRequestManager$_ensureUniqueIds(ids) {
  439. // given a single ID or an array of IDs that might be ClientIDs or UniqueIDs,
  440. // returns a list of the UniqueIDs. This is used from createPanelID and makes
  441. // it so beginAsyncPostBack and the panelstoUpdate property of the event args
  442. // supports a list of update panels by UniqueID or ClientID.
  443. // If an ID is not a ClientID we assume it is a UniqueID even though it could just be
  444. // that it does not exist.
  445. if (!ids) return ids;
  446. ids = ids instanceof Array ? ids : [ids];
  447. var uniqueIds = [];
  448. for (var i = 0, l = ids.length; i < l; i++) {
  449. var id = ids[i], index = Array.indexOf(this._updatePanelClientIDs, id);
  450. uniqueIds.push(index > -1 ? this._updatePanelIDs[index] : id);
  451. }
  452. return uniqueIds;
  453. },
  454. // Finds the nearest element to the given UniqueID. If an element is not
  455. // found for the exact UniqueID, it walks up the parent chain to look for it.
  456. _findNearestElement: function PageRequestManager$_findNearestElement(uniqueID) {
  457. while (uniqueID.length > 0) {
  458. var clientID = this._uniqueIDToClientID(uniqueID);
  459. var element = document.getElementById(clientID);
  460. if (element) {
  461. return element;
  462. }
  463. var indexOfLastDollar = uniqueID.lastIndexOf('$');
  464. if (indexOfLastDollar === -1) {
  465. return null;
  466. }
  467. uniqueID = uniqueID.substring(0, indexOfLastDollar);
  468. }
  469. return null;
  470. },
  471. _findText: function PageRequestManager$_findText(text, location) {
  472. var startIndex = Math.max(0, location - 20);
  473. var endIndex = Math.min(text.length, location + 20);
  474. return text.substring(startIndex, endIndex);
  475. },
  476. _fireDefaultButton: function PageRequestManager$_fireDefaultButton(event, target) {
  477. // This is a copy of the function WebForm_FireDefaultButton as defined in WebForms.js.
  478. // The purpose is to hook into the WebForm_FireDefaultButton call with the code commented in the middle.
  479. // Other than that, there have been a few minor changes to the code but the logic is the same.
  480. if (event.keyCode === 13) {
  481. var src = event.srcElement || event.target;
  482. if (!src || (src.tagName.toLowerCase() !== "textarea")) {
  483. var defaultButton = document.getElementById(target);
  484. if (defaultButton && (typeof(defaultButton.click) !== "undefined")) {
  485. // Beginning of new code...
  486. // In all but FF this causes the form.onclick event to fire with the button as the event target.
  487. // In FF the the form.onclick event has the current focus control as the target, which prevents the
  488. // default button's server-side click event from firing. So we ensure the correct control is determined
  489. // to have caused the postback by saving the default button before clicking on it. The code in
  490. // onFormSubmit looks for this field and ensures the postback target is the button.
  491. this._activeDefaultButton = defaultButton;
  492. this._activeDefaultButtonClicked = false;
  493. try {
  494. // click is synchronous -- it will immediately cause a form onclick event and then a form onsubmit event
  495. // assuming nothing uses preventDefault() to cancel the event.
  496. defaultButton.click();
  497. }
  498. finally {
  499. // form submission may or may not be occuring after this point
  500. this._activeDefaultButton = null;
  501. }
  502. // ...End of new code
  503. // cancel submission caused by hitting enter in the input control
  504. event.cancelBubble = true;
  505. if (typeof(event.stopPropagation) === "function") {
  506. event.stopPropagation();
  507. }
  508. return false;
  509. }
  510. }
  511. }
  512. return true;
  513. },
  514. _getPageLoadedEventArgs: function PageRequestManager$_getPageLoadedEventArgs(initialLoad, data) {
  515. // -------------+------------------------------------+-----------------------
  516. // Situation | In ID collections | In eventArg property
  517. // -------------+------------------------------------+-----------------------
  518. // Update (exp) | in panelsToRefresh | updated
  519. // Update (imp) | in new, in old, in childUP | created
  520. // Create (exp) | in new, not in old, not in childUP | created
  521. // Create (imp) | in new, not in old, in childUP | created
  522. // Delete (exp) | not in new, in old, not in childUP | ---
  523. // Delete (imp) | not in new, in old, in childUP | ---
  524. // -------------+------------------------------------+-----------------------
  525. // (exp) = explicit
  526. // (imp) = implicit (happened as result of parent UpdatePanel updating)
  527. // --------------------------------------------------------------------------
  528. // in panelsToRefresh = updated
  529. // not updated, in new = created
  530. // else = don't care
  531. // --------------------------------------------------------------------------
  532. var updated = [];
  533. var created = [];
  534. var version4 = data ? data.version4 : false;
  535. var upData = data ? data.updatePanelData : null;
  536. // All panels before update,
  537. // All panels after update,
  538. // Child panels created after update,
  539. // Parent panels created after update
  540. var newIDs, newClientIDs, childIDs, refreshedIDs;
  541. if (!upData) {
  542. // this is the initial load, consider the initialized update panels
  543. // to be the newly created ones
  544. newIDs = this._updatePanelIDs;
  545. newClientIDs = this._updatePanelClientIDs;
  546. childIDs = null;
  547. refreshedIDs = null;
  548. }
  549. else {
  550. newIDs = upData.updatePanelIDs;
  551. newClientIDs = upData.updatePanelClientIDs;
  552. childIDs = upData.childUpdatePanelIDs;
  553. refreshedIDs = upData.panelsToRefreshIDs;
  554. }
  555. var i, l, uniqueID, clientID;
  556. // in panelsToRefresh = updated
  557. if (refreshedIDs) {
  558. for (i = 0, l = refreshedIDs.length; i < l; i += (version4 ? 2 : 1)) {
  559. uniqueID = refreshedIDs[i];
  560. clientID = (version4 ? refreshedIDs[i+1] : "") || this._uniqueIDToClientID(uniqueID);
  561. Array.add(updated, document.getElementById(clientID));
  562. }
  563. }
  564. // If the panel is in the new list and it is either the initial load
  565. // of the page a refreshed child, it is 'created'.
  566. for (i = 0, l = newIDs.length; i < l; i++) {
  567. if (initialLoad || Array.indexOf(childIDs, newIDs[i]) !== -1) {
  568. Array.add(created, document.getElementById(newClientIDs[i]));
  569. }
  570. }
  571. return new Sys.WebForms.PageLoadedEventArgs(updated, created, data ? data.dataItems : {});
  572. },
  573. _getPageLoadingEventArgs: function PageRequestManager$_getPageLoadingEventArgs(data) {
  574. // -------------+------------------------------------+-----------------------
  575. // Situation | In ID collections | In eventArg property
  576. // -------------+------------------------------------+-----------------------
  577. // Update (exp) | in panelsToRefresh | updated
  578. // Update (imp) | in old, in new, in childUP | deleted
  579. // Create (exp) | not in old, in new, not in childUP | ---
  580. // Create (imp) | not in old, in new, in childUP | ---
  581. // Delete (exp) | in old, not in new, not in childUP | deleted
  582. // Delete (imp) | in old, not in new, in childUP | deleted
  583. // -------------+------------------------------------+-----------------------
  584. // (exp) = explicit
  585. // (imp) = implicit (happened as result of parent UpdatePanel updating)
  586. // --------------------------------------------------------------------------
  587. // in panelsToRefresh = updated
  588. // not updated, (not in new or in childUP) = deleted
  589. // else = don't care
  590. // --------------------------------------------------------------------------
  591. var updated = [],
  592. deleted = [],
  593. upData = data.updatePanelData,
  594. oldIDs = upData.oldUpdatePanelIDs,
  595. oldClientIDs = upData.oldUpdatePanelClientIDs,
  596. newIDs = upData.updatePanelIDs,
  597. childIDs = upData.childUpdatePanelIDs,
  598. refreshedIDs = upData.panelsToRefreshIDs,
  599. i, l, uniqueID, clientID,
  600. version4 = data.version4;
  601. // in panelsToRefresh = updated
  602. for (i = 0, l = refreshedIDs.length; i < l; i += (version4 ? 2 : 1)) {
  603. uniqueID = refreshedIDs[i];
  604. clientID = (version4 ? refreshedIDs[i+1] : "") || this._uniqueIDToClientID(uniqueID);
  605. Array.add(updated, document.getElementById(clientID));
  606. }
  607. // not in new or in childUP = deleted
  608. for (i = 0, l = oldIDs.length; i < l; i++) {
  609. uniqueID = oldIDs[i];
  610. if (Array.indexOf(refreshedIDs, uniqueID) === -1 &&
  611. (Array.indexOf(newIDs, uniqueID) === -1 || Array.indexOf(childIDs, uniqueID) > -1)) {
  612. Array.add(deleted, document.getElementById(oldClientIDs[i]));
  613. }
  614. }
  615. return new Sys.WebForms.PageLoadingEventArgs(updated, deleted, data.dataItems);
  616. },
  617. _getPostBackSettings: function PageRequestManager$_getPostBackSettings(element, elementUniqueID) {
  618. var originalElement = element;
  619. // Keep track of whether we have an AsyncPostBackControl but still
  620. // want to see if we're inside an UpdatePanel anyway.
  621. var proposedSettings = null;
  622. // Walk up DOM hierarchy to find out the nearest container of
  623. // the element that caused the postback.
  624. while (element) {
  625. if (element.id) {
  626. // First try an exact match for async postback, regular postback, or UpdatePanel
  627. if (!proposedSettings && Array.contains(this._asyncPostBackControlClientIDs, element.id)) {
  628. // The element explicitly causes an async postback
  629. proposedSettings = this._createPostBackSettings(true, null, elementUniqueID, originalElement);
  630. }
  631. else {
  632. if (!proposedSettings && Array.contains(this._postBackControlClientIDs, element.id)) {
  633. // The element explicitly doesn't cause an async postback
  634. return this._createPostBackSettings(false);
  635. }
  636. else {
  637. var indexOfPanel = Array.indexOf(this._updatePanelClientIDs, element.id);
  638. if (indexOfPanel !== -1) {
  639. // The element causes an async postback because it is inside an UpdatePanel
  640. if (this._updatePanelHasChildrenAsTriggers[indexOfPanel]) {
  641. // If it was in an UpdatePanel and the panel has ChildrenAsTriggers=true, then
  642. // we do an async postback and refresh the given panel
  643. // Although we do the search by looking at ClientIDs, we end
  644. // up sending a UniqueID back to the server so that we can
  645. // call FindControl() with it.
  646. return this._createPostBackSettings(true, [this._updatePanelIDs[indexOfPanel]], elementUniqueID, originalElement);
  647. }
  648. else {
  649. // The element was inside an UpdatePanel so we do an async postback,
  650. // but because it has ChildrenAsTriggers=false we don't update this panel.
  651. return this._createPostBackSettings(true, null, elementUniqueID, originalElement);
  652. }
  653. }
  654. }
  655. }
  656. // Then try near matches
  657. if (!proposedSettings && this._matchesParentIDInList(element.id, this._asyncPostBackControlClientIDs)) {
  658. // The element explicitly causes an async postback
  659. proposedSettings = this._createPostBackSettings(true, null, elementUniqueID, originalElement);
  660. }
  661. else {
  662. if (!proposedSettings && this._matchesParentIDInList(element.id, this._postBackControlClientIDs)) {
  663. // The element explicitly doesn't cause an async postback
  664. return this._createPostBackSettings(false);
  665. }
  666. }
  667. }
  668. element = element.parentNode;
  669. }
  670. // If we have proposed settings that means we found a match for an
  671. // AsyncPostBackControl but were still searching for an UpdatePanel.
  672. // If we got here that means we didn't find the UpdatePanel so we
  673. // just fall back to the original AsyncPostBackControl settings that
  674. // we created.
  675. if (!proposedSettings) {
  676. // The element doesn't cause an async postback
  677. return this._createPostBackSettings(false);
  678. }
  679. else {
  680. return proposedSettings;
  681. }
  682. },
  683. _getScrollPosition: function PageRequestManager$_getScrollPosition() {
  684. var d = document.documentElement;
  685. if (d && (this._validPosition(d.scrollLeft) || this._validPosition(d.scrollTop))) {
  686. return {
  687. x: d.scrollLeft,
  688. y: d.scrollTop
  689. };
  690. }
  691. else {
  692. d = document.body;
  693. if (d && (this._validPosition(d.scrollLeft) || this._validPosition(d.scrollTop))) {
  694. return {
  695. x: d.scrollLeft,
  696. y: d.scrollTop
  697. };
  698. }
  699. else {
  700. if (this._validPosition(window.pageXOffset) || this._validPosition(window.pageYOffset)) {
  701. return {
  702. x: window.pageXOffset,
  703. y: window.pageYOffset
  704. };
  705. }
  706. else {
  707. return {
  708. x: 0,
  709. y: 0
  710. };
  711. }
  712. }
  713. }
  714. },
  715. _initializeInternal: function PageRequestManager$_initializeInternal(scriptManagerID, formElement, updatePanelIDs, asyncPostBackControlIDs, postBackControlIDs, asyncPostBackTimeout, masterPageUniqueID) {
  716. if (this._prmInitialized) {
  717. throw Error.invalidOperation(Sys.WebForms.Res.PRM_CannotRegisterTwice);
  718. }
  719. this._prmInitialized = true;
  720. this._masterPageUniqueID = masterPageUniqueID;
  721. this._scriptManagerID = scriptManagerID;
  722. this._form = Sys.UI.DomElement.resolveElement(formElement);
  723. this._onsubmit = this._form.onsubmit;
  724. this._form.onsubmit = null;
  725. this._onFormSubmitHandler = Function.createDelegate(this, this._onFormSubmit);
  726. this._onFormElementClickHandler = Function.createDelegate(this, this._onFormElementClick);
  727. this._onWindowUnloadHandler = Function.createDelegate(this, this._onWindowUnload);
  728. Sys.UI.DomEvent.addHandler(this._form, 'submit', this._onFormSubmitHandler);
  729. Sys.UI.DomEvent.addHandler(this._form, 'click', this._onFormElementClickHandler);
  730. Sys.UI.DomEvent.addHandler(window, 'unload', this._onWindowUnloadHandler);
  731. this._originalDoPostBack = window.__doPostBack;
  732. if (this._originalDoPostBack) {
  733. window.__doPostBack = Function.createDelegate(this, this._doPostBack);
  734. }
  735. this._originalDoPostBackWithOptions = window.WebForm_DoPostBackWithOptions;
  736. if (this._originalDoPostBackWithOptions) {
  737. window.WebForm_DoPostBackWithOptions = Function.createDelegate(this, this._doPostBackWithOptions);
  738. }
  739. this._originalFireDefaultButton = window.WebForm_FireDefaultButton;
  740. if (this._originalFireDefaultButton) {
  741. window.WebForm_FireDefaultButton = Function.createDelegate(this, this._fireDefaultButton);
  742. }
  743. this._originalDoCallback = window.WebForm_DoCallback;
  744. if (this._originalDoCallback) {
  745. window.WebForm_DoCallback = Function.createDelegate(this, this._doCallback);
  746. }
  747. this._pageLoadedHandler = Function.createDelegate(this, this._pageLoadedInitialLoad);
  748. Sys.UI.DomEvent.addHandler(window, 'load', this._pageLoadedHandler);
  749. if (updatePanelIDs) {
  750. this._updateControls(updatePanelIDs, asyncPostBackControlIDs, postBackControlIDs, asyncPostBackTimeout, true);
  751. }
  752. },
  753. _matchesParentIDInList: function PageRequestManager$_matchesParentIDInList(clientID, parentIDList) {
  754. for (var i = 0, l = parentIDList.length; i < l; i++) {
  755. if (clientID.startsWith(parentIDList[i] + "_")) {
  756. return true;
  757. }
  758. }
  759. return false;
  760. },
  761. _onFormElementActive: function PageRequestManager$_onFormElementActive(element, offsetX, offsetY) {
  762. // element: the form element that is active
  763. // offsetX/Y: if the element is an image button, the coordinates of the click
  764. if (element.disabled) {
  765. return;
  766. }
  767. // Check if the element that was clicked on should cause an async postback
  768. this._postBackSettings = this._getPostBackSettings(element, element.name);
  769. if (element.name) {
  770. // DevDiv Bugs 146697: tagName needs to be case insensitive to work with xhtml content type
  771. var tagName = element.tagName.toUpperCase();
  772. if (tagName === 'INPUT') {
  773. var type = element.type;
  774. if (type === 'submit') {
  775. // DevDiv Bugs 109456: Encode the name as well as the value
  776. this._additionalInput = encodeURIComponent(element.name) + '=' + encodeURIComponent(element.value);
  777. }
  778. else if (type === 'image') {
  779. // DevDiv Bugs 109456: Encode the name as well as the value
  780. this._additionalInput = encodeURIComponent(element.name) + '.x=' + offsetX + '&' + encodeURIComponent(element.name) + '.y=' + offsetY;
  781. }
  782. }
  783. else if ((tagName === 'BUTTON') && (element.name.length !== 0) && (element.type === 'submit')) {
  784. // DevDiv Bugs 109456: Encode the name as well as the value
  785. this._additionalInput = encodeURIComponent(element.name) + '=' + encodeURIComponent(element.value);
  786. }
  787. }
  788. },
  789. _onFormElementClick: function PageRequestManager$_onFormElementClick(evt) {
  790. // flag used by fireDefaultButton to know whether calling click() on the default button raised this event.
  791. this._activeDefaultButtonClicked = (evt.target === this._activeDefaultButton);
  792. this._onFormElementActive(evt.target, evt.offsetX, evt.offsetY);
  793. },
  794. _onFormSubmit: function PageRequestManager$_onFormSubmit(evt) {
  795. var i, l, continueSubmit = true,
  796. isCrossPost = this._isCrossPost;
  797. // set to false so subsequent posts that don't go through DPWO aren't considered cross post
  798. this._isCrossPost = false;
  799. // Call the statically declared form onsubmit statement if there was one
  800. if (this._onsubmit) {
  801. continueSubmit = this._onsubmit();
  802. }
  803. // If necessary, call dynamically added form onsubmit statements
  804. if (continueSubmit) {
  805. for (i = 0, l = this._onSubmitStatements.length; i < l; i++) {
  806. if (!this._onSubmitStatements[i]()) {
  807. continueSubmit = false;
  808. break;
  809. }
  810. }
  811. }
  812. if (!continueSubmit) {
  813. if (evt) {
  814. evt.preventDefault();
  815. }
  816. return;
  817. }
  818. var form = this._form;
  819. if (isCrossPost) {
  820. // Allow the default form submit to take place. Since it's a cross-page postback.
  821. return;
  822. }
  823. // DevDiv Bugs 123782
  824. if (this._activeDefaultButton && !this._activeDefaultButtonClicked) {
  825. // we are submitting because a default button's click method was called by _fireDefaultButton
  826. // but calling click() explicitly did not cause a click event or raised it for a different element,
  827. // so we must manually create the correct postback options.
  828. // The button was clicked programmatically, so there are no offsetX or offsetY coordinates.
  829. this._onFormElementActive(this._activeDefaultButton, 0, 0);
  830. }
  831. // If the postback happened from outside an update panel, fall back
  832. // and do a normal postback.
  833. // Dev10 546632: postBackSettings may not exist in certain scenarios where
  834. // the form click events are cancelled.
  835. if (!this._postBackSettings || !this._postBackSettings.async) {
  836. return;
  837. }
  838. // Construct the form body
  839. var formBody = new Sys.StringBuilder(),
  840. count = form.elements.length,
  841. panelID = this._createPanelID(null, this._postBackSettings);
  842. // DevDiv Bugs 109456: ScriptManager and UpdatePanel IDs should be encoded as well
  843. formBody.append(panelID);
  844. for (i = 0; i < count; i++) {
  845. var element = form.elements[i];
  846. var name = element.name;
  847. if (typeof(name) === "undefined" || (name === null) || (name.length === 0) || (name === this._scriptManagerID)) {
  848. continue;
  849. }
  850. // DevDiv Bugs 146697: tagName needs to be case insensitive to work with xhtml content type
  851. var tagName = element.tagName.toUpperCase();
  852. if (tagName === 'INPUT') {
  853. var type = element.type;
  854. if ((type === 'text') ||
  855. (type === 'password') ||
  856. (type === 'hidden') ||
  857. (((type === 'checkbox') || (type === 'radio')) && element.checked)) {
  858. // DevDiv Bugs 109456: Encode the name as well as the value
  859. formBody.append(encodeURIComponent(name));
  860. formBody.append('=');
  861. formBody.append(encodeURIComponent(element.value));
  862. formBody.append('&');
  863. }
  864. }
  865. else if (tagName === 'SELECT') {
  866. var optionCount = element.options.length;
  867. for (var j = 0; j < optionCount; j++) {
  868. var option = element.options[j];
  869. if (option.selected) {
  870. // DevDiv Bugs 109456: Encode the name as well as the value
  871. formBody.append(encodeURIComponent(name));
  872. formBody.append('=');
  873. formBody.append(encodeURIComponent(option.value));
  874. formBody.append('&');
  875. }
  876. }
  877. }
  878. else if (tagName === 'TEXTAREA') {
  879. // DevDiv Bugs 109456: Encode the name as well as the value
  880. formBody.append(encodeURIComponent(name));
  881. formBody.append('=');
  882. formBody.append(encodeURIComponent(element.value));
  883. formBody.append('&');
  884. }
  885. }
  886. // DevDiv Bugs 188713: Some firewalls strip the X-MicrosoftAjax header, so send a custom form field as well.
  887. formBody.append("__ASYNCPOST=true&");
  888. if (this._additionalInput) {
  889. formBody.append(this._additionalInput);
  890. this._additionalInput = null;
  891. }
  892. var request = new Sys.Net.WebRequest();
  893. var action = form.action;
  894. if (Sys.Browser.agent === Sys.Browser.InternetExplorer) {
  895. // DevDiv Bugs 85367: In IE we must encode the path portion of the request because XHR doesn't do it for us.
  896. // First, we remove the url fragment, which can appear in history-related scenarios
  897. // but is not relevant to async postbacks.
  898. var fragmentIndex = action.indexOf('#');

Large files files are truncated, but you can click here to view the full file