index.tsx 1.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445
  1. import React, { useCallback } from 'react';
  2. import { useNode } from './Node.hooks';
  3. import { withErrorBoundary } from 'react-error-boundary';
  4. import { ErrorBoundaryFallbackComponent } from '../_shared/ErrorBoundaryFallbackComponent';
  5. import TextBlock from '../TextBlock';
  6. import { NodeContext } from '../_shared/SubscribeNode.hooks';
  7. import { BlockType } from '$app/interfaces/document';
  8. import HeadingBlock from '$app/components/document/HeadingBlock';
  9. function NodeComponent({ id, ...props }: { id: string } & React.HTMLAttributes<HTMLDivElement>) {
  10. const { node, childIds, isSelected, ref } = useNode(id);
  11. const renderBlock = useCallback(() => {
  12. switch (node.type) {
  13. case BlockType.TextBlock: {
  14. return <TextBlock node={node} childIds={childIds} />;
  15. }
  16. case BlockType.HeadingBlock: {
  17. return <HeadingBlock node={node} />;
  18. }
  19. default:
  20. return null;
  21. }
  22. }, [node, childIds]);
  23. if (!node) return null;
  24. return (
  25. <NodeContext.Provider value={node}>
  26. <div {...props} ref={ref} data-block-id={node.id} className={`relative px-2 ${props.className}`}>
  27. {renderBlock()}
  28. <div className='block-overlay' />
  29. {isSelected ? (
  30. <div className='pointer-events-none absolute inset-0 z-[-1] m-[1px] rounded-[4px] bg-[#E0F8FF]' />
  31. ) : null}
  32. </div>
  33. </NodeContext.Provider>
  34. );
  35. }
  36. const NodeWithErrorBoundary = withErrorBoundary(NodeComponent, {
  37. FallbackComponent: ErrorBoundaryFallbackComponent,
  38. });
  39. export default React.memo(NodeWithErrorBoundary);