@happyvertical/smrt-content 0.30.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 (291) hide show
  1. package/AGENTS.md +194 -0
  2. package/CLAUDE.md +1 -0
  3. package/LICENSE +7 -0
  4. package/README.md +634 -0
  5. package/dist/__smrt-register__.d.ts +2 -0
  6. package/dist/__smrt-register__.d.ts.map +1 -0
  7. package/dist/asset-associable.d.ts +115 -0
  8. package/dist/asset-associable.d.ts.map +1 -0
  9. package/dist/body-format.d.ts +29 -0
  10. package/dist/body-format.d.ts.map +1 -0
  11. package/dist/body-format.js +604 -0
  12. package/dist/body-format.js.map +1 -0
  13. package/dist/content-asset.d.ts +17 -0
  14. package/dist/content-asset.d.ts.map +1 -0
  15. package/dist/content-assets.d.ts +10 -0
  16. package/dist/content-assets.d.ts.map +1 -0
  17. package/dist/content-chat-handlers.d.ts +115 -0
  18. package/dist/content-chat-handlers.d.ts.map +1 -0
  19. package/dist/content-chat-prompts.d.ts +3 -0
  20. package/dist/content-chat-prompts.d.ts.map +1 -0
  21. package/dist/content-chat-session.d.ts +26 -0
  22. package/dist/content-chat-session.d.ts.map +1 -0
  23. package/dist/content-contribution-attachment.d.ts +42 -0
  24. package/dist/content-contribution-attachment.d.ts.map +1 -0
  25. package/dist/content-contribution-attachments.d.ts +8 -0
  26. package/dist/content-contribution-attachments.d.ts.map +1 -0
  27. package/dist/content-contribution-config.d.ts +84 -0
  28. package/dist/content-contribution-config.d.ts.map +1 -0
  29. package/dist/content-contribution-revision.d.ts +38 -0
  30. package/dist/content-contribution-revision.d.ts.map +1 -0
  31. package/dist/content-contribution-revisions.d.ts +8 -0
  32. package/dist/content-contribution-revisions.d.ts.map +1 -0
  33. package/dist/content-contribution-type.d.ts +51 -0
  34. package/dist/content-contribution-type.d.ts.map +1 -0
  35. package/dist/content-contribution-types.d.ts +7 -0
  36. package/dist/content-contribution-types.d.ts.map +1 -0
  37. package/dist/content-contribution.d.ts +161 -0
  38. package/dist/content-contribution.d.ts.map +1 -0
  39. package/dist/content-contributions.d.ts +53 -0
  40. package/dist/content-contributions.d.ts.map +1 -0
  41. package/dist/content-contributor.d.ts +30 -0
  42. package/dist/content-contributor.d.ts.map +1 -0
  43. package/dist/content-contributors.d.ts +13 -0
  44. package/dist/content-contributors.d.ts.map +1 -0
  45. package/dist/content-correction.d.ts +39 -0
  46. package/dist/content-correction.d.ts.map +1 -0
  47. package/dist/content-corrections.d.ts +9 -0
  48. package/dist/content-corrections.d.ts.map +1 -0
  49. package/dist/content-editor-assistant.d.ts +68 -0
  50. package/dist/content-editor-assistant.d.ts.map +1 -0
  51. package/dist/content-editor-assistant.js +97 -0
  52. package/dist/content-editor-assistant.js.map +1 -0
  53. package/dist/content-feed-parser.d.ts +19 -0
  54. package/dist/content-feed-parser.d.ts.map +1 -0
  55. package/dist/content-feed-source.d.ts +52 -0
  56. package/dist/content-feed-source.d.ts.map +1 -0
  57. package/dist/content-feed-sources.d.ts +11 -0
  58. package/dist/content-feed-sources.d.ts.map +1 -0
  59. package/dist/content-feed-sync.d.ts +23 -0
  60. package/dist/content-feed-sync.d.ts.map +1 -0
  61. package/dist/content-governance-assignment.d.ts +42 -0
  62. package/dist/content-governance-assignment.d.ts.map +1 -0
  63. package/dist/content-governance-assignments.d.ts +11 -0
  64. package/dist/content-governance-assignments.d.ts.map +1 -0
  65. package/dist/content-governance-policies.d.ts +7 -0
  66. package/dist/content-governance-policies.d.ts.map +1 -0
  67. package/dist/content-governance-policy.d.ts +29 -0
  68. package/dist/content-governance-policy.d.ts.map +1 -0
  69. package/dist/content-governance-profile.d.ts +31 -0
  70. package/dist/content-governance-profile.d.ts.map +1 -0
  71. package/dist/content-governance-profiles.d.ts +7 -0
  72. package/dist/content-governance-profiles.d.ts.map +1 -0
  73. package/dist/content-governance.d.ts +188 -0
  74. package/dist/content-governance.d.ts.map +1 -0
  75. package/dist/content-prompts.d.ts +10 -0
  76. package/dist/content-prompts.d.ts.map +1 -0
  77. package/dist/content-reference.d.ts +17 -0
  78. package/dist/content-reference.d.ts.map +1 -0
  79. package/dist/content-references.d.ts +55 -0
  80. package/dist/content-references.d.ts.map +1 -0
  81. package/dist/content-review.d.ts +34 -0
  82. package/dist/content-review.d.ts.map +1 -0
  83. package/dist/content-reviews.d.ts +21 -0
  84. package/dist/content-reviews.d.ts.map +1 -0
  85. package/dist/content-transparency.d.ts +72 -0
  86. package/dist/content-transparency.d.ts.map +1 -0
  87. package/dist/content-types.d.ts +51 -0
  88. package/dist/content-types.d.ts.map +1 -0
  89. package/dist/content-version.d.ts +38 -0
  90. package/dist/content-version.d.ts.map +1 -0
  91. package/dist/content-versions.d.ts +16 -0
  92. package/dist/content-versions.d.ts.map +1 -0
  93. package/dist/content.d.ts +736 -0
  94. package/dist/content.d.ts.map +1 -0
  95. package/dist/contents.d.ts +292 -0
  96. package/dist/contents.d.ts.map +1 -0
  97. package/dist/database-utils.d.ts +3 -0
  98. package/dist/database-utils.d.ts.map +1 -0
  99. package/dist/index.d.ts +78 -0
  100. package/dist/index.d.ts.map +1 -0
  101. package/dist/index.js +11602 -0
  102. package/dist/index.js.map +1 -0
  103. package/dist/manifest.json +12308 -0
  104. package/dist/mock-smrt-client.d.ts +493 -0
  105. package/dist/mock-smrt-client.d.ts.map +1 -0
  106. package/dist/mock-smrt-client.js +390 -0
  107. package/dist/mock-smrt-client.js.map +1 -0
  108. package/dist/playground.d.ts +2 -0
  109. package/dist/playground.d.ts.map +1 -0
  110. package/dist/playground.js +454 -0
  111. package/dist/playground.js.map +1 -0
  112. package/dist/publish-readiness.d.ts +30 -0
  113. package/dist/publish-readiness.d.ts.map +1 -0
  114. package/dist/publish-readiness.js +74 -0
  115. package/dist/publish-readiness.js.map +1 -0
  116. package/dist/safe-remote-url.d.ts +52 -0
  117. package/dist/safe-remote-url.d.ts.map +1 -0
  118. package/dist/serialization.d.ts +78 -0
  119. package/dist/serialization.d.ts.map +1 -0
  120. package/dist/smrt-knowledge.json +6130 -0
  121. package/dist/svelte/api.d.ts +3 -0
  122. package/dist/svelte/api.d.ts.map +1 -0
  123. package/dist/svelte/api.js +10 -0
  124. package/dist/svelte/components/ArticleCard.svelte +159 -0
  125. package/dist/svelte/components/ArticleCard.svelte.d.ts +17 -0
  126. package/dist/svelte/components/ArticleCard.svelte.d.ts.map +1 -0
  127. package/dist/svelte/components/ArticleList.svelte +75 -0
  128. package/dist/svelte/components/ArticleList.svelte.d.ts +21 -0
  129. package/dist/svelte/components/ArticleList.svelte.d.ts.map +1 -0
  130. package/dist/svelte/components/ContentAgentChat.svelte +652 -0
  131. package/dist/svelte/components/ContentAgentChat.svelte.d.ts +17 -0
  132. package/dist/svelte/components/ContentAgentChat.svelte.d.ts.map +1 -0
  133. package/dist/svelte/components/ContentBodyEditor.svelte +1446 -0
  134. package/dist/svelte/components/ContentBodyEditor.svelte.d.ts +25 -0
  135. package/dist/svelte/components/ContentBodyEditor.svelte.d.ts.map +1 -0
  136. package/dist/svelte/components/ContentBodyRenderer.svelte +152 -0
  137. package/dist/svelte/components/ContentBodyRenderer.svelte.d.ts +10 -0
  138. package/dist/svelte/components/ContentBodyRenderer.svelte.d.ts.map +1 -0
  139. package/dist/svelte/components/ContentClaimAuditTool.svelte +441 -0
  140. package/dist/svelte/components/ContentClaimAuditTool.svelte.d.ts +12 -0
  141. package/dist/svelte/components/ContentClaimAuditTool.svelte.d.ts.map +1 -0
  142. package/dist/svelte/components/ContentContributionForm.svelte +226 -0
  143. package/dist/svelte/components/ContentContributionForm.svelte.d.ts +23 -0
  144. package/dist/svelte/components/ContentContributionForm.svelte.d.ts.map +1 -0
  145. package/dist/svelte/components/ContentContributionInbox.svelte +322 -0
  146. package/dist/svelte/components/ContentContributionInbox.svelte.d.ts +22 -0
  147. package/dist/svelte/components/ContentContributionInbox.svelte.d.ts.map +1 -0
  148. package/dist/svelte/components/ContentContributionPortal.svelte +182 -0
  149. package/dist/svelte/components/ContentContributionPortal.svelte.d.ts +12 -0
  150. package/dist/svelte/components/ContentContributionPortal.svelte.d.ts.map +1 -0
  151. package/dist/svelte/components/ContentContributionTypeManager.svelte +281 -0
  152. package/dist/svelte/components/ContentContributionTypeManager.svelte.d.ts +10 -0
  153. package/dist/svelte/components/ContentContributionTypeManager.svelte.d.ts.map +1 -0
  154. package/dist/svelte/components/ContentContributorManager.svelte +140 -0
  155. package/dist/svelte/components/ContentContributorManager.svelte.d.ts +10 -0
  156. package/dist/svelte/components/ContentContributorManager.svelte.d.ts.map +1 -0
  157. package/dist/svelte/components/ContentCorrectionsTool.svelte +361 -0
  158. package/dist/svelte/components/ContentCorrectionsTool.svelte.d.ts +11 -0
  159. package/dist/svelte/components/ContentCorrectionsTool.svelte.d.ts.map +1 -0
  160. package/dist/svelte/components/ContentEditor.svelte +2166 -0
  161. package/dist/svelte/components/ContentEditor.svelte.d.ts +26 -0
  162. package/dist/svelte/components/ContentEditor.svelte.d.ts.map +1 -0
  163. package/dist/svelte/components/ContentGovernanceAssignmentEditor.svelte +199 -0
  164. package/dist/svelte/components/ContentGovernanceAssignmentEditor.svelte.d.ts +11 -0
  165. package/dist/svelte/components/ContentGovernanceAssignmentEditor.svelte.d.ts.map +1 -0
  166. package/dist/svelte/components/ContentGovernanceManager.svelte +340 -0
  167. package/dist/svelte/components/ContentGovernanceManager.svelte.d.ts +11 -0
  168. package/dist/svelte/components/ContentGovernanceManager.svelte.d.ts.map +1 -0
  169. package/dist/svelte/components/ContentGovernancePanel.svelte +2244 -0
  170. package/dist/svelte/components/ContentGovernancePanel.svelte.d.ts +26 -0
  171. package/dist/svelte/components/ContentGovernancePanel.svelte.d.ts.map +1 -0
  172. package/dist/svelte/components/ContentGovernancePolicyEditor.svelte +110 -0
  173. package/dist/svelte/components/ContentGovernancePolicyEditor.svelte.d.ts +10 -0
  174. package/dist/svelte/components/ContentGovernancePolicyEditor.svelte.d.ts.map +1 -0
  175. package/dist/svelte/components/ContentGovernanceProfileEditor.svelte +185 -0
  176. package/dist/svelte/components/ContentGovernanceProfileEditor.svelte.d.ts +11 -0
  177. package/dist/svelte/components/ContentGovernanceProfileEditor.svelte.d.ts.map +1 -0
  178. package/dist/svelte/components/ContentGovernanceTool.svelte +56 -0
  179. package/dist/svelte/components/ContentGovernanceTool.svelte.d.ts +13 -0
  180. package/dist/svelte/components/ContentGovernanceTool.svelte.d.ts.map +1 -0
  181. package/dist/svelte/components/ContentImageBrowser.svelte +243 -0
  182. package/dist/svelte/components/ContentImageBrowser.svelte.d.ts +18 -0
  183. package/dist/svelte/components/ContentImageBrowser.svelte.d.ts.map +1 -0
  184. package/dist/svelte/components/ContentImageChooser.svelte +134 -0
  185. package/dist/svelte/components/ContentImageChooser.svelte.d.ts +11 -0
  186. package/dist/svelte/components/ContentImageChooser.svelte.d.ts.map +1 -0
  187. package/dist/svelte/components/ContentList.svelte +906 -0
  188. package/dist/svelte/components/ContentList.svelte.d.ts +16 -0
  189. package/dist/svelte/components/ContentList.svelte.d.ts.map +1 -0
  190. package/dist/svelte/components/ContentMetadataFields.svelte +107 -0
  191. package/dist/svelte/components/ContentMetadataFields.svelte.d.ts +8 -0
  192. package/dist/svelte/components/ContentMetadataFields.svelte.d.ts.map +1 -0
  193. package/dist/svelte/components/ContentReferencesPanel.svelte +221 -0
  194. package/dist/svelte/components/ContentReferencesPanel.svelte.d.ts +20 -0
  195. package/dist/svelte/components/ContentReferencesPanel.svelte.d.ts.map +1 -0
  196. package/dist/svelte/components/ContentReviewStatusTray.svelte +151 -0
  197. package/dist/svelte/components/ContentReviewStatusTray.svelte.d.ts +20 -0
  198. package/dist/svelte/components/ContentReviewStatusTray.svelte.d.ts.map +1 -0
  199. package/dist/svelte/components/ContentStatusFields.svelte +85 -0
  200. package/dist/svelte/components/ContentStatusFields.svelte.d.ts +8 -0
  201. package/dist/svelte/components/ContentStatusFields.svelte.d.ts.map +1 -0
  202. package/dist/svelte/components/ContentTitleField.svelte +54 -0
  203. package/dist/svelte/components/ContentTitleField.svelte.d.ts +10 -0
  204. package/dist/svelte/components/ContentTitleField.svelte.d.ts.map +1 -0
  205. package/dist/svelte/components/ContentTransparencyReport.svelte +322 -0
  206. package/dist/svelte/components/ContentTransparencyReport.svelte.d.ts +10 -0
  207. package/dist/svelte/components/ContentTransparencyReport.svelte.d.ts.map +1 -0
  208. package/dist/svelte/components/ContentTransparencyTool.svelte +314 -0
  209. package/dist/svelte/components/ContentTransparencyTool.svelte.d.ts +10 -0
  210. package/dist/svelte/components/ContentTransparencyTool.svelte.d.ts.map +1 -0
  211. package/dist/svelte/components/ContentVersionsTool.svelte +291 -0
  212. package/dist/svelte/components/ContentVersionsTool.svelte.d.ts +10 -0
  213. package/dist/svelte/components/ContentVersionsTool.svelte.d.ts.map +1 -0
  214. package/dist/svelte/components/GovernedContentEditor.svelte +409 -0
  215. package/dist/svelte/components/GovernedContentEditor.svelte.d.ts +35 -0
  216. package/dist/svelte/components/GovernedContentEditor.svelte.d.ts.map +1 -0
  217. package/dist/svelte/components/ImageThumbnail.cache.d.ts +14 -0
  218. package/dist/svelte/components/ImageThumbnail.cache.d.ts.map +1 -0
  219. package/dist/svelte/components/ImageThumbnail.cache.js +36 -0
  220. package/dist/svelte/components/ImageThumbnail.svelte +159 -0
  221. package/dist/svelte/components/ImageThumbnail.svelte.d.ts +8 -0
  222. package/dist/svelte/components/ImageThumbnail.svelte.d.ts.map +1 -0
  223. package/dist/svelte/components/Markdown.svelte +125 -0
  224. package/dist/svelte/components/Markdown.svelte.d.ts +11 -0
  225. package/dist/svelte/components/Markdown.svelte.d.ts.map +1 -0
  226. package/dist/svelte/content-editor-form.d.ts +63 -0
  227. package/dist/svelte/content-editor-form.d.ts.map +1 -0
  228. package/dist/svelte/content-editor-form.js +94 -0
  229. package/dist/svelte/content-editor-media.d.ts +12 -0
  230. package/dist/svelte/content-editor-media.d.ts.map +1 -0
  231. package/dist/svelte/content-editor-media.js +84 -0
  232. package/dist/svelte/content-editor-state.svelte.d.ts +35 -0
  233. package/dist/svelte/content-editor-state.svelte.d.ts.map +1 -0
  234. package/dist/svelte/content-editor-state.svelte.js +141 -0
  235. package/dist/svelte/governance-manager-client.d.ts +22 -0
  236. package/dist/svelte/governance-manager-client.d.ts.map +1 -0
  237. package/dist/svelte/governance-manager-client.js +1 -0
  238. package/dist/svelte/i18n.contribution.d.ts +57 -0
  239. package/dist/svelte/i18n.contribution.d.ts.map +1 -0
  240. package/dist/svelte/i18n.contribution.js +64 -0
  241. package/dist/svelte/i18n.editor.d.ts +71 -0
  242. package/dist/svelte/i18n.editor.d.ts.map +1 -0
  243. package/dist/svelte/i18n.editor.js +87 -0
  244. package/dist/svelte/i18n.governance.d.ts +66 -0
  245. package/dist/svelte/i18n.governance.d.ts.map +1 -0
  246. package/dist/svelte/i18n.governance.js +66 -0
  247. package/dist/svelte/i18n.routes.d.ts +66 -0
  248. package/dist/svelte/i18n.routes.d.ts.map +1 -0
  249. package/dist/svelte/i18n.routes.js +75 -0
  250. package/dist/svelte/i18n.tools.d.ts +81 -0
  251. package/dist/svelte/i18n.tools.d.ts.map +1 -0
  252. package/dist/svelte/i18n.tools.js +90 -0
  253. package/dist/svelte/index.d.ts +101 -0
  254. package/dist/svelte/index.d.ts.map +1 -0
  255. package/dist/svelte/index.js +63 -0
  256. package/dist/svelte/playground.d.ts +281 -0
  257. package/dist/svelte/playground.d.ts.map +1 -0
  258. package/dist/svelte/playground.js +438 -0
  259. package/dist/svelte/routes/ContentContributionsRoute.svelte +809 -0
  260. package/dist/svelte/routes/ContentContributionsRoute.svelte.d.ts +10 -0
  261. package/dist/svelte/routes/ContentContributionsRoute.svelte.d.ts.map +1 -0
  262. package/dist/svelte/routes/ContentFactsRoute.svelte +612 -0
  263. package/dist/svelte/routes/ContentFactsRoute.svelte.d.ts +11 -0
  264. package/dist/svelte/routes/ContentFactsRoute.svelte.d.ts.map +1 -0
  265. package/dist/svelte/routes/ContentGovernanceRoute.svelte +218 -0
  266. package/dist/svelte/routes/ContentGovernanceRoute.svelte.d.ts +10 -0
  267. package/dist/svelte/routes/ContentGovernanceRoute.svelte.d.ts.map +1 -0
  268. package/dist/svelte/routes/ContentWorkspaceRoute.svelte +431 -0
  269. package/dist/svelte/routes/ContentWorkspaceRoute.svelte.d.ts +12 -0
  270. package/dist/svelte/routes/ContentWorkspaceRoute.svelte.d.ts.map +1 -0
  271. package/dist/svelte/routes/PublishedArticleRoute.svelte +194 -0
  272. package/dist/svelte/routes/PublishedArticleRoute.svelte.d.ts +10 -0
  273. package/dist/svelte/routes/PublishedArticleRoute.svelte.d.ts.map +1 -0
  274. package/dist/svelte/routes/index.d.ts +8 -0
  275. package/dist/svelte/routes/index.d.ts.map +1 -0
  276. package/dist/svelte/routes/index.js +6 -0
  277. package/dist/svelte/routes/shared.d.ts +90 -0
  278. package/dist/svelte/routes/shared.d.ts.map +1 -0
  279. package/dist/svelte/routes/shared.js +104 -0
  280. package/dist/svelte/types.d.ts +69 -0
  281. package/dist/svelte/types.d.ts.map +1 -0
  282. package/dist/svelte/types.js +6 -0
  283. package/dist/thumbnail-generator.d.ts +174 -0
  284. package/dist/thumbnail-generator.d.ts.map +1 -0
  285. package/dist/ui.d.ts +10 -0
  286. package/dist/ui.d.ts.map +1 -0
  287. package/dist/ui.js +42 -0
  288. package/dist/ui.js.map +1 -0
  289. package/dist/utils.d.ts +18 -0
  290. package/dist/utils.d.ts.map +1 -0
  291. package/package.json +119 -0
@@ -0,0 +1,226 @@
1
+ <script lang="ts">
2
+ import { useI18n } from '@happyvertical/smrt-ui/i18n';
3
+ import type {
4
+ ContentContributionData,
5
+ ContentContributionTypeData,
6
+ } from '../../mock-smrt-client';
7
+ import { M } from '../i18n.contribution.js';
8
+
9
+ const { t } = useI18n();
10
+
11
+ export interface ContentContributionFormSubmitData {
12
+ typeKey: string;
13
+ contributorEmail?: string;
14
+ contributorName?: string;
15
+ title: string;
16
+ description: string;
17
+ body: string;
18
+ files: File[];
19
+ }
20
+
21
+ export interface Props {
22
+ types?: ContentContributionTypeData[];
23
+ initial?: Partial<ContentContributionData>;
24
+ showContributorFields?: boolean;
25
+ submitLabel?: string;
26
+ action?: string;
27
+ onSubmit?: (payload: ContentContributionFormSubmitData) => void;
28
+ onCancel?: () => void;
29
+ }
30
+
31
+ let {
32
+ types = [],
33
+ initial = {},
34
+ showContributorFields = true,
35
+ submitLabel = 'Submit contribution',
36
+ action = undefined,
37
+ onSubmit,
38
+ onCancel = undefined,
39
+ }: Props = $props();
40
+
41
+ function createDraft(
42
+ source: Partial<ContentContributionData>,
43
+ availableTypes: ContentContributionTypeData[],
44
+ ) {
45
+ return {
46
+ typeKey: source.contributionTypeKey || availableTypes[0]?.key || '',
47
+ contributorEmail: source.contributorEmail || '',
48
+ contributorName: source.contributorName || '',
49
+ title: source.title || '',
50
+ description: source.description || '',
51
+ body: source.body || '',
52
+ files: [] as File[],
53
+ };
54
+ }
55
+
56
+ function getInitialSignature(source: Partial<ContentContributionData>) {
57
+ return JSON.stringify({
58
+ contributionTypeKey: source.contributionTypeKey || '',
59
+ contributorEmail: source.contributorEmail || '',
60
+ contributorName: source.contributorName || '',
61
+ title: source.title || '',
62
+ description: source.description || '',
63
+ body: source.body || '',
64
+ });
65
+ }
66
+
67
+ let draft = $state(createDraft({}, []));
68
+ let lastInitialSignature = $state('__unset__');
69
+
70
+ $effect(() => {
71
+ const nextInitialSignature = getInitialSignature(initial);
72
+ if (nextInitialSignature !== lastInitialSignature) {
73
+ draft = createDraft(initial, types);
74
+ lastInitialSignature = nextInitialSignature;
75
+ return;
76
+ }
77
+
78
+ if (types.length === 0) {
79
+ if (draft.typeKey) {
80
+ draft.typeKey = '';
81
+ }
82
+ return;
83
+ }
84
+
85
+ const hasValidType = types.some((type) => type.key === draft.typeKey);
86
+ if (!hasValidType) {
87
+ draft.typeKey = types[0]?.key || '';
88
+ }
89
+ });
90
+
91
+ const activeType = $derived(
92
+ types.find((type) => type.key === draft.typeKey) || null,
93
+ );
94
+
95
+ function handleFileChange(event: Event) {
96
+ const target = event.currentTarget as HTMLInputElement;
97
+ draft.files = Array.from(target.files || []);
98
+ }
99
+
100
+ function handleSubmit(event: SubmitEvent) {
101
+ if (!onSubmit) {
102
+ if (!action?.trim()) {
103
+ event.preventDefault();
104
+ }
105
+ return;
106
+ }
107
+
108
+ event.preventDefault();
109
+ onSubmit({
110
+ typeKey: draft.typeKey,
111
+ contributorEmail: draft.contributorEmail || undefined,
112
+ contributorName: draft.contributorName || undefined,
113
+ title: draft.title,
114
+ description: draft.description,
115
+ body: draft.body,
116
+ files: draft.files,
117
+ });
118
+ }
119
+ </script>
120
+
121
+ <form
122
+ class="contribution-form"
123
+ method="post"
124
+ enctype="multipart/form-data"
125
+ {action}
126
+ onsubmit={handleSubmit}
127
+ >
128
+ <h3>{t(M['content.contribution_form.heading'])}</h3>
129
+
130
+ <label>
131
+ {t(M['content.contribution_form.contribution_type'])}
132
+ <select name="typeKey" bind:value={draft.typeKey} required>
133
+ {#each types.filter((type) => type.enabled !== false) as type (type.key)}
134
+ <option value={type.key}>{type.label}</option>
135
+ {/each}
136
+ </select>
137
+ </label>
138
+
139
+ {#if showContributorFields}
140
+ <div class="grid">
141
+ <label>
142
+ Email
143
+ <input name="contributorEmail" type="email" bind:value={draft.contributorEmail} required />
144
+ </label>
145
+ <label>
146
+ Name
147
+ <input name="contributorName" type="text" bind:value={draft.contributorName} />
148
+ </label>
149
+ </div>
150
+ {/if}
151
+
152
+ <label>
153
+ Title
154
+ <input name="title" type="text" bind:value={draft.title} />
155
+ </label>
156
+
157
+ <label>
158
+ Description
159
+ <textarea name="description" bind:value={draft.description} rows="2"></textarea>
160
+ </label>
161
+
162
+ <label>
163
+ Body
164
+ <textarea name="body" bind:value={draft.body} rows="8"></textarea>
165
+ </label>
166
+
167
+ {#if activeType?.allowFiles !== false}
168
+ <label>
169
+ {t(M['content.contribution_form.attach_files'])}
170
+ <input
171
+ name="files"
172
+ type="file"
173
+ multiple
174
+ onchange={handleFileChange}
175
+ />
176
+ </label>
177
+ {#if draft.files.length > 0}
178
+ <div class="file-list">
179
+ {#each draft.files as file (file.name + file.size)}
180
+ <span>{file.name} ({Math.max(1, Math.round(file.size / 1024))} KB)</span>
181
+ {/each}
182
+ </div>
183
+ {/if}
184
+ {/if}
185
+
186
+ <div class="actions">
187
+ <button type="submit">{submitLabel}</button>
188
+ {#if onCancel}
189
+ <button type="button" class="secondary" onclick={() => onCancel?.()}>
190
+ Cancel
191
+ </button>
192
+ {/if}
193
+ </div>
194
+ </form>
195
+
196
+ <style>
197
+ .contribution-form {
198
+ display: grid;
199
+ gap: 0.85rem;
200
+ }
201
+
202
+ .grid {
203
+ display: grid;
204
+ gap: 0.75rem;
205
+ grid-template-columns: repeat(auto-fit, minmax(12rem, 1fr));
206
+ }
207
+
208
+ label {
209
+ display: grid;
210
+ gap: 0.35rem;
211
+ }
212
+
213
+ .file-list {
214
+ display: flex;
215
+ gap: 0.5rem;
216
+ flex-wrap: wrap;
217
+ font-size: var(--smrt-typography-body-medium-size, 0.9rem);
218
+ color: var(--smrt-color-on-surface-variant);
219
+ }
220
+
221
+ .actions {
222
+ display: flex;
223
+ gap: 0.75rem;
224
+ flex-wrap: wrap;
225
+ }
226
+ </style>
@@ -0,0 +1,23 @@
1
+ import type { ContentContributionData, ContentContributionTypeData } from '../../mock-smrt-client';
2
+ export interface ContentContributionFormSubmitData {
3
+ typeKey: string;
4
+ contributorEmail?: string;
5
+ contributorName?: string;
6
+ title: string;
7
+ description: string;
8
+ body: string;
9
+ files: File[];
10
+ }
11
+ export interface Props {
12
+ types?: ContentContributionTypeData[];
13
+ initial?: Partial<ContentContributionData>;
14
+ showContributorFields?: boolean;
15
+ submitLabel?: string;
16
+ action?: string;
17
+ onSubmit?: (payload: ContentContributionFormSubmitData) => void;
18
+ onCancel?: () => void;
19
+ }
20
+ declare const ContentContributionForm: import("svelte").Component<Props, {}, "">;
21
+ type ContentContributionForm = ReturnType<typeof ContentContributionForm>;
22
+ export default ContentContributionForm;
23
+ //# sourceMappingURL=ContentContributionForm.svelte.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ContentContributionForm.svelte.d.ts","sourceRoot":"","sources":["../../../src/svelte/components/ContentContributionForm.svelte.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EACV,uBAAuB,EACvB,2BAA2B,EAC5B,MAAM,wBAAwB,CAAC;AAIhC,MAAM,WAAW,iCAAiC;IAChD,OAAO,EAAE,MAAM,CAAC;IAChB,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,KAAK,EAAE,MAAM,CAAC;IACd,WAAW,EAAE,MAAM,CAAC;IACpB,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,IAAI,EAAE,CAAC;CACf;AAED,MAAM,WAAW,KAAK;IACpB,KAAK,CAAC,EAAE,2BAA2B,EAAE,CAAC;IACtC,OAAO,CAAC,EAAE,OAAO,CAAC,uBAAuB,CAAC,CAAC;IAC3C,qBAAqB,CAAC,EAAE,OAAO,CAAC;IAChC,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,QAAQ,CAAC,EAAE,CAAC,OAAO,EAAE,iCAAiC,KAAK,IAAI,CAAC;IAChE,QAAQ,CAAC,EAAE,MAAM,IAAI,CAAC;CACvB;AAwKD,QAAA,MAAM,uBAAuB,2CAAwC,CAAC;AACtE,KAAK,uBAAuB,GAAG,UAAU,CAAC,OAAO,uBAAuB,CAAC,CAAC;AAC1E,eAAe,uBAAuB,CAAC"}
@@ -0,0 +1,322 @@
1
+ <script lang="ts">
2
+ import { useI18n } from '@happyvertical/smrt-ui/i18n';
3
+ import type { ContentContributionData } from '../../mock-smrt-client';
4
+ import { M } from '../i18n.contribution.js';
5
+
6
+ const { t } = useI18n();
7
+
8
+ export interface Props {
9
+ contributions?: ContentContributionData[];
10
+ selectedId?: string | null;
11
+ emptyMessage?: string;
12
+ workflowFormAction?: string;
13
+ onSelect?: (contribution: ContentContributionData) => void;
14
+ onApprove?: (
15
+ contribution: ContentContributionData,
16
+ options: { targetStatus: 'draft' | 'review'; note: string },
17
+ ) => void;
18
+ onRequestChanges?: (
19
+ contribution: ContentContributionData,
20
+ options: { note: string },
21
+ ) => void;
22
+ onReject?: (
23
+ contribution: ContentContributionData,
24
+ options: { note: string },
25
+ ) => void;
26
+ }
27
+
28
+ let {
29
+ contributions = [],
30
+ selectedId = null,
31
+ emptyMessage = 'No contributions need review right now.',
32
+ workflowFormAction = undefined,
33
+ onSelect = undefined,
34
+ onApprove = undefined,
35
+ onRequestChanges = undefined,
36
+ onReject = undefined,
37
+ }: Props = $props();
38
+
39
+ let note = $state('');
40
+ let targetStatus = $state<'draft' | 'review'>('draft');
41
+ const workflowIntents = ['approve', 'request-changes', 'reject'] as const;
42
+ type WorkflowIntent = (typeof workflowIntents)[number];
43
+
44
+ const selectedContribution = $derived(
45
+ contributions.find((item) => item.id === selectedId) ||
46
+ contributions[0] ||
47
+ null,
48
+ );
49
+ const canSubmitWorkflow = $derived(
50
+ !workflowFormAction || Boolean(selectedContribution?.id),
51
+ );
52
+
53
+ function workflowStatus(
54
+ contribution: ContentContributionData | null | undefined,
55
+ ) {
56
+ return contribution?.status || 'submitted';
57
+ }
58
+
59
+ function approveActionLabel(contribution: ContentContributionData) {
60
+ return workflowStatus(contribution) === 'approved'
61
+ ? 'Promote'
62
+ : 'Approve and promote';
63
+ }
64
+
65
+ function isWorkflowIntent(
66
+ intent: string | undefined,
67
+ ): intent is WorkflowIntent {
68
+ return workflowIntents.includes(intent as WorkflowIntent);
69
+ }
70
+
71
+ function handleWorkflowSubmit(event: SubmitEvent) {
72
+ const submitter = event.submitter as HTMLButtonElement | null;
73
+ const intent = submitter?.value;
74
+ const hasNativeWorkflowTarget =
75
+ Boolean(workflowFormAction) &&
76
+ Boolean(selectedContribution?.id) &&
77
+ isWorkflowIntent(intent);
78
+ const shouldHandleWithCallback =
79
+ Boolean(selectedContribution) &&
80
+ ((intent === 'approve' && Boolean(onApprove)) ||
81
+ (intent === 'request-changes' && Boolean(onRequestChanges)) ||
82
+ (intent === 'reject' && Boolean(onReject)));
83
+
84
+ if (!hasNativeWorkflowTarget || shouldHandleWithCallback) {
85
+ event.preventDefault();
86
+ }
87
+
88
+ if (intent === 'approve' && onApprove && selectedContribution) {
89
+ onApprove(selectedContribution, {
90
+ targetStatus,
91
+ note,
92
+ });
93
+ return;
94
+ }
95
+
96
+ if (
97
+ intent === 'request-changes' &&
98
+ onRequestChanges &&
99
+ selectedContribution
100
+ ) {
101
+ onRequestChanges(selectedContribution, {
102
+ note,
103
+ });
104
+ return;
105
+ }
106
+
107
+ if (intent === 'reject' && onReject && selectedContribution) {
108
+ onReject(selectedContribution, {
109
+ note,
110
+ });
111
+ }
112
+ }
113
+
114
+ $effect(() => {
115
+ note = selectedContribution?.editorNotes || '';
116
+ targetStatus = 'draft';
117
+ });
118
+ </script>
119
+
120
+ <section class="inbox">
121
+ <header class="inbox__header">
122
+ <div>
123
+ <h3>{t(M['content.contribution_inbox.heading'])}</h3>
124
+ <p>{t(M['content.contribution_inbox.intro'])}</p>
125
+ </div>
126
+ <span class="pill">{contributions.length}</span>
127
+ </header>
128
+
129
+ {#if contributions.length === 0}
130
+ <p class="empty-copy">{emptyMessage}</p>
131
+ {:else}
132
+ <div class="inbox__layout">
133
+ <div class="inbox__list">
134
+ {#each contributions as contribution (contribution.id)}
135
+ <button
136
+ type="button"
137
+ class:selected={selectedContribution?.id === contribution.id}
138
+ onclick={() => onSelect?.(contribution)}
139
+ >
140
+ <strong>{contribution.title || contribution.contributionTypeKey || 'Untitled submission'}</strong>
141
+ <span>{contribution.contributorName || contribution.contributorEmail || 'Unknown contributor'}</span>
142
+ <span class="pill">{workflowStatus(contribution)}</span>
143
+ </button>
144
+ {/each}
145
+ </div>
146
+
147
+ {#if selectedContribution}
148
+ <article class="inbox__detail">
149
+ <header>
150
+ <div>
151
+ <h4>{selectedContribution.title || 'Untitled submission'}</h4>
152
+ <p>{selectedContribution.contributorName || selectedContribution.contributorEmail || 'Unknown contributor'}</p>
153
+ </div>
154
+ <span class="pill">{workflowStatus(selectedContribution)}</span>
155
+ </header>
156
+
157
+ {#if selectedContribution.body}
158
+ <div class="body-preview">
159
+ {selectedContribution.body}
160
+ </div>
161
+ {/if}
162
+
163
+ <dl>
164
+ <div>
165
+ <dt>Type</dt>
166
+ <dd>{selectedContribution.contributionTypeKey || 'n/a'}</dd>
167
+ </div>
168
+ <div>
169
+ <dt>Revisions</dt>
170
+ <dd>{selectedContribution.revisionCount || 0}</dd>
171
+ </div>
172
+ <div>
173
+ <dt>{t(M['content.contribution_inbox.promoted_content'])}</dt>
174
+ <dd>{selectedContribution.promotedContentId || 'Not promoted yet'}</dd>
175
+ </div>
176
+ {#if selectedContribution.intakeDecision}
177
+ <div>
178
+ <dt>{t(M['content.contribution_inbox.intake_decision'])}</dt>
179
+ <dd>{selectedContribution.intakeDecision}</dd>
180
+ </div>
181
+ {/if}
182
+ </dl>
183
+
184
+ <form
185
+ method="post"
186
+ action={workflowFormAction}
187
+ onsubmit={handleWorkflowSubmit}
188
+ >
189
+ {#if selectedContribution.id}
190
+ <input type="hidden" name="contributionId" value={selectedContribution.id} />
191
+ {/if}
192
+
193
+ <label>
194
+ {t(M['content.contribution_inbox.editorial_note'])}
195
+ <textarea name="editorNote" bind:value={note} rows="4"></textarea>
196
+ </label>
197
+
198
+ <div class="actions">
199
+ {#if onApprove || workflowFormAction}
200
+ <label class="inline">
201
+ {t(M['content.contribution_inbox.promote_to'])}
202
+ <select name="targetStatus" bind:value={targetStatus}>
203
+ <option value="draft">draft</option>
204
+ <option value="review">review</option>
205
+ </select>
206
+ </label>
207
+ <button
208
+ type="submit"
209
+ name="intent"
210
+ value="approve"
211
+ disabled={!canSubmitWorkflow}
212
+ >
213
+ {approveActionLabel(selectedContribution)}
214
+ </button>
215
+ {/if}
216
+
217
+ {#if onRequestChanges || workflowFormAction}
218
+ <button
219
+ type="submit"
220
+ name="intent"
221
+ value="request-changes"
222
+ class="secondary"
223
+ disabled={!canSubmitWorkflow}
224
+ >
225
+ {t(M['content.contribution_inbox.request_changes'])}
226
+ </button>
227
+ {/if}
228
+
229
+ {#if onReject || workflowFormAction}
230
+ <button
231
+ type="submit"
232
+ name="intent"
233
+ value="reject"
234
+ class="danger"
235
+ disabled={!canSubmitWorkflow}
236
+ >
237
+ Reject
238
+ </button>
239
+ {/if}
240
+ </div>
241
+ </form>
242
+ </article>
243
+ {/if}
244
+ </div>
245
+ {/if}
246
+ </section>
247
+
248
+ <style>
249
+ .inbox,
250
+ .inbox__header,
251
+ .inbox__layout,
252
+ .inbox__detail,
253
+ .inbox__detail form {
254
+ display: grid;
255
+ gap: 1rem;
256
+ }
257
+
258
+ .inbox__layout {
259
+ grid-template-columns: minmax(14rem, 20rem) minmax(0, 1fr);
260
+ }
261
+
262
+ .inbox__list {
263
+ display: grid;
264
+ gap: 0.5rem;
265
+ }
266
+
267
+ .inbox__list button,
268
+ .inbox__detail {
269
+ border: 1px solid var(--smrt-color-outline-variant);
270
+ border-radius: 0.75rem;
271
+ padding: 0.85rem;
272
+ background: var(--smrt-color-surface);
273
+ }
274
+
275
+ .inbox__list button {
276
+ display: grid;
277
+ gap: 0.35rem;
278
+ text-align: left;
279
+ }
280
+
281
+ .inbox__list button.selected {
282
+ border-color: var(--smrt-color-primary);
283
+ box-shadow: 0 0 0 1px var(--smrt-color-primary);
284
+ }
285
+
286
+ .body-preview {
287
+ padding: 0.85rem;
288
+ border-radius: 0.75rem;
289
+ background: var(--smrt-color-surface-container);
290
+ white-space: pre-wrap;
291
+ }
292
+
293
+ dl div {
294
+ display: flex;
295
+ justify-content: space-between;
296
+ gap: 1rem;
297
+ }
298
+
299
+ .actions,
300
+ .inline {
301
+ display: flex;
302
+ gap: 0.75rem;
303
+ align-items: center;
304
+ flex-wrap: wrap;
305
+ }
306
+
307
+ .pill {
308
+ display: inline-flex;
309
+ align-items: center;
310
+ justify-content: center;
311
+ min-width: 1.75rem;
312
+ padding: 0.15rem 0.5rem;
313
+ border-radius: var(--smrt-radius-full, 9999px);
314
+ background: var(--smrt-color-primary-container);
315
+ }
316
+
317
+ @media (max-width: 720px) {
318
+ .inbox__layout {
319
+ grid-template-columns: 1fr;
320
+ }
321
+ }
322
+ </style>
@@ -0,0 +1,22 @@
1
+ import type { ContentContributionData } from '../../mock-smrt-client';
2
+ export interface Props {
3
+ contributions?: ContentContributionData[];
4
+ selectedId?: string | null;
5
+ emptyMessage?: string;
6
+ workflowFormAction?: string;
7
+ onSelect?: (contribution: ContentContributionData) => void;
8
+ onApprove?: (contribution: ContentContributionData, options: {
9
+ targetStatus: 'draft' | 'review';
10
+ note: string;
11
+ }) => void;
12
+ onRequestChanges?: (contribution: ContentContributionData, options: {
13
+ note: string;
14
+ }) => void;
15
+ onReject?: (contribution: ContentContributionData, options: {
16
+ note: string;
17
+ }) => void;
18
+ }
19
+ declare const ContentContributionInbox: import("svelte").Component<Props, {}, "">;
20
+ type ContentContributionInbox = ReturnType<typeof ContentContributionInbox>;
21
+ export default ContentContributionInbox;
22
+ //# sourceMappingURL=ContentContributionInbox.svelte.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ContentContributionInbox.svelte.d.ts","sourceRoot":"","sources":["../../../src/svelte/components/ContentContributionInbox.svelte.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,uBAAuB,EAAE,MAAM,wBAAwB,CAAC;AAItE,MAAM,WAAW,KAAK;IACpB,aAAa,CAAC,EAAE,uBAAuB,EAAE,CAAC;IAC1C,UAAU,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC3B,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,kBAAkB,CAAC,EAAE,MAAM,CAAC;IAC5B,QAAQ,CAAC,EAAE,CAAC,YAAY,EAAE,uBAAuB,KAAK,IAAI,CAAC;IAC3D,SAAS,CAAC,EAAE,CACV,YAAY,EAAE,uBAAuB,EACrC,OAAO,EAAE;QAAE,YAAY,EAAE,OAAO,GAAG,QAAQ,CAAC;QAAC,IAAI,EAAE,MAAM,CAAA;KAAE,KACxD,IAAI,CAAC;IACV,gBAAgB,CAAC,EAAE,CACjB,YAAY,EAAE,uBAAuB,EACrC,OAAO,EAAE;QAAE,IAAI,EAAE,MAAM,CAAA;KAAE,KACtB,IAAI,CAAC;IACV,QAAQ,CAAC,EAAE,CACT,YAAY,EAAE,uBAAuB,EACrC,OAAO,EAAE;QAAE,IAAI,EAAE,MAAM,CAAA;KAAE,KACtB,IAAI,CAAC;CACX;AAgND,QAAA,MAAM,wBAAwB,2CAAwC,CAAC;AACvE,KAAK,wBAAwB,GAAG,UAAU,CAAC,OAAO,wBAAwB,CAAC,CAAC;AAC5E,eAAe,wBAAwB,CAAC"}