@carlonicora/nextjs-jsonapi 0.0.1 → 1.0.4

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 (303) hide show
  1. package/dist/BlockNoteEditor-VFWG6LXI.js.map +1 -1
  2. package/dist/JsonApiRequest-ZZLSP26T.js.map +1 -1
  3. package/dist/atoms/index.js.map +1 -1
  4. package/dist/chunk-2K3Q24UF.js.map +1 -1
  5. package/dist/chunk-3FBCC4G3.js.map +1 -1
  6. package/dist/chunk-4HCRAOS5.js.map +1 -1
  7. package/dist/chunk-6GKHCVF6.js.map +1 -1
  8. package/dist/chunk-7QVYU63E.js.map +1 -1
  9. package/dist/chunk-A5DDIABK.js.map +1 -1
  10. package/dist/chunk-AWONBQQP.js.map +1 -1
  11. package/dist/chunk-CXQOWQSY.js.map +1 -1
  12. package/dist/chunk-DO2HLAZO.js.map +1 -1
  13. package/dist/chunk-EFJEWLRL.js.map +1 -1
  14. package/dist/chunk-FY4SXJGU.js.map +1 -1
  15. package/dist/chunk-H6FMOA6B.js.map +1 -1
  16. package/dist/chunk-I2REI7OA.js.map +1 -1
  17. package/dist/chunk-IBS6NI7D.js.map +1 -1
  18. package/dist/chunk-J4Q36PMP.js.map +1 -1
  19. package/dist/chunk-JC3WJK65.js.map +1 -1
  20. package/dist/chunk-LXKSUWAV.js.map +1 -1
  21. package/dist/chunk-RAF7PNLG.js.map +1 -1
  22. package/dist/chunk-RUR22SVM.js.map +1 -1
  23. package/dist/chunk-TEGF6ZWG.js.map +1 -1
  24. package/dist/chunk-TMVHSY3Y.js.map +1 -1
  25. package/dist/chunk-V2JJPI7N.js.map +1 -1
  26. package/dist/client/index.js.map +1 -1
  27. package/dist/components/index.js.map +1 -1
  28. package/dist/contexts/index.js.map +1 -1
  29. package/dist/core/index.js.map +1 -1
  30. package/dist/features/index.js.map +1 -1
  31. package/dist/hooks/index.js.map +1 -1
  32. package/dist/index.js.map +1 -1
  33. package/dist/interfaces/index.js.map +1 -1
  34. package/dist/permissions/index.js.map +1 -1
  35. package/dist/request-QFS7NEIE.js.map +1 -1
  36. package/dist/request-ZYY6RI5X.js.map +1 -1
  37. package/dist/roles/index.js.map +1 -1
  38. package/dist/server/index.js.map +1 -1
  39. package/dist/shadcnui/index.js.map +1 -1
  40. package/dist/token-MJMC26ON.js.map +1 -1
  41. package/dist/token-UYE7CV6X.js.map +1 -1
  42. package/dist/utils/index.js.map +1 -1
  43. package/package.json +6 -1
  44. package/src/atoms/index.ts +1 -0
  45. package/src/atoms/recentPagesAtom.ts +10 -0
  46. package/src/client/context/JsonApiContext.ts +61 -0
  47. package/src/client/context/JsonApiProvider.tsx +27 -0
  48. package/src/client/context/index.ts +2 -0
  49. package/src/client/hooks/index.ts +3 -0
  50. package/src/client/hooks/useJsonApiGet.ts +188 -0
  51. package/src/client/hooks/useJsonApiMutation.ts +193 -0
  52. package/src/client/hooks/useRehydration.ts +47 -0
  53. package/src/client/index.ts +11 -0
  54. package/src/client/request.ts +97 -0
  55. package/src/client/token.ts +10 -0
  56. package/src/components/containers/PageContainer.tsx +15 -0
  57. package/src/components/containers/ReactMarkdownContainer.tsx +119 -0
  58. package/src/components/containers/TabsContainer.tsx +93 -0
  59. package/src/components/containers/index.ts +3 -0
  60. package/src/components/contents/AttributeElement.tsx +20 -0
  61. package/src/components/contents/index.ts +1 -0
  62. package/src/components/details/AllowedUsersDetails.tsx +23 -0
  63. package/src/components/details/index.ts +1 -0
  64. package/src/components/editors/BlockNoteDiffInlineContent.tsx +152 -0
  65. package/src/components/editors/BlockNoteEditor.tsx +404 -0
  66. package/src/components/editors/BlockNoteEditorContainer.tsx +13 -0
  67. package/src/components/editors/BlockNoteEditorFormattingToolbar.tsx +38 -0
  68. package/src/components/editors/index.ts +1 -0
  69. package/src/components/errors/ErrorDetails.tsx +41 -0
  70. package/src/components/errors/errorToast.ts +9 -0
  71. package/src/components/errors/index.ts +2 -0
  72. package/src/components/forms/CommonAssociationForm.tsx +162 -0
  73. package/src/components/forms/CommonDeleter.tsx +94 -0
  74. package/src/components/forms/CommonEditorButtons.tsx +30 -0
  75. package/src/components/forms/CommonEditorHeader.tsx +35 -0
  76. package/src/components/forms/CommonEditorTrigger.tsx +26 -0
  77. package/src/components/forms/DatePickerPopover.tsx +219 -0
  78. package/src/components/forms/DateRangeSelector.tsx +110 -0
  79. package/src/components/forms/FileUploader.tsx +324 -0
  80. package/src/components/forms/FormCheckbox.tsx +66 -0
  81. package/src/components/forms/FormContainerGeneric.tsx +39 -0
  82. package/src/components/forms/FormDate.tsx +247 -0
  83. package/src/components/forms/FormDateTime.tsx +231 -0
  84. package/src/components/forms/FormInput.tsx +110 -0
  85. package/src/components/forms/FormPassword.tsx +54 -0
  86. package/src/components/forms/FormPlaceAutocomplete.tsx +286 -0
  87. package/src/components/forms/FormSelect.tsx +72 -0
  88. package/src/components/forms/FormSlider.tsx +51 -0
  89. package/src/components/forms/FormSwitch.tsx +25 -0
  90. package/src/components/forms/FormTextarea.tsx +44 -0
  91. package/src/components/forms/MultiFileUploader.tsx +107 -0
  92. package/src/components/forms/PasswordInput.tsx +47 -0
  93. package/src/components/forms/index.ts +21 -0
  94. package/src/components/index.ts +11 -0
  95. package/src/components/navigations/Breadcrumb.tsx +83 -0
  96. package/src/components/navigations/ContentTitle.tsx +39 -0
  97. package/src/components/navigations/Header.tsx +27 -0
  98. package/src/components/navigations/ModeToggleSwitch.tsx +25 -0
  99. package/src/components/navigations/PageSection.tsx +64 -0
  100. package/src/components/navigations/RecentPagesNavigator.tsx +52 -0
  101. package/src/components/navigations/index.ts +6 -0
  102. package/src/components/pages/PageContainerContentDetails.tsx +76 -0
  103. package/src/components/pages/PageContentContainer.tsx +31 -0
  104. package/src/components/pages/index.ts +2 -0
  105. package/src/components/tables/ContentListTable.tsx +165 -0
  106. package/src/components/tables/ContentTableSearch.tsx +105 -0
  107. package/src/components/tables/cells/cell.component.tsx +18 -0
  108. package/src/components/tables/cells/cell.date.tsx +16 -0
  109. package/src/components/tables/cells/cell.id.tsx +27 -0
  110. package/src/components/tables/cells/cell.link.tsx +18 -0
  111. package/src/components/tables/cells/cell.text.tsx +12 -0
  112. package/src/components/tables/cells/cell.url.tsx +13 -0
  113. package/src/components/tables/cells/index.ts +5 -0
  114. package/src/components/tables/index.ts +3 -0
  115. package/src/contexts/SharedContext.tsx +35 -0
  116. package/src/contexts/index.ts +2 -0
  117. package/src/core/abstracts/AbstractApiData.ts +138 -0
  118. package/src/core/abstracts/AbstractService.ts +263 -0
  119. package/src/core/abstracts/index.ts +2 -0
  120. package/src/core/endpoint/EndpointCreator.ts +97 -0
  121. package/src/core/endpoint/index.ts +1 -0
  122. package/src/core/factories/JsonApiDataFactory.ts +12 -0
  123. package/src/core/factories/RehydrationFactory.ts +30 -0
  124. package/src/core/factories/index.ts +2 -0
  125. package/src/core/fields/FieldSelector.ts +15 -0
  126. package/src/core/fields/index.ts +1 -0
  127. package/src/core/index.ts +20 -0
  128. package/src/core/interfaces/ApiData.ts +8 -0
  129. package/src/core/interfaces/ApiDataInterface.ts +15 -0
  130. package/src/core/interfaces/ApiRequestDataTypeInterface.ts +14 -0
  131. package/src/core/interfaces/ApiResponseInterface.ts +17 -0
  132. package/src/core/interfaces/JsonApiHydratedDataInterface.ts +5 -0
  133. package/src/core/interfaces/index.ts +5 -0
  134. package/src/core/registry/DataClassRegistry.ts +51 -0
  135. package/src/core/registry/ModuleRegistrar.ts +43 -0
  136. package/src/core/registry/ModuleRegistry.ts +64 -0
  137. package/src/core/registry/index.ts +3 -0
  138. package/src/core/utils/index.ts +2 -0
  139. package/src/core/utils/rehydrate.ts +24 -0
  140. package/src/core/utils/translateResponse.ts +125 -0
  141. package/src/features/auth/auth.module.ts +9 -0
  142. package/src/features/auth/config.ts +57 -0
  143. package/src/features/auth/data/auth.interface.ts +31 -0
  144. package/src/features/auth/data/auth.service.ts +159 -0
  145. package/src/features/auth/data/auth.ts +54 -0
  146. package/src/features/auth/data/index.ts +3 -0
  147. package/src/features/auth/index.ts +3 -0
  148. package/src/features/company/company.module.ts +10 -0
  149. package/src/features/company/data/company.fields.ts +6 -0
  150. package/src/features/company/data/company.interface.ts +28 -0
  151. package/src/features/company/data/company.service.ts +73 -0
  152. package/src/features/company/data/company.ts +104 -0
  153. package/src/features/company/data/index.ts +4 -0
  154. package/src/features/company/index.ts +2 -0
  155. package/src/features/content/content.module.ts +20 -0
  156. package/src/features/content/data/content.fields.ts +13 -0
  157. package/src/features/content/data/content.interface.ts +23 -0
  158. package/src/features/content/data/content.service.ts +75 -0
  159. package/src/features/content/data/content.ts +85 -0
  160. package/src/features/content/data/index.ts +4 -0
  161. package/src/features/content/index.ts +2 -0
  162. package/src/features/feature/components/forms/FormFeatures.tsx +149 -0
  163. package/src/features/feature/components/index.ts +1 -0
  164. package/src/features/feature/data/feature.interface.ts +9 -0
  165. package/src/features/feature/data/feature.service.ts +19 -0
  166. package/src/features/feature/data/feature.ts +33 -0
  167. package/src/features/feature/data/index.ts +3 -0
  168. package/src/features/feature/feature.module.ts +10 -0
  169. package/src/features/feature/index.ts +3 -0
  170. package/src/features/index.ts +12 -0
  171. package/src/features/module/data/index.ts +2 -0
  172. package/src/features/module/data/module.interface.ts +12 -0
  173. package/src/features/module/data/module.ts +42 -0
  174. package/src/features/module/index.ts +2 -0
  175. package/src/features/module/module.module.ts +10 -0
  176. package/src/features/notification/data/index.ts +4 -0
  177. package/src/features/notification/data/notification.fields.ts +8 -0
  178. package/src/features/notification/data/notification.interface.ts +14 -0
  179. package/src/features/notification/data/notification.service.ts +34 -0
  180. package/src/features/notification/data/notification.ts +51 -0
  181. package/src/features/notification/index.ts +2 -0
  182. package/src/features/notification/notification.module.ts +10 -0
  183. package/src/features/push/data/index.ts +3 -0
  184. package/src/features/push/data/push.interface.ts +8 -0
  185. package/src/features/push/data/push.service.ts +17 -0
  186. package/src/features/push/data/push.ts +18 -0
  187. package/src/features/push/index.ts +2 -0
  188. package/src/features/push/push.module.ts +10 -0
  189. package/src/features/role/data/index.ts +4 -0
  190. package/src/features/role/data/role.fields.ts +8 -0
  191. package/src/features/role/data/role.interface.ts +16 -0
  192. package/src/features/role/data/role.service.ts +117 -0
  193. package/src/features/role/data/role.ts +62 -0
  194. package/src/features/role/index.ts +2 -0
  195. package/src/features/role/role.module.ts +10 -0
  196. package/src/features/s3/data/index.ts +3 -0
  197. package/src/features/s3/data/s3.interface.ts +11 -0
  198. package/src/features/s3/data/s3.service.ts +30 -0
  199. package/src/features/s3/data/s3.ts +60 -0
  200. package/src/features/s3/index.ts +2 -0
  201. package/src/features/s3/s3.module.ts +10 -0
  202. package/src/features/search/index.ts +1 -0
  203. package/src/features/search/interfaces/index.ts +1 -0
  204. package/src/features/search/interfaces/search.result.interface.ts +3 -0
  205. package/src/features/user/author.module.ts +10 -0
  206. package/src/features/user/components/index.ts +2 -0
  207. package/src/features/user/components/lists/ContributorsList.tsx +41 -0
  208. package/src/features/user/components/lists/index.ts +1 -0
  209. package/src/features/user/components/widgets/UserAvatar.tsx +86 -0
  210. package/src/features/user/components/widgets/index.ts +1 -0
  211. package/src/features/user/contexts/CurrentUserContext.tsx +156 -0
  212. package/src/features/user/contexts/index.ts +1 -0
  213. package/src/features/user/data/index.ts +4 -0
  214. package/src/features/user/data/user.fields.ts +8 -0
  215. package/src/features/user/data/user.interface.ts +41 -0
  216. package/src/features/user/data/user.service.ts +246 -0
  217. package/src/features/user/data/user.ts +162 -0
  218. package/src/features/user/index.ts +4 -0
  219. package/src/features/user/user.module.ts +21 -0
  220. package/src/hooks/TableGeneratorRegistry.ts +53 -0
  221. package/src/hooks/index.ts +33 -0
  222. package/src/hooks/types.ts +35 -0
  223. package/src/hooks/url.rewriter.ts +22 -0
  224. package/src/hooks/useCustomD3Graph.tsx +705 -0
  225. package/src/hooks/useDataListRetriever.ts +349 -0
  226. package/src/hooks/useDebounce.ts +33 -0
  227. package/src/hooks/usePageUrlGenerator.ts +50 -0
  228. package/src/hooks/useTableGenerator.ts +16 -0
  229. package/src/i18n/config.ts +73 -0
  230. package/src/i18n/index.ts +18 -0
  231. package/src/index.ts +16 -0
  232. package/src/interfaces/breadcrumb.item.data.interface.ts +4 -0
  233. package/src/interfaces/d3.link.interface.ts +7 -0
  234. package/src/interfaces/d3.node.interface.ts +12 -0
  235. package/src/interfaces/index.ts +3 -0
  236. package/src/permissions/check.ts +127 -0
  237. package/src/permissions/index.ts +2 -0
  238. package/src/permissions/types.ts +109 -0
  239. package/src/roles/config.ts +46 -0
  240. package/src/roles/index.ts +1 -0
  241. package/src/server/cache.ts +28 -0
  242. package/src/server/index.ts +3 -0
  243. package/src/server/request.ts +113 -0
  244. package/src/server/token.ts +10 -0
  245. package/src/shadcnui/custom/kanban.tsx +1001 -0
  246. package/src/shadcnui/custom/link.tsx +18 -0
  247. package/src/shadcnui/custom/multi-select.tsx +382 -0
  248. package/src/shadcnui/index.ts +49 -0
  249. package/src/shadcnui/ui/accordion.tsx +52 -0
  250. package/src/shadcnui/ui/alert-dialog.tsx +141 -0
  251. package/src/shadcnui/ui/alert.tsx +43 -0
  252. package/src/shadcnui/ui/avatar.tsx +50 -0
  253. package/src/shadcnui/ui/badge.tsx +40 -0
  254. package/src/shadcnui/ui/breadcrumb.tsx +115 -0
  255. package/src/shadcnui/ui/button.tsx +51 -0
  256. package/src/shadcnui/ui/calendar.tsx +73 -0
  257. package/src/shadcnui/ui/card.tsx +43 -0
  258. package/src/shadcnui/ui/carousel.tsx +225 -0
  259. package/src/shadcnui/ui/chart.tsx +320 -0
  260. package/src/shadcnui/ui/checkbox.tsx +29 -0
  261. package/src/shadcnui/ui/collapsible.tsx +11 -0
  262. package/src/shadcnui/ui/command.tsx +155 -0
  263. package/src/shadcnui/ui/context-menu.tsx +179 -0
  264. package/src/shadcnui/ui/dialog.tsx +96 -0
  265. package/src/shadcnui/ui/drawer.tsx +89 -0
  266. package/src/shadcnui/ui/dropdown-menu.tsx +205 -0
  267. package/src/shadcnui/ui/form.tsx +138 -0
  268. package/src/shadcnui/ui/hover-card.tsx +29 -0
  269. package/src/shadcnui/ui/input.tsx +21 -0
  270. package/src/shadcnui/ui/label.tsx +26 -0
  271. package/src/shadcnui/ui/navigation-menu.tsx +168 -0
  272. package/src/shadcnui/ui/popover.tsx +33 -0
  273. package/src/shadcnui/ui/progress.tsx +25 -0
  274. package/src/shadcnui/ui/radio-group.tsx +37 -0
  275. package/src/shadcnui/ui/resizable.tsx +47 -0
  276. package/src/shadcnui/ui/scroll-area.tsx +40 -0
  277. package/src/shadcnui/ui/select.tsx +164 -0
  278. package/src/shadcnui/ui/separator.tsx +28 -0
  279. package/src/shadcnui/ui/sheet.tsx +139 -0
  280. package/src/shadcnui/ui/sidebar.tsx +677 -0
  281. package/src/shadcnui/ui/skeleton.tsx +13 -0
  282. package/src/shadcnui/ui/slider.tsx +25 -0
  283. package/src/shadcnui/ui/sonner.tsx +25 -0
  284. package/src/shadcnui/ui/switch.tsx +31 -0
  285. package/src/shadcnui/ui/table.tsx +120 -0
  286. package/src/shadcnui/ui/tabs.tsx +55 -0
  287. package/src/shadcnui/ui/textarea.tsx +24 -0
  288. package/src/shadcnui/ui/toggle.tsx +39 -0
  289. package/src/shadcnui/ui/tooltip.tsx +61 -0
  290. package/src/unified/JsonApiRequest.ts +325 -0
  291. package/src/unified/index.ts +1 -0
  292. package/src/utils/blocknote-diff.util.ts +815 -0
  293. package/src/utils/blocknote-word-diff-renderer.util.ts +413 -0
  294. package/src/utils/cn.ts +6 -0
  295. package/src/utils/compose-refs.ts +61 -0
  296. package/src/utils/date-formatter.ts +53 -0
  297. package/src/utils/exists.ts +7 -0
  298. package/src/utils/index.ts +15 -0
  299. package/src/utils/schemas/entity.object.schema.ts +8 -0
  300. package/src/utils/schemas/index.ts +2 -0
  301. package/src/utils/schemas/user.object.schema.ts +9 -0
  302. package/src/utils/table-options.ts +67 -0
  303. package/src/utils/use-mobile.tsx +21 -0
@@ -0,0 +1,404 @@
1
+ "use client";
2
+
3
+ import { BlockNoteSchema, defaultInlineContentSpecs, PartialBlock } from "@blocknote/core";
4
+ import { createReactInlineContentSpec, useCreateBlockNote } from "@blocknote/react";
5
+ import { BlockNoteView } from "@blocknote/shadcn";
6
+ import "@blocknote/shadcn/style.css";
7
+ import { CheckIcon, XIcon } from "lucide-react";
8
+ import { useTranslations } from "next-intl";
9
+ import React, { useCallback, useEffect, useMemo, useRef, useState } from "react";
10
+ import { useCurrentUserContext } from "../../contexts";
11
+ import { S3Interface, S3Service, UserInterface } from "../../features";
12
+ import { Button } from "../../shadcnui";
13
+ import { BlockNoteDiffUtil, BlockNoteWordDiffRendererUtil, cn } from "../../utils";
14
+ import { errorToast } from "../errors";
15
+ import { BlockNoteEditorFormattingToolbar } from "./BlockNoteEditorFormattingToolbar";
16
+
17
+ export type BlockNoteEditorProps = {
18
+ id: string;
19
+ type: string;
20
+ initialContent?: PartialBlock[];
21
+ onChange?: (content: any, isEmpty: boolean, hasUnresolvedDiff: boolean) => void;
22
+ size?: "sm" | "md";
23
+ className?: string;
24
+ markdownContent?: string;
25
+ diffContent?: PartialBlock[];
26
+ placeholder?: string;
27
+ bordered?: boolean;
28
+ };
29
+
30
+ const createDiffActionsInlineContentSpec = (
31
+ handleAcceptChange: (diffId: string) => void,
32
+ handleRejectChange: (diffId: string) => void,
33
+ ) => {
34
+ return createReactInlineContentSpec(
35
+ {
36
+ type: "diffActions",
37
+ propSchema: {
38
+ diffIds: {
39
+ default: "",
40
+ },
41
+ },
42
+ content: "none",
43
+ },
44
+ {
45
+ render: (props) => {
46
+ const diffIds = props.inlineContent.props.diffIds;
47
+
48
+ return (
49
+ <span className="diff-actions-container mx-2 inline-flex items-center gap-1 align-middle">
50
+ <Button
51
+ title="Accept change"
52
+ onClick={(e) => {
53
+ e.preventDefault();
54
+ e.stopPropagation();
55
+ diffIds.split(",").forEach((id: string) => handleAcceptChange(id.trim()));
56
+ }}
57
+ >
58
+ <CheckIcon className="h-3 w-3 text-green-600" />
59
+ </Button>
60
+ <Button
61
+ title="Reject change"
62
+ className="mx-2 p-0"
63
+ onClick={(e) => {
64
+ e.preventDefault();
65
+ e.stopPropagation();
66
+ diffIds.split(",").forEach((id: string) => handleRejectChange(id.trim()));
67
+ }}
68
+ >
69
+ <XIcon className="h-3 w-3 text-red-600" />
70
+ </Button>
71
+ </span>
72
+ );
73
+ },
74
+ },
75
+ );
76
+ };
77
+
78
+ export default function BlockNoteEditor({
79
+ id,
80
+ type,
81
+ initialContent,
82
+ onChange,
83
+ size,
84
+ className,
85
+ markdownContent,
86
+ diffContent,
87
+ placeholder,
88
+ bordered,
89
+ }: BlockNoteEditorProps): React.JSX.Element {
90
+ const t = useTranslations();
91
+ const { company } = useCurrentUserContext<UserInterface>();
92
+
93
+ const [acceptedChanges, setAcceptedChanges] = useState<Set<string>>(new Set());
94
+ const [rejectedChanges, setRejectedChanges] = useState<Set<string>>(new Set());
95
+
96
+ const editorRef = useRef<HTMLDivElement>(null);
97
+
98
+ const handleAcceptChange = useCallback((diffId: string) => {
99
+ setAcceptedChanges((prev) => new Set([...prev, diffId]));
100
+ setRejectedChanges((prev) => {
101
+ const newSet = new Set(prev);
102
+ newSet.delete(diffId);
103
+ return newSet;
104
+ });
105
+ }, []);
106
+
107
+ const handleRejectChange = useCallback((diffId: string) => {
108
+ setRejectedChanges((prev) => new Set([...prev, diffId]));
109
+ setAcceptedChanges((prev) => {
110
+ const newSet = new Set(prev);
111
+ newSet.delete(diffId);
112
+ return newSet;
113
+ });
114
+ }, []);
115
+
116
+ const DiffActionsInlineContent = useMemo(
117
+ () => createDiffActionsInlineContentSpec(handleAcceptChange, handleRejectChange),
118
+ [handleAcceptChange, handleRejectChange],
119
+ );
120
+
121
+ const schema = useMemo(
122
+ () =>
123
+ BlockNoteSchema.create({
124
+ inlineContentSpecs: {
125
+ ...defaultInlineContentSpecs,
126
+ diffActions: DiffActionsInlineContent,
127
+ },
128
+ } as any),
129
+ [DiffActionsInlineContent],
130
+ );
131
+
132
+ const uploadImage = useCallback(
133
+ async (file: File): Promise<string> => {
134
+ if (!company) {
135
+ errorToast({
136
+ title: t(`generic.errors.upload`),
137
+ error: t(`generic.errors.upload_description`),
138
+ });
139
+ throw new Error(t(`generic.errors.upload`));
140
+ }
141
+
142
+ const fileType = file.type;
143
+ const key = `companies/${company.id}/${type}/${id}/${file.name}`;
144
+
145
+ const s3: S3Interface = await S3Service.getPreSignedUrl({
146
+ key: key,
147
+ contentType: fileType,
148
+ isPublic: true,
149
+ });
150
+
151
+ await fetch(s3.url, {
152
+ method: "PUT",
153
+ headers: s3.headers,
154
+ body: file,
155
+ });
156
+
157
+ const signedImage: S3Interface = await S3Service.getSignedUrl({
158
+ key: key,
159
+ isPublic: true,
160
+ });
161
+
162
+ return signedImage.url;
163
+ },
164
+ [company, id, t],
165
+ );
166
+
167
+ // Utility: Remove trailing empty blocks for read-only display
168
+ const removeTrailingEmptyBlocks = useCallback(
169
+ (blocks: PartialBlock[]): PartialBlock[] => {
170
+ if (!blocks || blocks.length === 0) return blocks;
171
+
172
+ // Only remove trailing empty blocks in read-only mode
173
+ if (onChange !== undefined) return blocks;
174
+
175
+ const result = [...blocks];
176
+
177
+ // Remove trailing empty paragraph blocks, but keep at least one block
178
+ while (result.length > 1) {
179
+ const lastBlock = result[result.length - 1];
180
+
181
+ // Check if it's an empty paragraph
182
+ const isEmptyParagraph =
183
+ lastBlock.type === "paragraph" &&
184
+ (!lastBlock.content ||
185
+ lastBlock.content.length === 0 ||
186
+ (Array.isArray(lastBlock.content) && lastBlock.content.every((c: any) => !c.text || c.text.trim() === "")));
187
+
188
+ if (isEmptyParagraph) {
189
+ result.pop();
190
+ } else {
191
+ break;
192
+ }
193
+ }
194
+
195
+ return result;
196
+ },
197
+ [onChange],
198
+ );
199
+
200
+ const processedContent = useMemo(() => {
201
+ if (diffContent && initialContent) {
202
+ try {
203
+ const diffResult = BlockNoteDiffUtil.diff(initialContent, diffContent);
204
+ const renderedDiff = BlockNoteWordDiffRendererUtil.renderWordDiffs(
205
+ diffResult.blocks,
206
+ handleAcceptChange,
207
+ handleRejectChange,
208
+ acceptedChanges,
209
+ rejectedChanges,
210
+ );
211
+ return removeTrailingEmptyBlocks(renderedDiff);
212
+ } catch (error) {
213
+ return initialContent && Array.isArray(initialContent) && initialContent.length > 0
214
+ ? removeTrailingEmptyBlocks(initialContent)
215
+ : [];
216
+ }
217
+ }
218
+
219
+ if (!initialContent) {
220
+ return [];
221
+ }
222
+
223
+ if (!Array.isArray(initialContent)) {
224
+ return [];
225
+ }
226
+
227
+ return initialContent.length > 0 ? removeTrailingEmptyBlocks(initialContent) : [];
228
+ }, [
229
+ initialContent,
230
+ diffContent,
231
+ handleAcceptChange,
232
+ handleRejectChange,
233
+ acceptedChanges,
234
+ rejectedChanges,
235
+ removeTrailingEmptyBlocks,
236
+ ]);
237
+
238
+ const validatedInitialContent = useMemo(() => {
239
+ if (processedContent && Array.isArray(processedContent) && processedContent.length > 0) {
240
+ const validatedContent = processedContent.filter((block) => {
241
+ if (!block || typeof block !== "object") return false;
242
+ if (!(block as any).type) return false;
243
+ return true;
244
+ });
245
+ return validatedContent.length > 0 ? (validatedContent as PartialBlock[]) : undefined;
246
+ }
247
+ return undefined;
248
+ }, [processedContent]);
249
+
250
+ const editor = useCreateBlockNote(
251
+ useMemo(
252
+ () => ({
253
+ placeholders: {
254
+ emptyDocument: placeholder || t(`generic.blocknote.placeholder`),
255
+ },
256
+ schema,
257
+ initialContent: validatedInitialContent,
258
+ uploadFile: uploadImage,
259
+ }),
260
+ [placeholder, t, schema, validatedInitialContent, uploadImage],
261
+ ),
262
+ );
263
+
264
+ const handleChange = useCallback(async () => {
265
+ if (!onChange) return;
266
+ const newBlocks = editor.document;
267
+
268
+ const markdownFromBlocks = (await editor.blocksToMarkdownLossy(editor.document)).trim();
269
+
270
+ function hasUnresolvedDiffsRecursive(block: any): boolean {
271
+ if (!block || typeof block !== "object") return false;
272
+ let diffId = undefined;
273
+ if (block.props && block.props.diffId) diffId = block.props.diffId;
274
+ if (!diffId && block.attrs && block.attrs.diffId) diffId = block.attrs.diffId;
275
+ if (diffId && !acceptedChanges.has(diffId) && !rejectedChanges.has(diffId)) {
276
+ return true;
277
+ }
278
+
279
+ if (block.content) {
280
+ const contentArr = Array.isArray(block.content) ? block.content : [block.content];
281
+ for (const inline of contentArr) {
282
+ if (inline && typeof inline === "object") {
283
+ if (inline.type === "diffActions" && inline.props && inline.props.diffIds) {
284
+ const ids =
285
+ typeof inline.props.diffIds === "string" ? inline.props.diffIds.split(",") : inline.props.diffIds;
286
+ for (const id of ids) {
287
+ const trimmed = (id || "").toString().trim();
288
+ if (trimmed && !acceptedChanges.has(trimmed) && !rejectedChanges.has(trimmed)) {
289
+ return true;
290
+ }
291
+ }
292
+ }
293
+ if (inline.props && inline.props.diffId) {
294
+ const diffIdInline = inline.props.diffId;
295
+ if (diffIdInline && !acceptedChanges.has(diffIdInline) && !rejectedChanges.has(diffIdInline)) {
296
+ return true;
297
+ }
298
+ }
299
+ if (inline.children && Array.isArray(inline.children)) {
300
+ for (const child of inline.children) {
301
+ if (hasUnresolvedDiffsRecursive(child)) return true;
302
+ }
303
+ }
304
+ }
305
+ }
306
+ }
307
+ if (Array.isArray(block.children)) {
308
+ for (const child of block.children) {
309
+ if (hasUnresolvedDiffsRecursive(child)) return true;
310
+ }
311
+ }
312
+ return false;
313
+ }
314
+
315
+ let hasUnresolvedDiff = false;
316
+ if (Array.isArray(newBlocks)) {
317
+ hasUnresolvedDiff = newBlocks.some((block: any) => hasUnresolvedDiffsRecursive(block));
318
+ }
319
+
320
+ onChange(newBlocks, !markdownFromBlocks.length, hasUnresolvedDiff);
321
+ }, [editor, onChange, id, acceptedChanges, rejectedChanges]);
322
+
323
+ // Utility: deep equality for arrays of blocks
324
+ const areBlocksEqual = (a: any[], b: any[]): boolean => {
325
+ return JSON.stringify(a) === JSON.stringify(b);
326
+ };
327
+
328
+ // Only initialize from markdownContent once per value, and only if different
329
+ const hasInitializedFromMarkdown = useRef<string | null>(null);
330
+ useEffect(() => {
331
+ const updateContent = async (markdown: string) => {
332
+ const blocks = await editor.tryParseMarkdownToBlocks(markdown);
333
+ if (!areBlocksEqual(blocks, editor.document)) {
334
+ editor.replaceBlocks(editor.document, blocks);
335
+ }
336
+ };
337
+
338
+ if (markdownContent && hasInitializedFromMarkdown.current !== markdownContent) {
339
+ hasInitializedFromMarkdown.current = markdownContent;
340
+ updateContent(markdownContent).then(() => handleChange());
341
+ }
342
+ }, [markdownContent, editor]);
343
+
344
+ // Update editor content when diff content changes, but only if different
345
+ // Prevent unnecessary replaceBlocks calls that reset scroll/cursor.
346
+ const previousContentHashRef = useRef<string | null>(null);
347
+ useEffect(() => {
348
+ if (!processedContent || !editor) return;
349
+ const hash = JSON.stringify(processedContent);
350
+ if (previousContentHashRef.current === hash) return; // no changes
351
+ const currentHash = JSON.stringify(editor.document);
352
+ if (currentHash === hash) {
353
+ previousContentHashRef.current = hash;
354
+ return; // already in sync
355
+ }
356
+ editor.replaceBlocks(editor.document, processedContent as PartialBlock[]);
357
+ previousContentHashRef.current = hash;
358
+ }, [processedContent, editor]);
359
+
360
+ // Handle audio received from whisper transcription
361
+ const handleAudioReceived = useCallback(
362
+ (message: string) => {
363
+ try {
364
+ // Ensure the editor has focus
365
+ const editorElement = editorRef.current?.querySelector('[contenteditable="true"]') as HTMLElement;
366
+ if (editorElement && document.activeElement !== editorElement) {
367
+ editorElement.focus();
368
+ }
369
+
370
+ // Insert the transcribed text at the current cursor position
371
+ editor.insertInlineContent(message);
372
+ } catch (error) {
373
+ console.error("Error inserting transcribed text:", error);
374
+ // Fallback: try to insert at the end of the document
375
+ try {
376
+ const blocks = editor.document;
377
+ if (blocks.length > 0) {
378
+ const lastBlock = blocks[blocks.length - 1];
379
+ editor.setTextCursorPosition(lastBlock.id, "end");
380
+ editor.insertInlineContent(message);
381
+ }
382
+ } catch (fallbackError) {
383
+ console.error("Fallback insertion also failed:", fallbackError);
384
+ }
385
+ }
386
+ },
387
+ [editor],
388
+ );
389
+
390
+ return (
391
+ <div ref={editorRef} className={cn(bordered ? "rounded-md border" : "", "w-full")}>
392
+ <BlockNoteView
393
+ editor={editor}
394
+ onChange={handleChange}
395
+ editable={onChange !== undefined}
396
+ formattingToolbar={false}
397
+ theme="light"
398
+ className={cn(`BlockNoteView ${onChange ? "min-h-96 p-4" : ""}`, className, size === "sm" && "small")}
399
+ >
400
+ <BlockNoteEditorFormattingToolbar />
401
+ </BlockNoteView>
402
+ </div>
403
+ );
404
+ }
@@ -0,0 +1,13 @@
1
+ "use client";
2
+
3
+ import dynamic from "next/dynamic";
4
+ import React from "react";
5
+ import { BlockNoteEditorProps } from "./BlockNoteEditor";
6
+
7
+ const BlockNoteEditor = dynamic(() => import("./BlockNoteEditor"), {
8
+ ssr: false,
9
+ });
10
+
11
+ export const BlockNoteEditorContainer = React.memo(function EditorContainer(props: BlockNoteEditorProps) {
12
+ return <BlockNoteEditor {...props} />;
13
+ });
@@ -0,0 +1,38 @@
1
+ "use client";
2
+
3
+ import {
4
+ BasicTextStyleButton,
5
+ BlockTypeSelect,
6
+ CreateLinkButton,
7
+ FileCaptionButton,
8
+ FileReplaceButton,
9
+ FormattingToolbar,
10
+ FormattingToolbarController,
11
+ TextAlignButton,
12
+ } from "@blocknote/react";
13
+
14
+ export function BlockNoteEditorFormattingToolbar() {
15
+ return (
16
+ <FormattingToolbarController
17
+ formattingToolbar={() => (
18
+ <FormattingToolbar>
19
+ <BlockTypeSelect key={"blockTypeSelect"} />
20
+
21
+ <FileCaptionButton key={"fileCaptionButton"} />
22
+ <FileReplaceButton key={"replaceFileButton"} />
23
+
24
+ <BasicTextStyleButton basicTextStyle={"bold"} key={"boldStyleButton"} />
25
+ <BasicTextStyleButton basicTextStyle={"italic"} key={"italicStyleButton"} />
26
+ <BasicTextStyleButton basicTextStyle={"underline"} key={"underlineStyleButton"} />
27
+ <BasicTextStyleButton basicTextStyle={"strike"} key={"strikeStyleButton"} />
28
+
29
+ <TextAlignButton textAlignment={"left"} key={"textAlignLeftButton"} />
30
+ <TextAlignButton textAlignment={"center"} key={"textAlignCenterButton"} />
31
+ <TextAlignButton textAlignment={"right"} key={"textAlignRightButton"} />
32
+
33
+ <CreateLinkButton key={"createLinkButton"} />
34
+ </FormattingToolbar>
35
+ )}
36
+ />
37
+ );
38
+ }
@@ -0,0 +1 @@
1
+ export * from "./BlockNoteEditorContainer";
@@ -0,0 +1,41 @@
1
+ "use client";
2
+
3
+ import Image from "next/image";
4
+ import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "../../shadcnui";
5
+
6
+ export function ErrorDetails({ title, message, code }: { title?: string; message: string; code: number }) {
7
+ if (code === 403)
8
+ return (
9
+ <div className="w-xl max-w-xl">
10
+ <Card className="w-full">
11
+ <CardHeader>
12
+ <CardTitle className="text-foreground flex flex-col items-center gap-y-4 pb-10 text-4xl">
13
+ <Image src="/phlow-logo.webp" alt="Phlow" width={100} height={100} priority />
14
+ {/* <div>{code}</div> */}
15
+ <div>Unauthorised</div>
16
+ </CardTitle>
17
+ <CardDescription className="text-center text-lg">
18
+ We are sorry, but you are not allowed to access this content.
19
+ </CardDescription>
20
+ </CardHeader>
21
+ <CardContent></CardContent>
22
+ </Card>
23
+ </div>
24
+ );
25
+
26
+ return (
27
+ <div className="w-xl max-w-xl">
28
+ <Card className="w-full">
29
+ <CardHeader>
30
+ <CardTitle className="text-foreground flex flex-col items-center gap-y-4 pb-10 text-center text-4xl">
31
+ <Image src="/phlow-logo.webp" alt="Phlow" width={100} height={100} priority />
32
+ <div>{code}</div>
33
+ <div>{title}</div>
34
+ </CardTitle>
35
+ <CardDescription className="text-center text-lg">{message}</CardDescription>
36
+ </CardHeader>
37
+ <CardContent></CardContent>
38
+ </Card>
39
+ </div>
40
+ );
41
+ }
@@ -0,0 +1,9 @@
1
+ // import { toast } from "@/hooks/use-toast";
2
+
3
+ import { toast } from "sonner";
4
+
5
+ export function errorToast(params: { title?: string; error: any }) {
6
+ toast.error(params?.title ?? "Error", {
7
+ description: params.error instanceof Error ? params.error.message : String(params.error),
8
+ });
9
+ }
@@ -0,0 +1,2 @@
1
+ export * from "./ErrorDetails";
2
+ export * from "./errorToast";
@@ -0,0 +1,162 @@
1
+ "use client";
2
+
3
+ import { useTranslations } from "next-intl";
4
+ import { ReactNode, useCallback, useEffect, useRef, useState } from "react";
5
+ import { toast } from "sonner";
6
+ import { DataListRetriever, useDebounce } from "../../hooks";
7
+ import {
8
+ Button,
9
+ Command,
10
+ CommandDialog,
11
+ CommandEmpty,
12
+ CommandInput,
13
+ CommandList,
14
+ DialogDescription,
15
+ DialogHeader,
16
+ DialogTitle,
17
+ } from "../../shadcnui";
18
+
19
+ type CommonAssociationTriggerProps = {
20
+ sourceType: string;
21
+ destinationType: string;
22
+ hasDestination?: boolean;
23
+ onTrigger: () => void;
24
+ };
25
+
26
+ export function CommonAssociationTrigger({
27
+ sourceType,
28
+ destinationType,
29
+ hasDestination,
30
+ onTrigger,
31
+ }: CommonAssociationTriggerProps) {
32
+ const t = useTranslations();
33
+
34
+ if (hasDestination)
35
+ return (
36
+ <div className="hover:text-accent cursor-pointer" onClick={onTrigger}>
37
+ Join
38
+ </div>
39
+ );
40
+
41
+ return (
42
+ <Button variant={`outline`} size={`sm`} onClick={onTrigger}>
43
+ {t(`generic.association.label`, {
44
+ source: sourceType,
45
+ destination: destinationType,
46
+ })}
47
+ </Button>
48
+ );
49
+ }
50
+
51
+ type CommonAssociationCommandDialogProps = {
52
+ show: boolean;
53
+ setShow: (show: boolean) => void;
54
+ data: DataListRetriever<any>;
55
+ source: string;
56
+ destination: string;
57
+ destinationName: string;
58
+ children: ReactNode;
59
+ };
60
+
61
+ export function CommonAssociationCommandDialog({
62
+ show,
63
+ setShow,
64
+ data,
65
+ source,
66
+ destination,
67
+ destinationName,
68
+ children,
69
+ }: CommonAssociationCommandDialogProps) {
70
+ const t = useTranslations();
71
+
72
+ const searchTermRef = useRef<string>("");
73
+ const [searchTerm, setSearchTerm] = useState<string>("");
74
+
75
+ const refreshList = useCallback(
76
+ async (searchedTerm: string) => {
77
+ if (searchedTerm === searchTermRef.current) return;
78
+ searchTermRef.current = searchedTerm;
79
+ await data.search(searchedTerm);
80
+ },
81
+ [searchTerm, data],
82
+ );
83
+
84
+ const updateSearchTerm = useDebounce(refreshList, 500);
85
+
86
+ useEffect(() => {
87
+ if (show) updateSearchTerm(searchTerm);
88
+ }, [show, searchTerm]);
89
+
90
+ return (
91
+ <CommandDialog open={show} onOpenChange={setShow}>
92
+ <DialogHeader className="flex flex-col items-start p-4 pb-0">
93
+ <DialogTitle>
94
+ {t(`generic.association.label`, {
95
+ source: source,
96
+ destination: destination,
97
+ })}
98
+ </DialogTitle>
99
+ <DialogDescription>
100
+ {t(`generic.association.description`, {
101
+ source: source,
102
+ destination: destination,
103
+ destination_name: destinationName,
104
+ })}
105
+ </DialogDescription>
106
+ </DialogHeader>
107
+ <Command shouldFilter={false} className="p-4">
108
+ <CommandInput
109
+ placeholder={t(`generic.search.placeholder`, { type: source })}
110
+ value={searchTerm}
111
+ onValueChange={setSearchTerm}
112
+ />
113
+ <CommandList className="mt-3 h-auto max-h-96 min-h-96 max-w-full overflow-y-auto overflow-x-hidden">
114
+ <CommandEmpty>{t(`generic.search.no_results`, { type: source })}</CommandEmpty>
115
+ {children}
116
+ </CommandList>
117
+ </Command>
118
+ </CommandDialog>
119
+ );
120
+ }
121
+
122
+ export const triggerAssociationToast = (params: {
123
+ t: any;
124
+ source: string;
125
+ destination: string;
126
+ source_name: string;
127
+ destination_name: string;
128
+ level?: string;
129
+ }) => {
130
+ if (params.level) {
131
+ toast.message(
132
+ params.t(`generic.association.label`, {
133
+ source: params.source,
134
+ destination: params.destination,
135
+ }),
136
+ {
137
+ description: params.t(`generic.association.success_level`, {
138
+ source: params.source,
139
+ destination: params.destination,
140
+ source_name: params.source_name,
141
+ destination_name: params.destination_name,
142
+ level: params.level,
143
+ }),
144
+ },
145
+ );
146
+ } else {
147
+ toast.message(
148
+ params.t(`generic.association.label`, {
149
+ source: params.source,
150
+ destination: params.destination,
151
+ }),
152
+ {
153
+ description: params.t(`generic.association.success`, {
154
+ source: params.source,
155
+ destination: params.destination,
156
+ source_name: params.source_name,
157
+ destination_name: params.destination_name,
158
+ }),
159
+ },
160
+ );
161
+ }
162
+ };