@btst/stack 2.7.0 → 2.8.1

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 (181) hide show
  1. package/README.md +1 -0
  2. package/dist/packages/stack/src/plugins/blog/client/components/loading/post-navigation-skeleton.cjs +13 -0
  3. package/dist/packages/stack/src/plugins/blog/client/components/loading/post-navigation-skeleton.mjs +11 -0
  4. package/dist/packages/stack/src/plugins/blog/client/components/loading/recent-posts-carousel-skeleton.cjs +17 -0
  5. package/dist/packages/stack/src/plugins/blog/client/components/loading/recent-posts-carousel-skeleton.mjs +15 -0
  6. package/dist/packages/stack/src/plugins/blog/client/components/pages/post-page.internal.cjs +18 -7
  7. package/dist/packages/stack/src/plugins/blog/client/components/pages/post-page.internal.mjs +18 -7
  8. package/dist/packages/stack/src/plugins/blog/client/components/shared/post-navigation.cjs +48 -52
  9. package/dist/packages/stack/src/plugins/blog/client/components/shared/post-navigation.mjs +49 -53
  10. package/dist/packages/stack/src/plugins/blog/client/components/shared/recent-posts-carousel.cjs +34 -37
  11. package/dist/packages/stack/src/plugins/blog/client/components/shared/recent-posts-carousel.mjs +35 -38
  12. package/dist/packages/stack/src/plugins/blog/client/hooks/blog-hooks.cjs +4 -21
  13. package/dist/packages/stack/src/plugins/blog/client/hooks/blog-hooks.mjs +4 -21
  14. package/dist/packages/stack/src/plugins/comments/api/getters.cjs +284 -0
  15. package/dist/packages/stack/src/plugins/comments/api/getters.mjs +280 -0
  16. package/dist/packages/stack/src/plugins/comments/api/mutations.cjs +118 -0
  17. package/dist/packages/stack/src/plugins/comments/api/mutations.mjs +112 -0
  18. package/dist/packages/stack/src/plugins/comments/api/plugin.cjs +335 -0
  19. package/dist/packages/stack/src/plugins/comments/api/plugin.mjs +333 -0
  20. package/dist/packages/stack/src/plugins/comments/api/query-key-defs.cjs +60 -0
  21. package/dist/packages/stack/src/plugins/comments/api/query-key-defs.mjs +55 -0
  22. package/dist/packages/stack/src/plugins/comments/api/serializers.cjs +23 -0
  23. package/dist/packages/stack/src/plugins/comments/api/serializers.mjs +21 -0
  24. package/dist/packages/stack/src/plugins/comments/client/components/comment-count.cjs +46 -0
  25. package/dist/packages/stack/src/plugins/comments/client/components/comment-count.mjs +44 -0
  26. package/dist/packages/stack/src/plugins/comments/client/components/comment-form.cjs +86 -0
  27. package/dist/packages/stack/src/plugins/comments/client/components/comment-form.mjs +84 -0
  28. package/dist/packages/stack/src/plugins/comments/client/components/comment-thread.cjs +540 -0
  29. package/dist/packages/stack/src/plugins/comments/client/components/comment-thread.mjs +538 -0
  30. package/dist/packages/stack/src/plugins/comments/client/components/pages/moderation-page.cjs +64 -0
  31. package/dist/packages/stack/src/plugins/comments/client/components/pages/moderation-page.internal.cjs +426 -0
  32. package/dist/packages/stack/src/plugins/comments/client/components/pages/moderation-page.internal.mjs +424 -0
  33. package/dist/packages/stack/src/plugins/comments/client/components/pages/moderation-page.mjs +62 -0
  34. package/dist/packages/stack/src/plugins/comments/client/components/pages/my-comments-page.cjs +66 -0
  35. package/dist/packages/stack/src/plugins/comments/client/components/pages/my-comments-page.internal.cjs +256 -0
  36. package/dist/packages/stack/src/plugins/comments/client/components/pages/my-comments-page.internal.mjs +254 -0
  37. package/dist/packages/stack/src/plugins/comments/client/components/pages/my-comments-page.mjs +64 -0
  38. package/dist/packages/stack/src/plugins/comments/client/components/pages/resource-comments-page.cjs +86 -0
  39. package/dist/packages/stack/src/plugins/comments/client/components/pages/resource-comments-page.internal.cjs +191 -0
  40. package/dist/packages/stack/src/plugins/comments/client/components/pages/resource-comments-page.internal.mjs +189 -0
  41. package/dist/packages/stack/src/plugins/comments/client/components/pages/resource-comments-page.mjs +84 -0
  42. package/dist/packages/stack/src/plugins/comments/client/components/shared/page-wrapper.cjs +27 -0
  43. package/dist/packages/stack/src/plugins/comments/client/components/shared/page-wrapper.mjs +25 -0
  44. package/dist/packages/stack/src/plugins/comments/client/components/shared/pagination.cjs +37 -0
  45. package/dist/packages/stack/src/plugins/comments/client/components/shared/pagination.mjs +35 -0
  46. package/dist/packages/stack/src/plugins/comments/client/hooks/use-comments.cjs +476 -0
  47. package/dist/packages/stack/src/plugins/comments/client/hooks/use-comments.mjs +464 -0
  48. package/dist/packages/stack/src/plugins/comments/client/localization/comments-moderation.cjs +67 -0
  49. package/dist/packages/stack/src/plugins/comments/client/localization/comments-moderation.mjs +65 -0
  50. package/dist/packages/stack/src/plugins/comments/client/localization/comments-my.cjs +27 -0
  51. package/dist/packages/stack/src/plugins/comments/client/localization/comments-my.mjs +25 -0
  52. package/dist/packages/stack/src/plugins/comments/client/localization/comments-thread.cjs +30 -0
  53. package/dist/packages/stack/src/plugins/comments/client/localization/comments-thread.mjs +28 -0
  54. package/dist/packages/stack/src/plugins/comments/client/localization/index.cjs +13 -0
  55. package/dist/packages/stack/src/plugins/comments/client/localization/index.mjs +11 -0
  56. package/dist/packages/stack/src/plugins/comments/client/plugin.cjs +116 -0
  57. package/dist/packages/stack/src/plugins/comments/client/plugin.mjs +114 -0
  58. package/dist/packages/stack/src/plugins/comments/client/utils.cjs +41 -0
  59. package/dist/packages/stack/src/plugins/comments/client/utils.mjs +37 -0
  60. package/dist/packages/stack/src/plugins/comments/db.cjs +75 -0
  61. package/dist/packages/stack/src/plugins/comments/db.mjs +73 -0
  62. package/dist/packages/stack/src/plugins/comments/schemas.cjs +45 -0
  63. package/dist/packages/stack/src/plugins/comments/schemas.mjs +38 -0
  64. package/dist/packages/stack/src/plugins/kanban/client/components/forms/task-form.cjs +0 -1
  65. package/dist/packages/stack/src/plugins/kanban/client/components/forms/task-form.mjs +0 -1
  66. package/dist/packages/stack/src/plugins/kanban/client/components/pages/board-page.internal.cjs +39 -22
  67. package/dist/packages/stack/src/plugins/kanban/client/components/pages/board-page.internal.mjs +40 -23
  68. package/dist/packages/ui/src/components/avatar.mjs +1 -1
  69. package/dist/packages/ui/src/components/pagination-controls.cjs +64 -0
  70. package/dist/packages/ui/src/components/pagination-controls.mjs +62 -0
  71. package/dist/packages/ui/src/components/when-visible.cjs +39 -0
  72. package/dist/packages/ui/src/components/when-visible.mjs +37 -0
  73. package/dist/plugins/blog/client/hooks/index.d.cts +1 -1
  74. package/dist/plugins/blog/client/hooks/index.d.mts +1 -1
  75. package/dist/plugins/blog/client/hooks/index.d.ts +1 -1
  76. package/dist/plugins/blog/client/index.d.cts +24 -2
  77. package/dist/plugins/blog/client/index.d.mts +24 -2
  78. package/dist/plugins/blog/client/index.d.ts +24 -2
  79. package/dist/plugins/comments/api/index.cjs +21 -0
  80. package/dist/plugins/comments/api/index.d.cts +126 -0
  81. package/dist/plugins/comments/api/index.d.mts +126 -0
  82. package/dist/plugins/comments/api/index.d.ts +126 -0
  83. package/dist/plugins/comments/api/index.mjs +5 -0
  84. package/dist/plugins/comments/client/components/index.cjs +15 -0
  85. package/dist/plugins/comments/client/components/index.d.cts +125 -0
  86. package/dist/plugins/comments/client/components/index.d.mts +125 -0
  87. package/dist/plugins/comments/client/components/index.d.ts +125 -0
  88. package/dist/plugins/comments/client/components/index.mjs +5 -0
  89. package/dist/plugins/comments/client/hooks/index.cjs +17 -0
  90. package/dist/plugins/comments/client/hooks/index.d.cts +200 -0
  91. package/dist/plugins/comments/client/hooks/index.d.mts +200 -0
  92. package/dist/plugins/comments/client/hooks/index.d.ts +200 -0
  93. package/dist/plugins/comments/client/hooks/index.mjs +1 -0
  94. package/dist/plugins/comments/client/index.cjs +9 -0
  95. package/dist/plugins/comments/client/index.d.cts +262 -0
  96. package/dist/plugins/comments/client/index.d.mts +262 -0
  97. package/dist/plugins/comments/client/index.d.ts +262 -0
  98. package/dist/plugins/comments/client/index.mjs +2 -0
  99. package/dist/plugins/comments/client.css +2 -0
  100. package/dist/plugins/comments/query-keys.cjs +113 -0
  101. package/dist/plugins/comments/query-keys.d.cts +71 -0
  102. package/dist/plugins/comments/query-keys.d.mts +71 -0
  103. package/dist/plugins/comments/query-keys.d.ts +71 -0
  104. package/dist/plugins/comments/query-keys.mjs +111 -0
  105. package/dist/plugins/comments/style.css +15 -0
  106. package/dist/plugins/kanban/api/index.d.cts +1 -1
  107. package/dist/plugins/kanban/api/index.d.mts +1 -1
  108. package/dist/plugins/kanban/api/index.d.ts +1 -1
  109. package/dist/plugins/kanban/client/hooks/index.d.cts +1 -1
  110. package/dist/plugins/kanban/client/hooks/index.d.mts +1 -1
  111. package/dist/plugins/kanban/client/hooks/index.d.ts +1 -1
  112. package/dist/plugins/kanban/client/index.d.cts +1 -1
  113. package/dist/plugins/kanban/client/index.d.mts +1 -1
  114. package/dist/plugins/kanban/client/index.d.ts +1 -1
  115. package/dist/plugins/kanban/query-keys.d.cts +1 -1
  116. package/dist/plugins/kanban/query-keys.d.mts +1 -1
  117. package/dist/plugins/kanban/query-keys.d.ts +1 -1
  118. package/dist/shared/{stack.FeaWkglm.d.ts → stack.BxFl46lB.d.cts} +24 -1
  119. package/dist/shared/stack.C-b3Sn8j.d.cts +142 -0
  120. package/dist/shared/stack.C-b3Sn8j.d.mts +142 -0
  121. package/dist/shared/stack.C-b3Sn8j.d.ts +142 -0
  122. package/dist/shared/stack.CJE9sAjV.d.ts +335 -0
  123. package/dist/shared/stack.CmHRdhl8.d.cts +335 -0
  124. package/dist/shared/{stack.CNLHlv7r.d.mts → stack.DOZ1EXjM.d.mts} +6 -12
  125. package/dist/shared/{stack.FeaWkglm.d.mts → stack.DRpeDS6X.d.ts} +24 -1
  126. package/dist/shared/{stack.CQAZwXhV.d.cts → stack.DX-tQ93o.d.cts} +6 -12
  127. package/dist/shared/stack.Dcz6636A.d.mts +335 -0
  128. package/dist/shared/{stack.FeaWkglm.d.cts → stack.Jb0kQDJC.d.mts} +24 -1
  129. package/dist/shared/stack.Ldfkr5b2.d.cts +112 -0
  130. package/dist/shared/stack.Ldfkr5b2.d.mts +112 -0
  131. package/dist/shared/stack.Ldfkr5b2.d.ts +112 -0
  132. package/dist/shared/{stack.D3BsrpAz.d.ts → stack.VF6FhyZw.d.ts} +6 -12
  133. package/package.json +69 -4
  134. package/src/plugins/blog/client/components/loading/post-navigation-skeleton.tsx +10 -0
  135. package/src/plugins/blog/client/components/loading/recent-posts-carousel-skeleton.tsx +18 -0
  136. package/src/plugins/blog/client/components/pages/post-page.internal.tsx +23 -8
  137. package/src/plugins/blog/client/components/shared/post-navigation.tsx +0 -5
  138. package/src/plugins/blog/client/components/shared/recent-posts-carousel.tsx +1 -5
  139. package/src/plugins/blog/client/hooks/blog-hooks.tsx +8 -33
  140. package/src/plugins/blog/client/overrides.ts +26 -1
  141. package/src/plugins/cms/client/components/shared/pagination.tsx +14 -42
  142. package/src/plugins/comments/api/getters.ts +444 -0
  143. package/src/plugins/comments/api/index.ts +21 -0
  144. package/src/plugins/comments/api/mutations.ts +206 -0
  145. package/src/plugins/comments/api/plugin.ts +628 -0
  146. package/src/plugins/comments/api/query-key-defs.ts +143 -0
  147. package/src/plugins/comments/api/serializers.ts +37 -0
  148. package/src/plugins/comments/client/components/comment-count.tsx +66 -0
  149. package/src/plugins/comments/client/components/comment-form.tsx +112 -0
  150. package/src/plugins/comments/client/components/comment-thread.tsx +799 -0
  151. package/src/plugins/comments/client/components/index.tsx +11 -0
  152. package/src/plugins/comments/client/components/pages/moderation-page.internal.tsx +550 -0
  153. package/src/plugins/comments/client/components/pages/moderation-page.tsx +70 -0
  154. package/src/plugins/comments/client/components/pages/my-comments-page.internal.tsx +367 -0
  155. package/src/plugins/comments/client/components/pages/my-comments-page.tsx +72 -0
  156. package/src/plugins/comments/client/components/pages/resource-comments-page.internal.tsx +225 -0
  157. package/src/plugins/comments/client/components/pages/resource-comments-page.tsx +97 -0
  158. package/src/plugins/comments/client/components/shared/page-wrapper.tsx +32 -0
  159. package/src/plugins/comments/client/components/shared/pagination.tsx +44 -0
  160. package/src/plugins/comments/client/hooks/index.tsx +13 -0
  161. package/src/plugins/comments/client/hooks/use-comments.tsx +717 -0
  162. package/src/plugins/comments/client/index.ts +14 -0
  163. package/src/plugins/comments/client/localization/comments-moderation.ts +75 -0
  164. package/src/plugins/comments/client/localization/comments-my.ts +32 -0
  165. package/src/plugins/comments/client/localization/comments-thread.ts +32 -0
  166. package/src/plugins/comments/client/localization/index.ts +11 -0
  167. package/src/plugins/comments/client/overrides.ts +164 -0
  168. package/src/plugins/comments/client/plugin.tsx +195 -0
  169. package/src/plugins/comments/client/utils.ts +67 -0
  170. package/src/plugins/comments/client.css +2 -0
  171. package/src/plugins/comments/db.ts +77 -0
  172. package/src/plugins/comments/query-keys.ts +189 -0
  173. package/src/plugins/comments/schemas.ts +72 -0
  174. package/src/plugins/comments/style.css +15 -0
  175. package/src/plugins/comments/types.ts +73 -0
  176. package/src/plugins/kanban/client/components/forms/task-form.tsx +0 -1
  177. package/src/plugins/kanban/client/components/pages/board-page.internal.tsx +46 -27
  178. package/src/plugins/kanban/client/overrides.ts +27 -1
  179. package/dist/shared/{stack.Rtcvl8sS.d.cts → stack.BOokfhZD.d.cts} +3 -3
  180. package/dist/shared/{stack.D4Cea8II.d.ts → stack.BvCR4-9H.d.ts} +3 -3
  181. package/dist/shared/{stack.HE_IvqV5.d.mts → stack.CWxAl9K3.d.mts} +3 -3
@@ -0,0 +1,538 @@
1
+ "use client";
2
+ import { jsx, jsxs, Fragment } from 'react/jsx-runtime';
3
+ import { useState, useEffect } from 'react';
4
+ import { WhenVisible } from '../../../../../../ui/src/components/when-visible.mjs';
5
+ import { Avatar, AvatarImage, AvatarFallback } from '../../../../../../ui/src/components/avatar.mjs';
6
+ import { Badge } from '../../../../../../ui/src/components/badge.mjs';
7
+ import { Button } from '../../../../../../ui/src/components/button.mjs';
8
+ import { Separator } from '../../../../../../ui/src/components/separator.mjs';
9
+ import { MessageSquare, LogIn, Heart, Pencil, X, ChevronUp, ChevronDown } from 'lucide-react';
10
+ import { formatDistanceToNow } from 'date-fns';
11
+ import { getInitials } from '../utils.mjs';
12
+ import { CommentForm } from './comment-form.mjs';
13
+ import { useInfiniteComments, usePostComment, useUpdateComment, useDeleteComment, useToggleLike, useComments } from '../hooks/use-comments.mjs';
14
+ import { COMMENTS_LOCALIZATION } from '../localization/index.mjs';
15
+ import { usePluginOverrides } from '@btst/stack/context';
16
+
17
+ const DEFAULT_RENDERER = ({ body }) => /* @__PURE__ */ jsx("p", { className: "text-sm whitespace-pre-wrap wrap-break-word", children: body });
18
+ function CommentCard({
19
+ comment,
20
+ currentUserId,
21
+ apiBaseURL,
22
+ apiBasePath,
23
+ resourceId,
24
+ resourceType,
25
+ headers,
26
+ components,
27
+ loc,
28
+ infiniteKey,
29
+ onReplyClick,
30
+ allowPosting,
31
+ allowEditing
32
+ }) {
33
+ const [isEditing, setIsEditing] = useState(false);
34
+ const Renderer = components?.Renderer ?? DEFAULT_RENDERER;
35
+ const config = { apiBaseURL, apiBasePath, headers };
36
+ const updateMutation = useUpdateComment(config);
37
+ const deleteMutation = useDeleteComment(config);
38
+ const toggleLikeMutation = useToggleLike(config, {
39
+ resourceId,
40
+ resourceType,
41
+ parentId: comment.parentId,
42
+ currentUserId,
43
+ infiniteKey
44
+ });
45
+ const isOwn = currentUserId && comment.authorId === currentUserId;
46
+ const isPending = comment.status === "pending";
47
+ const isApproved = comment.status === "approved";
48
+ const handleEdit = async (body) => {
49
+ await updateMutation.mutateAsync({ id: comment.id, body });
50
+ setIsEditing(false);
51
+ };
52
+ const handleDelete = async () => {
53
+ if (!window.confirm(loc.COMMENTS_DELETE_CONFIRM)) return;
54
+ await deleteMutation.mutateAsync(comment.id);
55
+ };
56
+ const handleLike = () => {
57
+ if (!currentUserId) return;
58
+ toggleLikeMutation.mutate({
59
+ commentId: comment.id,
60
+ authorId: currentUserId
61
+ });
62
+ };
63
+ return /* @__PURE__ */ jsxs(
64
+ "div",
65
+ {
66
+ className: "flex gap-3 py-3",
67
+ "data-testid": "comment-card",
68
+ "data-comment-id": comment.id,
69
+ children: [
70
+ /* @__PURE__ */ jsxs(Avatar, { className: "h-8 w-8 shrink-0 mt-0.5", children: [
71
+ comment.resolvedAvatarUrl && /* @__PURE__ */ jsx(
72
+ AvatarImage,
73
+ {
74
+ src: comment.resolvedAvatarUrl,
75
+ alt: comment.resolvedAuthorName
76
+ }
77
+ ),
78
+ /* @__PURE__ */ jsx(AvatarFallback, { className: "text-xs", children: getInitials(comment.resolvedAuthorName) })
79
+ ] }),
80
+ /* @__PURE__ */ jsxs("div", { className: "flex-1 min-w-0", children: [
81
+ /* @__PURE__ */ jsxs("div", { className: "flex flex-wrap items-center gap-2 mb-1", children: [
82
+ /* @__PURE__ */ jsx("span", { className: "text-sm font-medium", children: comment.resolvedAuthorName }),
83
+ /* @__PURE__ */ jsx("span", { className: "text-xs text-muted-foreground", children: formatDistanceToNow(new Date(comment.createdAt), {
84
+ addSuffix: true
85
+ }) }),
86
+ comment.editedAt && /* @__PURE__ */ jsx("span", { className: "text-xs text-muted-foreground italic", children: loc.COMMENTS_EDITED_BADGE }),
87
+ isPending && isOwn && /* @__PURE__ */ jsx(
88
+ Badge,
89
+ {
90
+ variant: "secondary",
91
+ className: "text-xs",
92
+ "data-testid": "pending-badge",
93
+ children: loc.COMMENTS_PENDING_BADGE
94
+ }
95
+ )
96
+ ] }),
97
+ isEditing ? /* @__PURE__ */ jsx(
98
+ CommentForm,
99
+ {
100
+ authorId: currentUserId ?? "",
101
+ initialBody: comment.body,
102
+ submitLabel: loc.COMMENTS_SAVE_EDIT,
103
+ InputComponent: components?.Input,
104
+ localization: loc,
105
+ onSubmit: handleEdit,
106
+ onCancel: () => setIsEditing(false)
107
+ }
108
+ ) : /* @__PURE__ */ jsx(Renderer, { body: comment.body }),
109
+ !isEditing && /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-1 mt-2", children: [
110
+ currentUserId && isApproved && /* @__PURE__ */ jsxs(
111
+ Button,
112
+ {
113
+ variant: "ghost",
114
+ size: "sm",
115
+ className: "h-7 px-2 text-xs gap-1",
116
+ onClick: handleLike,
117
+ "aria-label": comment.isLikedByCurrentUser ? loc.COMMENTS_UNLIKE_ARIA : loc.COMMENTS_LIKE_ARIA,
118
+ "data-testid": "like-button",
119
+ children: [
120
+ /* @__PURE__ */ jsx(
121
+ Heart,
122
+ {
123
+ className: `h-3.5 w-3.5 ${comment.isLikedByCurrentUser ? "fill-current text-red-500" : ""}`
124
+ }
125
+ ),
126
+ comment.likes > 0 && /* @__PURE__ */ jsx("span", { "data-testid": "like-count", children: comment.likes })
127
+ ]
128
+ }
129
+ ),
130
+ allowPosting && currentUserId && !comment.parentId && isApproved && /* @__PURE__ */ jsxs(
131
+ Button,
132
+ {
133
+ variant: "ghost",
134
+ size: "sm",
135
+ className: "h-7 px-2 text-xs",
136
+ onClick: () => onReplyClick(comment.id),
137
+ "data-testid": "reply-button",
138
+ children: [
139
+ /* @__PURE__ */ jsx(MessageSquare, { className: "h-3.5 w-3.5 mr-1" }),
140
+ loc.COMMENTS_REPLY_BUTTON
141
+ ]
142
+ }
143
+ ),
144
+ isOwn && /* @__PURE__ */ jsxs(Fragment, { children: [
145
+ allowEditing && isApproved && /* @__PURE__ */ jsxs(
146
+ Button,
147
+ {
148
+ variant: "ghost",
149
+ size: "sm",
150
+ className: "h-7 px-2 text-xs",
151
+ onClick: () => setIsEditing(true),
152
+ "data-testid": "edit-button",
153
+ children: [
154
+ /* @__PURE__ */ jsx(Pencil, { className: "h-3.5 w-3.5 mr-1" }),
155
+ loc.COMMENTS_EDIT_BUTTON
156
+ ]
157
+ }
158
+ ),
159
+ /* @__PURE__ */ jsxs(
160
+ Button,
161
+ {
162
+ variant: "ghost",
163
+ size: "sm",
164
+ className: "h-7 px-2 text-xs text-destructive hover:text-destructive",
165
+ onClick: handleDelete,
166
+ disabled: deleteMutation.isPending,
167
+ "data-testid": "delete-button",
168
+ children: [
169
+ /* @__PURE__ */ jsx(X, { className: "h-3.5 w-3.5 mr-1" }),
170
+ loc.COMMENTS_DELETE_BUTTON
171
+ ]
172
+ }
173
+ )
174
+ ] })
175
+ ] })
176
+ ] })
177
+ ]
178
+ }
179
+ );
180
+ }
181
+ const DEFAULT_PAGE_SIZE = 100;
182
+ const REPLIES_PAGE_SIZE = 20;
183
+ const OPTIMISTIC_ID_PREFIX = "optimistic-";
184
+ function CommentThreadInner({
185
+ resourceId,
186
+ resourceType,
187
+ apiBaseURL,
188
+ apiBasePath,
189
+ currentUserId,
190
+ loginHref,
191
+ headers,
192
+ components,
193
+ localization: localizationProp,
194
+ pageSize: pageSizeProp,
195
+ allowPosting: allowPostingProp,
196
+ allowEditing: allowEditingProp
197
+ }) {
198
+ const overrides = usePluginOverrides("comments", {});
199
+ const pageSize = pageSizeProp ?? overrides.defaultCommentPageSize ?? DEFAULT_PAGE_SIZE;
200
+ const allowPosting = allowPostingProp ?? overrides.allowPosting ?? true;
201
+ const allowEditing = allowEditingProp ?? overrides.allowEditing ?? true;
202
+ const loc = { ...COMMENTS_LOCALIZATION, ...localizationProp };
203
+ const [replyingTo, setReplyingTo] = useState(null);
204
+ const [expandedReplies, setExpandedReplies] = useState(
205
+ /* @__PURE__ */ new Set()
206
+ );
207
+ const [replyOffsets, setReplyOffsets] = useState({});
208
+ const config = { apiBaseURL, apiBasePath, headers };
209
+ const {
210
+ comments,
211
+ total,
212
+ isLoading,
213
+ loadMore,
214
+ hasMore,
215
+ isLoadingMore,
216
+ queryKey: threadQueryKey
217
+ } = useInfiniteComments(config, {
218
+ resourceId,
219
+ resourceType,
220
+ status: "approved",
221
+ parentId: null,
222
+ currentUserId,
223
+ pageSize
224
+ });
225
+ const postMutation = usePostComment(config, {
226
+ resourceId,
227
+ resourceType,
228
+ currentUserId,
229
+ infiniteKey: threadQueryKey,
230
+ pageSize
231
+ });
232
+ const handlePost = async (body) => {
233
+ if (!currentUserId) return;
234
+ await postMutation.mutateAsync({
235
+ body,
236
+ parentId: null
237
+ });
238
+ };
239
+ const handleReply = async (body, parentId) => {
240
+ if (!currentUserId) return;
241
+ await postMutation.mutateAsync({
242
+ body,
243
+ parentId,
244
+ limit: REPLIES_PAGE_SIZE,
245
+ offset: replyOffsets[parentId] ?? 0
246
+ });
247
+ setReplyingTo(null);
248
+ setExpandedReplies((prev) => new Set(prev).add(parentId));
249
+ };
250
+ return /* @__PURE__ */ jsxs("div", { className: "space-y-1", "data-testid": "comment-thread", children: [
251
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2 mb-4", children: [
252
+ /* @__PURE__ */ jsx(MessageSquare, { className: "h-5 w-5 text-muted-foreground" }),
253
+ /* @__PURE__ */ jsx("h3", { className: "font-semibold text-sm", children: total === 0 ? loc.COMMENTS_TITLE : `${total} ${loc.COMMENTS_TITLE}` })
254
+ ] }),
255
+ isLoading && /* @__PURE__ */ jsx("div", { className: "space-y-4", children: [1, 2].map((i) => /* @__PURE__ */ jsxs("div", { className: "flex gap-3 py-3 animate-pulse", children: [
256
+ /* @__PURE__ */ jsx("div", { className: "h-8 w-8 rounded-full bg-muted shrink-0" }),
257
+ /* @__PURE__ */ jsxs("div", { className: "flex-1 space-y-2", children: [
258
+ /* @__PURE__ */ jsx("div", { className: "h-3 w-24 rounded bg-muted" }),
259
+ /* @__PURE__ */ jsx("div", { className: "h-3 w-full rounded bg-muted" }),
260
+ /* @__PURE__ */ jsx("div", { className: "h-3 w-3/4 rounded bg-muted" })
261
+ ] })
262
+ ] }, i)) }),
263
+ !isLoading && comments.length > 0 && /* @__PURE__ */ jsx("div", { className: "divide-y divide-border", children: comments.map((comment) => /* @__PURE__ */ jsxs("div", { children: [
264
+ /* @__PURE__ */ jsx(
265
+ CommentCard,
266
+ {
267
+ comment,
268
+ currentUserId,
269
+ apiBaseURL,
270
+ apiBasePath,
271
+ resourceId,
272
+ resourceType,
273
+ headers,
274
+ components,
275
+ loc,
276
+ infiniteKey: threadQueryKey,
277
+ onReplyClick: (parentId) => {
278
+ setReplyingTo(replyingTo === parentId ? null : parentId);
279
+ },
280
+ allowPosting,
281
+ allowEditing
282
+ }
283
+ ),
284
+ /* @__PURE__ */ jsx(
285
+ RepliesSection,
286
+ {
287
+ parentId: comment.id,
288
+ resourceId,
289
+ resourceType,
290
+ apiBaseURL,
291
+ apiBasePath,
292
+ currentUserId,
293
+ headers,
294
+ components,
295
+ loc,
296
+ expanded: expandedReplies.has(comment.id),
297
+ replyCount: comment.replyCount,
298
+ onToggle: () => {
299
+ const isExpanded = expandedReplies.has(comment.id);
300
+ if (!isExpanded) {
301
+ setReplyOffsets((prev) => {
302
+ if ((prev[comment.id] ?? 0) === 0) return prev;
303
+ return { ...prev, [comment.id]: 0 };
304
+ });
305
+ }
306
+ setExpandedReplies((prev) => {
307
+ const next = new Set(prev);
308
+ next.has(comment.id) ? next.delete(comment.id) : next.add(comment.id);
309
+ return next;
310
+ });
311
+ },
312
+ onOffsetChange: (offset) => {
313
+ setReplyOffsets((prev) => {
314
+ if (prev[comment.id] === offset) return prev;
315
+ return { ...prev, [comment.id]: offset };
316
+ });
317
+ },
318
+ allowEditing
319
+ }
320
+ ),
321
+ allowPosting && replyingTo === comment.id && currentUserId && /* @__PURE__ */ jsx("div", { className: "pl-11 pb-3", children: /* @__PURE__ */ jsx(
322
+ CommentForm,
323
+ {
324
+ authorId: currentUserId,
325
+ parentId: comment.id,
326
+ submitLabel: loc.COMMENTS_FORM_POST_REPLY,
327
+ InputComponent: components?.Input,
328
+ localization: loc,
329
+ onSubmit: (body) => handleReply(body, comment.id),
330
+ onCancel: () => setReplyingTo(null)
331
+ }
332
+ ) })
333
+ ] }, comment.id)) }),
334
+ !isLoading && comments.length === 0 && /* @__PURE__ */ jsx("p", { className: "text-sm text-muted-foreground py-4 text-center", children: loc.COMMENTS_EMPTY }),
335
+ hasMore && /* @__PURE__ */ jsx("div", { className: "flex justify-center pt-2", children: /* @__PURE__ */ jsx(
336
+ Button,
337
+ {
338
+ variant: "outline",
339
+ size: "sm",
340
+ onClick: () => loadMore(),
341
+ disabled: isLoadingMore,
342
+ "data-testid": "load-more-comments",
343
+ children: isLoadingMore ? loc.COMMENTS_LOADING_MORE : loc.COMMENTS_LOAD_MORE
344
+ }
345
+ ) }),
346
+ allowPosting && /* @__PURE__ */ jsxs(Fragment, { children: [
347
+ /* @__PURE__ */ jsx(Separator, { className: "my-4" }),
348
+ currentUserId ? /* @__PURE__ */ jsx("div", { "data-testid": "comment-form-wrapper", children: /* @__PURE__ */ jsx(
349
+ CommentForm,
350
+ {
351
+ authorId: currentUserId,
352
+ submitLabel: loc.COMMENTS_FORM_POST_COMMENT,
353
+ InputComponent: components?.Input,
354
+ localization: loc,
355
+ onSubmit: handlePost
356
+ }
357
+ ) }) : /* @__PURE__ */ jsxs(
358
+ "div",
359
+ {
360
+ className: "flex flex-col items-center gap-3 py-6 text-center border rounded-lg bg-muted/30",
361
+ "data-testid": "login-prompt",
362
+ children: [
363
+ /* @__PURE__ */ jsx(LogIn, { className: "h-6 w-6 text-muted-foreground" }),
364
+ /* @__PURE__ */ jsx("p", { className: "text-sm text-muted-foreground", children: loc.COMMENTS_LOGIN_PROMPT }),
365
+ loginHref && /* @__PURE__ */ jsx(
366
+ "a",
367
+ {
368
+ href: loginHref,
369
+ className: "inline-flex items-center gap-1 text-sm font-medium text-primary underline underline-offset-4",
370
+ "data-testid": "login-link",
371
+ children: loc.COMMENTS_LOGIN_LINK
372
+ }
373
+ )
374
+ ]
375
+ }
376
+ )
377
+ ] })
378
+ ] });
379
+ }
380
+ function RepliesSection({
381
+ parentId,
382
+ resourceId,
383
+ resourceType,
384
+ apiBaseURL,
385
+ apiBasePath,
386
+ currentUserId,
387
+ headers,
388
+ components,
389
+ loc,
390
+ expanded,
391
+ replyCount,
392
+ onToggle,
393
+ onOffsetChange,
394
+ allowEditing
395
+ }) {
396
+ const config = { apiBaseURL, apiBasePath, headers };
397
+ const [replyOffset, setReplyOffset] = useState(0);
398
+ const [loadedReplies, setLoadedReplies] = useState([]);
399
+ const {
400
+ comments: repliesPage,
401
+ total: repliesTotal,
402
+ isFetching: isFetchingReplies
403
+ } = useComments(
404
+ config,
405
+ {
406
+ resourceId,
407
+ resourceType,
408
+ parentId,
409
+ status: "approved",
410
+ currentUserId,
411
+ limit: REPLIES_PAGE_SIZE,
412
+ offset: replyOffset
413
+ },
414
+ { enabled: expanded }
415
+ );
416
+ useEffect(() => {
417
+ if (expanded) {
418
+ setReplyOffset(0);
419
+ setLoadedReplies([]);
420
+ }
421
+ }, [expanded, parentId]);
422
+ useEffect(() => {
423
+ onOffsetChange(replyOffset);
424
+ }, [onOffsetChange, replyOffset]);
425
+ useEffect(() => {
426
+ if (!expanded) return;
427
+ setLoadedReplies((prev) => {
428
+ const byId = new Map(prev.map((item) => [item.id, item]));
429
+ for (const reply of repliesPage) {
430
+ byId.set(reply.id, reply);
431
+ }
432
+ const currentPageIds = new Set(repliesPage.map((reply) => reply.id));
433
+ const currentPageRealReplies = repliesPage.filter(
434
+ (reply) => !reply.id.startsWith(OPTIMISTIC_ID_PREFIX)
435
+ );
436
+ return Array.from(byId.values()).filter((reply) => {
437
+ if (!reply.id.startsWith(OPTIMISTIC_ID_PREFIX)) return true;
438
+ if (currentPageIds.has(reply.id)) return true;
439
+ return !currentPageRealReplies.some(
440
+ (realReply) => realReply.parentId === reply.parentId && realReply.authorId === reply.authorId && realReply.body === reply.body
441
+ );
442
+ });
443
+ });
444
+ }, [expanded, repliesPage]);
445
+ if (replyCount === 0 && !expanded) return null;
446
+ const displayCount = expanded ? loadedReplies.length || replyCount : replyCount;
447
+ const effectiveReplyTotal = repliesTotal || replyCount;
448
+ const hasMoreReplies = loadedReplies.length < effectiveReplyTotal;
449
+ return /* @__PURE__ */ jsxs("div", { className: "pl-11", children: [
450
+ /* @__PURE__ */ jsxs(
451
+ Button,
452
+ {
453
+ variant: "ghost",
454
+ size: "sm",
455
+ className: "h-7 px-2 text-xs mb-1",
456
+ onClick: onToggle,
457
+ "data-testid": expanded ? "hide-replies-button" : "show-replies-button",
458
+ children: [
459
+ expanded ? /* @__PURE__ */ jsx(ChevronUp, { className: "h-3 w-3 mr-1" }) : /* @__PURE__ */ jsx(ChevronDown, { className: "h-3 w-3 mr-1" }),
460
+ expanded ? loc.COMMENTS_HIDE_REPLIES : `${displayCount} ${displayCount === 1 ? loc.COMMENTS_REPLIES_SINGULAR : loc.COMMENTS_REPLIES_PLURAL}`
461
+ ]
462
+ }
463
+ ),
464
+ expanded && /* @__PURE__ */ jsxs(
465
+ "div",
466
+ {
467
+ className: "border-l-2 border-border pl-3 space-y-0",
468
+ "data-testid": "replies-list",
469
+ children: [
470
+ loadedReplies.map((reply) => /* @__PURE__ */ jsx(
471
+ CommentCard,
472
+ {
473
+ comment: reply,
474
+ currentUserId,
475
+ apiBaseURL,
476
+ apiBasePath,
477
+ resourceId,
478
+ resourceType,
479
+ headers,
480
+ components,
481
+ loc,
482
+ onReplyClick: () => {
483
+ },
484
+ allowPosting: false,
485
+ allowEditing
486
+ },
487
+ reply.id
488
+ )),
489
+ hasMoreReplies && /* @__PURE__ */ jsx("div", { className: "py-2", children: /* @__PURE__ */ jsx(
490
+ Button,
491
+ {
492
+ variant: "ghost",
493
+ size: "sm",
494
+ className: "h-7 px-2 text-xs",
495
+ onClick: () => setReplyOffset((prev) => prev + REPLIES_PAGE_SIZE),
496
+ disabled: isFetchingReplies,
497
+ "data-testid": "load-more-replies",
498
+ children: isFetchingReplies ? loc.COMMENTS_LOADING_MORE : loc.COMMENTS_LOAD_MORE
499
+ }
500
+ ) })
501
+ ]
502
+ }
503
+ )
504
+ ] });
505
+ }
506
+ function CommentThreadSkeleton() {
507
+ return /* @__PURE__ */ jsxs("div", { className: "space-y-1", children: [
508
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2 mb-4", children: [
509
+ /* @__PURE__ */ jsx("div", { className: "h-5 w-5 rounded bg-muted animate-pulse" }),
510
+ /* @__PURE__ */ jsx("div", { className: "h-4 w-24 rounded bg-muted animate-pulse" })
511
+ ] }),
512
+ [1, 2, 3].map((i) => /* @__PURE__ */ jsxs("div", { className: "flex gap-3 py-3", children: [
513
+ /* @__PURE__ */ jsx("div", { className: "h-8 w-8 rounded-full bg-muted shrink-0 mt-0.5 animate-pulse" }),
514
+ /* @__PURE__ */ jsxs("div", { className: "flex-1 space-y-2", children: [
515
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2", children: [
516
+ /* @__PURE__ */ jsx("div", { className: "h-3 w-20 rounded bg-muted animate-pulse" }),
517
+ /* @__PURE__ */ jsx("div", { className: "h-3 w-14 rounded bg-muted animate-pulse" })
518
+ ] }),
519
+ /* @__PURE__ */ jsx("div", { className: "h-3 w-full rounded bg-muted animate-pulse" }),
520
+ /* @__PURE__ */ jsx("div", { className: "h-3 w-4/5 rounded bg-muted animate-pulse" }),
521
+ /* @__PURE__ */ jsxs("div", { className: "flex gap-1 mt-1", children: [
522
+ /* @__PURE__ */ jsx("div", { className: "h-7 w-12 rounded bg-muted animate-pulse" }),
523
+ /* @__PURE__ */ jsx("div", { className: "h-7 w-14 rounded bg-muted animate-pulse" })
524
+ ] })
525
+ ] })
526
+ ] }, i)),
527
+ /* @__PURE__ */ jsx("div", { className: "h-px w-full bg-muted my-4" }),
528
+ /* @__PURE__ */ jsxs("div", { className: "space-y-2", children: [
529
+ /* @__PURE__ */ jsx("div", { className: "h-20 w-full rounded-md border bg-muted animate-pulse" }),
530
+ /* @__PURE__ */ jsx("div", { className: "flex justify-end", children: /* @__PURE__ */ jsx("div", { className: "h-9 w-28 rounded-md bg-muted animate-pulse" }) })
531
+ ] })
532
+ ] });
533
+ }
534
+ function CommentThread(props) {
535
+ return /* @__PURE__ */ jsx("div", { id: "comments", className: props.className, children: /* @__PURE__ */ jsx(WhenVisible, { fallback: /* @__PURE__ */ jsx(CommentThreadSkeleton, {}), rootMargin: "300px", children: /* @__PURE__ */ jsx(CommentThreadInner, { ...props }) }) });
536
+ }
537
+
538
+ export { CommentThread };
@@ -0,0 +1,64 @@
1
+ "use client";
2
+ 'use strict';
3
+
4
+ const jsxRuntime = require('react/jsx-runtime');
5
+ const React = require('react');
6
+ const components = require('@btst/stack/client/components');
7
+ const context = require('@btst/stack/context');
8
+ const index = require('../../localization/index.cjs');
9
+ const useRouteLifecycle = require('../../../../../../../ui/src/hooks/use-route-lifecycle.cjs');
10
+ const pageWrapper = require('../shared/page-wrapper.cjs');
11
+
12
+ const ModerationPageInternal = React.lazy(
13
+ () => import('./moderation-page.internal.cjs').then((m) => ({
14
+ default: m.ModerationPage
15
+ }))
16
+ );
17
+ function ModerationPageSkeleton() {
18
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "w-full max-w-5xl mx-auto space-y-4 animate-pulse", children: [
19
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "h-8 w-64 rounded bg-muted" }),
20
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "h-4 w-48 rounded bg-muted" }),
21
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "h-10 w-72 rounded bg-muted" }),
22
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "rounded-lg border h-64 bg-muted" })
23
+ ] });
24
+ }
25
+ function ModerationPageComponent() {
26
+ return /* @__PURE__ */ jsxRuntime.jsx(
27
+ components.ComposedRoute,
28
+ {
29
+ path: "/comments/moderation",
30
+ PageComponent: ModerationPageWrapper,
31
+ LoadingComponent: ModerationPageSkeleton,
32
+ onError: (error) => console.error("[btst/comments] Moderation error:", error)
33
+ }
34
+ );
35
+ }
36
+ function ModerationPageWrapper() {
37
+ const overrides = context.usePluginOverrides("comments");
38
+ const loc = { ...index.COMMENTS_LOCALIZATION, ...overrides.localization };
39
+ useRouteLifecycle.useRouteLifecycle({
40
+ routeName: "moderation",
41
+ context: {
42
+ path: "/comments/moderation",
43
+ isSSR: typeof window === "undefined"
44
+ },
45
+ overrides,
46
+ beforeRenderHook: (o, context) => {
47
+ if (o.onBeforeModerationPageRendered) {
48
+ return o.onBeforeModerationPageRendered(context);
49
+ }
50
+ return true;
51
+ }
52
+ });
53
+ return /* @__PURE__ */ jsxRuntime.jsx(pageWrapper.PageWrapper, { children: /* @__PURE__ */ jsxRuntime.jsx(
54
+ ModerationPageInternal,
55
+ {
56
+ apiBaseURL: overrides.apiBaseURL,
57
+ apiBasePath: overrides.apiBasePath,
58
+ headers: overrides.headers,
59
+ localization: loc
60
+ }
61
+ ) });
62
+ }
63
+
64
+ exports.ModerationPageComponent = ModerationPageComponent;