@carlonicora/nextjs-jsonapi 1.68.0 → 1.70.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 (103) hide show
  1. package/dist/{AuthComponent-NwQ_ZXsv.d.mts → AuthComponent-DXe3kPzb.d.mts} +1 -1
  2. package/dist/{AuthComponent-DL1D3y7f.d.ts → AuthComponent-Di8DsZ2I.d.ts} +1 -1
  3. package/dist/{BlockNoteEditor-6FDECIS2.mjs → BlockNoteEditor-6XV2IXLY.mjs} +15 -9
  4. package/dist/BlockNoteEditor-6XV2IXLY.mjs.map +1 -0
  5. package/dist/{BlockNoteEditor-DXHROT4C.js → BlockNoteEditor-NVPUPZXB.js} +25 -19
  6. package/dist/BlockNoteEditor-NVPUPZXB.js.map +1 -0
  7. package/dist/HowToInterface-DtVWAE1s.d.mts +17 -0
  8. package/dist/HowToInterface-NaqSG9sE.d.ts +17 -0
  9. package/dist/{auth.interface-BX_1qZZJ.d.ts → auth.interface-BTco8PWs.d.ts} +1 -1
  10. package/dist/{auth.interface-yeLelxdI.d.mts → auth.interface-C4uJzBec.d.mts} +1 -1
  11. package/dist/billing/index.js +346 -346
  12. package/dist/billing/index.mjs +3 -3
  13. package/dist/{chunk-37KYO2UD.js → chunk-56VU7A4I.js} +172 -18
  14. package/dist/chunk-56VU7A4I.js.map +1 -0
  15. package/dist/{chunk-WOJIRXIP.js → chunk-6ROMPIIP.js} +11 -11
  16. package/dist/{chunk-WOJIRXIP.js.map → chunk-6ROMPIIP.js.map} +1 -1
  17. package/dist/{chunk-IOMDNRX5.mjs → chunk-GZNHBAZF.mjs} +155 -1
  18. package/dist/chunk-GZNHBAZF.mjs.map +1 -0
  19. package/dist/{chunk-H4ZS3R76.mjs → chunk-LQEKQYUJ.mjs} +2569 -1603
  20. package/dist/chunk-LQEKQYUJ.mjs.map +1 -0
  21. package/dist/{chunk-WVTBEVAL.mjs → chunk-WJYWWOTG.mjs} +2 -2
  22. package/dist/{chunk-ELTHSXBI.js → chunk-ZKOLKFAS.js} +1664 -698
  23. package/dist/chunk-ZKOLKFAS.js.map +1 -0
  24. package/dist/client/index.d.mts +5 -6
  25. package/dist/client/index.d.ts +5 -6
  26. package/dist/client/index.js +4 -4
  27. package/dist/client/index.mjs +3 -3
  28. package/dist/components/index.d.mts +83 -10
  29. package/dist/components/index.d.ts +83 -10
  30. package/dist/components/index.js +26 -4
  31. package/dist/components/index.js.map +1 -1
  32. package/dist/components/index.mjs +25 -3
  33. package/dist/{config-D-mqttuF.d.mts → config-Bmr_0qTn.d.mts} +1 -1
  34. package/dist/{config-CyCAWW-d.d.ts → config-n0lfSf27.d.ts} +1 -1
  35. package/dist/contexts/index.d.mts +16 -4
  36. package/dist/contexts/index.d.ts +16 -4
  37. package/dist/contexts/index.js +8 -4
  38. package/dist/contexts/index.js.map +1 -1
  39. package/dist/contexts/index.mjs +7 -3
  40. package/dist/core/index.d.mts +61 -11
  41. package/dist/core/index.d.ts +61 -11
  42. package/dist/core/index.js +10 -2
  43. package/dist/core/index.js.map +1 -1
  44. package/dist/core/index.mjs +9 -1
  45. package/dist/index.d.mts +9 -10
  46. package/dist/index.d.ts +9 -10
  47. package/dist/index.js +11 -3
  48. package/dist/index.js.map +1 -1
  49. package/dist/index.mjs +10 -2
  50. package/dist/{notification.interface-ItBxq2au.d.ts → notification.interface-DYDZENx2.d.ts} +18 -1
  51. package/dist/{notification.interface-C6UcmJqu.d.mts → notification.interface-DrHu_1MM.d.mts} +18 -1
  52. package/dist/{s3.service-N1g0piXD.d.ts → s3.service-DK2KKXbR.d.ts} +2 -3
  53. package/dist/{s3.service-CHOTwfWA.d.mts → s3.service-TsN2unZr.d.mts} +2 -3
  54. package/dist/server/index.d.mts +3 -4
  55. package/dist/server/index.d.ts +3 -4
  56. package/dist/server/index.js +3 -3
  57. package/dist/server/index.mjs +1 -1
  58. package/dist/{useRbacState-CUj0hp8t.d.ts → useRbacState-BYaSdA78.d.ts} +1 -1
  59. package/dist/{useRbacState-Btk1gkQg.d.mts → useRbacState-CQEJ_ysV.d.mts} +1 -1
  60. package/dist/{useSocket-BSUN9s3p.d.ts → useSocket-Cjt_qvkI.d.ts} +1 -1
  61. package/dist/{useSocket-DKI92Fbg.d.mts → useSocket-VAGetcT3.d.mts} +1 -1
  62. package/package.json +1 -1
  63. package/src/components/editors/BlockNoteEditor.tsx +7 -1
  64. package/src/components/forms/FormBlockNote.tsx +6 -0
  65. package/src/components/forms/FormSelect.tsx +3 -0
  66. package/src/components/index.ts +1 -0
  67. package/src/contexts/index.ts +1 -0
  68. package/src/core/index.ts +2 -0
  69. package/src/core/registry/ModuleRegistry.ts +19 -0
  70. package/src/features/how-to/HowToModule.ts +18 -0
  71. package/src/features/how-to/components/containers/HowToCommand.tsx +230 -0
  72. package/src/features/how-to/components/containers/HowToCommandViewer.tsx +76 -0
  73. package/src/features/how-to/components/containers/HowToContainer.tsx +27 -0
  74. package/src/features/how-to/components/containers/HowToListContainer.tsx +17 -0
  75. package/src/features/how-to/components/details/HowToContent.tsx +16 -0
  76. package/src/features/how-to/components/details/HowToDetails.tsx +52 -0
  77. package/src/features/how-to/components/forms/HowToDeleter.tsx +31 -0
  78. package/src/features/how-to/components/forms/HowToEditor.tsx +270 -0
  79. package/src/features/how-to/components/forms/HowToMultiSelector.tsx +152 -0
  80. package/src/features/how-to/components/forms/HowToSelector.tsx +164 -0
  81. package/src/features/how-to/components/index.ts +11 -0
  82. package/src/features/how-to/components/lists/HowToList.tsx +39 -0
  83. package/src/features/how-to/contexts/HowToContext.tsx +101 -0
  84. package/src/features/how-to/data/HowTo.ts +69 -0
  85. package/src/features/how-to/data/HowToFields.ts +10 -0
  86. package/src/features/how-to/data/HowToInterface.ts +11 -0
  87. package/src/features/how-to/data/HowToService.ts +61 -0
  88. package/src/features/how-to/data/index.ts +4 -0
  89. package/src/features/how-to/hooks/useHowToTableStructure.tsx +86 -0
  90. package/src/features/how-to/index.ts +2 -0
  91. package/src/features/how-to/utils/blocknote.ts +108 -0
  92. package/src/features/how-to/utils/index.ts +1 -0
  93. package/dist/BlockNoteEditor-6FDECIS2.mjs.map +0 -1
  94. package/dist/BlockNoteEditor-DXHROT4C.js.map +0 -1
  95. package/dist/breadcrumb.item.data.interface-CgB4_1EE.d.mts +0 -6
  96. package/dist/breadcrumb.item.data.interface-CgB4_1EE.d.ts +0 -6
  97. package/dist/chunk-37KYO2UD.js.map +0 -1
  98. package/dist/chunk-ELTHSXBI.js.map +0 -1
  99. package/dist/chunk-H4ZS3R76.mjs.map +0 -1
  100. package/dist/chunk-IOMDNRX5.mjs.map +0 -1
  101. package/dist/content.interface-8T5-G84c.d.mts +0 -21
  102. package/dist/content.interface-D-xdYxjt.d.ts +0 -21
  103. /package/dist/{chunk-WVTBEVAL.mjs.map → chunk-WJYWWOTG.mjs.map} +0 -0
@@ -0,0 +1,39 @@
1
+ "use client";
2
+
3
+ import { useTranslations } from "next-intl";
4
+ import { ReactNode } from "react";
5
+ import { ContentListTable } from "../../../../components";
6
+ import { Modules } from "../../../../core";
7
+ import { DataListRetriever, useDataListRetriever } from "../../../../hooks";
8
+ import { HowToFields } from "../../data/HowToFields";
9
+ import { HowToInterface } from "../../data/HowToInterface";
10
+ import { HowToService } from "../../data/HowToService";
11
+ import "../../hooks/useHowToTableStructure";
12
+ import HowToEditor from "../forms/HowToEditor";
13
+
14
+ type HowToListProps = {
15
+ fullWidth?: boolean;
16
+ };
17
+
18
+ export default function HowToList({ fullWidth }: HowToListProps) {
19
+ const t = useTranslations();
20
+
21
+ const data: DataListRetriever<HowToInterface> = useDataListRetriever({
22
+ module: Modules.HowTo,
23
+ retriever: (params) => HowToService.findMany(params),
24
+ retrieverParams: {},
25
+ });
26
+
27
+ const functions: ReactNode[] = [<HowToEditor key="create-how-to" />];
28
+
29
+ return (
30
+ <ContentListTable
31
+ data={data}
32
+ fields={[HowToFields.name, HowToFields.pages, HowToFields.updatedAt]}
33
+ tableGeneratorType={Modules.HowTo}
34
+ functions={functions}
35
+ title={t(`entities.howtos`, { count: 2 })}
36
+ fullWidth={fullWidth}
37
+ />
38
+ );
39
+ }
@@ -0,0 +1,101 @@
1
+ "use client";
2
+
3
+ import { JsonApiHydratedDataInterface, Modules, rehydrate } from "../../../core";
4
+ import { BreadcrumbItemData } from "../../../interfaces";
5
+ import { usePageUrlGenerator } from "../../../hooks";
6
+ import { SharedProvider } from "../../../contexts/SharedContext";
7
+ import { HowToInterface } from "../data/HowToInterface";
8
+ import { HowToService } from "../data/HowToService";
9
+ import HowToDeleter from "../components/forms/HowToDeleter";
10
+ import HowToEditor from "../components/forms/HowToEditor";
11
+ import { useTranslations } from "next-intl";
12
+ import { createContext, ReactNode, useContext, useState } from "react";
13
+
14
+ interface HowToContextType {
15
+ howTo: HowToInterface | undefined;
16
+ setHowTo: (value: HowToInterface | undefined) => void;
17
+ reloadHowTo: () => Promise<void>;
18
+ }
19
+
20
+ const HowToContext = createContext<HowToContextType | undefined>(undefined);
21
+
22
+ type HowToProviderProps = {
23
+ children: ReactNode;
24
+ dehydratedHowTo?: JsonApiHydratedDataInterface;
25
+ };
26
+
27
+ export const HowToProvider = ({ children, dehydratedHowTo }: HowToProviderProps) => {
28
+ const generateUrl = usePageUrlGenerator();
29
+ const t = useTranslations();
30
+
31
+ const [howTo, setHowTo] = useState<HowToInterface | undefined>(
32
+ dehydratedHowTo ? rehydrate<HowToInterface>(Modules.HowTo, dehydratedHowTo) : undefined,
33
+ );
34
+
35
+ const reloadHowTo = async () => {
36
+ if (!howTo) return;
37
+
38
+ const freshHowTo = await HowToService.findOne({ id: howTo.id });
39
+ setHowTo(freshHowTo);
40
+ };
41
+
42
+ const breadcrumb = () => {
43
+ const response: BreadcrumbItemData[] = [];
44
+
45
+ response.push({
46
+ name: t(`entities.howtos`, { count: 2 }),
47
+ href: generateUrl({ page: Modules.HowTo }),
48
+ });
49
+
50
+ if (howTo)
51
+ response.push({
52
+ name: howTo.name,
53
+ href: generateUrl({ page: Modules.HowTo, id: howTo.id }),
54
+ });
55
+
56
+ return response;
57
+ };
58
+
59
+ const title = () => {
60
+ const response: any = {
61
+ type: t(`entities.howtos`, { count: howTo ? 1 : 2 }),
62
+ };
63
+
64
+ const functions: ReactNode[] = [];
65
+
66
+ if (howTo) {
67
+ response.element = howTo.name;
68
+
69
+ functions.push(<HowToDeleter key={`HowToDeleter`} howTo={howTo} />);
70
+ functions.push(<HowToEditor key={`HowToEditor`} howTo={howTo} propagateChanges={setHowTo} />);
71
+ } else {
72
+ functions.push(<HowToEditor key={`HowToEditor`} />);
73
+ }
74
+
75
+ if (functions.length > 0) response.functions = functions;
76
+
77
+ return response;
78
+ };
79
+
80
+ return (
81
+ <SharedProvider value={{ breadcrumbs: breadcrumb(), title: title() }}>
82
+ <HowToContext.Provider
83
+ value={{
84
+ howTo: howTo,
85
+ setHowTo: setHowTo,
86
+ reloadHowTo: reloadHowTo,
87
+ }}
88
+ >
89
+ {children}
90
+ </HowToContext.Provider>
91
+ </SharedProvider>
92
+ );
93
+ };
94
+
95
+ export const useHowToContext = (): HowToContextType => {
96
+ const context = useContext(HowToContext);
97
+ if (context === undefined) {
98
+ throw new Error("useHowToContext must be used within a HowToProvider");
99
+ }
100
+ return context;
101
+ };
@@ -0,0 +1,69 @@
1
+ import { JsonApiHydratedDataInterface, Modules } from "../../../core";
2
+ import { Content } from "../../content/data/content";
3
+ import { HowToInput, HowToInterface } from "./HowToInterface";
4
+
5
+ export class HowTo extends Content implements HowToInterface {
6
+ private _description?: any;
7
+ private _pages?: string;
8
+
9
+ /**
10
+ * Parse pages from backend JSON string (handles legacy single string + JSON array)
11
+ */
12
+ static parsePagesFromString(pagesStr?: string): string[] {
13
+ if (!pagesStr) return [];
14
+ try {
15
+ const parsed = JSON.parse(pagesStr);
16
+ return Array.isArray(parsed) ? parsed : [pagesStr];
17
+ } catch {
18
+ // Legacy: treat as single page if not valid JSON
19
+ return pagesStr ? [pagesStr] : [];
20
+ }
21
+ }
22
+
23
+ /**
24
+ * Serialize pages array to JSON string for backend
25
+ */
26
+ static serializePagesToString(pages: string[]): string | undefined {
27
+ const filtered = pages.filter((p) => p.trim());
28
+ return filtered.length > 0 ? JSON.stringify(filtered) : undefined;
29
+ }
30
+
31
+ get description(): any {
32
+ return this._description;
33
+ }
34
+
35
+ get pages(): string | undefined {
36
+ return this._pages;
37
+ }
38
+
39
+ rehydrate(data: JsonApiHydratedDataInterface): this {
40
+ super.rehydrate(data);
41
+
42
+ this._description = data.jsonApi.attributes.description
43
+ ? JSON.parse(data.jsonApi.attributes.description)
44
+ : undefined;
45
+ this._pages = data.jsonApi.attributes.pages;
46
+
47
+ return this;
48
+ }
49
+
50
+ createJsonApi(data: HowToInput) {
51
+ const response: any = {
52
+ data: {
53
+ type: Modules.HowTo.name,
54
+ id: data.id,
55
+ attributes: {},
56
+ meta: {},
57
+ relationships: {},
58
+ },
59
+ included: [],
60
+ };
61
+
62
+ super.addContentInput(response, data);
63
+
64
+ if (data.description !== undefined) response.data.attributes.description = JSON.stringify(data.description);
65
+ if (data.pages !== undefined) response.data.attributes.pages = data.pages;
66
+
67
+ return response;
68
+ }
69
+ }
@@ -0,0 +1,10 @@
1
+ export enum HowToFields {
2
+ howToId = "howToId",
3
+
4
+ name = "name",
5
+ description = "description",
6
+ pages = "pages",
7
+
8
+ createdAt = "createdAt",
9
+ updatedAt = "updatedAt",
10
+ }
@@ -0,0 +1,11 @@
1
+ import { ContentInput, ContentInterface } from "../../content/data/content.interface";
2
+
3
+ export type HowToInput = ContentInput & {
4
+ description?: any;
5
+ pages?: string | undefined | null;
6
+ };
7
+
8
+ export interface HowToInterface extends ContentInterface {
9
+ get description(): any;
10
+ get pages(): string | undefined;
11
+ }
@@ -0,0 +1,61 @@
1
+ import { AbstractService, EndpointCreator, HttpMethod, Modules, NextRef, PreviousRef } from "../../../core";
2
+ import { HowToInput, HowToInterface } from "./HowToInterface";
3
+
4
+ export class HowToService extends AbstractService {
5
+ static async findOne(params: { id: string }): Promise<HowToInterface> {
6
+ return this.callApi<HowToInterface>({
7
+ type: Modules.HowTo,
8
+ method: HttpMethod.GET,
9
+ endpoint: new EndpointCreator({ endpoint: Modules.HowTo, id: params.id }).generate(),
10
+ });
11
+ }
12
+
13
+ static async findMany(
14
+ params: {
15
+ search?: string;
16
+ fetchAll?: boolean;
17
+ next?: NextRef;
18
+ prev?: PreviousRef;
19
+ } = {},
20
+ ): Promise<HowToInterface[]> {
21
+ const endpoint = new EndpointCreator({ endpoint: Modules.HowTo });
22
+
23
+ if (params.fetchAll) endpoint.addAdditionalParam("fetchAll", "true");
24
+ if (params.search) endpoint.addAdditionalParam("search", params.search);
25
+ if (Modules.HowTo.inclusions?.lists?.fields) endpoint.limitToFields(Modules.HowTo.inclusions.lists.fields);
26
+ if (Modules.HowTo.inclusions?.lists?.types) endpoint.limitToType(Modules.HowTo.inclusions.lists.types);
27
+
28
+ return this.callApi({
29
+ type: Modules.HowTo,
30
+ method: HttpMethod.GET,
31
+ endpoint: endpoint.generate(),
32
+ next: params.next,
33
+ });
34
+ }
35
+
36
+ static async create(params: HowToInput): Promise<HowToInterface> {
37
+ return this.callApi({
38
+ type: Modules.HowTo,
39
+ method: HttpMethod.POST,
40
+ endpoint: new EndpointCreator({ endpoint: Modules.HowTo }).generate(),
41
+ input: params,
42
+ });
43
+ }
44
+
45
+ static async update(params: HowToInput): Promise<HowToInterface> {
46
+ return this.callApi({
47
+ type: Modules.HowTo,
48
+ method: HttpMethod.PUT,
49
+ endpoint: new EndpointCreator({ endpoint: Modules.HowTo, id: params.id }).generate(),
50
+ input: params,
51
+ });
52
+ }
53
+
54
+ static async delete(params: { howToId: string }): Promise<void> {
55
+ await this.callApi({
56
+ type: Modules.HowTo,
57
+ method: HttpMethod.DELETE,
58
+ endpoint: new EndpointCreator({ endpoint: Modules.HowTo, id: params.howToId }).generate(),
59
+ });
60
+ }
61
+ }
@@ -0,0 +1,4 @@
1
+ export * from "./HowTo";
2
+ export * from "./HowToFields";
3
+ export * from "./HowToInterface";
4
+ export * from "./HowToService";
@@ -0,0 +1,86 @@
1
+ "use client";
2
+
3
+ import { ColumnDef } from "@tanstack/react-table";
4
+ import { useTranslations } from "next-intl";
5
+ import { useMemo } from "react";
6
+
7
+ import { cellDate, cellId } from "../../../components";
8
+ import { Modules } from "../../../core";
9
+ import { registerTableGenerator, TableContent, usePageUrlGenerator, UseTableStructureHook } from "../../../hooks";
10
+ import { Link } from "../../../shadcnui";
11
+ import { HowToFields } from "../data/HowToFields";
12
+ import { HowToInterface } from "../data/HowToInterface";
13
+
14
+ export const useHowToTableStructure: UseTableStructureHook<HowToInterface, HowToFields> = (params) => {
15
+ const t = useTranslations();
16
+ const generateUrl = usePageUrlGenerator();
17
+
18
+ const tableData = useMemo(() => {
19
+ return params.data.map((howTo: HowToInterface) => {
20
+ const entry: TableContent<HowToInterface> = {
21
+ jsonApiData: howTo,
22
+ };
23
+ entry[HowToFields.howToId] = howTo.id;
24
+ params.fields.forEach((field) => {
25
+ entry[field] = howTo[field as keyof HowToInterface];
26
+ });
27
+ return entry;
28
+ });
29
+ }, [params.data, params.fields]);
30
+
31
+ const fieldColumnMap: Partial<Record<HowToFields, () => any>> = {
32
+ [HowToFields.howToId]: () =>
33
+ cellId({
34
+ name: "howToId",
35
+ checkedIds: params.checkedIds,
36
+ toggleId: params.toggleId,
37
+ }),
38
+ [HowToFields.name]: () => ({
39
+ id: "name",
40
+ accessorKey: "name",
41
+ header: t(`howto.fields.name.label`),
42
+ cell: ({ row }: { row: TableContent<HowToInterface> }) => {
43
+ const howTo: HowToInterface = row.original.jsonApiData;
44
+ return <Link href={generateUrl({ page: Modules.HowTo, id: howTo.id })}>{howTo.name}</Link>;
45
+ },
46
+ enableSorting: false,
47
+ enableHiding: false,
48
+ }),
49
+ [HowToFields.description]: () => ({
50
+ id: "description",
51
+ accessorKey: "description",
52
+ header: t(`howto.fields.description.label`),
53
+ cell: ({ row }: { row: TableContent<HowToInterface> }) => <>{row.getValue("description")}</>,
54
+ enableSorting: false,
55
+ enableHiding: false,
56
+ }),
57
+ [HowToFields.pages]: () => ({
58
+ id: "pages",
59
+ accessorKey: "pages",
60
+ header: t(`howto.fields.pages.label`),
61
+ cell: ({ row }: { row: TableContent<HowToInterface> }) => <>{row.getValue("pages")}</>,
62
+ enableSorting: false,
63
+ enableHiding: false,
64
+ }),
65
+ [HowToFields.createdAt]: () =>
66
+ cellDate({
67
+ name: "createdAt",
68
+ title: t(`common.date.create`),
69
+ }),
70
+ [HowToFields.updatedAt]: () =>
71
+ cellDate({
72
+ name: "updatedAt",
73
+ title: t(`common.date.update`),
74
+ }),
75
+ };
76
+
77
+ const columns = useMemo(() => {
78
+ return params.fields.map((field) => fieldColumnMap[field]?.()).filter((col) => col !== undefined) as ColumnDef<
79
+ TableContent<HowToInterface>
80
+ >[];
81
+ }, [params.fields, fieldColumnMap, t, generateUrl]);
82
+
83
+ return useMemo(() => ({ data: tableData, columns: columns }), [tableData, columns]);
84
+ };
85
+
86
+ registerTableGenerator("howtos", useHowToTableStructure);
@@ -0,0 +1,2 @@
1
+ export * from "./HowToModule";
2
+ export * from "./data";
@@ -0,0 +1,108 @@
1
+ export type TocHeading = {
2
+ id: string;
3
+ text: string;
4
+ level: number;
5
+ };
6
+
7
+ /**
8
+ * Extracts all text content from a BlockNote block recursively
9
+ */
10
+ function extractTextFromContent(content: any): string {
11
+ if (!content) return "";
12
+
13
+ if (typeof content === "string") return content;
14
+
15
+ if (Array.isArray(content)) {
16
+ return content.map(extractTextFromContent).join("");
17
+ }
18
+
19
+ if (content.text) return content.text;
20
+
21
+ if (content.content) {
22
+ return extractTextFromContent(content.content);
23
+ }
24
+
25
+ return "";
26
+ }
27
+
28
+ /**
29
+ * Extracts all text from BlockNote content for word counting
30
+ */
31
+ export function extractAllText(blocks: any[] | undefined): string {
32
+ if (!blocks || !Array.isArray(blocks)) return "";
33
+
34
+ return blocks
35
+ .map((block) => {
36
+ let text = extractTextFromContent(block.content);
37
+
38
+ // Recursively get text from children blocks
39
+ if (block.children && Array.isArray(block.children)) {
40
+ text += " " + extractAllText(block.children);
41
+ }
42
+
43
+ return text;
44
+ })
45
+ .join(" ")
46
+ .trim();
47
+ }
48
+
49
+ /**
50
+ * Calculates estimated reading time based on word count
51
+ * Assumes average reading speed of 200 words per minute
52
+ */
53
+ export function calculateReadingTime(blocks: any[] | undefined): number {
54
+ const text = extractAllText(blocks);
55
+ const wordCount = text.split(/\s+/).filter((word: string) => word.length > 0).length;
56
+ const minutes = Math.ceil(wordCount / 200);
57
+ return Math.max(1, minutes); // Minimum 1 minute
58
+ }
59
+
60
+ /**
61
+ * Generates a URL-friendly slug from text
62
+ */
63
+ function generateSlug(text: string, index: number): string {
64
+ const baseSlug = text
65
+ .toLowerCase()
66
+ .replace(/[^a-z0-9\s-]/g, "")
67
+ .replace(/\s+/g, "-")
68
+ .replace(/-+/g, "-")
69
+ .trim();
70
+
71
+ return baseSlug || `heading-${index}`;
72
+ }
73
+
74
+ /**
75
+ * Extracts headings from BlockNote content for table of contents
76
+ */
77
+ export function extractHeadings(blocks: any[] | undefined): TocHeading[] {
78
+ if (!blocks || !Array.isArray(blocks)) return [];
79
+
80
+ const headings: TocHeading[] = [];
81
+ let headingIndex = 0;
82
+
83
+ function processBlocks(blockArray: any[]) {
84
+ for (const block of blockArray) {
85
+ if (block.type === "heading") {
86
+ const level = block.props?.level || 1;
87
+ const text = extractTextFromContent(block.content);
88
+
89
+ if (text.trim()) {
90
+ headings.push({
91
+ id: generateSlug(text, headingIndex),
92
+ text: text.trim(),
93
+ level,
94
+ });
95
+ headingIndex++;
96
+ }
97
+ }
98
+
99
+ // Process children blocks recursively
100
+ if (block.children && Array.isArray(block.children)) {
101
+ processBlocks(block.children);
102
+ }
103
+ }
104
+ }
105
+
106
+ processBlocks(blocks);
107
+ return headings;
108
+ }
@@ -0,0 +1 @@
1
+ export * from "./blocknote";
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/components/editors/BlockNoteEditor.tsx","../src/components/editors/BlockNoteEditorFormattingToolbar.tsx"],"sourcesContent":["\"use client\";\n\nimport { BlockNoteSchema, defaultInlineContentSpecs, PartialBlock } from \"@blocknote/core\";\nimport { createReactInlineContentSpec, useCreateBlockNote } from \"@blocknote/react\";\nimport { BlockNoteView } from \"@blocknote/shadcn\";\nimport \"@blocknote/shadcn/style.css\";\nimport { CheckIcon, XIcon } from \"lucide-react\";\nimport { useTranslations } from \"next-intl\";\nimport React, { useCallback, useEffect, useMemo, useRef, useState } from \"react\";\nimport { useCurrentUserContext } from \"../../contexts\";\nimport { S3Interface } from \"../../features/s3/data\";\nimport { S3Service } from \"../../features/s3/data/s3.service\";\nimport { UserInterface } from \"../../features/user/data\";\nimport { Button } from \"../../shadcnui\";\nimport { BlockNoteDiffUtil, BlockNoteWordDiffRendererUtil, cn } from \"../../utils\";\nimport { errorToast } from \"../errors\";\nimport { BlockNoteEditorFormattingToolbar } from \"./BlockNoteEditorFormattingToolbar\";\n\nexport type BlockNoteEditorProps = {\n id: string;\n type: string;\n initialContent?: PartialBlock[];\n onChange?: (content: any, isEmpty: boolean, hasUnresolvedDiff: boolean) => void;\n size?: \"sm\" | \"md\";\n className?: string;\n markdownContent?: string;\n diffContent?: PartialBlock[];\n placeholder?: string;\n bordered?: boolean;\n};\n\nconst createDiffActionsInlineContentSpec = (\n handleAcceptChange: (diffId: string) => void,\n handleRejectChange: (diffId: string) => void,\n) => {\n return createReactInlineContentSpec(\n {\n type: \"diffActions\",\n propSchema: {\n diffIds: {\n default: \"\",\n },\n },\n content: \"none\",\n },\n {\n render: (props) => {\n const diffIds = props.inlineContent.props.diffIds;\n\n return (\n <span className=\"diff-actions-container mx-2 inline-flex items-center gap-1 align-middle\">\n <Button\n title=\"Accept change\"\n onClick={(e) => {\n e.preventDefault();\n e.stopPropagation();\n diffIds.split(\",\").forEach((id: string) => handleAcceptChange(id.trim()));\n }}\n >\n <CheckIcon className=\"h-3 w-3 text-green-600\" />\n </Button>\n <Button\n title=\"Reject change\"\n className=\"mx-2 p-0\"\n onClick={(e) => {\n e.preventDefault();\n e.stopPropagation();\n diffIds.split(\",\").forEach((id: string) => handleRejectChange(id.trim()));\n }}\n >\n <XIcon className=\"h-3 w-3 text-red-600\" />\n </Button>\n </span>\n );\n },\n },\n );\n};\n\nexport default function BlockNoteEditor({\n id,\n type,\n initialContent,\n onChange,\n size,\n className,\n markdownContent,\n diffContent,\n placeholder,\n bordered,\n}: BlockNoteEditorProps): React.JSX.Element {\n const t = useTranslations();\n const { company } = useCurrentUserContext<UserInterface>();\n\n const [acceptedChanges, setAcceptedChanges] = useState<Set<string>>(new Set());\n const [rejectedChanges, setRejectedChanges] = useState<Set<string>>(new Set());\n\n const editorRef = useRef<HTMLDivElement>(null);\n\n // Ensure side menu buttons don't trigger form submission\n useEffect(() => {\n if (!editorRef.current) return;\n const setButtonTypes = () => {\n editorRef.current?.querySelectorAll(\".bn-side-menu button\").forEach((btn) => {\n if (!btn.getAttribute(\"type\")) {\n btn.setAttribute(\"type\", \"button\");\n }\n });\n };\n const observer = new MutationObserver(setButtonTypes);\n observer.observe(editorRef.current, { childList: true, subtree: true });\n return () => observer.disconnect();\n }, []);\n\n const handleAcceptChange = useCallback((diffId: string) => {\n setAcceptedChanges((prev) => new Set([...prev, diffId]));\n setRejectedChanges((prev) => {\n const newSet = new Set(prev);\n newSet.delete(diffId);\n return newSet;\n });\n }, []);\n\n const handleRejectChange = useCallback((diffId: string) => {\n setRejectedChanges((prev) => new Set([...prev, diffId]));\n setAcceptedChanges((prev) => {\n const newSet = new Set(prev);\n newSet.delete(diffId);\n return newSet;\n });\n }, []);\n\n const DiffActionsInlineContent = useMemo(\n () => createDiffActionsInlineContentSpec(handleAcceptChange, handleRejectChange),\n [handleAcceptChange, handleRejectChange],\n );\n\n const schema = useMemo(\n () =>\n BlockNoteSchema.create({\n inlineContentSpecs: {\n ...defaultInlineContentSpecs,\n diffActions: DiffActionsInlineContent,\n },\n } as any),\n [DiffActionsInlineContent],\n );\n\n const uploadImage = useCallback(\n async (file: File): Promise<string> => {\n if (!company) {\n errorToast({\n title: t(`common.errors.upload`),\n error: t(`common.errors.upload_description`),\n });\n throw new Error(t(`common.errors.upload`));\n }\n\n const fileType = file.type;\n const key = `companies/${company.id}/${type}/${id}/${file.name}`;\n\n const s3: S3Interface = await S3Service.getPreSignedUrl({\n key: key,\n contentType: fileType,\n isPublic: true,\n });\n\n await fetch(s3.url, {\n method: \"PUT\",\n headers: s3.headers,\n body: file,\n });\n\n const signedImage: S3Interface = await S3Service.getSignedUrl({\n key: key,\n isPublic: true,\n });\n\n return signedImage.url;\n },\n [company, id, t],\n );\n\n // Utility: Remove trailing empty blocks for read-only display\n const removeTrailingEmptyBlocks = useCallback(\n (blocks: PartialBlock[]): PartialBlock[] => {\n if (!blocks || blocks.length === 0) return blocks;\n\n // Only remove trailing empty blocks in read-only mode\n if (onChange !== undefined) return blocks;\n\n const result = [...blocks];\n\n // Remove trailing empty paragraph blocks, but keep at least one block\n while (result.length > 1) {\n const lastBlock = result[result.length - 1];\n\n // Check if it's an empty paragraph\n const isEmptyParagraph =\n lastBlock.type === \"paragraph\" &&\n (!lastBlock.content ||\n lastBlock.content.length === 0 ||\n (Array.isArray(lastBlock.content) && lastBlock.content.every((c: any) => !c.text || c.text.trim() === \"\")));\n\n if (isEmptyParagraph) {\n result.pop();\n } else {\n break;\n }\n }\n\n return result;\n },\n [onChange],\n );\n\n const processedContent = useMemo(() => {\n if (diffContent && initialContent) {\n try {\n const diffResult = BlockNoteDiffUtil.diff(initialContent, diffContent);\n const renderedDiff = BlockNoteWordDiffRendererUtil.renderWordDiffs(\n diffResult.blocks,\n handleAcceptChange,\n handleRejectChange,\n acceptedChanges,\n rejectedChanges,\n );\n return removeTrailingEmptyBlocks(renderedDiff);\n } catch (_error) {\n return initialContent && Array.isArray(initialContent) && initialContent.length > 0\n ? removeTrailingEmptyBlocks(initialContent)\n : [];\n }\n }\n\n if (!initialContent) {\n return [];\n }\n\n if (!Array.isArray(initialContent)) {\n return [];\n }\n\n return initialContent.length > 0 ? removeTrailingEmptyBlocks(initialContent) : [];\n }, [\n initialContent,\n diffContent,\n handleAcceptChange,\n handleRejectChange,\n acceptedChanges,\n rejectedChanges,\n removeTrailingEmptyBlocks,\n ]);\n\n const validatedInitialContent = useMemo(() => {\n if (processedContent && Array.isArray(processedContent) && processedContent.length > 0) {\n const validatedContent = processedContent.filter((block) => {\n if (!block || typeof block !== \"object\") return false;\n if (!(block as any).type) return false;\n return true;\n });\n return validatedContent.length > 0 ? (validatedContent as PartialBlock[]) : undefined;\n }\n return undefined;\n }, [processedContent]);\n\n const editor = useCreateBlockNote(\n useMemo(\n () => ({\n placeholders: {\n emptyDocument: placeholder || t(`common.blocknote.placeholder`),\n },\n schema,\n initialContent: validatedInitialContent,\n uploadFile: uploadImage,\n }),\n [placeholder, t, schema, validatedInitialContent, uploadImage],\n ),\n );\n\n const handleChange = useCallback(async () => {\n if (!onChange) return;\n const newBlocks = editor.document;\n\n const markdownFromBlocks = (await editor.blocksToMarkdownLossy(editor.document)).trim();\n\n function hasUnresolvedDiffsRecursive(block: any): boolean {\n if (!block || typeof block !== \"object\") return false;\n let diffId = undefined;\n if (block.props && block.props.diffId) diffId = block.props.diffId;\n if (!diffId && block.attrs && block.attrs.diffId) diffId = block.attrs.diffId;\n if (diffId && !acceptedChanges.has(diffId) && !rejectedChanges.has(diffId)) {\n return true;\n }\n\n if (block.content) {\n const contentArr = Array.isArray(block.content) ? block.content : [block.content];\n for (const inline of contentArr) {\n if (inline && typeof inline === \"object\") {\n if (inline.type === \"diffActions\" && inline.props && inline.props.diffIds) {\n const ids =\n typeof inline.props.diffIds === \"string\" ? inline.props.diffIds.split(\",\") : inline.props.diffIds;\n for (const id of ids) {\n const trimmed = (id || \"\").toString().trim();\n if (trimmed && !acceptedChanges.has(trimmed) && !rejectedChanges.has(trimmed)) {\n return true;\n }\n }\n }\n if (inline.props && inline.props.diffId) {\n const diffIdInline = inline.props.diffId;\n if (diffIdInline && !acceptedChanges.has(diffIdInline) && !rejectedChanges.has(diffIdInline)) {\n return true;\n }\n }\n if (inline.children && Array.isArray(inline.children)) {\n for (const child of inline.children) {\n if (hasUnresolvedDiffsRecursive(child)) return true;\n }\n }\n }\n }\n }\n if (Array.isArray(block.children)) {\n for (const child of block.children) {\n if (hasUnresolvedDiffsRecursive(child)) return true;\n }\n }\n return false;\n }\n\n let hasUnresolvedDiff = false;\n if (Array.isArray(newBlocks)) {\n hasUnresolvedDiff = newBlocks.some((block: any) => hasUnresolvedDiffsRecursive(block));\n }\n\n onChange(newBlocks, !markdownFromBlocks.length, hasUnresolvedDiff);\n }, [editor, onChange, id, acceptedChanges, rejectedChanges]);\n\n // Utility: deep equality for arrays of blocks\n const areBlocksEqual = (a: any[], b: any[]): boolean => {\n return JSON.stringify(a) === JSON.stringify(b);\n };\n\n // Only initialize from markdownContent once per value, and only if different\n const hasInitializedFromMarkdown = useRef<string | null>(null);\n useEffect(() => {\n const updateContent = async (markdown: string) => {\n const blocks = await editor.tryParseMarkdownToBlocks(markdown);\n if (!areBlocksEqual(blocks, editor.document)) {\n editor.replaceBlocks(editor.document, blocks);\n }\n };\n\n if (markdownContent && hasInitializedFromMarkdown.current !== markdownContent) {\n hasInitializedFromMarkdown.current = markdownContent;\n updateContent(markdownContent).then(() => handleChange());\n }\n }, [markdownContent, editor]);\n\n // Update editor content when diff content changes, but only if different\n // Prevent unnecessary replaceBlocks calls that reset scroll/cursor.\n const previousContentHashRef = useRef<string | null>(null);\n useEffect(() => {\n if (!processedContent || !editor) return;\n const hash = JSON.stringify(processedContent);\n if (previousContentHashRef.current === hash) return; // no changes\n const currentHash = JSON.stringify(editor.document);\n if (currentHash === hash) {\n previousContentHashRef.current = hash;\n return; // already in sync\n }\n editor.replaceBlocks(editor.document, processedContent as PartialBlock[]);\n previousContentHashRef.current = hash;\n }, [processedContent, editor]);\n\n // Handle audio received from whisper transcription\n const _handleAudioReceived = useCallback(\n (message: string) => {\n try {\n // Ensure the editor has focus\n const editorElement = editorRef.current?.querySelector('[contenteditable=\"true\"]') as HTMLElement;\n if (editorElement && document.activeElement !== editorElement) {\n editorElement.focus();\n }\n\n // Insert the transcribed text at the current cursor position\n editor.insertInlineContent(message);\n } catch (error) {\n console.error(\"Error inserting transcribed text:\", error);\n // Fallback: try to insert at the end of the document\n try {\n const blocks = editor.document;\n if (blocks.length > 0) {\n const lastBlock = blocks[blocks.length - 1];\n editor.setTextCursorPosition(lastBlock.id, \"end\");\n editor.insertInlineContent(message);\n }\n } catch (fallbackError) {\n console.error(\"Fallback insertion also failed:\", fallbackError);\n }\n }\n },\n [editor],\n );\n\n return (\n <div\n ref={editorRef}\n className={cn(\n bordered ? \"rounded-md border border-input bg-input/20 dark:bg-input/30\" : \"\",\n \"flex flex-col w-full\",\n className,\n )}\n >\n <BlockNoteView\n editor={editor}\n onChange={handleChange}\n editable={onChange !== undefined}\n formattingToolbar={false}\n theme=\"light\"\n className={cn(`BlockNoteView flex-1 ${onChange ? \"p-4\" : \"\"}`, size === \"sm\" && \"small\")}\n >\n <BlockNoteEditorFormattingToolbar />\n </BlockNoteView>\n </div>\n );\n}\n","\"use client\";\n\nimport {\n BasicTextStyleButton,\n BlockTypeSelect,\n CreateLinkButton,\n FileCaptionButton,\n FileReplaceButton,\n FormattingToolbar,\n FormattingToolbarController,\n TextAlignButton,\n} from \"@blocknote/react\";\n\nexport function BlockNoteEditorFormattingToolbar() {\n return (\n <FormattingToolbarController\n formattingToolbar={() => (\n <FormattingToolbar>\n <BlockTypeSelect key={\"blockTypeSelect\"} />\n\n <FileCaptionButton key={\"fileCaptionButton\"} />\n <FileReplaceButton key={\"replaceFileButton\"} />\n\n <BasicTextStyleButton basicTextStyle={\"bold\"} key={\"boldStyleButton\"} />\n <BasicTextStyleButton basicTextStyle={\"italic\"} key={\"italicStyleButton\"} />\n <BasicTextStyleButton basicTextStyle={\"underline\"} key={\"underlineStyleButton\"} />\n <BasicTextStyleButton basicTextStyle={\"strike\"} key={\"strikeStyleButton\"} />\n\n <TextAlignButton textAlignment={\"left\"} key={\"textAlignLeftButton\"} />\n <TextAlignButton textAlignment={\"center\"} key={\"textAlignCenterButton\"} />\n <TextAlignButton textAlignment={\"right\"} key={\"textAlignRightButton\"} />\n\n <CreateLinkButton key={\"createLinkButton\"} />\n </FormattingToolbar>\n )}\n />\n );\n}\n"],"mappings":"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","names":["jsx","jsxs","id"]}
@@ -1 +0,0 @@
1
- {"version":3,"sources":["/home/runner/work/nextjs-jsonapi/nextjs-jsonapi/dist/BlockNoteEditor-DXHROT4C.js","../src/components/editors/BlockNoteEditor.tsx","../src/components/editors/BlockNoteEditorFormattingToolbar.tsx"],"names":["jsxs","jsx"],"mappings":"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","file":"/home/runner/work/nextjs-jsonapi/nextjs-jsonapi/dist/BlockNoteEditor-DXHROT4C.js","sourcesContent":[null,"\"use client\";\n\nimport { BlockNoteSchema, defaultInlineContentSpecs, PartialBlock } from \"@blocknote/core\";\nimport { createReactInlineContentSpec, useCreateBlockNote } from \"@blocknote/react\";\nimport { BlockNoteView } from \"@blocknote/shadcn\";\nimport \"@blocknote/shadcn/style.css\";\nimport { CheckIcon, XIcon } from \"lucide-react\";\nimport { useTranslations } from \"next-intl\";\nimport React, { useCallback, useEffect, useMemo, useRef, useState } from \"react\";\nimport { useCurrentUserContext } from \"../../contexts\";\nimport { S3Interface } from \"../../features/s3/data\";\nimport { S3Service } from \"../../features/s3/data/s3.service\";\nimport { UserInterface } from \"../../features/user/data\";\nimport { Button } from \"../../shadcnui\";\nimport { BlockNoteDiffUtil, BlockNoteWordDiffRendererUtil, cn } from \"../../utils\";\nimport { errorToast } from \"../errors\";\nimport { BlockNoteEditorFormattingToolbar } from \"./BlockNoteEditorFormattingToolbar\";\n\nexport type BlockNoteEditorProps = {\n id: string;\n type: string;\n initialContent?: PartialBlock[];\n onChange?: (content: any, isEmpty: boolean, hasUnresolvedDiff: boolean) => void;\n size?: \"sm\" | \"md\";\n className?: string;\n markdownContent?: string;\n diffContent?: PartialBlock[];\n placeholder?: string;\n bordered?: boolean;\n};\n\nconst createDiffActionsInlineContentSpec = (\n handleAcceptChange: (diffId: string) => void,\n handleRejectChange: (diffId: string) => void,\n) => {\n return createReactInlineContentSpec(\n {\n type: \"diffActions\",\n propSchema: {\n diffIds: {\n default: \"\",\n },\n },\n content: \"none\",\n },\n {\n render: (props) => {\n const diffIds = props.inlineContent.props.diffIds;\n\n return (\n <span className=\"diff-actions-container mx-2 inline-flex items-center gap-1 align-middle\">\n <Button\n title=\"Accept change\"\n onClick={(e) => {\n e.preventDefault();\n e.stopPropagation();\n diffIds.split(\",\").forEach((id: string) => handleAcceptChange(id.trim()));\n }}\n >\n <CheckIcon className=\"h-3 w-3 text-green-600\" />\n </Button>\n <Button\n title=\"Reject change\"\n className=\"mx-2 p-0\"\n onClick={(e) => {\n e.preventDefault();\n e.stopPropagation();\n diffIds.split(\",\").forEach((id: string) => handleRejectChange(id.trim()));\n }}\n >\n <XIcon className=\"h-3 w-3 text-red-600\" />\n </Button>\n </span>\n );\n },\n },\n );\n};\n\nexport default function BlockNoteEditor({\n id,\n type,\n initialContent,\n onChange,\n size,\n className,\n markdownContent,\n diffContent,\n placeholder,\n bordered,\n}: BlockNoteEditorProps): React.JSX.Element {\n const t = useTranslations();\n const { company } = useCurrentUserContext<UserInterface>();\n\n const [acceptedChanges, setAcceptedChanges] = useState<Set<string>>(new Set());\n const [rejectedChanges, setRejectedChanges] = useState<Set<string>>(new Set());\n\n const editorRef = useRef<HTMLDivElement>(null);\n\n // Ensure side menu buttons don't trigger form submission\n useEffect(() => {\n if (!editorRef.current) return;\n const setButtonTypes = () => {\n editorRef.current?.querySelectorAll(\".bn-side-menu button\").forEach((btn) => {\n if (!btn.getAttribute(\"type\")) {\n btn.setAttribute(\"type\", \"button\");\n }\n });\n };\n const observer = new MutationObserver(setButtonTypes);\n observer.observe(editorRef.current, { childList: true, subtree: true });\n return () => observer.disconnect();\n }, []);\n\n const handleAcceptChange = useCallback((diffId: string) => {\n setAcceptedChanges((prev) => new Set([...prev, diffId]));\n setRejectedChanges((prev) => {\n const newSet = new Set(prev);\n newSet.delete(diffId);\n return newSet;\n });\n }, []);\n\n const handleRejectChange = useCallback((diffId: string) => {\n setRejectedChanges((prev) => new Set([...prev, diffId]));\n setAcceptedChanges((prev) => {\n const newSet = new Set(prev);\n newSet.delete(diffId);\n return newSet;\n });\n }, []);\n\n const DiffActionsInlineContent = useMemo(\n () => createDiffActionsInlineContentSpec(handleAcceptChange, handleRejectChange),\n [handleAcceptChange, handleRejectChange],\n );\n\n const schema = useMemo(\n () =>\n BlockNoteSchema.create({\n inlineContentSpecs: {\n ...defaultInlineContentSpecs,\n diffActions: DiffActionsInlineContent,\n },\n } as any),\n [DiffActionsInlineContent],\n );\n\n const uploadImage = useCallback(\n async (file: File): Promise<string> => {\n if (!company) {\n errorToast({\n title: t(`common.errors.upload`),\n error: t(`common.errors.upload_description`),\n });\n throw new Error(t(`common.errors.upload`));\n }\n\n const fileType = file.type;\n const key = `companies/${company.id}/${type}/${id}/${file.name}`;\n\n const s3: S3Interface = await S3Service.getPreSignedUrl({\n key: key,\n contentType: fileType,\n isPublic: true,\n });\n\n await fetch(s3.url, {\n method: \"PUT\",\n headers: s3.headers,\n body: file,\n });\n\n const signedImage: S3Interface = await S3Service.getSignedUrl({\n key: key,\n isPublic: true,\n });\n\n return signedImage.url;\n },\n [company, id, t],\n );\n\n // Utility: Remove trailing empty blocks for read-only display\n const removeTrailingEmptyBlocks = useCallback(\n (blocks: PartialBlock[]): PartialBlock[] => {\n if (!blocks || blocks.length === 0) return blocks;\n\n // Only remove trailing empty blocks in read-only mode\n if (onChange !== undefined) return blocks;\n\n const result = [...blocks];\n\n // Remove trailing empty paragraph blocks, but keep at least one block\n while (result.length > 1) {\n const lastBlock = result[result.length - 1];\n\n // Check if it's an empty paragraph\n const isEmptyParagraph =\n lastBlock.type === \"paragraph\" &&\n (!lastBlock.content ||\n lastBlock.content.length === 0 ||\n (Array.isArray(lastBlock.content) && lastBlock.content.every((c: any) => !c.text || c.text.trim() === \"\")));\n\n if (isEmptyParagraph) {\n result.pop();\n } else {\n break;\n }\n }\n\n return result;\n },\n [onChange],\n );\n\n const processedContent = useMemo(() => {\n if (diffContent && initialContent) {\n try {\n const diffResult = BlockNoteDiffUtil.diff(initialContent, diffContent);\n const renderedDiff = BlockNoteWordDiffRendererUtil.renderWordDiffs(\n diffResult.blocks,\n handleAcceptChange,\n handleRejectChange,\n acceptedChanges,\n rejectedChanges,\n );\n return removeTrailingEmptyBlocks(renderedDiff);\n } catch (_error) {\n return initialContent && Array.isArray(initialContent) && initialContent.length > 0\n ? removeTrailingEmptyBlocks(initialContent)\n : [];\n }\n }\n\n if (!initialContent) {\n return [];\n }\n\n if (!Array.isArray(initialContent)) {\n return [];\n }\n\n return initialContent.length > 0 ? removeTrailingEmptyBlocks(initialContent) : [];\n }, [\n initialContent,\n diffContent,\n handleAcceptChange,\n handleRejectChange,\n acceptedChanges,\n rejectedChanges,\n removeTrailingEmptyBlocks,\n ]);\n\n const validatedInitialContent = useMemo(() => {\n if (processedContent && Array.isArray(processedContent) && processedContent.length > 0) {\n const validatedContent = processedContent.filter((block) => {\n if (!block || typeof block !== \"object\") return false;\n if (!(block as any).type) return false;\n return true;\n });\n return validatedContent.length > 0 ? (validatedContent as PartialBlock[]) : undefined;\n }\n return undefined;\n }, [processedContent]);\n\n const editor = useCreateBlockNote(\n useMemo(\n () => ({\n placeholders: {\n emptyDocument: placeholder || t(`common.blocknote.placeholder`),\n },\n schema,\n initialContent: validatedInitialContent,\n uploadFile: uploadImage,\n }),\n [placeholder, t, schema, validatedInitialContent, uploadImage],\n ),\n );\n\n const handleChange = useCallback(async () => {\n if (!onChange) return;\n const newBlocks = editor.document;\n\n const markdownFromBlocks = (await editor.blocksToMarkdownLossy(editor.document)).trim();\n\n function hasUnresolvedDiffsRecursive(block: any): boolean {\n if (!block || typeof block !== \"object\") return false;\n let diffId = undefined;\n if (block.props && block.props.diffId) diffId = block.props.diffId;\n if (!diffId && block.attrs && block.attrs.diffId) diffId = block.attrs.diffId;\n if (diffId && !acceptedChanges.has(diffId) && !rejectedChanges.has(diffId)) {\n return true;\n }\n\n if (block.content) {\n const contentArr = Array.isArray(block.content) ? block.content : [block.content];\n for (const inline of contentArr) {\n if (inline && typeof inline === \"object\") {\n if (inline.type === \"diffActions\" && inline.props && inline.props.diffIds) {\n const ids =\n typeof inline.props.diffIds === \"string\" ? inline.props.diffIds.split(\",\") : inline.props.diffIds;\n for (const id of ids) {\n const trimmed = (id || \"\").toString().trim();\n if (trimmed && !acceptedChanges.has(trimmed) && !rejectedChanges.has(trimmed)) {\n return true;\n }\n }\n }\n if (inline.props && inline.props.diffId) {\n const diffIdInline = inline.props.diffId;\n if (diffIdInline && !acceptedChanges.has(diffIdInline) && !rejectedChanges.has(diffIdInline)) {\n return true;\n }\n }\n if (inline.children && Array.isArray(inline.children)) {\n for (const child of inline.children) {\n if (hasUnresolvedDiffsRecursive(child)) return true;\n }\n }\n }\n }\n }\n if (Array.isArray(block.children)) {\n for (const child of block.children) {\n if (hasUnresolvedDiffsRecursive(child)) return true;\n }\n }\n return false;\n }\n\n let hasUnresolvedDiff = false;\n if (Array.isArray(newBlocks)) {\n hasUnresolvedDiff = newBlocks.some((block: any) => hasUnresolvedDiffsRecursive(block));\n }\n\n onChange(newBlocks, !markdownFromBlocks.length, hasUnresolvedDiff);\n }, [editor, onChange, id, acceptedChanges, rejectedChanges]);\n\n // Utility: deep equality for arrays of blocks\n const areBlocksEqual = (a: any[], b: any[]): boolean => {\n return JSON.stringify(a) === JSON.stringify(b);\n };\n\n // Only initialize from markdownContent once per value, and only if different\n const hasInitializedFromMarkdown = useRef<string | null>(null);\n useEffect(() => {\n const updateContent = async (markdown: string) => {\n const blocks = await editor.tryParseMarkdownToBlocks(markdown);\n if (!areBlocksEqual(blocks, editor.document)) {\n editor.replaceBlocks(editor.document, blocks);\n }\n };\n\n if (markdownContent && hasInitializedFromMarkdown.current !== markdownContent) {\n hasInitializedFromMarkdown.current = markdownContent;\n updateContent(markdownContent).then(() => handleChange());\n }\n }, [markdownContent, editor]);\n\n // Update editor content when diff content changes, but only if different\n // Prevent unnecessary replaceBlocks calls that reset scroll/cursor.\n const previousContentHashRef = useRef<string | null>(null);\n useEffect(() => {\n if (!processedContent || !editor) return;\n const hash = JSON.stringify(processedContent);\n if (previousContentHashRef.current === hash) return; // no changes\n const currentHash = JSON.stringify(editor.document);\n if (currentHash === hash) {\n previousContentHashRef.current = hash;\n return; // already in sync\n }\n editor.replaceBlocks(editor.document, processedContent as PartialBlock[]);\n previousContentHashRef.current = hash;\n }, [processedContent, editor]);\n\n // Handle audio received from whisper transcription\n const _handleAudioReceived = useCallback(\n (message: string) => {\n try {\n // Ensure the editor has focus\n const editorElement = editorRef.current?.querySelector('[contenteditable=\"true\"]') as HTMLElement;\n if (editorElement && document.activeElement !== editorElement) {\n editorElement.focus();\n }\n\n // Insert the transcribed text at the current cursor position\n editor.insertInlineContent(message);\n } catch (error) {\n console.error(\"Error inserting transcribed text:\", error);\n // Fallback: try to insert at the end of the document\n try {\n const blocks = editor.document;\n if (blocks.length > 0) {\n const lastBlock = blocks[blocks.length - 1];\n editor.setTextCursorPosition(lastBlock.id, \"end\");\n editor.insertInlineContent(message);\n }\n } catch (fallbackError) {\n console.error(\"Fallback insertion also failed:\", fallbackError);\n }\n }\n },\n [editor],\n );\n\n return (\n <div\n ref={editorRef}\n className={cn(\n bordered ? \"rounded-md border border-input bg-input/20 dark:bg-input/30\" : \"\",\n \"flex flex-col w-full\",\n className,\n )}\n >\n <BlockNoteView\n editor={editor}\n onChange={handleChange}\n editable={onChange !== undefined}\n formattingToolbar={false}\n theme=\"light\"\n className={cn(`BlockNoteView flex-1 ${onChange ? \"p-4\" : \"\"}`, size === \"sm\" && \"small\")}\n >\n <BlockNoteEditorFormattingToolbar />\n </BlockNoteView>\n </div>\n );\n}\n","\"use client\";\n\nimport {\n BasicTextStyleButton,\n BlockTypeSelect,\n CreateLinkButton,\n FileCaptionButton,\n FileReplaceButton,\n FormattingToolbar,\n FormattingToolbarController,\n TextAlignButton,\n} from \"@blocknote/react\";\n\nexport function BlockNoteEditorFormattingToolbar() {\n return (\n <FormattingToolbarController\n formattingToolbar={() => (\n <FormattingToolbar>\n <BlockTypeSelect key={\"blockTypeSelect\"} />\n\n <FileCaptionButton key={\"fileCaptionButton\"} />\n <FileReplaceButton key={\"replaceFileButton\"} />\n\n <BasicTextStyleButton basicTextStyle={\"bold\"} key={\"boldStyleButton\"} />\n <BasicTextStyleButton basicTextStyle={\"italic\"} key={\"italicStyleButton\"} />\n <BasicTextStyleButton basicTextStyle={\"underline\"} key={\"underlineStyleButton\"} />\n <BasicTextStyleButton basicTextStyle={\"strike\"} key={\"strikeStyleButton\"} />\n\n <TextAlignButton textAlignment={\"left\"} key={\"textAlignLeftButton\"} />\n <TextAlignButton textAlignment={\"center\"} key={\"textAlignCenterButton\"} />\n <TextAlignButton textAlignment={\"right\"} key={\"textAlignRightButton\"} />\n\n <CreateLinkButton key={\"createLinkButton\"} />\n </FormattingToolbar>\n )}\n />\n );\n}\n"]}
@@ -1,6 +0,0 @@
1
- type BreadcrumbItemData = {
2
- name: string;
3
- href?: string;
4
- };
5
-
6
- export type { BreadcrumbItemData as B };
@@ -1,6 +0,0 @@
1
- type BreadcrumbItemData = {
2
- name: string;
3
- href?: string;
4
- };
5
-
6
- export type { BreadcrumbItemData as B };