mirror of
				https://github.com/excalidraw/excalidraw.git
				synced 2025-10-26 08:24:20 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			56 lines
		
	
	
		
			1.3 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			56 lines
		
	
	
		
			1.3 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import { Tooltip } from "@excalidraw/excalidraw/components/Tooltip";
 | |
| import { warning } from "@excalidraw/excalidraw/components/icons";
 | |
| import clsx from "clsx";
 | |
| import { useEffect, useRef, useState } from "react";
 | |
| 
 | |
| import { atom } from "../app-jotai";
 | |
| 
 | |
| import "./CollabError.scss";
 | |
| 
 | |
| type ErrorIndicator = {
 | |
|   message: string | null;
 | |
|   /** used to rerun the useEffect responsible for animation */
 | |
|   nonce: number;
 | |
| };
 | |
| 
 | |
| export const collabErrorIndicatorAtom = atom<ErrorIndicator>({
 | |
|   message: null,
 | |
|   nonce: 0,
 | |
| });
 | |
| 
 | |
| const CollabError = ({ collabError }: { collabError: ErrorIndicator }) => {
 | |
|   const [isAnimating, setIsAnimating] = useState(false);
 | |
|   const clearAnimationRef = useRef<string | number>(0);
 | |
| 
 | |
|   useEffect(() => {
 | |
|     setIsAnimating(true);
 | |
|     clearAnimationRef.current = window.setTimeout(() => {
 | |
|       setIsAnimating(false);
 | |
|     }, 1000);
 | |
| 
 | |
|     return () => {
 | |
|       window.clearTimeout(clearAnimationRef.current);
 | |
|     };
 | |
|   }, [collabError.message, collabError.nonce]);
 | |
| 
 | |
|   if (!collabError.message) {
 | |
|     return null;
 | |
|   }
 | |
| 
 | |
|   return (
 | |
|     <Tooltip label={collabError.message} long={true}>
 | |
|       <div
 | |
|         className={clsx("collab-errors-button", {
 | |
|           "collab-errors-button-shake": isAnimating,
 | |
|         })}
 | |
|       >
 | |
|         {warning}
 | |
|       </div>
 | |
|     </Tooltip>
 | |
|   );
 | |
| };
 | |
| 
 | |
| CollabError.displayName = "CollabError";
 | |
| 
 | |
| export default CollabError;
 | 
