All files / components/Comments CommentItem.tsx

80% Statements 20/25
62.5% Branches 15/24
100% Functions 5/5
95.23% Lines 20/21

Press n or j to go to the next uncovered block, b, p or k for the previous block.

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139                              3x           5x 5x   5x                                                     1x                                 1x                                                             3x 5x 5x                                   5x 5x 5x   5x 5x 5x   5x 5x 5x 5x   5x            
import { memo } from 'react';
import { TouchableOpacity } from 'react-native';
import { Separator, Text, XStack, YStack } from 'tamagui';
import { IconSymbol } from '@/src/components/ui/IconSymbol';
import { AvatarImage } from '@/src/components/ui/OptimizedImage';
import type { Comment } from '@/src/lib/supabase';
import { primaryScale, errorScale } from '@/src/design-system/tokens';
 
interface CommentItemProps {
  comment: Comment;
  currentUserId?: string;
  onLike: (commentId: number) => void;
  onDelete: (commentId: number) => void;
}
 
export const CommentItem = memo(function CommentItem({
  comment,
  currentUserId,
  onLike,
  onDelete,
}: CommentItemProps) {
  const isOwner = comment.author.id === currentUserId;
  const likeCount = comment.likes || 0;
 
  return (
    <YStack paddingVertical="$3" gap="$2">
      <XStack gap="$3">
        <CommentAvatar avatar={comment.author.avatarUrl} authorName={comment.author.username} />
 
        <YStack flex={1} gap="$2">
          {/* 作者信息 */}
          <XStack alignItems="center" justifyContent="space-between">
            <XStack gap="$2" alignItems="center">
              <Text fontSize="$4" fontWeight="600" color="$foreground">
                {comment.author.username}
              </Text>
              {isOwner && (
                <YStack
                  backgroundColor={primaryScale.primary2}
                  paddingHorizontal="$2"
                  paddingVertical="$0.5"
                  borderRadius="$2"
                >
                  <Text fontSize="$1" color={primaryScale.primary9}>
                    我
                  </Text>
                </YStack>
              )}
            </XStack>
 
            {isOwner && (
              <TouchableOpacity onPress={() => onDelete(comment.id)} testID="delete-button">
                <IconSymbol name="trash" size={16} color="$foregroundSubtle" />
              </TouchableOpacity>
            )}
          </XStack>
 
          {/* 内容 */}
          <Text fontSize="$3" color="$foreground" lineHeight={20}>
            {comment.content}
          </Text>
 
          {/* 时间和点赞 */}
          <XStack gap="$4" alignItems="center">
            <Text fontSize="$2" color="$foregroundSubtle">
              {formatTime(comment.createdAt)}
            </Text>
 
            <TouchableOpacity onPress={() => onLike(comment.id)} testID="like-button">
              <XStack gap="$1" alignItems="center">
                <IconSymbol
                  name={comment.isLiked ? 'heart.fill' : 'heart'}
                  size={14}
                  color={comment.isLiked ? errorScale.error6 : '$foregroundSubtle'}
                />
                {likeCount > 0 && (
                  <Text
                    fontSize="$2"
                    color={comment.isLiked ? errorScale.error6 : '$foregroundSubtle'}
                  >
                    {likeCount}
                  </Text>
                )}
              </XStack>
            </TouchableOpacity>
          </XStack>
        </YStack>
      </XStack>
 
      <Separator marginTop="$2" borderColor="$borderMuted" />
    </YStack>
  );
});
 
interface CommentAvatarProps {
  avatar?: string | null;
  authorName: string;
}
 
const CommentAvatar = memo(function CommentAvatar({ avatar }: CommentAvatarProps) {
  Eif (avatar) {
    return <AvatarImage source={avatar} size={40} cachePolicy="memory-disk" />;
  }
 
  return (
    <YStack
      width={40}
      height={40}
      borderRadius={20}
      backgroundColor={primaryScale.primary2}
      alignItems="center"
      justifyContent="center"
    >
      <IconSymbol name="person.fill" size={20} color={primaryScale.primary7} />
    </YStack>
  );
});
 
function formatTime(dateString: string): string {
  const now = new Date();
  const date = new Date(dateString);
  const diff = now.getTime() - date.getTime();
 
  const minutes = Math.floor(diff / 60000);
  const hours = Math.floor(diff / 3600000);
  const days = Math.floor(diff / 86400000);
 
  Iif (minutes < 1) return '刚刚';
  Iif (minutes < 60) return `${minutes}分钟前`;
  Iif (hours < 24) return `${hours}小时前`;
  Iif (days < 7) return `${days}天前`;
 
  return date.toLocaleDateString('zh-CN', {
    year: 'numeric',
    month: '2-digit',
    day: '2-digit',
  });
}