mirror of
				https://github.com/excalidraw/excalidraw.git
				synced 2025-10-25 00:44:38 +02:00 
			
		
		
		
	 d6cd8b78f1
			
		
	
	d6cd8b78f1
	
	
	
		
			
			* feat: decouple package deps and introduce yarn workspaces * update root directory * fix * fix scripts * fix lint * update path in scripts * remove yarn.lock files from packages * ignore workspace * dummy * dummy * remove comment check * revert workflow changes * ignore ws when installing gh actions * remove log * update path * fix * fix typo
		
			
				
	
	
		
			91 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			91 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import { useEffect, useState } from "react";
 | |
| import { debounce, getVersion, nFormatter } from "../packages/excalidraw/utils";
 | |
| import {
 | |
|   getElementsStorageSize,
 | |
|   getTotalStorageSize,
 | |
| } from "./data/localStorage";
 | |
| import { DEFAULT_VERSION } from "../packages/excalidraw/constants";
 | |
| import { t } from "../packages/excalidraw/i18n";
 | |
| import { copyTextToSystemClipboard } from "../packages/excalidraw/clipboard";
 | |
| import { NonDeletedExcalidrawElement } from "../packages/excalidraw/element/types";
 | |
| import { UIAppState } from "../packages/excalidraw/types";
 | |
| 
 | |
| type StorageSizes = { scene: number; total: number };
 | |
| 
 | |
| const STORAGE_SIZE_TIMEOUT = 500;
 | |
| 
 | |
| const getStorageSizes = debounce((cb: (sizes: StorageSizes) => void) => {
 | |
|   cb({
 | |
|     scene: getElementsStorageSize(),
 | |
|     total: getTotalStorageSize(),
 | |
|   });
 | |
| }, STORAGE_SIZE_TIMEOUT);
 | |
| 
 | |
| type Props = {
 | |
|   setToast: (message: string) => void;
 | |
|   elements: readonly NonDeletedExcalidrawElement[];
 | |
|   appState: UIAppState;
 | |
| };
 | |
| const CustomStats = (props: Props) => {
 | |
|   const [storageSizes, setStorageSizes] = useState<StorageSizes>({
 | |
|     scene: 0,
 | |
|     total: 0,
 | |
|   });
 | |
| 
 | |
|   useEffect(() => {
 | |
|     getStorageSizes((sizes) => {
 | |
|       setStorageSizes(sizes);
 | |
|     });
 | |
|   }, [props.elements, props.appState]);
 | |
|   useEffect(() => () => getStorageSizes.cancel(), []);
 | |
| 
 | |
|   const version = getVersion();
 | |
|   let hash;
 | |
|   let timestamp;
 | |
| 
 | |
|   if (version !== DEFAULT_VERSION) {
 | |
|     timestamp = version.slice(0, 16).replace("T", " ");
 | |
|     hash = version.slice(21);
 | |
|   } else {
 | |
|     timestamp = t("stats.versionNotAvailable");
 | |
|   }
 | |
| 
 | |
|   return (
 | |
|     <>
 | |
|       <tr>
 | |
|         <th colSpan={2}>{t("stats.storage")}</th>
 | |
|       </tr>
 | |
|       <tr>
 | |
|         <td>{t("stats.scene")}</td>
 | |
|         <td>{nFormatter(storageSizes.scene, 1)}</td>
 | |
|       </tr>
 | |
|       <tr>
 | |
|         <td>{t("stats.total")}</td>
 | |
|         <td>{nFormatter(storageSizes.total, 1)}</td>
 | |
|       </tr>
 | |
|       <tr>
 | |
|         <th colSpan={2}>{t("stats.version")}</th>
 | |
|       </tr>
 | |
|       <tr>
 | |
|         <td
 | |
|           colSpan={2}
 | |
|           style={{ textAlign: "center", cursor: "pointer" }}
 | |
|           onClick={async () => {
 | |
|             try {
 | |
|               await copyTextToSystemClipboard(getVersion());
 | |
|               props.setToast(t("toast.copyToClipboard"));
 | |
|             } catch {}
 | |
|           }}
 | |
|           title={t("stats.versionCopy")}
 | |
|         >
 | |
|           {timestamp}
 | |
|           <br />
 | |
|           {hash}
 | |
|         </td>
 | |
|       </tr>
 | |
|     </>
 | |
|   );
 | |
| };
 | |
| 
 | |
| export default CustomStats;
 |