extension-background.js 53 KB

1
  1. !function(){"use strict";async function e(t,a){let n;try{n=await browser.downloads.download(t)}catch(n){if(n.message){const o=n.message.toLowerCase(),r=o.includes("illegal characters")||o.includes("invalid filename");if(r&&t.filename.startsWith("."))return t.filename=a+t.filename,e(t,a);if(r&&t.filename.includes(","))return t.filename=t.filename.replace(/,/g,a),e(t,a);if(r&&!t.filename.match(/^[\x00-\x7F]+$/))return t.filename=t.filename.replace(/[^\x00-\x7F]+/g,a),e(t,a);if((o.includes("'incognito'")||o.includes('"incognito"'))&&t.incognito)return delete t.incognito,e(t,a);if("conflictaction prompt not yet implemented"==o&&t.conflictAction)return delete t.conflictAction,e(t,a);if(o.includes("canceled"))return{};throw n}throw n}return new Promise(((e,t)=>{browser.downloads.onChanged.addListener((function a(o){o.id==n&&o.state&&("complete"==o.state.current&&(browser.downloads.search({id:n}).then((t=>e({filename:t[0]&&t[0].filename}))).catch((()=>e({}))),browser.downloads.onChanged.removeListener(a)),"interrupted"==o.state.current&&(o.error&&"USER_CANCELED"==o.error.current?e({}):t(new Error(o.state.current)),browser.downloads.onChanged.removeListener(a)))}))}))}let t,a,n;async function o(e){a&&delete a[e];const t=await s();if(t[e]){const a=t[e].autoSave;t[e]={autoSave:a},await i(t)}}function r(e){return a||(a={}),void 0===e||a[e]||(a[e]={}),a}async function s(e){if(!t){const e=await browser.storage.local.get();t=e.tabsData||{}}return async function(){if(!n){n=!0;const e=await browser.tabs.query({currentWindow:!0,highlighted:!0});Object.keys(t).filter((t=>{if("autoSaveAll"!=t&&"autoSaveUnpinned"!=t&&"profileName"!=t)return!e.find((e=>e.id==t))})).forEach((e=>delete t[e])),await browser.storage.local.set({tabsData:t})}}(),void 0===e||t[e]||(t[e]={}),t}async function i(e){t=e,await browser.storage.local.set({tabsData:e})}setTimeout((()=>s().then((e=>t=e))),0);const c="-",l="__Default_Settings__",d="__Disabled_Settings__",u="regexp:",f={removeHiddenElements:!0,removeUnusedStyles:!0,removeUnusedFonts:!0,removeFrames:!1,removeImports:!0,removeScripts:!0,compressHTML:!0,compressCSS:!1,loadDeferredImages:!0,loadDeferredImagesMaxIdleTime:1500,loadDeferredImagesBlockCookies:!1,loadDeferredImagesBlockStorage:!1,loadDeferredImagesKeepZoomLevel:!1,filenameTemplate:"{page-title} ({date-locale} {time-locale}).html",infobarTemplate:"",includeInfobar:!1,confirmInfobarContent:!1,autoClose:!1,confirmFilename:!1,filenameConflictAction:"uniquify",filenameMaxLength:192,filenameMaxLengthUnit:"bytes",filenameReplacedCharacters:["~","+","\\\\","?","%","*",":","|",'"',"<",">","\0-",""],filenameReplacementCharacter:"_",contextMenuEnabled:!0,tabMenuEnabled:!0,browserActionMenuEnabled:!0,shadowEnabled:!0,logsEnabled:!0,progressBarEnabled:!0,maxResourceSizeEnabled:!1,maxResourceSize:10,removeAudioSrc:!0,removeVideoSrc:!0,displayInfobar:!0,displayStats:!1,backgroundSave:!/Mobile.*Firefox/.test(navigator.userAgent),defaultEditorMode:"normal",applySystemTheme:!0,autoSaveDelay:1,autoSaveLoad:!1,autoSaveUnload:!1,autoSaveLoadOrUnload:!0,autoSaveDiscard:!1,autoSaveRemove:!1,autoSaveRepeat:!1,autoSaveRepeatDelay:10,removeAlternativeFonts:!0,removeAlternativeMedias:!0,removeAlternativeImages:!0,groupDuplicateImages:!0,saveRawPage:!1,saveToClipboard:!1,addProof:!1,saveToGDrive:!1,saveToGitHub:!1,githubToken:"",githubUser:"",githubRepository:"SingleFile-Archives",githubBranch:"main",saveWithCompanion:!1,forceWebAuthFlow:!1,extractAuthCode:!0,resolveFragmentIdentifierURLs:!1,userScriptEnabled:!1,openEditor:!1,openSavedPage:!1,autoOpenEditor:!1,saveCreatedBookmarks:!1,allowedBookmarkFolders:[],ignoredBookmarkFolders:[],replaceBookmarkURL:!0,saveFavicon:!0,includeBOM:!1,warnUnsavedPage:!0,autoSaveExternalSave:!1,insertMetaNoIndex:!1,insertMetaCSP:!0,passReferrerOnError:!1,insertSingleFileComment:!0,blockMixedContent:!1,saveOriginalURLs:!1,acceptHeaders:{font:"application/font-woff2;q=1.0,application/font-woff;q=0.9,*/*;q=0.8",image:"image/avif,image/webp,image/apng,image/svg+xml,image/*,*/*;q=0.8",stylesheet:"text/css,*/*;q=0.1",script:"*/*",document:"text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8"},moveStylesInHead:!1,woleetKey:""};let m,h=w();async function w(){const{sync:e}=await browser.storage.local.get();m=e?browser.storage.sync:browser.storage.local;const t=await m.get();if(t.profiles)t.rules||(t.rules=[]),Object.keys(t.profiles).forEach((e=>p(t.profiles[e]))),await m.remove(["profiles","defaultProfile","rules"]),await m.set({profiles:t.profiles,rules:t.rules});else{const e=t;delete e.tabsData,p(e);const a={profiles:{},rules:[]};a.profiles.__Default_Settings__=e,m.remove(Object.keys(f)),await m.set(a)}t.maxParallelWorkers||await m.set({maxParallelWorkers:navigator.hardwareConcurrency||4})}function p(e){Object.keys(f).forEach((t=>function(e,t){void 0===e[t]&&(e[t]=f[t])}(e,t)))}async function b(e,t){const a=await g(),n=a.rules.filter((e=>v(e)));let o=n.sort(y).find((t=>e&&e.match(new RegExp(t.url.split(u)[1]))));if(!o){const n=a.rules.filter((e=>!v(e)));o=n.sort(y).find((a=>!t&&"*"==a.url||e&&e.includes(a.url)))}return o}async function g(){return await h,m.get(["profiles","rules","maxParallelWorkers"])}function y(e,t){return t.url.length-e.url.length}function v(e){return e.url.toLowerCase().startsWith(u)}async function k(t){if(t.method.endsWith(".deleteRules")&&await async function(e){const t=await g();t.rules=t.rules=e?t.rules.filter((t=>t.autoSaveProfile!=e&&t.profile!=e)):[],await m.set({rules:t.rules})}(t.profileName),t.method.endsWith(".deleteRule")&&await async function(e){if(!e)throw new Error("URL is empty");const t=await g();t.rules=t.rules.filter((t=>t.url!=e)),await m.set({rules:t.rules})}(t.url),t.method.endsWith(".addRule")&&await S(t.url,t.profileName,t.autoSaveProfileName),t.method.endsWith(".createProfile")&&await async function(e,t){const a=await g();if(Object.keys(a.profiles).includes(e))throw new Error("Duplicate profile name");a.profiles[e]=JSON.parse(JSON.stringify(a.profiles[t])),await m.set({profiles:a.profiles})}(t.profileName,t.fromProfileName||l),t.method.endsWith(".renameProfile")&&await async function(e,t){const[a,n]=await Promise.all([g(),s()]);if(!Object.keys(a.profiles).includes(e))throw new Error("Profile not found");if(Object.keys(a.profiles).includes(t))throw new Error("Duplicate profile name");if(e==l)throw new Error("Default settings cannot be renamed");n.profileName==e&&(n.profileName=t,await i(n));a.profiles[t]=a.profiles[e],a.rules.forEach((a=>{a.profile==e&&(a.profile=t),a.autoSaveProfile==e&&(a.autoSaveProfile=t)})),delete a.profiles[e],await m.set({profiles:a.profiles,rules:a.rules})}(t.profileName,t.newProfileName),t.method.endsWith(".deleteProfile")&&await async function(e){const[t,a]=await Promise.all([g(),s()]);if(!Object.keys(t.profiles).includes(e))throw new Error("Profile not found");if(e==l)throw new Error("Default settings cannot be deleted");a.profileName==e&&(delete a.profileName,await i(a));t.rules.forEach((t=>{t.profile==e&&(t.profile=l),t.autoSaveProfile==e&&(t.autoSaveProfile=l)})),delete t.profiles[e],await m.set({profiles:t.profiles,rules:t.rules})}(t.profileName),t.method.endsWith(".resetProfiles")&&await async function(){await h;const e=await s();delete e.profileName,await i(e),await m.remove(["profiles","rules","maxParallelWorkers"]),await browser.storage.local.set({sync:!1}),m=browser.storage.local,await w()}(),t.method.endsWith(".resetProfile")&&await async function(e){const t=await g();if(!Object.keys(t.profiles).includes(e))throw new Error("Profile not found");t.profiles[e]=f,await m.set({profiles:t.profiles})}(t.profileName),t.method.endsWith(".importConfig")&&await async function(e){await m.remove(["profiles","rules","maxParallelWorkers"]),await m.set({profiles:e.profiles,rules:e.rules,maxParallelWorkers:e.maxParallelWorkers}),await w()}(t.config),t.method.endsWith(".updateProfile")&&await async function(e,t){const a=await g();if(!Object.keys(a.profiles).includes(e))throw new Error("Profile not found");Object.keys(t).forEach((n=>a.profiles[e][n]=t[n])),await m.set({profiles:a.profiles})}(t.profileName,t.profile),t.method.endsWith(".updateRule")&&await T(t.url,t.newUrl,t.profileName,t.autoSaveProfileName),t.method.endsWith(".getConstants"))return{DISABLED_PROFILE_NAME:d,DEFAULT_PROFILE_NAME:l,CURRENT_PROFILE_NAME:c};if(t.method.endsWith(".getRules"))return async function(){return(await g()).rules}();if(t.method.endsWith(".getProfiles"))return I();if(t.method.endsWith(".exportConfig"))return async function(){const t=await g(),a=URL.createObjectURL(new Blob([JSON.stringify({profiles:t.profiles,rules:t.rules,maxParallelWorkers:t.maxParallelWorkers},null,2)],{type:"text/json"})),n={url:a,filename:`singlefile-settings-${(new Date).toISOString().replace(/:/g,"_")}.json`,saveAs:!0};try{await e(n,"_")}finally{URL.revokeObjectURL(a)}}();if(t.method.endsWith(".enableSync")){await browser.storage.local.set({sync:!0});const e=await browser.storage.sync.get();if(!e||!e.profiles){const e=await browser.storage.local.get();await browser.storage.sync.set({profiles:e.profiles,rules:e.rules,maxParallelWorkers:e.maxParallelWorkers})}return m=browser.storage.sync,{}}if(t.method.endsWith(".disableSync")){await browser.storage.local.set({sync:!1});const e=await browser.storage.sync.get();e&&e.profiles&&await browser.storage.local.set({profiles:e.profiles,rules:e.rules,maxParallelWorkers:e.maxParallelWorkers}),m=browser.storage.local}return t.method.endsWith(".isSync")?{sync:(await browser.storage.local.get()).sync}:{}}async function I(){return(await g()).profiles}async function x(e,t){const[a,n,o]=await Promise.all([g(),b(e),s()]),r=o.profileName||l;let i;if(n){const e=n[t?"autoSaveProfile":"profile"];i=e==c?r:e}else i=r;return Object.assign({profileName:i},a.profiles[i])}async function S(e,t,a){if(!e)throw new Error("URL is empty");const n=await g();if(n.rules.find((t=>t.url==e)))throw new Error("URL already exists");n.rules.push({url:e,profile:t,autoSaveProfile:a}),await m.set({rules:n.rules})}async function T(e,t,a,n){if(!e||!t)throw new Error("URL is empty");const o=await g(),r=o.rules.find((t=>t.url==e));if(!r)throw new Error("URL not found");if(o.rules.find((a=>a.url==t&&a.url!=e)))throw new Error("New URL already exists");r.url=t,r.profile=a,r.autoSaveProfile=n,await m.set({rules:o.rules})}async function E(){return(await m.get()).authInfo}async function P(e){await m.set({authInfo:e})}async function M(){let e=E();e.revokableAccessToken?P({revokableAccessToken:e.revokableAccessToken}):await m.remove(["authInfo"])}async function W(e){if(e){const[t,a]=await Promise.all([s(),b(e.url)]);return Boolean(t.autoSaveAll||t.autoSaveUnpinned&&!e.pinned||t[e.id]&&t[e.id].autoSave)&&(!a||a.autoSaveProfile!=d)}}const A=33554432,L="/extension/ui/pages/editor.html",C=new Map,R=new Map,U=browser.runtime.getURL(L);function B(e){return e.url==U}const _=new Map,D="x-single-file-request-id";function O(e,t={},a){return new Promise(((n,o)=>{const r=new XMLHttpRequest;if(r.withCredentials=!0,r.responseType="arraybuffer",r.onerror=e=>o(new Error(e.detail)),r.onreadystatechange=()=>{r.readyState==XMLHttpRequest.DONE&&(r.status||r.response.byteLength?401!=r.status&&403!=r.status&&404!=r.status||a?n({array:Array.from(new Uint8Array(r.response)),headers:{"content-type":r.getResponseHeader("Content-Type")},status:r.status}):O(e,t,!0).then(n).catch(o):o())},r.open("GET",e,!0),t.headers)for(const e of Object.entries(t.headers))r.setRequestHeader(e[0],e[1]);if(a){const e=String(Math.random()).substring(2);s=e,i=t.referrer,_.set(s,i),r.setRequestHeader(D,e)}var s,i;r.send()}))}browser.runtime.onMessage.addListener(((e,t)=>{if(e.method&&e.method.startsWith("singlefile.fetch"))return new Promise((a=>{(function(e,t){if("singlefile.fetch"==e.method)return O(e.url,{referrer:e.referrer,headers:e.headers});if("singlefile.fetchFrame"==e.method)return browser.tabs.sendMessage(t.tab.id,e)})(e,t).then(a).catch((e=>a({error:e&&e.toString()})))}))}));let j=!1;function N(e){return e.method.endsWith(".enableReferrerOnError")?(q(),{}):e.method.endsWith(".disableReferrerOnError")?(function(){try{browser.webRequest.onBeforeSendHeaders.removeListener(F)}catch(e){}j=!1}(),{}):void 0}function F(e){if(j){let t=e.requestHeaders.find((e=>e.name===D));if(t){e.requestHeaders=e.requestHeaders.filter((e=>e.name!==D));const a=_.get(t.value);if(a){_.delete(t.value);if(!e.requestHeaders.find((e=>"referer"===e.name.toLowerCase())))return e.requestHeaders.push({name:"Referer",value:a}),{requestHeaders:e.requestHeaders}}}}}function q(){if(!j){try{browser.webRequest.onBeforeSendHeaders.addListener(F,{urls:["<all_urls>"]},["blocking","requestHeaders","extraHeaders"])}catch(e){browser.webRequest.onBeforeSendHeaders.addListener(F,{urls:["<all_urls>"]},["blocking","requestHeaders"])}j=!0}}const z=new Map;async function H(e){return(await browser.tabs.query(e)).sort(((e,t)=>e.index-t.index))}const J="/extension/ui/resources/icon_128.png",G="/extension/ui/resources/icon_128_wait",K=browser.i18n.getMessage("buttonDefaultTooltip"),$=browser.i18n.getMessage("buttonBlockedTooltip"),X=browser.i18n.getMessage("buttonInitializingBadge"),Y=browser.i18n.getMessage("buttonInitializingTooltip"),Z=browser.i18n.getMessage("buttonErrorBadge"),V=browser.i18n.getMessage("buttonBlockedBadge"),Q=browser.i18n.getMessage("buttonOKBadge"),ee=browser.i18n.getMessage("buttonSaveProgressTooltip"),te=browser.i18n.getMessage("buttonUploadProgressTooltip"),ae=browser.i18n.getMessage("buttonAutoSaveActiveBadge"),ne=browser.i18n.getMessage("buttonAutoSaveActiveTooltip"),oe=[2,147,20,192],re=[4,229,36,192],se={default:{setBadgeBackgroundColor:{color:oe},setBadgeText:{text:""},setTitle:{title:K},setIcon:{path:J}},inject:{setBadgeBackgroundColor:{color:oe},setBadgeText:{text:X},setTitle:{title:Y}},execute:{setBadgeBackgroundColor:{color:re},setBadgeText:{text:X}},progress:{setBadgeBackgroundColor:{color:re},setBadgeText:{text:""}},edit:{setBadgeBackgroundColor:{color:oe},setBadgeText:{text:""},setTitle:{title:K},setIcon:{path:J}},end:{setBadgeBackgroundColor:{color:re},setBadgeText:{text:Q},setTitle:{title:K},setIcon:{path:J}},error:{setBadgeBackgroundColor:{color:[229,4,12,192]},setBadgeText:{text:Z},setTitle:{title:""},setIcon:{path:J}},forbidden:{setBadgeBackgroundColor:{color:[255,255,255,1]},setBadgeText:{text:V},setTitle:{title:$},setIcon:{path:J}},autosave:{inject:{setBadgeBackgroundColor:{color:[64,64,64,192]},setBadgeText:{text:ae},setTitle:{title:ne},setIcon:{path:J}},default:{setBadgeBackgroundColor:{color:[208,208,208,192]},setBadgeText:{text:ae},setTitle:{title:ne},setIcon:{path:J}}}};let ie;function ce(e,t){if(e.method.endsWith(".processInit")){delete r(t.tab.id)[t.tab.id].button,me(t.tab)}var a,n,o;return e.method.endsWith(".processProgress")&&e.maxIndex&&(a=t.tab.id,n=e.index,o=e.maxIndex,fe(a,n,o,ee)),e.method.endsWith(".processEnd")&&de(t.tab.id),e.method.endsWith(".processError")&&(e.error&&console.error("Initialization error",e.error),le(t.tab.id)),e.method.endsWith(".processCancelled")&&ue(t.tab),Promise.resolve({})}function le(e){he(e,pe("error"))}function de(e,t){he(e,t?pe("default",!0):pe("end"))}function ue(e){me(e)}function fe(e,t,a,n){const o=Math.max(Math.min(20,Math.floor(t/a*20)),0),r=Math.min(Math.floor(t/a*8),8),s=G+r+".png",i=pe("progress");i.setTitle={title:n+5*o+"%"},i.setIcon={path:s},he(e,i)}async function me(e){const t=pe("default",await W(e));await he(e.id,t)}async function he(e,t){const a=r(e);if(t){a[e].button||(a[e].button={lastState:null});const n=a[e].button.lastState||{},o={};Object.keys(t).forEach((e=>{void 0!==t[e]&&JSON.stringify(n[e])!=JSON.stringify(t[e])&&(o[e]=t[e])})),Object.keys(o).length&&(a[e].button.lastState=t,await async function(e,t){for(const a of Object.keys(t))await we(e,a,t[a])}(e,o))}}async function we(e,t,a){if(browser.browserAction[t]){const n=JSON.parse(JSON.stringify(a));n.tabId=e,await browser.browserAction[t](n)}}function pe(e,t){return JSON.parse(JSON.stringify(t?se.autosave[e]:se[e]))}browser.browserAction.onClicked.addListener((async e=>{const t=await H({currentWindow:!0,highlighted:!0});t.length<=1?function(e){ie.isSavingTab(e)?ie.cancelTab(e.id):ie.saveTabs([e])}(e):ie.saveTabs(t)}));const be=browser.menus||browser.contextMenus,ge=be&&be.onClicked&&be.create&&be.update&&be.removeAll,ye="save-page",ve="edit-and-save-page",ke="save-with-profile",Ie="save-selected-links",xe="view-pendings",Se="select-profile",Te="wasve-with-profile-",Ee="select-profile-",Pe="associate-with-profile",Me="associate-with-profile-",We="save-selected",Ae="save-frame",Le="save-tabs",Ce="save-selected-tabs",Re="save-unpinned-tabs",Ue="save-all-tabs",Be="button-save-selected-tabs",_e="button-save-unpinned-tabs",De="button-save-all-tabs",Oe="auto-save",je="auto-save-disabled",Ne="auto-save-tab",Fe="auto-save-unpinned",qe="auto-save-all",ze=browser.i18n.getMessage("menuCreateDomainRule"),He=browser.i18n.getMessage("menuUpdateRule"),Je=browser.i18n.getMessage("menuSavePage"),Ge=browser.i18n.getMessage("menuSaveWithProfile"),Ke=browser.i18n.getMessage("menuSaveSelectedLinks"),$e=browser.i18n.getMessage("menuEditPage"),Xe=browser.i18n.getMessage("menuEditAndSavePage"),Ye=browser.i18n.getMessage("menuViewPendingSaves"),Ze=browser.i18n.getMessage("menuSaveSelection"),Ve=browser.i18n.getMessage("menuSaveFrame"),Qe=browser.i18n.getMessage("menuSaveTabs"),et=browser.i18n.getMessage("menuSaveSelectedTabs"),tt=browser.i18n.getMessage("menuSaveUnpinnedTabs"),at=browser.i18n.getMessage("menuSaveAllTabs"),nt=browser.i18n.getMessage("menuSelectProfile"),ot=browser.i18n.getMessage("profileDefaultSettings"),rt=browser.i18n.getMessage("menuAutoSave"),st=browser.i18n.getMessage("menuAutoSaveDisabled"),it=browser.i18n.getMessage("menuAutoSaveTab"),ct=browser.i18n.getMessage("menuAutoSaveUnpinnedTabs"),lt=browser.i18n.getMessage("menuAutoSaveAllTabs"),dt=[ve,Ie,We,Ae,Oe,Pe],ut=new Map,ft=new Map;let mt,ht,wt,pt=!0,bt=!0,gt=new Map;async function yt(e){const[t,a]=await Promise.all([I(),s()]),n=await x(e&&e.url);if(ge&&n){const o=["page","frame","image","link","video","audio","selection"],r=[];if(n.browserActionMenuEnabled&&r.push("browser_action"),n.tabMenuEnabled)try{be.create({id:"temporary-id",contexts:["tab"],title:"title"}),r.push("tab")}catch(e){n.tabMenuEnabled=!1}await be.removeAll();const s=r.concat(...o),i=n.contextMenuEnabled?s:r;if(be.create({id:ye,contexts:i,title:Je}),be.create({id:ve,contexts:i,title:Xe}),be.create({id:Ie,contexts:n.contextMenuEnabled?r.concat(["selection"]):r,title:Ke}),Object.keys(t).length>1&&be.create({id:ke,contexts:i,title:Ge}),n.contextMenuEnabled&&be.create({id:"separator-1",contexts:o,type:"separator"}),be.create({id:We,contexts:i,title:Ze}),n.contextMenuEnabled&&be.create({id:Ae,contexts:["frame"],title:Ve}),be.create({id:Le,contexts:r,title:Qe}),be.create({id:Be,contexts:r,title:et,parentId:Le}),be.create({id:_e,contexts:r,title:tt,parentId:Le}),be.create({id:De,contexts:r,title:at,parentId:Le}),n.contextMenuEnabled&&(be.create({id:Ce,contexts:o,title:et}),be.create({id:Re,contexts:o,title:tt}),be.create({id:Ue,contexts:o,title:at}),be.create({id:"separator-2",contexts:o,type:"separator"})),Object.keys(t).length>1){be.create({id:Se,title:nt,contexts:i}),be.create({id:"wasve-with-profile-default",contexts:i,title:ot,parentId:ke});const r="select-profile-default",s=!a.profileName||a.profileName==l;let d;be.create({id:r,type:"radio",contexts:i,title:ot,checked:s,parentId:Se}),ut.set(r,s),be.create({id:Pe,title:ze,contexts:i}),ft.set(Pe,ze),e&&e.url&&(d=await b(e.url,!0));const u="associate-with-profile-current",f=!d||d.profile==c;be.create({id:u,type:"radio",contexts:i,title:c,checked:f,parentId:Pe}),ut.set(u,f);const m="associate-with-profile-default",h=Boolean(d)&&d.profile==l;be.create({id:m,type:"radio",contexts:i,title:ot,checked:h,parentId:Pe}),ut.set(m,h),gt=new Map,Object.keys(t).forEach(((e,t)=>{if(e!=l){let n=Te+t;be.create({id:n,contexts:i,title:e,parentId:ke}),n=Ee+t;let o=a.profileName==e;be.create({id:n,type:"radio",contexts:i,title:e,checked:o,parentId:Se}),ut.set(n,o),n=Me+t,o=Boolean(d)&&d.profile==e,be.create({id:n,type:"radio",contexts:i,title:e,checked:o,parentId:Pe}),ut.set(n,o),gt.set(e,t)}})),n.contextMenuEnabled&&be.create({id:"separator-3",contexts:o,type:"separator"})}be.create({id:Oe,contexts:i,title:rt}),be.create({id:je,type:"radio",title:st,contexts:i,checked:!0,parentId:Oe}),ut.set(je,!0),be.create({id:Ne,type:"radio",title:it,contexts:i,checked:!1,parentId:Oe}),ut.set(Ne,!1),be.create({id:Fe,type:"radio",title:ct,contexts:i,checked:!1,parentId:Oe}),ut.set(Fe,!1),be.create({id:qe,type:"radio",title:lt,contexts:i,checked:!1,parentId:Oe}),ut.set(qe,!1),be.create({id:"separator-4",contexts:i,type:"separator"}),be.create({id:xe,contexts:i,title:Ye})}mt=!0,ht&&(ht=!1,(await browser.tabs.query({})).forEach((async e=>await kt(e))))}async function vt(e){const t=await s(e.id);await async function(){const e=await browser.tabs.query({});return Promise.all(e.map((async e=>{const[t,a]=await Promise.all([x(e.url,!0),W(e)]);try{await browser.tabs.sendMessage(e.id,{method:"content.init",autoSaveEnabled:a,options:t})}catch(e){}})))}(),await me(e);try{await browser.runtime.sendMessage({method:"options.refresh",profileName:t.profileName})}catch(e){}}async function kt(e){if(ge&&mt){const t=[],a=await s(e.id);if(a[e.id].editorDetected)It(!1);else if(It(!0),t.push(St(je,!a[e.id].autoSave)),t.push(St(Ne,a[e.id].autoSave)),t.push(St(Fe,Boolean(a.autoSaveUnpinned))),t.push(St(qe,Boolean(a.autoSaveAll))),e&&e.url){const n=await x(e.url);t.push(async function(e,t){const a=pt;pt=t,(void 0===a||a!=t)&&await yt(e)}(e,n.contextMenuEnabled)),t.push(xt(ve,a[e.id].savedPageDetected?$e:Xe)),t.push(be.update(We,{visible:!n.saveRawPage})),t.push(be.update(ve,{visible:!n.openEditor||a[e.id].savedPageDetected}));let o="associate-with-profile-default",r=ze;const[s,i]=await Promise.all([I(),b(e.url)]);if(i){const e=gt.get(i.profile);e&&(o=Me+e,r=He)}Object.keys(s).length>1&&(Object.keys(s).forEach(((e,a)=>{e==l?t.push(St("associate-with-profile-default","associate-with-profile-default"==o)):t.push(St(Me+a,o==Me+a))})),t.push(xt(Pe,r)))}await Promise.all(t)}}async function It(e){const t=bt;if(bt=e,void 0===t||t!=e){const t=[];try{dt.forEach((a=>t.push(be.update(a,{visible:e})))),await Promise.all(t)}catch(e){}}}function xt(e,t){const a=ft.get(e);return ft.set(e,t),void 0===a||a!=t?be.update(e,{title:t}):void 0}async function St(e,t){t=Boolean(t),ut.set(e,t),await be.update(e,{checked:t})}Promise.resolve().then((async function(){ge&&(yt(),be.onClicked.addListener((async(e,t)=>{if(e.menuItemId==ye&&(e.linkUrl?wt.saveUrls([e.linkUrl]):wt.saveTabs([t])),e.menuItemId==ve){(await s(t.id))[t.id].savedPageDetected?wt.openEditor(t):e.linkUrl?wt.saveUrls([e.linkUrl],{openEditor:!0}):wt.saveTabs([t],{openEditor:!0})}if(e.menuItemId==Ie&&wt.saveSelectedLinks(t),e.menuItemId==xe&&await browser.tabs.create({active:!0,url:"/extension/ui/pages/pendings.html"}),e.menuItemId==We&&wt.saveTabs([t],{selected:!0}),e.menuItemId==Ae&&wt.saveTabs([t],{frameId:e.frameId}),e.menuItemId==Ce||e.menuItemId==Be){const e=await H({currentWindow:!0,highlighted:!0});wt.saveTabs(e)}if(e.menuItemId==Re||e.menuItemId==_e){const e=await H({currentWindow:!0,pinned:!1});wt.saveTabs(e)}if(e.menuItemId==Ue||e.menuItemId==De){const e=await H({currentWindow:!0});wt.saveTabs(e)}if(e.menuItemId==Ne){const e=await s(t.id);e[t.id].autoSave=!0,await i(e),vt(t)}if(e.menuItemId==je){const e=await s();Object.keys(e).forEach((t=>{"object"==typeof e[t]&&e[t].autoSave&&(e[t].autoSave=!1)})),e.autoSaveUnpinned=e.autoSaveAll=!1,await i(e),vt(t)}if(e.menuItemId==qe){const a=await s();a.autoSaveAll=e.checked,await i(a),vt(t)}if(e.menuItemId==Fe){const a=await s();a.autoSaveUnpinned=e.checked,await i(a),vt(t)}if(e.menuItemId.startsWith(Te)){const a=await I(),n=e.menuItemId.split(Te)[1];let o;if("default"==n)o=l;else{const e=Number(n);o=Object.keys(a)[e]}a[o].profileName=o,wt.saveTabs([t],a[o])}if(e.menuItemId.startsWith(Ee)){const[a,n]=await Promise.all([I(),s()]),o=e.menuItemId.split(Ee)[1];if("default"==o)n.profileName=l;else{const e=Number(o);n.profileName=Object.keys(a)[e]}await i(n),vt(t)}if(e.menuItemId.startsWith(Me)){const[a,n]=await Promise.all([I(),b(t.url,!0)]),o=e.menuItemId.split(Me)[1];let r;if("default"==o)r=l;else if("current"==o)r=c;else{const e=Number(o);r=Object.keys(a)[e]}n?await T(n.url,n.url,r,r):(await xt(Pe,He),await S(new URL(t.url).hostname,r,r))}})),mt?ht=!0:(await browser.tabs.query({})).forEach((async e=>await kt(e))))}));const Tt=browser.commands;let Et,Pt,Mt;function Wt(e,t){return e.method.endsWith(".refreshMenu")?function(e){if(e.method.endsWith("refreshMenu"))return yt(),Promise.resolve({})}(e):ce(e,t)}function At(e){!function(e){he(e.id,pe("forbidden"))}(e)}function Lt(e,t,a){!function(e,t,a){let n;a?n=pe("inject",!0):(n=pe(1==t?"inject":"execute"),n.setTitle={title:Y+" ("+t+"/2)"},n.setIcon={path:G+"0.png"}),he(e,n)}(e,t,a)}async function Ct(e,t,a){le(e),t&&await browser.tabs.sendMessage(e,{method:"content.error",error:t.toString(),link:a})}function Rt(e){!function(e){he(e,pe("edit"))}(e)}function Ut(e,t){de(e,t)}function Bt(e,t,a){!function(e,t,a){fe(e,t,a,te)}(e,t,a)}function _t(e){kt(e)}Tt&&Tt.onCommand&&Tt.onCommand.addListener&&Tt.onCommand.addListener((async e=>{if("save-selected-tabs"==e){const e=await H({currentWindow:!0,highlighted:!0});Et.saveTabs(e,{optionallySelected:!0})}if("save-all-tabs"==e){const e=await H({currentWindow:!0});Et.saveTabs(e)}}));const Dt=["dist/chrome-browser-polyfill.js","dist/single-file.js"],Ot=["dist/chrome-browser-polyfill.js","dist/single-file-frames.js"];async function jt(e,t){let a;if(await async function(e){const t=e.extensionScriptFiles||[];Pt||Mt||([Pt,Mt]=await Promise.all([Nt(Dt.concat(t)),Nt(Ot)]))}(t),!t.removeFrames)try{await browser.tabs.executeScript(e,{code:Mt,allFrames:!0,matchAboutBlank:!0,runAt:"document_start"})}catch(e){}try{await browser.tabs.executeScript(e,{code:Pt,allFrames:!1,runAt:"document_idle"}),a=!0}catch(e){}return a&&t.frameId&&await browser.tabs.executeScript(e,{code:"document.documentElement.dataset.requestedFrameId = true",frameId:t.frameId,matchAboutBlank:!0,runAt:"document_start"}),a}async function Nt(e){const t=e.map((async e=>{if("function"==typeof e)return"("+e.toString()+")();";{const t=await fetch(browser.runtime.getURL("../../../"+e));return(new TextDecoder).decode(await t.arrayBuffer())}}));let a="";for(const e of t)a+=await e;return a}const Ft="single-file-response-fetch",qt=(e,t)=>window.fetch(e,t);async function zt(e,t={}){try{let a=await qt(e,{cache:"force-cache",headers:t.headers});return 401!=a.status&&403!=a.status&&404!=a.status||(a=await Gt(e)),a}catch(a){const n=await Jt({method:"singlefile.fetch",url:e,referrer:t.referrer,headers:t.headers});return{status:n.status,headers:{get:e=>n.headers&&n.headers[e]},arrayBuffer:async()=>new Uint8Array(n.array).buffer}}}async function Ht(e,t){const a=await Jt({method:"singlefile.fetchFrame",url:e,frameId:t.frameId,referrer:t.referrer,headers:t.headers});return{status:a.status,headers:new Map(a.headers),arrayBuffer:async()=>new Uint8Array(a.array).buffer}}async function Jt(e){const t=await browser.runtime.sendMessage(e);if(!t||t.error)throw new Error(t&&t.error&&t.error.toString());return t}function Gt(e){return new Promise(((t,a)=>{var n,o,r,s;n=new CustomEvent("single-file-request-fetch",{detail:e}),window.dispatchEvent(n),o=Ft,r=function n(o){var r,s,i;o.detail?o.detail.url==e&&(r=Ft,s=n,i=!1,window.removeEventListener(r,s,i),o.detail.response?t({status:o.detail.status,headers:new Map(o.detail.headers),arrayBuffer:async()=>o.detail.response}):a(o.detail.error)):a()},s=!1,window.addEventListener(o,r,s)}))}function Kt(e,t){return jt(e,t)}browser.runtime.onMessage.addListener((e=>{if("singlefile.fetchFrame"==e.method&&window.frameId&&window.frameId==e.frameId)return async function(e){try{let t=await qt(e.url,{cache:"force-cache",headers:e.headers});return 401!=t.status&&403!=t.status&&404!=t.status||(t=await Promise.race([Gt(e.url),new Promise(((e,t)=>setTimeout((()=>t()),5e3)))])),{status:t.status,headers:[...t.headers],array:Array.from(new Uint8Array(await t.arrayBuffer()))}}catch(e){return{error:e&&e.toString()}}}(e)}));const $t="Could not establish connection. Receiving end does not exist.",Xt="The message port closed before a response was received.",Yt="Message manager disconnected",Zt="Cannot access contents of url ",Vt="pending",Qt="processing",ea=["dist/infobar.js","dist/extension.js"],ta=[];let aa,na=0;var oa;async function ra(e,t={}){await ca(),await Promise.all(e.map((async e=>{const a=await x(e);Object.keys(t).forEach((e=>a[e]=t[e])),a.autoClose=!0,a.extensionScriptFiles=ea,a.passReferrerOnError&&await q(),ia({tab:{url:e},status:Vt,options:a,method:"content.save"})}))),la()}async function sa(e,t={}){await ca(),await Promise.all(e.map((async e=>{const a=e.id,n=await x(e.url);if(Object.keys(t).forEach((e=>n[e]=t[e])),n.tabId=a,n.tabIndex=e.index,n.extensionScriptFiles=ea,n.passReferrerOnError&&await q(),t.autoSave){if(W(e)){da(ia({status:Qt,tab:e,options:n,method:"content.autosave"}))}}else{Lt(a,1);await Kt(a,n)||B(e)?(Lt(a,2),ia({status:Vt,tab:e,options:n,method:"content.save"})):At(e)}}))),la()}function ia(e){const t={id:na,status:e.status,tab:e.tab,options:e.options,method:e.method,done:function(){ta.splice(ta.findIndex((e=>e.id==this.id)),1),la()}};return ta.push(t),na++,t}async function ca(){aa||(aa=(await g()).maxParallelWorkers)}function la(){const e=ta.filter((e=>e.status==Qt)).length;for(let t=0;t<Math.min(ta.length-e,aa-e);t++){const e=ta.find((e=>e.status==Vt));e&&da(e)}}async function da(e){const t=e.id;if(e.status=Qt,!e.tab.id){let t;try{const a=await async function(e){const t=await browser.tabs.create(e);return new Promise(((e,a)=>{function n(a,r){a==t.id&&"complete"==r.status&&(e(t),browser.tabs.onUpdated.removeListener(n),browser.tabs.onRemoved.removeListener(o))}function o(e){e==t.id&&(a(e),browser.tabs.onRemoved.removeListener(o))}browser.tabs.onUpdated.addListener(n),browser.tabs.onRemoved.addListener(o)}))}({url:e.tab.url,active:!1});e.tab.id=e.options.tabId=a.id,e.tab.index=e.options.tabIndex=a.index,Lt(e.tab.id,1),t=await Kt(e.tab.id,e.options)}catch(t){e.tab.id=t}if(!t)return void e.done();Lt(e.tab.id,2)}e.options.taskId=t;try{await browser.tabs.sendMessage(e.tab.id,{method:e.method,options:e.options})}catch(t){!t||t.message&&function(e){return e.message==Xt||e.message==$t||e.message==Yt||e.message.startsWith(Zt+JSON.stringify(U))}(t)||(console.log(t.message?t.message:t),Ct(e.tab.id,t.message,t.link),e.done())}}function ua(e){const t=ta.find((t=>t.id==e));t&&(t.options.autoClose&&!t.cancelled&&browser.tabs.remove(t.tab.id),t.done())}function fa(e,t){const a=ta.find((t=>t.id==e));a&&(a.cancel=t)}function ma(e){Array.from(ta).filter((t=>t.tab.id==e&&!t.options.autoSave)).forEach(wa)}function ha(e){return ta.find((t=>t.id==e))}function wa(e){const t=e.tab.id;e.cancelled=!0,browser.tabs.sendMessage(t,{method:"content.cancelSave",options:{loadDeferredImages:e.options.loadDeferredImages,loadDeferredImagesKeepZoomLevel:e.options.loadDeferredImagesKeepZoomLevel}}),e.cancel&&e.cancel(),"content.autosave"==e.method&&Ut(t,!0),function(e){ue(e)}(e.tab),e.done()}function pa(e){return{id:e.id,tabId:e.tab.id,index:e.tab.index,url:e.tab.url,title:e.tab.title,cancelled:e.cancelled,status:e.status}}(function(e){wt=e})(oa={isSavingTab:function(e){return Boolean(ta.find((t=>t.tab.id==e.id)))},saveTabs:sa,saveUrls:ra,cancelTab:ma,openEditor:function(e){browser.tabs.sendMessage(e.id,{method:"content.openEditor"})},saveSelectedLinks:async function(e){const t={extensionScriptFiles:ea,tabId:e.id,tabIndex:e.index};if(await Kt(e.id,t)){const t=await browser.tabs.sendMessage(e.id,{method:"content.getSelectedLinks"});t.urls&&t.urls.length&&await ra(t.urls)}else At(e)}}),function(e){ie=e}(oa),function(e){Et=e}(oa);async function ba(e){const t=browser.runtime.connectNative("singlefile_companion");t.postMessage({method:"save",pageData:e}),await new Promise(((e,a)=>{t.onDisconnect.addListener((()=>{t.error?a(new Error(t.error.message+" (Companion)")):browser.runtime.lastError&&!browser.runtime.lastError.message.includes("Native host has exited")||e()}))}))}async function ga(e){return e.method.endsWith(".saveCreatedBookmarks")?(ya(),{}):e.method.endsWith(".disable")?(async function(){let e;const t=await I();Object.keys(t).forEach((a=>e=e||!t[a].saveCreatedBookmarks)),e&&browser.bookmarks.onCreated.removeListener(va)}(),{}):void 0}async function ya(){try{browser.bookmarks.onCreated.removeListener(va)}catch(e){}let e;const t=await I();Object.keys(t).forEach((a=>{t[a].saveCreatedBookmarks&&(e=!0)})),e&&browser.bookmarks.onCreated.addListener(va)}async function va(e,t){const a=await browser.tabs.query({lastFocusedWindow:!0,active:!0}),n=await x(t.url);if(n.saveCreatedBookmarks){const r=await async function e(t,a=[]){if(t){const n=(await browser.bookmarks.get(t))[0];n&&n.title&&(a.unshift(n.title),await e(n.parentId,a))}return a}(t.parentId),s=n.allowedBookmarkFolders.toString(),i=r.find((e=>n.allowedBookmarkFolders.includes(e))),c=n.ignoredBookmarkFolders.toString(),l=r.find((e=>n.ignoredBookmarkFolders.includes(e)));if((s&&i||!s)&&(c&&!l||!c))if(a.length&&a[0].url==t.url)sa(a,{bookmarkId:e,bookmarkFolders:r});else{const a=await browser.tabs.query({});if(a.length){const n=a.find((e=>e.url==t.url));n?sa([n],{bookmarkId:e,bookmarkFolders:r}):t.url&&("about:blank"==t.url?browser.bookmarks.onChanged.addListener((function t(a,n){a==e&&n.url&&(browser.bookmarks.onChanged.removeListener(t),o(n.url))})):o(t.url))}}}function o(t){ra([t],{bookmarkId:e})}}Promise.resolve().then(ya);async function ka(e,t){let a=t||"eyJhbGciOiJIUzI1NiJ9.eyJzdWIiOiJhYzZmZTMzMi0wODNjLTRjZmMtYmYxNC0xNWU5MTJmMWY4OWIiLCJpYXQiOjE1NzYxNzQzNDV9.n31j9ctJj7R1Vjwyc5yd1d6Cmg0NDnpwSaLWsqtZJQA";const n=await fetch("https://api.woleet.io/v1/anchor",{method:"POST",headers:{Accept:"application/json","Content-Type":"application/json",Authorization:"Bearer "+a},body:JSON.stringify({name:e,hash:e,public:!0})});if(401==n.status){const e=new Error("Your access token on Woleet is invalid. Go to __DOC_LINK__ to create your account.");throw e.link="https://app.woleet.io/",e}if(402==n.status){const e=new Error("You have no more credits on Woleet. Go to __DOC_LINK__ to recharge them.");throw e.link="https://app.woleet.io/",e}if(n.status>=400)throw new Error((n.statusText||"Error "+n.status)+" (Woleet)");return n.json()}const Ia="https://oauth2.googleapis.com/token",xa="https://www.googleapis.com/drive/v3/files";let Sa,Ta=!0;class Ea{constructor(e){this.file=e.file,this.onProgress=e.onProgress,this.contentType=this.file.type||"application/octet-stream",this.metadata={name:e.filename,mimeType:this.contentType,parents:e.parents||["root"]},this.token=e.token,this.offset=0,this.chunkSize=e.chunkSize||524288}async upload(){const e=Ua(await fetch("https://www.googleapis.com/upload/drive/v3/files?uploadType=resumable",{method:"POST",headers:{Authorization:"Bearer "+this.token,"Content-Type":"application/json","X-Upload-Content-Length":this.file.size,"X-Upload-Content-Type":this.contentType},body:JSON.stringify(this.metadata)})).headers.get("Location");if(this.url=e,!this.cancelled)return this.onProgress&&this.onProgress(0,this.file.size),Ca(this)}}async function Pa(e,t){const a=await fetch(Ia,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:"client_id="+e.clientId+"&grant_type=authorization_code&code="+t.code+"&redirect_uri="+e.redirectURI}),n=await Ra(a);return e.accessToken=n.access_token,e.refreshToken=n.refresh_token,e.expirationDate=Date.now()+1e3*n.expires_in,{accessToken:e.accessToken,refreshToken:e.refreshToken,expirationDate:e.expirationDate}}function Ma(e,t={}){return e.redirectURI=encodeURIComponent("urn:ietf:wg:oauth:2.0:oob"+(t.auto?":auto":"")),e.authURL="https://accounts.google.com/o/oauth2/v2/auth?client_id="+e.clientId+"&response_type=code&access_type=offline&redirect_uri="+e.redirectURI+"&scope="+e.scopes.join(" "),e.authURL}function Wa(e={}){return Boolean(browser.identity&&browser.identity.getAuthToken)&&!e.forceWebAuthFlow}async function Aa(e,t,a=!0){const n=t.split("/");n.pop();const o=e.folderIds.get(n.join("/"));if(o)return o;let r="root";if(n.length){let o="";for(const s of n){o&&(o+="/"),o+=s;const n=e.folderIds.get(o);if(n)r=n;else try{r=await La(e,s,r),e.folderIds.set(o,r)}catch(n){if("path_not_found"==n.message&&a)return e.folderIds.clear(),Aa(e,t,!1);throw n}}}return r}async function La(e,t,a){const n=await async function(e,t,a){return Ra(await fetch(xa+"?q=mimeType = 'application/vnd.google-apps.folder' and name = '"+t+"' and trashed != true and '"+a+"' in parents",{headers:{Authorization:"Bearer "+e.accessToken}}))}(e,t,a);if(n.files.length)return n.files[0].id;{const n=await async function(e,t,a){return Ra(await fetch(xa,{method:"POST",headers:{Authorization:"Bearer "+e.accessToken,"Content-Type":"application/json"},body:JSON.stringify({name:t,parents:[a],mimeType:"application/vnd.google-apps.folder"})}))}(e,t,a);return n.id}}async function Ca(e){let t=e.file,a=e.file.size;(e.offset||e.chunkSize)&&(e.chunkSize&&(a=Math.min(e.offset+e.chunkSize,e.file.size)),t=t.slice(e.offset,a));const n=await fetch(e.url,{method:"PUT",headers:{Authorization:"Bearer "+e.token,"Content-Type":e.contentType,"Content-Range":"bytes "+e.offset+"-"+(a-1)+"/"+e.file.size,"X-Upload-Content-Type":e.contentType},body:t});if(e.onProgress&&!e.cancelled&&e.onProgress(e.offset+e.chunkSize,e.file.size),200==n.status||201==n.status)return n.json();if(308==n.status){const t=n.headers.get("Range");if(t&&(e.offset=parseInt(t.match(/\d+/g).pop(),10)+1),e.cancelled)throw new Error("upload_cancelled");return Ca(e)}Ua(n)}async function Ra(e){e=Ua(e);const t=await e.json();if(t.error)throw new Error(t.error);return t}function Ua(e){if(200==e.status)return e;throw 404==e.status?new Error("path_not_found"):401==e.status?new Error("invalid_token"):new Error("unknown_error ("+e.status+")")}async function Ba(e,t,a,n,o,r){for(;Sa;)await Sa;const s=new AbortController;return Sa=(async()=>{try{await async function({path:o,content:r,message:s=""},i){try{const c=await fetch(`https://api.github.com/repos/${t}/${a}/contents/${o.replace(/#/g,"%23")}`,{method:"PUT",headers:new Map([["Authorization",`token ${e}`],["Accept","application/vnd.github.v3+json"]]),body:JSON.stringify({content:btoa(unescape(encodeURIComponent(r))),message:s,branch:n}),signal:i}),l=await c.json();if(c.status<400)return l;throw new Error(l.message)}catch(e){if("AbortError"!=e.name)throw e}}({path:o,content:r},s.signal)}finally{Sa=null}})(),{cancelPush:()=>s.abort(),pushPromise:Sa}}const _a=new Map,Da="text/html",Oa=/([{}()^$&.*?/+|[\\\\]|\]|-)/g,ja=browser.runtime.getManifest(),Na=ja.optional_permissions&&ja.optional_permissions.includes("identity"),Fa=new class{constructor(e,t){this.clientId=e,this.scopes=t,this.folderIds=new Map,setInterval((()=>this.folderIds.clear()),6e4)}async auth(e={interactive:!0,auto:!0}){if(e.requestPermissionIdentity&&Ta)try{await browser.permissions.request({permissions:["identity"]}),Ta=!1}catch(e){}return Wa(e)?(this.accessToken=await browser.identity.getAuthToken({interactive:e.interactive}),{revokableAccessToken:this.accessToken}):(Ma(this,e),e.code?Pa(this,e):async function(e,t){let a;t.extractAuthCode&&t.extractAuthCode(Ma(e,t)).then((e=>a=e)).catch((()=>{}));try{if(browser.identity&&browser.identity.launchWebAuthFlow&&!t.forceWebAuthFlow)return await browser.identity.launchWebAuthFlow({interactive:t.interactive,url:e.authURL});if(t.launchWebAuthFlow)return await t.launchWebAuthFlow({url:e.authURL});throw new Error("auth_not_supported")}catch(n){if(n.message&&("code_required"==n.message||n.message.includes("access"))){if(t.auto||a||!t.promptAuthCode||(a=await t.promptAuthCode()),a)return t.code=a,await Pa(e,t);throw new Error("code_required")}throw n}}(this,e))}setAuthInfo(e,t){Wa(t)||(e?(this.accessToken=e.accessToken,this.refreshToken=e.refreshToken,this.expirationDate=e.expirationDate):(delete this.accessToken,delete this.refreshToken,delete this.expirationDate))}async refreshAuthToken(){if(this.refreshToken){const e=await fetch(Ia,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:"client_id="+this.clientId+"&refresh_token="+this.refreshToken+"&grant_type=refresh_token"});if(400==e.status)throw new Error("unknown_token");const t=await Ra(e);return this.accessToken=t.access_token,t.refresh_token&&(this.refreshToken=t.refresh_token),t.expires_in&&(this.expirationDate=Date.now()+1e3*t.expires_in),{accessToken:this.accessToken,refreshToken:this.refreshToken,expirationDate:this.expirationDate}}}async revokeAuthToken(e){if(e){if(browser.identity&&browser.identity.removeCachedAuthToken)try{await browser.identity.removeCachedAuthToken({token:e})}catch(e){}const t=await fetch("https://accounts.google.com/o/oauth2/revoke",{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:"token="+e});try{await Ra(t)}catch(e){if("invalid_token"!=e.message)throw e}finally{delete this.accessToken,delete this.refreshToken,delete this.expirationDate}}}async upload(e,t,a,n=!0){const o=await Aa(this,e),r=e.split("/").pop(),s=new Ea({token:this.accessToken,file:t,parents:[o],filename:r,onProgress:a.onProgress});try{return{cancelUpload:()=>s.cancelled=!0,uploadPromise:s.upload()}}catch(o){if("path_not_found"==o.message&&n)return this.folderIds.clear(),this.upload(e,t,a,!1);throw o}}}("207618107333-3pj2pmelhnl4sf3rpctghs9cean3q8nj.apps.googleusercontent.com",["https://www.googleapis.com/auth/drive.file"]);async function qa(e,t){if(e.method.endsWith(".download"))return async function(e,t){let a;e.truncated?(a=_a.get(t.id),a||(a=[],_a.set(t.id,a)),a.push(e.content),e.finished&&_a.delete(t.id)):e.content&&(a=[e.content]);e.truncated&&!e.finished||(e.openEditor?(Rt(t.id),await async function({tabIndex:e,content:t,filename:a}){const n={active:!0,url:L};null!=e&&(n.index=e);const o=await browser.tabs.create(n);C.set(o.id,{content:t,filename:a})}({tabIndex:t.index+1,filename:e.filename,content:a.join("")})):e.saveToClipboard?(e.content=a.join(""),function(e){const t="copy";function a(t){t.clipboardData.setData(Da,e.content),t.clipboardData.setData("text/plain",e.content),t.preventDefault()}document.addEventListener(t,a),document.execCommand(t),document.removeEventListener(t,a)}(e),Ut(t.id)):await async function(e,t,a,n){try{if(n.saveToGDrive?await(await Ja(n.taskId,n.filename,new Blob(e,{type:Da}),{forceWebAuthFlow:n.forceWebAuthFlow,extractAuthCode:n.extractAuthCode},{onProgress:(e,a)=>Bt(t.id,e,a)})).uploadPromise:n.saveToGitHub?await(await Ha(n.taskId,n.filename,e.join(""),n.githubToken,n.githubUser,n.githubRepository,n.githubBranch)).pushPromise:n.saveWithCompanion?await ba({filename:n.filename,content:n.content,filenameConflictAction:n.filenameConflictAction}):(n.url=URL.createObjectURL(new Blob(e,{type:Da})),await Ga(n,{confirmFilename:n.confirmFilename,incognito:a,filenameConflictAction:n.filenameConflictAction,filenameReplacementCharacter:n.filenameReplacementCharacter,includeInfobar:n.includeInfobar})),Ut(t.id),n.openSavedPage){const a={active:!0,url:URL.createObjectURL(new Blob(e,{type:Da}))};null!=t.index&&(a.index=t.index+1),browser.tabs.create(a)}}catch(e){e.message&&"upload_cancelled"==e.message||(console.error(e),Ct(t.id,e.message,e.link))}finally{n.url&&URL.revokeObjectURL(n.url)}}(a,t,t.incognito,e));return{}}(e,t.tab);if(e.method.endsWith(".disableGDrive")){const e=await E();return M(),await Fa.revokeAuthToken(e&&(e.accessToken||e.revokableAccessToken)),{}}if(e.method.endsWith(".end")){if(e.hash)try{await ka(e.hash,e.woleetKey)}catch(e){Ct(t.tab.id,e.message,e.link)}return ua(e.taskId),{}}return e.method.endsWith(".getInfo")?ta.map(pa):e.method.endsWith(".cancel")?(a=e.taskId,wa(ta.find((e=>e.id==a))),{}):e.method.endsWith(".cancelAll")?(Array.from(ta).forEach(wa),{}):e.method.endsWith(".saveUrls")?(ra(e.urls),{}):void 0;var a}async function za(e,t){let a=await E();const n={interactive:!0,auto:e.extractAuthCode,forceWebAuthFlow:e.forceWebAuthFlow,requestPermissionIdentity:Na,launchWebAuthFlow:e=>async function(e){const t=await browser.tabs.create({url:e.url,active:!0});return new Promise(((e,a)=>{browser.tabs.onRemoved.addListener((function e(n){n==t.id&&(browser.tabs.onRemoved.removeListener(e),a(new Error("code_required")))}))}))}(e),extractAuthCode:e=>function(e){return new Promise(((t,a)=>{let n;function o(a,s){s&&s.url==e&&(n=a),n==a&&s&&s.title&&s.title.startsWith("Success code=")&&(browser.tabs.onUpdated.removeListener(o),browser.tabs.onUpdated.removeListener(r),t(s.title.substring(13,s.title.length-49)))}function r(e){e==n&&(browser.tabs.onUpdated.removeListener(o),browser.tabs.onUpdated.removeListener(r),a())}browser.tabs.onUpdated.addListener(o),browser.tabs.onRemoved.addListener(r)}))}(e),promptAuthCode:()=>async function(e){const t=await browser.tabs.query({currentWindow:!0,active:!0});return new Promise(((a,n)=>{const o=t[0].id;browser.tabs.onRemoved.addListener((function e(t){t==o&&(z.delete(t),browser.tabs.onUpdated.removeListener(e),n())})),z.set(o,{resolve:a,reject:n}),browser.tabs.sendMessage(o,{method:"common.promptValueRequest",promptMessage:e})}))}("Please enter the access code for Google Drive")};return Fa.setAuthInfo(a,n),a&&a.accessToken&&!t||(a=await Fa.auth(n),a?await P(a):await M()),a}async function Ha(e,t,a,n,o,r,s){const i=ha(e);if(!i||!i.cancelled){const i=Ba(n,o,r,s,t,a);fa(e,i.cancelPush);try{return await(await i).pushPromise,i}catch(e){throw new Error(e.message+" (GitHub)")}}}async function Ja(e,t,a,n,o){try{await za(n);const r=ha(e);if(!r||!r.cancelled){const n=await Fa.upload(t,a,o);return fa(e,n.cancelUpload),n}}catch(r){if("invalid_token"!=r.message)throw new Error(r.message+" (Google Drive)");{let r;try{r=await Fa.refreshAuthToken()}catch(e){if("unknown_token"!=e.message)throw new Error(e.message+" (Google Drive)");r=await za(n,!0)}r?await P(r):await M(),await Ja(e,t,a,n,o)}}}async function Ga(t,a){let n;if("skip"==a.filenameConflictAction){(await browser.downloads.search({filenameRegex:"(\\\\|/)"+(o=t.filename,o.replace(Oa,"\\$1")+"$"),exists:!0})).length?n=!0:a.filenameConflictAction="uniquify"}var o;if(!n){const n={url:t.url,saveAs:a.confirmFilename,filename:t.filename,conflictAction:a.filenameConflictAction};a.incognito&&(n.incognito=!0);const o=await e(n,a.filenameReplacementCharacter);o.filename&&t.bookmarkId&&t.replaceBookmarkURL&&(o.filename.startsWith("file:")||(o.filename.startsWith("/")&&(o.filename=o.filename.substring(1)),o.filename="file:///"+o.filename.replace(/#/g,"%23")),await async function(e,t){try{await browser.bookmarks.update(e,t)}catch(e){}}(t.bookmarkId,{url:o.filename}))}}const Ka={},$a={};async function Xa(e,t){if("enableAutoSave"==e.method){const a=await s(t.id);a[t.id].autoSave=e.enabled,await i(a),async function(e){Promise.all([yt(e),me(e)])}(t)}if("isAutoSaveEnabled"==e.method)return W(t)}async function Ya(e,t){const a=t.id,n=await x(t.url,!0);if(n){let o;Lt(a,1,!0),n.content=e.content,n.url=e.url,n.frames=e.frames,n.canvases=e.canvases,n.fonts=e.fonts,n.stylesheets=e.stylesheets,n.images=e.images,n.posters=e.posters,n.usedFonts=e.usedFonts,n.shadowRoots=e.shadowRoots,n.imports=e.imports,n.referrer=e.referrer,n.updatedResources=e.updatedResources,n.visitDate=new Date(e.visitDate),n.backgroundTab=!0,n.autoSave=!0,n.incognito=t.incognito,n.tabId=a,n.tabIndex=t.index;try{if(n.autoSaveExternalSave)await async function(e){e.autoSaveExternalSave=!1;const t=browser.runtime.connectNative("singlefile_companion");t.postMessage({method:"externalSave",pageData:e}),await new Promise(((e,a)=>{t.onDisconnect.addListener((()=>{t.error?a(new Error(t.error.message+" (Companion)")):browser.runtime.lastError&&!browser.runtime.lastError.message.includes("Native host has exited")||e()}))}))}(n);else{if(o=await function(e,t,a,n={fetch:zt,frameFetch:Ht}){return globalThis.singlefile.getPageData(e,n,t,a)}(n,null,null,{fetch:Za}),n.includeInfobar&&await infobar.includeScript(o),n.saveToGDrive){const t=new Blob([o.content],{type:"text/html"});await(await Ja(e.taskId,o.filename,t,n,{})).uploadPromise}else if(n.saveToGitHub)await(await Ha(e.taskId,o.filename,o.content,n.githubToken,n.githubUser,n.githubRepository,n.githubBranch)).pushPromise;else if(n.saveWithCompanion)await ba({filename:o.filename,content:o.content,filenameConflictAction:o.filenameConflictAction});else{const e=new Blob([o.content],{type:"text/html"});if(o.url=URL.createObjectURL(e),await Ga(o,n),n.openSavedPage){const n={active:!0,url:URL.createObjectURL(e),windowId:t.windowId},o=t.index;try{await browser.tabs.get(a),n.index=o+1}catch(e){n.index=o}browser.tabs.create(n)}}o.hash&&await ka(o.hash,n.woleetKey)}}finally{e.taskId?ua(e.taskId):n.autoClose&&(browser.tabs.remove($a[a]||a),delete $a[a]),o&&o.url&&URL.revokeObjectURL(o.url),Ut(a,!0)}}}function Za(e,t={}){return new Promise(((a,n)=>{const o=new XMLHttpRequest;if(o.withCredentials=!0,o.responseType="arraybuffer",o.onerror=e=>n(new Error(e.detail)),o.onreadystatechange=()=>{o.readyState==XMLHttpRequest.DONE&&a({status:o.status,headers:{get:e=>o.getResponseHeader(e)},arrayBuffer:async()=>o.response})},o.open("GET",e,!0),t.headers)for(const e of Object.entries(t.headers))o.setRequestHeader(e[0],e[1]);o.send()}))}async function Va(e,t){if(e.method.endsWith(".init")&&(await async function(e,t){await o(e.id);const a=await s(e.id);a[e.id].savedPageDetected=t.savedPageDetected,await i(a)}(t.tab,e),kt(t.tab),function(e){ma(e.id)}(t.tab),async function(e){const[t,a]=await Promise.all([x(e.url,!0),W(e)]);t&&(t.autoSaveLoad||t.autoSaveLoadOrUnload)&&a&&sa([e],{autoSave:!0})}(t.tab)),e.method.endsWith(".promptValueResponse")&&async function(e,t){const a=z.get(t.tab.id);a&&(a.resolve(e.value),z.delete(t.tab.id))}(e,t),e.method.endsWith(".getOptions"))return x(e.url);e.method.endsWith(".activate")&&await browser.tabs.update(e.tabId,{active:!0})}browser.tabs.onCreated.addListener((e=>function(e){!function(e){kt(e)}(e)}(e))),browser.tabs.onActivated.addListener((e=>async function(e){_t(await browser.tabs.get(e.tabId))}(e))),browser.tabs.onRemoved.addListener((e=>function(e){o(e),function(e){C.delete(e)}(e),ma(e),async function(e){const t=Ka[e];t?t.autoSaveRemove&&(delete Ka[e],await Ya(t,t.tab)):Ka[e]={removed:!0}}(e)}(e))),browser.tabs.onUpdated.addListener(((e,t)=>async function(e,t){if("complete"==t.status){setTimeout((async()=>{try{await browser.tabs.sendMessage(e,{method:"content.maybeInit"})}catch(e){}}),1500),function(e){delete Ka[e]}(e);const t=await browser.tabs.get(e);if(B(t)){const e=await s(t.id);e[t.id].editorDetected=!0,await i(e),_t(t)}}t.discarded&&async function(e){const t=Ka[e];t?(delete Ka[e],await Ya(t,t.tab)):Ka[e]={discarded:!0}}(e)}(e,t))),browser.tabs.onReplaced.addListener(((e,t)=>function(e,t){!async function(e,t){Ka[t]&&!Ka[e]&&(Ka[e]=Ka[t],delete Ka[t],$a[t]=e);const a=await s();a[t]&&!a[e]&&(a[e]=a[t],delete a[t],await i(a))}(e,t)}(e,t))),browser.runtime.onMessage.addListener(((e,t)=>{if("singlefile.frameTree.initResponse"==e.method||"singlefile.frameTree.ackInitRequest"==e.method)return browser.tabs.sendMessage(t.tab.id,e,{frameId:0}),Promise.resolve({})}));const Qa=new Map;function en(e,t){e.delete(t)}browser.runtime.onMessage.addListener(((e,t)=>{if("singlefile.lazyTimeout.setTimeout"==e.method){let a,n=Qa.get(t.tab.id);if(n)if(a=n.get(t.frameId),a){const t=a.get(e.type);t&&clearTimeout(t)}else a=new Map;const o=setTimeout((async()=>{try{const a=Qa.get(t.tab.id),n=a.get(t.frameId);a&&n&&en(n,e.type),await browser.tabs.sendMessage(t.tab.id,{method:"singlefile.lazyTimeout.onTimeout",type:e.type})}catch(e){}}),e.delay);return n||(n=new Map,a=new Map,n.set(t.frameId,a),Qa.set(t.tab.id,n)),a.set(e.type,o),Promise.resolve({})}if("singlefile.lazyTimeout.clearTimeout"==e.method){let a=Qa.get(t.tab.id);if(a){const n=a.get(t.frameId);if(n){const t=n.get(e.type);t&&clearTimeout(t),en(n,e.type)}}return Promise.resolve({})}})),browser.tabs.onRemoved.addListener((e=>Qa.delete(e))),browser.runtime.onMessage.addListener(((e,t)=>e.method.startsWith("tabs.")?Va(e,t):e.method.startsWith("downloads.")?qa(e,t):e.method.startsWith("autosave.")?async function(e,t){if(e.method.endsWith(".save"))return e.autoSaveDiscard||e.autoSaveRemove?(t.tab?(e.tab=t.tab,Ka[t.tab.id]=e):Ka[e.tabId]&&(Ka[e.tabId].removed&&e.autoSaveRemove||Ka[e.tabId].discarded&&e.autoSaveDiscard)&&(delete Ka[e.tabId],await Ya(e,{id:e.tabId,index:e.tabIndex,url:t.url})),e.autoSaveUnload&&(delete Ka[e.tabId],await Ya(e,t.tab))):(delete Ka[e.tabId],await Ya(e,t.tab)),{}}(e,t):e.method.startsWith("ui.")?Wt(e,t):e.method.startsWith("config.")?k(e):e.method.startsWith("tabsData.")?function(e){return e.method.endsWith(".get")?s():e.method.endsWith(".set")?i(e.tabsData):void 0}(e):e.method.startsWith("devtools.")?async function(e){e.method.endsWith(".resourceCommitted")&&e.tabId&&e.url&&("stylesheet"==e.type||"script"==e.type)&&await browser.tabs.sendMessage(e.tabId,e)}(e):e.method.startsWith("editor.")?async function(e,t){if(e.method.endsWith(".getTabData")){const e=t.tab,a=C.get(e.id);if(a){const t=await x(a.url),n=JSON.stringify(a);for(let a=0;a*A<n.length;a++){const o={method:"editor.setTabData"};o.truncated=n.length>A,o.truncated?(o.finished=(a+1)*A>n.length,o.content=n.substring(a*A,(a+1)*A)):(o.content=n,o.options=t),await browser.tabs.sendMessage(e.id,o)}}}if(e.method.endsWith(".open")){let a;const n=t.tab;if(e.truncated?(a=R.get(n.id),a||(a=[],R.set(n.id,a)),a.push(e.content),e.finished&&R.delete(n.id)):e.content&&(a=[e.content]),!e.truncated||e.finished){const t={url:L};await browser.tabs.update(n.id,t),C.set(n.id,{url:n.url,content:a.join(""),filename:e.filename})}}}(e,t):e.method.startsWith("bookmarks.")?ga(e):e.method.startsWith("companion.")?async function(e){if(e.method.endsWith(".state"))return{enabled:!0}}(e):e.method.startsWith("requests.")?N(e):e.method.startsWith("bootstrap.")?async function(e,t){if(e.method.endsWith(".init")){const[e,a]=await Promise.all([x(t.tab.url,!0),W(t.tab)]);return{options:e,autoSaveEnabled:a,tabId:t.tab.id,tabIndex:t.tab.index}}}(e,t):void 0)),browser.runtime.onMessageExternal&&browser.runtime.onMessageExternal.addListener((async(e,t)=>{const a=(await browser.tabs.query({currentWindow:!0,active:!0}))[0];return!!a&&Xa(e,a)}))}();