1
0

single-file-extension-background.js 57 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 r=n.message.toLowerCase(),o=r.includes("illegal characters")||r.includes("invalid filename");if(o&&t.filename.startsWith("."))return t.filename=a+t.filename,e(t,a);if(o&&t.filename.includes(","))return t.filename=t.filename.replace(/,/g,a),e(t,a);if(o&&t.filename.match(/\u200C/))return t.filename=t.filename.replace(/\u200C/g,a),e(t,a);if(o&&!t.filename.match(/^[\x00-\x7F]+$/))return t.filename=t.filename.replace(/[^\x00-\x7F]+/g,a),e(t,a);if((r.includes("'incognito'")||r.includes('"incognito"'))&&t.incognito)return delete t.incognito,e(t,a);if("conflictaction prompt not yet implemented"==r&&t.conflictAction)return delete t.conflictAction,e(t,a);if(r.includes("canceled"))return{};throw n}throw n}return new Promise(((e,t)=>{browser.downloads.onChanged.addListener((function a(r){r.id==n&&r.state&&("complete"==r.state.current&&(browser.downloads.search({id:n}).then((t=>e({filename:t[0]&&t[0].filename}))).catch((()=>e({}))),browser.downloads.onChanged.removeListener(a)),"interrupted"==r.state.current&&(r.error&&"USER_CANCELED"==r.error.current?e({}):t(new Error(r.state.current)),browser.downloads.onChanged.removeListener(a)))}))}))}let t,a,n;async function r(e,t,a){e[a]&&!e[t]&&(e[t]=e[a],delete e[a])}async function o(e){a&&delete a[e];const t=await i();if(t[e]){const a=t[e].autoSave;t[e]={autoSave:a},await c(t)}}function s(e){return a||(a={}),void 0===e||a[e]||(a[e]={}),a}async function i(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 c(e){t=e,await browser.storage.local.set({tabsData:e})}setTimeout((()=>i().then((e=>t=e))),0);const l="-",d="__Default_Settings__",u="__Disabled_Settings__",f="regexp:",h=!/Safari/.test(navigator.userAgent)||/Chrome/.test(navigator.userAgent),w=!(/Mobile.*Firefox/.test(navigator.userAgent)||/Safari/.test(navigator.userAgent)&&!/Chrome/.test(navigator.userAgent)),m=h,b=h,p=h,g=h,y=h,v=h,k=h,I=h,S=h,T=h,x=h,E={removeHiddenElements:!0,removeUnusedStyles:!0,removeUnusedFonts:!0,removeFrames:!1,compressHTML:!0,compressCSS:!1,loadDeferredImages:!0,loadDeferredImagesMaxIdleTime:1500,loadDeferredImagesBlockCookies:!1,loadDeferredImagesBlockStorage:!1,loadDeferredImagesKeepZoomLevel:!1,loadDeferredImagesDispatchScrollEvent:!1,filenameTemplate:"{page-title} ({date-locale} {time-locale}).html",infobarTemplate:"",includeInfobar:!v,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,displayInfobar:!0,displayStats:!1,backgroundSave:w,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,maxSizeDuplicateImages:524288,saveRawPage:!1,saveToClipboard:!1,addProof:!1,saveToGDrive:!1,saveWithWebDAV:!1,webDAVURL:"",webDAVUser:"",webDAVPassword:"",saveToGitHub:!1,githubToken:"",githubUser:"",githubRepository:"SingleFile-Archives",githubBranch:"main",saveWithCompanion:!1,forceWebAuthFlow:!1,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",video:"video/webm,video/ogg,video/*;q=0.9,application/ogg;q=0.7,audio/*;q=0.6,*/*;q=0.5",audio:"audio/webm,audio/ogg,audio/wav,audio/*;q=0.9,application/ogg;q=0.7,video/*;q=0.6,*/*;q=0.5"},moveStylesInHead:!1,networkTimeout:0,woleetKey:"",blockImages:!1,blockStylesheets:!1,blockFonts:!1,blockScripts:!0,blockVideos:!0,blockAudios:!0},P=[{url:"file:",profile:"__Default_Settings__",autoSaveProfile:"__Disabled_Settings__"}];let A,R=W();async function W(){const{sync:e}=await browser.storage.local.get();A=e?browser.storage.sync:browser.storage.local;const t=await A.get();if(t.profiles)t.rules||(t.rules=P),Object.keys(t.profiles).forEach((e=>L(t.profiles[e]))),await A.remove(["profiles","rules"]),await A.set({profiles:t.profiles,rules:t.rules});else{const e=t;delete e.tabsData,L(e);const a={profiles:{},rules:P};a.profiles.__Default_Settings__=e,A.remove(Object.keys(E)),await A.set(a)}t.maxParallelWorkers||await A.set({maxParallelWorkers:navigator.hardwareConcurrency||4})}function L(e){M(e,"blockScripts","removeScripts"),M(e,"blockVideos","removeVideoSrc"),M(e,"blockAudios","removeAudioSrc"),Object.keys(E).forEach((t=>function(e,t){void 0===e[t]&&(e[t]=E[t])}(e,t)))}function M(e,t,a){void 0===e[t]&&void 0!==e[a]&&(e[t]=e[a],delete e[a])}async function D(e,t){const a=await _(),n=a.rules.filter((e=>U(e)));let r=n.sort(C).find((t=>e&&e.match(new RegExp(t.url.split(f)[1]))));if(!r){const n=a.rules.filter((e=>!U(e)));r=n.sort(C).find((a=>!t&&"*"==a.url||e&&e.includes(a.url)))}return r}async function _(){return await R,A.get(["profiles","rules","maxParallelWorkers"])}function C(e,t){return t.url.length-e.url.length}function U(e){return e.url.toLowerCase().startsWith(f)}async function O(t){if(t.method.endsWith(".deleteRules")&&await async function(e){const t=await _();t.rules=t.rules=e?t.rules.filter((t=>t.autoSaveProfile!=e&&t.profile!=e)):[],await A.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 _();t.rules=t.rules.filter((t=>t.url!=e)),await A.set({rules:t.rules})}(t.url),t.method.endsWith(".addRule")&&await j(t.url,t.profileName,t.autoSaveProfileName),t.method.endsWith(".createProfile")&&await async function(e,t){const a=await _();if(Object.keys(a.profiles).includes(e))throw new Error("Duplicate profile name");a.profiles[e]=JSON.parse(JSON.stringify(a.profiles[t])),await A.set({profiles:a.profiles})}(t.profileName,t.fromProfileName||d),t.method.endsWith(".renameProfile")&&await async function(e,t){const[a,n]=await Promise.all([_(),i()]);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==d)throw new Error("Default settings cannot be renamed");n.profileName==e&&(n.profileName=t,await c(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 A.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([_(),i()]);if(!Object.keys(t.profiles).includes(e))throw new Error("Profile not found");if(e==d)throw new Error("Default settings cannot be deleted");a.profileName==e&&(delete a.profileName,await c(a));t.rules.forEach((t=>{t.profile==e&&(t.profile=d),t.autoSaveProfile==e&&(t.autoSaveProfile=d)})),delete t.profiles[e],await A.set({profiles:t.profiles,rules:t.rules})}(t.profileName),t.method.endsWith(".resetProfiles")&&await async function(){await R;const e=await i();delete e.profileName,await c(e),await A.remove(["profiles","rules","maxParallelWorkers"]),await browser.storage.local.set({sync:!1}),A=browser.storage.local,await W()}(),t.method.endsWith(".resetProfile")&&await async function(e){const t=await _();if(!Object.keys(t.profiles).includes(e))throw new Error("Profile not found");t.profiles[e]=E,await A.set({profiles:t.profiles})}(t.profileName),t.method.endsWith(".importConfig")&&await async function(e){await A.remove(["profiles","rules","maxParallelWorkers"]),await A.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 _();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 A.set({profiles:a.profiles})}(t.profileName,t.profile),t.method.endsWith(".updateRule")&&await q(t.url,t.newUrl,t.profileName,t.autoSaveProfileName),t.method.endsWith(".getConstants"))return{DISABLED_PROFILE_NAME:u,DEFAULT_PROFILE_NAME:d,CURRENT_PROFILE_NAME:l,BACKGROUND_SAVE_SUPPORTED:w,BADGE_COLOR_SUPPORTED:m,AUTO_SAVE_SUPPORTED:b,SELECTABLE_TABS_SUPPORTED:p,OPEN_SAVED_PAGE_SUPPORTED:y,AUTO_OPEN_EDITOR_SUPPORTED:g,INFOBAR_SUPPORTED:v,BOOKMARKS_API_SUPPORTED:k,IDENTITY_API_SUPPORTED:I,CLIPBOARD_API_SUPPORTED:S,NATIVE_API_API_SUPPORTED:T,WEB_BLOCKING_API_SUPPORTED:x};if(t.method.endsWith(".getRules"))return async function(){return(await _()).rules}();if(t.method.endsWith(".getProfiles"))return B();if(t.method.endsWith(".exportConfig"))return async function(){const t=await _(),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 A=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}),A=browser.storage.local}return t.method.endsWith(".isSync")?{sync:(await browser.storage.local.get()).sync}:{}}async function B(){return(await _()).profiles}async function N(e,t){const[a,n,r]=await Promise.all([_(),D(e),i()]),o=r.profileName||d;let s;if(n){const e=n[t?"autoSaveProfile":"profile"];s=e==l?o:e}else s=o;return Object.assign({profileName:s},a.profiles[s])}async function j(e,t,a){if(!e)throw new Error("URL is empty");const n=await _();if(n.rules.find((t=>t.url==e)))throw new Error("URL already exists");n.rules.push({url:e,profile:t,autoSaveProfile:a}),await A.set({rules:n.rules})}async function q(e,t,a,n){if(!e||!t)throw new Error("URL is empty");const r=await _(),o=r.rules.find((t=>t.url==e));if(!o)throw new Error("URL not found");if(r.rules.find((a=>a.url==t&&a.url!=e)))throw new Error("New URL already exists");o.url=t,o.profile=a,o.autoSaveProfile=n,await A.set({rules:r.rules})}async function F(){return(await A.get()).authInfo}async function z(e){await A.set({authInfo:e})}async function H(){let e=F();e.revokableAccessToken?z({revokableAccessToken:e.revokableAccessToken}):await A.remove(["authInfo"])}async function V(e){if(e){const[t,a]=await Promise.all([i(),D(e.url)]);return Boolean(t.autoSaveAll||t.autoSaveUnpinned&&!e.pinned||t[e.id]&&t[e.id].autoSave)&&(!a||a.autoSaveProfile!=u)}}const J=33554432,G="/src/ui/pages/editor.html",K=new Map,Y=new Map,$=browser.runtime.getURL(G);function X(e){return e.url==$}const Z=new Map,Q="x-single-file-request-id",ee=8388608;async function te(e,t,a){for(let n=0;n*ee<=a.array.length;n++){const r={method:"singlefile.fetchResponse",requestId:t,headers:a.headers,status:a.status,error:a.error};r.truncated=a.array.length>ee,r.truncated?(r.finished=(n+1)*ee>a.array.length,r.array=a.array.slice(n*ee,(n+1)*ee)):r.array=a.array,await browser.tabs.sendMessage(e,r)}return{}}function ae(e,t={},a){return new Promise(((n,r)=>{const o=new XMLHttpRequest;if(o.withCredentials=!0,o.responseType="arraybuffer",o.onerror=e=>r(new Error(e.detail)),o.onreadystatechange=()=>{o.readyState==XMLHttpRequest.DONE&&(o.status||o.response.byteLength?401!=o.status&&403!=o.status&&404!=o.status||a?n({array:Array.from(new Uint8Array(o.response)),headers:{"content-type":o.getResponseHeader("Content-Type")},status:o.status}):ae(e,t,!0).then(n).catch(r):r())},o.open("GET",e,!0),t.headers)for(const e of Object.entries(t.headers))o.setRequestHeader(e[0],e[1]);if(a){const e=String(Math.random()).substring(2);!function(e,t){Z.set(e,t)}(e,t.referrer),o.setRequestHeader(Q,e)}o.send()}))}browser.runtime.onMessage.addListener(((e,t)=>{if(e.method&&e.method.startsWith("singlefile.fetch"))return new Promise((a=>{(async function(e,t){if("singlefile.fetch"==e.method)try{const a=await ae(e.url,{referrer:e.referrer,headers:e.headers});return te(t.tab.id,e.requestId,a)}catch(a){return te(t.tab.id,e.requestId,{error:a.message,arrray:[]})}else 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 ne=!1;function re(e){return e.method.endsWith(".enableReferrerOnError")?(se(),{}):e.method.endsWith(".disableReferrerOnError")?(function(){try{browser.webRequest.onBeforeSendHeaders.removeListener(oe)}catch(e){}ne=!1}(),{}):void 0}function oe(e){if(ne){let t=e.requestHeaders.find((e=>e.name===Q));if(t){e.requestHeaders=e.requestHeaders.filter((e=>e.name!==Q));const a=Z.get(t.value);if(a){Z.delete(t.value);if(!e.requestHeaders.find((e=>"referer"===e.name.toLowerCase())))return e.requestHeaders.push({name:"Referer",value:a}),{requestHeaders:e.requestHeaders}}}}}function se(){if(!ne){try{browser.webRequest.onBeforeSendHeaders.addListener(oe,{urls:["<all_urls>"]},["blocking","requestHeaders","extraHeaders"])}catch(e){browser.webRequest.onBeforeSendHeaders.addListener(oe,{urls:["<all_urls>"]},["blocking","requestHeaders"])}ne=!0}}async function ie(e){return(await browser.tabs.query(e)).sort(((e,t)=>e.index-t.index))}const ce="/src/ui/resources/icon_128.png",le="/src/ui/resources/icon_128_wait",de=browser.i18n.getMessage("buttonDefaultTooltip"),ue=browser.i18n.getMessage("buttonBlockedTooltip"),fe=browser.i18n.getMessage("buttonInitializingBadge"),he=browser.i18n.getMessage("buttonInitializingTooltip"),we=browser.i18n.getMessage("buttonErrorBadge"),me=browser.i18n.getMessage("buttonBlockedBadge"),be=browser.i18n.getMessage("buttonOKBadge"),pe=browser.i18n.getMessage("buttonSaveProgressTooltip"),ge=browser.i18n.getMessage("buttonUploadProgressTooltip"),ye=browser.i18n.getMessage("buttonAutoSaveActiveBadge"),ve=browser.i18n.getMessage("buttonAutoSaveActiveTooltip"),ke=[2,147,20,192],Ie=[4,229,36,192],Se={default:{setBadgeBackgroundColor:{color:ke},setBadgeText:{text:""},setTitle:{title:de},setIcon:{path:ce}},inject:{setBadgeBackgroundColor:{color:ke},setBadgeText:{text:fe},setTitle:{title:he}},execute:{setBadgeBackgroundColor:{color:Ie},setBadgeText:{text:fe}},progress:{setBadgeBackgroundColor:{color:Ie},setBadgeText:{text:""}},edit:{setBadgeBackgroundColor:{color:ke},setBadgeText:{text:""},setTitle:{title:de},setIcon:{path:ce}},end:{setBadgeBackgroundColor:{color:Ie},setBadgeText:{text:be},setTitle:{title:de},setIcon:{path:ce}},error:{setBadgeBackgroundColor:{color:[229,4,12,192]},setBadgeText:{text:we},setTitle:{title:""},setIcon:{path:ce}},forbidden:{setBadgeBackgroundColor:{color:[255,255,255,1]},setBadgeText:{text:me},setTitle:{title:ue},setIcon:{path:ce}},autosave:{inject:{setBadgeBackgroundColor:{color:[64,64,64,192]},setBadgeText:{text:ye},setTitle:{title:ve},setIcon:{path:ce}},default:{setBadgeBackgroundColor:{color:[208,208,208,192]},setBadgeText:{text:ye},setTitle:{title:ve},setIcon:{path:ce}}}};let Te;function xe(e,t){if(e.method.endsWith(".processInit")){delete s(t.tab.id)[t.tab.id].button,We(t.tab)}var a,n,r;return e.method.endsWith(".processProgress")&&e.maxIndex&&(a=t.tab.id,n=e.index,r=e.maxIndex,Re(a,n,r,pe)),e.method.endsWith(".processEnd")&&Pe(t.tab.id),e.method.endsWith(".processError")&&(e.error&&console.error("Initialization error",e.error),Ee(t.tab.id)),e.method.endsWith(".processCancelled")&&Ae(t.tab),Promise.resolve({})}function Ee(e){Le(e,De("error"))}function Pe(e,t){Le(e,t?De("default",!0):De("end"))}function Ae(e){We(e)}function Re(e,t,a,n){const r=Math.max(Math.min(20,Math.floor(t/a*20)),0),o=Math.min(Math.floor(t/a*8),8),s=le+o+".png",i=De("progress");i.setTitle={title:n+5*r+"%"},i.setIcon={path:s},Le(e,i)}async function We(e){const t=De("default",await V(e));await Le(e.id,t)}async function Le(e,t){try{const a=s(e);if(t){a[e].button||(a[e].button={lastState:null});const n=a[e].button.lastState||{},r={};Object.keys(t).forEach((e=>{void 0!==t[e]&&JSON.stringify(n[e])!=JSON.stringify(t[e])&&(r[e]=t[e])})),Object.keys(r).length&&(a[e].button.lastState=t,await async function(e,t){for(const a of Object.keys(t))await Me(e,a,t[a])}(e,r))}}catch(e){}}async function Me(e,t,a){const n="setBadgeBackgroundColor"!=t||m;if(browser.browserAction[t]&&n){const n=JSON.parse(JSON.stringify(a));n.tabId=e,await browser.browserAction[t](n)}}function De(e,t){return JSON.parse(JSON.stringify(t?Se.autosave[e]:Se[e]))}browser.browserAction.onClicked.addListener((async e=>{const t=await ie({currentWindow:!0,highlighted:!0});t.length<=1?function(e){Te.isSavingTab(e)?Te.cancelTab(e.id):Te.saveTabs([e])}(e):Te.saveTabs(t)}));const _e=browser.menus,Ce=_e&&_e.onClicked&&_e.create&&_e.update&&_e.removeAll,Ue="save-page",Oe="edit-and-save-page",Be="save-with-profile",Ne="save-selected-links",je="view-pendings",qe="select-profile",Fe="wasve-with-profile-",ze="select-profile-",He="associate-with-profile",Ve="associate-with-profile-",Je="save-selected",Ge="save-frame",Ke="save-tabs",Ye="save-selected-tabs",$e="save-unpinned-tabs",Xe="save-all-tabs",Ze="batch-save-urls",Qe="button-save-selected-tabs",et="button-save-unpinned-tabs",tt="button-save-all-tabs",at="auto-save",nt="auto-save-disabled",rt="auto-save-tab",ot="auto-save-unpinned",st="auto-save-all",it=browser.i18n.getMessage("menuCreateDomainRule"),ct=browser.i18n.getMessage("menuUpdateRule"),lt=browser.i18n.getMessage("menuSavePage"),dt=browser.i18n.getMessage("menuSaveWithProfile"),ut=browser.i18n.getMessage("menuSaveSelectedLinks"),ft=browser.i18n.getMessage("menuEditPage"),ht=browser.i18n.getMessage("menuEditAndSavePage"),wt=browser.i18n.getMessage("menuViewPendingSaves"),mt=browser.i18n.getMessage("menuSaveSelection"),bt=browser.i18n.getMessage("menuSaveFrame"),pt=browser.i18n.getMessage("menuSaveTabs"),gt=browser.i18n.getMessage("menuSaveSelectedTabs"),yt=browser.i18n.getMessage("menuSaveUnpinnedTabs"),vt=browser.i18n.getMessage("menuSaveAllTabs"),kt=browser.i18n.getMessage("menuBatchSaveUrls"),It=browser.i18n.getMessage("menuSelectProfile"),St=browser.i18n.getMessage("profileDefaultSettings"),Tt=browser.i18n.getMessage("menuAutoSave"),xt=browser.i18n.getMessage("menuAutoSaveDisabled"),Et=browser.i18n.getMessage("menuAutoSaveTab"),Pt=browser.i18n.getMessage("menuAutoSaveUnpinnedTabs"),At=browser.i18n.getMessage("menuAutoSaveAllTabs"),Rt=[Oe,Ne,Je,Ge,at,He],Wt=new Map,Lt=new Map;let Mt,Dt,_t,Ct=!0,Ut=!0,Ot=new Map;async function Bt(e){const[t,a]=await Promise.all([B(),i()]),n=await N(e&&e.url);if(Ce&&n){const r=["page","frame","image","link","video","audio","selection"],o=[];if(n.browserActionMenuEnabled&&o.push("browser_action"),n.tabMenuEnabled)try{await _e.create({id:"temporary-id",contexts:["tab"],title:"title"}),o.push("tab")}catch(e){n.tabMenuEnabled=!1}await _e.removeAll();const s=o.concat(...r),i=n.contextMenuEnabled?s:o;if(_e.create({id:Ue,contexts:i,title:lt}),_e.create({id:Oe,contexts:i,title:ht}),_e.create({id:Ne,contexts:n.contextMenuEnabled?o.concat(["selection"]):o,title:ut}),Object.keys(t).length>1&&_e.create({id:Be,contexts:i,title:dt}),n.contextMenuEnabled&&_e.create({id:"separator-1",contexts:r,type:"separator"}),_e.create({id:Je,contexts:i,title:mt}),n.contextMenuEnabled&&_e.create({id:Ge,contexts:["frame"],title:bt}),_e.create({id:Ke,contexts:o,title:pt}),_e.create({id:Qe,contexts:o,title:gt,parentId:Ke}),_e.create({id:et,contexts:o,title:yt,parentId:Ke}),_e.create({id:tt,contexts:o,title:vt,parentId:Ke}),n.contextMenuEnabled&&(p&&_e.create({id:Ye,contexts:r,title:gt}),_e.create({id:$e,contexts:r,title:yt}),_e.create({id:Xe,contexts:r,title:vt}),_e.create({id:"separator-2",contexts:r,type:"separator"})),Object.keys(t).length>1){_e.create({id:qe,title:It,contexts:i}),_e.create({id:"wasve-with-profile-default",contexts:i,title:St,parentId:Be});const o="select-profile-default",s=!a.profileName||a.profileName==d;let c;_e.create({id:o,type:"radio",contexts:i,title:St,checked:s,parentId:qe}),Wt.set(o,s),_e.create({id:He,title:it,contexts:i}),Lt.set(He,it),e&&e.url&&(c=await D(e.url,!0));const u="associate-with-profile-current",f=!c||c.profile==l;_e.create({id:u,type:"radio",contexts:i,title:l,checked:f,parentId:He}),Wt.set(u,f);const h="associate-with-profile-default",w=Boolean(c)&&c.profile==d;_e.create({id:h,type:"radio",contexts:i,title:St,checked:w,parentId:He}),Wt.set(h,w),Ot=new Map,Object.keys(t).forEach(((e,t)=>{if(e!=d){let n=Fe+t;_e.create({id:n,contexts:i,title:e,parentId:Be}),n=ze+t;let r=a.profileName==e;_e.create({id:n,type:"radio",contexts:i,title:e,checked:r,parentId:qe}),Wt.set(n,r),n=Ve+t,r=Boolean(c)&&c.profile==e,_e.create({id:n,type:"radio",contexts:i,title:e,checked:r,parentId:He}),Wt.set(n,r),Ot.set(e,t)}})),n.contextMenuEnabled&&_e.create({id:"separator-3",contexts:r,type:"separator"})}b&&(_e.create({id:at,contexts:i,title:Tt}),_e.create({id:nt,type:"radio",title:xt,contexts:i,checked:!0,parentId:at}),Wt.set(nt,!0),_e.create({id:rt,type:"radio",title:Et,contexts:i,checked:!1,parentId:at}),Wt.set(rt,!1),_e.create({id:ot,type:"radio",title:Pt,contexts:i,checked:!1,parentId:at}),Wt.set(ot,!1),_e.create({id:st,type:"radio",title:At,contexts:i,checked:!1,parentId:at}),Wt.set(st,!1),_e.create({id:"separator-4",contexts:i,type:"separator"})),_e.create({id:Ze,contexts:i,title:kt}),_e.create({id:je,contexts:i,title:wt})}Mt=!0,Dt&&(Dt=!1,(await browser.tabs.query({})).forEach((async e=>await jt(e))))}async function Nt(e){const t=await i(e.id);await async function(){const e=await browser.tabs.query({});return Promise.all(e.map((async e=>{const[t,a]=await Promise.all([N(e.url,!0),V(e)]);try{await browser.tabs.sendMessage(e.id,{method:"content.init",autoSaveEnabled:a,options:t})}catch(e){}})))}(),await We(e);try{await browser.runtime.sendMessage({method:"options.refresh",profileName:t.profileName})}catch(e){}}async function jt(e){if(Ce&&Mt){const t=[],a=await i(e.id);if(a[e.id].editorDetected)qt(!1);else if(qt(!0),b&&(t.push(zt(nt,!a[e.id].autoSave)),t.push(zt(rt,a[e.id].autoSave)),t.push(zt(ot,Boolean(a.autoSaveUnpinned))),t.push(zt(st,Boolean(a.autoSaveAll)))),e&&e.url){const n=await N(e.url);t.push(async function(e,t){const a=Ct;Ct=t,(void 0===a||a!=t)&&await Bt(e)}(e,n.contextMenuEnabled)),t.push(Ft(Oe,a[e.id].savedPageDetected?ft:ht)),p&&t.push(_e.update(Je,{visible:!n.saveRawPage})),t.push(_e.update(Oe,{visible:!n.openEditor||a[e.id].savedPageDetected}));let r="associate-with-profile-default",o=it;const[s,i]=await Promise.all([B(),D(e.url)]);if(i){const e=Ot.get(i.profile);e&&(r=Ve+e,o=ct)}Object.keys(s).length>1&&(Object.keys(s).forEach(((e,a)=>{e==d?t.push(zt("associate-with-profile-default","associate-with-profile-default"==r)):t.push(zt(Ve+a,r==Ve+a))})),t.push(Ft(He,o)))}await Promise.all(t)}}async function qt(e){const t=Ut;if(Ut=e,void 0===t||t!=e){const t=[];try{Rt.forEach((a=>t.push(_e.update(a,{visible:e})))),await Promise.all(t)}catch(e){}}}function Ft(e,t){const a=Lt.get(e);return Lt.set(e,t),void 0===a||a!=t?_e.update(e,{title:t}):void 0}async function zt(e,t){t=Boolean(t),Wt.set(e,t),await _e.update(e,{checked:t})}Promise.resolve().then((async function(){Ce&&(Bt(),_e.onClicked.addListener((async(e,t)=>{if(e.menuItemId==Ue&&(e.linkUrl?_t.saveUrls([e.linkUrl]):_t.saveTabs([t])),e.menuItemId==Oe){(await i(t.id))[t.id].savedPageDetected?_t.openEditor(t):e.linkUrl?_t.saveUrls([e.linkUrl],{openEditor:!0}):_t.saveTabs([t],{openEditor:!0})}if(e.menuItemId==Ne&&_t.saveSelectedLinks(t),e.menuItemId==je&&await browser.tabs.create({active:!0,url:"/src/ui/pages/pendings.html"}),e.menuItemId==Je&&_t.saveTabs([t],{selected:!0}),e.menuItemId==Ge&&_t.saveTabs([t],{frameId:e.frameId}),e.menuItemId==Ye||e.menuItemId==Qe){const e=await ie({currentWindow:!0,highlighted:!0});_t.saveTabs(e)}if(e.menuItemId==$e||e.menuItemId==et){const e=await ie({currentWindow:!0,pinned:!1});_t.saveTabs(e)}if(e.menuItemId==Xe||e.menuItemId==tt){const e=await ie({currentWindow:!0});_t.saveTabs(e)}if(e.menuItemId==Ze&&_t.batchSaveUrls(),e.menuItemId==rt){const e=await i(t.id);e[t.id].autoSave=!0,await c(e),Nt(t)}if(e.menuItemId==nt){const e=await i();Object.keys(e).forEach((t=>{"object"==typeof e[t]&&e[t].autoSave&&(e[t].autoSave=!1)})),e.autoSaveUnpinned=e.autoSaveAll=!1,await c(e),Nt(t)}if(e.menuItemId==st){const a=await i();a.autoSaveAll=e.checked,await c(a),Nt(t)}if(e.menuItemId==ot){const a=await i();a.autoSaveUnpinned=e.checked,await c(a),Nt(t)}if(e.menuItemId.startsWith(Fe)){const a=await B(),n=e.menuItemId.split(Fe)[1];let r;if("default"==n)r=d;else{const e=Number(n);r=Object.keys(a)[e]}a[r].profileName=r,_t.saveTabs([t],a[r])}if(e.menuItemId.startsWith(ze)){const[a,n]=await Promise.all([B(),i()]),r=e.menuItemId.split(ze)[1];if("default"==r)n.profileName=d;else{const e=Number(r);n.profileName=Object.keys(a)[e]}await c(n),Nt(t)}if(e.menuItemId.startsWith(Ve)){const[a,n]=await Promise.all([B(),D(t.url,!0)]),r=e.menuItemId.split(Ve)[1];let o;if("default"==r)o=d;else if("current"==r)o=l;else{const e=Number(r);o=Object.keys(a)[e]}n?await q(n.url,n.url,o,o):(await Ft(He,ct),await j(new URL(t.url).hostname,o,o))}})),Mt?Dt=!0:(await browser.tabs.query({})).forEach((async e=>await jt(e))))}));const Ht=browser.commands;let Vt,Jt,Gt;function Kt(e,t){return e.method.endsWith(".refreshMenu")?function(e){if(e.method.endsWith("refreshMenu"))return Bt(),Promise.resolve({})}(e):xe(e,t)}function Yt(e){!function(e){Le(e.id,De("forbidden"))}(e)}function $t(e,t,a){!function(e,t,a){let n;a?n=De("inject",!0):(n=De(1==t?"inject":"execute"),n.setTitle={title:he+" ("+t+"/2)"},n.setIcon={path:le+"0.png"}),Le(e,n)}(e,t,a)}async function Xt(e,t,a){Ee(e);try{t&&await browser.tabs.sendMessage(e,{method:"content.error",error:t.toString(),link:a})}catch(e){}}function Zt(e){!function(e){Le(e,De("edit"))}(e)}function Qt(e,t){Pe(e,t)}function ea(e,t,a){!function(e,t,a){Re(e,t,a,ge)}(e,t,a)}function ta(e){e&&jt(e)}Ht&&Ht.onCommand&&Ht.onCommand.addListener&&Ht.onCommand.addListener((async e=>{if("save-selected-tabs"==e){const e=await ie({currentWindow:!0,highlighted:!0});Vt.saveTabs(e,{optionallySelected:!0})}if("save-all-tabs"==e){const e=await ie({currentWindow:!0});Vt.saveTabs(e)}}));const aa=["lib/chrome-browser-polyfill.js","lib/single-file.js"],na=["lib/chrome-browser-polyfill.js","lib/single-file-frames.js"];async function ra(e,t){let a;if(await async function(e){const t=e.extensionScriptFiles||[];Jt||Gt||([Jt,Gt]=await Promise.all([oa(aa.concat(t)),oa(na)]))}(t),!t.removeFrames)try{await browser.tabs.executeScript(e,{code:Gt,allFrames:!0,matchAboutBlank:!0,runAt:"document_start"})}catch(e){}try{await browser.tabs.executeScript(e,{code:Jt,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 oa(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 sa="single-file-ack-fetch",ia="single-file-response-fetch",ca="Host fetch error (SingleFile)",la=Boolean(window.wrappedJSObject),da=(e,t,a)=>window.addEventListener(e,t,a),ua=(e,t,a)=>window.removeEventListener(e,t,a),fa=(e,t)=>window.fetch(e,t);let ha=0,wa=new Map;async function ma(e,t){const a=new Promise(((a,n)=>{var r;r=new CustomEvent("single-file-request-fetch",{detail:JSON.stringify({url:e,options:t})}),window.dispatchEvent(r),da(sa,i,!1),da(ia,s,!1);const o=setTimeout((()=>{c(),n(new Error(ca))}),2500);function s(t){t.detail?t.detail.url==e&&(c(),t.detail.response?a({status:t.detail.status,headers:new Map(t.detail.headers),arrayBuffer:async()=>t.detail.response}):n(t.detail.error)):n()}function i(){clearTimeout(o)}function c(){ua(ia,s,!1),ua(sa,i,!1)}}));try{return await a}catch(a){if(a&&a.message==ca)return fa(e,t);throw a}}async function ba(e,t={}){try{const a={cache:"force-cache",headers:t.headers};return await(t.referrer&&la?ma(e,a):fa(e,a))}catch(a){ha++;const n=new Promise(((e,t)=>wa.set(ha,{resolve:e,reject:t})));return await ga({method:"singlefile.fetch",url:e,requestId:ha,referrer:t.referrer,headers:t.headers}),n}}async function pa(e,t){const a=await ga({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 ga(e){const t=await browser.runtime.sendMessage(e);if(!t||t.error)throw new Error(t&&t.error&&t.error.toString());return t}function ya(e,t){return ra(e,t)}browser.runtime.onMessage.addListener((e=>"singlefile.fetchFrame"==e.method&&window.frameId&&window.frameId==e.frameId?async function(e){try{const t=await fa(e.url,{cache:"force-cache",headers:e.headers});return{status:t.status,headers:[...t.headers],array:Array.from(new Uint8Array(await t.arrayBuffer()))}}catch(e){return{error:e&&e.toString()}}}(e):"singlefile.fetchResponse"==e.method?async function(e){const t=wa.get(e.requestId);t&&(e.error?(t.reject(new Error(e.error)),wa.delete(e.requestId)):(e.truncated&&(t.array?t.array=t.array.concat(e.array):(t.array=e.array,wa.set(e.requestId,t)),e.finished&&(e.array=t.array)),e.truncated&&!e.finished||(t.resolve({status:e.status,headers:{get:t=>e.headers&&e.headers[t]},arrayBuffer:async()=>new Uint8Array(e.array).buffer}),wa.delete(e.requestId))));return{}}(e):void 0));const va="Could not establish connection. Receiving end does not exist.",ka="The message port closed before a response was received.",Ia="Message manager disconnected",Sa="Cannot access contents of url ",Ta="pending",xa="processing",Ea=["lib/single-file-extension-infobar.js","lib/single-file-extension.js"],Pa=[];let Aa,Ra=0;var Wa;async function La(){return browser.tabs.create({active:!0,url:"/src/ui/pages/batch-save-urls.html"})}async function Ma(e,t={}){await Ca(),await Promise.all(e.map((async e=>{const a=await N(e);Object.keys(t).forEach((e=>a[e]=t[e])),a.autoClose=!0,a.extensionScriptFiles=Ea,a.passReferrerOnError&&await se(),_a({tab:{url:e},status:Ta,options:a,method:"content.save"})}))),Ua()}async function Da(e,t={}){await Ca(),await Promise.all(e.map((async e=>{const a=e.id,n=await N(e.url);Object.keys(t).forEach((e=>n[e]=t[e])),n.tabId=a,n.tabIndex=e.index,n.extensionScriptFiles=Ea,n.passReferrerOnError&&await se();const r={id:e.id,index:e.index,url:e.url,title:e.title};if(t.autoSave){if(V(e)){Oa(_a({status:xa,tab:r,options:n,method:"content.autosave"}))}}else{$t(a,1);await ya(a,n)||X(e)?($t(a,2),_a({status:Ta,tab:r,options:n,method:"content.save"})):Yt(e)}}))),Ua()}function _a(e){const t={id:Ra,status:e.status,tab:e.tab,options:e.options,method:e.method,done:function(){Pa.splice(Pa.findIndex((e=>e.id==this.id)),1),Ua()}};return Pa.push(t),Ra++,t}async function Ca(){Aa||(Aa=(await _()).maxParallelWorkers)}function Ua(){const e=Pa.filter((e=>e.status==xa)).length;for(let t=0;t<Math.min(Pa.length-e,Aa-e);t++){const e=Pa.find((e=>e.status==Ta));e&&Oa(e)}}async function Oa(e){const t=e.id;if(e.status=xa,!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,o){a==t.id&&"complete"==o.status&&(e(t),browser.tabs.onUpdated.removeListener(n),browser.tabs.onRemoved.removeListener(r))}function r(e){e==t.id&&(a(e),browser.tabs.onRemoved.removeListener(r))}browser.tabs.onUpdated.addListener(n),browser.tabs.onRemoved.addListener(r)}))}({url:e.tab.url,active:!1});e.tab.id=e.options.tabId=a.id,e.tab.index=e.options.tabIndex=a.index,$t(e.tab.id,1),t=await ya(e.tab.id,e.options)}catch(t){e.tab.id=t}if(!t)return void e.done();$t(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==ka||e.message==va||e.message==Ia||e.message.startsWith(Sa+JSON.stringify($))}(t)||(console.log(t.message?t.message:t),Xt(e.tab.id,t.message,t.link),e.done())}}function Ba(e){const t=Pa.find((t=>t.id==e));t&&(t.options.autoClose&&!t.cancelled&&browser.tabs.remove(t.tab.id),t.done())}function Na(e,t){const a=Pa.find((t=>t.id==e));a&&(a.cancel=t)}function ja(e){Array.from(Pa).filter((t=>t.tab.id==e&&!t.options.autoSave)).forEach(Fa)}function qa(e){return Pa.find((t=>t.id==e))}function Fa(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&&Qt(t,!0),function(e){Ae(e)}(e.tab),e.done()}function za(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}}Wa={isSavingTab:function(e){return Boolean(Pa.find((t=>t.tab.id==e.id)))},saveTabs:Da,saveUrls:Ma,cancelTab:ja,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 ya(e.id,t)){const t=await browser.tabs.sendMessage(e.id,{method:"content.getSelectedLinks"});if(t.urls&&t.urls.length){const e=await La(),a=(n,r)=>{"complete"==r.status&&n==e.id&&(browser.tabs.onUpdated.removeListener(a),browser.tabs.sendMessage(e.id,{method:"newUrls.addURLs",urls:t.urls}))};browser.tabs.onUpdated.addListener(a)}}else Yt(e)},batchSaveUrls:La},function(e){_t=e}(Wa),function(e){Te=e}(Wa),function(e){Vt=e}(Wa);async function Ha(e){let t;try{t=await browser.runtime.sendNativeMessage("singlefile_companion",{method:"save",pageData:e})}catch(e){if(!e.message||!e.message.includes("Native host has exited"))throw e}if(t&&t.error)throw new Error(t.error+" (Companion)")}const Va=new Set;async function Ja(e){return e.method.endsWith(".saveCreatedBookmarks")?(Ga(),{}):e.method.endsWith(".disable")?(async function(){let e;const t=await B();Object.keys(t).forEach((a=>e=e||!t[a].saveCreatedBookmarks)),e&&(browser.bookmarks.onCreated.removeListener(Ka),browser.bookmarks.onMoved.removeListener(Ya))}(),{}):void 0}async function Ga(){try{browser.bookmarks.onCreated.removeListener(Ka),browser.bookmarks.onMoved.removeListener(Ya)}catch(e){}let e;const t=await B();Object.keys(t).forEach((a=>{t[a].saveCreatedBookmarks&&(e=!0)})),e&&(browser.bookmarks.onCreated.addListener(Ka),browser.bookmarks.onMoved.addListener(Ya))}async function Ka(e,t){Va.add(e),await $a(e,t.url,t)}async function Ya(e,t){if(Va.has(e)){const a=await browser.bookmarks.get(e);a[0]&&await $a(e,a[0].url,t)}}async function $a(e,t,a){const n=await browser.tabs.query({lastFocusedWindow:!0,active:!0}),r=await N(t);if(r.saveCreatedBookmarks){const s=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}(a.parentId),i=r.allowedBookmarkFolders.toString(),c=s.find((e=>r.allowedBookmarkFolders.includes(e))),l=r.ignoredBookmarkFolders.toString(),d=s.find((e=>r.ignoredBookmarkFolders.includes(e)));if((i&&c||!i)&&(l&&!d||!l))if(n.length&&n[0].url==t)Va.delete(e),Da(n,{bookmarkId:e,bookmarkFolders:s});else{const a=await browser.tabs.query({});if(a.length){const n=a.find((e=>e.url==t));n?(Va.delete(e),Da([n],{bookmarkId:e,bookmarkFolders:s})):t&&("about:blank"==t?browser.bookmarks.onChanged.addListener((function t(a,n){a==e&&n.url&&(browser.bookmarks.onChanged.removeListener(t),o(n.url))})):o(t))}}}function o(t){Va.delete(e),Ma([t],{bookmarkId:e})}}Promise.resolve().then(Ga);async function Xa(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 Za="https://oauth2.googleapis.com/token",Qa="https://www.googleapis.com/drive/v3/files";class en{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=cn(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),on(this)}}async function tn(e,t){const a=await fetch(Za,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:"client_id="+e.clientId+"&client_secret="+e.clientKey+"&grant_type=authorization_code&code="+t.code+"&redirect_uri="+browser.identity.getRedirectURL()}),n=await sn(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 an(e={}){return Boolean(browser.identity&&browser.identity.getAuthToken)&&!e.forceWebAuthFlow}async function nn(e,t,a=!0){const n=t.split("/");n.pop();const r=e.folderIds.get(n.join("/"));if(r)return r;let o="root";if(n.length){let r="";for(const s of n){r&&(r+="/"),r+=s;const n=e.folderIds.get(r);if(n)o=n;else try{o=await rn(e,s,o),e.folderIds.set(r,o)}catch(n){if("path_not_found"==n.message&&a)return e.folderIds.clear(),nn(e,t,!1);throw n}}}return o}async function rn(e,t,a){const n=await async function(e,t,a){return sn(await fetch(Qa+"?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 sn(await fetch(Qa,{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 on(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 on(e)}cn(n)}async function sn(e){e=cn(e);const t=await e.json();if(t.error)throw new Error(t.error);return t}function cn(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+")")}let ln;async function dn(e,t,a,n,r,o){for(;ln;)await ln;const s=new AbortController;return ln=(async()=>{try{await async function({path:r,content:o,message:s=""},i){try{const c=await fetch(`https://api.github.com/repos/${t}/${a}/contents/${r}`,{method:"PUT",headers:new Map([["Authorization",`token ${e}`],["Accept","application/vnd.github.v3+json"]]),body:JSON.stringify({content:btoa(unescape(encodeURIComponent(o))),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:r,content:o},s.signal)}finally{ln=null}})(),{cancelPush:()=>s.abort(),pushPromise:ln}}const un=new Map,fn="text/html",hn=/([{}()^$&.*?/+|[\\\\]|\]|-)/g,wn=new class{constructor(e,t,a){this.clientId=e,this.clientKey=t,this.scopes=a,this.folderIds=new Map,setInterval((()=>this.folderIds.clear()),6e4)}async auth(e={interactive:!0}){return an(e)?(this.accessToken=await browser.identity.getAuthToken({interactive:e.interactive}),{revokableAccessToken:this.accessToken}):(this.authURL="https://accounts.google.com/o/oauth2/v2/auth?client_id="+this.clientId+"&response_type=code&access_type=offline&redirect_uri="+browser.identity.getRedirectURL()+"&scope="+this.scopes.join(" "),e.code?tn(this,e):async function(e,t){let a;try{if(browser.identity&&browser.identity.launchWebAuthFlow&&!t.forceWebAuthFlow){const a=await browser.identity.launchWebAuthFlow({interactive:t.interactive,url:e.authURL});return t.code=new URLSearchParams(new URL(a).search).get("code"),await tn(e,t)}if(t.launchWebAuthFlow)return t.extractAuthCode(browser.identity.getRedirectURL()).then((e=>a=e)).catch((()=>{})),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(a)return t.code=a,await tn(e,t);throw new Error("code_required")}throw n}}(this,e))}setAuthInfo(e,t){an(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(Za,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:"client_id="+this.clientId+"&refresh_token="+this.refreshToken+"&grant_type=refresh_token&client_secret="+this.clientKey});if(400==e.status)throw new Error("unknown_token");const t=await sn(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}}try{return browser.identity&&browser.identity.removeCachedAuthToken&&this.accessToken&&await browser.identity.removeCachedAuthToken({token:this.accessToken}),this.accessToken=await browser.identity.getAuthToken({interactive:!1}),{revokableAccessToken:this.accessToken}}catch(e){delete this.accessToken}}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 sn(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,r=!0){const o=await nn(this,e),s=e.split("/").pop(),i=new en({token:this.accessToken,file:t,parents:[o],filename:s,onProgress:a.onProgress});try{n&&n((()=>i.cancelled=!0)),await i.upload()}catch(o){if("path_not_found"==o.message&&r)return this.folderIds.clear(),this.upload(e,t,a,n);throw o}}}("207618107333-h1220p1oasj3050kr5r416661adm091a.apps.googleusercontent.com","VQJ8Gq8Vxx72QyxPyeLtWvUt",["https://www.googleapis.com/auth/drive.file"]);async function mn(e,t){if(e.method.endsWith(".download"))return async function(e,t){let a;e.truncated?(a=un.get(t.id),a||(a=[],un.set(t.id,a)),a.push(e.content),e.finished&&un.delete(t.id)):e.content&&(a=[e.content]);e.truncated&&!e.finished||(e.openEditor?(Zt(t.id),await async function({tabIndex:e,content:t,filename:a}){const n={active:!0,url:G};null!=e&&(n.index=e);const r=await browser.tabs.create(n);K.set(r.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(fn,e.content),t.clipboardData.setData("text/plain",e.content),t.preventDefault()}document.addEventListener(t,a),document.execCommand(t),document.removeEventListener(t,a)}(e),Qt(t.id)):await async function(e,t,a,n){try{if(n.saveWithWebDAV?await yn(n.taskId,bn(n.filename),e.join(""),n.webDAVURL,n.webDAVUser,n.webDAVPassword):n.saveToGDrive?await vn(n.taskId,bn(n.filename),new Blob(e,{type:fn}),{forceWebAuthFlow:n.forceWebAuthFlow},{onProgress:(e,a)=>ea(t.id,e,a)}):n.saveToGitHub?await(await gn(n.taskId,bn(n.filename),e.join(""),n.githubToken,n.githubUser,n.githubRepository,n.githubBranch)).pushPromise:n.saveWithCompanion?await Ha({filename:n.filename,content:n.content,filenameConflictAction:n.filenameConflictAction}):(n.url=URL.createObjectURL(new Blob(e,{type:fn})),await kn(n,{confirmFilename:n.confirmFilename,incognito:a,filenameConflictAction:n.filenameConflictAction,filenameReplacementCharacter:n.filenameReplacementCharacter,includeInfobar:n.includeInfobar})),Qt(t.id),n.openSavedPage){const a={active:!0,url:URL.createObjectURL(new Blob(e,{type:fn}))};null!=t.index&&(a.index=t.index+1),browser.tabs.create(a)}}catch(e){e.message&&"upload_cancelled"==e.message||(console.error(e),Xt(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 F();return H(),await wn.revokeAuthToken(e&&(e.accessToken||e.revokableAccessToken)),{}}if(e.method.endsWith(".end")){if(e.hash)try{await Xa(e.hash,e.woleetKey)}catch(e){Xt(t.tab.id,e.message,e.link)}return Ba(e.taskId),{}}return e.method.endsWith(".getInfo")?Pa.map(za):e.method.endsWith(".cancel")?(a=e.taskId,Fa(Pa.find((e=>e.id==a))),{}):e.method.endsWith(".cancelAll")?(Array.from(Pa).forEach(Fa),{}):e.method.endsWith(".saveUrls")?(Ma(e.urls),{}):void 0;var a}function bn(e){return e.replace(/#/g,"%23")}async function pn(e,t){let a=await F();const n={interactive:!0,forceWebAuthFlow:e.forceWebAuthFlow,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)=>{browser.tabs.onUpdated.addListener((function n(r,o){if(o&&o.url.startsWith(e)){browser.tabs.onUpdated.removeListener(n);const e=new URLSearchParams(new URL(o.url).search).get("code");e?(browser.tabs.remove(r),t(e)):a()}}))}))}(e)};return wn.setAuthInfo(a,n),a&&a.accessToken&&!t||(a=await wn.auth(n),a?await z(a):await H()),a}async function gn(e,t,a,n,r,o,s){const i=qa(e);if(!i||!i.cancelled){const i=dn(n,r,o,s,t,a);Na(e,i.cancelPush);try{return await(await i).pushPromise,i}catch(e){throw new Error(e.message+" (GitHub)")}}}async function yn(e,t,a,n,r,o){const s=qa(e),i=new AbortController,{signal:c}=i,l="Basic "+btoa(r+":"+o);if(n.endsWith("/")||(n+="/"),!s||!s.cancelled){Na(e,(()=>i.abort()));try{const s=await d(n+t,"PUT",a);if(404==s.status&&t.includes("/")){const s=t.split(/\/+/);s.pop();let i="";for(const e of s)if(e){i+=e;if(404==(await d(n+i,"PROPFIND")).status){const e=await d(n+i,"MKCOL");if(e.status>=400)throw new Error("Error "+e.status+" (WebDAV)")}i+="/"}return yn(e,t,a,n,r,o)}if(s.status>=400)throw new Error("Error "+s.status+" (WebDAV)");return s}catch(e){if("AbortError"!=e.name)throw new Error(e.message+" (WebDAV)")}}function d(e,t,a){const n={Authorization:l};return a&&(n["Content-Type"]="text/html"),fetch(e,{method:t,headers:n,signal:c,body:a,credentials:"omit"})}}async function vn(e,t,a,n,r){try{await pn(n);const o=qa(e);if(!o||!o.cancelled)return wn.upload(t,a,r,(t=>Na(e,t)))}catch(o){if("invalid_token"==o.message){let o;try{o=await wn.refreshAuthToken()}catch(e){if("unknown_token"!=e.message)throw new Error(e.message+" (Google Drive)");o=await pn(n,!0)}return o?await z(o):await H(),await vn(e,t,a,n,r)}throw new Error(o.message+" (Google Drive)")}}async function kn(t,a){let n;if("skip"==a.filenameConflictAction){(await browser.downloads.search({filenameRegex:"(\\\\|/)"+(r=t.filename,r.replace(hn,"\\$1")+"$"),exists:!0})).length?n=!0:a.filenameConflictAction="uniquify"}var r;if(!n){const n={url:t.url,saveAs:a.confirmFilename,filename:t.filename,conflictAction:a.filenameConflictAction};a.incognito&&(n.incognito=!0);const r=await e(n,a.filenameReplacementCharacter);r.filename&&t.bookmarkId&&t.replaceBookmarkURL&&(r.filename.startsWith("file:")||(r.filename.startsWith("/")&&(r.filename=r.filename.substring(1)),r.filename="file:///"+bn(r.filename)),await async function(e,t){try{await browser.bookmarks.update(e,t)}catch(e){}}(t.bookmarkId,{url:r.filename}))}}const In={},Sn={};async function Tn(e,t){if("enableAutoSave"==e.method){const a=await i(t.id);a[t.id].autoSave=e.enabled,await c(a),async function(e){Promise.all([Bt(e),We(e)])}(t)}if("isAutoSaveEnabled"==e.method)return V(t)}async function xn(e,t){const a=t.id,n=await N(t.url,!0);if(n){let r;$t(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.videos=e.videos,n.usedFonts=e.usedFonts,n.shadowRoots=e.shadowRoots,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,n.keepFilename=n.saveToGDrive||n.saveToGitHub||n.saveWithWebDAV;try{if(n.autoSaveExternalSave)await async function(e){let t;e.autoSaveExternalSave=!1;try{t=await browser.runtime.sendNativeMessage("singlefile_companion",{method:"externalSave",pageData:e})}catch(e){if(!e.message||!e.message.includes("Native host has exited"))throw e}if(t&&t.error)throw new Error(t.error+" (Companion)")}(n);else{if(r=await function(e,t,a,n={fetch:ba,frameFetch:pa}){return globalThis.singlefile.getPageData(e,n,t,a)}(n,null,null,{fetch:En}),n.includeInfobar&&(r.content+=await infobar.getScript()),n.saveToGDrive){const t=new Blob([r.content],{type:"text/html"});await vn(e.taskId,bn(r.filename),t,n,{forceWebAuthFlow:n.forceWebAuthFlow})}else if(n.saveWithWebDAV)await yn(e.taskId,bn(r.filename),r.content,n.webDAVURL,n.webDAVUser,n.webDAVPassword);else if(n.saveToGitHub)await(await gn(e.taskId,bn(r.filename),r.content,n.githubToken,n.githubUser,n.githubRepository,n.githubBranch)).pushPromise;else if(n.saveWithCompanion)await Ha({filename:r.filename,content:r.content,filenameConflictAction:r.filenameConflictAction});else{const e=new Blob([r.content],{type:"text/html"});if(r.url=URL.createObjectURL(e),await kn(r,n),n.openSavedPage){const n={active:!0,url:URL.createObjectURL(e),windowId:t.windowId},r=t.index;try{await browser.tabs.get(a),n.index=r+1}catch(e){n.index=r}browser.tabs.create(n)}}r.hash&&await Xa(r.hash,n.woleetKey)}}finally{e.taskId?Ba(e.taskId):n.autoClose&&(browser.tabs.remove(Sn[a]||a),delete Sn[a]),r&&r.url&&URL.revokeObjectURL(r.url),Qt(a,!0)}}}function En(e,t={}){return new Promise(((a,n)=>{const r=new XMLHttpRequest;if(r.withCredentials=!0,r.responseType="arraybuffer",r.onerror=e=>n(new Error(e.detail)),r.onreadystatechange=()=>{r.readyState==XMLHttpRequest.DONE&&a({status:r.status,headers:{get:e=>r.getResponseHeader(e)},arrayBuffer:async()=>r.response})},r.open("GET",e,!0),t.headers)for(const e of Object.entries(t.headers))r.setRequestHeader(e[0],e[1]);r.send()}))}async function Pn(e,t){if(e.method.endsWith(".init")&&(await async function(e,t){await o(e.id);const a=await i(e.id);a[e.id].savedPageDetected=t.savedPageDetected,await c(a)}(t.tab,e),jt(t.tab),function(e){ja(e.id)}(t.tab),async function(e){const[t,a]=await Promise.all([N(e.url,!0),V(e)]);t&&(t.autoSaveLoad||t.autoSaveLoadOrUnload)&&a&&Da([e],{autoSave:!0})}(t.tab)),e.method.endsWith(".getOptions"))return N(e.url);e.method.endsWith(".activate")&&await browser.tabs.update(e.tabId,{active:!0})}browser.tabs.onCreated.addListener((e=>function(e){!function(e){jt(e)}(e)}(e))),browser.tabs.onActivated.addListener((e=>async function(e){ta(await browser.tabs.get(e.tabId))}(e))),browser.tabs.onRemoved.addListener((e=>function(e){o(e),function(e){K.delete(e)}(e),ja(e),async function(e){const t=In[e];t?t.autoSaveRemove&&(delete In[e],await xn(t,t.tab)):In[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 In[e]}(e);const t=await browser.tabs.get(e);if(X(t)){const e=await i(t.id);e[t.id].editorDetected=!0,await c(e),ta(t)}}t.discarded&&async function(e){const t=In[e];t?(delete In[e],await xn(t,t.tab)):In[e]={discarded:!0}}(e)}(e,t))),browser.tabs.onReplaced.addListener(((e,t)=>function(e,t){(async function(e,t){let n=await i();await r(n,e,t),c(n),await r(a,e,t)})(e,t),async function(e,t){In[t]&&!In[e]&&(In[e]=In[t],delete In[t],Sn[t]=e)}(e,t),function(e,t){Pa.forEach((a=>{a.tab.id==t&&(a.tab.id=e)}))}(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 An=new Map;function Rn(e,t){e.delete(t)}browser.runtime.onMessage.addListener(((e,t)=>{if("singlefile.lazyTimeout.setTimeout"==e.method){let a,n=An.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 r=setTimeout((async()=>{try{const a=An.get(t.tab.id),n=a.get(t.frameId);a&&n&&Rn(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),An.set(t.tab.id,n)),a.set(e.type,r),Promise.resolve({})}if("singlefile.lazyTimeout.clearTimeout"==e.method){let a=An.get(t.tab.id);if(a){const n=a.get(t.frameId);if(n){const t=n.get(e.type);t&&clearTimeout(t),Rn(n,e.type)}}return Promise.resolve({})}})),browser.tabs.onRemoved.addListener((e=>An.delete(e))),browser.runtime.onMessage.addListener(((e,t)=>e.method.startsWith("tabs.")?Pn(e,t):e.method.startsWith("downloads.")?mn(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,In[t.tab.id]=e):In[e.tabId]&&(In[e.tabId].removed&&e.autoSaveRemove||In[e.tabId].discarded&&e.autoSaveDiscard)&&(delete In[e.tabId],await xn(e,{id:e.tabId,index:e.tabIndex,url:t.url})),e.autoSaveUnload&&(delete In[e.tabId],await xn(e,t.tab))):(delete In[e.tabId],await xn(e,t.tab)),{}}(e,t):e.method.startsWith("ui.")?Kt(e,t):e.method.startsWith("config.")?O(e):e.method.startsWith("tabsData.")?function(e){return e.method.endsWith(".get")?i():e.method.endsWith(".set")?c(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=K.get(e.id);if(a){const t=await N(a.url),n=JSON.stringify(a);for(let a=0;a*J<n.length;a++){const r={method:"editor.setTabData"};r.truncated=n.length>J,r.truncated?(r.finished=(a+1)*J>n.length,r.content=n.substring(a*J,(a+1)*J),r.finished&&(r.options=t)):(r.content=n,r.options=t),await browser.tabs.sendMessage(e.id,r)}}return{}}if(e.method.endsWith(".open")){let a;const n=t.tab;if(e.truncated?(a=Y.get(n.id),a||(a=[],Y.set(n.id,a)),a.push(e.content),e.finished&&Y.delete(n.id)):e.content&&(a=[e.content]),!e.truncated||e.finished){const t={url:G};await browser.tabs.update(n.id,t),K.set(n.id,{url:n.url,content:a.join(""),filename:e.filename})}return{}}if(e.method.endsWith(".ping"))return{}}(e,t):e.method.startsWith("bookmarks.")?Ja(e):e.method.startsWith("companion.")?async function(e){if(e.method.endsWith(".state"))return{enabled:!0}}(e):e.method.startsWith("requests.")?re(e):e.method.startsWith("bootstrap.")?async function(e,t){if(e.method.endsWith(".init")){const[e,a,n]=await Promise.all([N(t.tab.url,!0),N(t.tab.url),V(t.tab)]);return{optionsAutoSave:e,options:a,autoSaveEnabled:n,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&&Tn(e,a)}))}();