ui-options.js 85 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558
  1. /*
  2. * Copyright 2010-2020 Gildas Lormeau
  3. * contact : gildas.lormeau <at> gmail.com
  4. *
  5. * This file is part of SingleFile.
  6. *
  7. * The code in this file is free software: you can redistribute it and/or
  8. * modify it under the terms of the GNU Affero General Public License
  9. * (GNU AGPL) as published by the Free Software Foundation, either version 3
  10. * of the License, or (at your option) any later version.
  11. *
  12. * The code in this file is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero
  15. * General Public License for more details.
  16. *
  17. * As additional permission under GNU AGPL version 3 section 7, you may
  18. * distribute UNMODIFIED VERSIONS OF THIS file without the copy of the GNU
  19. * AGPL normally required by section 4, provided you include this license
  20. * notice and a URL through which recipients can access the Corresponding
  21. * Source.
  22. */
  23. /* global browser, window, document, localStorage, FileReader, location, fetch, TextDecoder, DOMParser, HTMLElement, MouseEvent */
  24. const HELP_ICON_URL = "data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABQAAAAUCAYAAACNiR0NAAABIUlEQVQ4y+2TsarCMBSGvxTBRdqiUZAWOrhJB9EXcPKFfCvfQYfulUKHDqXg4CYUJSioYO4mSDX3ttzt3n87fMlHTpIjlsulxpDZbEYYhgghSNOUOI5Ny2mZYBAELBYLer0eAJ7ncTweKYri4x7LJJRS0u12n7XrukgpjSc0CpVSXK/XZ32/31FKNW85z3PW6zXT6RSAJEnIsqy5UGvNZrNhu90CcDqd+C6tT6J+v//2Th+PB2VZ1hN2Oh3G4zGTyQTbtl/YbrdjtVpxu91+Ljyfz0RRhG3bzOfzF+Y4TvNXvlwuaK2pE4tfzr/wzwsty0IIURlL0998KxRCMBqN8H2/wlzXJQxD2u12vVkeDoeUZUkURRU+GAw4HA7s9/sK+wK6CWHasQ/S/wAAAABJRU5ErkJggg==";
  25. const HELP_PAGE_PATH = "/src/ui/pages/help.html";
  26. let DEFAULT_PROFILE_NAME,
  27. DISABLED_PROFILE_NAME,
  28. CURRENT_PROFILE_NAME,
  29. BACKGROUND_SAVE_SUPPORTED,
  30. AUTO_SAVE_SUPPORTED,
  31. AUTO_OPEN_EDITOR_SUPPORTED,
  32. INFOBAR_SUPPORTED,
  33. BOOKMARKS_API_SUPPORTED,
  34. IDENTITY_API_SUPPORTED,
  35. CLIPBOARD_API_SUPPORTED,
  36. NATIVE_API_API_SUPPORTED,
  37. WEB_BLOCKING_API_SUPPORTED,
  38. SHARE_API_SUPPORTED;
  39. browser.runtime.sendMessage({ method: "config.getConstants" }).then(data => {
  40. ({
  41. DEFAULT_PROFILE_NAME,
  42. DISABLED_PROFILE_NAME,
  43. CURRENT_PROFILE_NAME,
  44. BACKGROUND_SAVE_SUPPORTED,
  45. AUTO_SAVE_SUPPORTED,
  46. AUTO_OPEN_EDITOR_SUPPORTED,
  47. INFOBAR_SUPPORTED,
  48. BOOKMARKS_API_SUPPORTED,
  49. IDENTITY_API_SUPPORTED,
  50. CLIPBOARD_API_SUPPORTED,
  51. NATIVE_API_API_SUPPORTED,
  52. WEB_BLOCKING_API_SUPPORTED,
  53. SHARE_API_SUPPORTED
  54. } = data);
  55. init();
  56. });
  57. const removeHiddenElementsLabel = document.getElementById("removeHiddenElementsLabel");
  58. const removeUnusedStylesLabel = document.getElementById("removeUnusedStylesLabel");
  59. const removeUnusedFontsLabel = document.getElementById("removeUnusedFontsLabel");
  60. const removeFramesLabel = document.getElementById("removeFramesLabel");
  61. const blockScriptsLabel = document.getElementById("blockScriptsLabel");
  62. const blockAudiosLabel = document.getElementById("blockAudiosLabel");
  63. const blockVideosLabel = document.getElementById("blockVideosLabel");
  64. const blockFontsLabel = document.getElementById("blockFontsLabel");
  65. const blockStylesheetsLabel = document.getElementById("blockStylesheetsLabel");
  66. const blockImagesLabel = document.getElementById("blockImagesLabel");
  67. const acceptHeaderDocumentLabel = document.getElementById("acceptHeaderDocumentLabel");
  68. const acceptHeaderScriptLabel = document.getElementById("acceptHeaderScriptLabel");
  69. const acceptHeaderAudioLabel = document.getElementById("acceptHeaderAudioLabel");
  70. const acceptHeaderVideoLabel = document.getElementById("acceptHeaderVideoLabel");
  71. const acceptHeaderFontLabel = document.getElementById("acceptHeaderFontLabel");
  72. const acceptHeaderStylesheetLabel = document.getElementById("acceptHeaderStylesheetLabel");
  73. const acceptHeaderImageLabel = document.getElementById("acceptHeaderImageLabel");
  74. const saveRawPageLabel = document.getElementById("saveRawPageLabel");
  75. const insertMetaCSPLabel = document.getElementById("insertMetaCSPLabel");
  76. const saveToClipboardLabel = document.getElementById("saveToClipboardLabel");
  77. const saveToFilesystemLabel = document.getElementById("saveToFilesystemLabel");
  78. const sharePageLabel = document.getElementById("sharePageLabel");
  79. const addProofLabel = document.getElementById("addProofLabel");
  80. const woleetKeyLabel = document.getElementById("woleetKeyLabel");
  81. const saveToGDriveLabel = document.getElementById("saveToGDriveLabel");
  82. const saveToDropboxLabel = document.getElementById("saveToDropboxLabel");
  83. const saveWithWebDAVLabel = document.getElementById("saveWithWebDAVLabel");
  84. const webDAVURLLabel = document.getElementById("webDAVURLLabel");
  85. const webDAVUserLabel = document.getElementById("webDAVUserLabel");
  86. const webDAVPasswordLabel = document.getElementById("webDAVPasswordLabel");
  87. const saveToGitHubLabel = document.getElementById("saveToGitHubLabel");
  88. const githubTokenLabel = document.getElementById("githubTokenLabel");
  89. const githubUserLabel = document.getElementById("githubUserLabel");
  90. const githubRepositoryLabel = document.getElementById("githubRepositoryLabel");
  91. const githubBranchLabel = document.getElementById("githubBranchLabel");
  92. const saveWithCompanionLabel = document.getElementById("saveWithCompanionLabel");
  93. const saveToS3Label = document.getElementById("saveToS3Label");
  94. const S3DomainLabel = document.getElementById("S3DomainLabel");
  95. const S3RegionLabel = document.getElementById("S3RegionLabel");
  96. const S3BucketLabel = document.getElementById("S3BucketLabel");
  97. const S3AccessKeyLabel = document.getElementById("S3AccessKeyLabel");
  98. const S3SecretKeyLabel = document.getElementById("S3SecretKeyLabel");
  99. const compressHTMLLabel = document.getElementById("compressHTMLLabel");
  100. const insertTextBodyLabel = document.getElementById("insertTextBodyLabel");
  101. const insertEmbeddedImageLabel = document.getElementById("insertEmbeddedImageLabel");
  102. const insertEmbeddedCustomImageLabel = document.getElementById("insertEmbeddedCustomImageLabel");
  103. const insertEmbeddedScreenshotImageLabel = document.getElementById("insertEmbeddedScreenshotImageLabel");
  104. const compressCSSLabel = document.getElementById("compressCSSLabel");
  105. const moveStylesInHeadLabel = document.getElementById("moveStylesInHeadLabel");
  106. const loadDeferredImagesLabel = document.getElementById("loadDeferredImagesLabel");
  107. const loadDeferredImagesMaxIdleTimeLabel = document.getElementById("loadDeferredImagesMaxIdleTimeLabel");
  108. const loadDeferredImagesKeepZoomLevelLabel = document.getElementById("loadDeferredImagesKeepZoomLevelLabel");
  109. const loadDeferredImagesDispatchScrollEventLabel = document.getElementById("loadDeferredImagesDispatchScrollEventLabel");
  110. const loadDeferredImagesBeforeFramesLabel = document.getElementById("loadDeferredImagesBeforeFramesLabel");
  111. const addMenuEntryLabel = document.getElementById("addMenuEntryLabel");
  112. const filenameTemplateLabel = document.getElementById("filenameTemplateLabel");
  113. const filenameMaxLengthLabel = document.getElementById("filenameMaxLengthLabel");
  114. const filenameMaxLengthBytesUnitLabel = document.getElementById("filenameMaxLengthBytesUnitLabel");
  115. const filenameMaxLengthCharsUnitLabel = document.getElementById("filenameMaxLengthCharsUnitLabel");
  116. const filenameReplacementCharacterLabel = document.getElementById("filenameReplacementCharacterLabel");
  117. const replaceEmojisInFilenameLabel = document.getElementById("replaceEmojisInFilenameLabel");
  118. const saveFilenameTemplateDataLabel = document.getElementById("saveFilenameTemplateDataLabel");
  119. const shadowEnabledLabel = document.getElementById("shadowEnabledLabel");
  120. const setMaxResourceSizeLabel = document.getElementById("setMaxResourceSizeLabel");
  121. const maxResourceSizeLabel = document.getElementById("maxResourceSizeLabel");
  122. const setMaxResourceDelayLabel = document.getElementById("setMaxResourceDelayLabel");
  123. const maxResourceDelayLabel = document.getElementById("maxResourceDelayLabel");
  124. const confirmFilenameLabel = document.getElementById("confirmFilenameLabel");
  125. const filenameConflictActionLabel = document.getElementById("filenameConflictActionLabel");
  126. const filenameConflictActionUniquifyLabel = document.getElementById("filenameConflictActionUniquifyLabel");
  127. const filenameConflictActionOverwriteLabel = document.getElementById("filenameConflictActionOverwriteLabel");
  128. const filenameConflictActionPromptLabel = document.getElementById("filenameConflictActionPromptLabel");
  129. const filenameConflictActionSkipLabel = document.getElementById("filenameConflictActionSkipLabel");
  130. const displayInfobarLabel = document.getElementById("displayInfobarLabel");
  131. const displayStatsLabel = document.getElementById("displayStatsLabel");
  132. const backgroundSaveLabel = document.getElementById("backgroundSaveLabel");
  133. const autoSaveDelayLabel = document.getElementById("autoSaveDelayLabel");
  134. const autoSaveLoadLabel = document.getElementById("autoSaveLoadLabel");
  135. const autoSaveUnloadLabel = document.getElementById("autoSaveUnloadLabel");
  136. const autoSaveLoadOrUnloadLabel = document.getElementById("autoSaveLoadOrUnloadLabel");
  137. const autoSaveDiscardLabel = document.getElementById("autoSaveDiscardLabel");
  138. const autoSaveRemoveLabel = document.getElementById("autoSaveRemoveLabel");
  139. const autoSaveRepeatLabel = document.getElementById("autoSaveRepeatLabel");
  140. const autoSaveRepeatDelayLabel = document.getElementById("autoSaveRepeatDelayLabel");
  141. const autoSaveExternalSaveLabel = document.getElementById("autoSaveExternalSaveLabel");
  142. const removeAlternativeFontsLabel = document.getElementById("removeAlternativeFontsLabel");
  143. const removeAlternativeImagesLabel = document.getElementById("removeAlternativeImagesLabel");
  144. const removeAlternativeMediasLabel = document.getElementById("removeAlternativeMediasLabel");
  145. const saveCreatedBookmarksLabel = document.getElementById("saveCreatedBookmarksLabel");
  146. const passReferrerOnErrorLabel = document.getElementById("passReferrerOnErrorLabel");
  147. const replaceBookmarkURLLabel = document.getElementById("replaceBookmarkURLLabel");
  148. const allowedBookmarkFoldersLabel = document.getElementById("allowedBookmarkFoldersLabel");
  149. const ignoredBookmarkFoldersLabel = document.getElementById("ignoredBookmarkFoldersLabel");
  150. const createRootDirectoryLabel = document.getElementById("createRootDirectoryLabel");
  151. const preventAppendedDataLabel = document.getElementById("preventAppendedDataLabel");
  152. const passwordLabel = document.getElementById("passwordLabel");
  153. const titleLabel = document.getElementById("titleLabel");
  154. const userInterfaceLabel = document.getElementById("userInterfaceLabel");
  155. const filenameLabel = document.getElementById("filenameLabel");
  156. const htmlContentLabel = document.getElementById("htmlContentLabel");
  157. const fileFormatLabel = document.getElementById("fileFormatLabel");
  158. const fileFormatSelectHTMLLabel = document.getElementById("fileFormatSelectHTMLLabel");
  159. const fileFormatSelectSelfExtractingUniversalLabel = document.getElementById("fileFormatSelectSelfExtractingUniversalLabel");
  160. const fileFormatSelectSelfExtractingLabel = document.getElementById("fileFormatSelectSelfExtractingLabel");
  161. const fileFormatSelectZIPLabel = document.getElementById("fileFormatSelectZIPLabel");
  162. const fileFormatSelectLabel = document.getElementById("fileFormatSelectLabel");
  163. const infobarLabel = document.getElementById("infobarLabel");
  164. const imagesLabel = document.getElementById("imagesLabel");
  165. const stylesheetsLabel = document.getElementById("stylesheetsLabel");
  166. const fontsLabel = document.getElementById("fontsLabel");
  167. const networkLabel = document.getElementById("networkLabel");
  168. const blockResourcesLabel = document.getElementById("blockResourcesLabel");
  169. const acceptHeadersLabel = document.getElementById("acceptHeadersLabel");
  170. const destinationLabel = document.getElementById("destinationLabel");
  171. const bookmarksLabel = document.getElementById("bookmarksLabel");
  172. const autoSaveLabel = document.getElementById("autoSaveLabel");
  173. const autoSettingsLabel = document.getElementById("autoSettingsLabel");
  174. const autoSettingsUrlLabel = document.getElementById("autoSettingsUrlLabel");
  175. const autoSettingsProfileLabel = document.getElementById("autoSettingsProfileLabel");
  176. const autoSettingsAutoSaveProfileLabel = document.getElementById("autoSettingsAutoSaveProfileLabel");
  177. const showAllProfilesLabel = document.getElementById("showAllProfilesLabel");
  178. const showAutoSaveProfileLabel = document.getElementById("showAutoSaveProfileLabel");
  179. const groupDuplicateImagesLabel = document.getElementById("groupDuplicateImagesLabel");
  180. const confirmInfobarLabel = document.getElementById("confirmInfobarLabel");
  181. const autoCloseLabel = document.getElementById("autoCloseLabel");
  182. const editorLabel = document.getElementById("editorLabel");
  183. const openEditorLabel = document.getElementById("openEditorLabel");
  184. const openSavedPageLabel = document.getElementById("openSavedPageLabel");
  185. const autoOpenEditorLabel = document.getElementById("autoOpenEditorLabel");
  186. const defaultEditorModeLabel = document.getElementById("defaultEditorModeLabel");
  187. const applySystemThemeLabel = document.getElementById("applySystemThemeLabel");
  188. const warnUnsavedPageLabel = document.getElementById("warnUnsavedPageLabel");
  189. const displayInfobarInEditorLabel = document.getElementById("displayInfobarInEditorLabel");
  190. const infobarTemplateLabel = document.getElementById("infobarTemplateLabel");
  191. const blockMixedContentLabel = document.getElementById("blockMixedContentLabel");
  192. const saveOriginalURLsLabel = document.getElementById("saveOriginalURLsLabel");
  193. const includeInfobarLabel = document.getElementById("includeInfobarLabel");
  194. const openInfobarLabel = document.getElementById("openInfobarLabel");
  195. const removeInfobarSavedDateLabel = document.getElementById("removeInfobarSavedDateLabel");
  196. const miscLabel = document.getElementById("miscLabel");
  197. const helpLabel = document.getElementById("helpLabel");
  198. const synchronizeLabel = document.getElementById("synchronizeLabel");
  199. const addProfileButton = document.getElementById("addProfileButton");
  200. const deleteProfileButton = document.getElementById("deleteProfileButton");
  201. const renameProfileButton = document.getElementById("renameProfileButton");
  202. const resetButton = document.getElementById("resetButton");
  203. const exportButton = document.getElementById("exportButton");
  204. const importButton = document.getElementById("importButton");
  205. const fileInput = document.getElementById("fileInput");
  206. const profileNamesInput = document.getElementById("profileNamesInput");
  207. const removeHiddenElementsInput = document.getElementById("removeHiddenElementsInput");
  208. const removeUnusedStylesInput = document.getElementById("removeUnusedStylesInput");
  209. const removeUnusedFontsInput = document.getElementById("removeUnusedFontsInput");
  210. const removeFramesInput = document.getElementById("removeFramesInput");
  211. const blockScriptsInput = document.getElementById("blockScriptsInput");
  212. const blockVideosInput = document.getElementById("blockVideosInput");
  213. const blockAudiosInput = document.getElementById("blockAudiosInput");
  214. const blockFontsInput = document.getElementById("blockFontsInput");
  215. const blockStylesheetsInput = document.getElementById("blockStylesheetsInput");
  216. const blockImagesInput = document.getElementById("blockImagesInput");
  217. const acceptHeaderDocumentInput = document.getElementById("acceptHeaderDocumentInput");
  218. const acceptHeaderScriptInput = document.getElementById("acceptHeaderScriptInput");
  219. const acceptHeaderAudioInput = document.getElementById("acceptHeaderAudioInput");
  220. const acceptHeaderVideoInput = document.getElementById("acceptHeaderVideoInput");
  221. const acceptHeaderFontInput = document.getElementById("acceptHeaderFontInput");
  222. const acceptHeaderStylesheetInput = document.getElementById("acceptHeaderStylesheetInput");
  223. const acceptHeaderImageInput = document.getElementById("acceptHeaderImageInput");
  224. const saveRawPageInput = document.getElementById("saveRawPageInput");
  225. const insertMetaCSPInput = document.getElementById("insertMetaCSPInput");
  226. const saveToClipboardInput = document.getElementById("saveToClipboardInput");
  227. const addProofInput = document.getElementById("addProofInput");
  228. const woleetKeyInput = document.getElementById("woleetKeyInput");
  229. const saveToGDriveInput = document.getElementById("saveToGDriveInput");
  230. const saveToDropboxInput = document.getElementById("saveToDropboxInput");
  231. const saveWithWebDAVInput = document.getElementById("saveWithWebDAVInput");
  232. const webDAVURLInput = document.getElementById("webDAVURLInput");
  233. const webDAVUserInput = document.getElementById("webDAVUserInput");
  234. const webDAVPasswordInput = document.getElementById("webDAVPasswordInput");
  235. const saveToGitHubInput = document.getElementById("saveToGitHubInput");
  236. const saveToS3Input = document.getElementById("saveToS3Input");
  237. const githubTokenInput = document.getElementById("githubTokenInput");
  238. const githubUserInput = document.getElementById("githubUserInput");
  239. const githubRepositoryInput = document.getElementById("githubRepositoryInput");
  240. const githubBranchInput = document.getElementById("githubBranchInput");
  241. const saveWithCompanionInput = document.getElementById("saveWithCompanionInput");
  242. const sharePageInput = document.getElementById("sharePageInput");
  243. const saveToFilesystemInput = document.getElementById("saveToFilesystemInput");
  244. const compressHTMLInput = document.getElementById("compressHTMLInput");
  245. const insertTextBodyInput = document.getElementById("insertTextBodyInput");
  246. const insertEmbeddedImageInput = document.getElementById("insertEmbeddedImageInput");
  247. const insertEmbeddedCustomImageInput = document.getElementById("insertEmbeddedCustomImageInput");
  248. const insertEmbeddedScreenshotImageInput = document.getElementById("insertEmbeddedScreenshotImageInput");
  249. const compressCSSInput = document.getElementById("compressCSSInput");
  250. const moveStylesInHeadInput = document.getElementById("moveStylesInHeadInput");
  251. const loadDeferredImagesInput = document.getElementById("loadDeferredImagesInput");
  252. const loadDeferredImagesMaxIdleTimeInput = document.getElementById("loadDeferredImagesMaxIdleTimeInput");
  253. const loadDeferredImagesKeepZoomLevelInput = document.getElementById("loadDeferredImagesKeepZoomLevelInput");
  254. const loadDeferredImagesDispatchScrollEventInput = document.getElementById("loadDeferredImagesDispatchScrollEventInput");
  255. const loadDeferredImagesBeforeFramesInput = document.getElementById("loadDeferredImagesBeforeFramesInput");
  256. const contextMenuEnabledInput = document.getElementById("contextMenuEnabledInput");
  257. const filenameTemplateInput = document.getElementById("filenameTemplateInput");
  258. const filenameMaxLengthInput = document.getElementById("filenameMaxLengthInput");
  259. const filenameMaxLengthUnitInput = document.getElementById("filenameMaxLengthUnitInput");
  260. const filenameReplacementCharacterInput = document.getElementById("filenameReplacementCharacterInput");
  261. const replaceEmojisInFilenameInput = document.getElementById("replaceEmojisInFilenameInput");
  262. const saveFilenameTemplateDataInput = document.getElementById("saveFilenameTemplateDataInput");
  263. const shadowEnabledInput = document.getElementById("shadowEnabledInput");
  264. const maxResourceSizeInput = document.getElementById("maxResourceSizeInput");
  265. const maxResourceSizeEnabledInput = document.getElementById("maxResourceSizeEnabledInput");
  266. const maxResourceDelayInput = document.getElementById("maxResourceDelayInput");
  267. const maxResourceDelayEnabledInput = document.getElementById("maxResourceDelayEnabledInput");
  268. const confirmFilenameInput = document.getElementById("confirmFilenameInput");
  269. const filenameConflictActionInput = document.getElementById("filenameConflictActionInput");
  270. const displayInfobarInput = document.getElementById("displayInfobarInput");
  271. const displayStatsInput = document.getElementById("displayStatsInput");
  272. const backgroundSaveInput = document.getElementById("backgroundSaveInput");
  273. const autoSaveDelayInput = document.getElementById("autoSaveDelayInput");
  274. const autoSaveLoadInput = document.getElementById("autoSaveLoadInput");
  275. const autoSaveUnloadInput = document.getElementById("autoSaveUnloadInput");
  276. const autoSaveDiscardInput = document.getElementById("autoSaveDiscardInput");
  277. const autoSaveRemoveInput = document.getElementById("autoSaveRemoveInput");
  278. const autoSaveLoadOrUnloadInput = document.getElementById("autoSaveLoadOrUnloadInput");
  279. const autoSaveRepeatInput = document.getElementById("autoSaveRepeatInput");
  280. const autoSaveRepeatDelayInput = document.getElementById("autoSaveRepeatDelayInput");
  281. const autoSaveExternalSaveInput = document.getElementById("autoSaveExternalSaveInput");
  282. const removeAlternativeFontsInput = document.getElementById("removeAlternativeFontsInput");
  283. const removeAlternativeImagesInput = document.getElementById("removeAlternativeImagesInput");
  284. const removeAlternativeMediasInput = document.getElementById("removeAlternativeMediasInput");
  285. const saveCreatedBookmarksInput = document.getElementById("saveCreatedBookmarksInput");
  286. const passReferrerOnErrorInput = document.getElementById("passReferrerOnErrorInput");
  287. const replaceBookmarkURLInput = document.getElementById("replaceBookmarkURLInput");
  288. const allowedBookmarkFoldersInput = document.getElementById("allowedBookmarkFoldersInput");
  289. const ignoredBookmarkFoldersInput = document.getElementById("ignoredBookmarkFoldersInput");
  290. const fileFormatSelectInput = document.getElementById("fileFormatSelectInput");
  291. const createRootDirectoryInput = document.getElementById("createRootDirectoryInput");
  292. const preventAppendedDataInput = document.getElementById("preventAppendedDataInput");
  293. const passwordInput = document.getElementById("passwordInput");
  294. const groupDuplicateImagesInput = document.getElementById("groupDuplicateImagesInput");
  295. const infobarTemplateInput = document.getElementById("infobarTemplateInput");
  296. const blockMixedContentInput = document.getElementById("blockMixedContentInput");
  297. const saveOriginalURLsInput = document.getElementById("saveOriginalURLsInput");
  298. const includeInfobarInput = document.getElementById("includeInfobarInput");
  299. const openInfobarInput = document.getElementById("openInfobarInput");
  300. const removeInfobarSavedDateInput = document.getElementById("removeInfobarSavedDateInput");
  301. const confirmInfobarInput = document.getElementById("confirmInfobarInput");
  302. const autoCloseInput = document.getElementById("autoCloseInput");
  303. const openEditorInput = document.getElementById("openEditorInput");
  304. const openSavedPageInput = document.getElementById("openSavedPageInput");
  305. const autoOpenEditorInput = document.getElementById("autoOpenEditorInput");
  306. const defaultEditorModeInput = document.getElementById("defaultEditorModeInput");
  307. const defaultEditorModeNormalLabel = document.getElementById("defaultEditorModeNormalLabel");
  308. const defaultEditorModeEditLabel = document.getElementById("defaultEditorModeEditLabel");
  309. const defaultEditorModeFormatLabel = document.getElementById("defaultEditorModeFormatLabel");
  310. const defaultEditorModeCutLabel = document.getElementById("defaultEditorModeCutLabel");
  311. const defaultEditorModeCutExternalLabel = document.getElementById("defaultEditorModeCutExternalLabel");
  312. const applySystemThemeInput = document.getElementById("applySystemThemeInput");
  313. const warnUnsavedPageInput = document.getElementById("warnUnsavedPageInput");
  314. const displayInfobarInEditorInput = document.getElementById("displayInfobarInEditorInput");
  315. const expandAllButton = document.getElementById("expandAllButton");
  316. const rulesDeleteAllButton = document.getElementById("rulesDeleteAllButton");
  317. const ruleUrlInput = document.getElementById("ruleUrlInput");
  318. const ruleProfileInput = document.getElementById("ruleProfileInput");
  319. const ruleAutoSaveProfileInput = document.getElementById("ruleAutoSaveProfileInput");
  320. const ruleEditProfileInput = document.getElementById("ruleEditProfileInput");
  321. const ruleEditAutoSaveProfileInput = document.getElementById("ruleEditAutoSaveProfileInput");
  322. const ruleAddButton = document.getElementById("ruleAddButton");
  323. const ruleCancelButton = document.getElementById("ruleCancelButton");
  324. const rulesElement = document.querySelector(".rules-table");
  325. const rulesContainerElement = document.querySelector(".rules-table-container");
  326. const ruleEditUrlInput = document.getElementById("ruleEditUrlInput");
  327. const ruleEditButton = document.getElementById("ruleEditButton");
  328. const createURLElement = rulesElement.querySelector(".rule-create");
  329. const showAllProfilesInput = document.getElementById("showAllProfilesInput");
  330. const showAutoSaveProfileInput = document.getElementById("showAutoSaveProfileInput");
  331. const synchronizeInput = document.getElementById("synchronizeInput");
  332. const resetAllButton = document.getElementById("resetAllButton");
  333. const resetCurrentButton = document.getElementById("resetCurrentButton");
  334. const resetCancelButton = document.getElementById("resetCancelButton");
  335. const confirmButton = document.getElementById("confirmButton");
  336. const cancelButton = document.getElementById("cancelButton");
  337. const promptInput = document.getElementById("promptInput");
  338. const promptCancelButton = document.getElementById("promptCancelButton");
  339. const promptConfirmButton = document.getElementById("promptConfirmButton");
  340. const manifest = browser.runtime.getManifest();
  341. const requestPermissionIdentity = manifest.optional_permissions && manifest.optional_permissions.includes("identity");
  342. const saveToRestFormApiLabel = document.getElementById("saveToRestFormApiLabel");
  343. const saveToRestFormApiUrlLabel = document.getElementById("saveToRestFormApiUrlLabel");
  344. const saveToRestFormApiFileFieldNameLabel = document.getElementById("saveToRestFormApiFileFieldNameLabel");
  345. const saveToRestFormApiUrlFieldNameLabel = document.getElementById("saveToRestFormApiUrlFieldNameLabel");
  346. const saveToRestFormApiTokenLabel = document.getElementById("saveToRestFormApiTokenLabel");
  347. const saveToRestFormApiInput = document.getElementById("saveToRestFormApiInput");
  348. const saveToRestFormApiUrlInput = document.getElementById("saveToRestFormApiUrlInput");
  349. const saveToRestFormApiFileFieldNameInput = document.getElementById("saveToRestFormApiFileFieldNameInput");
  350. const saveToRestFormApiUrlFieldNameInput = document.getElementById("saveToRestFormApiUrlFieldNameInput");
  351. const saveToRestFormApiTokenInput = document.getElementById("saveToRestFormApiTokenInput");
  352. const S3DomainInput = document.getElementById("S3DomainInput");
  353. const S3RegionInput = document.getElementById("S3RegionInput");
  354. const S3BucketInput = document.getElementById("S3BucketInput");
  355. const S3AccessKeyInput = document.getElementById("S3AccessKeyInput");
  356. const S3SecretKeyInput = document.getElementById("S3SecretKeyInput");
  357. let sidePanelDisplay, lastURLValue = "", lastProfileValue = "", lastAutoSaveProfileValue = "";
  358. if (location.href.endsWith("#side-panel")) {
  359. sidePanelDisplay = true;
  360. document.querySelector(".options-title").remove();
  361. document.documentElement.classList.add("side-panel");
  362. }
  363. browser.runtime.onMessage.addListener(message => {
  364. if (message.method == "options.refresh" || (message.method == "options.refreshPanel" && sidePanelDisplay)) {
  365. refresh(message.profileName);
  366. }
  367. });
  368. let pendingSave = Promise.resolve();
  369. let autoSaveProfileChanged;
  370. ruleProfileInput.onchange = () => {
  371. if (!autoSaveProfileChanged && ruleProfileInput.value != CURRENT_PROFILE_NAME) {
  372. ruleAutoSaveProfileInput.value = ruleProfileInput.value;
  373. }
  374. };
  375. ruleAutoSaveProfileInput.onchange = () => {
  376. autoSaveProfileChanged = true;
  377. };
  378. rulesDeleteAllButton.addEventListener("click", async event => {
  379. if (await confirm(browser.i18n.getMessage("optionsDeleteDisplayedRulesConfirm"), event.clientY - 100)) {
  380. await browser.runtime.sendMessage({ method: "config.deleteRules", profileName: !showAllProfilesInput.checked && profileNamesInput.value });
  381. await refresh();
  382. await refreshExternalComponents();
  383. }
  384. }, false);
  385. createURLElement.onsubmit = async event => {
  386. event.preventDefault();
  387. try {
  388. await browser.runtime.sendMessage({ method: "config.addRule", url: ruleUrlInput.value, profileName: ruleProfileInput.value, autoSaveProfileName: ruleAutoSaveProfileInput.value });
  389. } catch (error) {
  390. // ignored
  391. }
  392. ruleUrlInput.value = "";
  393. ruleProfileInput.value = ruleAutoSaveProfileInput.value = DEFAULT_PROFILE_NAME;
  394. autoSaveProfileChanged = false;
  395. await refresh();
  396. await refreshExternalComponents();
  397. ruleUrlInput.focus();
  398. };
  399. ruleUrlInput.onclick = ruleUrlInput.onkeyup = ruleUrlInput.onchange = async () => {
  400. ruleAddButton.disabled = !ruleUrlInput.value;
  401. const rules = await browser.runtime.sendMessage({ method: "config.getRules" });
  402. if (rules.find(rule => rule.url == ruleUrlInput.value)) {
  403. ruleAddButton.disabled = true;
  404. }
  405. };
  406. ruleEditUrlInput.onclick = ruleEditUrlInput.onkeyup = ruleEditUrlInput.onchange = async () => {
  407. ruleEditButton.disabled = !ruleEditUrlInput.value;
  408. const rules = await browser.runtime.sendMessage({ method: "config.getRules" });
  409. if (rules.find(rule => rule.url == ruleEditUrlInput.value)) {
  410. ruleEditButton.disabled = true;
  411. }
  412. };
  413. ruleCancelButton.onclick = async () => {
  414. ruleEditUrlInput.value = lastURLValue;
  415. ruleEditProfileInput.value = lastProfileValue;
  416. ruleEditAutoSaveProfileInput.value = lastAutoSaveProfileValue;
  417. lastURLValue = "";
  418. lastProfileValue = "";
  419. lastAutoSaveProfileValue = "";
  420. };
  421. if (getLocalStorageItem("optionShowAutoSaveProfile")) {
  422. showAutoSaveProfileInput.checked = true;
  423. rulesContainerElement.classList.remove("compact");
  424. }
  425. showAutoSaveProfileInput.addEventListener("click", () => {
  426. if (showAutoSaveProfileInput.checked) {
  427. setLocalStorageItem("optionShowAutoSaveProfile", 1);
  428. rulesContainerElement.classList.remove("compact");
  429. } else {
  430. removeLocalStorageItem("optionShowAutoSaveProfile");
  431. rulesContainerElement.classList.add("compact");
  432. }
  433. }, false);
  434. if (getLocalStorageItem("optionShowAllProfiles")) {
  435. showAllProfilesInput.checked = true;
  436. }
  437. showAllProfilesInput.addEventListener("click", () => {
  438. if (showAllProfilesInput.checked) {
  439. setLocalStorageItem("optionShowAllProfiles", 1);
  440. } else {
  441. removeLocalStorageItem("optionShowAllProfiles");
  442. }
  443. }, false);
  444. addProfileButton.addEventListener("click", async event => {
  445. const profileName = await prompt(browser.i18n.getMessage("profileAddPrompt"), event.clientY + 50);
  446. if (profileName) {
  447. try {
  448. await browser.runtime.sendMessage({ method: "config.createProfile", profileName, fromProfileName: profileNamesInput.value });
  449. } catch (error) {
  450. // ignored
  451. }
  452. if (sidePanelDisplay) {
  453. await refresh();
  454. } else {
  455. await refresh(profileName);
  456. }
  457. await refreshExternalComponents();
  458. }
  459. }, false);
  460. deleteProfileButton.addEventListener("click", async event => {
  461. if (await confirm(browser.i18n.getMessage("profileDeleteConfirm"), event.clientY + 50)) {
  462. try {
  463. await browser.runtime.sendMessage({ method: "config.deleteProfile", profileName: profileNamesInput.value });
  464. } catch (error) {
  465. // ignored
  466. }
  467. profileNamesInput.value = null;
  468. await refresh();
  469. await refreshExternalComponents();
  470. }
  471. }, false);
  472. renameProfileButton.addEventListener("click", async event => {
  473. const profileName = await prompt(browser.i18n.getMessage("profileRenamePrompt"), event.clientY + 50, profileNamesInput.value);
  474. if (profileName) {
  475. try {
  476. await browser.runtime.sendMessage({ method: "config.renameProfile", profileName: profileNamesInput.value, newProfileName: profileName });
  477. } catch (error) {
  478. // ignored
  479. }
  480. await refresh(profileName);
  481. await refreshExternalComponents();
  482. }
  483. }, false);
  484. resetButton.addEventListener("click", async event => {
  485. const choice = await reset(event.clientY - 250);
  486. if (choice) {
  487. if (choice == "all") {
  488. await browser.runtime.sendMessage({ method: "config.resetProfiles" });
  489. await refresh(DEFAULT_PROFILE_NAME);
  490. await refreshExternalComponents();
  491. }
  492. if (choice == "current") {
  493. await browser.runtime.sendMessage({ method: "config.resetProfile", profileName: profileNamesInput.value });
  494. await refresh();
  495. await refreshExternalComponents();
  496. }
  497. await update();
  498. }
  499. }, false);
  500. exportButton.addEventListener("click", async () => {
  501. const response = await browser.runtime.sendMessage({ method: "config.exportConfig" });
  502. if (response.filename && response.textContent) {
  503. const link = document.createElement("a");
  504. link.download = response.filename;
  505. link.href = "data:application/octet-stream," + response.textContent;
  506. link.target = "_blank";
  507. link.dispatchEvent(new MouseEvent("click"));
  508. }
  509. }, false);
  510. importButton.addEventListener("click", () => {
  511. fileInput.onchange = async () => {
  512. if (fileInput.files.length) {
  513. const reader = new FileReader();
  514. reader.readAsText(fileInput.files[0]);
  515. const serializedConfig = await new Promise((resolve, reject) => {
  516. reader.addEventListener("load", () => resolve(reader.result), false);
  517. reader.addEventListener("error", reject, false);
  518. });
  519. const config = JSON.parse(serializedConfig);
  520. Object.keys(config.profiles).forEach(profileName => {
  521. const profile = config.profiles[profileName];
  522. if (profile.saveToGDrive && !profile.forceWebAuthFlow) {
  523. profile.saveToGDrive = false;
  524. }
  525. profile.saveToClipboard = false;
  526. profile.saveWithCompanion = false;
  527. profile.saveCreatedBookmarks = false;
  528. profile.passReferrerOnError = false;
  529. });
  530. await browser.runtime.sendMessage({ method: "config.importConfig", config });
  531. await refresh(DEFAULT_PROFILE_NAME);
  532. await refreshExternalComponents();
  533. fileInput.value = "";
  534. }
  535. };
  536. fileInput.click();
  537. }, false);
  538. autoSaveUnloadInput.addEventListener("click", async () => {
  539. if (!autoSaveLoadInput.checked && !autoSaveUnloadInput.checked) {
  540. autoSaveLoadOrUnloadInput.checked = true;
  541. }
  542. }, false);
  543. autoSaveLoadInput.addEventListener("click", async () => {
  544. if (!autoSaveLoadInput.checked && !autoSaveUnloadInput.checked) {
  545. autoSaveLoadOrUnloadInput.checked = true;
  546. }
  547. }, false);
  548. autoSaveLoadOrUnloadInput.addEventListener("click", async () => {
  549. if (autoSaveLoadOrUnloadInput.checked) {
  550. autoSaveUnloadInput.checked = autoSaveLoadInput.checked = false;
  551. } else {
  552. autoSaveUnloadInput.checked = false;
  553. }
  554. }, false);
  555. expandAllButton.addEventListener("click", () => {
  556. if (expandAllButton.className) {
  557. expandAllButton.className = "";
  558. } else {
  559. expandAllButton.className = "opened";
  560. }
  561. document.querySelectorAll("details").forEach(detailElement => detailElement.open = Boolean(expandAllButton.className));
  562. }, false);
  563. saveToFilesystemInput.addEventListener("click", () => disableDestinationPermissions(["clipboardWrite", "nativeMessaging"]), false);
  564. saveToClipboardInput.addEventListener("click", () => disableDestinationPermissions(["nativeMessaging"]), false);
  565. saveWithCompanionInput.addEventListener("click", () => disableDestinationPermissions(["clipboardWrite"]), false);
  566. saveToGDriveInput.addEventListener("click", () => disableDestinationPermissions(["clipboardWrite", "nativeMessaging"], false), false);
  567. saveToDropboxInput.addEventListener("click", () => disableDestinationPermissions(["clipboardWrite", "nativeMessaging"], true, false), false);
  568. saveWithWebDAVInput.addEventListener("click", () => disableDestinationPermissions(["clipboardWrite", "nativeMessaging"]), false);
  569. saveToRestFormApiInput.addEventListener("click", () => disableDestinationPermissions(["clipboardWrite", "nativeMessaging"]), false);
  570. sharePageInput.addEventListener("click", () => disableDestinationPermissions(["clipboardWrite", "nativeMessaging"]), false);
  571. saveCreatedBookmarksInput.addEventListener("click", saveCreatedBookmarks, false);
  572. passReferrerOnErrorInput.addEventListener("click", passReferrerOnError, false);
  573. autoSaveExternalSaveInput.addEventListener("click", () => enableExternalSave(autoSaveExternalSaveInput), false);
  574. saveWithCompanionInput.addEventListener("click", () => enableExternalSave(saveWithCompanionInput), false);
  575. saveToClipboardInput.addEventListener("click", onClickSaveToClipboard, false);
  576. saveToGDriveInput.addEventListener("click", onClickSaveToGDrive, false);
  577. addProofInput.addEventListener("click", async event => {
  578. if (addProofInput.checked) {
  579. addProofInput.checked = false;
  580. if (await confirm(browser.i18n.getMessage("optionsAddProofConfirm"), event.clientY - 100)) {
  581. addProofInput.checked = true;
  582. woleetKeyInput.disabled = false;
  583. }
  584. await update();
  585. }
  586. });
  587. browser.runtime.sendMessage({ method: "config.isSync" }).then(data => synchronizeInput.checked = data.sync);
  588. synchronizeInput.addEventListener("click", async () => {
  589. if (synchronizeInput.checked) {
  590. await browser.runtime.sendMessage({ method: "config.enableSync" });
  591. await refresh(DEFAULT_PROFILE_NAME);
  592. } else {
  593. await browser.runtime.sendMessage({ method: "config.disableSync" });
  594. await refresh();
  595. }
  596. }, false);
  597. insertEmbeddedImageInput.addEventListener("click", () => {
  598. if (insertEmbeddedImageInput.checked) {
  599. insertEmbeddedScreenshotImageInput.checked = true;
  600. } else {
  601. insertEmbeddedScreenshotImageInput.checked = false;
  602. insertEmbeddedCustomImageInput.checked = false;
  603. }
  604. }, false);
  605. fileFormatSelectInput.addEventListener("change", () => {
  606. if (fileFormatSelectInput.value == "html") {
  607. insertEmbeddedScreenshotImageInput.checked = false;
  608. insertEmbeddedCustomImageInput.checked = false;
  609. }
  610. }, false);
  611. document.body.onchange = async event => {
  612. let target = event.target;
  613. if (target != ruleUrlInput &&
  614. target != ruleProfileInput &&
  615. target != ruleAutoSaveProfileInput &&
  616. target != ruleEditUrlInput &&
  617. target != ruleEditProfileInput &&
  618. target != ruleEditAutoSaveProfileInput &&
  619. target != showAutoSaveProfileInput &&
  620. target != saveCreatedBookmarksInput &&
  621. target != passReferrerOnErrorInput) {
  622. if (target != profileNamesInput && target != showAllProfilesInput) {
  623. await update();
  624. }
  625. if (target == profileNamesInput) {
  626. await refresh(profileNamesInput.value);
  627. if (sidePanelDisplay) {
  628. const tabsData = await browser.runtime.sendMessage({ method: "tabsData.get" });
  629. tabsData.profileName = profileNamesInput.value;
  630. await browser.runtime.sendMessage({ method: "tabsData.set", tabsData: tabsData });
  631. await browser.runtime.sendMessage({ method: "ui.refreshMenu" });
  632. }
  633. } else {
  634. if (target == contextMenuEnabledInput) {
  635. await browser.runtime.sendMessage({ method: "ui.refreshMenu" });
  636. }
  637. if (target == openEditorInput) {
  638. await browser.runtime.sendMessage({ method: "ui.refreshMenu" });
  639. }
  640. await refresh();
  641. }
  642. }
  643. };
  644. addProfileButton.title = browser.i18n.getMessage("profileAddButtonTooltip");
  645. deleteProfileButton.title = browser.i18n.getMessage("profileDeleteButtonTooltip");
  646. renameProfileButton.title = browser.i18n.getMessage("profileRenameButtonTooltip");
  647. removeHiddenElementsLabel.textContent = browser.i18n.getMessage("optionRemoveHiddenElements");
  648. removeUnusedStylesLabel.textContent = browser.i18n.getMessage("optionRemoveUnusedStyles");
  649. removeUnusedFontsLabel.textContent = browser.i18n.getMessage("optionRemoveUnusedFonts");
  650. removeFramesLabel.textContent = browser.i18n.getMessage("optionRemoveFrames");
  651. blockScriptsLabel.textContent = browser.i18n.getMessage("optionResourceScript");
  652. blockAudiosLabel.textContent = browser.i18n.getMessage("optionResourceAudio");
  653. blockVideosLabel.textContent = browser.i18n.getMessage("optionResourceVideo");
  654. blockFontsLabel.textContent = browser.i18n.getMessage("optionResourceFont");
  655. blockStylesheetsLabel.textContent = browser.i18n.getMessage("optionResourceStylesheet");
  656. blockImagesLabel.textContent = browser.i18n.getMessage("optionResourceImage");
  657. acceptHeaderDocumentLabel.textContent = browser.i18n.getMessage("optionResourceDocument");
  658. acceptHeaderScriptLabel.textContent = browser.i18n.getMessage("optionResourceScript");
  659. acceptHeaderAudioLabel.textContent = browser.i18n.getMessage("optionResourceAudio");
  660. acceptHeaderVideoLabel.textContent = browser.i18n.getMessage("optionResourceVideo");
  661. acceptHeaderFontLabel.textContent = browser.i18n.getMessage("optionResourceFont");
  662. acceptHeaderStylesheetLabel.textContent = browser.i18n.getMessage("optionResourceStylesheet");
  663. acceptHeaderImageLabel.textContent = browser.i18n.getMessage("optionResourceImage");
  664. saveRawPageLabel.textContent = browser.i18n.getMessage("optionSaveRawPage");
  665. insertMetaCSPLabel.textContent = browser.i18n.getMessage("optionInsertMetaCSP");
  666. saveToClipboardLabel.textContent = browser.i18n.getMessage("optionSaveToClipboard");
  667. saveToFilesystemLabel.textContent = browser.i18n.getMessage("optionSaveToFilesystem");
  668. sharePageLabel.textContent = browser.i18n.getMessage("optionSharePage");
  669. addProofLabel.textContent = browser.i18n.getMessage("optionAddProof");
  670. woleetKeyLabel.textContent = browser.i18n.getMessage("optionWoleetKey");
  671. saveToGDriveLabel.textContent = browser.i18n.getMessage("optionSaveToGDrive");
  672. saveToDropboxLabel.textContent = browser.i18n.getMessage("optionSaveToDropbox");
  673. saveWithWebDAVLabel.textContent = browser.i18n.getMessage("optionSaveWithWebDAV");
  674. webDAVURLLabel.textContent = browser.i18n.getMessage("optionWebDAVURL");
  675. webDAVUserLabel.textContent = browser.i18n.getMessage("optionWebDAVUser");
  676. webDAVPasswordLabel.textContent = browser.i18n.getMessage("optionWebDAVPassword");
  677. saveToGitHubLabel.textContent = browser.i18n.getMessage("optionSaveToGitHub");
  678. githubTokenLabel.textContent = browser.i18n.getMessage("optionGitHubToken");
  679. githubUserLabel.textContent = browser.i18n.getMessage("optionGitHubUser");
  680. githubRepositoryLabel.textContent = browser.i18n.getMessage("optionGitHubRepository");
  681. githubBranchLabel.textContent = browser.i18n.getMessage("optionGitHubBranch");
  682. saveToS3Label.textContent = browser.i18n.getMessage("optionSaveToS3");
  683. S3DomainLabel.textContent = browser.i18n.getMessage("optionS3Domain");
  684. S3RegionLabel.textContent = browser.i18n.getMessage("optionS3Region");
  685. S3BucketLabel.textContent = browser.i18n.getMessage("optionS3Bucket");
  686. S3AccessKeyLabel.textContent = browser.i18n.getMessage("optionS3AccessKey");
  687. S3SecretKeyLabel.textContent = browser.i18n.getMessage("optionS3SecretKey");
  688. saveWithCompanionLabel.textContent = browser.i18n.getMessage("optionSaveWithCompanion");
  689. compressHTMLLabel.textContent = browser.i18n.getMessage("optionCompressHTML");
  690. insertTextBodyLabel.textContent = browser.i18n.getMessage("optionInsertTextBody");
  691. insertEmbeddedImageLabel.textContent = browser.i18n.getMessage("optionInsertEmbeddedImage");
  692. insertEmbeddedCustomImageLabel.textContent = browser.i18n.getMessage("optionInsertEmbeddedCustomImage");
  693. insertEmbeddedScreenshotImageLabel.textContent = browser.i18n.getMessage("optionInsertEmbeddedScreenshotImage");
  694. compressCSSLabel.textContent = browser.i18n.getMessage("optionCompressCSS");
  695. moveStylesInHeadLabel.textContent = browser.i18n.getMessage("optionMoveStylesInHead");
  696. loadDeferredImagesLabel.textContent = browser.i18n.getMessage("optionLoadDeferredImages");
  697. loadDeferredImagesMaxIdleTimeLabel.textContent = browser.i18n.getMessage("optionLoadDeferredImagesMaxIdleTime");
  698. loadDeferredImagesKeepZoomLevelLabel.textContent = browser.i18n.getMessage("optionLoadDeferredImagesKeepZoomLevel");
  699. loadDeferredImagesDispatchScrollEventLabel.textContent = browser.i18n.getMessage("optionLoadDeferredImagesDispatchScrollEvent");
  700. loadDeferredImagesBeforeFramesLabel.textContent = browser.i18n.getMessage("optionLoadDeferredImagesBeforeFrames");
  701. addMenuEntryLabel.textContent = browser.i18n.getMessage("optionAddMenuEntry");
  702. filenameTemplateLabel.textContent = browser.i18n.getMessage("optionFilenameTemplate");
  703. filenameMaxLengthLabel.textContent = browser.i18n.getMessage("optionFilenameMaxLength");
  704. filenameMaxLengthBytesUnitLabel.textContent = browser.i18n.getMessage("optionFilenameMaxLengthBytesUnit");
  705. filenameMaxLengthCharsUnitLabel.textContent = browser.i18n.getMessage("optionFilenameMaxLengthCharsUnit");
  706. filenameReplacementCharacterLabel.textContent = browser.i18n.getMessage("optionFilenameReplacementCharacter");
  707. replaceEmojisInFilenameLabel.textContent = browser.i18n.getMessage("optionReplaceEmojisInFilename");
  708. saveFilenameTemplateDataLabel.textContent = browser.i18n.getMessage("optionSaveFilenameTemplateData");
  709. shadowEnabledLabel.textContent = browser.i18n.getMessage("optionDisplayShadow");
  710. setMaxResourceSizeLabel.textContent = browser.i18n.getMessage("optionSetMaxResourceSize");
  711. maxResourceSizeLabel.textContent = browser.i18n.getMessage("optionMaxResourceSize");
  712. setMaxResourceDelayLabel.textContent = browser.i18n.getMessage("optionSetMaxResourceDelay");
  713. maxResourceDelayLabel.textContent = browser.i18n.getMessage("optionMaxResourceDelay");
  714. confirmFilenameLabel.textContent = browser.i18n.getMessage("optionConfirmFilename");
  715. filenameConflictActionLabel.textContent = browser.i18n.getMessage("optionFilenameConflictAction");
  716. filenameConflictActionUniquifyLabel.textContent = browser.i18n.getMessage("optionFilenameConflictActionUniquify");
  717. filenameConflictActionOverwriteLabel.textContent = browser.i18n.getMessage("optionFilenameConflictActionOverwrite");
  718. filenameConflictActionPromptLabel.textContent = browser.i18n.getMessage("optionFilenameConflictActionPrompt");
  719. filenameConflictActionSkipLabel.textContent = browser.i18n.getMessage("optionFilenameConflictActionSkip");
  720. displayInfobarLabel.textContent = browser.i18n.getMessage("optionDisplayInfobar");
  721. displayStatsLabel.textContent = browser.i18n.getMessage("optionDisplayStats");
  722. backgroundSaveLabel.textContent = browser.i18n.getMessage("optionBackgroundSave");
  723. autoSaveDelayLabel.textContent = browser.i18n.getMessage("optionAutoSaveDelay");
  724. autoSaveLoadLabel.textContent = browser.i18n.getMessage("optionAutoSaveLoad");
  725. autoSaveUnloadLabel.textContent = browser.i18n.getMessage("optionAutoSaveUnload");
  726. autoSaveLoadOrUnloadLabel.textContent = browser.i18n.getMessage("optionAutoSaveLoadOrUnload");
  727. autoSaveDiscardLabel.textContent = browser.i18n.getMessage("optionAutoSaveDiscard");
  728. autoSaveRemoveLabel.textContent = browser.i18n.getMessage("optionAutoSaveRemove");
  729. autoSaveRepeatLabel.textContent = browser.i18n.getMessage("optionAutoSaveRepeat");
  730. autoSaveRepeatDelayLabel.textContent = browser.i18n.getMessage("optionAutoSaveRepeatDelay");
  731. autoSaveExternalSaveLabel.textContent = browser.i18n.getMessage("optionAutoSaveExternalSave");
  732. removeAlternativeFontsLabel.textContent = browser.i18n.getMessage("optionRemoveAlternativeFonts");
  733. removeAlternativeImagesLabel.textContent = browser.i18n.getMessage("optionRemoveAlternativeImages");
  734. removeAlternativeMediasLabel.textContent = browser.i18n.getMessage("optionRemoveAlternativeMedias");
  735. saveCreatedBookmarksLabel.textContent = browser.i18n.getMessage("optionSaveCreatedBookmarks");
  736. passReferrerOnErrorLabel.textContent = browser.i18n.getMessage("optionPassReferrerOnError");
  737. replaceBookmarkURLLabel.textContent = browser.i18n.getMessage("optionReplaceBookmarkURL");
  738. allowedBookmarkFoldersLabel.textContent = browser.i18n.getMessage("optionAllowedBookmarkFolders");
  739. ignoredBookmarkFoldersLabel.textContent = browser.i18n.getMessage("optionIgnoredBookmarkFolders");
  740. createRootDirectoryLabel.textContent = browser.i18n.getMessage("optionCreateRootDirectory");
  741. preventAppendedDataLabel.textContent = browser.i18n.getMessage("optionPreventAppendedData");
  742. passwordLabel.textContent = browser.i18n.getMessage("optionPassword");
  743. groupDuplicateImagesLabel.textContent = browser.i18n.getMessage("optionGroupDuplicateImages");
  744. titleLabel.textContent = browser.i18n.getMessage("optionsTitle");
  745. userInterfaceLabel.textContent = browser.i18n.getMessage("optionsUserInterfaceSubTitle");
  746. filenameLabel.textContent = browser.i18n.getMessage("optionsFileNameSubTitle");
  747. htmlContentLabel.textContent = browser.i18n.getMessage("optionsHTMLContentSubTitle");
  748. fileFormatLabel.textContent = browser.i18n.getMessage("optionsFileFormatSubTitle");
  749. fileFormatSelectHTMLLabel.textContent = browser.i18n.getMessage("optionFileFormatSelectHTML");
  750. fileFormatSelectSelfExtractingUniversalLabel.textContent = browser.i18n.getMessage("optionFileFormatSelectSelfExtractingUniversal");
  751. fileFormatSelectSelfExtractingLabel.textContent = browser.i18n.getMessage("optionFileFormatSelectSelfExtracting");
  752. fileFormatSelectZIPLabel.textContent = browser.i18n.getMessage("optionFileFormatSelectZIP");
  753. fileFormatSelectLabel.textContent = browser.i18n.getMessage("optionFileFormat");
  754. infobarLabel.textContent = browser.i18n.getMessage("optionsInfobarSubTitle");
  755. imagesLabel.textContent = browser.i18n.getMessage("optionsImagesSubTitle");
  756. stylesheetsLabel.textContent = browser.i18n.getMessage("optionsStylesheetsSubTitle");
  757. fontsLabel.textContent = browser.i18n.getMessage("optionsFontsSubTitle");
  758. networkLabel.textContent = browser.i18n.getMessage("optionsNetworkSubTitle");
  759. blockResourcesLabel.textContent = browser.i18n.getMessage("optionsBlockedResources");
  760. acceptHeadersLabel.textContent = browser.i18n.getMessage("optionsAcceptHeaders");
  761. destinationLabel.textContent = browser.i18n.getMessage("optionsDestinationSubTitle");
  762. bookmarksLabel.textContent = browser.i18n.getMessage("optionsBookmarkSubTitle");
  763. autoSaveLabel.textContent = browser.i18n.getMessage("optionsAutoSaveSubTitle");
  764. miscLabel.textContent = browser.i18n.getMessage("optionsMiscSubTitle");
  765. helpLabel.textContent = browser.i18n.getMessage("optionsHelpLink");
  766. infobarTemplateLabel.textContent = browser.i18n.getMessage("optionInfobarTemplate");
  767. blockMixedContentLabel.textContent = browser.i18n.getMessage("optionBlockMixedContent");
  768. saveOriginalURLsLabel.textContent = browser.i18n.getMessage("optionSaveOriginalURLs");
  769. includeInfobarLabel.textContent = browser.i18n.getMessage("optionIncludeInfobar");
  770. openInfobarLabel.textContent = browser.i18n.getMessage("optionOpenInfobar");
  771. removeInfobarSavedDateLabel.textContent = browser.i18n.getMessage("optionRemoveInfobarSavedDate");
  772. confirmInfobarLabel.textContent = browser.i18n.getMessage("optionConfirmInfobar");
  773. autoCloseLabel.textContent = browser.i18n.getMessage("optionAutoClose");
  774. editorLabel.textContent = browser.i18n.getMessage("optionsEditorSubTitle");
  775. openEditorLabel.textContent = browser.i18n.getMessage("optionOpenEditor");
  776. openSavedPageLabel.textContent = browser.i18n.getMessage("optionOpenSavedPage");
  777. autoOpenEditorLabel.textContent = browser.i18n.getMessage("optionAutoOpenEditor");
  778. defaultEditorModeLabel.textContent = browser.i18n.getMessage("optionDefaultEditorMode");
  779. defaultEditorModeNormalLabel.textContent = browser.i18n.getMessage("optionDefaultEditorModeNormal");
  780. defaultEditorModeEditLabel.textContent = browser.i18n.getMessage("optionDefaultEditorModeEdit");
  781. defaultEditorModeFormatLabel.textContent = browser.i18n.getMessage("optionDefaultEditorModeFormat");
  782. defaultEditorModeCutLabel.textContent = browser.i18n.getMessage("optionDefaultEditorModeCut");
  783. defaultEditorModeCutExternalLabel.textContent = browser.i18n.getMessage("optionDefaultEditorModeCutExternal");
  784. applySystemThemeLabel.textContent = browser.i18n.getMessage("optionApplySystemTheme");
  785. warnUnsavedPageLabel.textContent = browser.i18n.getMessage("optionWarnUnsavedPage");
  786. displayInfobarInEditorLabel.textContent = browser.i18n.getMessage("optiondisplayInfobarInEditor");
  787. resetButton.textContent = browser.i18n.getMessage("optionsResetButton");
  788. exportButton.textContent = browser.i18n.getMessage("optionsExportButton");
  789. importButton.textContent = browser.i18n.getMessage("optionsImportButton");
  790. resetButton.title = browser.i18n.getMessage("optionsResetTooltip");
  791. autoSettingsLabel.textContent = browser.i18n.getMessage("optionsAutoSettingsSubTitle");
  792. autoSettingsUrlLabel.textContent = browser.i18n.getMessage("optionsAutoSettingsUrl");
  793. autoSettingsProfileLabel.textContent = browser.i18n.getMessage("optionsAutoSettingsProfile");
  794. autoSettingsAutoSaveProfileLabel.textContent = browser.i18n.getMessage("optionsAutoSettingsAutoSaveProfile");
  795. ruleAddButton.title = browser.i18n.getMessage("optionsAddRuleTooltip");
  796. ruleEditButton.title = browser.i18n.getMessage("optionsValidateChangesTooltip");
  797. ruleCancelButton.title = browser.i18n.getMessage("optionsCancelChangesTooltip");
  798. rulesDeleteAllButton.title = browser.i18n.getMessage("optionsDeleteRulesTooltip");
  799. showAllProfilesLabel.textContent = browser.i18n.getMessage("optionsAutoSettingsShowAllProfiles");
  800. showAutoSaveProfileLabel.textContent = browser.i18n.getMessage("optionsAutoSettingsShowAutoSaveProfile");
  801. ruleUrlInput.placeholder = ruleEditUrlInput.placeholder = browser.i18n.getMessage("optionsAutoSettingsUrlPlaceholder");
  802. synchronizeLabel.textContent = browser.i18n.getMessage("optionSynchronize");
  803. resetAllButton.textContent = browser.i18n.getMessage("optionsResetAllButton");
  804. resetCurrentButton.textContent = browser.i18n.getMessage("optionsResetCurrentButton");
  805. resetCancelButton.textContent = promptCancelButton.textContent = cancelButton.textContent = browser.i18n.getMessage("optionsCancelButton");
  806. confirmButton.textContent = promptConfirmButton.textContent = browser.i18n.getMessage("optionsOKButton");
  807. document.getElementById("resetConfirmLabel").textContent = browser.i18n.getMessage("optionsResetConfirm");
  808. saveToRestFormApiLabel.textContent = browser.i18n.getMessage("optionSaveToRestFormApi");
  809. saveToRestFormApiUrlLabel.textContent = browser.i18n.getMessage("optionRestFormApiUrl");
  810. saveToRestFormApiFileFieldNameLabel.textContent = browser.i18n.getMessage("optionRestFormApiFileFieldName");
  811. saveToRestFormApiUrlFieldNameLabel.textContent = browser.i18n.getMessage("optionRestFormApiUrlFieldName");
  812. saveToRestFormApiTokenLabel.textContent = browser.i18n.getMessage("optionRestFormApiToken");
  813. if (location.href.endsWith("#")) {
  814. document.querySelector(".new-window-link").remove();
  815. document.documentElement.classList.add("maximized");
  816. }
  817. let tabsData;
  818. browser.runtime.sendMessage({ method: "tabsData.get" }).then(allTabsData => {
  819. tabsData = allTabsData;
  820. return refresh(tabsData.profileName);
  821. });
  822. getHelpContents();
  823. function init() {
  824. if (!AUTO_SAVE_SUPPORTED) {
  825. document.getElementById("autoSaveSection").hidden = true;
  826. document.getElementById("showAutoSaveProfileOption").hidden = true;
  827. rulesContainerElement.classList.add("compact");
  828. }
  829. if (!BACKGROUND_SAVE_SUPPORTED) {
  830. document.getElementById("backgroundSaveOptions").hidden = true;
  831. document.getElementById("confirmFilenameOption").hidden = true;
  832. document.getElementById("filenameConflictAction").hidden = true;
  833. }
  834. if (!BOOKMARKS_API_SUPPORTED) {
  835. document.getElementById("bookmarksOptions").hidden = true;
  836. }
  837. if (!AUTO_OPEN_EDITOR_SUPPORTED) {
  838. document.getElementById("autoOpenEditorOption").hidden = true;
  839. }
  840. if (!INFOBAR_SUPPORTED) {
  841. document.getElementById("displayInfobarOption").hidden = true;
  842. }
  843. if (!IDENTITY_API_SUPPORTED) {
  844. document.getElementById("saveToGDriveOption").hidden = true;
  845. document.getElementById("saveToDropboxOption").hidden = true;
  846. }
  847. if (!CLIPBOARD_API_SUPPORTED) {
  848. document.getElementById("saveToClipboardOption").hidden = true;
  849. }
  850. if (!NATIVE_API_API_SUPPORTED) {
  851. document.getElementById("saveWithCompanionOption").hidden = true;
  852. }
  853. if (!WEB_BLOCKING_API_SUPPORTED) {
  854. document.getElementById("passReferrerOnErrorOption").hidden = true;
  855. }
  856. if (!SHARE_API_SUPPORTED) {
  857. document.getElementById("sharePageOption").hidden = true;
  858. }
  859. }
  860. async function refresh(profileName) {
  861. const [profiles, rules, companionState] = await Promise.all([
  862. browser.runtime.sendMessage({ method: "config.getProfiles" }),
  863. browser.runtime.sendMessage({ method: "config.getRules" }),
  864. browser.runtime.sendMessage({ method: "companion.state" })]);
  865. const selectedProfileName = profileName || profileNamesInput.value || DEFAULT_PROFILE_NAME;
  866. Array.from(profileNamesInput.childNodes).forEach(node => node.remove());
  867. profileNamesInput.options.length = 0;
  868. ruleProfileInput.options.length = 0;
  869. ruleAutoSaveProfileInput.options.length = 0;
  870. ruleEditProfileInput.options.length = 0;
  871. ruleEditAutoSaveProfileInput.options.length = 0;
  872. let optionElement = document.createElement("option");
  873. optionElement.value = DEFAULT_PROFILE_NAME;
  874. optionElement.textContent = browser.i18n.getMessage("profileDefaultSettings");
  875. [CURRENT_PROFILE_NAME].concat(...Object.keys(profiles)).forEach(profileName => {
  876. const optionElement = document.createElement("option");
  877. optionElement.value = optionElement.textContent = profileName;
  878. if (profileName == DEFAULT_PROFILE_NAME) {
  879. optionElement.textContent = browser.i18n.getMessage("profileDefaultSettings");
  880. }
  881. if (profileName != CURRENT_PROFILE_NAME) {
  882. profileNamesInput.appendChild(optionElement);
  883. }
  884. ruleProfileInput.appendChild(optionElement.cloneNode(true));
  885. ruleAutoSaveProfileInput.appendChild(optionElement.cloneNode(true));
  886. ruleEditProfileInput.appendChild(optionElement.cloneNode(true));
  887. ruleEditAutoSaveProfileInput.appendChild(optionElement.cloneNode(true));
  888. });
  889. profileNamesInput.disabled = profileNamesInput.options.length == 1;
  890. optionElement = document.createElement("option");
  891. optionElement.value = DISABLED_PROFILE_NAME;
  892. optionElement.textContent = browser.i18n.getMessage("profileDisabled");
  893. ruleEditProfileInput.appendChild(optionElement.cloneNode(true));
  894. ruleProfileInput.appendChild(optionElement.cloneNode(true));
  895. ruleAutoSaveProfileInput.appendChild(optionElement);
  896. ruleEditAutoSaveProfileInput.appendChild(optionElement.cloneNode(true));
  897. const rulesDataElement = rulesElement.querySelector(".rules-data");
  898. Array.from(rulesDataElement.childNodes).forEach(node => (!node.className || !node.className.includes("rule-edit")) && node.remove());
  899. const editURLElement = rulesElement.querySelector(".rule-edit");
  900. createURLElement.hidden = false;
  901. editURLElement.hidden = true;
  902. ruleProfileInput.value = ruleAutoSaveProfileInput.value = selectedProfileName;
  903. let rulesDisplayed;
  904. rules.forEach(rule => {
  905. if (showAllProfilesInput.checked || selectedProfileName == rule.profile || selectedProfileName == rule.autoSaveProfile) {
  906. rulesDisplayed = true;
  907. const ruleElement = rulesElement.querySelector(".rule-view").cloneNode(true);
  908. const ruleUrlElement = ruleElement.querySelector(".rule-url");
  909. const ruleProfileElement = ruleElement.querySelector(".rule-profile");
  910. const ruleAutoSaveProfileElement = ruleElement.querySelector(".rule-autosave-profile");
  911. ruleUrlElement.textContent = ruleUrlElement.title = rule.url;
  912. ruleProfileElement.textContent = ruleProfileElement.title = getProfileText(rule.profile);
  913. ruleAutoSaveProfileElement.textContent = ruleAutoSaveProfileElement.title = getProfileText(rule.autoSaveProfile);
  914. ruleElement.hidden = false;
  915. ruleElement.className = "tr data";
  916. rulesDataElement.appendChild(ruleElement);
  917. const ruleDeleteButton = ruleElement.querySelector(".rule-delete-button");
  918. const ruleUpdateButton = ruleElement.querySelector(".rule-update-button");
  919. ruleDeleteButton.title = browser.i18n.getMessage("optionsDeleteRuleTooltip");
  920. ruleDeleteButton.addEventListener("click", async event => {
  921. if (await confirm(browser.i18n.getMessage("optionsDeleteRuleConfirm"), event.clientY - 100)) {
  922. await browser.runtime.sendMessage({ method: "config.deleteRule", url: rule.url });
  923. await refresh();
  924. await refreshExternalComponents();
  925. }
  926. }, false);
  927. ruleUpdateButton.title = browser.i18n.getMessage("optionsUpdateRuleTooltip");
  928. ruleUpdateButton.addEventListener("click", async () => {
  929. if (editURLElement.hidden) {
  930. createURLElement.hidden = true;
  931. editURLElement.hidden = false;
  932. rulesDataElement.replaceChild(editURLElement, ruleElement);
  933. ruleEditUrlInput.value = lastURLValue = rule.url;
  934. ruleEditProfileInput.value = lastProfileValue = rule.profile;
  935. ruleEditAutoSaveProfileInput.value = lastAutoSaveProfileValue = rule.autoSaveProfile;
  936. ruleEditUrlInput.focus();
  937. editURLElement.onsubmit = async event => {
  938. event.preventDefault();
  939. rulesElement.appendChild(editURLElement);
  940. await browser.runtime.sendMessage({ method: "config.updateRule", url: rule.url, newUrl: ruleEditUrlInput.value, profileName: ruleEditProfileInput.value, autoSaveProfileName: ruleEditAutoSaveProfileInput.value });
  941. await refresh();
  942. await refreshExternalComponents();
  943. ruleUrlInput.focus();
  944. };
  945. editURLElement.onkeyup = event => {
  946. if (event.key == "Escape") {
  947. ruleCancelButton.click();
  948. }
  949. };
  950. }
  951. }, false);
  952. }
  953. });
  954. rulesDeleteAllButton.disabled = !rulesDisplayed;
  955. rulesElement.appendChild(createURLElement);
  956. profileNamesInput.value = selectedProfileName;
  957. renameProfileButton.disabled = deleteProfileButton.disabled = profileNamesInput.value == DEFAULT_PROFILE_NAME;
  958. const profileOptions = profiles[selectedProfileName];
  959. removeHiddenElementsInput.checked = profileOptions.removeHiddenElements;
  960. removeUnusedStylesInput.checked = profileOptions.removeUnusedStyles;
  961. removeUnusedFontsInput.checked = profileOptions.removeUnusedFonts;
  962. removeFramesInput.checked = profileOptions.removeFrames;
  963. blockScriptsInput.checked = profileOptions.blockScripts;
  964. blockVideosInput.checked = profileOptions.blockVideos;
  965. blockAudiosInput.checked = profileOptions.blockAudios;
  966. blockFontsInput.checked = profileOptions.blockFonts;
  967. blockStylesheetsInput.checked = profileOptions.blockStylesheets;
  968. blockImagesInput.checked = profileOptions.blockImages;
  969. acceptHeaderDocumentInput.value = profileOptions.acceptHeaders.document;
  970. acceptHeaderScriptInput.value = profileOptions.acceptHeaders.script;
  971. acceptHeaderAudioInput.value = profileOptions.acceptHeaders.audio;
  972. acceptHeaderVideoInput.value = profileOptions.acceptHeaders.video;
  973. acceptHeaderFontInput.value = profileOptions.acceptHeaders.font;
  974. acceptHeaderStylesheetInput.value = profileOptions.acceptHeaders.stylesheet;
  975. acceptHeaderImageInput.value = profileOptions.acceptHeaders.image;
  976. saveRawPageInput.checked = profileOptions.saveRawPage;
  977. insertMetaCSPInput.checked = profileOptions.insertMetaCSP;
  978. saveToClipboardInput.checked = profileOptions.saveToClipboard;
  979. addProofInput.checked = profileOptions.addProof;
  980. woleetKeyInput.value = profileOptions.woleetKey;
  981. woleetKeyInput.disabled = !profileOptions.addProof;
  982. saveToGDriveInput.checked = profileOptions.saveToGDrive;
  983. saveToDropboxInput.checked = profileOptions.saveToDropbox;
  984. saveWithWebDAVInput.checked = profileOptions.saveWithWebDAV;
  985. webDAVURLInput.value = profileOptions.webDAVURL;
  986. webDAVURLInput.disabled = !profileOptions.saveWithWebDAV;
  987. webDAVUserInput.value = profileOptions.webDAVUser;
  988. webDAVUserInput.disabled = !profileOptions.saveWithWebDAV;
  989. webDAVPasswordInput.value = profileOptions.webDAVPassword;
  990. webDAVPasswordInput.disabled = !profileOptions.saveWithWebDAV;
  991. saveToGitHubInput.checked = profileOptions.saveToGitHub;
  992. githubTokenInput.value = profileOptions.githubToken;
  993. githubTokenInput.disabled = !profileOptions.saveToGitHub;
  994. githubUserInput.value = profileOptions.githubUser;
  995. githubUserInput.disabled = !profileOptions.saveToGitHub;
  996. githubRepositoryInput.value = profileOptions.githubRepository;
  997. githubRepositoryInput.disabled = !profileOptions.saveToGitHub;
  998. githubBranchInput.value = profileOptions.githubBranch;
  999. githubBranchInput.disabled = !profileOptions.saveToGitHub;
  1000. saveWithCompanionInput.checked = profileOptions.saveWithCompanion;
  1001. saveToRestFormApiInput.checked = profileOptions.saveToRestFormApi;
  1002. saveToRestFormApiUrlInput.value = profileOptions.saveToRestFormApiUrl;
  1003. saveToRestFormApiUrlInput.disabled = !profileOptions.saveToRestFormApi;
  1004. saveToRestFormApiTokenInput.value = profileOptions.saveToRestFormApiToken;
  1005. saveToRestFormApiTokenInput.disabled = !profileOptions.saveToRestFormApi;
  1006. saveToRestFormApiFileFieldNameInput.value = profileOptions.saveToRestFormApiFileFieldName;
  1007. saveToRestFormApiFileFieldNameInput.disabled = !profileOptions.saveToRestFormApi;
  1008. saveToRestFormApiUrlFieldNameInput.value = profileOptions.saveToRestFormApiUrlFieldName;
  1009. saveToRestFormApiUrlFieldNameInput.disabled = !profileOptions.saveToRestFormApi;
  1010. saveToS3Input.checked = profileOptions.saveToS3;
  1011. S3DomainInput.value = profileOptions.S3Domain;
  1012. S3DomainInput.disabled = !profileOptions.saveToS3;
  1013. S3RegionInput.value = profileOptions.S3Region;
  1014. S3RegionInput.disabled = !profileOptions.saveToS3;
  1015. S3BucketInput.value = profileOptions.S3Bucket;
  1016. S3BucketInput.disabled = !profileOptions.saveToS3;
  1017. S3AccessKeyInput.value = profileOptions.S3AccessKey;
  1018. S3AccessKeyInput.disabled = !profileOptions.saveToS3;
  1019. S3SecretKeyInput.value = profileOptions.S3SecretKey;
  1020. S3SecretKeyInput.disabled = !profileOptions.saveToS3;
  1021. sharePageInput.checked = profileOptions.sharePage;
  1022. saveToFilesystemInput.checked = !profileOptions.saveToGDrive && !profileOptions.saveToGitHub && !profileOptions.saveToS3 && !profileOptions.saveWithCompanion && !profileOptions.saveToClipboard && !profileOptions.saveWithWebDAV && !profileOptions.saveToDropbox && !profileOptions.saveToRestFormApi && !profileOptions.sharePage;
  1023. compressHTMLInput.checked = profileOptions.compressHTML;
  1024. compressCSSInput.checked = profileOptions.compressCSS;
  1025. moveStylesInHeadInput.checked = profileOptions.moveStylesInHead;
  1026. loadDeferredImagesInput.checked = profileOptions.loadDeferredImages;
  1027. loadDeferredImagesMaxIdleTimeInput.value = profileOptions.loadDeferredImagesMaxIdleTime;
  1028. loadDeferredImagesKeepZoomLevelInput.checked = profileOptions.loadDeferredImagesKeepZoomLevel;
  1029. loadDeferredImagesKeepZoomLevelInput.disabled = !profileOptions.loadDeferredImages;
  1030. loadDeferredImagesMaxIdleTimeInput.disabled = !profileOptions.loadDeferredImages;
  1031. loadDeferredImagesDispatchScrollEventInput.checked = profileOptions.loadDeferredImagesDispatchScrollEvent;
  1032. loadDeferredImagesDispatchScrollEventInput.disabled = !profileOptions.loadDeferredImages;
  1033. loadDeferredImagesBeforeFramesInput.checked = profileOptions.loadDeferredImagesBeforeFrames;
  1034. loadDeferredImagesBeforeFramesInput.disabled = !profileOptions.loadDeferredImages;
  1035. contextMenuEnabledInput.checked = profileOptions.contextMenuEnabled;
  1036. filenameTemplateInput.value = profileOptions.filenameTemplate;
  1037. filenameMaxLengthInput.value = profileOptions.filenameMaxLength;
  1038. filenameMaxLengthUnitInput.value = profileOptions.filenameMaxLengthUnit;
  1039. filenameReplacementCharacterInput.value = profileOptions.filenameReplacementCharacter;
  1040. replaceEmojisInFilenameInput.checked = profileOptions.replaceEmojisInFilename;
  1041. saveFilenameTemplateDataInput.checked = profileOptions.saveFilenameTemplateData;
  1042. shadowEnabledInput.checked = profileOptions.shadowEnabled;
  1043. maxResourceSizeEnabledInput.checked = profileOptions.maxResourceSizeEnabled;
  1044. maxResourceSizeInput.value = profileOptions.maxResourceSizeEnabled ? profileOptions.maxResourceSize : 10;
  1045. maxResourceSizeInput.disabled = !profileOptions.maxResourceSizeEnabled;
  1046. maxResourceDelayEnabledInput.checked = Boolean(profileOptions.networkTimeout);
  1047. maxResourceDelayInput.value = profileOptions.networkTimeout ? profileOptions.networkTimeout / 1000 : 60;
  1048. maxResourceDelayInput.disabled = !profileOptions.networkTimeout;
  1049. confirmFilenameInput.checked = profileOptions.confirmFilename;
  1050. filenameConflictActionInput.value = profileOptions.filenameConflictAction;
  1051. displayInfobarInput.checked = profileOptions.displayInfobar;
  1052. displayStatsInput.checked = profileOptions.displayStats;
  1053. backgroundSaveInput.checked = profileOptions.backgroundSave;
  1054. autoSaveDelayInput.value = profileOptions.autoSaveDelay;
  1055. autoSaveLoadInput.checked = !profileOptions.autoSaveLoadOrUnload && profileOptions.autoSaveLoad;
  1056. autoSaveLoadOrUnloadInput.checked = profileOptions.autoSaveLoadOrUnload;
  1057. autoSaveUnloadInput.checked = !profileOptions.autoSaveLoadOrUnload && profileOptions.autoSaveUnload;
  1058. autoSaveLoadInput.disabled = profileOptions.autoSaveLoadOrUnload;
  1059. autoSaveUnloadInput.disabled = profileOptions.autoSaveLoadOrUnload;
  1060. autoSaveDiscardInput.checked = profileOptions.autoSaveDiscard;
  1061. autoSaveRemoveInput.checked = profileOptions.autoSaveRemove;
  1062. autoSaveRepeatInput.checked = profileOptions.autoSaveRepeat;
  1063. autoSaveRepeatDelayInput.value = profileOptions.autoSaveRepeatDelay;
  1064. autoSaveRepeatDelayInput.disabled = !profileOptions.autoSaveRepeat;
  1065. autoSaveExternalSaveInput.checked = profileOptions.autoSaveExternalSave;
  1066. autoSaveExternalSaveInput.parentElement.hidden = !companionState.enabled;
  1067. removeAlternativeFontsInput.checked = profileOptions.removeAlternativeFonts;
  1068. removeAlternativeImagesInput.checked = profileOptions.removeAlternativeImages;
  1069. groupDuplicateImagesInput.checked = profileOptions.groupDuplicateImages;
  1070. removeAlternativeMediasInput.checked = profileOptions.removeAlternativeMedias;
  1071. saveCreatedBookmarksInput.checked = profileOptions.saveCreatedBookmarks;
  1072. passReferrerOnErrorInput.checked = profileOptions.passReferrerOnError;
  1073. replaceBookmarkURLInput.checked = profileOptions.replaceBookmarkURL;
  1074. replaceBookmarkURLInput.disabled = !profileOptions.saveCreatedBookmarks;
  1075. allowedBookmarkFoldersInput.value = profileOptions.allowedBookmarkFolders.map(folder => folder.replace(/,/g, "\\,")).join(",");
  1076. allowedBookmarkFoldersInput.disabled = !profileOptions.saveCreatedBookmarks;
  1077. ignoredBookmarkFoldersInput.value = profileOptions.ignoredBookmarkFolders.map(folder => folder.replace(/,/g, "\\,")).join(",");
  1078. ignoredBookmarkFoldersInput.disabled = !profileOptions.saveCreatedBookmarks;
  1079. fileFormatSelectInput.value = profileOptions.compressContent ? profileOptions.selfExtractingArchive ? profileOptions.extractDataFromPage ?
  1080. "self-extracting-zip-universal" : "self-extracting-zip" : "zip" : "html";
  1081. createRootDirectoryInput.checked = profileOptions.createRootDirectory;
  1082. createRootDirectoryInput.disabled = !profileOptions.compressContent;
  1083. preventAppendedDataInput.checked = profileOptions.preventAppendedData;
  1084. preventAppendedDataInput.disabled = !profileOptions.compressContent && !profileOptions.selfExtractingArchive;
  1085. passwordInput.value = profileOptions.password;
  1086. passwordInput.disabled = !profileOptions.compressContent;
  1087. insertTextBodyInput.checked = profileOptions.insertTextBody;
  1088. insertTextBodyInput.disabled = !profileOptions.compressContent || (!profileOptions.selfExtractingArchive && !profileOptions.extractDataFromPage);
  1089. insertEmbeddedCustomImageInput.checked = profileOptions.insertEmbeddedImage;
  1090. insertEmbeddedCustomImageInput.disabled = !profileOptions.compressContent || (!profileOptions.insertEmbeddedImage && !profileOptions.insertEmbeddedScreenshotImage);
  1091. insertEmbeddedScreenshotImageInput.checked = profileOptions.insertEmbeddedScreenshotImage;
  1092. insertEmbeddedScreenshotImageInput.disabled = !profileOptions.compressContent || (!profileOptions.insertEmbeddedImage && !profileOptions.insertEmbeddedScreenshotImage);
  1093. insertEmbeddedImageInput.checked = profileOptions.compressContent && (profileOptions.insertEmbeddedImage || profileOptions.insertEmbeddedScreenshotImage);
  1094. insertEmbeddedImageInput.disabled = !profileOptions.compressContent;
  1095. infobarTemplateInput.value = profileOptions.infobarTemplate;
  1096. blockMixedContentInput.checked = profileOptions.blockMixedContent;
  1097. saveOriginalURLsInput.checked = profileOptions.saveOriginalURLs;
  1098. includeInfobarInput.checked = profileOptions.includeInfobar;
  1099. openInfobarInput.checked = profileOptions.openInfobar;
  1100. removeInfobarSavedDateInput.checked = profileOptions.removeSavedDate;
  1101. confirmInfobarInput.checked = profileOptions.confirmInfobarContent;
  1102. autoCloseInput.checked = profileOptions.autoClose;
  1103. openEditorInput.checked = profileOptions.openEditor;
  1104. openSavedPageInput.checked = profileOptions.openSavedPage;
  1105. autoOpenEditorInput.checked = profileOptions.autoOpenEditor;
  1106. defaultEditorModeInput.value = profileOptions.defaultEditorMode;
  1107. applySystemThemeInput.checked = profileOptions.applySystemTheme;
  1108. warnUnsavedPageInput.checked = profileOptions.warnUnsavedPage;
  1109. displayInfobarInEditorInput.checked = profileOptions.displayInfobarInEditor;
  1110. }
  1111. function getProfileText(profileName) {
  1112. return profileName == DEFAULT_PROFILE_NAME ? browser.i18n.getMessage("profileDefaultSettings") : profileName == DISABLED_PROFILE_NAME ? browser.i18n.getMessage("profileDisabled") : profileName;
  1113. }
  1114. async function update() {
  1115. try {
  1116. await pendingSave;
  1117. } catch (error) {
  1118. // ignored
  1119. }
  1120. pendingSave = browser.runtime.sendMessage({
  1121. method: "config.updateProfile",
  1122. profileName: profileNamesInput.value,
  1123. profile: {
  1124. removeHiddenElements: removeHiddenElementsInput.checked,
  1125. removeUnusedStyles: removeUnusedStylesInput.checked,
  1126. removeUnusedFonts: removeUnusedFontsInput.checked,
  1127. removeFrames: removeFramesInput.checked,
  1128. blockScripts: blockScriptsInput.checked,
  1129. blockVideos: blockVideosInput.checked,
  1130. blockAudios: blockAudiosInput.checked,
  1131. blockFonts: blockFontsInput.checked,
  1132. blockStylesheets: blockStylesheetsInput.checked,
  1133. blockImages: blockImagesInput.checked,
  1134. acceptHeaders: {
  1135. document: acceptHeaderDocumentInput.value,
  1136. script: acceptHeaderScriptInput.value,
  1137. audio: acceptHeaderAudioInput.value,
  1138. video: acceptHeaderVideoInput.value,
  1139. font: acceptHeaderFontInput.value,
  1140. stylesheet: acceptHeaderStylesheetInput.value,
  1141. image: acceptHeaderImageInput.value
  1142. },
  1143. saveRawPage: saveRawPageInput.checked,
  1144. insertMetaCSP: insertMetaCSPInput.checked,
  1145. saveToClipboard: saveToClipboardInput.checked,
  1146. addProof: addProofInput.checked,
  1147. woleetKey: woleetKeyInput.value,
  1148. saveToGDrive: saveToGDriveInput.checked,
  1149. saveToDropbox: saveToDropboxInput.checked,
  1150. saveWithWebDAV: saveWithWebDAVInput.checked,
  1151. webDAVURL: webDAVURLInput.value,
  1152. webDAVUser: webDAVUserInput.value,
  1153. webDAVPassword: webDAVPasswordInput.value,
  1154. saveToGitHub: saveToGitHubInput.checked,
  1155. githubToken: githubTokenInput.value,
  1156. githubUser: githubUserInput.value,
  1157. githubRepository: githubRepositoryInput.value,
  1158. githubBranch: githubBranchInput.value,
  1159. saveWithCompanion: saveWithCompanionInput.checked,
  1160. sharePage: sharePageInput.checked,
  1161. compressHTML: compressHTMLInput.checked,
  1162. insertTextBody: insertTextBodyInput.checked,
  1163. insertEmbeddedImage: insertEmbeddedCustomImageInput.checked,
  1164. insertEmbeddedScreenshotImage: insertEmbeddedScreenshotImageInput.checked,
  1165. compressCSS: compressCSSInput.checked,
  1166. moveStylesInHead: moveStylesInHeadInput.checked,
  1167. loadDeferredImages: loadDeferredImagesInput.checked,
  1168. loadDeferredImagesMaxIdleTime: Math.max(loadDeferredImagesMaxIdleTimeInput.value, 0),
  1169. loadDeferredImagesKeepZoomLevel: loadDeferredImagesKeepZoomLevelInput.checked,
  1170. loadDeferredImagesDispatchScrollEvent: loadDeferredImagesDispatchScrollEventInput.checked,
  1171. loadDeferredImagesBeforeFrames: loadDeferredImagesBeforeFramesInput.checked,
  1172. contextMenuEnabled: contextMenuEnabledInput.checked,
  1173. filenameTemplate: filenameTemplateInput.value,
  1174. filenameMaxLength: filenameMaxLengthInput.value,
  1175. filenameMaxLengthUnit: filenameMaxLengthUnitInput.value,
  1176. filenameReplacementCharacter: filenameReplacementCharacterInput.value,
  1177. replaceEmojisInFilename: replaceEmojisInFilenameInput.checked,
  1178. saveFilenameTemplateData: saveFilenameTemplateDataInput.checked,
  1179. shadowEnabled: shadowEnabledInput.checked,
  1180. maxResourceSizeEnabled: maxResourceSizeEnabledInput.checked,
  1181. maxResourceSize: maxResourceSizeEnabledInput.checked ? Math.max(maxResourceSizeInput.value, 0) : 10,
  1182. networkTimeout: maxResourceDelayEnabledInput.checked ? Math.max(maxResourceDelayInput.value * 1000, 60) : 0,
  1183. confirmFilename: confirmFilenameInput.checked,
  1184. filenameConflictAction: filenameConflictActionInput.value,
  1185. displayInfobar: displayInfobarInput.checked,
  1186. displayStats: displayStatsInput.checked,
  1187. backgroundSave: backgroundSaveInput.checked,
  1188. autoSaveDelay: Math.max(autoSaveDelayInput.value, 0),
  1189. autoSaveLoad: autoSaveLoadInput.checked,
  1190. autoSaveUnload: autoSaveUnloadInput.checked,
  1191. autoSaveDiscard: autoSaveDiscardInput.checked,
  1192. autoSaveRemove: autoSaveRemoveInput.checked,
  1193. autoSaveLoadOrUnload: autoSaveLoadOrUnloadInput.checked,
  1194. autoSaveRepeat: autoSaveRepeatInput.checked,
  1195. autoSaveRepeatDelay: Math.max(autoSaveRepeatDelayInput.value, 1),
  1196. autoSaveExternalSave: autoSaveExternalSaveInput.checked,
  1197. removeAlternativeFonts: removeAlternativeFontsInput.checked,
  1198. removeAlternativeImages: removeAlternativeImagesInput.checked,
  1199. removeAlternativeMedias: removeAlternativeMediasInput.checked,
  1200. saveCreatedBookmarks: saveCreatedBookmarksInput.checked,
  1201. passReferrerOnError: passReferrerOnErrorInput.checked,
  1202. replaceBookmarkURL: replaceBookmarkURLInput.checked,
  1203. allowedBookmarkFolders: allowedBookmarkFoldersInput.value.replace(/([^\\]),/g, "$1 ,").split(/[^\\],/).map(folder => folder.replace(/\\,/g, ",")),
  1204. ignoredBookmarkFolders: ignoredBookmarkFoldersInput.value.replace(/([^\\]),/g, "$1 ,").split(/[^\\],/).map(folder => folder.replace(/\\,/g, ",")),
  1205. compressContent: fileFormatSelectInput.value.includes("zip"),
  1206. createRootDirectory: createRootDirectoryInput.checked,
  1207. preventAppendedData: preventAppendedDataInput.checked,
  1208. selfExtractingArchive: fileFormatSelectInput.value.includes("self-extracting"),
  1209. extractDataFromPage: fileFormatSelectInput.value == "self-extracting-zip-universal",
  1210. password: passwordInput.value,
  1211. groupDuplicateImages: groupDuplicateImagesInput.checked,
  1212. infobarTemplate: infobarTemplateInput.value,
  1213. blockMixedContent: blockMixedContentInput.checked,
  1214. saveOriginalURLs: saveOriginalURLsInput.checked,
  1215. includeInfobar: includeInfobarInput.checked,
  1216. openInfobar: openInfobarInput.checked,
  1217. removeSavedDate: removeInfobarSavedDateInput.checked,
  1218. confirmInfobarContent: confirmInfobarInput.checked,
  1219. autoClose: autoCloseInput.checked,
  1220. openEditor: openEditorInput.checked,
  1221. openSavedPage: openSavedPageInput.checked,
  1222. autoOpenEditor: autoOpenEditorInput.checked,
  1223. defaultEditorMode: defaultEditorModeInput.value,
  1224. applySystemTheme: applySystemThemeInput.checked,
  1225. warnUnsavedPage: warnUnsavedPageInput.checked,
  1226. displayInfobarInEditor: displayInfobarInEditorInput.checked,
  1227. saveToRestFormApi: saveToRestFormApiInput.checked,
  1228. saveToRestFormApiUrl: saveToRestFormApiUrlInput.value,
  1229. saveToRestFormApiToken: saveToRestFormApiTokenInput.value,
  1230. saveToRestFormApiFileFieldName: saveToRestFormApiFileFieldNameInput.value,
  1231. saveToRestFormApiUrlFieldName: saveToRestFormApiUrlFieldNameInput.value,
  1232. saveToS3: saveToS3Input.checked,
  1233. S3Domain: S3DomainInput.value,
  1234. S3Region: S3RegionInput.value,
  1235. S3Bucket: S3BucketInput.value,
  1236. S3AccessKey: S3AccessKeyInput.value,
  1237. S3SecretKey: S3SecretKeyInput.value
  1238. }
  1239. });
  1240. try {
  1241. await pendingSave;
  1242. } catch (error) {
  1243. // ignored
  1244. }
  1245. }
  1246. async function refreshExternalComponents() {
  1247. try {
  1248. await browser.runtime.sendMessage({ method: "ui.refreshMenu" });
  1249. if (sidePanelDisplay) {
  1250. await browser.runtime.sendMessage({ method: "options.refresh", profileName: profileNamesInput.value });
  1251. } else {
  1252. await browser.runtime.sendMessage({ method: "options.refreshPanel", profileName: profileNamesInput.value });
  1253. }
  1254. } catch (error) {
  1255. // ignored
  1256. }
  1257. }
  1258. async function saveCreatedBookmarks() {
  1259. if (saveCreatedBookmarksInput.checked) {
  1260. saveCreatedBookmarksInput.checked = false;
  1261. try {
  1262. const permissionGranted = await browser.permissions.request({ permissions: ["bookmarks"] });
  1263. if (permissionGranted) {
  1264. saveCreatedBookmarksInput.checked = true;
  1265. await update();
  1266. await refresh();
  1267. await browser.runtime.sendMessage({ method: "bookmarks.saveCreatedBookmarks" });
  1268. } else {
  1269. await disableOption();
  1270. }
  1271. } catch (error) {
  1272. saveCreatedBookmarksInput.checked = false;
  1273. await disableOption();
  1274. }
  1275. } else {
  1276. try {
  1277. await browser.permissions.remove({ permissions: ["bookmarks"] });
  1278. } catch (error) {
  1279. // ignored
  1280. }
  1281. await disableOption();
  1282. }
  1283. async function disableOption() {
  1284. await update();
  1285. await refresh();
  1286. await browser.runtime.sendMessage({ method: "bookmarks.disable" });
  1287. }
  1288. }
  1289. async function onClickSaveToClipboard() {
  1290. if (saveToClipboardInput.checked) {
  1291. saveToClipboardInput.checked = false;
  1292. try {
  1293. const permissionGranted = await browser.permissions.request({ permissions: ["clipboardWrite"] });
  1294. if (permissionGranted) {
  1295. saveToClipboardInput.checked = true;
  1296. await browser.runtime.sendMessage({ method: "downloads.disableGDrive" });
  1297. }
  1298. } catch (error) {
  1299. saveToClipboardInput.checked = false;
  1300. }
  1301. }
  1302. await update();
  1303. await refresh();
  1304. }
  1305. async function onClickSaveToGDrive() {
  1306. if (saveToGDriveInput.checked) {
  1307. saveToGDriveInput.checked = false;
  1308. try {
  1309. if (requestPermissionIdentity) {
  1310. const permissionGranted = await browser.permissions.request({ permissions: ["identity"] });
  1311. if (permissionGranted) {
  1312. saveToGDriveInput.checked = true;
  1313. }
  1314. } else {
  1315. saveToGDriveInput.checked = true;
  1316. }
  1317. } catch (error) {
  1318. saveToGDriveInput.checked = false;
  1319. await browser.runtime.sendMessage({ method: "downloads.disableGDrive" });
  1320. }
  1321. }
  1322. await update();
  1323. await refresh();
  1324. }
  1325. async function disableDestinationPermissions(permissions, disableGDrive = true, disableDropbox = true) {
  1326. if (disableGDrive) {
  1327. await browser.runtime.sendMessage({ method: "downloads.disableGDrive" });
  1328. }
  1329. if (disableDropbox) {
  1330. await browser.runtime.sendMessage({ method: "downloads.disableDropbox" });
  1331. }
  1332. try {
  1333. await browser.permissions.remove({ permissions });
  1334. } catch (error) {
  1335. //ignored
  1336. }
  1337. }
  1338. async function passReferrerOnError() {
  1339. if (passReferrerOnErrorInput.checked) {
  1340. passReferrerOnErrorInput.checked = false;
  1341. try {
  1342. const permissionGranted = await browser.permissions.request({ permissions: ["webRequest", "webRequestBlocking"] });
  1343. if (permissionGranted) {
  1344. passReferrerOnErrorInput.checked = true;
  1345. await update();
  1346. await refresh();
  1347. await browser.runtime.sendMessage({ method: "requests.enableReferrerOnError" });
  1348. } else {
  1349. await disableOption();
  1350. }
  1351. } catch (error) {
  1352. await disableOption();
  1353. }
  1354. } else {
  1355. await disableOption();
  1356. }
  1357. async function disableOption() {
  1358. await update();
  1359. await refresh();
  1360. await browser.runtime.sendMessage({ method: "requests.disableReferrerOnError" });
  1361. await browser.permissions.remove({ permissions: ["webRequest", "webRequestBlocking"] });
  1362. }
  1363. }
  1364. async function enableExternalSave(input) {
  1365. if (input.checked) {
  1366. input.checked = false;
  1367. try {
  1368. const permissionGranted = await browser.permissions.request({ permissions: ["nativeMessaging"] });
  1369. if (permissionGranted) {
  1370. input.checked = true;
  1371. await refreshOption();
  1372. if (window.chrome) {
  1373. window.chrome.runtime.reload();
  1374. }
  1375. } else {
  1376. await refreshOption();
  1377. }
  1378. } catch (error) {
  1379. input.checked = true;
  1380. await refreshOption();
  1381. }
  1382. } else {
  1383. await refreshOption();
  1384. }
  1385. async function refreshOption() {
  1386. await update();
  1387. await refresh();
  1388. }
  1389. }
  1390. async function confirm(message, positionY) {
  1391. document.getElementById("confirmLabel").textContent = message;
  1392. document.getElementById("formConfirmContainer").style.setProperty("display", "flex");
  1393. document.querySelector("#formConfirmContainer .popup-content").style.setProperty("margin-top", positionY + "px");
  1394. confirmButton.focus();
  1395. document.body.style.setProperty("overflow-y", "hidden");
  1396. return new Promise(resolve => {
  1397. confirmButton.onclick = event => hideAndResolve(event, true);
  1398. cancelButton.onclick = event => hideAndResolve(event);
  1399. window.onkeyup = event => {
  1400. if (event.key == "Escape") {
  1401. hideAndResolve(event);
  1402. }
  1403. };
  1404. function hideAndResolve(event, value) {
  1405. event.preventDefault();
  1406. document.getElementById("formConfirmContainer").style.setProperty("display", "none");
  1407. document.body.style.setProperty("overflow-y", "");
  1408. resolve(value);
  1409. }
  1410. });
  1411. }
  1412. async function reset(positionY) {
  1413. document.getElementById("formResetContainer").style.setProperty("display", "flex");
  1414. document.querySelector("#formResetContainer .popup-content").style.setProperty("margin-top", positionY + "px");
  1415. resetCancelButton.focus();
  1416. document.body.style.setProperty("overflow-y", "hidden");
  1417. return new Promise(resolve => {
  1418. resetAllButton.onclick = event => hideAndResolve(event, "all");
  1419. resetCurrentButton.onclick = event => hideAndResolve(event, "current");
  1420. resetCancelButton.onclick = event => hideAndResolve(event);
  1421. window.onkeyup = event => {
  1422. if (event.key == "Escape") {
  1423. hideAndResolve(event);
  1424. }
  1425. };
  1426. function hideAndResolve(event, value) {
  1427. event.preventDefault();
  1428. document.getElementById("formResetContainer").style.setProperty("display", "none");
  1429. document.body.style.setProperty("overflow-y", "");
  1430. resolve(value);
  1431. }
  1432. });
  1433. }
  1434. async function prompt(message, positionY, defaultValue = "") {
  1435. document.getElementById("promptLabel").textContent = message;
  1436. document.getElementById("formPromptContainer").style.setProperty("display", "flex");
  1437. document.querySelector("#formPromptContainer .popup-content").style.setProperty("margin-top", positionY + "px");
  1438. promptInput.value = defaultValue;
  1439. promptInput.focus();
  1440. document.body.style.setProperty("overflow-y", "hidden");
  1441. return new Promise(resolve => {
  1442. promptConfirmButton.onclick = event => hideAndResolve(event, promptInput.value);
  1443. promptCancelButton.onclick = event => hideAndResolve(event);
  1444. window.onkeyup = event => {
  1445. if (event.key == "Escape") {
  1446. hideAndResolve(event);
  1447. }
  1448. };
  1449. function hideAndResolve(event, value) {
  1450. event.preventDefault();
  1451. document.getElementById("formPromptContainer").style.setProperty("display", "none");
  1452. document.body.style.setProperty("overflow-y", "");
  1453. resolve(value);
  1454. }
  1455. });
  1456. }
  1457. async function getHelpContents() {
  1458. const helpPage = await fetch(browser.runtime.getURL(HELP_PAGE_PATH));
  1459. const content = new TextDecoder().decode(await helpPage.arrayBuffer());
  1460. const doc = (new DOMParser()).parseFromString(content, "text/html");
  1461. const items = doc.querySelectorAll("[data-options-label]");
  1462. items.forEach(itemElement => {
  1463. const optionLabel = document.getElementById(itemElement.dataset.optionsLabel);
  1464. if (optionLabel) {
  1465. const helpIconWrapper = document.createElement("span");
  1466. const helpIconContainer = document.createElement("span");
  1467. const helpIcon = document.createElement("img");
  1468. helpIcon.src = HELP_ICON_URL;
  1469. helpIconWrapper.className = "help-icon-wrapper";
  1470. const labelWords = optionLabel.textContent.split(/\s+/);
  1471. if (labelWords.length > 1) {
  1472. helpIconWrapper.textContent = labelWords.pop();
  1473. optionLabel.textContent = labelWords.join(" ") + " ";
  1474. }
  1475. helpIconContainer.className = "help-icon";
  1476. helpIconContainer.onclick = () => {
  1477. helpContent.hidden = !helpContent.hidden;
  1478. return false;
  1479. };
  1480. helpIcon.tabIndex = 0;
  1481. helpIconContainer.onkeyup = event => {
  1482. if (event.code == "Enter") {
  1483. helpContent.hidden = !helpContent.hidden;
  1484. return false;
  1485. }
  1486. };
  1487. helpIconContainer.appendChild(helpIcon);
  1488. helpIconWrapper.appendChild(helpIconContainer);
  1489. optionLabel.appendChild(helpIconWrapper);
  1490. const helpContent = document.createElement("div");
  1491. helpContent.hidden = true;
  1492. helpContent.className = "help-content";
  1493. itemElement.childNodes.forEach(node => {
  1494. if (node instanceof HTMLElement && node.className != "option") {
  1495. helpContent.appendChild(document.importNode(node, true));
  1496. }
  1497. });
  1498. helpContent.querySelectorAll("a[href]").forEach(linkElement => {
  1499. const hrefValue = linkElement.getAttribute("href");
  1500. if (hrefValue.startsWith("#")) {
  1501. linkElement.href = browser.runtime.getURL(HELP_PAGE_PATH + linkElement.getAttribute("href"));
  1502. linkElement.target = "_blank";
  1503. }
  1504. });
  1505. optionLabel.parentElement.insertAdjacentElement("afterEnd", helpContent);
  1506. }
  1507. });
  1508. }
  1509. function getLocalStorageItem(key) {
  1510. try {
  1511. return localStorage.getItem(key);
  1512. } catch (error) {
  1513. // ignored
  1514. }
  1515. }
  1516. function setLocalStorageItem(key, value) {
  1517. try {
  1518. return localStorage.setItem(key, value);
  1519. } catch (error) {
  1520. // ignored
  1521. }
  1522. }
  1523. function removeLocalStorageItem(key) {
  1524. try {
  1525. return localStorage.removeItem(key);
  1526. } catch (error) {
  1527. // ignored
  1528. }
  1529. }