webamp/js/fileUtils.ts
2019-03-19 06:55:24 -07:00

138 lines
4.0 KiB
TypeScript

import invariant from "invariant";
import { IMusicMetadataBrowserApi } from "./types";
import { IAudioMetadata } from "music-metadata-browser"; // Import music-metadata type definitions
type MediaDataType = string | ArrayBuffer | Blob;
export function genMediaTags(
file: MediaDataType,
musicMetadata: IMusicMetadataBrowserApi
): Promise<IAudioMetadata> {
invariant(
file != null,
"Attempted to get the tags of media file without passing a file"
);
const options = {
duration: true,
skipPostHeaders: true, // avoid unnecessary data to be read
};
if (typeof file === "string") {
return musicMetadata.fetchFromUrl(file, options);
}
// Assume Blob
return musicMetadata.parseBlob(file as Blob, options);
}
export function genMediaDuration(url: string): Promise<number> {
invariant(
typeof url === "string",
"Attempted to get the duration of media file without passing a url"
);
return new Promise((resolve, reject) => {
// TODO: Does this actually stop downloading the file once it's
// got the duration?
const audio = document.createElement("audio");
audio.crossOrigin = "anonymous";
const durationChange = () => {
resolve(audio.duration);
audio.removeEventListener("durationchange", durationChange);
audio.src = "";
// TODO: Not sure if this really gets cleaned up.
};
audio.addEventListener("durationchange", durationChange);
audio.addEventListener("error", e => {
reject(e);
});
audio.src = url;
});
}
export async function genArrayBufferFromFileReference(
fileReference: File
): Promise<any> {
invariant(
fileReference != null,
"Attempt to get an ArrayBuffer without assigning a fileReference"
);
return new Promise((resolve, reject) => {
const reader = new FileReader();
reader.onload = () => {
resolve(reader.result as ArrayBuffer);
};
reader.onerror = reject;
reader.readAsArrayBuffer(fileReference);
});
}
export async function genStringFromFileReference(
fileReference: File
): Promise<string> {
return new Promise((resolve, reject) => {
const reader = new FileReader();
reader.onload = () => {
resolve(reader.result as string);
};
reader.onerror = reject;
reader.readAsText(fileReference);
});
}
interface PromptForFileReferenceOptions {
accept?: string | null;
directory?: boolean;
}
export async function promptForFileReferences(
{ accept, directory = false }: PromptForFileReferenceOptions = {
accept: null,
directory: false,
}
): Promise<FileList> {
return new Promise<FileList>(resolve => {
// Does this represent a memory leak somehow?
// Can this fail? Do we ever reject?
const fileInput = document.createElement("input");
if (accept) fileInput.setAttribute("accept", accept);
fileInput.type = "file";
fileInput.multiple = true;
// @ts-ignore Non-standard
fileInput.webkitdirectory = directory;
// @ts-ignore Non-standard
fileInput.directory = directory;
// @ts-ignore Non-standard
fileInput.mozdirectory = directory;
// Not entirely sure why this is needed, since the input
// was just created, but somehow this helps prevent change
// events from getting swallowed.
// https://stackoverflow.com/a/12102992/1263117
// @ts-ignore Technically you can't set this to null, it has to be a string.
// But I don't feel like retesting it, so I'll leave it as null
fileInput.value = null;
fileInput.addEventListener("change", (e: Event) => {
const files = (<HTMLInputElement>e.target).files;
resolve(files as FileList);
});
fileInput.click();
});
}
function urlIsBlobUrl(url: string): boolean {
return /^blob:/.test(url);
}
// This is not perfect, but... meh: https://stackoverflow.com/a/36756650/1263117
export function filenameFromUrl(url: string): string | null {
if (urlIsBlobUrl(url)) {
return null;
}
const lastSegment = url.split("/").pop();
if (lastSegment == null) {
return null;
}
return lastSegment.split("#")[0].split("?")[0];
}