@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,426 @@
1
+ "use client";
2
+ 'use strict';
3
+
4
+ const jsxRuntime = require('react/jsx-runtime');
5
+ const React = require('react');
6
+ const table = require('../../../../../../../ui/src/components/table.cjs');
7
+ const dialog = require('../../../../../../../ui/src/components/dialog.cjs');
8
+ const alertDialog = require('../../../../../../../ui/src/components/alert-dialog.cjs');
9
+ const button = require('../../../../../../../ui/src/components/button.cjs');
10
+ const badge = require('../../../../../../../ui/src/components/badge.cjs');
11
+ const tabs = require('../../../../../../../ui/src/components/tabs.cjs');
12
+ const checkbox = require('../../../../../../../ui/src/components/checkbox.cjs');
13
+ const avatar = require('../../../../../../../ui/src/components/avatar.cjs');
14
+ const LucideIcons = require('lucide-react');
15
+ const sonner = require('sonner');
16
+ const dateFns = require('date-fns');
17
+ const context = require('@btst/stack/plugins/ai-chat/client/context');
18
+ const useComments = require('../../hooks/use-comments.cjs');
19
+ const index = require('../../localization/index.cjs');
20
+ const utils = require('../../utils.cjs');
21
+ const pagination = require('../shared/pagination.cjs');
22
+
23
+ function StatusBadge({ status }) {
24
+ const variants = {
25
+ pending: "secondary",
26
+ approved: "default",
27
+ spam: "destructive"
28
+ };
29
+ return /* @__PURE__ */ jsxRuntime.jsx(badge.Badge, { variant: variants[status], children: status });
30
+ }
31
+ function ModerationPage({
32
+ apiBaseURL,
33
+ apiBasePath,
34
+ headers,
35
+ localization: localizationProp
36
+ }) {
37
+ const loc = { ...index.COMMENTS_LOCALIZATION, ...localizationProp };
38
+ const [activeTab, setActiveTab] = React.useState("pending");
39
+ const [currentPage, setCurrentPage] = React.useState(1);
40
+ const [selected, setSelected] = React.useState(/* @__PURE__ */ new Set());
41
+ const [viewComment, setViewComment] = React.useState(
42
+ null
43
+ );
44
+ const [deleteIds, setDeleteIds] = React.useState([]);
45
+ const config = { apiBaseURL, apiBasePath, headers };
46
+ const { comments, total, limit, offset, totalPages, refetch } = useComments.useSuspenseModerationComments(config, {
47
+ status: activeTab,
48
+ page: currentPage
49
+ });
50
+ const updateStatus = useComments.useUpdateCommentStatus(config);
51
+ const deleteMutation = useComments.useDeleteComment(config);
52
+ context.useRegisterPageAIContext({
53
+ routeName: "comments-moderation",
54
+ pageDescription: `${total} ${activeTab} comments in the moderation queue.
55
+
56
+ Top ${activeTab} comments:
57
+ ${comments.slice(0, 5).map(
58
+ (c) => `- "${c.body.slice(0, 80)}${c.body.length > 80 ? "\u2026" : ""}" by ${c.resolvedAuthorName} on ${c.resourceType}/${c.resourceId}`
59
+ ).join("\n")}`,
60
+ suggestions: [
61
+ "Approve all safe-looking comments",
62
+ "Flag spam comments",
63
+ "Summarize today's discussion"
64
+ ]
65
+ });
66
+ const toggleSelect = (id) => {
67
+ setSelected((prev) => {
68
+ const next = new Set(prev);
69
+ next.has(id) ? next.delete(id) : next.add(id);
70
+ return next;
71
+ });
72
+ };
73
+ const toggleSelectAll = () => {
74
+ if (selected.size === comments.length) {
75
+ setSelected(/* @__PURE__ */ new Set());
76
+ } else {
77
+ setSelected(new Set(comments.map((c) => c.id)));
78
+ }
79
+ };
80
+ const handleApprove = async (id) => {
81
+ try {
82
+ await updateStatus.mutateAsync({ id, status: "approved" });
83
+ sonner.toast.success(loc.COMMENTS_MODERATION_TOAST_APPROVED);
84
+ await refetch();
85
+ } catch {
86
+ sonner.toast.error(loc.COMMENTS_MODERATION_TOAST_APPROVE_ERROR);
87
+ }
88
+ };
89
+ const handleSpam = async (id) => {
90
+ try {
91
+ await updateStatus.mutateAsync({ id, status: "spam" });
92
+ sonner.toast.success(loc.COMMENTS_MODERATION_TOAST_SPAM);
93
+ await refetch();
94
+ } catch {
95
+ sonner.toast.error(loc.COMMENTS_MODERATION_TOAST_SPAM_ERROR);
96
+ }
97
+ };
98
+ const handleDelete = async (ids) => {
99
+ try {
100
+ await Promise.all(ids.map((id) => deleteMutation.mutateAsync(id)));
101
+ sonner.toast.success(
102
+ ids.length === 1 ? loc.COMMENTS_MODERATION_TOAST_DELETED : loc.COMMENTS_MODERATION_TOAST_DELETED_PLURAL.replace(
103
+ "{n}",
104
+ String(ids.length)
105
+ )
106
+ );
107
+ setSelected(/* @__PURE__ */ new Set());
108
+ setDeleteIds([]);
109
+ await refetch();
110
+ } catch {
111
+ sonner.toast.error(loc.COMMENTS_MODERATION_TOAST_DELETE_ERROR);
112
+ }
113
+ };
114
+ const handleBulkApprove = async () => {
115
+ const ids = [...selected];
116
+ try {
117
+ await Promise.all(
118
+ ids.map((id) => updateStatus.mutateAsync({ id, status: "approved" }))
119
+ );
120
+ sonner.toast.success(
121
+ loc.COMMENTS_MODERATION_TOAST_BULK_APPROVED.replace(
122
+ "{n}",
123
+ String(ids.length)
124
+ )
125
+ );
126
+ setSelected(/* @__PURE__ */ new Set());
127
+ await refetch();
128
+ } catch {
129
+ sonner.toast.error(loc.COMMENTS_MODERATION_TOAST_BULK_APPROVE_ERROR);
130
+ }
131
+ };
132
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "w-full max-w-5xl space-y-6", "data-testid": "moderation-page", children: [
133
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
134
+ /* @__PURE__ */ jsxRuntime.jsx("h1", { className: "text-2xl font-bold", children: loc.COMMENTS_MODERATION_TITLE }),
135
+ /* @__PURE__ */ jsxRuntime.jsx("p", { className: "text-muted-foreground text-sm mt-1", children: loc.COMMENTS_MODERATION_DESCRIPTION })
136
+ ] }),
137
+ /* @__PURE__ */ jsxRuntime.jsx(
138
+ tabs.Tabs,
139
+ {
140
+ value: activeTab,
141
+ onValueChange: (v) => {
142
+ setActiveTab(v);
143
+ setCurrentPage(1);
144
+ setSelected(/* @__PURE__ */ new Set());
145
+ },
146
+ children: /* @__PURE__ */ jsxRuntime.jsxs(tabs.TabsList, { children: [
147
+ /* @__PURE__ */ jsxRuntime.jsx(tabs.TabsTrigger, { value: "pending", "data-testid": "tab-pending", children: loc.COMMENTS_MODERATION_TAB_PENDING }),
148
+ /* @__PURE__ */ jsxRuntime.jsx(tabs.TabsTrigger, { value: "approved", "data-testid": "tab-approved", children: loc.COMMENTS_MODERATION_TAB_APPROVED }),
149
+ /* @__PURE__ */ jsxRuntime.jsx(tabs.TabsTrigger, { value: "spam", "data-testid": "tab-spam", children: loc.COMMENTS_MODERATION_TAB_SPAM })
150
+ ] })
151
+ }
152
+ ),
153
+ selected.size > 0 && /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center gap-2 p-3 bg-muted rounded-lg", children: [
154
+ /* @__PURE__ */ jsxRuntime.jsx("span", { className: "text-sm text-muted-foreground", children: loc.COMMENTS_MODERATION_SELECTED.replace(
155
+ "{n}",
156
+ String(selected.size)
157
+ ) }),
158
+ activeTab !== "approved" && /* @__PURE__ */ jsxRuntime.jsxs(
159
+ button.Button,
160
+ {
161
+ size: "sm",
162
+ variant: "outline",
163
+ onClick: handleBulkApprove,
164
+ disabled: updateStatus.isPending,
165
+ children: [
166
+ /* @__PURE__ */ jsxRuntime.jsx(LucideIcons.CheckCircle, { className: "h-4 w-4 mr-1" }),
167
+ loc.COMMENTS_MODERATION_APPROVE_SELECTED
168
+ ]
169
+ }
170
+ ),
171
+ /* @__PURE__ */ jsxRuntime.jsxs(
172
+ button.Button,
173
+ {
174
+ size: "sm",
175
+ variant: "outline",
176
+ className: "text-destructive border-destructive hover:bg-destructive hover:text-destructive-foreground",
177
+ onClick: () => setDeleteIds([...selected]),
178
+ children: [
179
+ /* @__PURE__ */ jsxRuntime.jsx(LucideIcons.Trash2, { className: "h-4 w-4 mr-1" }),
180
+ loc.COMMENTS_MODERATION_DELETE_SELECTED
181
+ ]
182
+ }
183
+ )
184
+ ] }),
185
+ comments.length === 0 ? /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex flex-col items-center gap-2 py-16 text-muted-foreground", children: [
186
+ /* @__PURE__ */ jsxRuntime.jsx(LucideIcons.CheckCircle, { className: "h-8 w-8" }),
187
+ /* @__PURE__ */ jsxRuntime.jsx("p", { className: "text-sm", children: loc.COMMENTS_MODERATION_EMPTY.replace("{status}", activeTab) })
188
+ ] }) : /* @__PURE__ */ jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [
189
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "rounded-lg border", children: /* @__PURE__ */ jsxRuntime.jsxs(table.Table, { children: [
190
+ /* @__PURE__ */ jsxRuntime.jsx(table.TableHeader, { children: /* @__PURE__ */ jsxRuntime.jsxs(table.TableRow, { children: [
191
+ /* @__PURE__ */ jsxRuntime.jsx(table.TableHead, { className: "w-10", children: /* @__PURE__ */ jsxRuntime.jsx(
192
+ checkbox.Checkbox,
193
+ {
194
+ checked: selected.size === comments.length && comments.length > 0,
195
+ onCheckedChange: toggleSelectAll,
196
+ "aria-label": loc.COMMENTS_MODERATION_SELECT_ALL
197
+ }
198
+ ) }),
199
+ /* @__PURE__ */ jsxRuntime.jsx(table.TableHead, { children: loc.COMMENTS_MODERATION_COL_AUTHOR }),
200
+ /* @__PURE__ */ jsxRuntime.jsx(table.TableHead, { children: loc.COMMENTS_MODERATION_COL_COMMENT }),
201
+ /* @__PURE__ */ jsxRuntime.jsx(table.TableHead, { children: loc.COMMENTS_MODERATION_COL_RESOURCE }),
202
+ /* @__PURE__ */ jsxRuntime.jsx(table.TableHead, { children: loc.COMMENTS_MODERATION_COL_DATE }),
203
+ /* @__PURE__ */ jsxRuntime.jsx(table.TableHead, { className: "w-36", children: loc.COMMENTS_MODERATION_COL_ACTIONS })
204
+ ] }) }),
205
+ /* @__PURE__ */ jsxRuntime.jsx(table.TableBody, { children: comments.map((comment) => /* @__PURE__ */ jsxRuntime.jsxs(
206
+ table.TableRow,
207
+ {
208
+ "data-testid": "moderation-row",
209
+ "data-comment-id": comment.id,
210
+ children: [
211
+ /* @__PURE__ */ jsxRuntime.jsx(table.TableCell, { children: /* @__PURE__ */ jsxRuntime.jsx(
212
+ checkbox.Checkbox,
213
+ {
214
+ checked: selected.has(comment.id),
215
+ onCheckedChange: () => toggleSelect(comment.id),
216
+ "aria-label": loc.COMMENTS_MODERATION_SELECT_ONE
217
+ }
218
+ ) }),
219
+ /* @__PURE__ */ jsxRuntime.jsx(table.TableCell, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center gap-2", children: [
220
+ /* @__PURE__ */ jsxRuntime.jsxs(avatar.Avatar, { className: "h-7 w-7", children: [
221
+ comment.resolvedAvatarUrl && /* @__PURE__ */ jsxRuntime.jsx(avatar.AvatarImage, { src: comment.resolvedAvatarUrl }),
222
+ /* @__PURE__ */ jsxRuntime.jsx(avatar.AvatarFallback, { className: "text-xs", children: utils.getInitials(comment.resolvedAuthorName) })
223
+ ] }),
224
+ /* @__PURE__ */ jsxRuntime.jsx("span", { className: "text-sm font-medium truncate max-w-[100px]", children: comment.resolvedAuthorName })
225
+ ] }) }),
226
+ /* @__PURE__ */ jsxRuntime.jsx(table.TableCell, { children: /* @__PURE__ */ jsxRuntime.jsx("p", { className: "text-sm text-muted-foreground max-w-xs truncate", children: comment.body }) }),
227
+ /* @__PURE__ */ jsxRuntime.jsx(table.TableCell, { children: /* @__PURE__ */ jsxRuntime.jsxs("span", { className: "text-xs text-muted-foreground", children: [
228
+ comment.resourceType,
229
+ "/",
230
+ comment.resourceId
231
+ ] }) }),
232
+ /* @__PURE__ */ jsxRuntime.jsx(table.TableCell, { className: "text-xs text-muted-foreground whitespace-nowrap", children: dateFns.formatDistanceToNow(new Date(comment.createdAt), {
233
+ addSuffix: true
234
+ }) }),
235
+ /* @__PURE__ */ jsxRuntime.jsx(table.TableCell, { children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center gap-1", children: [
236
+ /* @__PURE__ */ jsxRuntime.jsx(
237
+ button.Button,
238
+ {
239
+ variant: "ghost",
240
+ size: "icon",
241
+ className: "h-7 w-7",
242
+ title: loc.COMMENTS_MODERATION_ACTION_VIEW,
243
+ onClick: () => setViewComment(comment),
244
+ "data-testid": "view-button",
245
+ children: /* @__PURE__ */ jsxRuntime.jsx(LucideIcons.Eye, { className: "h-4 w-4" })
246
+ }
247
+ ),
248
+ activeTab !== "approved" && /* @__PURE__ */ jsxRuntime.jsx(
249
+ button.Button,
250
+ {
251
+ variant: "ghost",
252
+ size: "icon",
253
+ className: "h-7 w-7 text-green-600 hover:text-green-700",
254
+ title: loc.COMMENTS_MODERATION_ACTION_APPROVE,
255
+ onClick: () => handleApprove(comment.id),
256
+ disabled: updateStatus.isPending,
257
+ "data-testid": "approve-button",
258
+ children: /* @__PURE__ */ jsxRuntime.jsx(LucideIcons.CheckCircle, { className: "h-4 w-4" })
259
+ }
260
+ ),
261
+ activeTab !== "spam" && /* @__PURE__ */ jsxRuntime.jsx(
262
+ button.Button,
263
+ {
264
+ variant: "ghost",
265
+ size: "icon",
266
+ className: "h-7 w-7 text-orange-500 hover:text-orange-600",
267
+ title: loc.COMMENTS_MODERATION_ACTION_SPAM,
268
+ onClick: () => handleSpam(comment.id),
269
+ disabled: updateStatus.isPending,
270
+ "data-testid": "spam-button",
271
+ children: /* @__PURE__ */ jsxRuntime.jsx(LucideIcons.ShieldOff, { className: "h-4 w-4" })
272
+ }
273
+ ),
274
+ /* @__PURE__ */ jsxRuntime.jsx(
275
+ button.Button,
276
+ {
277
+ variant: "ghost",
278
+ size: "icon",
279
+ className: "h-7 w-7 text-destructive hover:text-destructive",
280
+ title: loc.COMMENTS_MODERATION_ACTION_DELETE,
281
+ onClick: () => setDeleteIds([comment.id]),
282
+ "data-testid": "delete-button",
283
+ children: /* @__PURE__ */ jsxRuntime.jsx(LucideIcons.Trash2, { className: "h-4 w-4" })
284
+ }
285
+ )
286
+ ] }) })
287
+ ]
288
+ },
289
+ comment.id
290
+ )) })
291
+ ] }) }),
292
+ /* @__PURE__ */ jsxRuntime.jsx(
293
+ pagination.Pagination,
294
+ {
295
+ currentPage,
296
+ totalPages,
297
+ onPageChange: setCurrentPage,
298
+ total,
299
+ limit,
300
+ offset
301
+ }
302
+ )
303
+ ] }),
304
+ /* @__PURE__ */ jsxRuntime.jsx(dialog.Dialog, { open: !!viewComment, onOpenChange: () => setViewComment(null), children: /* @__PURE__ */ jsxRuntime.jsxs(dialog.DialogContent, { className: "max-w-2xl", children: [
305
+ /* @__PURE__ */ jsxRuntime.jsx(dialog.DialogHeader, { children: /* @__PURE__ */ jsxRuntime.jsx(dialog.DialogTitle, { children: loc.COMMENTS_MODERATION_DIALOG_TITLE }) }),
306
+ viewComment && /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "space-y-4", children: [
307
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center gap-3", children: [
308
+ /* @__PURE__ */ jsxRuntime.jsxs(avatar.Avatar, { className: "h-10 w-10", children: [
309
+ viewComment.resolvedAvatarUrl && /* @__PURE__ */ jsxRuntime.jsx(avatar.AvatarImage, { src: viewComment.resolvedAvatarUrl }),
310
+ /* @__PURE__ */ jsxRuntime.jsx(avatar.AvatarFallback, { children: utils.getInitials(viewComment.resolvedAuthorName) })
311
+ ] }),
312
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
313
+ /* @__PURE__ */ jsxRuntime.jsx("p", { className: "font-medium text-sm", children: viewComment.resolvedAuthorName }),
314
+ /* @__PURE__ */ jsxRuntime.jsx("p", { className: "text-xs text-muted-foreground", children: new Date(viewComment.createdAt).toLocaleString() })
315
+ ] }),
316
+ /* @__PURE__ */ jsxRuntime.jsx(StatusBadge, { status: viewComment.status })
317
+ ] }),
318
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "grid grid-cols-2 gap-3 text-sm", children: [
319
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
320
+ /* @__PURE__ */ jsxRuntime.jsx("p", { className: "text-muted-foreground text-xs", children: loc.COMMENTS_MODERATION_DIALOG_RESOURCE }),
321
+ /* @__PURE__ */ jsxRuntime.jsxs("p", { className: "font-mono text-xs", children: [
322
+ viewComment.resourceType,
323
+ "/",
324
+ viewComment.resourceId
325
+ ] })
326
+ ] }),
327
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
328
+ /* @__PURE__ */ jsxRuntime.jsx("p", { className: "text-muted-foreground text-xs", children: loc.COMMENTS_MODERATION_DIALOG_LIKES }),
329
+ /* @__PURE__ */ jsxRuntime.jsx("p", { children: viewComment.likes })
330
+ ] }),
331
+ viewComment.parentId && /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
332
+ /* @__PURE__ */ jsxRuntime.jsx("p", { className: "text-muted-foreground text-xs", children: loc.COMMENTS_MODERATION_DIALOG_REPLY_TO }),
333
+ /* @__PURE__ */ jsxRuntime.jsx("p", { className: "font-mono text-xs", children: viewComment.parentId })
334
+ ] }),
335
+ viewComment.editedAt && /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
336
+ /* @__PURE__ */ jsxRuntime.jsx("p", { className: "text-muted-foreground text-xs", children: loc.COMMENTS_MODERATION_DIALOG_EDITED }),
337
+ /* @__PURE__ */ jsxRuntime.jsx("p", { className: "text-xs", children: new Date(viewComment.editedAt).toLocaleString() })
338
+ ] })
339
+ ] }),
340
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
341
+ /* @__PURE__ */ jsxRuntime.jsx("p", { className: "text-muted-foreground text-xs mb-1", children: loc.COMMENTS_MODERATION_DIALOG_BODY }),
342
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "p-3 bg-muted rounded-lg text-sm whitespace-pre-wrap break-words", children: viewComment.body })
343
+ ] }),
344
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex justify-end gap-2", children: [
345
+ viewComment.status !== "approved" && /* @__PURE__ */ jsxRuntime.jsxs(
346
+ button.Button,
347
+ {
348
+ size: "sm",
349
+ onClick: async () => {
350
+ await handleApprove(viewComment.id);
351
+ setViewComment(null);
352
+ },
353
+ disabled: updateStatus.isPending,
354
+ "data-testid": "dialog-approve-button",
355
+ children: [
356
+ /* @__PURE__ */ jsxRuntime.jsx(LucideIcons.CheckCircle, { className: "h-4 w-4 mr-1" }),
357
+ loc.COMMENTS_MODERATION_DIALOG_APPROVE
358
+ ]
359
+ }
360
+ ),
361
+ viewComment.status !== "spam" && /* @__PURE__ */ jsxRuntime.jsxs(
362
+ button.Button,
363
+ {
364
+ size: "sm",
365
+ variant: "outline",
366
+ onClick: async () => {
367
+ await handleSpam(viewComment.id);
368
+ setViewComment(null);
369
+ },
370
+ disabled: updateStatus.isPending,
371
+ children: [
372
+ /* @__PURE__ */ jsxRuntime.jsx(LucideIcons.ShieldOff, { className: "h-4 w-4 mr-1" }),
373
+ loc.COMMENTS_MODERATION_DIALOG_MARK_SPAM
374
+ ]
375
+ }
376
+ ),
377
+ /* @__PURE__ */ jsxRuntime.jsxs(
378
+ button.Button,
379
+ {
380
+ size: "sm",
381
+ variant: "destructive",
382
+ onClick: () => {
383
+ setDeleteIds([viewComment.id]);
384
+ setViewComment(null);
385
+ },
386
+ children: [
387
+ /* @__PURE__ */ jsxRuntime.jsx(LucideIcons.Trash2, { className: "h-4 w-4 mr-1" }),
388
+ loc.COMMENTS_MODERATION_DIALOG_DELETE
389
+ ]
390
+ }
391
+ )
392
+ ] })
393
+ ] })
394
+ ] }) }),
395
+ /* @__PURE__ */ jsxRuntime.jsx(
396
+ alertDialog.AlertDialog,
397
+ {
398
+ open: deleteIds.length > 0,
399
+ onOpenChange: (open) => !open && setDeleteIds([]),
400
+ children: /* @__PURE__ */ jsxRuntime.jsxs(alertDialog.AlertDialogContent, { children: [
401
+ /* @__PURE__ */ jsxRuntime.jsxs(alertDialog.AlertDialogHeader, { children: [
402
+ /* @__PURE__ */ jsxRuntime.jsx(alertDialog.AlertDialogTitle, { children: deleteIds.length === 1 ? loc.COMMENTS_MODERATION_DELETE_TITLE_SINGULAR : loc.COMMENTS_MODERATION_DELETE_TITLE_PLURAL.replace(
403
+ "{n}",
404
+ String(deleteIds.length)
405
+ ) }),
406
+ /* @__PURE__ */ jsxRuntime.jsx(alertDialog.AlertDialogDescription, { children: deleteIds.length === 1 ? loc.COMMENTS_MODERATION_DELETE_DESCRIPTION_SINGULAR : loc.COMMENTS_MODERATION_DELETE_DESCRIPTION_PLURAL })
407
+ ] }),
408
+ /* @__PURE__ */ jsxRuntime.jsxs(alertDialog.AlertDialogFooter, { children: [
409
+ /* @__PURE__ */ jsxRuntime.jsx(alertDialog.AlertDialogCancel, { children: loc.COMMENTS_MODERATION_DELETE_CANCEL }),
410
+ /* @__PURE__ */ jsxRuntime.jsx(
411
+ alertDialog.AlertDialogAction,
412
+ {
413
+ className: "bg-destructive text-destructive-foreground hover:bg-destructive/90",
414
+ onClick: () => handleDelete(deleteIds),
415
+ "data-testid": "confirm-delete-button",
416
+ children: deleteMutation.isPending ? loc.COMMENTS_MODERATION_DELETE_DELETING : loc.COMMENTS_MODERATION_DELETE_CONFIRM
417
+ }
418
+ )
419
+ ] })
420
+ ] })
421
+ }
422
+ )
423
+ ] });
424
+ }
425
+
426
+ exports.ModerationPage = ModerationPage;