| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507 |
- (function () {
- 'use strict';
- /* global TextEncoder, TextDecoder */
- const DEFAULT_CHUNK_SIZE = 8 * 1024 * 1024;
- const TYPE_REFERENCE = 0;
- const SPECIAL_TYPES = [TYPE_REFERENCE];
- const EMPTY_SLOT_VALUE = Symbol();
- const textEncoder = new TextEncoder();
- const textDecoder = new TextDecoder();
- const types = new Array(256);
- let typeIndex = 0;
- registerType(serializeCircularReference, parseCircularReference, testCircularReference, TYPE_REFERENCE);
- registerType(null, parseObject, testObject);
- registerType(serializeArray, parseArray, testArray);
- registerType(serializeString, parseString, testString);
- registerType(serializeTypedArray, parseFloat64Array, testFloat64Array);
- registerType(serializeTypedArray, parseFloat32Array, testFloat32Array);
- registerType(serializeTypedArray, parseUint32Array, testUint32Array);
- registerType(serializeTypedArray, parseInt32Array, testInt32Array);
- registerType(serializeTypedArray, parseUint16Array, testUint16Array);
- registerType(serializeTypedArray, parseInt16Array, testInt16Array);
- registerType(serializeTypedArray, parseUint8ClampedArray, testUint8ClampedArray);
- registerType(serializeTypedArray, parseUint8Array, testUint8Array);
- registerType(serializeTypedArray, parseInt8Array, testInt8Array);
- registerType(serializeArrayBuffer, parseArrayBuffer, testArrayBuffer);
- registerType(serializeNumber, parseNumber, testNumber);
- registerType(serializeUint32, parseUint32, testUint32);
- registerType(serializeInt32, parseInt32, testInt32);
- registerType(serializeUint16, parseUint16, testUint16);
- registerType(serializeInt16, parseInt16, testInt16);
- registerType(serializeUint8, parseUint8, testUint8);
- registerType(serializeInt8, parseInt8, testInt8);
- registerType(null, parseUndefined, testUndefined);
- registerType(null, parseNull, testNull);
- registerType(null, parseNaN, testNaN);
- registerType(serializeBoolean, parseBoolean, testBoolean);
- registerType(serializeSymbol, parseSymbol, testSymbol);
- registerType(null, parseEmptySlot, testEmptySlot);
- registerType(serializeMap, parseMap, testMap);
- registerType(serializeSet, parseSet, testSet);
- registerType(serializeDate, parseDate, testDate);
- registerType(serializeError, parseError, testError);
- registerType(serializeRegExp, parseRegExp, testRegExp);
- registerType(serializeStringObject, parseStringObject, testStringObject);
- registerType(serializeNumberObject, parseNumberObject, testNumberObject);
- registerType(serializeBooleanObject, parseBooleanObject, testBooleanObject);
- function registerType(serialize, parse, test, type) {
- if (type === undefined) {
- typeIndex++;
- if (types.length - typeIndex >= SPECIAL_TYPES.length) {
- types[types.length - typeIndex] = { serialize, parse, test };
- } else {
- throw new Error("Reached maximum number of custom types");
- }
- } else {
- types[type] = { serialize, parse, test };
- }
- }
- async function serialize(object, options) {
- const serializer = getSerializer(object, options);
- let result = new Uint8Array([]);
- for await (const chunk of serializer) {
- const previousResult = result;
- result = new Uint8Array(previousResult.length + chunk.length);
- result.set(previousResult, 0);
- result.set(chunk, previousResult.length);
- }
- return result;
- }
- class SerializerData {
- constructor(appendData, chunkSize) {
- this.stream = new WriteStream(appendData, chunkSize);
- this.objects = [];
- }
- append(array) {
- return this.stream.append(array);
- }
- flush() {
- return this.stream.flush();
- }
- addObject(value) {
- this.objects.push(testReferenceable(value) && !testCircularReference(value, this) ? value : undefined);
- }
- }
- class WriteStream {
- constructor(appendData, chunkSize) {
- this.offset = 0;
- this.appendData = appendData;
- this.value = new Uint8Array(chunkSize);
- }
- async append(array) {
- if (this.offset + array.length > this.value.length) {
- const offset = this.value.length - this.offset;
- await this.append(array.subarray(0, offset));
- await this.appendData({ value: this.value });
- this.offset = 0;
- await this.append(array.subarray(offset));
- } else {
- this.value.set(array, this.offset);
- this.offset += array.length;
- }
- }
- async flush() {
- if (this.offset) {
- await this.appendData({ value: this.value.subarray(0, this.offset), done: true });
- }
- }
- }
- function getSerializer(value, { chunkSize = DEFAULT_CHUNK_SIZE } = {}) {
- let serializerData, result, setResult, iterationDone, previousResult, resolvePreviousResult;
- return {
- [Symbol.asyncIterator]() {
- return {
- next() {
- return iterationDone ? { done: iterationDone } : getResult();
- },
- return() {
- return { done: true };
- }
- };
- }
- };
- async function getResult() {
- if (resolvePreviousResult) {
- resolvePreviousResult();
- } else {
- initSerializerData().catch(() => { /* ignored */ });
- }
- initPreviousData();
- const value = await getValue();
- return { value };
- }
- async function initSerializerData() {
- initResult();
- serializerData = new SerializerData(appendData, chunkSize);
- await serializeValue(serializerData, value);
- await serializerData.flush();
- }
- function initResult() {
- result = new Promise(resolve => setResult = resolve);
- }
- function initPreviousData() {
- previousResult = new Promise(resolve => resolvePreviousResult = resolve);
- }
- async function appendData(result) {
- setResult(result);
- await previousResult;
- }
- async function getValue() {
- const { value, done } = await result;
- iterationDone = done;
- if (!done) {
- initResult();
- }
- return value;
- }
- }
- async function serializeValue(data, value) {
- const type = types.findIndex(({ test } = {}) => test && test(value, data));
- data.addObject(value);
- await data.append(new Uint8Array([type]));
- const serialize = types[type].serialize;
- if (serialize) {
- await serialize(data, value);
- }
- if (type != TYPE_REFERENCE && testObject(value)) {
- await serializeSymbols(data, value);
- await serializeOwnProperties(data, value);
- }
- }
- async function serializeSymbols(data, value) {
- const ownPropertySymbols = Object.getOwnPropertySymbols(value);
- const symbols = ownPropertySymbols.map(propertySymbol => [propertySymbol, value[propertySymbol]]);
- await serializeArray(data, symbols);
- }
- async function serializeOwnProperties(data, value) {
- if (!ArrayBuffer.isView(value)) {
- let entries = Object.entries(value);
- if (testArray(value)) {
- entries = entries.filter(([key]) => !testInteger(Number(key)));
- }
- await serializeValue(data, entries.length);
- for (const [key, value] of entries) {
- await serializeString(data, key);
- await serializeValue(data, value);
- }
- } else {
- await serializeValue(data, 0);
- }
- }
- async function serializeCircularReference(data, value) {
- const index = data.objects.indexOf(value);
- await serializeValue(data, index);
- }
- async function serializeArray(data, array) {
- await serializeValue(data, array.length);
- const notEmptyIndexes = Object.keys(array).filter(key => testInteger(Number(key))).map(key => Number(key));
- let indexNotEmptyIndexes = 0, currentNotEmptyIndex = notEmptyIndexes[indexNotEmptyIndexes];
- for (const [indexArray, value] of array.entries()) {
- if (currentNotEmptyIndex == indexArray) {
- currentNotEmptyIndex = notEmptyIndexes[++indexNotEmptyIndexes];
- await serializeValue(data, value);
- } else {
- await serializeValue(data, EMPTY_SLOT_VALUE);
- }
- }
- }
- async function serializeString(data, string) {
- const encodedString = textEncoder.encode(string);
- await serializeValue(data, encodedString.length);
- await data.append(encodedString);
- }
- async function serializeTypedArray(data, array) {
- await serializeValue(data, array.length);
- await data.append(array.constructor.name == "Uint8Array" ? array : new Uint8Array(array.buffer));
- }
- async function serializeArrayBuffer(data, arrayBuffer) {
- await serializeValue(data, arrayBuffer.byteLength);
- await data.append(new Uint8Array(arrayBuffer));
- }
- async function serializeNumber(data, number) {
- const serializedNumber = new Uint8Array(new Float64Array([number]).buffer);
- await data.append(serializedNumber);
- }
- async function serializeUint32(data, number) {
- const serializedNumber = new Uint8Array(new Uint32Array([number]).buffer);
- await data.append(serializedNumber);
- }
- async function serializeInt32(data, number) {
- const serializedNumber = new Uint8Array(new Int32Array([number]).buffer);
- await data.append(serializedNumber);
- }
- async function serializeUint16(data, number) {
- const serializedNumber = new Uint8Array(new Uint16Array([number]).buffer);
- await data.append(serializedNumber);
- }
- async function serializeInt16(data, number) {
- const serializedNumber = new Uint8Array(new Int16Array([number]).buffer);
- await data.append(serializedNumber);
- }
- async function serializeUint8(data, number) {
- const serializedNumber = new Uint8Array([number]);
- await data.append(serializedNumber);
- }
- async function serializeInt8(data, number) {
- const serializedNumber = new Uint8Array(new Int8Array([number]).buffer);
- await data.append(serializedNumber);
- }
- async function serializeBoolean(data, boolean) {
- const serializedBoolean = new Uint8Array([Number(boolean)]);
- await data.append(serializedBoolean);
- }
- async function serializeMap(data, map) {
- const entries = map.entries();
- await serializeValue(data, map.size);
- for (const [key, value] of entries) {
- await serializeValue(data, key);
- await serializeValue(data, value);
- }
- }
- async function serializeSet(data, set) {
- await serializeValue(data, set.size);
- for (const value of set) {
- await serializeValue(data, value);
- }
- }
- async function serializeDate(data, date) {
- await serializeNumber(data, date.getTime());
- }
- async function serializeError(data, error) {
- await serializeString(data, error.message);
- await serializeString(data, error.stack);
- }
- async function serializeRegExp(data, regExp) {
- await serializeString(data, regExp.source);
- await serializeString(data, regExp.flags);
- }
- async function serializeStringObject(data, string) {
- await serializeString(data, string.valueOf());
- }
- async function serializeNumberObject(data, number) {
- await serializeNumber(data, number.valueOf());
- }
- async function serializeBooleanObject(data, boolean) {
- await serializeBoolean(data, boolean.valueOf());
- }
- async function serializeSymbol(data, symbol) {
- await serializeString(data, symbol.description);
- }
- class Reference {
- constructor(index, data) {
- this.index = index;
- this.data = data;
- }
- getObject() {
- return this.data.objects[this.index];
- }
- }
- class ParserData {
- constructor(consumeData) {
- this.stream = new ReadStream(consumeData);
- this.objects = [];
- this.setters = [];
- }
- consume(size) {
- return this.stream.consume(size);
- }
- getObjectId() {
- const objectIndex = this.objects.length;
- this.objects.push(undefined);
- return objectIndex;
- }
- resolveObject(objectId, value) {
- if (testReferenceable(value) && !testReference(value)) {
- this.objects[objectId] = value;
- }
- }
- setObject(functionArguments, setterFunction) {
- this.setters.push({ functionArguments, setterFunction });
- }
- executeSetters() {
- this.setters.forEach(({ functionArguments, setterFunction }) => {
- const resolvedArguments = functionArguments.map(argument => testReference(argument) ? argument.getObject() : argument);
- setterFunction(...resolvedArguments);
- });
- }
- }
- class ReadStream {
- constructor(consumeData) {
- this.offset = 0;
- this.value = new Uint8Array(0);
- this.consumeData = consumeData;
- }
- async consume(size) {
- if (this.offset + size > this.value.length) {
- const pending = this.value.subarray(this.offset, this.value.length);
- const value = await this.consumeData();
- if (pending.length + value.length != this.value.length) {
- this.value = new Uint8Array(pending.length + value.length);
- }
- this.value.set(pending);
- this.value.set(value, pending.length);
- this.offset = 0;
- return this.consume(size);
- } else {
- const result = this.value.slice(this.offset, this.offset + size);
- this.offset += result.length;
- return result;
- }
- }
- }
- function getParser() {
- let parserData, input, setInput, value, previousData, resolvePreviousData;
- return {
- async next(input) {
- return input ? getResult(input) : { value: await value, done: true };
- },
- return() {
- return { done: true };
- }
- };
- async function getResult(input) {
- if (previousData) {
- await previousData;
- } else {
- initParserData().catch(() => { /* ignored */ });
- }
- initPreviousData();
- setInput(input);
- return { done: false };
- }
- async function initParserData() {
- let setValue;
- value = new Promise(resolve => setValue = resolve);
- parserData = new ParserData(consumeData);
- initChunk();
- const data = await parseValue(parserData);
- parserData.executeSetters();
- setValue(data);
- }
- function initChunk() {
- input = new Promise(resolve => setInput = resolve);
- }
- function initPreviousData() {
- previousData = new Promise(resolve => resolvePreviousData = resolve);
- }
- async function consumeData() {
- const data = await input;
- initChunk();
- if (resolvePreviousData) {
- resolvePreviousData();
- }
- return data;
- }
- }
- async function parseValue(data) {
- const array = await data.consume(1);
- const parserType = array[0];
- const parse = types[parserType].parse;
- const valueId = data.getObjectId();
- const result = await parse(data);
- if (parserType != TYPE_REFERENCE && testObject(result)) {
- await parseSymbols(data, result);
- await parseOwnProperties(data, result);
- }
- data.resolveObject(valueId, result);
- return result;
- }
- async function parseSymbols(data, value) {
- const symbols = await parseArray(data);
- data.setObject([symbols], symbols => symbols.forEach(([symbol, propertyValue]) => value[symbol] = propertyValue));
- }
- async function parseOwnProperties(data, object) {
- const size = await parseValue(data);
- if (size) {
- await parseNextProperty();
- }
- async function parseNextProperty(indexKey = 0) {
- const key = await parseString(data);
- const value = await parseValue(data);
- data.setObject([value], value => object[key] = value);
- if (indexKey < size - 1) {
- await parseNextProperty(indexKey + 1);
- }
- }
- }
- async function parseCircularReference(data) {
- const index = await parseValue(data);
- const result = new Reference(index, data);
- return result;
- }
- function parseObject() {
- return {};
- }
- async function parseArray(data) {
- const length = await parseValue(data);
- const array = new Array(length);
- if (length) {
- await parseNextSlot();
- }
- return array;
- async function parseNextSlot(indexArray = 0) {
- const value = await parseValue(data);
- if (!testEmptySlot(value)) {
- data.setObject([value], value => array[indexArray] = value);
- }
- if (indexArray < length - 1) {
- await parseNextSlot(indexArray + 1);
- }
- }
- }
- function parseEmptySlot() {
- return EMPTY_SLOT_VALUE;
- }
- async function parseString(data) {
- const size = await parseValue(data);
- const array = await data.consume(size);
- return textDecoder.decode(array);
- }
- async function parseFloat64Array(data) {
- const length = await parseValue(data);
- const array = await data.consume(length * 8);
- return new Float64Array(array.buffer);
- }
- async function parseFloat32Array(data) {
- const length = await parseValue(data);
- const array = await data.consume(length * 4);
- return new Float32Array(array.buffer);
- }
- async function parseUint32Array(data) {
- const length = await parseValue(data);
- const array = await data.consume(length * 4);
- return new Uint32Array(array.buffer);
- }
- async function parseInt32Array(data) {
- const length = await parseValue(data);
- const array = await data.consume(length * 4);
- return new Int32Array(array.buffer);
- }
- async function parseUint16Array(data) {
- const length = await parseValue(data);
- const array = await data.consume(length * 2);
- return new Uint16Array(array.buffer);
- }
- async function parseInt16Array(data) {
- const length = await parseValue(data);
- const array = await data.consume(length * 2);
- return new Int16Array(array.buffer);
- }
- async function parseUint8ClampedArray(data) {
- const length = await parseValue(data);
- const array = await data.consume(length);
- return new Uint8ClampedArray(array.buffer);
- }
- async function parseUint8Array(data) {
- const length = await parseValue(data);
- const array = await data.consume(length);
- return array;
- }
- async function parseInt8Array(data) {
- const length = await parseValue(data);
- const array = await data.consume(length);
- return new Int8Array(array.buffer);
- }
- async function parseArrayBuffer(data) {
- const length = await parseValue(data);
- const array = await data.consume(length);
- return array.buffer;
- }
- async function parseNumber(data) {
- const array = await data.consume(8);
- return new Float64Array(array.buffer)[0];
- }
- async function parseUint32(data) {
- const array = await data.consume(4);
- return new Uint32Array(array.buffer)[0];
- }
- async function parseInt32(data) {
- const array = await data.consume(4);
- return new Int32Array(array.buffer)[0];
- }
- async function parseUint16(data) {
- const array = await data.consume(2);
- return new Uint16Array(array.buffer)[0];
- }
- async function parseInt16(data) {
- const array = await data.consume(2);
- return new Int16Array(array.buffer)[0];
- }
- async function parseUint8(data) {
- const array = await data.consume(1);
- return new Uint8Array(array.buffer)[0];
- }
- async function parseInt8(data) {
- const array = await data.consume(1);
- return new Int8Array(array.buffer)[0];
- }
- function parseUndefined() {
- return undefined;
- }
- function parseNull() {
- return null;
- }
- function parseNaN() {
- return NaN;
- }
- async function parseBoolean(data) {
- const array = await data.consume(1);
- return Boolean(array[0]);
- }
- async function parseMap(data) {
- const size = await parseValue(data);
- const map = new Map();
- if (size) {
- await parseNextEntry();
- }
- return map;
- async function parseNextEntry(indexKey = 0) {
- const key = await parseValue(data);
- const value = await parseValue(data);
- data.setObject([key, value], (key, value) => map.set(key, value));
- if (indexKey < size - 1) {
- await parseNextEntry(indexKey + 1);
- }
- }
- }
- async function parseSet(data) {
- const size = await parseValue(data);
- const set = new Set();
- if (size) {
- await parseNextEntry();
- }
- return set;
- async function parseNextEntry(indexKey = 0) {
- const value = await parseValue(data);
- data.setObject([value], value => set.add(value));
- if (indexKey < size - 1) {
- await parseNextEntry(indexKey + 1);
- }
- }
- }
- async function parseDate(data) {
- const milliseconds = await parseNumber(data);
- return new Date(milliseconds);
- }
- async function parseError(data) {
- const message = await parseString(data);
- const stack = await parseString(data);
- const error = new Error(message);
- error.stack = stack;
- return error;
- }
- async function parseRegExp(data) {
- const source = await parseString(data);
- const flags = await parseString(data);
- return new RegExp(source, flags);
- }
- async function parseStringObject(data) {
- return new String(await parseString(data));
- }
- async function parseNumberObject(data) {
- return new Number(await parseNumber(data));
- }
- async function parseBooleanObject(data) {
- return new Boolean(await parseBoolean(data));
- }
- async function parseSymbol(data) {
- const description = await parseString(data);
- return Symbol(description);
- }
- function testCircularReference(value, data) {
- return testObject(value) && data.objects.includes(value);
- }
- function testReference(value) {
- return value instanceof Reference;
- }
- function testObject(value) {
- return value === Object(value);
- }
- function testArray(value) {
- return typeof value.length == "number";
- }
- function testEmptySlot(value) {
- return value === EMPTY_SLOT_VALUE;
- }
- function testString(value) {
- return typeof value == "string";
- }
- function testFloat64Array(value) {
- return value.constructor.name == "Float64Array";
- }
- function testUint32Array(value) {
- return value.constructor.name == "Uint32Array";
- }
- function testInt32Array(value) {
- return value.constructor.name == "Int32Array";
- }
- function testUint16Array(value) {
- return value.constructor.name == "Uint16Array";
- }
- function testFloat32Array(value) {
- return value.constructor.name == "Float32Array";
- }
- function testInt16Array(value) {
- return value.constructor.name == "Int16Array";
- }
- function testUint8ClampedArray(value) {
- return value.constructor.name == "Uint8ClampedArray";
- }
- function testUint8Array(value) {
- return value.constructor.name == "Uint8Array";
- }
- function testInt8Array(value) {
- return value.constructor.name == "Int8Array";
- }
- function testArrayBuffer(value) {
- return value.constructor.name == "ArrayBuffer";
- }
- function testNumber(value) {
- return typeof value == "number";
- }
- function testUint32(value) {
- return testInteger(value) && value >= 0 && value <= 4294967295;
- }
- function testInt32(value) {
- return testInteger(value) && value >= -2147483648 && value <= 2147483647;
- }
- function testUint16(value) {
- return testInteger(value) && value >= 0 && value <= 65535;
- }
- function testInt16(value) {
- return testInteger(value) && value >= -32768 && value <= 32767;
- }
- function testUint8(value) {
- return testInteger(value) && value >= 0 && value <= 255;
- }
- function testInt8(value) {
- return testInteger(value) && value >= -128 && value <= 127;
- }
- function testInteger(value) {
- return testNumber(value) && Number.isInteger(value);
- }
- function testUndefined(value) {
- return value === undefined;
- }
- function testNull(value) {
- return value === null;
- }
- function testNaN(value) {
- return Number.isNaN(value);
- }
- function testBoolean(value) {
- return typeof value == "boolean";
- }
- function testMap(value) {
- return value instanceof Map;
- }
- function testSet(value) {
- return value instanceof Set;
- }
- function testDate(value) {
- return value instanceof Date;
- }
- function testError(value) {
- return value instanceof Error;
- }
- function testRegExp(value) {
- return value instanceof RegExp;
- }
- function testStringObject(value) {
- return value instanceof String;
- }
- function testNumberObject(value) {
- return value instanceof Number;
- }
- function testBooleanObject(value) {
- return value instanceof Boolean;
- }
- function testSymbol(value) {
- return typeof value == "symbol";
- }
- function testReferenceable(value) {
- return testObject(value) || testSymbol(value);
- }
- /*
- * Copyright 2010-2020 Gildas Lormeau
- * contact : gildas.lormeau <at> gmail.com
- *
- * This file is part of SingleFile.
- *
- * The code in this file is free software: you can redistribute it and/or
- * modify it under the terms of the GNU Affero General Public License
- * (GNU AGPL) as published by the Free Software Foundation, either version 3
- * of the License, or (at your option) any later version.
- *
- * The code in this file is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero
- * General Public License for more details.
- *
- * As additional permission under GNU AGPL version 3 section 7, you may
- * distribute UNMODIFIED VERSIONS OF THIS file without the copy of the GNU
- * AGPL normally required by section 4, provided you include this license
- * notice and a URL through which recipients can access the Corresponding
- * Source.
- */
- /* global document, globalThis, getComputedStyle, FileReader, Image, OffscreenCanvas, createImageBitmap */
- const singlefile$2 = globalThis.singlefile;
- const CLOSE_ICON = "";
- const SINGLE_FILE_UI_ELEMENT_CLASS$1 = singlefile$2.helper.SINGLE_FILE_UI_ELEMENT_CLASS;
- const ERROR_BAR_TAGNAME = "singlefile-error-bar";
- const OPEN_FILE_BAR_TAGNAME = "singlefile-open-file-bar";
- const SHARE_PAGE_BAR_TAGNAME = "singlefile-share-page-bar";
- let EMBEDDED_IMAGE_BUTTON_MESSAGE$2, SHARE_PAGE_BUTTON_MESSAGE$2, SHARE_SELECTION_BUTTON_MESSAGE$2, ERROR_TITLE_MESSAGE$2;
- const CSS_PROPERTIES$1 = new Set(Array.from(getComputedStyle(document.documentElement)));
- function setLabels(labels) {
- ({ EMBEDDED_IMAGE_BUTTON_MESSAGE: EMBEDDED_IMAGE_BUTTON_MESSAGE$2, SHARE_PAGE_BUTTON_MESSAGE: SHARE_PAGE_BUTTON_MESSAGE$2, SHARE_SELECTION_BUTTON_MESSAGE: SHARE_SELECTION_BUTTON_MESSAGE$2, ERROR_TITLE_MESSAGE: ERROR_TITLE_MESSAGE$2 } = labels);
- }
- function onError(message, link) {
- console.error("SingleFile", message, link); // eslint-disable-line no-console
- displayBar(ERROR_BAR_TAGNAME, ERROR_TITLE_MESSAGE$2 + message, { link });
- }
- function getOpenFileBar() {
- let resolvePromise;
- return {
- display: async function () {
- return new Promise(resolve => {
- resolvePromise = resolve;
- displayBar(OPEN_FILE_BAR_TAGNAME, "", { buttonLabel: EMBEDDED_IMAGE_BUTTON_MESSAGE$2, buttonOnclick: resolve });
- });
- },
- hide: function () {
- const barElement = document.querySelector(OPEN_FILE_BAR_TAGNAME);
- if (barElement) {
- barElement.remove();
- }
- },
- cancel: function () {
- this.hide();
- if (resolvePromise) {
- resolvePromise(true);
- }
- }
- };
- }
- function getSharePageBar() {
- let resolvePromise;
- return {
- display: async function (selectedContent) {
- return new Promise(resolve => {
- resolvePromise = resolve;
- displayBar(SHARE_PAGE_BAR_TAGNAME, "", {
- buttonLabel: selectedContent ? SHARE_SELECTION_BUTTON_MESSAGE$2 : SHARE_PAGE_BUTTON_MESSAGE$2,
- buttonOnclick: resolve
- });
- });
- },
- hide: function () {
- const barElement = document.querySelector(SHARE_PAGE_BAR_TAGNAME);
- if (barElement) {
- barElement.remove();
- }
- },
- cancel: function () {
- this.hide();
- if (resolvePromise) {
- resolvePromise(true);
- }
- }
- };
- }
- function openFile({ accept } = { accept: "image/*" }) {
- const inputElement = document.createElement("input");
- inputElement.type = "file";
- inputElement.accept = accept;
- inputElement.click();
- return new Promise(resolve => {
- inputElement.addEventListener("change", async event => {
- if (event.target.files.length) {
- const file = event.target.files[0];
- let mimeType = file.type;
- if (mimeType == "image/png") {
- const fileReader = new FileReader();
- fileReader.addEventListener("load", async () => resolve(new Uint8Array(fileReader.result)));
- fileReader.addEventListener("error", () => resolve());
- fileReader.readAsArrayBuffer(file);
- } else {
- const dataURI = await new Promise(resolve => {
- const fileReader = new FileReader();
- fileReader.addEventListener("load", () => resolve(fileReader.result));
- fileReader.addEventListener("error", () => resolve());
- fileReader.readAsDataURL(file);
- });
- if (dataURI) {
- const imageBitmap = await createImageBitmap(file);
- const image = new Image();
- image.src = dataURI;
- image.addEventListener("error", () => resolve());
- await new Promise(resolve => image.addEventListener("load", resolve));
- const canvas = new OffscreenCanvas(image.width, image.height);
- const context = canvas.getContext("2d");
- context.drawImage(imageBitmap, 0, 0);
- const blob = await canvas.convertToBlob({ type: "image/png" });
- const fileReader = new FileReader();
- fileReader.addEventListener("load", () => resolve(new Uint8Array(fileReader.result)));
- fileReader.addEventListener("error", () => resolve());
- fileReader.readAsArrayBuffer(blob);
- } else {
- resolve();
- }
- }
- } else {
- resolve();
- }
- });
- inputElement.addEventListener("cancel", () => resolve());
- });
- }
- function displayBar(tagName, message, { link, buttonLabel, buttonOnclick } = {}) {
- try {
- const barElement = document.querySelector(tagName);
- if (!barElement) {
- const barElement = createElement$1(tagName);
- const shadowRoot = barElement.attachShadow({ mode: "open" });
- const styleElement = document.createElement("style");
- styleElement.textContent = `
- .container {
- background-color: #ff6c00;
- color: white;
- display: flex;
- position: fixed;
- top: 0px;
- left: 0px;
- right: 0px;
- height: auto;
- width: auto;
- min-height: 24px;
- min-width: 24px;
- z-index: 2147483647;
- margin: 0;
- padding: 2px;
- font-family: Arial;
- }
- .singlefile-open-file-bar.container, .singlefile-share-page-bar.container {
- background-color: gainsboro;
- border-block-end: gray 1px solid;
- }
- .text {
- flex: 1;
- padding-top: 4px;
- padding-bottom: 4px;
- padding-left: 8px;
- }
- button {
- background-color: grey;
- color: white;
- border: 1px solid darkgrey;
- padding: 3px;
- padding-left: 8px;
- padding-right: 8px;
- border-radius: 4px;
- cursor: pointer;
- }
- .close-button {
- opacity: .7;
- padding-left: 8px;
- padding-right: 8px;
- cursor: pointer;
- transition: opacity 250ms;
- height: 16px;
- font-size: .8rem;
- align-self: center;
- }
- .singlefile-open-file-bar button, .singlefile-share-page-bar button{
- background-color: dimgrey;
- }
- .singlefile-open-file-bar .close-button, .singlefile-share-page-bar .close-button{
- filter: invert(1);
- }
- a {
- color: #303036;
- }
- .close-button:hover {
- opacity: 1;
- }
- `;
- shadowRoot.appendChild(styleElement);
- const containerElement = document.createElement("div");
- containerElement.classList.add(tagName);
- containerElement.classList.add("container");
- const textElement = document.createElement("span");
- textElement.classList.add("text");
- const content = message.split("__DOC_LINK__");
- textElement.textContent = content[0];
- if (link && content.length == 2) {
- const linkElement = document.createElement("a");
- linkElement.textContent = link;
- linkElement.href = link;
- linkElement.target = "_blank";
- textElement.appendChild(linkElement);
- textElement.appendChild(document.createTextNode(content[1]));
- }
- if (buttonLabel && buttonOnclick) {
- const buttonElement = document.createElement("button");
- buttonElement.textContent = buttonLabel;
- buttonElement.onclick = () => buttonOnclick();
- textElement.appendChild(buttonElement);
- }
- containerElement.appendChild(textElement);
- const closeElement = document.createElement("img");
- closeElement.classList.add("close-button");
- containerElement.appendChild(closeElement);
- shadowRoot.appendChild(containerElement);
- closeElement.src = CLOSE_ICON;
- closeElement.onclick = event => {
- if (event.button === 0) {
- if (buttonOnclick) {
- buttonOnclick(true);
- }
- barElement.remove();
- }
- };
- document.documentElement.appendChild(barElement);
- }
- } catch (error) {
- // iignored
- }
- }
- function createElement$1(tagName, parentElement) {
- const element = document.createElement(tagName);
- element.className = SINGLE_FILE_UI_ELEMENT_CLASS$1;
- if (parentElement) {
- parentElement.appendChild(element);
- }
- CSS_PROPERTIES$1.forEach(property => element.style.setProperty(property, "initial", "important"));
- return element;
- }
- /*
- * Copyright 2010-2020 Gildas Lormeau
- * contact : gildas.lormeau <at> gmail.com
- *
- * This file is part of SingleFile.
- *
- * The code in this file is free software: you can redistribute it and/or
- * modify it under the terms of the GNU Affero General Public License
- * (GNU AGPL) as published by the Free Software Foundation, either version 3
- * of the License, or (at your option) any later version.
- *
- * The code in this file is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero
- * General Public License for more details.
- *
- * As additional permission under GNU AGPL version 3 section 7, you may
- * distribute UNMODIFIED VERSIONS OF THIS file without the copy of the GNU
- * AGPL normally required by section 4, provided you include this license
- * notice and a URL through which recipients can access the Corresponding
- * Source.
- */
- const MAX_CONTENT_SIZE = 16 * (1024 * 1024);
- let EMBEDDED_IMAGE_BUTTON_MESSAGE$1, SHARE_PAGE_BUTTON_MESSAGE$1, SHARE_SELECTION_BUTTON_MESSAGE$1, ERROR_TITLE_MESSAGE$1;
- try {
- EMBEDDED_IMAGE_BUTTON_MESSAGE$1 = browser.i18n.getMessage("topPanelEmbeddedImageButton");
- SHARE_PAGE_BUTTON_MESSAGE$1 = browser.i18n.getMessage("topPanelSharePageButton");
- SHARE_SELECTION_BUTTON_MESSAGE$1 = browser.i18n.getMessage("topPanelShareSelectionButton");
- ERROR_TITLE_MESSAGE$1 = browser.i18n.getMessage("topPanelError");
- } catch (error) {
- // ignored
- }
- let sharePageBar;
- setLabels({
- EMBEDDED_IMAGE_BUTTON_MESSAGE: EMBEDDED_IMAGE_BUTTON_MESSAGE$1,
- SHARE_PAGE_BUTTON_MESSAGE: SHARE_PAGE_BUTTON_MESSAGE$1,
- SHARE_SELECTION_BUTTON_MESSAGE: SHARE_SELECTION_BUTTON_MESSAGE$1,
- ERROR_TITLE_MESSAGE: ERROR_TITLE_MESSAGE$1
- });
- async function downloadPage(pageData, options) {
- if (options.includeBOM) {
- pageData.content = "\ufeff" + pageData.content;
- }
- const embeddedImage = options.embeddedImage;
- const message = {
- method: "downloads.download",
- taskId: options.taskId,
- insertTextBody: options.insertTextBody,
- confirmFilename: options.confirmFilename,
- filenameConflictAction: options.filenameConflictAction,
- filename: pageData.filename,
- mimeType: pageData.mimeType,
- saveToClipboard: options.saveToClipboard,
- saveToGDrive: options.saveToGDrive,
- saveToDropbox: options.saveToDropbox,
- saveWithWebDAV: options.saveWithWebDAV,
- webDAVURL: options.webDAVURL,
- webDAVUser: options.webDAVUser,
- webDAVPassword: options.webDAVPassword,
- saveToGitHub: options.saveToGitHub,
- githubToken: options.githubToken,
- githubUser: options.githubUser,
- githubRepository: options.githubRepository,
- githubBranch: options.githubBranch,
- saveWithCompanion: options.saveWithCompanion,
- forceWebAuthFlow: options.forceWebAuthFlow,
- filenameReplacementCharacter: options.filenameReplacementCharacter,
- openEditor: options.openEditor,
- openSavedPage: options.openSavedPage,
- compressHTML: options.compressHTML,
- backgroundSave: options.backgroundSave,
- bookmarkId: options.bookmarkId,
- replaceBookmarkURL: options.replaceBookmarkURL,
- applySystemTheme: options.applySystemTheme,
- defaultEditorMode: options.defaultEditorMode,
- includeInfobar: options.includeInfobar,
- warnUnsavedPage: options.warnUnsavedPage,
- createRootDirectory: options.createRootDirectory,
- selfExtractingArchive: options.selfExtractingArchive,
- embeddedImage: embeddedImage ? Array.from(embeddedImage) : null,
- preventAppendedData: options.preventAppendedData,
- extractDataFromPage: options.extractDataFromPage,
- insertCanonicalLink: options.insertCanonicalLink,
- insertMetaNoIndex: options.insertMetaNoIndex,
- insertMetaCSP: options.insertMetaCSP,
- password: options.password,
- compressContent: options.compressContent,
- foregroundSave: options.foregroundSave,
- sharePage: options.sharePage,
- saveToRestFormApi: options.saveToRestFormApi,
- saveToRestFormApiUrl: options.saveToRestFormApiUrl,
- saveToRestFormApiFileFieldName: options.saveToRestFormApiFileFieldName,
- saveToRestFormApiUrlFieldName: options.saveToRestFormApiUrlFieldName,
- saveToRestFormApiToken: options.saveToRestFormApiToken,
- };
- if (options.compressContent) {
- const blob = new Blob([await serialize(pageData)], { type: pageData.mimeType });
- const blobURL = URL.createObjectURL(blob);
- message.blobURL = blobURL;
- const result = await browser.runtime.sendMessage(message);
- URL.revokeObjectURL(blobURL);
- if (result.error) {
- message.embeddedImage = embeddedImage;
- message.blobURL = null;
- message.pageData = pageData;
- let data, indexData = 0;
- const dataArray = await serialize(message);
- do {
- data = Array.from(dataArray.slice(indexData, indexData + MAX_CONTENT_SIZE));
- indexData += MAX_CONTENT_SIZE;
- await browser.runtime.sendMessage({
- method: "downloads.download",
- compressContent: true,
- data
- });
- } while (data.length);
- await browser.runtime.sendMessage({
- method: "downloads.download",
- compressContent: true,
- mimeType: pageData.mimeType
- });
- }
- if (options.backgroundSave) {
- await browser.runtime.sendMessage({ method: "downloads.end", taskId: options.taskId });
- }
- } else {
- if ((options.backgroundSave && !options.sharePage) || options.openEditor || options.saveToGDrive || options.saveToGitHub || options.saveWithCompanion || options.saveWithWebDAV || options.saveToDropbox || options.saveToRestFormApi) {
- const blobURL = URL.createObjectURL(new Blob([pageData.content], { type: pageData.mimeType }));
- message.blobURL = blobURL;
- const result = await browser.runtime.sendMessage(message);
- URL.revokeObjectURL(blobURL);
- if (result.error) {
- message.blobURL = null;
- for (let blockIndex = 0; blockIndex * MAX_CONTENT_SIZE < pageData.content.length; blockIndex++) {
- message.truncated = pageData.content.length > MAX_CONTENT_SIZE;
- if (message.truncated) {
- message.finished = (blockIndex + 1) * MAX_CONTENT_SIZE > pageData.content.length;
- message.content = pageData.content.substring(blockIndex * MAX_CONTENT_SIZE, (blockIndex + 1) * MAX_CONTENT_SIZE);
- } else {
- message.content = pageData.content;
- }
- await browser.runtime.sendMessage(message);
- }
- }
- } else {
- if (options.saveToClipboard) {
- saveToClipboard(pageData);
- } else {
- await downloadPageForeground(pageData, options);
- }
- if (options.openSavedPage) {
- open(URL.createObjectURL(new Blob([pageData.content], { type: pageData.mimeType })));
- }
- browser.runtime.sendMessage({ method: "ui.processEnd" });
- }
- await browser.runtime.sendMessage({ method: "downloads.end", taskId: options.taskId, hash: pageData.hash, woleetKey: options.woleetKey });
- }
- }
- async function downloadPageForeground(pageData, options) {
- if (Array.isArray(pageData.content)) {
- pageData.content = new Uint8Array(pageData.content);
- }
- if (options.sharePage && navigator.share) {
- await sharePage(pageData, options);
- } else {
- if (pageData.filename && pageData.filename.length) {
- const link = document.createElement("a");
- link.download = pageData.filename;
- link.href = URL.createObjectURL(new Blob([pageData.content], { type: pageData.mimeType }));
- link.dispatchEvent(new MouseEvent("click"));
- return new Promise(resolve => setTimeout(() => { URL.revokeObjectURL(link.href); resolve(); }, 1000));
- }
- }
- }
- async function sharePage(pageData, options) {
- sharePageBar = getSharePageBar();
- const cancelled = await sharePageBar.display(options.selected);
- if (!cancelled) {
- const data = { files: [new File([pageData.content], pageData.filename, { type: pageData.mimeType })] };
- try {
- await navigator.share(data);
- sharePageBar.hide();
- } catch (error) {
- sharePageBar.hide();
- if (error.name === "AbortError") {
- await sharePage(pageData, options);
- } else {
- throw error;
- }
- }
- }
- }
- function saveToClipboard(pageData) {
- const command = "copy";
- document.addEventListener(command, listener);
- document.execCommand(command);
- document.removeEventListener(command, listener);
- function listener(event) {
- event.clipboardData.setData(pageData.mimeType, pageData.content);
- event.clipboardData.setData("text/plain", pageData.content);
- event.preventDefault();
- }
- }
- /*
- * Copyright 2010-2020 Gildas Lormeau
- * contact : gildas.lormeau <at> gmail.com
- *
- * This file is part of SingleFile.
- *
- * The code in this file is free software: you can redistribute it and/or
- * modify it under the terms of the GNU Affero General Public License
- * (GNU AGPL) as published by the Free Software Foundation, either version 3
- * of the License, or (at your option) any later version.
- *
- * The code in this file is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero
- * General Public License for more details.
- *
- * As additional permission under GNU AGPL version 3 section 7, you may
- * distribute UNMODIFIED VERSIONS OF THIS file without the copy of the GNU
- * AGPL normally required by section 4, provided you include this license
- * notice and a URL through which recipients can access the Corresponding
- * Source.
- */
- /* global browser, window, document, CustomEvent, setTimeout, clearTimeout */
- const FETCH_REQUEST_EVENT = "single-file-request-fetch";
- const FETCH_ACK_EVENT = "single-file-ack-fetch";
- const FETCH_RESPONSE_EVENT = "single-file-response-fetch";
- const ERR_HOST_FETCH = "Host fetch error (SingleFile)";
- const HOST_FETCH_MAX_DELAY = 2500;
- const USE_HOST_FETCH = Boolean(window.wrappedJSObject);
- const fetch = (url, options) => window.fetch(url, options);
- let requestId = 0, pendingResponses = new Map();
- browser.runtime.onMessage.addListener(message => {
- if (message.method == "singlefile.fetchFrame" && window.frameId && window.frameId == message.frameId) {
- return onFetchFrame(message);
- }
- if (message.method == "singlefile.fetchResponse") {
- return onFetchResponse(message);
- }
- });
- async function onFetchFrame(message) {
- try {
- const response = await fetch(message.url, { cache: "force-cache", headers: message.headers });
- return {
- status: response.status,
- headers: [...response.headers],
- array: Array.from(new Uint8Array(await response.arrayBuffer()))
- };
- } catch (error) {
- return {
- error: error && error.toString()
- };
- }
- }
- async function onFetchResponse(message) {
- const pendingResponse = pendingResponses.get(message.requestId);
- if (pendingResponse) {
- if (message.error) {
- pendingResponse.reject(new Error(message.error));
- pendingResponses.delete(message.requestId);
- } else {
- if (message.truncated) {
- if (pendingResponse.array) {
- pendingResponse.array = pendingResponse.array.concat(message.array);
- } else {
- pendingResponse.array = message.array;
- pendingResponses.set(message.requestId, pendingResponse);
- }
- if (message.finished) {
- message.array = pendingResponse.array;
- }
- }
- if (!message.truncated || message.finished) {
- pendingResponse.resolve({
- status: message.status,
- headers: { get: headerName => message.headers && message.headers[headerName] },
- arrayBuffer: async () => new Uint8Array(message.array).buffer
- });
- pendingResponses.delete(message.requestId);
- }
- }
- }
- return {};
- }
- async function hostFetch(url, options) {
- const result = new Promise((resolve, reject) => {
- document.dispatchEvent(new CustomEvent(FETCH_REQUEST_EVENT, { detail: JSON.stringify({ url, options }) }));
- document.addEventListener(FETCH_ACK_EVENT, onAckFetch, false);
- document.addEventListener(FETCH_RESPONSE_EVENT, onResponseFetch, false);
- const timeout = setTimeout(() => {
- removeListeners();
- reject(new Error(ERR_HOST_FETCH));
- }, HOST_FETCH_MAX_DELAY);
- function onResponseFetch(event) {
- if (event.detail) {
- if (event.detail.url == url) {
- removeListeners();
- if (event.detail.response) {
- resolve({
- status: event.detail.status,
- headers: new Map(event.detail.headers),
- arrayBuffer: async () => event.detail.response
- });
- } else {
- reject(event.detail.error);
- }
- }
- } else {
- reject();
- }
- }
- function onAckFetch() {
- clearTimeout(timeout);
- }
- function removeListeners() {
- document.removeEventListener(FETCH_RESPONSE_EVENT, onResponseFetch, false);
- document.removeEventListener(FETCH_ACK_EVENT, onAckFetch, false);
- }
- });
- try {
- return await result;
- } catch (error) {
- if (error && error.message == ERR_HOST_FETCH) {
- return fetch(url, options);
- } else {
- throw error;
- }
- }
- }
- async function fetchResource(url, options = {}) {
- try {
- const fetchOptions = { cache: "force-cache", headers: options.headers };
- return await (options.referrer && USE_HOST_FETCH ? hostFetch(url, fetchOptions) : fetch(url, fetchOptions));
- }
- catch (error) {
- requestId++;
- const promise = new Promise((resolve, reject) => pendingResponses.set(requestId, { resolve, reject }));
- await sendMessage({ method: "singlefile.fetch", url, requestId, referrer: options.referrer, headers: options.headers });
- return promise;
- }
- }
- async function frameFetch(url, options) {
- const response = await sendMessage({ method: "singlefile.fetchFrame", url, frameId: options.frameId, referrer: options.referrer, headers: options.headers });
- return {
- status: response.status,
- headers: new Map(response.headers),
- arrayBuffer: async () => new Uint8Array(response.array).buffer
- };
- }
- async function sendMessage(message) {
- const response = await browser.runtime.sendMessage(message);
- if (!response || response.error) {
- throw new Error(response && response.error && response.error.toString());
- } else {
- return response;
- }
- }
- /*
- * Copyright 2010-2020 Gildas Lormeau
- * contact : gildas.lormeau <at> gmail.com
- *
- * This file is part of SingleFile.
- *
- * The code in this file is free software: you can redistribute it and/or
- * modify it under the terms of the GNU Affero General Public License
- * (GNU AGPL) as published by the Free Software Foundation, either version 3
- * of the License, or (at your option) any later version.
- *
- * The code in this file is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero
- * General Public License for more details.
- *
- * As additional permission under GNU AGPL version 3 section 7, you may
- * distribute UNMODIFIED VERSIONS OF THIS file without the copy of the GNU
- * AGPL normally required by section 4, provided you include this license
- * notice and a URL through which recipients can access the Corresponding
- * Source.
- */
- /* global browser, document, globalThis, prompt, getComputedStyle, addEventListener, removeEventListener, requestAnimationFrame, setTimeout, getSelection, Node */
- const singlefile$1 = globalThis.singlefile;
- const SELECTED_CONTENT_ATTRIBUTE_NAME = singlefile$1.helper.SELECTED_CONTENT_ATTRIBUTE_NAME;
- const MASK_TAGNAME = "singlefile-mask";
- const MASK_CONTENT_CLASSNAME = "singlefile-mask-content";
- const PROGRESSBAR_CLASSNAME = "singlefile-progress-bar";
- const PROGRESSBAR_CONTENT_CLASSNAME = "singlefile-progress-bar-content";
- const SELECTION_ZONE_TAGNAME = "single-file-selection-zone";
- const LOGS_WINDOW_TAGNAME = "singlefile-logs-window";
- const LOGS_CLASSNAME = "singlefile-logs";
- const LOGS_LINE_CLASSNAME = "singlefile-logs-line";
- const LOGS_LINE_TEXT_ELEMENT_CLASSNAME = "singlefile-logs-line-text";
- const LOGS_LINE_STATUS_ELEMENT_CLASSNAME = "singlefile-logs-line-icon";
- const SINGLE_FILE_UI_ELEMENT_CLASS = singlefile$1.helper.SINGLE_FILE_UI_ELEMENT_CLASS;
- const SELECT_PX_THRESHOLD = 8;
- const LOG_PANEL_DEFERRED_IMAGES_MESSAGE = browser.i18n.getMessage("logPanelDeferredImages");
- const LOG_PANEL_FRAME_CONTENTS_MESSAGE = browser.i18n.getMessage("logPanelFrameContents");
- const LOG_PANEL_EMBEDDED_IMAGE_MESSAGE = browser.i18n.getMessage("logPanelEmbeddedImage");
- const LOG_PANEL_STEP_MESSAGE = browser.i18n.getMessage("logPanelStep");
- const LOG_PANEL_WIDTH = browser.i18n.getMessage("logPanelWidth");
- const CSS_PROPERTIES = new Set(Array.from(getComputedStyle(document.documentElement)));
- let selectedAreaElement, logsWindowElement;
- createLogsWindowElement();
- function promptMessage(message, defaultValue) {
- return prompt(message, defaultValue);
- }
- function setVisible(visible) {
- const maskElement = document.querySelector(MASK_TAGNAME);
- if (maskElement) {
- maskElement.style.setProperty("display", visible ? "block" : "none");
- }
- if (logsWindowElement) {
- logsWindowElement.style.setProperty("display", visible ? "block" : "none");
- }
- }
- function onStartPage(options) {
- let maskElement = document.querySelector(MASK_TAGNAME);
- if (!maskElement) {
- if (options.logsEnabled) {
- document.documentElement.appendChild(logsWindowElement);
- }
- if (options.shadowEnabled) {
- const maskElement = createMaskElement();
- if (options.progressBarEnabled) {
- createProgressBarElement(maskElement);
- }
- }
- }
- }
- function onEndPage() {
- const maskElement = document.querySelector(MASK_TAGNAME);
- if (maskElement) {
- maskElement.remove();
- }
- logsWindowElement.remove();
- clearLogs();
- }
- function onLoadResource(index, maxIndex, options) {
- if (options.shadowEnabled && options.progressBarEnabled) {
- updateProgressBar(index, maxIndex);
- }
- }
- function onLoadingDeferResources(options) {
- updateLog("load-deferred-images", LOG_PANEL_DEFERRED_IMAGES_MESSAGE, "…", options);
- }
- function onLoadDeferResources(options) {
- updateLog("load-deferred-images", LOG_PANEL_DEFERRED_IMAGES_MESSAGE, "✓", options);
- }
- function onInsertingEmbeddedImage(options) {
- updateLog("insert-embedded-image", LOG_PANEL_EMBEDDED_IMAGE_MESSAGE, "…", options);
- }
- function onInsertEmbeddedImage(options) {
- updateLog("insert-embedded-image", LOG_PANEL_EMBEDDED_IMAGE_MESSAGE, "✓", options);
- }
- function onLoadingFrames(options) {
- updateLog("load-frames", LOG_PANEL_FRAME_CONTENTS_MESSAGE, "…", options);
- }
- function onLoadFrames(options) {
- updateLog("load-frames", LOG_PANEL_FRAME_CONTENTS_MESSAGE, "✓", options);
- }
- function onStartStage(step, options) {
- updateLog("step-" + step, `${LOG_PANEL_STEP_MESSAGE} ${step + 1} / 3`, "…", options);
- }
- function onEndStage(step, options) {
- updateLog("step-" + step, `${LOG_PANEL_STEP_MESSAGE} ${step + 1} / 3`, "✓", options);
- }
- function getSelectedLinks() {
- let selectionFound;
- const links = [];
- const selection = getSelection();
- for (let indexRange = 0; indexRange < selection.rangeCount; indexRange++) {
- let range = selection.getRangeAt(indexRange);
- if (range && range.commonAncestorContainer) {
- const treeWalker = document.createTreeWalker(range.commonAncestorContainer);
- let rangeSelectionFound = false;
- let finished = false;
- while (!finished) {
- if (rangeSelectionFound || treeWalker.currentNode == range.startContainer || treeWalker.currentNode == range.endContainer) {
- rangeSelectionFound = true;
- if (range.startContainer != range.endContainer || range.startOffset != range.endOffset) {
- selectionFound = true;
- if (treeWalker.currentNode.tagName == "A" && treeWalker.currentNode.href) {
- links.push(treeWalker.currentNode.href);
- }
- }
- }
- if (treeWalker.currentNode == range.endContainer) {
- finished = true;
- } else {
- treeWalker.nextNode();
- }
- }
- if (selectionFound && treeWalker.currentNode == range.endContainer && treeWalker.currentNode.querySelectorAll) {
- treeWalker.currentNode.querySelectorAll("*").forEach(descendantElement => {
- if (descendantElement.tagName == "A" && descendantElement.href) {
- links.push(treeWalker.currentNode.href);
- }
- });
- }
- }
- }
- return Array.from(new Set(links));
- }
- async function markSelection(optionallySelected) {
- let selectionFound = markSelectedContent();
- if (selectionFound || optionallySelected) {
- return selectionFound;
- } else {
- selectionFound = await selectArea();
- if (selectionFound) {
- return markSelectedContent();
- }
- }
- }
- function markSelectedContent() {
- const selection = getSelection();
- let selectionFound;
- for (let indexRange = 0; indexRange < selection.rangeCount; indexRange++) {
- let range = selection.getRangeAt(indexRange);
- if (range && range.commonAncestorContainer) {
- const treeWalker = document.createTreeWalker(range.commonAncestorContainer);
- let rangeSelectionFound = false;
- let finished = false;
- while (!finished) {
- if (rangeSelectionFound || treeWalker.currentNode == range.startContainer || treeWalker.currentNode == range.endContainer) {
- rangeSelectionFound = true;
- if (range.startContainer != range.endContainer || range.startOffset != range.endOffset) {
- selectionFound = true;
- markSelectedNode(treeWalker.currentNode);
- }
- }
- if (selectionFound && treeWalker.currentNode == range.startContainer) {
- markSelectedParents(treeWalker.currentNode);
- }
- if (treeWalker.currentNode == range.endContainer) {
- finished = true;
- } else {
- treeWalker.nextNode();
- }
- }
- if (selectionFound && treeWalker.currentNode == range.endContainer && treeWalker.currentNode.querySelectorAll) {
- for (
- let offset = range.startContainer === range.endContainer ? range.startOffset : 0;
- offset < range.endOffset;
- offset++
- ) {
- const node = range.endContainer.childNodes[offset];
- if (node) {
- markSelectedNode(node);
- if (node.querySelectorAll) {
- node.querySelectorAll("*").forEach(markSelectedNode);
- }
- }
- }
- }
- }
- }
- return selectionFound;
- }
- function markSelectedNode(node) {
- const element = node.nodeType == Node.ELEMENT_NODE ? node : node.parentElement;
- element.setAttribute(SELECTED_CONTENT_ATTRIBUTE_NAME, "");
- }
- function markSelectedParents(node) {
- if (node.parentElement) {
- markSelectedNode(node);
- markSelectedParents(node.parentElement);
- }
- }
- function unmarkSelection() {
- document.querySelectorAll("[" + SELECTED_CONTENT_ATTRIBUTE_NAME + "]").forEach(selectedContent => selectedContent.removeAttribute(SELECTED_CONTENT_ATTRIBUTE_NAME));
- }
- function selectArea() {
- return new Promise(resolve => {
- let selectedRanges = [];
- addEventListener("mousemove", mousemoveListener, true);
- addEventListener("click", clickListener, true);
- addEventListener("keyup", keypressListener, true);
- document.addEventListener("contextmenu", contextmenuListener, true);
- getSelection().removeAllRanges();
- function contextmenuListener(event) {
- selectedRanges = [];
- select();
- event.preventDefault();
- }
- function mousemoveListener(event) {
- const targetElement = getTarget(event);
- if (targetElement) {
- selectedAreaElement = targetElement;
- moveAreaSelector(targetElement);
- }
- }
- function clickListener(event) {
- event.preventDefault();
- event.stopPropagation();
- if (event.button == 0) {
- select(selectedAreaElement, event.ctrlKey);
- } else {
- cancel();
- }
- }
- function keypressListener(event) {
- if (event.key == "Escape") {
- cancel();
- }
- }
- function cancel() {
- if (selectedRanges.length) {
- getSelection().removeAllRanges();
- }
- selectedRanges = [];
- cleanupAndResolve();
- }
- function select(selectedElement, multiSelect) {
- if (selectedElement) {
- if (!multiSelect) {
- restoreSelectedRanges();
- }
- const range = document.createRange();
- range.selectNodeContents(selectedElement);
- cleanupSelectionRanges();
- getSelection().addRange(range);
- saveSelectedRanges();
- if (!multiSelect) {
- cleanupAndResolve();
- }
- } else {
- cleanupAndResolve();
- }
- }
- function cleanupSelectionRanges() {
- const selection = getSelection();
- for (let indexRange = selection.rangeCount - 1; indexRange >= 0; indexRange--) {
- const range = selection.getRangeAt(indexRange);
- if (range.startOffset == range.endOffset) {
- selection.removeRange(range);
- indexRange--;
- }
- }
- }
- function cleanupAndResolve() {
- getAreaSelector().remove();
- removeEventListener("mousemove", mousemoveListener, true);
- removeEventListener("click", clickListener, true);
- removeEventListener("keyup", keypressListener, true);
- selectedAreaElement = null;
- resolve(Boolean(selectedRanges.length));
- setTimeout(() => document.removeEventListener("contextmenu", contextmenuListener, true), 0);
- }
- function restoreSelectedRanges() {
- getSelection().removeAllRanges();
- selectedRanges.forEach(range => getSelection().addRange(range));
- }
- function saveSelectedRanges() {
- selectedRanges = [];
- for (let indexRange = 0; indexRange < getSelection().rangeCount; indexRange++) {
- const range = getSelection().getRangeAt(indexRange);
- selectedRanges.push(range);
- }
- }
- });
- }
- function getTarget(event) {
- let newTarget, target = event.target, boundingRect = target.getBoundingClientRect();
- newTarget = determineTargetElement("floor", target, event.clientX - boundingRect.left, getMatchedParents(target, "left"));
- if (newTarget == target) {
- newTarget = determineTargetElement("ceil", target, boundingRect.left + boundingRect.width - event.clientX, getMatchedParents(target, "right"));
- }
- if (newTarget == target) {
- newTarget = determineTargetElement("floor", target, event.clientY - boundingRect.top, getMatchedParents(target, "top"));
- }
- if (newTarget == target) {
- newTarget = determineTargetElement("ceil", target, boundingRect.top + boundingRect.height - event.clientY, getMatchedParents(target, "bottom"));
- }
- target = newTarget;
- while (target && target.clientWidth <= SELECT_PX_THRESHOLD && target.clientHeight <= SELECT_PX_THRESHOLD) {
- target = target.parentElement;
- }
- return target;
- }
- function moveAreaSelector(target) {
- requestAnimationFrame(() => {
- const selectorElement = getAreaSelector();
- const boundingRect = target.getBoundingClientRect();
- const scrollingElement = document.scrollingElement || document.documentElement;
- selectorElement.style.setProperty("top", (scrollingElement.scrollTop + boundingRect.top - 10) + "px");
- selectorElement.style.setProperty("left", (scrollingElement.scrollLeft + boundingRect.left - 10) + "px");
- selectorElement.style.setProperty("width", (boundingRect.width + 20) + "px");
- selectorElement.style.setProperty("height", (boundingRect.height + 20) + "px");
- });
- }
- function getAreaSelector() {
- let selectorElement = document.querySelector(SELECTION_ZONE_TAGNAME);
- if (!selectorElement) {
- selectorElement = createElement(SELECTION_ZONE_TAGNAME, document.body);
- selectorElement.style.setProperty("box-sizing", "border-box", "important");
- selectorElement.style.setProperty("background-color", "#3ea9d7", "important");
- selectorElement.style.setProperty("border", "10px solid #0b4892", "important");
- selectorElement.style.setProperty("border-radius", "2px", "important");
- selectorElement.style.setProperty("opacity", ".25", "important");
- selectorElement.style.setProperty("pointer-events", "none", "important");
- selectorElement.style.setProperty("position", "absolute", "important");
- selectorElement.style.setProperty("transition", "all 100ms", "important");
- selectorElement.style.setProperty("cursor", "pointer", "important");
- selectorElement.style.setProperty("z-index", "2147483647", "important");
- selectorElement.style.removeProperty("border-inline-end");
- selectorElement.style.removeProperty("border-inline-start");
- selectorElement.style.removeProperty("inline-size");
- selectorElement.style.removeProperty("block-size");
- selectorElement.style.removeProperty("inset-block-start");
- selectorElement.style.removeProperty("inset-inline-end");
- selectorElement.style.removeProperty("inset-block-end");
- selectorElement.style.removeProperty("inset-inline-start");
- }
- return selectorElement;
- }
- function createMaskElement() {
- try {
- let maskElement = document.querySelector(MASK_TAGNAME);
- if (!maskElement) {
- maskElement = createElement(MASK_TAGNAME, document.documentElement);
- const shadowRoot = maskElement.attachShadow({ mode: "open" });
- const styleElement = document.createElement("style");
- styleElement.textContent = `
- @keyframes single-file-progress {
- 0% {
- left: -50px;
- }
- 100% {
- left: 0;
- }
- }
- .${PROGRESSBAR_CLASSNAME} {
- position: fixed;
- top: 0;
- left: 0;
- width: 0;
- height: 8px;
- z-index: 2147483646;
- opacity: .5;
- overflow: hidden;
- transition: width 200ms ease-in-out;
- }
- .${PROGRESSBAR_CONTENT_CLASSNAME} {
- position: absolute;
- left: 0;
- animation: single-file-progress 3s linear infinite reverse;
- background:
- white
- linear-gradient(-45deg, rgba(0, 0, 0, 0.075) 25%,
- transparent 25%,
- transparent 50%,
- rgba(0, 0, 0, 0.075) 50%,
- rgba(0, 0, 0, 0.075) 75%,
- transparent 75%, transparent)
- repeat scroll 0% 0% / 50px 50px padding-box border-box;
- width: calc(100% + 50px);
- height: 100%;
- }
- .${MASK_CONTENT_CLASSNAME} {
- position: fixed;
- top: 0;
- left: 0;
- width: 100%;
- height: 100%;
- z-index: 2147483646;
- opacity: 0;
- background-color: black;
- transition: opacity 250ms;
- }
- `;
- shadowRoot.appendChild(styleElement);
- let maskElementContent = document.createElement("div");
- maskElementContent.classList.add(MASK_CONTENT_CLASSNAME);
- shadowRoot.appendChild(maskElementContent);
- maskElement.offsetWidth;
- maskElementContent.style.setProperty("opacity", .3);
- maskElement.offsetWidth;
- }
- return maskElement;
- } catch (error) {
- // ignored
- }
- }
- function createProgressBarElement(maskElement) {
- try {
- let progressBarElement = maskElement.shadowRoot.querySelector("." + PROGRESSBAR_CLASSNAME);
- if (!progressBarElement) {
- let progressBarContent = document.createElement("div");
- progressBarContent.classList.add(PROGRESSBAR_CLASSNAME);
- maskElement.shadowRoot.appendChild(progressBarContent);
- const progressBarContentElement = document.createElement("div");
- progressBarContentElement.classList.add(PROGRESSBAR_CONTENT_CLASSNAME);
- progressBarContent.appendChild(progressBarContentElement);
- }
- } catch (error) {
- // ignored
- }
- }
- function createLogsWindowElement() {
- try {
- logsWindowElement = document.querySelector(LOGS_WINDOW_TAGNAME);
- if (!logsWindowElement) {
- logsWindowElement = createElement(LOGS_WINDOW_TAGNAME);
- const shadowRoot = logsWindowElement.attachShadow({ mode: "open" });
- const styleElement = document.createElement("style");
- styleElement.textContent = `
- @keyframes single-file-pulse {
- 0% {
- opacity: .25;
- }
- 100% {
- opacity: 1;
- }
- }
- .${LOGS_CLASSNAME} {
- position: fixed;
- bottom: 24px;
- left: 8px;
- z-index: 2147483647;
- opacity: 0.9;
- padding: 4px;
- background-color: white;
- min-width: ${LOG_PANEL_WIDTH}px;
- min-height: 16px;
- transition: height 100ms;
- }
- .${LOGS_LINE_CLASSNAME} {
- display: flex;
- justify-content: space-between;
- padding: 2px;
- font-family: arial, sans-serif;
- color: black;
- background-color: white;
- }
- .${LOGS_LINE_TEXT_ELEMENT_CLASSNAME} {
- font-size: 13px;
- opacity: 1;
- transition: opacity 200ms;
- }
- .${LOGS_LINE_STATUS_ELEMENT_CLASSNAME} {
- font-size: 11px;
- min-width: 15px;
- text-align: center;
- position: relative;
- top: 1px;
- }
- `;
- shadowRoot.appendChild(styleElement);
- const logsContentElement = document.createElement("div");
- logsContentElement.classList.add(LOGS_CLASSNAME);
- shadowRoot.appendChild(logsContentElement);
- }
- } catch (error) {
- // ignored
- }
- }
- function updateLog(id, textContent, textStatus, options) {
- try {
- if (options.logsEnabled) {
- const logsContentElement = logsWindowElement.shadowRoot.querySelector("." + LOGS_CLASSNAME);
- let lineElement = logsContentElement.querySelector("[data-id='" + id + "']");
- if (!lineElement) {
- lineElement = document.createElement("div");
- lineElement.classList.add(LOGS_LINE_CLASSNAME);
- logsContentElement.appendChild(lineElement);
- lineElement.setAttribute("data-id", id);
- const textElement = document.createElement("div");
- textElement.classList.add(LOGS_LINE_TEXT_ELEMENT_CLASSNAME);
- lineElement.appendChild(textElement);
- textElement.textContent = textContent;
- const statusElement = document.createElement("div");
- statusElement.classList.add(LOGS_LINE_STATUS_ELEMENT_CLASSNAME);
- lineElement.appendChild(statusElement);
- }
- updateLogLine(lineElement, textContent, textStatus);
- }
- } catch (error) {
- // ignored
- }
- }
- function updateLogLine(lineElement, textContent, textStatus) {
- const textElement = lineElement.childNodes[0];
- const statusElement = lineElement.childNodes[1];
- textElement.textContent = textContent;
- statusElement.style.setProperty("color", textStatus == "✓" ? "#055000" : "black");
- if (textStatus == "✓") {
- textElement.style.setProperty("opacity", ".5");
- statusElement.style.setProperty("opacity", ".5");
- statusElement.style.setProperty("animation", "none");
- } else {
- statusElement.style.setProperty("animation", "1s ease-in-out 0s infinite alternate none running single-file-pulse");
- }
- statusElement.textContent = textStatus;
- }
- function updateProgressBar(index, maxIndex) {
- try {
- const maskElement = document.querySelector(MASK_TAGNAME);
- if (maskElement) {
- const progressBarElement = maskElement.shadowRoot.querySelector("." + PROGRESSBAR_CLASSNAME);
- if (progressBarElement && maxIndex) {
- const width = Math.floor((index / maxIndex) * 100) + "%";
- if (progressBarElement.style.getPropertyValue("width") != width) {
- progressBarElement.style.setProperty("width", width);
- progressBarElement.offsetWidth;
- }
- }
- }
- } catch (error) {
- // ignored
- }
- }
- function clearLogs() {
- createLogsWindowElement();
- }
- function getMatchedParents(target, property) {
- let element = target, matchedParent, parents = [];
- do {
- const boundingRect = element.getBoundingClientRect();
- if (element.parentElement) {
- const parentBoundingRect = element.parentElement.getBoundingClientRect();
- matchedParent = Math.abs(parentBoundingRect[property] - boundingRect[property]) <= SELECT_PX_THRESHOLD;
- if (matchedParent) {
- if (element.parentElement.clientWidth > SELECT_PX_THRESHOLD && element.parentElement.clientHeight > SELECT_PX_THRESHOLD &&
- ((element.parentElement.clientWidth - element.clientWidth > SELECT_PX_THRESHOLD) || (element.parentElement.clientHeight - element.clientHeight > SELECT_PX_THRESHOLD))) {
- parents.push(element.parentElement);
- }
- element = element.parentElement;
- }
- } else {
- matchedParent = false;
- }
- } while (matchedParent && element);
- return parents;
- }
- function determineTargetElement(roundingMethod, target, widthDistance, parents) {
- if (Math[roundingMethod](widthDistance / SELECT_PX_THRESHOLD) <= parents.length) {
- target = parents[parents.length - Math[roundingMethod](widthDistance / SELECT_PX_THRESHOLD) - 1];
- }
- return target;
- }
- function createElement(tagName, parentElement) {
- const element = document.createElement(tagName);
- element.className = SINGLE_FILE_UI_ELEMENT_CLASS;
- if (parentElement) {
- parentElement.appendChild(element);
- }
- CSS_PROPERTIES.forEach(property => element.style.setProperty(property, "initial", "important"));
- return element;
- }
- /*
- * Copyright 2010-2020 Gildas Lormeau
- * contact : gildas.lormeau <at> gmail.com
- *
- * This file is part of SingleFile.
- *
- * The code in this file is free software: you can redistribute it and/or
- * modify it under the terms of the GNU Affero General Public License
- * (GNU AGPL) as published by the Free Software Foundation, either version 3
- * of the License, or (at your option) any later version.
- *
- * The code in this file is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero
- * General Public License for more details.
- *
- * As additional permission under GNU AGPL version 3 section 7, you may
- * distribute UNMODIFIED VERSIONS OF THIS file without the copy of the GNU
- * AGPL normally required by section 4, provided you include this license
- * notice and a URL through which recipients can access the Corresponding
- * Source.
- */
- const singlefile = globalThis.singlefile;
- const bootstrap = globalThis.singlefileBootstrap;
- const MOZ_EXTENSION_PROTOCOL = "moz-extension:";
- const EMBEDDED_IMAGE_BUTTON_MESSAGE = browser.i18n.getMessage("topPanelEmbeddedImageButton");
- const SHARE_PAGE_BUTTON_MESSAGE = browser.i18n.getMessage("topPanelSharePageButton");
- const SHARE_SELECTION_BUTTON_MESSAGE = browser.i18n.getMessage("topPanelShareSelectionButton");
- const ERROR_TITLE_MESSAGE = browser.i18n.getMessage("topPanelError");
- let processor, processing, downloadParser, openFileInfobar, scrollY, transform, overflow;
- setLabels({
- EMBEDDED_IMAGE_BUTTON_MESSAGE,
- SHARE_PAGE_BUTTON_MESSAGE,
- SHARE_SELECTION_BUTTON_MESSAGE,
- ERROR_TITLE_MESSAGE
- });
- if (!bootstrap || !bootstrap.initializedSingleFile) {
- singlefile.init({ fetch: fetchResource, frameFetch });
- browser.runtime.onMessage.addListener(message => {
- if (message.method == "content.save" ||
- message.method == "content.cancelSave" ||
- message.method == "content.download" ||
- message.method == "content.getSelectedLinks" ||
- message.method == "content.error" ||
- message.method == "content.prompt" ||
- message.method == "content.beginScrollTo" ||
- message.method == "content.scrollTo" ||
- message.method == "content.endScrollTo") {
- return onMessage(message);
- }
- });
- if (bootstrap) {
- bootstrap.initializedSingleFile = true;
- } else {
- globalThis.singlefileBootstrap = { initializedSingleFile: true };
- }
- }
- async function onMessage(message) {
- if (!location.href.startsWith(MOZ_EXTENSION_PROTOCOL)) {
- if (message.method == "content.save") {
- await savePage(message);
- return {};
- }
- if (message.method == "content.cancelSave") {
- if (processor) {
- processor.cancel();
- onEndPage();
- if (openFileInfobar) {
- openFileInfobar.cancel();
- openFileInfobar = null;
- }
- browser.runtime.sendMessage({ method: "ui.processCancelled" });
- }
- if (message.options.loadDeferredImages) {
- singlefile.processors.lazy.resetZoomLevel(message.options);
- }
- return {};
- }
- if (message.method == "content.getSelectedLinks") {
- return {
- urls: getSelectedLinks()
- };
- }
- if (message.method == "content.download") {
- if (!downloadParser) {
- downloadParser = getParser();
- }
- const result = await downloadParser.next(message.data);
- if (result.done) {
- downloadParser = null;
- try {
- await downloadPageForeground(result.value, {
- foregroundSave: result.value.foregroundSave,
- sharePage: result.value.sharePage,
- });
- } catch (error) {
- return {
- error: error.toString()
- };
- } finally {
- await browser.runtime.sendMessage({ method: "downloads.end", taskId: result.value.taskId });
- }
- }
- return {};
- }
- if (message.method == "content.error") {
- onError(message.error, message.link);
- return {};
- }
- if (message.method == "content.prompt") {
- return promptMessage(message.message, message.value);
- }
- if (message.method == "content.beginScrollTo") {
- scrollY = globalThis.scrollY;
- transform = document.documentElement.style.getPropertyValue("transform");
- overflow = document.documentElement.style.getPropertyValue("overflow");
- globalThis.scrollTo(0, 0);
- document.documentElement.style.setProperty("transform", "translateY(0px)");
- document.documentElement.style.setProperty("overflow", "hidden");
- return {};
- }
- if (message.method == "content.scrollTo") {
- document.documentElement.style.setProperty("transform", "translateY(-" + message.y + "px)");
- await new Promise(resolve => setTimeout(resolve, 500));
- return {};
- }
- if (message.method == "content.endScrollTo") {
- globalThis.scrollTo(0, scrollY);
- document.documentElement.style.setProperty("transform", transform);
- document.documentElement.style.setProperty("overflow", overflow);
- return {};
- }
- }
- }
- async function savePage(message) {
- const options = message.options;
- let selectionFound;
- if (options.selected || options.optionallySelected) {
- selectionFound = await markSelection(options.optionallySelected);
- }
- if (!processing && (!bootstrap || !bootstrap.pageInfo.processing)) {
- options.updatedResources = bootstrap ? bootstrap.pageInfo.updatedResources : {};
- options.visitDate = bootstrap ? bootstrap.pageInfo.visitDate : new Date();
- Object.keys(options.updatedResources).forEach(url => options.updatedResources[url].retrieved = false);
- if (options.optionallySelected && selectionFound) {
- options.selected = true;
- }
- if (!options.selected || selectionFound) {
- if (bootstrap) {
- bootstrap.pageInfo.processing = true;
- }
- processing = true;
- try {
- const pageData = await processPage(options);
- if (pageData) {
- if (((!options.backgroundSave && !options.saveToClipboard) || options.saveToGDrive || options.saveToGitHub || options.saveWithCompanion || options.saveWithWebDAV || options.saveToDropbox || options.saveToRestFormApi) && options.confirmFilename) {
- pageData.filename = promptMessage("Save as", pageData.filename) || pageData.filename;
- }
- await downloadPage(pageData, options);
- }
- } catch (error) {
- if (!processor.cancelled) {
- console.error(error); // eslint-disable-line no-console
- const errorMessage = error.toString();
- browser.runtime.sendMessage({ method: "ui.processError", error: errorMessage });
- onError(errorMessage);
- }
- }
- } else {
- browser.runtime.sendMessage({ method: "ui.processCancelled" });
- }
- processing = false;
- if (bootstrap) {
- bootstrap.pageInfo.processing = false;
- }
- }
- }
- async function processPage(options) {
- const frames = singlefile.processors.frameTree;
- let framesSessionId;
- options.keepFilename = options.saveToGDrive || options.saveToGitHub || options.saveWithWebDAV || options.saveToDropbox || options.saveToRestFormApi;
- singlefile.helper.initDoc(document);
- onStartPage(options);
- processor = new singlefile.SingleFile(options);
- const preInitializationPromises = [];
- options.insertCanonicalLink = true;
- let index = 0, maxIndex = 0, initializing;
- options.onprogress = async event => {
- const { options } = event.detail;
- if (!processor.cancelled) {
- if (event.type == event.RESOURCES_INITIALIZING) {
- if (!initializing && options.insertEmbeddedScreenshotImage && options.compressContent) {
- initializing = true;
- setVisible(false);
- const screenshotBlobURI = await browser.runtime.sendMessage({
- method: "tabs.getScreenshot",
- width: document.documentElement.scrollWidth,
- height: document.documentElement.scrollHeight,
- innerHeight: globalThis.innerHeight
- });
- setVisible(true);
- onInsertingEmbeddedImage(options);
- options.embeddedImage = new Uint8Array(await (await fetchResource(screenshotBlobURI)).arrayBuffer());
- URL.revokeObjectURL(screenshotBlobURI);
- onInsertEmbeddedImage(options);
- }
- }
- if (event.type == event.RESOURCES_INITIALIZED) {
- maxIndex = event.detail.max;
- if (options.loadDeferredImages) {
- singlefile.processors.lazy.resetZoomLevel(options);
- }
- }
- if (event.type == event.RESOURCES_INITIALIZED || event.type == event.RESOURCE_LOADED) {
- if (event.type == event.RESOURCE_LOADED) {
- index++;
- }
- await browser.runtime.sendMessage({ method: "ui.processProgress", index, maxIndex });
- onLoadResource(index, maxIndex, options);
- } else if (!event.detail.frame) {
- if (event.type == event.PAGE_LOADING) ; else if (event.type == event.PAGE_LOADED) ; else if (event.type == event.STAGE_STARTED) {
- if (event.detail.step < 3) {
- onStartStage(event.detail.step, options);
- }
- } else if (event.type == event.STAGE_ENDED) {
- if (event.detail.step < 3) {
- onEndStage(event.detail.step, options);
- }
- }
- }
- }
- };
- const cancelProcessor = processor.cancel.bind(processor);
- if (options.insertEmbeddedImage && options.compressContent) {
- onInsertingEmbeddedImage(options);
- openFileInfobar = getOpenFileBar();
- const cancelled = await openFileInfobar.display();
- if (!cancelled) {
- options.embeddedImage = await openFile({ accept: "image/*" });
- openFileInfobar.hide();
- }
- onInsertEmbeddedImage(options);
- }
- if (!options.saveRawPage && !processor.cancelled) {
- let lazyLoadPromise;
- if (options.loadDeferredImages) {
- lazyLoadPromise = singlefile.processors.lazy.process(options);
- onLoadingDeferResources(options);
- lazyLoadPromise.then(() => {
- if (!processor.cancelled) {
- onLoadDeferResources(options);
- }
- });
- if (options.loadDeferredImagesBeforeFrames) {
- await lazyLoadPromise;
- }
- }
- if (!options.removeFrames && frames && globalThis.frames) {
- let frameTreePromise;
- if (options.loadDeferredImages) {
- frameTreePromise = new Promise(resolve => globalThis.setTimeout(() => resolve(frames.getAsync(options)), options.loadDeferredImagesBeforeFrames || !options.loadDeferredImages ? 0 : options.loadDeferredImagesMaxIdleTime));
- } else {
- frameTreePromise = frames.getAsync(options);
- }
- onLoadingFrames(options);
- frameTreePromise.then(() => {
- if (!processor.cancelled) {
- onLoadFrames(options);
- }
- });
- if (options.loadDeferredImagesBeforeFrames) {
- options.frames = await new Promise(resolve => {
- processor.cancel = function () {
- cancelProcessor();
- resolve([]);
- };
- frameTreePromise.then(resolve);
- });
- } else {
- preInitializationPromises.push(frameTreePromise);
- }
- }
- if (options.loadDeferredImages && !options.loadDeferredImagesBeforeFrames) {
- preInitializationPromises.push(lazyLoadPromise);
- }
- }
- if (!options.loadDeferredImagesBeforeFrames && !processor.cancelled) {
- [options.frames] = await new Promise(resolve => {
- const preInitializationAllPromises = Promise.all(preInitializationPromises);
- processor.cancel = function () {
- cancelProcessor();
- resolve([[]]);
- };
- preInitializationAllPromises.then(() => resolve(preInitializationAllPromises));
- });
- }
- if (options.delayBeforeProcessing) {
- await new Promise(resolve => setTimeout(resolve, options.delayBeforeProcessing * 1000));
- }
- framesSessionId = options.frames && options.frames.sessionId;
- const selectedFrame = options.frames && options.frames.find(frameData => frameData.requestedFrame);
- options.win = globalThis;
- if (selectedFrame) {
- options.content = selectedFrame.content;
- options.url = selectedFrame.baseURI;
- options.canvases = selectedFrame.canvases;
- options.fonts = selectedFrame.fonts;
- options.stylesheets = selectedFrame.stylesheets;
- options.images = selectedFrame.images;
- options.posters = selectedFrame.posters;
- options.videos = selectedFrame.videos;
- options.usedFonts = selectedFrame.usedFonts;
- options.shadowRoots = selectedFrame.shadowRoots;
- options.adoptedStyleSheets = selectedFrame.adoptedStyleSheets;
- } else {
- options.doc = document;
- }
- if (!processor.cancelled) {
- await processor.run();
- }
- if (framesSessionId) {
- frames.cleanup(framesSessionId);
- }
- let page;
- if (!processor.cancelled) {
- if (options.confirmInfobarContent) {
- options.infobarContent = promptMessage("Infobar content", options.infobarContent) || "";
- }
- page = await processor.getPageData();
- if (options.selected || options.optionallySelected) {
- unmarkSelection();
- }
- onEndPage();
- if (options.displayStats) {
- console.log("SingleFile stats"); // eslint-disable-line no-console
- console.table(page.stats); // eslint-disable-line no-console
- }
- }
- return page;
- }
- })();
|