mirror of
https://github.com/kamranahmedse/developer-roadmap.git
synced 2026-03-17 12:21:43 +08:00
Compare commits
3 Commits
fix/activi
...
fix/activi
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
adca6350ce | ||
|
|
89c6b36090 | ||
|
|
cd35c77df1 |
@@ -3,7 +3,8 @@ import { getRelativeTimeString } from '../../lib/date';
|
||||
import type { ResourceType } from '../../lib/resource-progress';
|
||||
import { EmptyStream } from './EmptyStream';
|
||||
import { ActivityTopicsModal } from './ActivityTopicsModal.tsx';
|
||||
import {Book, BookOpen, ChevronsDown, ChevronsDownUp, ChevronsUp, ChevronsUpDown} from 'lucide-react';
|
||||
import { ChevronsDown, ChevronsUp } from 'lucide-react';
|
||||
import { ActivityTopicTitles } from './ActivityTopicTitles.tsx';
|
||||
|
||||
export const allowedActivityActionType = [
|
||||
'in_progress',
|
||||
@@ -21,7 +22,7 @@ export type UserStreamActivity = {
|
||||
resourceSlug?: string;
|
||||
isCustomResource?: boolean;
|
||||
actionType: AllowedActivityActionType;
|
||||
topicIds?: string[];
|
||||
topicTitles?: string[];
|
||||
createdAt: Date;
|
||||
updatedAt: Date;
|
||||
};
|
||||
@@ -38,7 +39,9 @@ export function ActivityStream(props: ActivityStreamProps) {
|
||||
useState<UserStreamActivity | null>(null);
|
||||
|
||||
const sortedActivities = activities
|
||||
.filter((activity) => activity?.topicIds && activity.topicIds.length > 0)
|
||||
.filter(
|
||||
(activity) => activity?.topicTitles && activity.topicTitles.length > 0,
|
||||
)
|
||||
.sort((a, b) => {
|
||||
return new Date(b.updatedAt).getTime() - new Date(a.updatedAt).getTime();
|
||||
})
|
||||
@@ -57,8 +60,8 @@ export function ActivityStream(props: ActivityStreamProps) {
|
||||
resourceId={selectedActivity.resourceId}
|
||||
resourceType={selectedActivity.resourceType}
|
||||
isCustomResource={selectedActivity.isCustomResource}
|
||||
topicIds={selectedActivity.topicIds || []}
|
||||
topicCount={selectedActivity.topicIds?.length || 0}
|
||||
topicTitles={selectedActivity.topicTitles || []}
|
||||
topicCount={selectedActivity.topicTitles?.length || 0}
|
||||
actionType={selectedActivity.actionType}
|
||||
/>
|
||||
)}
|
||||
@@ -73,7 +76,7 @@ export function ActivityStream(props: ActivityStreamProps) {
|
||||
resourceTitle,
|
||||
actionType,
|
||||
updatedAt,
|
||||
topicIds,
|
||||
topicTitles,
|
||||
isCustomResource,
|
||||
} = activity;
|
||||
|
||||
@@ -96,7 +99,7 @@ export function ActivityStream(props: ActivityStreamProps) {
|
||||
</a>
|
||||
);
|
||||
|
||||
const topicCount = topicIds?.length || 0;
|
||||
const topicCount = topicTitles?.length || 0;
|
||||
|
||||
const timeAgo = (
|
||||
<span className="ml-1 text-xs text-gray-400">
|
||||
@@ -108,32 +111,35 @@ export function ActivityStream(props: ActivityStreamProps) {
|
||||
<li key={_id} className="py-2 text-sm text-gray-600">
|
||||
{actionType === 'in_progress' && (
|
||||
<>
|
||||
Started{' '}
|
||||
<button
|
||||
className="font-medium underline underline-offset-2 hover:text-black"
|
||||
onClick={() => setSelectedActivity(activity)}
|
||||
>
|
||||
{topicCount} topic{topicCount > 1 ? 's' : ''}
|
||||
</button>{' '}
|
||||
in {resourceLinkComponent} {timeAgo}
|
||||
<p className="mb-1">
|
||||
Started {topicCount} topic
|
||||
{topicCount > 1 ? 's' : ''} in
|
||||
{resourceLinkComponent}
|
||||
{timeAgo}
|
||||
</p>
|
||||
<ActivityTopicTitles topicTitles={topicTitles || []} />
|
||||
</>
|
||||
)}
|
||||
{actionType === 'done' && (
|
||||
<>
|
||||
Completed{' '}
|
||||
<button
|
||||
className="font-medium underline underline-offset-2 hover:text-black"
|
||||
onClick={() => setSelectedActivity(activity)}
|
||||
>
|
||||
{topicCount} topic{topicCount > 1 ? 's' : ''}
|
||||
</button>{' '}
|
||||
in {resourceLinkComponent} {timeAgo}
|
||||
<p className="mb-1">
|
||||
Completed {topicCount} topic
|
||||
{topicCount > 1 ? 's' : ''} in
|
||||
{resourceLinkComponent}
|
||||
{timeAgo}
|
||||
</p>
|
||||
<ActivityTopicTitles topicTitles={topicTitles || []} />
|
||||
</>
|
||||
)}
|
||||
{actionType === 'answered' && (
|
||||
<>
|
||||
Answered {topicCount} question{topicCount > 1 ? 's' : ''} in{' '}
|
||||
{resourceLinkComponent} {timeAgo}
|
||||
<p className="mb-1">
|
||||
Answered {topicCount} question
|
||||
{topicCount > 1 ? 's' : ''} in
|
||||
{resourceLinkComponent}
|
||||
{timeAgo}
|
||||
</p>
|
||||
<ActivityTopicTitles topicTitles={topicTitles || []} />
|
||||
</>
|
||||
)}
|
||||
</li>
|
||||
@@ -146,16 +152,20 @@ export function ActivityStream(props: ActivityStreamProps) {
|
||||
|
||||
{activities.length > 10 && (
|
||||
<button
|
||||
className="mt-3 gap-2 flex items-center rounded-md border border-black pl-1.5 pr-2 py-1 text-xs uppercase tracking-wide text-black transition-colors hover:border-black hover:bg-black hover:text-white"
|
||||
className="mt-3 flex items-center gap-2 rounded-md border border-black py-1 pl-1.5 pr-2 text-xs uppercase tracking-wide text-black transition-colors hover:border-black hover:bg-black hover:text-white"
|
||||
onClick={() => setShowAll(!showAll)}
|
||||
>
|
||||
{showAll ? <>
|
||||
<ChevronsUp size={14} />
|
||||
Show less
|
||||
</> : <>
|
||||
<ChevronsDown size={14} />
|
||||
Show more
|
||||
</>}
|
||||
{showAll ? (
|
||||
<>
|
||||
<ChevronsUp size={14} />
|
||||
Show less
|
||||
</>
|
||||
) : (
|
||||
<>
|
||||
<ChevronsDown size={14} />
|
||||
Show more
|
||||
</>
|
||||
)}
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
|
||||
43
src/components/Activity/ActivityTopicTitles.tsx
Normal file
43
src/components/Activity/ActivityTopicTitles.tsx
Normal file
@@ -0,0 +1,43 @@
|
||||
import { useState } from 'react';
|
||||
import { cn } from '../../lib/classname';
|
||||
|
||||
type ActivityTopicTitlesProps = {
|
||||
topicTitles: string[];
|
||||
className?: string;
|
||||
onSelectActivity?: () => void;
|
||||
};
|
||||
|
||||
export function ActivityTopicTitles(props: ActivityTopicTitlesProps) {
|
||||
const { topicTitles, onSelectActivity, className } = props;
|
||||
|
||||
const [showAll, setShowAll] = useState(false);
|
||||
const filteredTopicTitles = topicTitles.slice(
|
||||
0,
|
||||
showAll ? topicTitles.length : 3,
|
||||
);
|
||||
|
||||
const shouldShowButton = topicTitles.length > 3;
|
||||
|
||||
return (
|
||||
<div
|
||||
className={cn(
|
||||
'flex flex-wrap gap-1 text-sm font-normal text-gray-600',
|
||||
className,
|
||||
)}
|
||||
>
|
||||
{filteredTopicTitles.map((topicTitle, index) => (
|
||||
<span key={index} className="rounded-md bg-gray-200 px-1">
|
||||
{topicTitle}
|
||||
</span>
|
||||
))}
|
||||
{shouldShowButton && (
|
||||
<button
|
||||
onClick={() => setShowAll(!showAll)}
|
||||
className="text-gray-600 underline underline-offset-2 hover:text-black"
|
||||
>
|
||||
{showAll ? '- Show less' : `+${topicTitles.length - 3} more`}
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
@@ -11,7 +11,7 @@ type ActivityTopicDetailsProps = {
|
||||
resourceId: string;
|
||||
resourceType: ResourceType | 'question';
|
||||
isCustomResource?: boolean;
|
||||
topicIds: string[];
|
||||
topicTitles: string[];
|
||||
topicCount: number;
|
||||
actionType: AllowedActivityActionType;
|
||||
onClose: () => void;
|
||||
@@ -22,56 +22,12 @@ export function ActivityTopicsModal(props: ActivityTopicDetailsProps) {
|
||||
resourceId,
|
||||
resourceType,
|
||||
isCustomResource,
|
||||
topicIds = [],
|
||||
topicTitles = [],
|
||||
topicCount,
|
||||
actionType,
|
||||
onClose,
|
||||
} = props;
|
||||
|
||||
const [isLoading, setIsLoading] = useState(true);
|
||||
const [topicTitles, setTopicTitles] = useState<Record<string, string>>({});
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
|
||||
const loadTopicTitles = async () => {
|
||||
setIsLoading(true);
|
||||
setError(null);
|
||||
|
||||
const { response, error } = await httpPost(
|
||||
`${import.meta.env.PUBLIC_API_URL}/v1-get-topic-titles`,
|
||||
{
|
||||
resourceId,
|
||||
resourceType,
|
||||
isCustomResource,
|
||||
topicIds,
|
||||
},
|
||||
);
|
||||
|
||||
if (error || !response) {
|
||||
setError(error?.message || 'Failed to load topic titles');
|
||||
setIsLoading(false);
|
||||
return;
|
||||
}
|
||||
|
||||
setTopicTitles(response);
|
||||
setIsLoading(false);
|
||||
};
|
||||
|
||||
useEffect(() => {
|
||||
loadTopicTitles().finally(() => {
|
||||
setIsLoading(false);
|
||||
});
|
||||
}, []);
|
||||
|
||||
if (isLoading || error) {
|
||||
return (
|
||||
<ModalLoader
|
||||
error={error!}
|
||||
text={'Loading topics..'}
|
||||
isLoading={isLoading}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
let pageUrl = '';
|
||||
if (resourceType === 'roadmap') {
|
||||
pageUrl = isCustomResource ? `/r/${resourceId}` : `/${resourceId}`;
|
||||
@@ -85,8 +41,6 @@ export function ActivityTopicsModal(props: ActivityTopicDetailsProps) {
|
||||
<Modal
|
||||
onClose={() => {
|
||||
onClose();
|
||||
setError(null);
|
||||
setIsLoading(false);
|
||||
}}
|
||||
>
|
||||
<div className={`popup-body relative rounded-lg bg-white p-4 shadow`}>
|
||||
@@ -108,9 +62,7 @@ export function ActivityTopicsModal(props: ActivityTopicDetailsProps) {
|
||||
</a>
|
||||
</span>
|
||||
<ul className="flex max-h-[50vh] flex-col gap-1 overflow-y-auto max-md:max-h-full">
|
||||
{topicIds.map((topicId) => {
|
||||
const topicTitle = topicTitles[topicId] || 'Unknown Topic';
|
||||
|
||||
{topicTitles.map((topicTitle) => {
|
||||
const ActivityIcon =
|
||||
actionType === 'done'
|
||||
? Check
|
||||
@@ -119,7 +71,7 @@ export function ActivityTopicsModal(props: ActivityTopicDetailsProps) {
|
||||
: Check;
|
||||
|
||||
return (
|
||||
<li key={topicId} className="flex items-start gap-2">
|
||||
<li key={topicTitle} className="flex items-start gap-2">
|
||||
<ActivityIcon
|
||||
strokeWidth={3}
|
||||
className="relative top-[4px] text-green-500"
|
||||
|
||||
@@ -4,7 +4,7 @@ export function EmptyStream() {
|
||||
return (
|
||||
<div className="rounded-md">
|
||||
<div className="flex flex-col items-center p-7 text-center">
|
||||
<List className="mb-2 h-[60px] w-[60px] opacity-10 sm:h-[120px] sm:w-[120px]" />
|
||||
<List className="mb-4 h-[60px] w-[60px] opacity-10 sm:h-[60px] sm:w-[60px]" />
|
||||
|
||||
<h2 className="text-lg font-bold sm:text-xl">No Activities</h2>
|
||||
<p className="my-1 max-w-[400px] text-balance text-sm text-gray-500 sm:my-2 sm:text-base">
|
||||
|
||||
@@ -2,6 +2,8 @@ import { useState } from 'react';
|
||||
import { getRelativeTimeString } from '../../lib/date';
|
||||
import type { TeamStreamActivity } from './TeamActivityPage';
|
||||
import { ChevronsDown, ChevronsUp } from 'lucide-react';
|
||||
import { ActivityTopicTitles } from '../Activity/ActivityTopicTitles';
|
||||
import { cn } from '../../lib/classname';
|
||||
|
||||
type TeamActivityItemProps = {
|
||||
onTopicClick?: (activity: TeamStreamActivity) => void;
|
||||
@@ -66,50 +68,61 @@ export function TeamActivityItem(props: TeamActivityItemProps) {
|
||||
src={userAvatar}
|
||||
alt={user.name}
|
||||
/>
|
||||
<span className="font-medium">{user?.name || 'Unknown'}</span>{' '}
|
||||
<span className="font-medium">{user?.name || 'Unknown'}</span>
|
||||
</>
|
||||
);
|
||||
|
||||
if (activities.length === 1) {
|
||||
const activity = activities[0];
|
||||
const { actionType, topicIds } = activity;
|
||||
const topicCount = topicIds?.length || 0;
|
||||
const { actionType, topicTitles } = activity;
|
||||
const topicCount = topicTitles?.length || 0;
|
||||
|
||||
return (
|
||||
<li
|
||||
key={user._id}
|
||||
className="flex flex-wrap items-center gap-1 rounded-md border px-2 py-2.5 text-sm"
|
||||
className="flex flex-wrap items-center rounded-md border px-2 py-2.5 text-sm"
|
||||
>
|
||||
{actionType === 'in_progress' && (
|
||||
<>
|
||||
{username} started{' '}
|
||||
<button
|
||||
className="font-medium underline underline-offset-2 hover:text-black"
|
||||
onClick={() => onTopicClick?.(activity)}
|
||||
>
|
||||
{topicCount} topic{topicCount > 1 ? 's' : ''}
|
||||
</button>{' '}
|
||||
in {resourceLink(activity)} {timeAgo(activity.updatedAt)}
|
||||
<p className="mb-1">
|
||||
{username} started
|
||||
{topicCount} topic{topicCount > 1 ? 's' : ''} in
|
||||
{resourceLink(activity)}
|
||||
{timeAgo(activity.updatedAt)}
|
||||
</p>
|
||||
<ActivityTopicTitles
|
||||
className="pl-5"
|
||||
topicTitles={topicTitles || []}
|
||||
/>
|
||||
</>
|
||||
)}
|
||||
|
||||
{actionType === 'done' && (
|
||||
<>
|
||||
{username} completed{' '}
|
||||
<button
|
||||
className="font-medium underline underline-offset-2 hover:text-black"
|
||||
onClick={() => onTopicClick?.(activity)}
|
||||
>
|
||||
{topicCount} topic{topicCount > 1 ? 's' : ''}
|
||||
</button>{' '}
|
||||
in {resourceLink(activity)} {timeAgo(activity.updatedAt)}
|
||||
<p className="mb-1">
|
||||
{username} completed
|
||||
{topicCount} topic{topicCount > 1 ? 's' : ''} in
|
||||
{resourceLink(activity)}
|
||||
{timeAgo(activity.updatedAt)}
|
||||
</p>
|
||||
<ActivityTopicTitles
|
||||
className="pl-5"
|
||||
topicTitles={topicTitles || []}
|
||||
/>
|
||||
</>
|
||||
)}
|
||||
{actionType === 'answered' && (
|
||||
<>
|
||||
{username} answered {topicCount} question
|
||||
{topicCount > 1 ? 's' : ''} in {resourceLink(activity)}{' '}
|
||||
{timeAgo(activity.updatedAt)}
|
||||
<p className="mb-1">
|
||||
{username} answered
|
||||
{topicCount} question{topicCount > 1 ? 's' : ''}
|
||||
in
|
||||
{resourceLink(activity)}
|
||||
{timeAgo(activity.updatedAt)}
|
||||
</p>
|
||||
<ActivityTopicTitles
|
||||
className="pl-5"
|
||||
topicTitles={topicTitles || []}
|
||||
/>
|
||||
</>
|
||||
)}
|
||||
</li>
|
||||
@@ -129,42 +142,51 @@ export function TeamActivityItem(props: TeamActivityItemProps) {
|
||||
resource(s)
|
||||
</h3>
|
||||
<div className="py-3">
|
||||
<ul className="ml-2 flex flex-col gap-2 sm:ml-[36px]">
|
||||
{activities.slice(0, activityLimit).map((activity) => {
|
||||
const { actionType, topicIds } = activity;
|
||||
const topicCount = topicIds?.length || 0;
|
||||
<ul className="ml-2 flex flex-col divide-y pr-2 sm:ml-[36px]">
|
||||
{activities.slice(0, activityLimit).map((activity, counter) => {
|
||||
const { actionType, topicTitles } = activity;
|
||||
const topicCount = topicTitles?.length || 0;
|
||||
|
||||
return (
|
||||
<li key={activity._id} className="text-sm text-gray-600">
|
||||
<li
|
||||
key={activity._id}
|
||||
className={cn(
|
||||
'text-sm text-gray-600',
|
||||
counter === 0 ? 'pb-2.5' : 'py-2.5',
|
||||
counter === activities.length - 1 ? 'pb-0' : '',
|
||||
)}
|
||||
>
|
||||
{actionType === 'in_progress' && (
|
||||
<>
|
||||
Started{' '}
|
||||
<button
|
||||
className="font-medium underline underline-offset-2 hover:text-black"
|
||||
onClick={() => onTopicClick?.(activity)}
|
||||
>
|
||||
{topicCount} topic{topicCount > 1 ? 's' : ''}
|
||||
</button>{' '}
|
||||
in {resourceLink(activity)} {timeAgo(activity.updatedAt)}
|
||||
<p className="mb-1">
|
||||
Started {topicCount} topic
|
||||
{topicCount > 1 ? 's' : ''} in
|
||||
{resourceLink(activity)}
|
||||
{timeAgo(activity.updatedAt)}
|
||||
</p>
|
||||
<ActivityTopicTitles topicTitles={topicTitles || []} />
|
||||
</>
|
||||
)}
|
||||
{actionType === 'done' && (
|
||||
<>
|
||||
Completed{' '}
|
||||
<button
|
||||
className="font-medium underline underline-offset-2 hover:text-black"
|
||||
onClick={() => onTopicClick?.(activity)}
|
||||
>
|
||||
{topicCount} topic{topicCount > 1 ? 's' : ''}
|
||||
</button>{' '}
|
||||
in {resourceLink(activity)} {timeAgo(activity.updatedAt)}
|
||||
<p className="mb-1">
|
||||
Completed {topicCount} topic
|
||||
{topicCount > 1 ? 's' : ''} in
|
||||
{resourceLink(activity)}
|
||||
{timeAgo(activity.updatedAt)}
|
||||
</p>
|
||||
<ActivityTopicTitles topicTitles={topicTitles || []} />
|
||||
</>
|
||||
)}
|
||||
{actionType === 'answered' && (
|
||||
<>
|
||||
Answered {topicCount} question
|
||||
{topicCount > 1 ? 's' : ''} in {resourceLink(activity)}{' '}
|
||||
{timeAgo(activity.updatedAt)}
|
||||
<p className="mb-1">
|
||||
Answered {topicCount} question
|
||||
{topicCount > 1 ? 's' : ''} in
|
||||
{resourceLink(activity)}
|
||||
{timeAgo(activity.updatedAt)}
|
||||
</p>
|
||||
<ActivityTopicTitles topicTitles={topicTitles || []} />
|
||||
</>
|
||||
)}
|
||||
</li>
|
||||
|
||||
@@ -18,7 +18,7 @@ export type TeamStreamActivity = {
|
||||
resourceSlug?: string;
|
||||
isCustomResource?: boolean;
|
||||
actionType: AllowedActivityActionType;
|
||||
topicIds?: string[];
|
||||
topicTitles?: string[];
|
||||
createdAt: Date;
|
||||
updatedAt: Date;
|
||||
};
|
||||
@@ -102,7 +102,7 @@ export function TeamActivityPage() {
|
||||
return activities?.filter((activity) => {
|
||||
return (
|
||||
activity.activity.length > 0 &&
|
||||
activity.activity.some((t) => (t?.topicIds?.length || 0) > 0)
|
||||
activity.activity.some((t) => (t?.topicTitles?.length || 0) > 0)
|
||||
);
|
||||
});
|
||||
}, [activities]);
|
||||
@@ -137,7 +137,7 @@ export function TeamActivityPage() {
|
||||
const userActivities = uniqueActivities
|
||||
.filter((activity) => activity.userId === user._id)
|
||||
.flatMap((activity) => activity.activity)
|
||||
.filter((activity) => (activity?.topicIds?.length || 0) > 0)
|
||||
.filter((activity) => (activity?.topicTitles?.length || 0) > 0)
|
||||
.sort((a, b) => {
|
||||
return (
|
||||
new Date(b.updatedAt).getTime() -
|
||||
|
||||
@@ -16,54 +16,10 @@ export function TeamActivityTopicsModal(props: TeamActivityTopicsModalProps) {
|
||||
resourceId,
|
||||
resourceType,
|
||||
isCustomResource,
|
||||
topicIds = [],
|
||||
topicTitles = [],
|
||||
actionType,
|
||||
} = activity;
|
||||
|
||||
const [isLoading, setIsLoading] = useState(true);
|
||||
const [topicTitles, setTopicTitles] = useState<Record<string, string>>({});
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
|
||||
const loadTopicTitles = async () => {
|
||||
setIsLoading(true);
|
||||
setError(null);
|
||||
|
||||
const { response, error } = await httpPost(
|
||||
`${import.meta.env.PUBLIC_API_URL}/v1-get-topic-titles`,
|
||||
{
|
||||
resourceId,
|
||||
resourceType,
|
||||
isCustomResource,
|
||||
topicIds,
|
||||
},
|
||||
);
|
||||
|
||||
if (error || !response) {
|
||||
setError(error?.message || 'Failed to load topic titles');
|
||||
setIsLoading(false);
|
||||
return;
|
||||
}
|
||||
|
||||
setTopicTitles(response);
|
||||
setIsLoading(false);
|
||||
};
|
||||
|
||||
useEffect(() => {
|
||||
loadTopicTitles().finally(() => {
|
||||
setIsLoading(false);
|
||||
});
|
||||
}, []);
|
||||
|
||||
if (isLoading || error) {
|
||||
return (
|
||||
<ModalLoader
|
||||
error={error!}
|
||||
text={'Loading topics..'}
|
||||
isLoading={isLoading}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
let pageUrl = '';
|
||||
if (resourceType === 'roadmap') {
|
||||
pageUrl = isCustomResource ? `/r/${resourceId}` : `/${resourceId}`;
|
||||
@@ -77,8 +33,6 @@ export function TeamActivityTopicsModal(props: TeamActivityTopicsModalProps) {
|
||||
<Modal
|
||||
onClose={() => {
|
||||
onClose();
|
||||
setError(null);
|
||||
setIsLoading(false);
|
||||
}}
|
||||
>
|
||||
<div className={`popup-body relative rounded-lg bg-white p-4 shadow`}>
|
||||
@@ -100,9 +54,7 @@ export function TeamActivityTopicsModal(props: TeamActivityTopicsModalProps) {
|
||||
</a>
|
||||
</span>
|
||||
<ul className="flex max-h-[50vh] flex-col gap-1 overflow-y-auto max-md:max-h-full">
|
||||
{topicIds.map((topicId) => {
|
||||
const topicTitle = topicTitles[topicId] || 'Unknown Topic';
|
||||
|
||||
{topicTitles.map((topicTitle) => {
|
||||
const ActivityIcon =
|
||||
actionType === 'done'
|
||||
? Check
|
||||
@@ -111,7 +63,7 @@ export function TeamActivityTopicsModal(props: TeamActivityTopicsModalProps) {
|
||||
: Check;
|
||||
|
||||
return (
|
||||
<li key={topicId} className="flex items-start gap-2">
|
||||
<li key={topicTitle} className="flex items-start gap-2">
|
||||
<ActivityIcon
|
||||
strokeWidth={3}
|
||||
className="relative top-[4px] text-green-500"
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
import { Activity, List, ListTodo } from 'lucide-react';
|
||||
import { ListTodo } from 'lucide-react';
|
||||
|
||||
type TeamActivityItemProps = {
|
||||
teamId: string;
|
||||
|
||||
Reference in New Issue
Block a user