@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,182 @@
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
+ onSelect?: (contribution: ContentContributionData) => void;
13
+ onWithdraw?: (contribution: ContentContributionData) => void;
14
+ }
15
+
16
+ let {
17
+ contributions = [],
18
+ selectedId = null,
19
+ emptyMessage = 'No submissions yet.',
20
+ onSelect = undefined,
21
+ onWithdraw = undefined,
22
+ }: Props = $props();
23
+
24
+ const selectedContribution = $derived(
25
+ contributions.find((item) => item.id === selectedId) ||
26
+ contributions[0] ||
27
+ null,
28
+ );
29
+
30
+ function canWithdraw(contribution: ContentContributionData) {
31
+ return !['promoted', 'rejected', 'withdrawn'].includes(
32
+ contribution.status || '',
33
+ );
34
+ }
35
+ </script>
36
+
37
+ <section class="portal">
38
+ <header class="portal__header">
39
+ <div>
40
+ <h3>{t(M['content.contribution_portal.heading'])}</h3>
41
+ <p>{t(M['content.contribution_portal.intro'])}</p>
42
+ </div>
43
+ <span class="pill">{contributions.length}</span>
44
+ </header>
45
+
46
+ {#if contributions.length === 0}
47
+ <p class="empty-copy">{emptyMessage}</p>
48
+ {:else}
49
+ <div class="portal__layout">
50
+ <div class="portal__list">
51
+ {#each contributions as contribution (contribution.id)}
52
+ <button
53
+ type="button"
54
+ class:selected={selectedContribution?.id === contribution.id}
55
+ onclick={() => onSelect?.(contribution)}
56
+ >
57
+ <strong>{contribution.title || contribution.contributionTypeKey || 'Untitled submission'}</strong>
58
+ <span>{contribution.status || 'submitted'}</span>
59
+ </button>
60
+ {/each}
61
+ </div>
62
+
63
+ {#if selectedContribution}
64
+ <article class="portal__detail">
65
+ <header>
66
+ <h4>{selectedContribution.title || 'Untitled submission'}</h4>
67
+ <span class="pill">{selectedContribution.status || 'submitted'}</span>
68
+ </header>
69
+ {#if selectedContribution.description}
70
+ <p>{selectedContribution.description}</p>
71
+ {/if}
72
+ {#if selectedContribution.editorNotes}
73
+ <div class="callout">
74
+ <strong>{t(M['content.contribution_portal.editor_notes'])}</strong>
75
+ <p>{selectedContribution.editorNotes}</p>
76
+ </div>
77
+ {/if}
78
+ <dl>
79
+ <div>
80
+ <dt>Type</dt>
81
+ <dd>{selectedContribution.contributionTypeKey || 'n/a'}</dd>
82
+ </div>
83
+ <div>
84
+ <dt>Revisions</dt>
85
+ <dd>{selectedContribution.revisionCount || 0}</dd>
86
+ </div>
87
+ <div>
88
+ <dt>Updated</dt>
89
+ <dd>{selectedContribution.updatedAt || selectedContribution.createdAt || 'n/a'}</dd>
90
+ </div>
91
+ </dl>
92
+
93
+ {#if onWithdraw && canWithdraw(selectedContribution)}
94
+ <button type="button" class="secondary" onclick={() => onWithdraw?.(selectedContribution)}>
95
+ {t(M['content.contribution_portal.withdraw_submission'])}
96
+ </button>
97
+ {/if}
98
+ </article>
99
+ {/if}
100
+ </div>
101
+ {/if}
102
+ </section>
103
+
104
+ <style>
105
+ .portal {
106
+ display: grid;
107
+ gap: 1rem;
108
+ }
109
+
110
+ .portal__header,
111
+ .portal__layout {
112
+ display: grid;
113
+ gap: 1rem;
114
+ }
115
+
116
+ .portal__layout {
117
+ grid-template-columns: minmax(14rem, 20rem) minmax(0, 1fr);
118
+ }
119
+
120
+ .portal__list {
121
+ display: grid;
122
+ gap: 0.5rem;
123
+ }
124
+
125
+ .portal__list button,
126
+ .portal__detail {
127
+ border: 1px solid var(--smrt-color-outline-variant);
128
+ border-radius: 0.75rem;
129
+ padding: 0.85rem;
130
+ background: var(--smrt-color-surface);
131
+ }
132
+
133
+ .portal__list button {
134
+ display: grid;
135
+ gap: 0.35rem;
136
+ text-align: left;
137
+ }
138
+
139
+ .portal__list button.selected {
140
+ border-color: var(--smrt-color-primary);
141
+ box-shadow: 0 0 0 1px var(--smrt-color-primary);
142
+ }
143
+
144
+ .portal__detail {
145
+ display: grid;
146
+ gap: 0.85rem;
147
+ }
148
+
149
+ .portal__detail header,
150
+ dl {
151
+ display: grid;
152
+ gap: 0.5rem;
153
+ }
154
+
155
+ dl div {
156
+ display: flex;
157
+ justify-content: space-between;
158
+ gap: 1rem;
159
+ }
160
+
161
+ .pill {
162
+ display: inline-flex;
163
+ align-items: center;
164
+ justify-content: center;
165
+ min-width: 1.75rem;
166
+ padding: 0.15rem 0.5rem;
167
+ border-radius: var(--smrt-radius-full, 9999px);
168
+ background: var(--smrt-color-primary-container);
169
+ }
170
+
171
+ .callout {
172
+ padding: 0.75rem;
173
+ border-radius: 0.75rem;
174
+ background: var(--smrt-color-surface-container);
175
+ }
176
+
177
+ @media (max-width: 720px) {
178
+ .portal__layout {
179
+ grid-template-columns: 1fr;
180
+ }
181
+ }
182
+ </style>
@@ -0,0 +1,12 @@
1
+ import type { ContentContributionData } from '../../mock-smrt-client';
2
+ export interface Props {
3
+ contributions?: ContentContributionData[];
4
+ selectedId?: string | null;
5
+ emptyMessage?: string;
6
+ onSelect?: (contribution: ContentContributionData) => void;
7
+ onWithdraw?: (contribution: ContentContributionData) => void;
8
+ }
9
+ declare const ContentContributionPortal: import("svelte").Component<Props, {}, "">;
10
+ type ContentContributionPortal = ReturnType<typeof ContentContributionPortal>;
11
+ export default ContentContributionPortal;
12
+ //# sourceMappingURL=ContentContributionPortal.svelte.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ContentContributionPortal.svelte.d.ts","sourceRoot":"","sources":["../../../src/svelte/components/ContentContributionPortal.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,QAAQ,CAAC,EAAE,CAAC,YAAY,EAAE,uBAAuB,KAAK,IAAI,CAAC;IAC3D,UAAU,CAAC,EAAE,CAAC,YAAY,EAAE,uBAAuB,KAAK,IAAI,CAAC;CAC9D;AAiGD,QAAA,MAAM,yBAAyB,2CAAwC,CAAC;AACxE,KAAK,yBAAyB,GAAG,UAAU,CAAC,OAAO,yBAAyB,CAAC,CAAC;AAC9E,eAAe,yBAAyB,CAAC"}
@@ -0,0 +1,281 @@
1
+ <script lang="ts">
2
+ import { useI18n } from '@happyvertical/smrt-ui/i18n';
3
+ import type { ContentContributionTypeData } from '../../mock-smrt-client';
4
+ import { M } from '../i18n.contribution.js';
5
+
6
+ const { t } = useI18n();
7
+
8
+ export interface Props {
9
+ types?: ContentContributionTypeData[];
10
+ onSave: (type: Partial<ContentContributionTypeData>) => void;
11
+ onDelete?: (type: ContentContributionTypeData) => void;
12
+ }
13
+
14
+ let { types = [], onSave, onDelete = undefined }: Props = $props();
15
+
16
+ function createDraft(source: Partial<ContentContributionTypeData> = {}) {
17
+ return {
18
+ ...source,
19
+ key: source.key || '',
20
+ label: source.label || '',
21
+ enabled: source.enabled ?? true,
22
+ allowedChannels: source.allowedChannels || ['web'],
23
+ allowText: source.allowText ?? true,
24
+ allowFiles: source.allowFiles ?? false,
25
+ allowEmptyText: source.allowEmptyText ?? false,
26
+ intakeRules: {
27
+ maxFiles: source.intakeRules?.maxFiles ?? '',
28
+ maxTotalBytes: source.intakeRules?.maxTotalBytes ?? '',
29
+ allowedMimePatterns: (source.intakeRules?.allowedMimePatterns || []).join(
30
+ ', ',
31
+ ),
32
+ blockedMimePatterns: (source.intakeRules?.blockedMimePatterns || []).join(
33
+ ', ',
34
+ ),
35
+ quarantineMimePatterns: (
36
+ source.intakeRules?.quarantineMimePatterns || []
37
+ ).join(', '),
38
+ blockedTextPatterns: (source.intakeRules?.blockedTextPatterns || []).join(
39
+ ', ',
40
+ ),
41
+ quarantineTextPatterns: (
42
+ source.intakeRules?.quarantineTextPatterns || []
43
+ ).join(', '),
44
+ trustedOnly: source.intakeRules?.trustedOnly ?? false,
45
+ },
46
+ promotion: {
47
+ targetContentType: source.promotion?.targetContentType || '',
48
+ targetContentVariant: source.promotion?.targetContentVariant || '',
49
+ targetContentStatus: source.promotion?.targetContentStatus || 'draft',
50
+ autoPromoteTrusted: source.promotion?.autoPromoteTrusted ?? false,
51
+ createAssets: source.promotion?.createAssets ?? true,
52
+ assetRelationship: source.promotion?.assetRelationship || 'attachment',
53
+ },
54
+ };
55
+ }
56
+
57
+ let editing = $state<Partial<ContentContributionTypeData> | null>(null);
58
+ let draft = $state(createDraft());
59
+
60
+ $effect(() => {
61
+ draft = createDraft(editing || {});
62
+ });
63
+
64
+ function toggleChannel(channel: string) {
65
+ draft.allowedChannels = draft.allowedChannels.includes(channel)
66
+ ? draft.allowedChannels.filter((item: string) => item !== channel)
67
+ : [...draft.allowedChannels, channel];
68
+ }
69
+
70
+ function csv(value: string) {
71
+ return value
72
+ .split(',')
73
+ .map((item) => item.trim())
74
+ .filter(Boolean);
75
+ }
76
+
77
+ function handleSubmit() {
78
+ onSave({
79
+ ...editing,
80
+ key: draft.key,
81
+ label: draft.label,
82
+ enabled: draft.enabled,
83
+ allowedChannels: draft.allowedChannels,
84
+ allowText: draft.allowText,
85
+ allowFiles: draft.allowFiles,
86
+ allowEmptyText: draft.allowEmptyText,
87
+ intakeRules: {
88
+ maxFiles: draft.intakeRules.maxFiles
89
+ ? Number(draft.intakeRules.maxFiles)
90
+ : null,
91
+ maxTotalBytes: draft.intakeRules.maxTotalBytes
92
+ ? Number(draft.intakeRules.maxTotalBytes)
93
+ : null,
94
+ allowedMimePatterns: csv(draft.intakeRules.allowedMimePatterns),
95
+ blockedMimePatterns: csv(draft.intakeRules.blockedMimePatterns),
96
+ quarantineMimePatterns: csv(draft.intakeRules.quarantineMimePatterns),
97
+ blockedTextPatterns: csv(draft.intakeRules.blockedTextPatterns),
98
+ quarantineTextPatterns: csv(draft.intakeRules.quarantineTextPatterns),
99
+ trustedOnly: draft.intakeRules.trustedOnly,
100
+ },
101
+ promotion: {
102
+ targetContentType: draft.promotion.targetContentType,
103
+ targetContentVariant: draft.promotion.targetContentVariant || null,
104
+ targetContentStatus: draft.promotion.targetContentStatus,
105
+ autoPromoteTrusted: draft.promotion.autoPromoteTrusted,
106
+ createAssets: draft.promotion.createAssets,
107
+ assetRelationship: draft.promotion.assetRelationship,
108
+ },
109
+ });
110
+ }
111
+ </script>
112
+
113
+ <section class="manager">
114
+ <header>
115
+ <div>
116
+ <h3>{t(M['content.contribution_type_manager.heading'])}</h3>
117
+ <p>{t(M['content.contribution_type_manager.intro'])}</p>
118
+ </div>
119
+ <button type="button" onclick={() => (editing = {})}>{t(M['content.contribution_type_manager.add_type'])}</button>
120
+ </header>
121
+
122
+ <div class="layout">
123
+ <div class="list">
124
+ {#each types as type (type.id ?? type.key)}
125
+ <article class="card">
126
+ <div>
127
+ <strong>{type.label}</strong>
128
+ <div>{type.key}</div>
129
+ </div>
130
+ <div class="actions">
131
+ <button type="button" class="secondary" onclick={() => (editing = type)}>Edit</button>
132
+ {#if onDelete && type.id}
133
+ <button type="button" class="danger" onclick={() => onDelete?.(type)}>Delete</button>
134
+ {/if}
135
+ </div>
136
+ </article>
137
+ {/each}
138
+ </div>
139
+
140
+ <form
141
+ class="editor"
142
+ onsubmit={(event) => {
143
+ event.preventDefault();
144
+ handleSubmit();
145
+ }}
146
+ >
147
+ <label>
148
+ Key
149
+ <input type="text" bind:value={draft.key} required />
150
+ </label>
151
+ <label>
152
+ Label
153
+ <input type="text" bind:value={draft.label} required />
154
+ </label>
155
+
156
+ <div class="checkbox-grid">
157
+ <label><input type="checkbox" bind:checked={draft.enabled} /> Enabled</label>
158
+ <label><input type="checkbox" bind:checked={draft.allowText} /> {t(M['content.contribution_type_manager.allow_text'])}</label>
159
+ <label><input type="checkbox" bind:checked={draft.allowFiles} /> {t(M['content.contribution_type_manager.allow_files'])}</label>
160
+ <label><input type="checkbox" bind:checked={draft.allowEmptyText} /> {t(M['content.contribution_type_manager.allow_empty_text'])}</label>
161
+ </div>
162
+
163
+ <div class="checkbox-grid">
164
+ <label>
165
+ <input
166
+ type="checkbox"
167
+ checked={draft.allowedChannels.includes('web')}
168
+ onchange={() => toggleChannel('web')}
169
+ />
170
+ Web
171
+ </label>
172
+ <label>
173
+ <input
174
+ type="checkbox"
175
+ checked={draft.allowedChannels.includes('email')}
176
+ onchange={() => toggleChannel('email')}
177
+ />
178
+ Email
179
+ </label>
180
+ </div>
181
+
182
+ <label>
183
+ {t(M['content.contribution_type_manager.promotion_content_type'])}
184
+ <input type="text" bind:value={draft.promotion.targetContentType} required />
185
+ </label>
186
+ <label>
187
+ {t(M['content.contribution_type_manager.promotion_variant'])}
188
+ <input type="text" bind:value={draft.promotion.targetContentVariant} />
189
+ </label>
190
+ <label>
191
+ {t(M['content.contribution_type_manager.promotion_status'])}
192
+ <select bind:value={draft.promotion.targetContentStatus}>
193
+ <option value="draft">draft</option>
194
+ <option value="review">review</option>
195
+ </select>
196
+ </label>
197
+
198
+ <div class="checkbox-grid">
199
+ <label><input type="checkbox" bind:checked={draft.promotion.autoPromoteTrusted} /> {t(M['content.contribution_type_manager.auto_promote_trusted'])}</label>
200
+ <label><input type="checkbox" bind:checked={draft.promotion.createAssets} /> {t(M['content.contribution_type_manager.create_assets_on_promotion'])}</label>
201
+ <label><input type="checkbox" bind:checked={draft.intakeRules.trustedOnly} /> {t(M['content.contribution_type_manager.trusted_contributors_only'])}</label>
202
+ </div>
203
+
204
+ <label>
205
+ {t(M['content.contribution_type_manager.max_files'])}
206
+ <input type="number" min="0" bind:value={draft.intakeRules.maxFiles} />
207
+ </label>
208
+ <label>
209
+ {t(M['content.contribution_type_manager.max_total_bytes'])}
210
+ <input type="number" min="0" bind:value={draft.intakeRules.maxTotalBytes} />
211
+ </label>
212
+ <label>
213
+ {t(M['content.contribution_type_manager.allowed_mime_patterns'])}
214
+ <input type="text" bind:value={draft.intakeRules.allowedMimePatterns} placeholder={t(M['content.contribution_type_manager.allowed_mime_patterns_placeholder'])} />
215
+ </label>
216
+ <label>
217
+ {t(M['content.contribution_type_manager.blocked_mime_patterns'])}
218
+ <input type="text" bind:value={draft.intakeRules.blockedMimePatterns} />
219
+ </label>
220
+ <label>
221
+ {t(M['content.contribution_type_manager.quarantine_mime_patterns'])}
222
+ <input type="text" bind:value={draft.intakeRules.quarantineMimePatterns} />
223
+ </label>
224
+ <label>
225
+ {t(M['content.contribution_type_manager.blocked_text_patterns'])}
226
+ <textarea bind:value={draft.intakeRules.blockedTextPatterns} rows="2"></textarea>
227
+ </label>
228
+ <label>
229
+ {t(M['content.contribution_type_manager.quarantine_text_patterns'])}
230
+ <textarea bind:value={draft.intakeRules.quarantineTextPatterns} rows="2"></textarea>
231
+ </label>
232
+
233
+ <div class="actions">
234
+ <button type="submit">{t(M['content.contribution_type_manager.save_type'])}</button>
235
+ </div>
236
+ </form>
237
+ </div>
238
+ </section>
239
+
240
+ <style>
241
+ .manager,
242
+ .layout,
243
+ .list,
244
+ .editor {
245
+ display: grid;
246
+ gap: 1rem;
247
+ }
248
+
249
+ .layout {
250
+ grid-template-columns: minmax(14rem, 20rem) minmax(0, 1fr);
251
+ }
252
+
253
+ .card,
254
+ .editor {
255
+ border: 1px solid var(--smrt-color-outline-variant);
256
+ border-radius: 0.75rem;
257
+ padding: 0.9rem;
258
+ background: var(--smrt-color-surface);
259
+ }
260
+
261
+ .card,
262
+ .actions,
263
+ .checkbox-grid,
264
+ header {
265
+ display: flex;
266
+ justify-content: space-between;
267
+ gap: 0.75rem;
268
+ flex-wrap: wrap;
269
+ }
270
+
271
+ .editor label {
272
+ display: grid;
273
+ gap: 0.35rem;
274
+ }
275
+
276
+ @media (max-width: 720px) {
277
+ .layout {
278
+ grid-template-columns: 1fr;
279
+ }
280
+ }
281
+ </style>
@@ -0,0 +1,10 @@
1
+ import type { ContentContributionTypeData } from '../../mock-smrt-client';
2
+ export interface Props {
3
+ types?: ContentContributionTypeData[];
4
+ onSave: (type: Partial<ContentContributionTypeData>) => void;
5
+ onDelete?: (type: ContentContributionTypeData) => void;
6
+ }
7
+ declare const ContentContributionTypeManager: import("svelte").Component<Props, {}, "">;
8
+ type ContentContributionTypeManager = ReturnType<typeof ContentContributionTypeManager>;
9
+ export default ContentContributionTypeManager;
10
+ //# sourceMappingURL=ContentContributionTypeManager.svelte.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ContentContributionTypeManager.svelte.d.ts","sourceRoot":"","sources":["../../../src/svelte/components/ContentContributionTypeManager.svelte.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,2BAA2B,EAAE,MAAM,wBAAwB,CAAC;AAI1E,MAAM,WAAW,KAAK;IACpB,KAAK,CAAC,EAAE,2BAA2B,EAAE,CAAC;IACtC,MAAM,EAAE,CAAC,IAAI,EAAE,OAAO,CAAC,2BAA2B,CAAC,KAAK,IAAI,CAAC;IAC7D,QAAQ,CAAC,EAAE,CAAC,IAAI,EAAE,2BAA2B,KAAK,IAAI,CAAC;CACxD;AAoOD,QAAA,MAAM,8BAA8B,2CAAwC,CAAC;AAC7E,KAAK,8BAA8B,GAAG,UAAU,CAAC,OAAO,8BAA8B,CAAC,CAAC;AACxF,eAAe,8BAA8B,CAAC"}
@@ -0,0 +1,140 @@
1
+ <script lang="ts">
2
+ import { useI18n } from '@happyvertical/smrt-ui/i18n';
3
+ import type { ContentContributorData } from '../../mock-smrt-client';
4
+ import { M } from '../i18n.contribution.js';
5
+
6
+ const { t } = useI18n();
7
+
8
+ export interface Props {
9
+ contributors?: ContentContributorData[];
10
+ onSave: (contributor: Partial<ContentContributorData>) => void;
11
+ onDelete?: (contributor: ContentContributorData) => void;
12
+ }
13
+
14
+ let { contributors = [], onSave, onDelete = undefined }: Props = $props();
15
+
16
+ function createDraft(source: Partial<ContentContributorData> = {}) {
17
+ return {
18
+ ...source,
19
+ email: source.email || '',
20
+ name: source.name || '',
21
+ trustLevel: source.trustLevel || 'standard',
22
+ };
23
+ }
24
+
25
+ let editing = $state<Partial<ContentContributorData> | null>(null);
26
+ let draft = $state(createDraft());
27
+
28
+ $effect(() => {
29
+ draft = createDraft(editing || {});
30
+ });
31
+
32
+ function handleSubmit() {
33
+ onSave({
34
+ ...editing,
35
+ email: draft.email,
36
+ name: draft.name,
37
+ trustLevel: draft.trustLevel,
38
+ });
39
+ }
40
+ </script>
41
+
42
+ <section class="manager">
43
+ <header>
44
+ <div>
45
+ <h3>Contributors</h3>
46
+ <p>{t(M['content.contributor_manager.intro'])}</p>
47
+ </div>
48
+ <button type="button" onclick={() => (editing = {})}>{t(M['content.contributor_manager.add_contributor'])}</button>
49
+ </header>
50
+
51
+ <div class="layout">
52
+ <div class="list">
53
+ {#each contributors as contributor (contributor.id ?? contributor.email)}
54
+ <article class="card">
55
+ <div>
56
+ <strong>{contributor.name || contributor.email}</strong>
57
+ <div>{contributor.trustLevel || 'standard'}</div>
58
+ </div>
59
+ <div class="actions">
60
+ <button type="button" class="secondary" onclick={() => (editing = contributor)}>Edit</button>
61
+ {#if onDelete}
62
+ <button type="button" class="danger" onclick={() => onDelete?.(contributor)}>Delete</button>
63
+ {/if}
64
+ </div>
65
+ </article>
66
+ {/each}
67
+ </div>
68
+
69
+ <form
70
+ class="editor"
71
+ onsubmit={(event) => {
72
+ event.preventDefault();
73
+ handleSubmit();
74
+ }}
75
+ >
76
+ <label>
77
+ Email
78
+ <input type="email" bind:value={draft.email} required />
79
+ </label>
80
+ <label>
81
+ Name
82
+ <input type="text" bind:value={draft.name} />
83
+ </label>
84
+ <label>
85
+ {t(M['content.contributor_manager.trust_level'])}
86
+ <select bind:value={draft.trustLevel}>
87
+ <option value="standard">standard</option>
88
+ <option value="trusted">trusted</option>
89
+ <option value="blocked">blocked</option>
90
+ </select>
91
+ </label>
92
+
93
+ <div class="actions">
94
+ <button type="submit">{t(M['content.contributor_manager.save_contributor'])}</button>
95
+ </div>
96
+ </form>
97
+ </div>
98
+ </section>
99
+
100
+ <style>
101
+ .manager,
102
+ .layout,
103
+ .list,
104
+ .editor {
105
+ display: grid;
106
+ gap: 1rem;
107
+ }
108
+
109
+ .layout {
110
+ grid-template-columns: minmax(14rem, 20rem) minmax(0, 1fr);
111
+ }
112
+
113
+ .card,
114
+ .editor {
115
+ border: 1px solid var(--smrt-color-outline-variant);
116
+ border-radius: 0.75rem;
117
+ padding: 0.9rem;
118
+ background: var(--smrt-color-surface);
119
+ }
120
+
121
+ .card,
122
+ .actions,
123
+ header {
124
+ display: flex;
125
+ justify-content: space-between;
126
+ gap: 0.75rem;
127
+ flex-wrap: wrap;
128
+ }
129
+
130
+ .editor label {
131
+ display: grid;
132
+ gap: 0.35rem;
133
+ }
134
+
135
+ @media (max-width: 720px) {
136
+ .layout {
137
+ grid-template-columns: 1fr;
138
+ }
139
+ }
140
+ </style>
@@ -0,0 +1,10 @@
1
+ import type { ContentContributorData } from '../../mock-smrt-client';
2
+ export interface Props {
3
+ contributors?: ContentContributorData[];
4
+ onSave: (contributor: Partial<ContentContributorData>) => void;
5
+ onDelete?: (contributor: ContentContributorData) => void;
6
+ }
7
+ declare const ContentContributorManager: import("svelte").Component<Props, {}, "">;
8
+ type ContentContributorManager = ReturnType<typeof ContentContributorManager>;
9
+ export default ContentContributorManager;
10
+ //# sourceMappingURL=ContentContributorManager.svelte.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ContentContributorManager.svelte.d.ts","sourceRoot":"","sources":["../../../src/svelte/components/ContentContributorManager.svelte.ts"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,sBAAsB,EAAE,MAAM,wBAAwB,CAAC;AAIrE,MAAM,WAAW,KAAK;IACpB,YAAY,CAAC,EAAE,sBAAsB,EAAE,CAAC;IACxC,MAAM,EAAE,CAAC,WAAW,EAAE,OAAO,CAAC,sBAAsB,CAAC,KAAK,IAAI,CAAC;IAC/D,QAAQ,CAAC,EAAE,CAAC,WAAW,EAAE,sBAAsB,KAAK,IAAI,CAAC;CAC1D;AAgGD,QAAA,MAAM,yBAAyB,2CAAwC,CAAC;AACxE,KAAK,yBAAyB,GAAG,UAAU,CAAC,OAAO,yBAAyB,CAAC,CAAC;AAC9E,eAAe,yBAAyB,CAAC"}