patchright-core 1.48.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (333) hide show
  1. package/README.md +3 -0
  2. package/ThirdPartyNotices.txt +1548 -0
  3. package/bin/PrintDeps.exe +0 -0
  4. package/bin/README.md +2 -0
  5. package/bin/install_media_pack.ps1 +5 -0
  6. package/bin/reinstall_chrome_beta_linux.sh +40 -0
  7. package/bin/reinstall_chrome_beta_mac.sh +13 -0
  8. package/bin/reinstall_chrome_beta_win.ps1 +24 -0
  9. package/bin/reinstall_chrome_stable_linux.sh +40 -0
  10. package/bin/reinstall_chrome_stable_mac.sh +12 -0
  11. package/bin/reinstall_chrome_stable_win.ps1 +24 -0
  12. package/bin/reinstall_msedge_beta_linux.sh +40 -0
  13. package/bin/reinstall_msedge_beta_mac.sh +11 -0
  14. package/bin/reinstall_msedge_beta_win.ps1 +23 -0
  15. package/bin/reinstall_msedge_dev_linux.sh +40 -0
  16. package/bin/reinstall_msedge_dev_mac.sh +11 -0
  17. package/bin/reinstall_msedge_dev_win.ps1 +23 -0
  18. package/bin/reinstall_msedge_stable_linux.sh +40 -0
  19. package/bin/reinstall_msedge_stable_mac.sh +11 -0
  20. package/bin/reinstall_msedge_stable_win.ps1 +24 -0
  21. package/browsers.json +57 -0
  22. package/cli.js +18 -0
  23. package/index.d.ts +17 -0
  24. package/index.js +33 -0
  25. package/index.mjs +28 -0
  26. package/lib/androidServerImpl.js +69 -0
  27. package/lib/browserServerImpl.js +92 -0
  28. package/lib/cli/driver.js +95 -0
  29. package/lib/cli/program.js +589 -0
  30. package/lib/cli/programWithTestStub.js +67 -0
  31. package/lib/client/accessibility.js +50 -0
  32. package/lib/client/android.js +473 -0
  33. package/lib/client/api.js +285 -0
  34. package/lib/client/artifact.js +79 -0
  35. package/lib/client/browser.js +145 -0
  36. package/lib/client/browserContext.js +583 -0
  37. package/lib/client/browserType.js +241 -0
  38. package/lib/client/cdpSession.js +53 -0
  39. package/lib/client/channelOwner.js +235 -0
  40. package/lib/client/clientHelper.js +57 -0
  41. package/lib/client/clientInstrumentation.js +50 -0
  42. package/lib/client/clock.js +69 -0
  43. package/lib/client/connection.js +333 -0
  44. package/lib/client/consoleMessage.js +55 -0
  45. package/lib/client/coverage.js +41 -0
  46. package/lib/client/dialog.js +57 -0
  47. package/lib/client/download.js +62 -0
  48. package/lib/client/electron.js +135 -0
  49. package/lib/client/elementHandle.js +321 -0
  50. package/lib/client/errors.js +77 -0
  51. package/lib/client/eventEmitter.js +314 -0
  52. package/lib/client/events.js +94 -0
  53. package/lib/client/fetch.js +391 -0
  54. package/lib/client/fileChooser.js +45 -0
  55. package/lib/client/frame.js +504 -0
  56. package/lib/client/harRouter.js +99 -0
  57. package/lib/client/input.js +111 -0
  58. package/lib/client/jsHandle.js +121 -0
  59. package/lib/client/jsonPipe.js +35 -0
  60. package/lib/client/localUtils.js +36 -0
  61. package/lib/client/locator.js +441 -0
  62. package/lib/client/network.js +762 -0
  63. package/lib/client/page.js +770 -0
  64. package/lib/client/playwright.js +80 -0
  65. package/lib/client/selectors.js +67 -0
  66. package/lib/client/stream.js +54 -0
  67. package/lib/client/tracing.js +134 -0
  68. package/lib/client/types.js +24 -0
  69. package/lib/client/video.js +51 -0
  70. package/lib/client/waiter.js +158 -0
  71. package/lib/client/webError.js +37 -0
  72. package/lib/client/worker.js +71 -0
  73. package/lib/client/writableStream.js +54 -0
  74. package/lib/common/socksProxy.js +569 -0
  75. package/lib/common/timeoutSettings.js +73 -0
  76. package/lib/common/types.js +5 -0
  77. package/lib/generated/clockSource.js +7 -0
  78. package/lib/generated/consoleApiSource.js +7 -0
  79. package/lib/generated/injectedScriptSource.js +7 -0
  80. package/lib/generated/pollingRecorderSource.js +7 -0
  81. package/lib/generated/utilityScriptSource.js +7 -0
  82. package/lib/generated/webSocketMockSource.js +7 -0
  83. package/lib/image_tools/colorUtils.js +98 -0
  84. package/lib/image_tools/compare.js +108 -0
  85. package/lib/image_tools/imageChannel.js +70 -0
  86. package/lib/image_tools/stats.js +102 -0
  87. package/lib/inProcessFactory.js +54 -0
  88. package/lib/inprocess.js +20 -0
  89. package/lib/outofprocess.js +67 -0
  90. package/lib/protocol/debug.js +27 -0
  91. package/lib/protocol/serializers.js +173 -0
  92. package/lib/protocol/transport.js +82 -0
  93. package/lib/protocol/validator.js +2760 -0
  94. package/lib/protocol/validatorPrimitives.js +139 -0
  95. package/lib/remote/playwrightConnection.js +274 -0
  96. package/lib/remote/playwrightServer.js +110 -0
  97. package/lib/server/accessibility.js +62 -0
  98. package/lib/server/android/android.js +441 -0
  99. package/lib/server/android/backendAdb.js +172 -0
  100. package/lib/server/artifact.js +104 -0
  101. package/lib/server/bidi/bidiBrowser.js +311 -0
  102. package/lib/server/bidi/bidiChromium.js +124 -0
  103. package/lib/server/bidi/bidiConnection.js +206 -0
  104. package/lib/server/bidi/bidiExecutionContext.js +159 -0
  105. package/lib/server/bidi/bidiFirefox.js +104 -0
  106. package/lib/server/bidi/bidiInput.js +158 -0
  107. package/lib/server/bidi/bidiNetworkManager.js +338 -0
  108. package/lib/server/bidi/bidiOverCdp.js +103 -0
  109. package/lib/server/bidi/bidiPage.js +529 -0
  110. package/lib/server/bidi/bidiPdf.js +140 -0
  111. package/lib/server/bidi/third_party/bidiDeserializer.js +93 -0
  112. package/lib/server/bidi/third_party/bidiKeyboard.js +238 -0
  113. package/lib/server/bidi/third_party/bidiProtocol.js +139 -0
  114. package/lib/server/bidi/third_party/bidiSerializer.js +144 -0
  115. package/lib/server/bidi/third_party/firefoxPrefs.js +221 -0
  116. package/lib/server/browser.js +148 -0
  117. package/lib/server/browserContext.js +666 -0
  118. package/lib/server/browserType.js +335 -0
  119. package/lib/server/chromium/appIcon.png +0 -0
  120. package/lib/server/chromium/chromium.js +350 -0
  121. package/lib/server/chromium/chromiumSwitches.js +36 -0
  122. package/lib/server/chromium/crAccessibility.js +237 -0
  123. package/lib/server/chromium/crBrowser.js +522 -0
  124. package/lib/server/chromium/crConnection.js +228 -0
  125. package/lib/server/chromium/crCoverage.js +246 -0
  126. package/lib/server/chromium/crDevTools.js +104 -0
  127. package/lib/server/chromium/crDragDrop.js +143 -0
  128. package/lib/server/chromium/crExecutionContext.js +149 -0
  129. package/lib/server/chromium/crInput.js +171 -0
  130. package/lib/server/chromium/crNetworkManager.js +809 -0
  131. package/lib/server/chromium/crPage.js +1235 -0
  132. package/lib/server/chromium/crPdf.js +153 -0
  133. package/lib/server/chromium/crProtocolHelper.js +133 -0
  134. package/lib/server/chromium/crServiceWorker.js +111 -0
  135. package/lib/server/chromium/defaultFontFamilies.js +145 -0
  136. package/lib/server/chromium/videoRecorder.js +155 -0
  137. package/lib/server/clock.js +133 -0
  138. package/lib/server/codegen/csharp.js +299 -0
  139. package/lib/server/codegen/java.js +235 -0
  140. package/lib/server/codegen/javascript.js +223 -0
  141. package/lib/server/codegen/jsonl.js +47 -0
  142. package/lib/server/codegen/language.js +88 -0
  143. package/lib/server/codegen/languages.js +30 -0
  144. package/lib/server/codegen/python.js +265 -0
  145. package/lib/server/codegen/types.js +5 -0
  146. package/lib/server/console.js +57 -0
  147. package/lib/server/cookieStore.js +185 -0
  148. package/lib/server/debugController.js +234 -0
  149. package/lib/server/debugger.js +132 -0
  150. package/lib/server/deviceDescriptors.js +26 -0
  151. package/lib/server/deviceDescriptorsSource.json +1669 -0
  152. package/lib/server/dialog.js +71 -0
  153. package/lib/server/dispatchers/androidDispatcher.js +193 -0
  154. package/lib/server/dispatchers/artifactDispatcher.js +118 -0
  155. package/lib/server/dispatchers/browserContextDispatcher.js +368 -0
  156. package/lib/server/dispatchers/browserDispatcher.js +170 -0
  157. package/lib/server/dispatchers/browserTypeDispatcher.js +55 -0
  158. package/lib/server/dispatchers/cdpSessionDispatcher.js +48 -0
  159. package/lib/server/dispatchers/debugControllerDispatcher.js +103 -0
  160. package/lib/server/dispatchers/dialogDispatcher.js +44 -0
  161. package/lib/server/dispatchers/dispatcher.js +395 -0
  162. package/lib/server/dispatchers/electronDispatcher.js +93 -0
  163. package/lib/server/dispatchers/elementHandlerDispatcher.js +228 -0
  164. package/lib/server/dispatchers/frameDispatcher.js +286 -0
  165. package/lib/server/dispatchers/jsHandleDispatcher.js +97 -0
  166. package/lib/server/dispatchers/jsonPipeDispatcher.js +59 -0
  167. package/lib/server/dispatchers/localUtilsDispatcher.js +413 -0
  168. package/lib/server/dispatchers/networkDispatchers.js +221 -0
  169. package/lib/server/dispatchers/pageDispatcher.js +367 -0
  170. package/lib/server/dispatchers/playwrightDispatcher.js +107 -0
  171. package/lib/server/dispatchers/selectorsDispatcher.js +36 -0
  172. package/lib/server/dispatchers/streamDispatcher.js +62 -0
  173. package/lib/server/dispatchers/tracingDispatcher.js +54 -0
  174. package/lib/server/dispatchers/webSocketRouteDispatcher.js +189 -0
  175. package/lib/server/dispatchers/writableStreamDispatcher.js +58 -0
  176. package/lib/server/dom.js +845 -0
  177. package/lib/server/download.js +60 -0
  178. package/lib/server/electron/electron.js +296 -0
  179. package/lib/server/electron/loader.js +57 -0
  180. package/lib/server/errors.js +68 -0
  181. package/lib/server/fetch.js +656 -0
  182. package/lib/server/fileChooser.js +42 -0
  183. package/lib/server/fileUploadUtils.js +75 -0
  184. package/lib/server/firefox/ffAccessibility.js +216 -0
  185. package/lib/server/firefox/ffBrowser.js +460 -0
  186. package/lib/server/firefox/ffConnection.js +168 -0
  187. package/lib/server/firefox/ffExecutionContext.js +135 -0
  188. package/lib/server/firefox/ffInput.js +150 -0
  189. package/lib/server/firefox/ffNetworkManager.js +233 -0
  190. package/lib/server/firefox/ffPage.js +559 -0
  191. package/lib/server/firefox/firefox.js +99 -0
  192. package/lib/server/formData.js +75 -0
  193. package/lib/server/frameSelectors.js +171 -0
  194. package/lib/server/frames.js +1808 -0
  195. package/lib/server/har/harRecorder.js +139 -0
  196. package/lib/server/har/harTracer.js +542 -0
  197. package/lib/server/helper.js +103 -0
  198. package/lib/server/index.js +114 -0
  199. package/lib/server/input.js +310 -0
  200. package/lib/server/instrumentation.js +70 -0
  201. package/lib/server/isomorphic/utilityScriptSerializers.js +226 -0
  202. package/lib/server/javascript.js +299 -0
  203. package/lib/server/launchApp.js +91 -0
  204. package/lib/server/macEditingCommands.js +139 -0
  205. package/lib/server/network.js +617 -0
  206. package/lib/server/page.js +819 -0
  207. package/lib/server/pipeTransport.js +85 -0
  208. package/lib/server/playwright.js +88 -0
  209. package/lib/server/progress.js +102 -0
  210. package/lib/server/protocolError.js +49 -0
  211. package/lib/server/recorder/contextRecorder.js +299 -0
  212. package/lib/server/recorder/recorderApp.js +196 -0
  213. package/lib/server/recorder/recorderCollection.js +116 -0
  214. package/lib/server/recorder/recorderFrontend.js +5 -0
  215. package/lib/server/recorder/recorderInTraceViewer.js +144 -0
  216. package/lib/server/recorder/recorderRunner.js +155 -0
  217. package/lib/server/recorder/recorderUtils.js +112 -0
  218. package/lib/server/recorder/throttledFile.js +46 -0
  219. package/lib/server/recorder.js +329 -0
  220. package/lib/server/registry/browserFetcher.js +168 -0
  221. package/lib/server/registry/dependencies.js +322 -0
  222. package/lib/server/registry/index.js +1005 -0
  223. package/lib/server/registry/nativeDeps.js +490 -0
  224. package/lib/server/registry/oopDownloadBrowserMain.js +138 -0
  225. package/lib/server/screenshotter.js +348 -0
  226. package/lib/server/selectors.js +73 -0
  227. package/lib/server/socksClientCertificatesInterceptor.js +340 -0
  228. package/lib/server/socksInterceptor.js +100 -0
  229. package/lib/server/trace/recorder/snapshotter.js +172 -0
  230. package/lib/server/trace/recorder/snapshotterInjected.js +493 -0
  231. package/lib/server/trace/recorder/tracing.js +542 -0
  232. package/lib/server/trace/test/inMemorySnapshotter.js +93 -0
  233. package/lib/server/trace/viewer/traceViewer.js +213 -0
  234. package/lib/server/transport.js +191 -0
  235. package/lib/server/types.js +24 -0
  236. package/lib/server/usKeyboardLayout.js +555 -0
  237. package/lib/server/webkit/webkit.js +87 -0
  238. package/lib/server/webkit/wkAccessibility.js +194 -0
  239. package/lib/server/webkit/wkBrowser.js +329 -0
  240. package/lib/server/webkit/wkConnection.js +173 -0
  241. package/lib/server/webkit/wkExecutionContext.js +143 -0
  242. package/lib/server/webkit/wkInput.js +169 -0
  243. package/lib/server/webkit/wkInterceptableRequest.js +162 -0
  244. package/lib/server/webkit/wkPage.js +1219 -0
  245. package/lib/server/webkit/wkProvisionalPage.js +94 -0
  246. package/lib/server/webkit/wkWorkers.js +104 -0
  247. package/lib/third_party/diff_match_patch.js +2222 -0
  248. package/lib/third_party/pixelmatch.js +255 -0
  249. package/lib/utils/ascii.js +31 -0
  250. package/lib/utils/comparators.js +171 -0
  251. package/lib/utils/crypto.js +174 -0
  252. package/lib/utils/debug.js +46 -0
  253. package/lib/utils/debugLogger.js +91 -0
  254. package/lib/utils/env.js +49 -0
  255. package/lib/utils/eventsHelper.js +38 -0
  256. package/lib/utils/expectUtils.js +33 -0
  257. package/lib/utils/fileUtils.js +205 -0
  258. package/lib/utils/happy-eyeballs.js +194 -0
  259. package/lib/utils/headers.js +52 -0
  260. package/lib/utils/hostPlatform.js +133 -0
  261. package/lib/utils/httpServer.js +237 -0
  262. package/lib/utils/index.js +368 -0
  263. package/lib/utils/isomorphic/cssParser.js +250 -0
  264. package/lib/utils/isomorphic/cssTokenizer.js +979 -0
  265. package/lib/utils/isomorphic/locatorGenerators.js +642 -0
  266. package/lib/utils/isomorphic/locatorParser.js +179 -0
  267. package/lib/utils/isomorphic/locatorUtils.js +62 -0
  268. package/lib/utils/isomorphic/mimeType.js +29 -0
  269. package/lib/utils/isomorphic/recorderUtils.js +195 -0
  270. package/lib/utils/isomorphic/selectorParser.js +397 -0
  271. package/lib/utils/isomorphic/stringUtils.js +139 -0
  272. package/lib/utils/isomorphic/traceUtils.js +39 -0
  273. package/lib/utils/isomorphic/urlMatch.js +120 -0
  274. package/lib/utils/linuxUtils.js +78 -0
  275. package/lib/utils/manualPromise.js +109 -0
  276. package/lib/utils/multimap.js +75 -0
  277. package/lib/utils/network.js +160 -0
  278. package/lib/utils/processLauncher.js +248 -0
  279. package/lib/utils/profiler.js +53 -0
  280. package/lib/utils/rtti.js +44 -0
  281. package/lib/utils/semaphore.js +51 -0
  282. package/lib/utils/spawnAsync.js +45 -0
  283. package/lib/utils/stackTrace.js +121 -0
  284. package/lib/utils/task.js +58 -0
  285. package/lib/utils/time.js +37 -0
  286. package/lib/utils/timeoutRunner.js +66 -0
  287. package/lib/utils/traceUtils.js +44 -0
  288. package/lib/utils/userAgent.js +105 -0
  289. package/lib/utils/wsServer.js +127 -0
  290. package/lib/utils/zipFile.js +75 -0
  291. package/lib/utils/zones.js +62 -0
  292. package/lib/utilsBundle.js +82 -0
  293. package/lib/utilsBundleImpl/index.js +53 -0
  294. package/lib/utilsBundleImpl/xdg-open +1066 -0
  295. package/lib/vite/htmlReport/index.html +66 -0
  296. package/lib/vite/recorder/assets/codeMirrorModule-baozm8ur.js +24 -0
  297. package/lib/vite/recorder/assets/codeMirrorModule-ez37Vkbh.css +1 -0
  298. package/lib/vite/recorder/assets/codicon-DCmgc-ay.ttf +0 -0
  299. package/lib/vite/recorder/assets/index-2ElAIWFB.js +42 -0
  300. package/lib/vite/recorder/assets/index-BW-aOBcL.css +1 -0
  301. package/lib/vite/recorder/index.html +29 -0
  302. package/lib/vite/recorder/playwright-logo.svg +9 -0
  303. package/lib/vite/traceViewer/assets/codeMirrorModule-Bh1rfd2w.js +24 -0
  304. package/lib/vite/traceViewer/assets/inspectorTab-7GHnKvSD.js +64 -0
  305. package/lib/vite/traceViewer/assets/testServerConnection-DeE2kSzz.js +1 -0
  306. package/lib/vite/traceViewer/assets/workbench-DPQnTHYP.js +9 -0
  307. package/lib/vite/traceViewer/assets/xtermModule-BeNbaIVa.js +9 -0
  308. package/lib/vite/traceViewer/codeMirrorModule.ez37Vkbh.css +1 -0
  309. package/lib/vite/traceViewer/codicon.DCmgc-ay.ttf +0 -0
  310. package/lib/vite/traceViewer/embedded.BlHoW5LY.js +2 -0
  311. package/lib/vite/traceViewer/embedded.html +18 -0
  312. package/lib/vite/traceViewer/embedded.w7WN2u1R.css +1 -0
  313. package/lib/vite/traceViewer/index.CrbWWHbf.css +1 -0
  314. package/lib/vite/traceViewer/index.DaWVfou1.js +2 -0
  315. package/lib/vite/traceViewer/index.html +29 -0
  316. package/lib/vite/traceViewer/inspectorTab.DLjBDrQR.css +1 -0
  317. package/lib/vite/traceViewer/playwright-logo.svg +9 -0
  318. package/lib/vite/traceViewer/recorder.B_SY1GJM.css +0 -0
  319. package/lib/vite/traceViewer/recorder.C4zxcvd2.js +2 -0
  320. package/lib/vite/traceViewer/recorder.html +17 -0
  321. package/lib/vite/traceViewer/snapshot.html +21 -0
  322. package/lib/vite/traceViewer/sw.bundle.js +3 -0
  323. package/lib/vite/traceViewer/uiMode.CAYqod-m.css +1 -0
  324. package/lib/vite/traceViewer/uiMode.html +20 -0
  325. package/lib/vite/traceViewer/uiMode.mTXWniJb.js +5 -0
  326. package/lib/vite/traceViewer/workbench.D3JVcA9K.css +1 -0
  327. package/lib/vite/traceViewer/xtermModule.DSXBckUd.css +32 -0
  328. package/lib/zipBundle.js +25 -0
  329. package/lib/zipBundleImpl.js +5 -0
  330. package/package.json +44 -0
  331. package/types/protocol.d.ts +21571 -0
  332. package/types/structs.d.ts +45 -0
  333. package/types/types.d.ts +22519 -0
@@ -0,0 +1,155 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.VideoRecorder = void 0;
7
+ var _utils = require("../../utils");
8
+ var _page = require("../page");
9
+ var _processLauncher = require("../../utils/processLauncher");
10
+ var _progress = require("../progress");
11
+ var _instrumentation = require("../instrumentation");
12
+ /**
13
+ * Copyright (c) Microsoft Corporation.
14
+ *
15
+ * Licensed under the Apache License, Version 2.0 (the "License");
16
+ * you may not use this file except in compliance with the License.
17
+ * You may obtain a copy of the License at
18
+ *
19
+ * http://www.apache.org/licenses/LICENSE-2.0
20
+ *
21
+ * Unless required by applicable law or agreed to in writing, software
22
+ * distributed under the License is distributed on an "AS IS" BASIS,
23
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
24
+ * See the License for the specific language governing permissions and
25
+ * limitations under the License.
26
+ */
27
+
28
+ const fps = 25;
29
+ class VideoRecorder {
30
+ static async launch(page, ffmpegPath, options) {
31
+ if (!options.outputFile.endsWith('.webm')) throw new Error('File must have .webm extension');
32
+ const controller = new _progress.ProgressController((0, _instrumentation.serverSideCallMetadata)(), page);
33
+ controller.setLogName('browser');
34
+ return await controller.run(async progress => {
35
+ const recorder = new VideoRecorder(page, ffmpegPath, progress);
36
+ await recorder._launch(options);
37
+ return recorder;
38
+ });
39
+ }
40
+ constructor(page, ffmpegPath, progress) {
41
+ this._process = null;
42
+ this._gracefullyClose = null;
43
+ this._lastWritePromise = Promise.resolve();
44
+ this._lastFrameTimestamp = 0;
45
+ this._lastFrameBuffer = null;
46
+ this._lastWriteTimestamp = 0;
47
+ this._progress = void 0;
48
+ this._frameQueue = [];
49
+ this._isStopped = false;
50
+ this._ffmpegPath = void 0;
51
+ this._progress = progress;
52
+ this._ffmpegPath = ffmpegPath;
53
+ page.on(_page.Page.Events.ScreencastFrame, frame => this.writeFrame(frame.buffer, frame.frameSwapWallTime / 1000));
54
+ }
55
+ async _launch(options) {
56
+ // How to tune the codec:
57
+ // 1. Read vp8 documentation to figure out the options.
58
+ // https://www.webmproject.org/docs/encoder-parameters/
59
+ // 2. Use the following command to map the options to ffmpeg arguments.
60
+ // $ ./third_party/ffmpeg/ffmpeg-mac -h encoder=vp8
61
+ // 3. A bit more about passing vp8 options to ffmpeg.
62
+ // https://trac.ffmpeg.org/wiki/Encode/VP8
63
+ // 4. Tuning for VP9:
64
+ // https://developers.google.com/media/vp9/live-encoding
65
+ //
66
+ // How to stress-test video recording (runs 10 recorders in parallel to book all cpus available):
67
+ // $ node ./utils/video_stress.js
68
+ //
69
+ // We use the following vp8 options:
70
+ // "-qmin 0 -qmax 50" - quality variation from 0 to 50.
71
+ // Suggested here: https://trac.ffmpeg.org/wiki/Encode/VP8
72
+ // "-crf 8" - constant quality mode, 4-63, lower means better quality.
73
+ // "-deadline realtime -speed 8" - do not use too much cpu to keep up with incoming frames.
74
+ // "-b:v 1M" - video bitrate. Default value is too low for vp8
75
+ // Suggested here: https://trac.ffmpeg.org/wiki/Encode/VP8
76
+ // Note that we can switch to "-qmin 20 -qmax 50 -crf 30" for smaller video size but worse quality.
77
+ //
78
+ // We use "pad" and "crop" video filters (-vf option) to resize incoming frames
79
+ // that might be of the different size to the desired video size.
80
+ // https://ffmpeg.org/ffmpeg-filters.html#pad-1
81
+ // https://ffmpeg.org/ffmpeg-filters.html#crop
82
+ //
83
+ // We use "image2pipe" mode to pipe frames and get a single video - https://trac.ffmpeg.org/wiki/Slideshow
84
+ // "-f image2pipe -c:v mjpeg -i -" forces input to be read from standard input, and forces
85
+ // mjpeg input image format.
86
+ // "-avioflags direct" reduces general buffering.
87
+ // "-fpsprobesize 0 -probesize 32 -analyzeduration 0" reduces initial buffering
88
+ // while analyzing input fps and other stats.
89
+ //
90
+ // "-y" means overwrite output.
91
+ // "-an" means no audio.
92
+ // "-threads 1" means using one thread. This drastically reduces stalling when
93
+ // cpu is overbooked. By default vp8 tries to use all available threads?
94
+
95
+ const w = options.width;
96
+ const h = options.height;
97
+ const args = `-loglevel error -f image2pipe -avioflags direct -fpsprobesize 0 -probesize 32 -analyzeduration 0 -c:v mjpeg -i pipe:0 -y -an -r ${fps} -c:v vp8 -qmin 0 -qmax 50 -crf 8 -deadline realtime -speed 8 -b:v 1M -threads 1 -vf pad=${w}:${h}:0:0:gray,crop=${w}:${h}:0:0`.split(' ');
98
+ args.push(options.outputFile);
99
+ const progress = this._progress;
100
+ const {
101
+ launchedProcess,
102
+ gracefullyClose
103
+ } = await (0, _processLauncher.launchProcess)({
104
+ command: this._ffmpegPath,
105
+ args,
106
+ stdio: 'stdin',
107
+ log: message => progress.log(message),
108
+ tempDirectories: [],
109
+ attemptToGracefullyClose: async () => {
110
+ progress.log('Closing stdin...');
111
+ launchedProcess.stdin.end();
112
+ },
113
+ onExit: (exitCode, signal) => {
114
+ progress.log(`ffmpeg onkill exitCode=${exitCode} signal=${signal}`);
115
+ }
116
+ });
117
+ launchedProcess.stdin.on('finish', () => {
118
+ progress.log('ffmpeg finished input.');
119
+ });
120
+ launchedProcess.stdin.on('error', () => {
121
+ progress.log('ffmpeg error.');
122
+ });
123
+ this._process = launchedProcess;
124
+ this._gracefullyClose = gracefullyClose;
125
+ }
126
+ writeFrame(frame, timestamp) {
127
+ (0, _utils.assert)(this._process);
128
+ if (this._isStopped) return;
129
+ if (this._lastFrameBuffer) {
130
+ const durationSec = timestamp - this._lastFrameTimestamp;
131
+ const repeatCount = Math.max(1, Math.round(fps * durationSec));
132
+ for (let i = 0; i < repeatCount; ++i) this._frameQueue.push(this._lastFrameBuffer);
133
+ this._lastWritePromise = this._lastWritePromise.then(() => this._sendFrames());
134
+ }
135
+ this._lastFrameBuffer = frame;
136
+ this._lastFrameTimestamp = timestamp;
137
+ this._lastWriteTimestamp = (0, _utils.monotonicTime)();
138
+ }
139
+ async _sendFrames() {
140
+ while (this._frameQueue.length) await this._sendFrame(this._frameQueue.shift());
141
+ }
142
+ async _sendFrame(frame) {
143
+ return new Promise(f => this._process.stdin.write(frame, f)).then(error => {
144
+ if (error) this._progress.log(`ffmpeg failed to write: ${String(error)}`);
145
+ });
146
+ }
147
+ async stop() {
148
+ if (this._isStopped) return;
149
+ this.writeFrame(Buffer.from([]), this._lastFrameTimestamp + ((0, _utils.monotonicTime)() - this._lastWriteTimestamp) / 1000);
150
+ this._isStopped = true;
151
+ await this._lastWritePromise;
152
+ await this._gracefullyClose();
153
+ }
154
+ }
155
+ exports.VideoRecorder = VideoRecorder;
@@ -0,0 +1,133 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.Clock = void 0;
7
+ var clockSource = _interopRequireWildcard(require("../generated/clockSource"));
8
+ function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function (e) { return e ? t : r; })(e); }
9
+ function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && Object.prototype.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; }
10
+ /**
11
+ * Copyright (c) Microsoft Corporation.
12
+ *
13
+ * Licensed under the Apache License, Version 2.0 (the "License");
14
+ * you may not use this file except in compliance with the License.
15
+ * You may obtain a copy of the License at
16
+ *
17
+ * http://www.apache.org/licenses/LICENSE-2.0
18
+ *
19
+ * Unless required by applicable law or agreed to in writing, software
20
+ * distributed under the License is distributed on an "AS IS" BASIS,
21
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
22
+ * See the License for the specific language governing permissions and
23
+ * limitations under the License.
24
+ */
25
+
26
+ class Clock {
27
+ constructor(browserContext) {
28
+ this._browserContext = void 0;
29
+ this._scriptInstalled = false;
30
+ this._browserContext = browserContext;
31
+ }
32
+ markAsUninstalled() {
33
+ this._scriptInstalled = false;
34
+ }
35
+ async fastForward(ticks) {
36
+ await this._installIfNeeded();
37
+ const ticksMillis = parseTicks(ticks);
38
+ await this._browserContext.addInitScript(`globalThis.__pwClock.controller.log('fastForward', ${Date.now()}, ${ticksMillis})`);
39
+ await this._evaluateInFrames(`globalThis.__pwClock.controller.fastForward(${ticksMillis})`);
40
+ }
41
+ async install(time) {
42
+ await this._installIfNeeded();
43
+ const timeMillis = time !== undefined ? parseTime(time) : Date.now();
44
+ await this._browserContext.addInitScript(`globalThis.__pwClock.controller.log('install', ${Date.now()}, ${timeMillis})`);
45
+ await this._evaluateInFrames(`globalThis.__pwClock.controller.install(${timeMillis})`);
46
+ }
47
+ async pauseAt(ticks) {
48
+ await this._installIfNeeded();
49
+ const timeMillis = parseTime(ticks);
50
+ await this._browserContext.addInitScript(`globalThis.__pwClock.controller.log('pauseAt', ${Date.now()}, ${timeMillis})`);
51
+ await this._evaluateInFrames(`globalThis.__pwClock.controller.pauseAt(${timeMillis})`);
52
+ }
53
+ async resume() {
54
+ await this._installIfNeeded();
55
+ await this._browserContext.addInitScript(`globalThis.__pwClock.controller.log('resume', ${Date.now()})`);
56
+ await this._evaluateInFrames(`globalThis.__pwClock.controller.resume()`);
57
+ }
58
+ async setFixedTime(time) {
59
+ await this._installIfNeeded();
60
+ const timeMillis = parseTime(time);
61
+ await this._browserContext.addInitScript(`globalThis.__pwClock.controller.log('setFixedTime', ${Date.now()}, ${timeMillis})`);
62
+ await this._evaluateInFrames(`globalThis.__pwClock.controller.setFixedTime(${timeMillis})`);
63
+ }
64
+ async setSystemTime(time) {
65
+ await this._installIfNeeded();
66
+ const timeMillis = parseTime(time);
67
+ await this._browserContext.addInitScript(`globalThis.__pwClock.controller.log('setSystemTime', ${Date.now()}, ${timeMillis})`);
68
+ await this._evaluateInFrames(`globalThis.__pwClock.controller.setSystemTime(${timeMillis})`);
69
+ }
70
+ async runFor(ticks) {
71
+ await this._installIfNeeded();
72
+ const ticksMillis = parseTicks(ticks);
73
+ await this._browserContext.addInitScript(`globalThis.__pwClock.controller.log('runFor', ${Date.now()}, ${ticksMillis})`);
74
+ await this._evaluateInFrames(`globalThis.__pwClock.controller.runFor(${ticksMillis})`);
75
+ }
76
+ async _installIfNeeded() {
77
+ if (this._scriptInstalled) return;
78
+ this._scriptInstalled = true;
79
+ const script = `(() => {
80
+ const module = {};
81
+ ${clockSource.source}
82
+ globalThis.__pwClock = (module.exports.inject())(globalThis);
83
+ })();`;
84
+ await this._browserContext.addInitScript(script);
85
+ await this._evaluateInFrames(script);
86
+ }
87
+ async _evaluateInFrames(script) {
88
+ // Dont ask me why this works
89
+ await Promise.all(this._browserContext.pages().map(async page => {
90
+ await Promise.all(page.frames().map(async frame => {
91
+ try {
92
+ await frame.evaluateExpression("");
93
+ } catch (e) {}
94
+ }));
95
+ }));
96
+ await this._browserContext.safeNonStallingEvaluateInAllFrames(script, 'main', {
97
+ throwOnJSErrors: true
98
+ });
99
+ }
100
+ }
101
+
102
+ /**
103
+ * Parse strings like '01:10:00' (meaning 1 hour, 10 minutes, 0 seconds) into
104
+ * number of milliseconds. This is used to support human-readable strings passed
105
+ * to clock.tick()
106
+ */
107
+ exports.Clock = Clock;
108
+ function parseTicks(value) {
109
+ if (typeof value === 'number') return value;
110
+ if (!value) return 0;
111
+ const str = value;
112
+ const strings = str.split(':');
113
+ const l = strings.length;
114
+ let i = l;
115
+ let ms = 0;
116
+ let parsed;
117
+ if (l > 3 || !/^(\d\d:){0,2}\d\d?$/.test(str)) {
118
+ throw new Error(`Clock only understands numbers, 'mm:ss' and 'hh:mm:ss'`);
119
+ }
120
+ while (i--) {
121
+ parsed = parseInt(strings[i], 10);
122
+ if (parsed >= 60) throw new Error(`Invalid time ${str}`);
123
+ ms += parsed * Math.pow(60, l - i - 1);
124
+ }
125
+ return ms * 1000;
126
+ }
127
+ function parseTime(epoch) {
128
+ if (!epoch) return 0;
129
+ if (typeof epoch === 'number') return epoch;
130
+ const parsed = new Date(epoch);
131
+ if (!isFinite(parsed.getTime())) throw new Error(`Invalid date: ${epoch}`);
132
+ return parsed.getTime();
133
+ }
@@ -0,0 +1,299 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.CSharpLanguageGenerator = void 0;
7
+ var _language = require("./language");
8
+ var _utils = require("../../utils");
9
+ var _deviceDescriptors = require("../deviceDescriptors");
10
+ /**
11
+ * Copyright (c) Microsoft Corporation.
12
+ *
13
+ * Licensed under the Apache License, Version 2.0 (the "License");
14
+ * you may not use this file except in compliance with the License.
15
+ * You may obtain a copy of the License at
16
+ *
17
+ * http://www.apache.org/licenses/LICENSE-2.0
18
+ *
19
+ * Unless required by applicable law or agreed to in writing, software
20
+ * distributed under the License is distributed on an "AS IS" BASIS,
21
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
22
+ * See the License for the specific language governing permissions and
23
+ * limitations under the License.
24
+ */
25
+
26
+ class CSharpLanguageGenerator {
27
+ constructor(mode) {
28
+ this.id = void 0;
29
+ this.groupName = '.NET C#';
30
+ this.name = void 0;
31
+ this.highlighter = 'csharp';
32
+ this._mode = void 0;
33
+ if (mode === 'library') {
34
+ this.name = 'Library';
35
+ this.id = 'csharp';
36
+ } else if (mode === 'mstest') {
37
+ this.name = 'MSTest';
38
+ this.id = 'csharp-mstest';
39
+ } else if (mode === 'nunit') {
40
+ this.name = 'NUnit';
41
+ this.id = 'csharp-nunit';
42
+ } else {
43
+ throw new Error(`Unknown C# language mode: ${mode}`);
44
+ }
45
+ this._mode = mode;
46
+ }
47
+ generateAction(actionInContext) {
48
+ const action = this._generateActionInner(actionInContext);
49
+ if (action) return action;
50
+ return '';
51
+ }
52
+ _generateActionInner(actionInContext) {
53
+ const action = actionInContext.action;
54
+ if (this._mode !== 'library' && (action.name === 'openPage' || action.name === 'closePage')) return '';
55
+ let pageAlias = actionInContext.frame.pageAlias;
56
+ if (this._mode !== 'library') pageAlias = pageAlias.replace('page', 'Page');
57
+ const formatter = new CSharpFormatter(this._mode === 'library' ? 0 : 8);
58
+ if (action.name === 'openPage') {
59
+ formatter.add(`var ${pageAlias} = await context.NewPageAsync();`);
60
+ if (action.url && action.url !== 'about:blank' && action.url !== 'chrome://newtab/') formatter.add(`await ${pageAlias}.GotoAsync(${quote(action.url)});`);
61
+ return formatter.format();
62
+ }
63
+ const locators = actionInContext.frame.framePath.map(selector => `.${this._asLocator(selector)}.ContentFrame`);
64
+ const subject = `${pageAlias}${locators.join('')}`;
65
+ const signals = (0, _language.toSignalMap)(action);
66
+ if (signals.dialog) {
67
+ formatter.add(` void ${pageAlias}_Dialog${signals.dialog.dialogAlias}_EventHandler(object sender, IDialog dialog)
68
+ {
69
+ Console.WriteLine($"Dialog message: {dialog.Message}");
70
+ dialog.DismissAsync();
71
+ ${pageAlias}.Dialog -= ${pageAlias}_Dialog${signals.dialog.dialogAlias}_EventHandler;
72
+ }
73
+ ${pageAlias}.Dialog += ${pageAlias}_Dialog${signals.dialog.dialogAlias}_EventHandler;`);
74
+ }
75
+ const lines = [];
76
+ lines.push(this._generateActionCall(subject, actionInContext));
77
+ if (signals.download) {
78
+ lines.unshift(`var download${signals.download.downloadAlias} = await ${pageAlias}.RunAndWaitForDownloadAsync(async () =>\n{`);
79
+ lines.push(`});`);
80
+ }
81
+ if (signals.popup) {
82
+ lines.unshift(`var ${signals.popup.popupAlias} = await ${pageAlias}.RunAndWaitForPopupAsync(async () =>\n{`);
83
+ lines.push(`});`);
84
+ }
85
+ for (const line of lines) formatter.add(line);
86
+ return formatter.format();
87
+ }
88
+ _generateActionCall(subject, actionInContext) {
89
+ const action = actionInContext.action;
90
+ switch (action.name) {
91
+ case 'openPage':
92
+ throw Error('Not reached');
93
+ case 'closePage':
94
+ return `await ${subject}.CloseAsync();`;
95
+ case 'click':
96
+ {
97
+ let method = 'Click';
98
+ if (action.clickCount === 2) method = 'DblClick';
99
+ const options = (0, _language.toClickOptionsForSourceCode)(action);
100
+ if (!Object.entries(options).length) return `await ${subject}.${this._asLocator(action.selector)}.${method}Async();`;
101
+ const optionsString = formatObject(options, ' ', 'Locator' + method + 'Options');
102
+ return `await ${subject}.${this._asLocator(action.selector)}.${method}Async(${optionsString});`;
103
+ }
104
+ case 'check':
105
+ return `await ${subject}.${this._asLocator(action.selector)}.CheckAsync();`;
106
+ case 'uncheck':
107
+ return `await ${subject}.${this._asLocator(action.selector)}.UncheckAsync();`;
108
+ case 'fill':
109
+ return `await ${subject}.${this._asLocator(action.selector)}.FillAsync(${quote(action.text)});`;
110
+ case 'setInputFiles':
111
+ return `await ${subject}.${this._asLocator(action.selector)}.SetInputFilesAsync(${formatObject(action.files)});`;
112
+ case 'press':
113
+ {
114
+ const modifiers = (0, _language.toKeyboardModifiers)(action.modifiers);
115
+ const shortcut = [...modifiers, action.key].join('+');
116
+ return `await ${subject}.${this._asLocator(action.selector)}.PressAsync(${quote(shortcut)});`;
117
+ }
118
+ case 'navigate':
119
+ return `await ${subject}.GotoAsync(${quote(action.url)});`;
120
+ case 'select':
121
+ return `await ${subject}.${this._asLocator(action.selector)}.SelectOptionAsync(${formatObject(action.options)});`;
122
+ case 'assertText':
123
+ return `await Expect(${subject}.${this._asLocator(action.selector)}).${action.substring ? 'ToContainTextAsync' : 'ToHaveTextAsync'}(${quote(action.text)});`;
124
+ case 'assertChecked':
125
+ return `await Expect(${subject}.${this._asLocator(action.selector)})${action.checked ? '' : '.Not'}.ToBeCheckedAsync();`;
126
+ case 'assertVisible':
127
+ return `await Expect(${subject}.${this._asLocator(action.selector)}).ToBeVisibleAsync();`;
128
+ case 'assertValue':
129
+ {
130
+ const assertion = action.value ? `ToHaveValueAsync(${quote(action.value)})` : `ToBeEmptyAsync()`;
131
+ return `await Expect(${subject}.${this._asLocator(action.selector)}).${assertion};`;
132
+ }
133
+ }
134
+ }
135
+ _asLocator(selector) {
136
+ return (0, _utils.asLocator)('csharp', selector);
137
+ }
138
+ generateHeader(options) {
139
+ if (this._mode === 'library') return this.generateStandaloneHeader(options);
140
+ return this.generateTestRunnerHeader(options);
141
+ }
142
+ generateStandaloneHeader(options) {
143
+ const formatter = new CSharpFormatter(0);
144
+ formatter.add(`
145
+ using Microsoft.Playwright;
146
+ using System;
147
+ using System.Threading.Tasks;
148
+
149
+ using var playwright = await Playwright.CreateAsync();
150
+ await using var browser = await playwright.${toPascal(options.browserName)}.LaunchAsync(${formatObject(options.launchOptions, ' ', 'BrowserTypeLaunchOptions')});
151
+ var context = await browser.NewContextAsync(${formatContextOptions(options.contextOptions, options.deviceName)});`);
152
+ formatter.newLine();
153
+ return formatter.format();
154
+ }
155
+ generateTestRunnerHeader(options) {
156
+ const formatter = new CSharpFormatter(0);
157
+ formatter.add(`
158
+ using Microsoft.Playwright.${this._mode === 'nunit' ? 'NUnit' : 'MSTest'};
159
+ using Microsoft.Playwright;
160
+
161
+ ${this._mode === 'nunit' ? `[Parallelizable(ParallelScope.Self)]
162
+ [TestFixture]` : '[TestClass]'}
163
+ public class Tests : PageTest
164
+ {`);
165
+ const formattedContextOptions = formatContextOptions(options.contextOptions, options.deviceName);
166
+ if (formattedContextOptions) {
167
+ formatter.add(`public override BrowserNewContextOptions ContextOptions()
168
+ {
169
+ return ${formattedContextOptions};
170
+ }`);
171
+ formatter.newLine();
172
+ }
173
+ formatter.add(` [${this._mode === 'nunit' ? 'Test' : 'TestMethod'}]
174
+ public async Task MyTest()
175
+ {`);
176
+ return formatter.format();
177
+ }
178
+ generateFooter(saveStorage) {
179
+ const offset = this._mode === 'library' ? '' : ' ';
180
+ let storageStateLine = saveStorage ? `\n${offset}await context.StorageStateAsync(new BrowserContextStorageStateOptions\n${offset}{\n${offset} Path = ${quote(saveStorage)}\n${offset}});\n` : '';
181
+ if (this._mode !== 'library') storageStateLine += ` }\n}\n`;
182
+ return storageStateLine;
183
+ }
184
+ }
185
+ exports.CSharpLanguageGenerator = CSharpLanguageGenerator;
186
+ function formatObject(value, indent = ' ', name = '') {
187
+ if (typeof value === 'string') {
188
+ if (['permissions', 'colorScheme', 'modifiers', 'button', 'recordHarContent', 'recordHarMode', 'serviceWorkers'].includes(name)) return `${getClassName(name)}.${toPascal(value)}`;
189
+ return quote(value);
190
+ }
191
+ if (Array.isArray(value)) return `new[] { ${value.map(o => formatObject(o, indent, name)).join(', ')} }`;
192
+ if (typeof value === 'object') {
193
+ const keys = Object.keys(value).filter(key => value[key] !== undefined).sort();
194
+ if (!keys.length) return name ? `new ${getClassName(name)}` : '';
195
+ const tokens = [];
196
+ for (const key of keys) {
197
+ const property = getPropertyName(key);
198
+ tokens.push(`${property} = ${formatObject(value[key], indent, key)},`);
199
+ }
200
+ if (name) return `new ${getClassName(name)}\n{\n${indent}${tokens.join(`\n${indent}`)}\n${indent}}`;
201
+ return `{\n${indent}${tokens.join(`\n${indent}`)}\n${indent}}`;
202
+ }
203
+ if (name === 'latitude' || name === 'longitude') return String(value) + 'm';
204
+ return String(value);
205
+ }
206
+ function getClassName(value) {
207
+ switch (value) {
208
+ case 'viewport':
209
+ return 'ViewportSize';
210
+ case 'proxy':
211
+ return 'ProxySettings';
212
+ case 'permissions':
213
+ return 'ContextPermission';
214
+ case 'modifiers':
215
+ return 'KeyboardModifier';
216
+ case 'button':
217
+ return 'MouseButton';
218
+ case 'recordHarMode':
219
+ return 'HarMode';
220
+ case 'recordHarContent':
221
+ return 'HarContentPolicy';
222
+ case 'serviceWorkers':
223
+ return 'ServiceWorkerPolicy';
224
+ default:
225
+ return toPascal(value);
226
+ }
227
+ }
228
+ function getPropertyName(key) {
229
+ switch (key) {
230
+ case 'storageState':
231
+ return 'StorageStatePath';
232
+ case 'viewport':
233
+ return 'ViewportSize';
234
+ default:
235
+ return toPascal(key);
236
+ }
237
+ }
238
+ function toPascal(value) {
239
+ return value[0].toUpperCase() + value.slice(1);
240
+ }
241
+ function convertContextOptions(options) {
242
+ const result = {
243
+ ...options
244
+ };
245
+ if (options.recordHar) {
246
+ result['recordHarPath'] = options.recordHar.path;
247
+ result['recordHarContent'] = options.recordHar.content;
248
+ result['recordHarMode'] = options.recordHar.mode;
249
+ result['recordHarOmitContent'] = options.recordHar.omitContent;
250
+ result['recordHarUrlFilter'] = options.recordHar.urlFilter;
251
+ delete result.recordHar;
252
+ }
253
+ return result;
254
+ }
255
+ function formatContextOptions(options, deviceName) {
256
+ const device = deviceName && _deviceDescriptors.deviceDescriptors[deviceName];
257
+ if (!device) {
258
+ if (!Object.entries(options).length) return '';
259
+ return formatObject(convertContextOptions(options), ' ', 'BrowserNewContextOptions');
260
+ }
261
+ options = (0, _language.sanitizeDeviceOptions)(device, options);
262
+ if (!Object.entries(options).length) return `playwright.Devices[${quote(deviceName)}]`;
263
+ return formatObject(convertContextOptions(options), ' ', `BrowserNewContextOptions(playwright.Devices[${quote(deviceName)}])`);
264
+ }
265
+ class CSharpFormatter {
266
+ constructor(offset = 0) {
267
+ this._baseIndent = void 0;
268
+ this._baseOffset = void 0;
269
+ this._lines = [];
270
+ this._baseIndent = ' '.repeat(4);
271
+ this._baseOffset = ' '.repeat(offset);
272
+ }
273
+ prepend(text) {
274
+ this._lines = text.trim().split('\n').map(line => line.trim()).concat(this._lines);
275
+ }
276
+ add(text) {
277
+ this._lines.push(...text.trim().split('\n').map(line => line.trim()));
278
+ }
279
+ newLine() {
280
+ this._lines.push('');
281
+ }
282
+ format() {
283
+ let spaces = '';
284
+ let previousLine = '';
285
+ return this._lines.map(line => {
286
+ if (line === '') return line;
287
+ if (line.startsWith('}') || line.startsWith(']') || line.includes('});') || line === ');') spaces = spaces.substring(this._baseIndent.length);
288
+ const extraSpaces = /^(for|while|if).*\(.*\)$/.test(previousLine) ? this._baseIndent : '';
289
+ previousLine = line;
290
+ line = spaces + extraSpaces + line;
291
+ if (line.endsWith('{') || line.endsWith('[') || line.endsWith('(')) spaces += this._baseIndent;
292
+ if (line.endsWith('));')) spaces = spaces.substring(this._baseIndent.length);
293
+ return this._baseOffset + line;
294
+ }).join('\n');
295
+ }
296
+ }
297
+ function quote(text) {
298
+ return (0, _utils.escapeWithQuotes)(text, '\"');
299
+ }