mirror of https://github.com/usememos/memos
235 lines
5.9 KiB
TypeScript
235 lines
5.9 KiB
TypeScript
import { assign } from "lodash-es";
|
|
|
|
export function getNowTimeStamp(): number {
|
|
return Date.now();
|
|
}
|
|
|
|
export function getOSVersion(): "Windows" | "MacOS" | "Linux" | "Unknown" {
|
|
const appVersion = navigator.userAgent;
|
|
let detectedOS: "Windows" | "MacOS" | "Linux" | "Unknown" = "Unknown";
|
|
|
|
if (appVersion.indexOf("Win") != -1) {
|
|
detectedOS = "Windows";
|
|
} else if (appVersion.indexOf("Mac") != -1) {
|
|
detectedOS = "MacOS";
|
|
} else if (appVersion.indexOf("Linux") != -1) {
|
|
detectedOS = "Linux";
|
|
}
|
|
|
|
return detectedOS;
|
|
}
|
|
|
|
export function getTimeStampByDate(t: Date | number | string): number {
|
|
if (typeof t === "string") {
|
|
t = t.replaceAll("-", "/");
|
|
}
|
|
const d = new Date(t);
|
|
|
|
return d.getTime();
|
|
}
|
|
|
|
export function getDateStampByDate(t: Date | number | string): number {
|
|
const d = new Date(getTimeStampByDate(t));
|
|
|
|
return new Date(d.getFullYear(), d.getMonth(), d.getDate()).getTime();
|
|
}
|
|
|
|
export function getDateString(t: Date | number | string): string {
|
|
const d = new Date(getTimeStampByDate(t));
|
|
|
|
const year = d.getFullYear();
|
|
const month = d.getMonth() + 1;
|
|
const date = d.getDate();
|
|
|
|
return `${year}/${month}/${date}`;
|
|
}
|
|
|
|
export function getTimeString(t: Date | number | string): string {
|
|
const d = new Date(getTimeStampByDate(t));
|
|
|
|
const hours = d.getHours();
|
|
const mins = d.getMinutes();
|
|
|
|
const hoursStr = hours < 10 ? "0" + hours : hours;
|
|
const minsStr = mins < 10 ? "0" + mins : mins;
|
|
|
|
return `${hoursStr}:${minsStr}`;
|
|
}
|
|
|
|
// For example: 2021-4-8 17:52:17
|
|
export function getDateTimeString(t: Date | number | string): string {
|
|
const d = new Date(getTimeStampByDate(t));
|
|
|
|
const year = d.getFullYear();
|
|
const month = d.getMonth() + 1;
|
|
const date = d.getDate();
|
|
const hours = d.getHours();
|
|
const mins = d.getMinutes();
|
|
const secs = d.getSeconds();
|
|
|
|
const monthStr = month < 10 ? "0" + month : month;
|
|
const dateStr = date < 10 ? "0" + date : date;
|
|
const hoursStr = hours < 10 ? "0" + hours : hours;
|
|
const minsStr = mins < 10 ? "0" + mins : mins;
|
|
const secsStr = secs < 10 ? "0" + secs : secs;
|
|
|
|
return `${year}/${monthStr}/${dateStr} ${hoursStr}:${minsStr}:${secsStr}`;
|
|
}
|
|
|
|
export function dedupe<T>(data: T[]): T[] {
|
|
return Array.from(new Set(data));
|
|
}
|
|
|
|
export function dedupeObjectWithId<T extends { id: string | number }>(data: T[]): T[] {
|
|
const idSet = new Set<string | number>();
|
|
const result = [];
|
|
|
|
for (const d of data) {
|
|
if (!idSet.has(d.id)) {
|
|
idSet.add(d.id);
|
|
result.push(d);
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
export function debounce(fn: FunctionType, delay: number) {
|
|
let timer: number | null = null;
|
|
|
|
return () => {
|
|
if (timer) {
|
|
clearTimeout(timer);
|
|
timer = setTimeout(fn, delay);
|
|
} else {
|
|
timer = setTimeout(fn, delay);
|
|
}
|
|
};
|
|
}
|
|
|
|
export function throttle(fn: FunctionType, delay: number) {
|
|
let valid = true;
|
|
|
|
return () => {
|
|
if (!valid) {
|
|
return false;
|
|
}
|
|
valid = false;
|
|
setTimeout(() => {
|
|
fn();
|
|
valid = true;
|
|
}, delay);
|
|
};
|
|
}
|
|
|
|
export function filterObjectNullKeys(object: KVObject): KVObject {
|
|
if (!object) {
|
|
return {};
|
|
}
|
|
|
|
const finalObject: KVObject = {};
|
|
const keys = Object.keys(object).sort();
|
|
|
|
for (const key of keys) {
|
|
const val = object[key];
|
|
if (typeof val === "object") {
|
|
const temp = filterObjectNullKeys(JSON.parse(JSON.stringify(val)));
|
|
if (temp && Object.keys(temp).length > 0) {
|
|
finalObject[key] = temp;
|
|
}
|
|
} else {
|
|
if (val) {
|
|
finalObject[key] = val;
|
|
}
|
|
}
|
|
}
|
|
|
|
return finalObject;
|
|
}
|
|
|
|
export async function copyTextToClipboard(text: string) {
|
|
if (navigator.clipboard && navigator.clipboard.writeText) {
|
|
try {
|
|
await navigator.clipboard.writeText(text);
|
|
} catch (error: unknown) {
|
|
console.warn("Copy to clipboard failed.", error);
|
|
}
|
|
} else {
|
|
console.warn("Copy to clipboard failed, methods not supports.");
|
|
}
|
|
}
|
|
|
|
export function getImageSize(src: string): Promise<{ width: number; height: number }> {
|
|
return new Promise((resolve) => {
|
|
const imgEl = new Image();
|
|
|
|
imgEl.onload = () => {
|
|
const { width, height } = imgEl;
|
|
|
|
if (width > 0 && height > 0) {
|
|
resolve({ width, height });
|
|
} else {
|
|
resolve({ width: 0, height: 0 });
|
|
}
|
|
};
|
|
|
|
imgEl.onerror = () => {
|
|
resolve({ width: 0, height: 0 });
|
|
};
|
|
|
|
imgEl.className = "hidden";
|
|
imgEl.src = src;
|
|
document.body.appendChild(imgEl);
|
|
imgEl.remove();
|
|
});
|
|
}
|
|
|
|
export const getElementBounding = (element: HTMLElement, relativeEl?: HTMLElement) => {
|
|
const scrollTop = window.pageYOffset || document.documentElement.scrollTop || document.body.scrollTop;
|
|
const scrollLeft = window.pageXOffset || document.documentElement.scrollLeft || document.body.scrollLeft;
|
|
|
|
relativeEl = relativeEl || document.body;
|
|
|
|
const elementRect = element.getBoundingClientRect();
|
|
const relativeElRect = relativeEl.getBoundingClientRect();
|
|
const relativeElPosition = window.getComputedStyle(relativeEl).getPropertyValue("position");
|
|
|
|
const bounding = {
|
|
width: elementRect.width,
|
|
height: elementRect.height,
|
|
};
|
|
|
|
if ((relativeEl.tagName !== "BODY" && relativeElPosition === "relative") || relativeElPosition === "sticky") {
|
|
return assign(bounding, {
|
|
top: elementRect.top - relativeElRect.top,
|
|
left: elementRect.left - relativeElRect.left,
|
|
});
|
|
}
|
|
|
|
const isElementFixed = (element: HTMLElement): boolean => {
|
|
const parentNode = element.parentNode;
|
|
|
|
if (!parentNode || parentNode.nodeName === "HTML") {
|
|
return false;
|
|
}
|
|
|
|
if (window.getComputedStyle(element).getPropertyValue("position") === "fixed") {
|
|
return true;
|
|
}
|
|
|
|
return isElementFixed(parentNode as HTMLElement);
|
|
};
|
|
|
|
if (isElementFixed(element)) {
|
|
return assign(bounding, {
|
|
top: elementRect.top,
|
|
left: elementRect.left,
|
|
});
|
|
}
|
|
|
|
return assign(bounding, {
|
|
top: elementRect.top + scrollTop,
|
|
left: elementRect.left + scrollLeft,
|
|
});
|
|
};
|