@stream-io/feeds-client 0.1.10 → 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (168) hide show
  1. package/CHANGELOG.md +30 -0
  2. package/dist/@react-bindings/contexts/StreamFeedContext.d.ts +1 -1
  3. package/dist/@react-bindings/contexts/StreamFeedsContext.d.ts +1 -1
  4. package/dist/@react-bindings/hooks/feed-state-hooks/useComments.d.ts +1 -1
  5. package/dist/@react-bindings/hooks/feed-state-hooks/useFeedActivities.d.ts +1 -1
  6. package/dist/@react-bindings/hooks/feed-state-hooks/useFeedMetadata.d.ts +1 -1
  7. package/dist/@react-bindings/hooks/feed-state-hooks/useFollowers.d.ts +1 -1
  8. package/dist/@react-bindings/hooks/feed-state-hooks/useFollowing.d.ts +1 -1
  9. package/dist/@react-bindings/hooks/feed-state-hooks/useOwnCapabilities.d.ts +1 -1
  10. package/dist/@react-bindings/hooks/feed-state-hooks/useOwnFollows.d.ts +1 -1
  11. package/dist/@react-bindings/hooks/useCreateFeedsClient.d.ts +1 -1
  12. package/dist/@react-bindings/wrappers/StreamFeed.d.ts +1 -1
  13. package/dist/index-react-bindings.browser.cjs +1720 -1601
  14. package/dist/index-react-bindings.browser.cjs.map +1 -1
  15. package/dist/index-react-bindings.browser.js +1720 -1601
  16. package/dist/index-react-bindings.browser.js.map +1 -1
  17. package/dist/index-react-bindings.node.cjs +1720 -1601
  18. package/dist/index-react-bindings.node.cjs.map +1 -1
  19. package/dist/index-react-bindings.node.js +1720 -1601
  20. package/dist/index-react-bindings.node.js.map +1 -1
  21. package/dist/index.browser.cjs +1724 -1602
  22. package/dist/index.browser.cjs.map +1 -1
  23. package/dist/index.browser.js +1722 -1603
  24. package/dist/index.browser.js.map +1 -1
  25. package/dist/index.d.ts +2 -2
  26. package/dist/index.node.cjs +1724 -1602
  27. package/dist/index.node.cjs.map +1 -1
  28. package/dist/index.node.js +1722 -1603
  29. package/dist/index.node.js.map +1 -1
  30. package/dist/src/common/ActivitySearchSource.d.ts +1 -1
  31. package/dist/src/common/FeedSearchSource.d.ts +2 -2
  32. package/dist/src/common/Poll.d.ts +1 -1
  33. package/dist/src/common/UserSearchSource.d.ts +1 -1
  34. package/dist/src/common/real-time/StableWSConnection.d.ts +3 -3
  35. package/dist/src/feed/event-handlers/activity/handle-activity-added.d.ts +7 -0
  36. package/dist/src/feed/event-handlers/activity/handle-activity-deleted.d.ts +8 -0
  37. package/dist/src/feed/event-handlers/activity/handle-activity-reaction-added.d.ts +8 -0
  38. package/dist/src/feed/event-handlers/activity/handle-activity-reaction-deleted.d.ts +8 -0
  39. package/dist/src/feed/event-handlers/activity/handle-activity-removed-from-feed.d.ts +3 -0
  40. package/dist/src/feed/event-handlers/activity/handle-activity-updated.d.ts +8 -0
  41. package/dist/src/feed/event-handlers/activity/index.d.ts +6 -0
  42. package/dist/src/feed/event-handlers/bookmark/handle-bookmark-added.d.ts +8 -0
  43. package/dist/src/feed/event-handlers/bookmark/handle-bookmark-deleted.d.ts +9 -0
  44. package/dist/src/feed/event-handlers/bookmark/handle-bookmark-updated.d.ts +8 -0
  45. package/dist/src/feed/event-handlers/bookmark/index.d.ts +3 -0
  46. package/dist/src/feed/event-handlers/comment/handle-comment-added.d.ts +3 -0
  47. package/dist/src/feed/event-handlers/comment/handle-comment-deleted.d.ts +3 -0
  48. package/dist/src/feed/event-handlers/comment/handle-comment-reaction.d.ts +3 -0
  49. package/dist/src/feed/event-handlers/comment/handle-comment-updated.d.ts +3 -0
  50. package/dist/src/feed/event-handlers/comment/index.d.ts +4 -0
  51. package/dist/src/feed/event-handlers/feed/handle-feed-updated.d.ts +3 -0
  52. package/dist/src/feed/event-handlers/feed/index.d.ts +1 -0
  53. package/dist/src/feed/event-handlers/feed-member/handle-feed-member-added.d.ts +3 -0
  54. package/dist/src/feed/event-handlers/feed-member/handle-feed-member-removed.d.ts +3 -0
  55. package/dist/src/feed/event-handlers/feed-member/handle-feed-member-updated.d.ts +3 -0
  56. package/dist/src/feed/event-handlers/feed-member/index.d.ts +3 -0
  57. package/dist/src/feed/event-handlers/follow/handle-follow-created.d.ts +7 -0
  58. package/dist/src/feed/event-handlers/follow/handle-follow-deleted.d.ts +7 -0
  59. package/dist/src/feed/event-handlers/follow/handle-follow-updated.d.ts +3 -0
  60. package/dist/src/feed/event-handlers/follow/index.d.ts +3 -0
  61. package/dist/src/feed/event-handlers/index.d.ts +8 -0
  62. package/dist/src/feed/event-handlers/notification-feed/handle-notification-feed-updated.d.ts +3 -0
  63. package/dist/src/feed/event-handlers/notification-feed/index.d.ts +1 -0
  64. package/dist/src/feed/event-handlers/watch/handle-watch-started.d.ts +2 -0
  65. package/dist/src/feed/event-handlers/watch/handle-watch-stopped.d.ts +2 -0
  66. package/dist/src/feed/event-handlers/watch/index.d.ts +2 -0
  67. package/dist/src/{Feed.d.ts → feed/feed.d.ts} +16 -43
  68. package/dist/src/feed/index.d.ts +2 -0
  69. package/dist/src/feeds-client/event-handlers/index.d.ts +1 -0
  70. package/dist/src/feeds-client/event-handlers/user/handle-user-updated.d.ts +3 -0
  71. package/dist/src/feeds-client/feeds-client.d.ts +76 -0
  72. package/dist/src/feeds-client/index.d.ts +2 -0
  73. package/dist/src/gen/feeds/FeedsApi.d.ts +27 -23
  74. package/dist/src/gen/models/index.d.ts +168 -23
  75. package/dist/src/gen-imports.d.ts +1 -1
  76. package/dist/src/test-utils/index.d.ts +1 -0
  77. package/dist/src/test-utils/response-generators.d.ts +9 -0
  78. package/dist/src/types-internal.d.ts +7 -0
  79. package/dist/src/types.d.ts +1 -1
  80. package/dist/src/utils/check-has-another-page.d.ts +1 -0
  81. package/dist/src/utils/constants.d.ts +3 -0
  82. package/dist/src/utils/index.d.ts +5 -0
  83. package/dist/src/utils/state-update-queue.d.ts +6 -0
  84. package/dist/src/utils/type-assertions.d.ts +7 -0
  85. package/dist/src/utils/unique-array-merge.d.ts +1 -0
  86. package/dist/tsconfig.tsbuildinfo +1 -1
  87. package/index.ts +2 -2
  88. package/package.json +3 -2
  89. package/src/common/ActivitySearchSource.ts +1 -1
  90. package/src/common/FeedSearchSource.ts +2 -2
  91. package/src/common/Poll.ts +1 -1
  92. package/src/common/UserSearchSource.ts +1 -1
  93. package/src/{state-updates → feed/event-handlers/activity}/activity-reaction-utils.test.ts +12 -2
  94. package/src/{state-updates → feed/event-handlers/activity}/activity-utils.test.ts +3 -2
  95. package/src/{state-updates/activity-utils.ts → feed/event-handlers/activity/handle-activity-added.ts} +16 -36
  96. package/src/feed/event-handlers/activity/handle-activity-deleted.ts +30 -0
  97. package/src/feed/event-handlers/activity/handle-activity-reaction-added.ts +67 -0
  98. package/src/feed/event-handlers/activity/handle-activity-reaction-deleted.ts +75 -0
  99. package/src/feed/event-handlers/activity/handle-activity-removed-from-feed.ts +16 -0
  100. package/src/feed/event-handlers/activity/handle-activity-updated.ts +47 -0
  101. package/src/feed/event-handlers/activity/index.ts +6 -0
  102. package/src/{state-updates → feed/event-handlers/bookmark}/bookmark-utils.test.ts +2 -2
  103. package/src/feed/event-handlers/bookmark/handle-bookmark-added.ts +63 -0
  104. package/src/feed/event-handlers/bookmark/handle-bookmark-deleted.ts +84 -0
  105. package/src/feed/event-handlers/bookmark/handle-bookmark-updated.ts +76 -0
  106. package/src/feed/event-handlers/bookmark/index.ts +3 -0
  107. package/src/feed/event-handlers/comment/handle-comment-added.ts +38 -0
  108. package/src/feed/event-handlers/comment/handle-comment-deleted.ts +35 -0
  109. package/src/feed/event-handlers/comment/handle-comment-reaction.ts +61 -0
  110. package/src/feed/event-handlers/comment/handle-comment-updated.ts +35 -0
  111. package/src/feed/event-handlers/comment/index.ts +4 -0
  112. package/src/feed/event-handlers/feed/handle-feed-updated.ts +9 -0
  113. package/src/feed/event-handlers/feed/index.ts +1 -0
  114. package/src/feed/event-handlers/feed-member/handle-feed-member-added.ts +31 -0
  115. package/src/feed/event-handlers/feed-member/handle-feed-member-removed.ts +24 -0
  116. package/src/feed/event-handlers/feed-member/handle-feed-member-updated.ts +40 -0
  117. package/src/feed/event-handlers/feed-member/index.ts +3 -0
  118. package/src/feed/event-handlers/follow/handle-follow-created.test.ts +250 -0
  119. package/src/feed/event-handlers/follow/handle-follow-created.ts +90 -0
  120. package/src/feed/event-handlers/follow/handle-follow-deleted.test.ts +268 -0
  121. package/src/feed/event-handlers/follow/handle-follow-deleted.ts +95 -0
  122. package/src/feed/event-handlers/follow/handle-follow-updated.test.ts +174 -0
  123. package/src/feed/event-handlers/follow/handle-follow-updated.ts +85 -0
  124. package/src/feed/event-handlers/follow/index.ts +3 -0
  125. package/src/feed/event-handlers/index.ts +8 -0
  126. package/src/feed/event-handlers/notification-feed/handle-notification-feed-updated.ts +10 -0
  127. package/src/feed/event-handlers/notification-feed/index.ts +1 -0
  128. package/src/feed/event-handlers/watch/handle-watch-started.ts +5 -0
  129. package/src/feed/event-handlers/watch/handle-watch-stopped.ts +5 -0
  130. package/src/feed/event-handlers/watch/index.ts +2 -0
  131. package/src/{Feed.ts → feed/feed.ts} +87 -516
  132. package/src/feed/index.ts +2 -0
  133. package/src/feeds-client/event-handlers/index.ts +1 -0
  134. package/src/feeds-client/event-handlers/user/handle-user-updated.test.ts +53 -0
  135. package/src/feeds-client/event-handlers/user/handle-user-updated.ts +28 -0
  136. package/src/{FeedsClient.ts → feeds-client/feeds-client.ts} +63 -36
  137. package/src/feeds-client/index.ts +2 -0
  138. package/src/gen/feeds/FeedsApi.ts +164 -138
  139. package/src/gen/model-decoders/decoders.ts +22 -0
  140. package/src/gen/models/index.ts +288 -29
  141. package/src/gen-imports.ts +1 -1
  142. package/src/test-utils/index.ts +1 -0
  143. package/src/test-utils/response-generators.ts +101 -0
  144. package/src/types-internal.ts +11 -0
  145. package/src/types.ts +1 -1
  146. package/src/utils/check-has-another-page.ts +6 -0
  147. package/src/utils/constants.ts +3 -0
  148. package/src/utils/index.ts +5 -0
  149. package/src/{state-updates → utils}/state-update-queue.test.ts +6 -6
  150. package/src/utils/state-update-queue.ts +42 -0
  151. package/src/utils/type-assertions.ts +22 -0
  152. package/src/{utils.test.ts → utils/unique-array-merge.test.ts} +7 -3
  153. package/src/utils/unique-array-merge.ts +19 -0
  154. package/dist/src/FeedsClient.d.ts +0 -75
  155. package/dist/src/state-updates/activity-reaction-utils.d.ts +0 -10
  156. package/dist/src/state-updates/activity-utils.d.ts +0 -13
  157. package/dist/src/state-updates/bookmark-utils.d.ts +0 -14
  158. package/dist/src/state-updates/follow-utils.d.ts +0 -19
  159. package/dist/src/state-updates/state-update-queue.d.ts +0 -15
  160. package/dist/src/utils.d.ts +0 -10
  161. package/src/state-updates/activity-reaction-utils.ts +0 -107
  162. package/src/state-updates/bookmark-utils.ts +0 -167
  163. package/src/state-updates/follow-utils.test.ts +0 -552
  164. package/src/state-updates/follow-utils.ts +0 -126
  165. package/src/state-updates/state-update-queue.ts +0 -35
  166. package/src/utils.ts +0 -48
  167. /package/dist/src/{ModerationClient.d.ts → moderation-client.d.ts} +0 -0
  168. /package/src/{ModerationClient.ts → moderation-client.ts} +0 -0
@@ -0,0 +1,76 @@
1
+ import type { Feed } from '../../../feed';
2
+ import type {
3
+ ActivityResponse,
4
+ BookmarkUpdatedEvent,
5
+ } from '../../../gen/models';
6
+ import type { EventPayload, UpdateStateResult } from '../../../types-internal';
7
+
8
+ import { updateActivityInState } from '../activity';
9
+ import { isSameBookmark } from './handle-bookmark-deleted';
10
+
11
+ export const updateBookmarkInActivity = (
12
+ event: BookmarkUpdatedEvent,
13
+ activity: ActivityResponse,
14
+ isCurrentUser: boolean,
15
+ ): UpdateStateResult<ActivityResponse> => {
16
+ // Update own_bookmarks if the bookmark is from the current user
17
+ let ownBookmarks = activity.own_bookmarks || [];
18
+ if (isCurrentUser) {
19
+ const bookmarkIndex = ownBookmarks.findIndex((bookmark) =>
20
+ isSameBookmark(bookmark, event.bookmark),
21
+ );
22
+ if (bookmarkIndex !== -1) {
23
+ ownBookmarks = [...ownBookmarks];
24
+ ownBookmarks[bookmarkIndex] = event.bookmark;
25
+ }
26
+ }
27
+
28
+ return {
29
+ ...activity,
30
+ own_bookmarks: ownBookmarks,
31
+ changed: true,
32
+ };
33
+ };
34
+
35
+ export const updateBookmarkInActivities = (
36
+ event: BookmarkUpdatedEvent,
37
+ activities: ActivityResponse[] | undefined,
38
+ isCurrentUser: boolean,
39
+ ): UpdateStateResult<{ activities: ActivityResponse[] }> => {
40
+ if (!activities) {
41
+ return { changed: false, activities: [] };
42
+ }
43
+
44
+ const activityIndex = activities.findIndex(
45
+ (a) => a.id === event.bookmark.activity.id,
46
+ );
47
+ if (activityIndex === -1) {
48
+ return { changed: false, activities };
49
+ }
50
+
51
+ const activity = activities[activityIndex];
52
+ const updatedActivity = updateBookmarkInActivity(
53
+ event,
54
+ activity,
55
+ isCurrentUser,
56
+ );
57
+ return updateActivityInState(updatedActivity, activities, true);
58
+ };
59
+
60
+ export function handleBookmarkUpdated(
61
+ this: Feed,
62
+ event: EventPayload<'feeds.bookmark.updated'>,
63
+ ) {
64
+ const currentActivities = this.currentState.activities;
65
+ const { connected_user: connectedUser } = this.client.state.getLatestValue();
66
+ const isCurrentUser = event.bookmark.user.id === connectedUser?.id;
67
+
68
+ const result = updateBookmarkInActivities(
69
+ event,
70
+ currentActivities,
71
+ isCurrentUser,
72
+ );
73
+ if (result.changed) {
74
+ this.state.partialNext({ activities: result.activities });
75
+ }
76
+ }
@@ -0,0 +1,3 @@
1
+ export * from './handle-bookmark-added';
2
+ export * from './handle-bookmark-deleted';
3
+ export * from './handle-bookmark-updated';
@@ -0,0 +1,38 @@
1
+ import type { Feed } from '../../../feed';
2
+ import type { EventPayload } from '../../../types-internal';
3
+
4
+ export function handleCommentAdded(
5
+ this: Feed,
6
+ event: EventPayload<'feeds.comment.added'>,
7
+ ) {
8
+ const { comment } = event;
9
+ const entityId = comment.parent_id ?? comment.object_id;
10
+
11
+ this.state.next((currentState) => {
12
+ const entityState = currentState.comments_by_entity_id[entityId];
13
+
14
+ if (typeof entityState?.comments === 'undefined') {
15
+ return currentState;
16
+ }
17
+
18
+ const newComments = entityState?.comments ? [...entityState.comments] : [];
19
+
20
+ if (entityState.pagination?.sort === 'last') {
21
+ newComments.unshift(comment);
22
+ } else {
23
+ // 'first' and other sort options
24
+ newComments.push(comment);
25
+ }
26
+
27
+ return {
28
+ ...currentState,
29
+ comments_by_entity_id: {
30
+ ...currentState.comments_by_entity_id,
31
+ [entityId]: {
32
+ ...currentState.comments_by_entity_id[entityId],
33
+ comments: newComments,
34
+ },
35
+ },
36
+ };
37
+ });
38
+ }
@@ -0,0 +1,35 @@
1
+ import { Feed } from '../../../feed';
2
+ import { EventPayload } from '../../../types-internal';
3
+
4
+ export function handleCommentDeleted(
5
+ this: Feed,
6
+ { comment }: EventPayload<'feeds.comment.deleted'>,
7
+ ) {
8
+ const entityId = comment.parent_id ?? comment.object_id;
9
+
10
+ this.state.next((currentState) => {
11
+ const newCommentsByEntityId = {
12
+ ...currentState.comments_by_entity_id,
13
+ [entityId]: {
14
+ ...currentState.comments_by_entity_id[entityId],
15
+ },
16
+ };
17
+
18
+ const index = this.getCommentIndex(comment, currentState);
19
+
20
+ if (newCommentsByEntityId?.[entityId]?.comments?.length && index !== -1) {
21
+ newCommentsByEntityId[entityId].comments = [
22
+ ...newCommentsByEntityId[entityId].comments,
23
+ ];
24
+
25
+ newCommentsByEntityId[entityId]?.comments?.splice(index, 1);
26
+ }
27
+
28
+ delete newCommentsByEntityId[comment.id];
29
+
30
+ return {
31
+ ...currentState,
32
+ comments_by_entity_id: newCommentsByEntityId,
33
+ };
34
+ });
35
+ }
@@ -0,0 +1,61 @@
1
+ import { Feed } from '../../../feed';
2
+ import { CommentResponse } from '../../../gen/models';
3
+ import { EventPayload } from '../../../types-internal';
4
+
5
+ export function handleCommentReaction(
6
+ this: Feed,
7
+ event: EventPayload<
8
+ 'feeds.comment.reaction.added' | 'feeds.comment.reaction.deleted'
9
+ >,
10
+ ) {
11
+ const { comment, reaction } = event;
12
+ const connectedUser = this.client.state.getLatestValue().connected_user;
13
+
14
+ this.state.next((currentState) => {
15
+ const forId = comment.parent_id ?? comment.object_id;
16
+ const entityState = currentState.comments_by_entity_id[forId];
17
+
18
+ const commentIndex = this.getCommentIndex(comment, currentState);
19
+
20
+ if (commentIndex === -1) return currentState;
21
+
22
+ const newComments = entityState?.comments?.concat([]) ?? [];
23
+
24
+ const commentCopy: Partial<CommentResponse> = { ...comment };
25
+
26
+ delete commentCopy.own_reactions;
27
+
28
+ const newComment: CommentResponse = {
29
+ ...newComments[commentIndex],
30
+ ...commentCopy,
31
+ // TODO: FIXME this should be handled by the backend
32
+ latest_reactions: commentCopy.latest_reactions ?? [],
33
+ reaction_groups: commentCopy.reaction_groups ?? {},
34
+ };
35
+
36
+ newComments[commentIndex] = newComment;
37
+
38
+ if (reaction.user.id === connectedUser?.id) {
39
+ if (event.type === 'feeds.comment.reaction.added') {
40
+ newComment.own_reactions = newComment.own_reactions.concat(
41
+ reaction,
42
+ ) ?? [reaction];
43
+ } else if (event.type === 'feeds.comment.reaction.deleted') {
44
+ newComment.own_reactions = newComment.own_reactions.filter(
45
+ (r) => r.type !== reaction.type,
46
+ );
47
+ }
48
+ }
49
+
50
+ return {
51
+ ...currentState,
52
+ comments_by_entity_id: {
53
+ ...currentState.comments_by_entity_id,
54
+ [forId]: {
55
+ ...entityState,
56
+ comments: newComments,
57
+ },
58
+ },
59
+ };
60
+ });
61
+ }
@@ -0,0 +1,35 @@
1
+ import { Feed } from '../../../feed';
2
+ import { EventPayload } from '../../../types-internal';
3
+
4
+ export function handleCommentUpdated(
5
+ this: Feed,
6
+ event: EventPayload<'feeds.comment.updated'>,
7
+ ) {
8
+ const { comment } = event;
9
+ const entityId = comment.parent_id ?? comment.object_id;
10
+
11
+ this.state.next((currentState) => {
12
+ const entityState = currentState.comments_by_entity_id[entityId];
13
+
14
+ if (!entityState?.comments?.length) return currentState;
15
+
16
+ const index = this.getCommentIndex(comment, currentState);
17
+
18
+ if (index === -1) return currentState;
19
+
20
+ const newComments = [...entityState.comments];
21
+
22
+ newComments[index] = comment;
23
+
24
+ return {
25
+ ...currentState,
26
+ comments_by_entity_id: {
27
+ ...currentState.comments_by_entity_id,
28
+ [entityId]: {
29
+ ...currentState.comments_by_entity_id[entityId],
30
+ comments: newComments,
31
+ },
32
+ },
33
+ };
34
+ });
35
+ }
@@ -0,0 +1,4 @@
1
+ export * from './handle-comment-added';
2
+ export * from './handle-comment-deleted';
3
+ export * from './handle-comment-updated';
4
+ export * from './handle-comment-reaction';
@@ -0,0 +1,9 @@
1
+ import { Feed } from '../../../feed';
2
+ import { EventPayload } from '../../../types-internal';
3
+
4
+ export function handleFeedUpdated(
5
+ this: Feed,
6
+ event: EventPayload<'feeds.feed.updated'>,
7
+ ) {
8
+ this.state.partialNext({ ...event.feed });
9
+ }
@@ -0,0 +1 @@
1
+ export * from './handle-feed-updated';
@@ -0,0 +1,31 @@
1
+ import { Feed, FeedState } from '../../../feed';
2
+ import { EventPayload } from '../../../types-internal';
3
+
4
+ export function handleFeedMemberAdded(
5
+ this: Feed,
6
+ event: EventPayload<'feeds.feed_member.added'>,
7
+ ) {
8
+ const { connected_user: connectedUser } = this.client.state.getLatestValue();
9
+
10
+ this.state.next((currentState) => {
11
+ let newState: FeedState | undefined;
12
+
13
+ if (typeof currentState.members !== 'undefined') {
14
+ newState ??= {
15
+ ...currentState,
16
+ };
17
+
18
+ newState.members = [event.member, ...currentState.members];
19
+ }
20
+
21
+ if (connectedUser?.id === event.member.user.id) {
22
+ newState ??= {
23
+ ...currentState,
24
+ };
25
+
26
+ newState.own_membership = event.member;
27
+ }
28
+
29
+ return newState ?? currentState;
30
+ });
31
+ }
@@ -0,0 +1,24 @@
1
+ import { Feed } from '../../../feed';
2
+ import { EventPayload } from '../../../types-internal';
3
+
4
+ export function handleFeedMemberRemoved(
5
+ this: Feed,
6
+ event: EventPayload<'feeds.feed_member.removed'>,
7
+ ) {
8
+ const { connected_user: connectedUser } = this.client.state.getLatestValue();
9
+
10
+ this.state.next((currentState) => {
11
+ const newState = {
12
+ ...currentState,
13
+ members: currentState.members?.filter(
14
+ (member) => member.user.id !== event.user?.id,
15
+ ),
16
+ };
17
+
18
+ if (connectedUser?.id === event.member_id) {
19
+ delete newState.own_membership;
20
+ }
21
+
22
+ return newState;
23
+ });
24
+ }
@@ -0,0 +1,40 @@
1
+ import { Feed, FeedState } from '../../../feed';
2
+ import { EventPayload } from '../../../types-internal';
3
+
4
+ export function handleFeedMemberUpdated(
5
+ this: Feed,
6
+ event: EventPayload<'feeds.feed_member.updated'>,
7
+ ) {
8
+ const { connected_user: connectedUser } = this.client.state.getLatestValue();
9
+
10
+ this.state.next((currentState) => {
11
+ const memberIndex =
12
+ currentState.members?.findIndex(
13
+ (member) => member.user.id === event.member.user.id,
14
+ ) ?? -1;
15
+
16
+ let newState: FeedState | undefined;
17
+
18
+ if (memberIndex !== -1) {
19
+ // if there's an index, there's a member to update
20
+ const newMembers = [...currentState.members!];
21
+ newMembers[memberIndex] = event.member;
22
+
23
+ newState ??= {
24
+ ...currentState,
25
+ };
26
+
27
+ newState.members = newMembers;
28
+ }
29
+
30
+ if (connectedUser?.id === event.member.user.id) {
31
+ newState ??= {
32
+ ...currentState,
33
+ };
34
+
35
+ newState.own_membership = event.member;
36
+ }
37
+
38
+ return newState ?? currentState;
39
+ });
40
+ }
@@ -0,0 +1,3 @@
1
+ export * from './handle-feed-member-added';
2
+ export * from './handle-feed-member-updated';
3
+ export * from './handle-feed-member-removed';
@@ -0,0 +1,250 @@
1
+ import {
2
+ FeedResponse,
3
+ FollowResponse,
4
+ UserResponse,
5
+ } from '../../../gen/models';
6
+ import { generateFollowResponse } from '../../../test-utils';
7
+ import { updateStateFollowCreated } from './handle-follow-created';
8
+
9
+ import { describe, it, expect, beforeEach } from 'vitest';
10
+
11
+ describe('handle-follow-created', () => {
12
+ describe(updateStateFollowCreated.name, () => {
13
+ let mockFollow: FollowResponse;
14
+ let mockFeed: FeedResponse;
15
+ let mockUser: UserResponse;
16
+
17
+ beforeEach(() => {
18
+ mockFollow = generateFollowResponse();
19
+ mockFeed = mockFollow.source_feed;
20
+ mockUser = mockFeed.created_by;
21
+ });
22
+
23
+ it('should return unchanged state for non-accepted follows', () => {
24
+ const follow: FollowResponse = {
25
+ ...mockFollow,
26
+ status: 'pending',
27
+ };
28
+
29
+ // @ts-expect-error - we're not testing the full state here
30
+ const currentState: FeedState = {
31
+ followers: [],
32
+ following: [],
33
+ };
34
+
35
+ const result = updateStateFollowCreated(
36
+ follow,
37
+ currentState,
38
+ 'user:feed-1',
39
+ 'user-1',
40
+ );
41
+
42
+ expect(result.changed).toBe(false);
43
+ });
44
+
45
+ it('should handle when this feed follows someone', () => {
46
+ const follow: FollowResponse = {
47
+ ...mockFollow,
48
+ source_feed: {
49
+ ...mockFeed,
50
+ id: 'feed-x',
51
+ feed: 'user:feed-x',
52
+ created_by: {
53
+ ...mockUser,
54
+ id: 'user-x',
55
+ },
56
+ following_count: 1,
57
+ },
58
+ target_feed: {
59
+ ...mockFeed,
60
+ id: 'other-feed',
61
+ feed: 'user:other-feed',
62
+ created_by: mockUser,
63
+ },
64
+ };
65
+
66
+ // @ts-expect-error - we're not testing the full state here
67
+ const currentState: FeedState = {
68
+ following: [],
69
+ following_count: 0,
70
+ };
71
+
72
+ const result = updateStateFollowCreated(
73
+ follow,
74
+ currentState,
75
+ 'user:feed-x',
76
+ 'user-1',
77
+ );
78
+
79
+ expect(result.changed).toBe(true);
80
+ expect(result.data.following).toHaveLength(1);
81
+ expect(result.data.following?.[0]).toEqual(follow);
82
+ expect(result.data).toMatchObject(follow.source_feed);
83
+ expect(result.data.own_follows).toBeUndefined();
84
+ expect(result.data.following_count).toEqual(1);
85
+ });
86
+
87
+ it('should handle when someone follows this feed', () => {
88
+ const follow: FollowResponse = {
89
+ ...mockFollow,
90
+ source_feed: {
91
+ ...mockFeed,
92
+ id: 'other-feed',
93
+ feed: 'user:other-feed',
94
+ created_by: {
95
+ ...mockUser,
96
+ id: 'other-user',
97
+ },
98
+ },
99
+ target_feed: {
100
+ ...mockFeed,
101
+ id: 'feed-1',
102
+ feed: 'user:feed-1',
103
+ created_by: mockUser,
104
+ follower_count: 1,
105
+ },
106
+ };
107
+
108
+ // @ts-expect-error - we're not testing the full state here
109
+ const currentState: FeedState = {
110
+ followers: [],
111
+ follower_count: 0,
112
+ };
113
+
114
+ const result = updateStateFollowCreated(
115
+ follow,
116
+ currentState,
117
+ 'user:feed-1',
118
+ 'user-1',
119
+ );
120
+
121
+ expect(result.changed).toBe(true);
122
+ expect(result.data.followers).toHaveLength(1);
123
+ expect(result.data.followers?.[0]).toEqual(follow);
124
+ expect(result.data).toMatchObject(follow.target_feed);
125
+ expect(result.data.own_follows).toBeUndefined();
126
+ expect(result.data.follower_count).toEqual(1);
127
+ });
128
+
129
+ it('should add to own_follows when connected user is the source', () => {
130
+ const follow: FollowResponse = {
131
+ ...mockFollow,
132
+ source_feed: {
133
+ ...mockFeed,
134
+ id: 'feed-1',
135
+ feed: 'user:feed-1',
136
+ created_by: { ...mockUser, id: 'user-1' },
137
+ },
138
+ target_feed: {
139
+ ...mockFeed,
140
+ id: 'feed-x',
141
+ feed: 'user:feed-x',
142
+ created_by: {
143
+ ...mockUser,
144
+ id: 'user-x',
145
+ },
146
+ },
147
+ };
148
+
149
+ // @ts-expect-error - we're not testing the full state here
150
+ const currentState: FeedState = {
151
+ followers: [],
152
+ own_follows: [],
153
+ };
154
+
155
+ const result = updateStateFollowCreated(
156
+ follow,
157
+ currentState,
158
+ 'user:feed-x',
159
+ 'user-1',
160
+ );
161
+
162
+ expect(result.changed).toBe(true);
163
+ expect(result.data.own_follows).toHaveLength(1);
164
+ expect(result.data.own_follows?.[0]).toEqual(follow);
165
+ });
166
+
167
+ it('should not update followers/following when they are undefined', () => {
168
+ const follow: FollowResponse = {
169
+ ...mockFollow,
170
+ source_feed: {
171
+ ...mockFeed,
172
+ id: 'other-feed',
173
+ feed: 'user:other-feed',
174
+ created_by: mockUser,
175
+ },
176
+ target_feed: {
177
+ ...mockFeed,
178
+ id: 'feed-1',
179
+ feed: 'user:feed-1',
180
+ created_by: mockUser,
181
+ },
182
+ };
183
+
184
+ // @ts-expect-error - we're not testing the full state here
185
+ const currentState: FeedState = {
186
+ followers: undefined,
187
+ following: undefined,
188
+ own_follows: undefined,
189
+ };
190
+
191
+ const result = updateStateFollowCreated(
192
+ follow,
193
+ currentState,
194
+ 'user:feed-1',
195
+ 'user-1',
196
+ );
197
+
198
+ expect(result.changed).toBe(true);
199
+ expect(result.data.followers).toBeUndefined();
200
+ expect(result.data).toMatchObject(follow.target_feed);
201
+ });
202
+
203
+ it('should add new followers to the top of existing arrays', () => {
204
+ const existingFollow: FollowResponse = {
205
+ ...mockFollow,
206
+ source_feed: {
207
+ ...mockFeed,
208
+ id: 'existing-feed',
209
+ feed: 'user:existing-feed',
210
+ created_by: mockUser,
211
+ },
212
+ };
213
+
214
+ const follow: FollowResponse = {
215
+ ...mockFollow,
216
+ source_feed: {
217
+ ...mockFeed,
218
+ id: 'other-feed',
219
+ feed: 'user:other-feed',
220
+ created_by: mockUser,
221
+ },
222
+ target_feed: {
223
+ ...mockFeed,
224
+ id: 'feed-1',
225
+ feed: 'user:feed-1',
226
+ created_by: mockUser,
227
+ },
228
+ };
229
+
230
+ // @ts-expect-error - we're not testing the full state here
231
+ const currentState: FeedState = {
232
+ followers: [existingFollow],
233
+ following: undefined,
234
+ own_follows: undefined,
235
+ };
236
+
237
+ const result = updateStateFollowCreated(
238
+ follow,
239
+ currentState,
240
+ 'user:feed-1',
241
+ 'user-1',
242
+ );
243
+
244
+ expect(result.changed).toBe(true);
245
+ expect(result.data.followers).toHaveLength(2);
246
+ expect(result.data.followers?.[0]).toEqual(follow);
247
+ expect(result.data.followers?.[1]).toEqual(existingFollow);
248
+ });
249
+ });
250
+ });