moved to root
This commit is contained in:
@@ -0,0 +1,275 @@
|
||||
/*
|
||||
* 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",
|
||||
};
|
||||
Reference in New Issue
Block a user