slate/common/user-behaviors.js

399 lines
11 KiB
JavaScript

import * as Websockets from "~/common/browser-websockets";
import * as Credentials from "~/common/credentials";
import * as Actions from "~/common/actions";
import * as Window from "~/common/window";
import * as Validations from "~/common/validations";
import * as Strings from "~/common/strings";
import * as Store from "~/common/store";
import * as FileUtilities from "~/common/file-utilities";
import * as Events from "~/common/custom-events";
import Cookies from "universal-cookie";
import JSZip from "jszip";
import { v4 as uuid } from "uuid";
import { saveAs } from "file-saver";
//NOTE(martina): this file is for utility *API-calling* functions
//For non API related utility functions, see common/utilities.js
//And for uploading related utility functions, see common/file-utilities.js
const cookies = new Cookies();
export const authenticate = async (state) => {
// NOTE(jim): Kills existing session cookie if there is one.
const jwt = cookies.get(Credentials.session.key);
if (jwt) {
cookies.remove(Credentials.session.key);
}
let response = await Actions.signIn(state);
if (Events.hasError(response)) {
return false;
}
if (response.token) {
// NOTE(jim):
// + One week.
// + Only requests to the same site.
// + Not using sessionStorage so the cookie doesn't leave when the browser dies.
cookies.set(Credentials.session.key, response.token, true, {
path: "/",
maxAge: 3600 * 24 * 7,
sameSite: "strict",
});
}
return response;
};
// NOTE(jim): Signs a user out and redirects to the sign in screen.
export const signOut = async ({ viewer }) => {
let wsclient = Websockets.getClient();
if (wsclient) {
wsclient.send(JSON.stringify({ type: "UNSUBSCRIBE_VIEWER", data: viewer }));
await Websockets.deleteClient();
wsclient = null;
}
const jwt = cookies.get(Credentials.session.key);
if (jwt) {
cookies.remove(Credentials.session.key);
}
window.location.replace("/_");
return;
};
// NOTE(jim): Permanently deletes you, forever.
export const deleteMe = async ({ viewer }) => {
const message = "Do you really want to delete your account? It will be permanently removed";
if (!window.confirm(message)) {
return false;
}
await Actions.updateSearch("delete-user");
let response = await Actions.deleteViewer();
if (Events.hasError(response)) {
return response;
}
await signOut({ viewer });
let wsclient = Websockets.getClient();
if (wsclient) {
await Websockets.deleteClient();
wsclient = null;
}
return response;
};
export const hydrate = async () => {
const response = await Actions.hydrateAuthenticatedUser();
if (Events.hasError(response)) {
return null;
}
return JSON.parse(JSON.stringify(response.data));
};
export const formatPastedImages = ({ clipboardItems }) => {
let files = [];
let fileLoading = {};
for (let i = 0; i < clipboardItems.length; i++) {
// Note(Amine): skip content if it's not an image
if (clipboardItems[i].type.indexOf("image") === -1) continue;
const file = clipboardItems[i].getAsFile();
files.push(file);
fileLoading[`${file.lastModified}-${file.name}`] = {
name: file.name,
loaded: 0,
total: file.size,
};
}
return { fileLoading, toUpload: files };
};
export const formatDroppedFiles = async ({ dataTransfer }) => {
// NOTE(jim): If this is true, then drag and drop came from a slate object.
const data = dataTransfer.getData("slate-object-drag-data");
if (data) {
return;
}
const files = [];
let fileLoading = {};
if (dataTransfer.items && dataTransfer.items.length) {
for (var i = 0; i < dataTransfer.items.length; i++) {
const data = dataTransfer.items[i];
let file = null;
if (data.kind === "file") {
file = data.getAsFile();
} else if (data.kind == "string" && data.type == "text/uri-list") {
try {
const dataAsString = new Promise((resolve, reject) =>
data.getAsString((d) => resolve(d))
);
const resp = await fetch(await dataAsString);
const blob = resp.blob();
file = new File(blob, `data-${uuid()}`);
file.name = `data-${uuid()}`;
} catch (e) {
Events.dispatchMessage({
message: "File type not supported. Please try a different file",
});
return { error: true };
}
}
files.push(file);
fileLoading[`${file.lastModified}-${file.name}`] = {
name: file.name,
loaded: 0,
total: file.size,
};
}
}
if (!files.length) {
Events.dispatchMessage({ message: "File type not supported. Please try a different file" });
}
return { fileLoading, files, numFailed: dataTransfer.items.length - files.length };
};
export const formatUploadedFiles = ({ files }) => {
let toUpload = [];
let fileLoading = {};
for (let i = 0; i < files.length; i++) {
let file = files[i];
if (!file) {
continue;
}
toUpload.push(file);
fileLoading[`${file.lastModified}-${file.name}`] = {
name: file.name,
loaded: 0,
total: file.size,
};
}
if (!toUpload.length) {
Events.dispatchMessage({ message: "We could not find any files to upload." });
return false;
}
return { toUpload, fileLoading, numFailed: files.length - toUpload.length };
};
export const uploadImage = async (file, resources, excludeFromLibrary) => {
if (!file) {
Events.dispatchMessage({ message: "Something went wrong with the upload. Please try again" });
return;
}
if (!Validations.isPreviewableImage(file.type)) {
Events.dispatchMessage({ message: "Upload failed. Only images and gifs are allowed" });
return;
}
const response = await FileUtilities.upload({ file, routes: resources });
if (Events.hasError(response)) {
return false;
}
return response;
};
export const deleteFiles = async (fileIds = [], noAlert) => {
let ids = Array.isArray(fileIds) ? fileIds : [fileIds];
const response = await Actions.deleteFiles({ ids });
if (!noAlert) {
if (Events.hasError(response)) {
return false;
}
Events.dispatchMessage({ message: "Files successfully deleted!", status: "INFO" });
return response;
}
};
export const removeFromSlate = async ({ slate, ids }) => {
const response = await Actions.removeFileFromSlate({
slateId: slate.id,
ids,
});
if (Events.hasError(response)) {
return false;
}
return response;
};
export const addToSlate = async ({ slate, files }) => {
const addResponse = await Actions.addFileToSlate({ slate, files });
if (Events.hasError(addResponse)) {
return false;
}
const { added, skipped } = addResponse;
let message = Strings.formatAsUploadMessage(added, skipped, true);
Events.dispatchMessage({ message, status: !added ? null : "INFO" });
return true;
};
export const saveCopy = async ({ files }) => {
let response = await Actions.saveCopy({ files });
if (Events.hasError(response)) {
return false;
}
let message = Strings.formatAsUploadMessage(response.data.added, response.data.skipped);
Events.dispatchMessage({ message, status: !response.data.added ? null : "INFO" });
return response;
};
export const download = (file) => {
let uri = Strings.getURLfromCID(file.cid);
Window.saveAs(uri, file.filename);
};
export const downloadZip = async (file) => {
try {
const { data } = await Actions.getZipFilePaths(file);
const filesPaths = data.filesPaths.map((item) => item.replace(`/${file.id}/`, ""));
const baseUrl = Strings.getURLfromCID(file.cid);
const zipFileName = file.filename;
let zip = new JSZip();
await Promise.all(
filesPaths.map(async (filePath) => {
let url = `${baseUrl}/${filePath}`;
const blob = await Window.getBlobFromUrl(url);
zip.file(filePath, blob);
})
);
zip.generateAsync({ type: "blob" }).then((blob) => {
saveAs(blob, zipFileName);
});
return {
decorator: "UNITY_ZIP_DOWNLOAD_SUCCESS",
error: false,
};
} catch (e) {
return {
decorator: "UNITY_ZIP_DOWNLOAD_FAILED",
error: true,
};
}
};
const _nativeDownload = ({ url, onError }) => {
const iframe = document.createElement("iframe");
iframe.style.display = "none";
iframe.src = url;
const ERROR_MESSAGE = "SLATE_DOWNLOAD_ERROR";
const handleIframeErrors = (e) => {
if (e.data === ERROR_MESSAGE && onError) {
onError(e.data);
}
};
window.addEventListener("message", handleIframeErrors);
iframe.onload = (e) => window.removeEventListener("message", handleIframeErrors);
document.body.appendChild(iframe);
};
export const compressAndDownloadFiles = async ({ files, name = "slate.zip", resourceURI }) => {
const errorMessage = "Something went wrong with the download. Please try again";
try {
if (!(files && files.length > 0)) {
Events.dispatchMessage({ message: "No files in collection to download" });
return;
}
Events.dispatchMessage({ message: "We're preparing your files to download", status: "INFO" });
let downloadFiles = [];
for (const file of files) {
if (file.type === "application/unity") {
const { data } = await Actions.getZipFilePaths(file);
const unityFiles = data.filesPaths.map((item) => ({
url: item.replace(`/${file.id}/`, `${Strings.getURLfromCID(file.cid)}/`),
name: item.replace(`/${file.id}/`, `/${file.filename}/`),
}));
downloadFiles = downloadFiles.concat(unityFiles);
continue;
}
downloadFiles.push({
name: file.filename,
url: Strings.getURLfromCID(file.cid),
});
}
const res = await Actions.createZipToken({ files: downloadFiles, resourceURI });
const downloadLink = Actions.downloadZip({ token: res.data.token, name, resourceURI });
await _nativeDownload({
url: downloadLink,
onError: (err) =>
Events.dispatchMessage({
message: errorMessage,
}),
});
} catch (e) {
console.error(e);
Events.dispatchMessage({ message: errorMessage });
}
};
// export const createSlate = async (data) => {
// let response = await Actions.createSlate({
// name: data.name,
// isPublic: data.isPublic,
// body: data.body,
// });
// return response;
// }
// export const createWalletAddress = async (data) => {
// let response = await Actions.updateViewer({
// type: "CREATE_FILECOIN_ADDRESS",
// address: {
// name: data.name,
// type: data.wallet_type,
// makeDefault: data.makeDefault,
// },
// });
// return response;
// }
// export const sendWalletAddressFilecoin = (data) => {
// let response = await Actions.sendFilecoin({
// source: data.source,
// target: data.target,
// amount: data.amount,
// });
// return response;
// }