extractify.zip/composables/files-manager.ts
2023-06-06 04:08:33 +03:00
Ask

170 lines
6.1 KiB
TypeScript

{0df85edab39ae8faf3c29d7f4e645083147d3733 true 6211 files-manager.ts 0xc002e3ac40}

// class to load 7zip wasm module
// and extract files from archive
import SevenZip, { SevenZipModule } from "7z-wasm";
export class FilesManager {
sevenZip?: SevenZipModule;
consoleOutputBuffer: string[] = [];
path: Ref<string> = useSelectedPath();
constructor(private filesList: Ref<any[]>) {
this.init();
}
async init() {
this.sevenZip = await SevenZip({
wasmBinary: await fetch("/7zz.wasm").then((res) => res.arrayBuffer()),
print: (text) => {
if (text.lastIndexOf("\b")) {
text = text.substring(text.lastIndexOf("\b") + 1);
}
this.consoleOutputBuffer.push(text);
},
});
}
execute(commands: string[]) {
if (!this.sevenZip) return;
this.consoleOutputBuffer = [];
this.sevenZip.callMain(commands);
return this.consoleOutputBuffer;
}
async loadArchive(file: File) {
if (!this.sevenZip) return;
const archiveName = file.name;
const stream = this.sevenZip.FS.open(archiveName, "w+");
let archiveData = new Uint8Array(await file.arrayBuffer());
this.sevenZip.FS.write(stream, archiveData, 0, archiveData.byteLength);
this.sevenZip.FS.close(stream);
// 7zip get files list
let filesString = this.execute(["l", "-ba", archiveName]);
// parse files list
let unorganizedFiles = filesString!.map((fileString) => {
let file: RegExpMatchArray = /[\s+|(\d{4}-\d{2}-\d{2}\s+\d{2}:\d{2}:\d{2})]\s+(?<type>[AD.]+)\s+(?<size>\d+)\s+(?<compressed>\d+)\s+(?<path>.+)[\n\r]{0,}/.exec(fileString)!;
let isFolder = file.groups!.type?.indexOf("D") > -1 ? true : false;
return {
name: file.groups!.path.lastIndexOf('/') > -1 ? file.groups!.path.substring(file.groups!.path.lastIndexOf('/') + 1) : file.groups!.path,
path: `/${file.groups!.path}`,
isFolder: isFolder ? true : false,
content: isFolder ? [] as any[] : undefined,
}
});
// sort unorganized files by depth
unorganizedFiles = unorganizedFiles.sort((a, b) => {
const aPathArrayLength = (a.path.match(/\//g) || []).length;
const bPathArrayLength = (b.path.match(/\//g) || []).length;
if (aPathArrayLength > bPathArrayLength) return -1;
if (aPathArrayLength < bPathArrayLength) return 1;
return 0;
});
// sort files and folder inside each folder
const pathArrays: any = {};
for (let file of unorganizedFiles) {
// get parent folder file
const parentPath = file.path.substring(0, file.path.lastIndexOf("/"));
let parentFolderFile = pathArrays[parentPath] || unorganizedFiles.find(_file => _file.path === parentPath);
if (!parentFolderFile) {
unorganizedFiles.push({
name: parentPath.substring(parentPath.lastIndexOf('/') + 1),
path: parentPath,
isFolder: true,
content: [],
});
parentFolderFile = unorganizedFiles.find(_file => _file.path === parentPath);
};
// add file to parent folder content
if (!parentFolderFile.content) parentFolderFile.content = [];
parentFolderFile.content.push(file);
// cache split path array to avoid calling split() multiple times
const pathArray = pathArrays[file.path] || file.path.split("/");
pathArrays[file.path] = pathArray;
}
// remove folders from root
const files = unorganizedFiles.filter(file => (file.path.match(/\//g) || []).length == 1).sort((a:any, b:any) => {
// sort by folder and from a to z
if (a.isFolder && !b.isFolder) return -1;
if (!a.isFolder && b.isFolder) return 1;
if (a.name < b.name) return -1;
if (a.name > b.name) return 1;
return 0;
});
this.filesList.value = files;
return files;
}
// wait until console output is empty
async waitConsoleOutput() {
if (!this.sevenZip) return;
let breakLoop = false;
let lastLength = this.consoleOutputBuffer.length;
while (!breakLoop) {
await new Promise((resolve) => {
if (lastLength == this.consoleOutputBuffer.length) {
breakLoop = true;
return;
}
lastLength = this.consoleOutputBuffer.length;
setTimeout(resolve, 10)
});
}
return;
}
getFile(path: string, innerList = undefined): any {
if (path == "/") {
return {
content: this.filesList.value.sort((a:any, b:any) => {
// sort by folder and from a to z
if (a.isFolder && !b.isFolder) return -1;
if (!a.isFolder && b.isFolder) return 1;
if (a.name < b.name) return -1;
if (a.name > b.name) return 1;
return 0;
}),
isFolder: this.filesList.value.length ? true : false,
};
}
for (const file of (innerList || this.filesList.value)) {
if (file.path == path) {
file.content = file.content?.sort((a:any, b:any) => {
// sort by folder and from a to z
if (a.isFolder && !b.isFolder) return -1;
if (!a.isFolder && b.isFolder) return 1;
if (a.name < b.name) return -1;
if (a.name > b.name) return 1;
return 0;
});
return file;
}
if (file.isFolder && path.includes(file.path)) {
const recursiveFile = this.getFile(path, file.content);
if (recursiveFile) {
return recursiveFile;
}
}
}
return undefined;
}
}