blob: 9e06e13dc8a28e93c58e20b8d77d5110f8635214 [file] [log] [blame]
/*
* Copyright (C) 2017 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
WI.FileUtilities = class FileUtilities {
static screenshotString()
{
let date = new Date;
let values = [
date.getFullYear(),
Number.zeroPad(date.getMonth() + 1, 2),
Number.zeroPad(date.getDate(), 2),
Number.zeroPad(date.getHours(), 2),
Number.zeroPad(date.getMinutes(), 2),
Number.zeroPad(date.getSeconds(), 2),
];
return WI.UIString("Screen Shot %s-%s-%s at %s.%s.%s").format(...values);
}
static sanitizeFilename(filename)
{
return filename.replace(/:+/g, "-");
}
static inspectorURLForFilename(filename)
{
return "web-inspector:///" + encodeURIComponent(FileUtilities.sanitizeFilename(filename));
}
static canSave(saveMode)
{
console.assert(Object.values(WI.FileUtilities.SaveMode).includes(saveMode), saveMode);
return InspectorFrontendHost.canSave(saveMode);
}
static async save(saveMode, fileVariants, forceSaveAs)
{
console.assert(WI.FileUtilities.canSave(saveMode), saveMode);
console.assert(fileVariants);
if (!fileVariants) {
InspectorFrontendHost.beep();
return;
}
let isFileVariantsMode = saveMode === WI.FileUtilities.SaveMode.FileVariants;
if (isFileVariantsMode)
forceSaveAs = true;
if (typeof fileVariants.customSaveHandler === "function") {
fileVariants.customSaveHandler(forceSaveAs);
return;
}
if (!isFileVariantsMode && !Array.isArray(fileVariants))
fileVariants = [fileVariants];
console.assert(Array.isArray(fileVariants), fileVariants);
if (!Array.isArray(fileVariants)) {
InspectorFrontendHost.beep();
return;
}
let promises = fileVariants.map((fileVariant) => {
let content = fileVariant.content;
console.assert(content, fileVariant);
if (!content)
return null;
let displayType = fileVariant.displayType || "";
console.assert(!isFileVariantsMode || fileVariant.displayType, fileVariant);
if (!fileVariant.displayType && isFileVariantsMode)
return null;
let suggestedName = fileVariant.suggestedName;
if (!suggestedName) {
let url = fileVariant.url || "";
suggestedName = parseURL(url).lastPathComponent;
if (!suggestedName) {
suggestedName = WI.UIString("Untitled");
let dataURLTypeMatch = /^data:([^;]+)/.exec(url);
if (dataURLTypeMatch) {
let fileExtension = WI.fileExtensionForMIMEType(dataURLTypeMatch[1]);
if (fileExtension)
suggestedName += "." + fileExtension;
}
}
}
let url = WI.FileUtilities.inspectorURLForFilename(suggestedName);
if (typeof content === "string") {
return Promise.resolve({
displayType,
url,
content,
base64Encoded: !!fileVariant.base64Encoded,
});
}
let wrappedPromise = new WI.WrappedPromise;
let fileReader = new FileReader;
fileReader.addEventListener("loadend", () => {
wrappedPromise.resolve({
displayType,
url,
content: parseDataURL(fileReader.result).data,
base64Encoded: true,
});
});
fileReader.readAsDataURL(content);
return wrappedPromise.promise;
});
if (promises.includes(null)) {
InspectorFrontendHost.beep();
return;
}
let saveDatas = await Promise.all(promises);
console.assert(isFileVariantsMode || saveDatas.length === 1, saveDatas);
console.assert(!isFileVariantsMode || new Set(saveDatas.map((saveData) => saveData.displayType)).size === saveDatas.length, saveDatas);
console.assert(!isFileVariantsMode || new Set(saveDatas.map((saveData) => WI.urlWithoutExtension(saveData.url))).size === 1, saveDatas);
InspectorFrontendHost.save(saveDatas, !!forceSaveAs);
}
static import(callback, {multiple} = {})
{
let inputElement = document.createElement("input");
inputElement.type = "file";
inputElement.value = null;
inputElement.multiple = !!multiple;
inputElement.addEventListener("change", (event) => {
callback(inputElement.files);
});
inputElement.click();
// Cache the last used import element so that it doesn't get GCd while the native file
// picker is shown, which would prevent the "change" event listener from firing.
FileUtilities.importInputElement = inputElement;
}
static importText(callback, options = {})
{
FileUtilities.import((files) => {
FileUtilities.readText(files, callback);
}, options);
}
static importJSON(callback, options = {})
{
FileUtilities.import((files) => {
FileUtilities.readJSON(files, callback);
}, options);
}
static importData(callback, options = {})
{
FileUtilities.import((files) => {
FileUtilities.readData(files, callback);
}, options);
}
static async readText(fileOrList, callback)
{
await FileUtilities._read(fileOrList, async (file, result) => {
await new Promise((resolve, reject) => {
let reader = new FileReader;
reader.addEventListener("loadend", (event) => {
result.text = reader.result;
resolve(event);
});
reader.addEventListener("error", reject);
reader.readAsText(file);
});
}, callback);
}
static async readJSON(fileOrList, callback)
{
await WI.FileUtilities.readText(fileOrList, async (result) => {
if (result.text && !result.error) {
try {
result.json = JSON.parse(result.text);
} catch (e) {
result.error = e;
}
}
await callback(result);
});
}
static async readData(fileOrList, callback)
{
await FileUtilities._read(fileOrList, async (file, result) => {
await new Promise((resolve, reject) => {
let reader = new FileReader;
reader.addEventListener("loadend", (event) => {
let {mimeType, base64, data} = parseDataURL(reader.result);
// In case no mime type was determined, try to derive one from the file extension.
if (!mimeType || mimeType === "text/plain") {
let extension = WI.fileExtensionForFilename(result.filename);
if (extension)
mimeType = WI.mimeTypeForFileExtension(extension);
}
result.mimeType = mimeType;
result.base64Encoded = base64;
result.content = data;
resolve(event);
});
reader.addEventListener("error", reject);
reader.readAsDataURL(file);
});
}, callback);
}
// Private
static async _read(fileOrList, operation, callback)
{
console.assert(fileOrList instanceof File || fileOrList instanceof FileList);
let files = [];
if (fileOrList instanceof File)
files.push(fileOrList);
else if (fileOrList instanceof FileList)
files = Array.from(fileOrList);
for (let file of files) {
let result = {
filename: file.name,
};
try {
await operation(file, result);
} catch (e) {
result.error = e;
}
await callback(result);
}
}
};
// Keep in sync with `InspectorFrontendClient::SaveMode` and `InspectorFrontendHost::SaveMode`.
WI.FileUtilities.SaveMode = {
SingleFile: "single-file",
FileVariants: "file-variants",
};