overlay.js 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665
  1. function _typeof(o) { "@babel/helpers - typeof"; return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (o) { return typeof o; } : function (o) { return o && "function" == typeof Symbol && o.constructor === Symbol && o !== Symbol.prototype ? "symbol" : typeof o; }, _typeof(o); }
  2. function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
  3. function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
  4. function _defineProperty(e, r, t) { return (r = _toPropertyKey(r)) in e ? Object.defineProperty(e, r, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : e[r] = t, e; }
  5. function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == _typeof(i) ? i : i + ""; }
  6. function _toPrimitive(t, r) { if ("object" != _typeof(t) || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != _typeof(i)) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
  7. // The error overlay is inspired (and mostly copied) from Create React App (https://github.com/facebookincubator/create-react-app)
  8. // They, in turn, got inspired by webpack-hot-middleware (https://github.com/glenjamin/webpack-hot-middleware).
  9. import ansiHTML from "ansi-html-community";
  10. /** @typedef {import("./index").EXPECTED_ANY} EXPECTED_ANY */
  11. /**
  12. * @type {(input: string, position: number) => number | undefined}
  13. */
  14. // @ts-expect-error
  15. var getCodePoint = String.prototype.codePointAt ?
  16. // @ts-expect-error
  17. function (input, position) {
  18. return input.codePointAt(position);
  19. } : function (input, position) {
  20. return (input.charCodeAt(position) - 0xd800) * 0x400 + input.charCodeAt(position + 1) - 0xdc00 + 0x10000;
  21. };
  22. /**
  23. * @param {string} macroText macro text
  24. * @param {RegExp} macroRegExp macro reg exp
  25. * @param {(input: string) => string} macroReplacer macro replacer
  26. * @returns {string} result
  27. */
  28. var replaceUsingRegExp = function replaceUsingRegExp(macroText, macroRegExp, macroReplacer) {
  29. macroRegExp.lastIndex = 0;
  30. var replaceMatch = macroRegExp.exec(macroText);
  31. var replaceResult;
  32. if (replaceMatch) {
  33. replaceResult = "";
  34. var replaceLastIndex = 0;
  35. do {
  36. if (replaceLastIndex !== replaceMatch.index) {
  37. replaceResult += macroText.slice(replaceLastIndex, replaceMatch.index);
  38. }
  39. var replaceInput = replaceMatch[0];
  40. replaceResult += macroReplacer(replaceInput);
  41. replaceLastIndex = replaceMatch.index + replaceInput.length;
  42. } while (replaceMatch = macroRegExp.exec(macroText));
  43. if (replaceLastIndex !== macroText.length) {
  44. replaceResult += macroText.slice(replaceLastIndex);
  45. }
  46. } else {
  47. replaceResult = macroText;
  48. }
  49. return replaceResult;
  50. };
  51. var references = {
  52. "<": "&lt;",
  53. ">": "&gt;",
  54. '"': "&quot;",
  55. "'": "&apos;",
  56. "&": "&amp;"
  57. };
  58. /**
  59. * @param {string} text text
  60. * @returns {string} encoded text
  61. */
  62. function encode(text) {
  63. if (!text) {
  64. return "";
  65. }
  66. return replaceUsingRegExp(text, /[<>'"&]/g, function (input) {
  67. var result = references[(/** @type {keyof typeof references} */input)];
  68. if (!result) {
  69. var code = input.length > 1 ? getCodePoint(input, 0) : input.charCodeAt(0);
  70. result = "&#".concat(code, ";");
  71. }
  72. return result;
  73. });
  74. }
  75. /**
  76. * @typedef {object} Context
  77. * @property {'warning' | 'error'} level level
  78. * @property {(string | Message)[]} messages messages
  79. * @property {'build' | 'runtime'} messageSource message source
  80. */
  81. /** @typedef {{ type: string } & Record<string, EXPECTED_ANY>} Event */
  82. /**
  83. * @typedef {object} Options
  84. * @property {{ [state: string]: { on: Record<string, { target: string; actions?: Array<string> }> } }} states states
  85. * @property {Context} context context
  86. * @property {string} initial initial
  87. */
  88. /**
  89. * @typedef {object} Implementation
  90. * @property {{ [actionName: string]: (ctx: Context, event: Event) => Context | void }} actions actions
  91. */
  92. /**
  93. * @typedef {{ send: (event: Event) => void }} StateMachine
  94. */
  95. /**
  96. * A simplified `createMachine` from `@xstate/fsm` with the following differences:
  97. * - the returned machine is technically a "service". No `interpret(machine).start()` is needed.
  98. * - the state definition only support `on` and target must be declared with { target: 'nextState', actions: [] } explicitly.
  99. * - event passed to `send` must be an object with `type` property.
  100. * - actions implementation will be [assign action](https://xstate.js.org/docs/guides/context.html#assign-action) if you return any value.
  101. * Do not return anything if you just want to invoke side effect.
  102. *
  103. * The goal of this custom function is to avoid installing the entire `'xstate/fsm'` package, while enabling modeling using
  104. * state machine. You can copy the first parameter into the editor at https://stately.ai/viz to visualize the state machine.
  105. * @param {Options} options options
  106. * @param {Implementation} implementation implementation
  107. * @returns {StateMachine} state machine
  108. */
  109. function createMachine(_ref, _ref2) {
  110. var states = _ref.states,
  111. context = _ref.context,
  112. initial = _ref.initial;
  113. var actions = _ref2.actions;
  114. var currentState = initial;
  115. var currentContext = context;
  116. return {
  117. send: function send(event) {
  118. var currentStateOn = states[currentState].on;
  119. var transitionConfig = currentStateOn && currentStateOn[event.type];
  120. if (transitionConfig) {
  121. currentState = transitionConfig.target;
  122. if (transitionConfig.actions) {
  123. transitionConfig.actions.forEach(function (actName) {
  124. var actionImpl = actions[actName];
  125. var nextContextValue = actionImpl && actionImpl(currentContext, event);
  126. if (nextContextValue) {
  127. currentContext = _objectSpread(_objectSpread({}, currentContext), nextContextValue);
  128. }
  129. });
  130. }
  131. }
  132. }
  133. };
  134. }
  135. /**
  136. * @typedef {object} ShowOverlayData
  137. * @property {'warning' | 'error'} level level
  138. * @property {(string | Message)[]} messages messages
  139. * @property {'build' | 'runtime'} messageSource message source
  140. */
  141. /**
  142. * @typedef {object} CreateOverlayMachineOptions
  143. * @property {(data: ShowOverlayData) => void} showOverlay show overlay
  144. * @property {() => void} hideOverlay hide overlay
  145. */
  146. /**
  147. * @param {CreateOverlayMachineOptions} options options
  148. * @returns {StateMachine} state machine
  149. */
  150. var createOverlayMachine = function createOverlayMachine(options) {
  151. var hideOverlay = options.hideOverlay,
  152. showOverlay = options.showOverlay;
  153. return createMachine({
  154. initial: "hidden",
  155. context: {
  156. level: "error",
  157. messages: [],
  158. messageSource: "build"
  159. },
  160. states: {
  161. hidden: {
  162. on: {
  163. BUILD_ERROR: {
  164. target: "displayBuildError",
  165. actions: ["setMessages", "showOverlay"]
  166. },
  167. RUNTIME_ERROR: {
  168. target: "displayRuntimeError",
  169. actions: ["setMessages", "showOverlay"]
  170. }
  171. }
  172. },
  173. displayBuildError: {
  174. on: {
  175. DISMISS: {
  176. target: "hidden",
  177. actions: ["dismissMessages", "hideOverlay"]
  178. },
  179. BUILD_ERROR: {
  180. target: "displayBuildError",
  181. actions: ["appendMessages", "showOverlay"]
  182. }
  183. }
  184. },
  185. displayRuntimeError: {
  186. on: {
  187. DISMISS: {
  188. target: "hidden",
  189. actions: ["dismissMessages", "hideOverlay"]
  190. },
  191. RUNTIME_ERROR: {
  192. target: "displayRuntimeError",
  193. actions: ["appendMessages", "showOverlay"]
  194. },
  195. BUILD_ERROR: {
  196. target: "displayBuildError",
  197. actions: ["setMessages", "showOverlay"]
  198. }
  199. }
  200. }
  201. }
  202. }, {
  203. actions: {
  204. dismissMessages: function dismissMessages() {
  205. return {
  206. messages: [],
  207. level: "error",
  208. messageSource: "build"
  209. };
  210. },
  211. appendMessages: function appendMessages(context, event) {
  212. return {
  213. messages: context.messages.concat(event.messages),
  214. level: event.level || context.level,
  215. messageSource: event.type === "RUNTIME_ERROR" ? "runtime" : "build"
  216. };
  217. },
  218. setMessages: function setMessages(context, event) {
  219. return {
  220. messages: event.messages,
  221. level: event.level || context.level,
  222. messageSource: event.type === "RUNTIME_ERROR" ? "runtime" : "build"
  223. };
  224. },
  225. hideOverlay: hideOverlay,
  226. showOverlay: showOverlay
  227. }
  228. });
  229. };
  230. /**
  231. * @param {Error} error error
  232. * @returns {undefined | string[]} stack
  233. */
  234. var parseErrorToStacks = function parseErrorToStacks(error) {
  235. if (!error || !(error instanceof Error)) {
  236. throw new Error("parseErrorToStacks expects Error object");
  237. }
  238. if (typeof error.stack === "string") {
  239. return error.stack.split("\n").filter(function (stack) {
  240. return stack !== "Error: ".concat(error.message);
  241. });
  242. }
  243. };
  244. /**
  245. * @callback ErrorCallback
  246. * @param {ErrorEvent} error
  247. * @returns {void}
  248. */
  249. /**
  250. * @param {ErrorCallback} callback callback
  251. * @returns {() => void} cleanup
  252. */
  253. var listenToRuntimeError = function listenToRuntimeError(callback) {
  254. window.addEventListener("error", callback);
  255. return function cleanup() {
  256. window.removeEventListener("error", callback);
  257. };
  258. };
  259. /**
  260. * @callback UnhandledRejectionCallback
  261. * @param {PromiseRejectionEvent} rejectionEvent
  262. * @returns {void}
  263. */
  264. /**
  265. * @param {UnhandledRejectionCallback} callback callback
  266. * @returns {() => void} cleanup
  267. */
  268. var listenToUnhandledRejection = function listenToUnhandledRejection(callback) {
  269. window.addEventListener("unhandledrejection", callback);
  270. return function cleanup() {
  271. window.removeEventListener("unhandledrejection", callback);
  272. };
  273. };
  274. // Styles are inspired by `react-error-overlay`
  275. var msgStyles = {
  276. error: {
  277. backgroundColor: "rgba(206, 17, 38, 0.1)",
  278. color: "#fccfcf"
  279. },
  280. warning: {
  281. backgroundColor: "rgba(251, 245, 180, 0.1)",
  282. color: "#fbf5b4"
  283. }
  284. };
  285. var iframeStyle = {
  286. position: "fixed",
  287. top: "0px",
  288. left: "0px",
  289. right: "0px",
  290. bottom: "0px",
  291. width: "100vw",
  292. height: "100vh",
  293. border: "none",
  294. "z-index": 9999999999
  295. };
  296. var containerStyle = {
  297. position: "fixed",
  298. boxSizing: "border-box",
  299. left: "0px",
  300. top: "0px",
  301. right: "0px",
  302. bottom: "0px",
  303. width: "100vw",
  304. height: "100vh",
  305. fontSize: "large",
  306. padding: "2rem 2rem 4rem 2rem",
  307. lineHeight: "1.2",
  308. whiteSpace: "pre-wrap",
  309. overflow: "auto",
  310. backgroundColor: "rgba(0, 0, 0, 0.9)",
  311. color: "white"
  312. };
  313. var headerStyle = {
  314. color: "#e83b46",
  315. fontSize: "2em",
  316. whiteSpace: "pre-wrap",
  317. fontFamily: "sans-serif",
  318. margin: "0 2rem 2rem 0",
  319. flex: "0 0 auto",
  320. maxHeight: "50%",
  321. overflow: "auto"
  322. };
  323. var dismissButtonStyle = {
  324. color: "#ffffff",
  325. lineHeight: "1rem",
  326. fontSize: "1.5rem",
  327. padding: "1rem",
  328. cursor: "pointer",
  329. position: "absolute",
  330. right: "0px",
  331. top: "0px",
  332. backgroundColor: "transparent",
  333. border: "none"
  334. };
  335. var msgTypeStyle = {
  336. color: "#e83b46",
  337. fontSize: "1.2em",
  338. marginBottom: "1rem",
  339. fontFamily: "sans-serif"
  340. };
  341. var msgTextStyle = {
  342. lineHeight: "1.5",
  343. fontSize: "1rem",
  344. fontFamily: "Menlo, Consolas, monospace"
  345. };
  346. // ANSI HTML
  347. var colors = {
  348. reset: ["transparent", "transparent"],
  349. black: "181818",
  350. red: "E36049",
  351. green: "B3CB74",
  352. yellow: "FFD080",
  353. blue: "7CAFC2",
  354. magenta: "7FACCA",
  355. cyan: "C3C2EF",
  356. lightgrey: "EBE7E3",
  357. darkgrey: "6D7891"
  358. };
  359. ansiHTML.setColors(colors);
  360. /** @typedef {Error & { file?: string, moduleName?: string, moduleIdentifier?: string, loc?: string, message?: string; stack?: string | string[] }} Message */
  361. /**
  362. * @param {string} type type
  363. * @param {string | Message} item item
  364. * @returns {{ header: string, body: string }} formatted problem
  365. */
  366. var formatProblem = function formatProblem(type, item) {
  367. var header = type === "warning" ? "WARNING" : "ERROR";
  368. var body = "";
  369. if (typeof item === "string") {
  370. body += item;
  371. } else {
  372. var file = item.file || "";
  373. var moduleName = item.moduleName ? item.moduleName.indexOf("!") !== -1 ? "".concat(item.moduleName.replace(/^(\s|\S)*!/, ""), " (").concat(item.moduleName, ")") : "".concat(item.moduleName) : "";
  374. var loc = item.loc;
  375. header += "".concat(moduleName || file ? " in ".concat(moduleName ? "".concat(moduleName).concat(file ? " (".concat(file, ")") : "") : file).concat(loc ? " ".concat(loc) : "") : "");
  376. body += item.message || "";
  377. }
  378. if (typeof item !== "string" && Array.isArray(item.stack)) {
  379. item.stack.forEach(function (stack) {
  380. if (typeof stack === "string") {
  381. body += "\r\n".concat(stack);
  382. }
  383. });
  384. }
  385. return {
  386. header: header,
  387. body: body
  388. };
  389. };
  390. /**
  391. * @typedef {object} CreateOverlayOptions
  392. * @property {(false | string)=} trustedTypesPolicyName trusted types policy name
  393. * @property {(boolean | ((error: Error) => void))=} catchRuntimeError runtime error catcher
  394. */
  395. /**
  396. * @param {CreateOverlayOptions} options options
  397. * @returns {StateMachine} overlay
  398. */
  399. var createOverlay = function createOverlay(options) {
  400. /** @type {HTMLIFrameElement | null | undefined} */
  401. var iframeContainerElement;
  402. /** @type {HTMLDivElement | null | undefined} */
  403. var containerElement;
  404. /** @type {HTMLDivElement | null | undefined} */
  405. var headerElement;
  406. /** @type {Array<(element: HTMLDivElement) => void>} */
  407. var onLoadQueue = [];
  408. /** @type {Omit<TrustedTypePolicy, "createScript" | "createScriptURL"> | undefined} */
  409. var overlayTrustedTypesPolicy;
  410. /** @typedef {Extract<keyof CSSStyleDeclaration, "string">} CSSStyleDeclarationKeys */
  411. /**
  412. * @param {HTMLElement} element element
  413. * @param {Partial<CSSStyleDeclaration>} style style
  414. */
  415. function applyStyle(element, style) {
  416. Object.keys(style).forEach(function (prop) {
  417. element.style[(/** @type {CSSStyleDeclarationKeys} */prop)] = /** @type {string} */
  418. style[(/** @type {CSSStyleDeclarationKeys} */prop)];
  419. });
  420. }
  421. /**
  422. * @param {string | false | undefined} trustedTypesPolicyName trusted types police name
  423. */
  424. function createContainer(trustedTypesPolicyName) {
  425. // Enable Trusted Types if they are available in the current browser.
  426. if (window.trustedTypes) {
  427. overlayTrustedTypesPolicy = window.trustedTypes.createPolicy(trustedTypesPolicyName || "webpack-dev-server#overlay", {
  428. createHTML: function createHTML(value) {
  429. return value;
  430. }
  431. });
  432. }
  433. iframeContainerElement = document.createElement("iframe");
  434. iframeContainerElement.id = "webpack-dev-server-client-overlay";
  435. iframeContainerElement.src = "about:blank";
  436. applyStyle(iframeContainerElement, iframeStyle);
  437. iframeContainerElement.onload = function () {
  438. var contentElement = /** @type {Document} */
  439. (/** @type {HTMLIFrameElement} */
  440. iframeContainerElement.contentDocument).createElement("div");
  441. containerElement = /** @type {Document} */
  442. (/** @type {HTMLIFrameElement} */
  443. iframeContainerElement.contentDocument).createElement("div");
  444. contentElement.id = "webpack-dev-server-client-overlay-div";
  445. applyStyle(contentElement, containerStyle);
  446. headerElement = document.createElement("div");
  447. headerElement.innerText = "Compiled with problems:";
  448. applyStyle(headerElement, headerStyle);
  449. var closeButtonElement = document.createElement("button");
  450. applyStyle(closeButtonElement, dismissButtonStyle);
  451. closeButtonElement.innerText = "×";
  452. closeButtonElement.ariaLabel = "Dismiss";
  453. closeButtonElement.addEventListener("click", function () {
  454. // eslint-disable-next-line no-use-before-define
  455. overlayService.send({
  456. type: "DISMISS"
  457. });
  458. });
  459. contentElement.appendChild(headerElement);
  460. contentElement.appendChild(closeButtonElement);
  461. contentElement.appendChild(containerElement);
  462. /** @type {Document} */
  463. (/** @type {HTMLIFrameElement} */
  464. iframeContainerElement.contentDocument).body.appendChild(contentElement);
  465. onLoadQueue.forEach(function (onLoad) {
  466. onLoad(/** @type {HTMLDivElement} */contentElement);
  467. });
  468. onLoadQueue = [];
  469. /** @type {HTMLIFrameElement} */
  470. iframeContainerElement.onload = null;
  471. };
  472. document.body.appendChild(iframeContainerElement);
  473. }
  474. /**
  475. * @param {(element: HTMLDivElement) => void} callback callback
  476. * @param {string | false | undefined} trustedTypesPolicyName trusted types policy name
  477. */
  478. function ensureOverlayExists(callback, trustedTypesPolicyName) {
  479. if (containerElement) {
  480. // @ts-expect-error https://github.com/microsoft/TypeScript/issues/30024
  481. containerElement.innerHTML = overlayTrustedTypesPolicy ? overlayTrustedTypesPolicy.createHTML("") : "";
  482. // Everything is ready, call the callback right away.
  483. callback(containerElement);
  484. return;
  485. }
  486. onLoadQueue.push(callback);
  487. if (iframeContainerElement) {
  488. return;
  489. }
  490. createContainer(trustedTypesPolicyName);
  491. }
  492. // Successful compilation.
  493. /**
  494. * @returns {void}
  495. */
  496. function hide() {
  497. if (!iframeContainerElement) {
  498. return;
  499. }
  500. // Clean up and reset internal state.
  501. document.body.removeChild(iframeContainerElement);
  502. iframeContainerElement = null;
  503. containerElement = null;
  504. }
  505. // Compilation with errors (e.g. syntax error or missing modules).
  506. /**
  507. * @param {string} type type
  508. * @param {(string | Message)[]} messages messages
  509. * @param {undefined | false | string} trustedTypesPolicyName trusted types policy name
  510. * @param {'build' | 'runtime'} messageSource message source
  511. */
  512. function show(type, messages, trustedTypesPolicyName, messageSource) {
  513. ensureOverlayExists(function () {
  514. /** @type {HTMLDivElement} */
  515. headerElement.innerText = messageSource === "runtime" ? "Uncaught runtime errors:" : "Compiled with problems:";
  516. messages.forEach(function (message) {
  517. var entryElement = document.createElement("div");
  518. var msgStyle = type === "warning" ? msgStyles.warning : msgStyles.error;
  519. applyStyle(entryElement, _objectSpread(_objectSpread({}, msgStyle), {}, {
  520. padding: "1rem 1rem 1.5rem 1rem"
  521. }));
  522. var typeElement = document.createElement("div");
  523. var _formatProblem = formatProblem(type, message),
  524. header = _formatProblem.header,
  525. body = _formatProblem.body;
  526. typeElement.innerText = header;
  527. applyStyle(typeElement, msgTypeStyle);
  528. if (typeof message !== "string" && message.moduleIdentifier) {
  529. applyStyle(typeElement, {
  530. cursor: "pointer"
  531. });
  532. // element.dataset not supported in IE
  533. typeElement.setAttribute("data-can-open", "true");
  534. typeElement.addEventListener("click", function () {
  535. fetch("/webpack-dev-server/open-editor?fileName=".concat(message.moduleIdentifier));
  536. });
  537. }
  538. // Make it look similar to our terminal.
  539. var text = ansiHTML(encode(body));
  540. var messageTextNode = document.createElement("div");
  541. applyStyle(messageTextNode, msgTextStyle);
  542. // @ts-expect-error https://github.com/microsoft/TypeScript/issues/30024
  543. messageTextNode.innerHTML = overlayTrustedTypesPolicy ? overlayTrustedTypesPolicy.createHTML(text) : text;
  544. entryElement.appendChild(typeElement);
  545. entryElement.appendChild(messageTextNode);
  546. /** @type {HTMLDivElement} */
  547. containerElement.appendChild(entryElement);
  548. });
  549. }, trustedTypesPolicyName);
  550. }
  551. /** @type {(event: KeyboardEvent) => void} */
  552. var handleEscapeKey;
  553. /**
  554. * @returns {void}
  555. */
  556. var hideOverlayWithEscCleanup = function hideOverlayWithEscCleanup() {
  557. window.removeEventListener("keydown", handleEscapeKey);
  558. hide();
  559. };
  560. var overlayService = createOverlayMachine({
  561. showOverlay: function showOverlay(_ref3) {
  562. var _ref3$level = _ref3.level,
  563. level = _ref3$level === void 0 ? "error" : _ref3$level,
  564. messages = _ref3.messages,
  565. messageSource = _ref3.messageSource;
  566. return show(level, messages, options.trustedTypesPolicyName, messageSource);
  567. },
  568. hideOverlay: hideOverlayWithEscCleanup
  569. });
  570. /**
  571. * ESC key press to dismiss the overlay.
  572. * @param {KeyboardEvent} event Keydown event
  573. */
  574. handleEscapeKey = function handleEscapeKey(event) {
  575. if (event.key === "Escape" || event.key === "Esc" || event.keyCode === 27) {
  576. overlayService.send({
  577. type: "DISMISS"
  578. });
  579. }
  580. };
  581. window.addEventListener("keydown", handleEscapeKey);
  582. if (options.catchRuntimeError) {
  583. /**
  584. * @param {Error | undefined} error error
  585. * @param {string} fallbackMessage fallback message
  586. */
  587. var handleError = function handleError(error, fallbackMessage) {
  588. var errorObject = error instanceof Error ? error : new Error(error || fallbackMessage, {
  589. cause: error
  590. });
  591. var shouldDisplay = typeof options.catchRuntimeError === "function" ? options.catchRuntimeError(errorObject) : true;
  592. if (shouldDisplay) {
  593. overlayService.send({
  594. type: "RUNTIME_ERROR",
  595. messages: [{
  596. message: errorObject.message,
  597. stack: parseErrorToStacks(errorObject)
  598. }]
  599. });
  600. }
  601. };
  602. listenToRuntimeError(function (errorEvent) {
  603. // error property may be empty in older browser like IE
  604. var error = errorEvent.error,
  605. message = errorEvent.message;
  606. if (!error && !message) {
  607. return;
  608. }
  609. // if error stack indicates a React error boundary caught the error, do not show overlay.
  610. if (error && error.stack && error.stack.includes("invokeGuardedCallbackDev")) {
  611. return;
  612. }
  613. handleError(error, message);
  614. });
  615. listenToUnhandledRejection(function (promiseRejectionEvent) {
  616. var reason = promiseRejectionEvent.reason;
  617. handleError(reason, "Unknown promise rejection reason");
  618. });
  619. }
  620. return overlayService;
  621. };
  622. export { createOverlay, formatProblem };