From c438f9442f50b539120830e893e99b39f2d72efd Mon Sep 17 00:00:00 2001 From: Michael Barrett Date: Thu, 19 Dec 2024 11:58:24 +0000 Subject: [PATCH] Filtered notifications that show in `admin-x-activitypub` (#21903) refs [AP-627](https://linear.app/ghost/issue/AP-627/sanitising-note-content) Adding client side filtering of the notifications in `admin-x-activitypub` - This is a stop-gap until we have a dedicated endpoint for returning notifications. Filtering includes: - Do not show our own likes - Only show `like` notifications when it is on our own posts - Only show replies to our own posts --- apps/admin-x-activitypub/package.json | 2 +- .../src/components/Activities.tsx | 70 +++++++++++++++++-- 2 files changed, 66 insertions(+), 6 deletions(-) diff --git a/apps/admin-x-activitypub/package.json b/apps/admin-x-activitypub/package.json index e719c34dae..fa09ee4c44 100644 --- a/apps/admin-x-activitypub/package.json +++ b/apps/admin-x-activitypub/package.json @@ -1,6 +1,6 @@ { "name": "@tryghost/admin-x-activitypub", - "version": "0.3.39", + "version": "0.3.40", "license": "MIT", "repository": { "type": "git", diff --git a/apps/admin-x-activitypub/src/components/Activities.tsx b/apps/admin-x-activitypub/src/components/Activities.tsx index 97e167fc53..34f19c278c 100644 --- a/apps/admin-x-activitypub/src/components/Activities.tsx +++ b/apps/admin-x-activitypub/src/components/Activities.tsx @@ -13,7 +13,11 @@ import Separator from './global/Separator'; import getUsername from '../utils/get-username'; import stripHtml from '../utils/strip-html'; import truncate from '../utils/truncate'; -import {GET_ACTIVITIES_QUERY_KEY_NOTIFICATIONS, useActivitiesForUser} from '../hooks/useActivityPubQueries'; +import { + GET_ACTIVITIES_QUERY_KEY_NOTIFICATIONS, + useActivitiesForUser, + useUserDataForUser +} from '../hooks/useActivityPubQueries'; import {type NotificationType} from './activities/NotificationIcon'; import {handleProfileClick} from '../utils/handle-profile-click'; @@ -144,7 +148,14 @@ const getGroupDescription = (group: GroupedActivity): JSX.Element => { return <>{actorText} liked your post {group.object?.name || ''}; case ACTIVITY_TYPE.CREATE: if (group.object?.inReplyTo && typeof group.object?.inReplyTo !== 'string') { - const content = stripHtml(group.object.inReplyTo.name); + let content = stripHtml(group.object.inReplyTo.content); + + // If the post has a name, use that instead of the content (short + // form posts do not have a name) + if (group.object.inReplyTo.name) { + content = stripHtml(group.object.inReplyTo.name); + } + return <>{actorText} replied to your post {truncate(content, 80)}; } } @@ -153,6 +164,7 @@ const getGroupDescription = (group: GroupedActivity): JSX.Element => { const Activities: React.FC = ({}) => { const user = 'index'; + const [openStates, setOpenStates] = React.useState<{[key: string]: boolean}>({}); const toggleOpen = (groupId: string) => { @@ -164,19 +176,66 @@ const Activities: React.FC = ({}) => { const maxAvatars = 5; + const {data: userProfile, isLoading: isLoadingProfile} = useUserDataForUser(user) as {data: ActorProperties | null, isLoading: boolean}; + const {getActivitiesQuery} = useActivitiesForUser({ handle: user, includeOwn: true, includeReplies: true, filter: { - type: ['Follow', 'Like', `Create:Note:isReplyToOwn`] + type: ['Follow', 'Like', `Create:Note`] }, key: GET_ACTIVITIES_QUERY_KEY_NOTIFICATIONS }); - const {data, fetchNextPage, hasNextPage, isFetchingNextPage, isLoading} = getActivitiesQuery; + const {data, fetchNextPage, hasNextPage, isFetchingNextPage, isLoading: isLoadingActivities} = getActivitiesQuery; + + const isLoading = isLoadingProfile === true || isLoadingActivities === true; + const groupedActivities = (data?.pages.flatMap((page) => { - const filtered = page.data.filter((activity, index, self) => index === self.findIndex(a => a.id === activity.id)); + const filtered = page.data + // Remove duplicates + .filter( + (activity, index, self) => index === self.findIndex(a => a.id === activity.id) + ) + // Remove our own likes + .filter((activity) => { + if (activity.type === ACTIVITY_TYPE.LIKE && activity.actor?.id === userProfile?.id) { + return false; + } + + return true; + }) + // Remove follower likes if they are not for our own posts + .filter((activity) => { + if (activity.type === ACTIVITY_TYPE.LIKE && activity.object?.attributedTo?.id !== userProfile?.id) { + return false; + } + + return true; + }) + // Remove create activities that are not replies to our own posts + .filter((activity) => { + if ( + activity.type === ACTIVITY_TYPE.CREATE && + activity.object?.inReplyTo?.attributedTo?.id !== userProfile?.id + ) { + return false; + } + + return true; + }) + // Remove our own create activities + .filter((activity) => { + if ( + activity.type === ACTIVITY_TYPE.CREATE && + activity.actor?.id === userProfile?.id + ) { + return false; + } + + return true; + }); return groupActivities(filtered); }) ?? []); @@ -234,6 +293,7 @@ const Activities: React.FC = ({}) => { break; } }; + return ( <>