mirror of
https://github.com/makeplane/plane.git
synced 2025-12-22 22:59:33 +01:00
* fix: svg not supported in image uploads * fix: svg image file error message fixed * feat: add custom image node for uploads * fix: combine two extensions * fix: added new image extension to backend * fix: type errors * style: image drop node * style: image resize handler * fix: removed unused stuff * fix: types of updateAttributes * fix: image insertion at pos and loading effect added * fix: resize image real time sync * fix: drag drop menu * feat: custom image component editor * fix: reverted back styles * fix: reverted back document info changes * fix: css image css * style: image selected and hover states * refactor: custom image extension folder structure * style: read-only image * chore: remove file handler * fix: fixed multi time file opener * fix: editor readonly content set properly * fix: old images not rendered as new ones * fix: drop upload fixed * chore: remove console logs * fix: src of image node as dependency * fix: helper library build fix * fix: improved reflow/layout and fixed resizing --------- Co-authored-by: Aaryan Khandelwal <aaryankhandu123@gmail.com>
81 lines
2.6 KiB
TypeScript
81 lines
2.6 KiB
TypeScript
import DOMPurify from "dompurify";
|
|
|
|
export const addSpaceIfCamelCase = (str: string) => str.replace(/([a-z])([A-Z])/g, "$1 $2");
|
|
|
|
const fallbackCopyTextToClipboard = (text: string) => {
|
|
const textArea = document.createElement("textarea");
|
|
textArea.value = text;
|
|
|
|
// Avoid scrolling to bottom
|
|
textArea.style.top = "0";
|
|
textArea.style.left = "0";
|
|
textArea.style.position = "fixed";
|
|
|
|
document.body.appendChild(textArea);
|
|
textArea.focus();
|
|
textArea.select();
|
|
|
|
try {
|
|
// FIXME: Even though we are using this as a fallback, execCommand is deprecated 👎. We should find a better way to do this.
|
|
// https://developer.mozilla.org/en-US/docs/Web/API/Document/execCommand
|
|
document.execCommand("copy");
|
|
} catch (err) {}
|
|
|
|
document.body.removeChild(textArea);
|
|
};
|
|
|
|
export const copyTextToClipboard = async (text: string) => {
|
|
if (!navigator.clipboard) {
|
|
fallbackCopyTextToClipboard(text);
|
|
return;
|
|
}
|
|
await navigator.clipboard.writeText(text);
|
|
};
|
|
|
|
/**
|
|
* @returns {boolean} true if email is valid, false otherwise
|
|
* @description Returns true if email is valid, false otherwise
|
|
* @param {string} email string to check if it is a valid email
|
|
* @example checkEmailIsValid("hello world") => false
|
|
* @example checkEmailIsValid("example@plane.so") => true
|
|
*/
|
|
export const checkEmailValidity = (email: string): boolean => {
|
|
if (!email) return false;
|
|
|
|
const isEmailValid =
|
|
/^(([^<>()[\]\\.,;:\s@\"]+(\.[^<>()[\]\\.,;:\s@\"]+)*)|(\".+\"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/.test(
|
|
email
|
|
);
|
|
|
|
return isEmailValid;
|
|
};
|
|
|
|
export const isEmptyHtmlString = (htmlString: string, allowedHTMLTags: string[] = []) => {
|
|
// Remove HTML tags using regex
|
|
const cleanText = DOMPurify.sanitize(htmlString, { ALLOWED_TAGS: allowedHTMLTags });
|
|
// Trim the string and check if it's empty
|
|
return cleanText.trim() === "";
|
|
};
|
|
|
|
/**
|
|
* @description this function returns whether a comment is empty or not by checking for the following conditions-
|
|
* 1. If comment is undefined
|
|
* 2. If comment is an empty string
|
|
* 3. If comment is "<p></p>"
|
|
* @param {string | undefined} comment
|
|
* @returns {boolean}
|
|
*/
|
|
export const isCommentEmpty = (comment: string | undefined): boolean => {
|
|
// return true if comment is undefined
|
|
if (!comment) return true;
|
|
return (
|
|
comment?.trim() === "" ||
|
|
comment === "<p></p>" ||
|
|
isEmptyHtmlString(comment ?? "", ["img", "mention-component", "image-component"])
|
|
);
|
|
};
|
|
|
|
export const replaceUnderscoreIfSnakeCase = (str: string) => str.replace(/_/g, " ");
|
|
|
|
export const capitalizeFirstLetter = (str: string) => str.charAt(0).toUpperCase() + str.slice(1);
|