create-nextblock 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (206) hide show
  1. package/bin/create-nextblock.js +997 -0
  2. package/package.json +25 -0
  3. package/scripts/sync-template.js +284 -0
  4. package/templates/nextblock-template/.env.example +37 -0
  5. package/templates/nextblock-template/.swcrc +30 -0
  6. package/templates/nextblock-template/README.md +194 -0
  7. package/templates/nextblock-template/app/(auth-pages)/forgot-password/page.tsx +57 -0
  8. package/templates/nextblock-template/app/(auth-pages)/layout.tsx +9 -0
  9. package/templates/nextblock-template/app/(auth-pages)/post-sign-in/page.tsx +28 -0
  10. package/templates/nextblock-template/app/(auth-pages)/sign-in/page.tsx +67 -0
  11. package/templates/nextblock-template/app/(auth-pages)/sign-up/page.tsx +70 -0
  12. package/templates/nextblock-template/app/ToasterProvider.tsx +17 -0
  13. package/templates/nextblock-template/app/[slug]/PageClientContent.tsx +147 -0
  14. package/templates/nextblock-template/app/[slug]/page.tsx +145 -0
  15. package/templates/nextblock-template/app/[slug]/page.utils.ts +183 -0
  16. package/templates/nextblock-template/app/actions/email.ts +31 -0
  17. package/templates/nextblock-template/app/actions/formActions.ts +65 -0
  18. package/templates/nextblock-template/app/actions/languageActions.ts +130 -0
  19. package/templates/nextblock-template/app/actions/postActions.ts +80 -0
  20. package/templates/nextblock-template/app/actions.ts +146 -0
  21. package/templates/nextblock-template/app/api/process-image/route.ts +210 -0
  22. package/templates/nextblock-template/app/api/revalidate/route.ts +86 -0
  23. package/templates/nextblock-template/app/api/revalidate-log/route.ts +23 -0
  24. package/templates/nextblock-template/app/api/upload/presigned-url/route.ts +106 -0
  25. package/templates/nextblock-template/app/api/upload/proxy/route.ts +84 -0
  26. package/templates/nextblock-template/app/auth/callback/route.ts +58 -0
  27. package/templates/nextblock-template/app/blog/[slug]/PostClientContent.tsx +169 -0
  28. package/templates/nextblock-template/app/blog/[slug]/page.tsx +177 -0
  29. package/templates/nextblock-template/app/blog/[slug]/page.utils.ts +136 -0
  30. package/templates/nextblock-template/app/blog/page.tsx +77 -0
  31. package/templates/nextblock-template/app/cms/CmsClientLayout.tsx +321 -0
  32. package/templates/nextblock-template/app/cms/blocks/actions.ts +434 -0
  33. package/templates/nextblock-template/app/cms/blocks/components/BackgroundSelector.tsx +348 -0
  34. package/templates/nextblock-template/app/cms/blocks/components/BlockEditorArea.tsx +567 -0
  35. package/templates/nextblock-template/app/cms/blocks/components/BlockEditorModal.tsx +98 -0
  36. package/templates/nextblock-template/app/cms/blocks/components/BlockTypeCard.tsx +58 -0
  37. package/templates/nextblock-template/app/cms/blocks/components/BlockTypeSelector.tsx +62 -0
  38. package/templates/nextblock-template/app/cms/blocks/components/ColumnEditor.tsx +276 -0
  39. package/templates/nextblock-template/app/cms/blocks/components/DeleteBlockButtonClient.tsx +47 -0
  40. package/templates/nextblock-template/app/cms/blocks/components/EditableBlock.tsx +182 -0
  41. package/templates/nextblock-template/app/cms/blocks/components/MediaLibraryModal.tsx +120 -0
  42. package/templates/nextblock-template/app/cms/blocks/components/SectionConfigPanel.tsx +133 -0
  43. package/templates/nextblock-template/app/cms/blocks/components/SortableBlockItem.tsx +46 -0
  44. package/templates/nextblock-template/app/cms/blocks/editors/ButtonBlockEditor.tsx +85 -0
  45. package/templates/nextblock-template/app/cms/blocks/editors/FormBlockEditor.tsx +182 -0
  46. package/templates/nextblock-template/app/cms/blocks/editors/HeadingBlockEditor.tsx +111 -0
  47. package/templates/nextblock-template/app/cms/blocks/editors/ImageBlockEditor.tsx +150 -0
  48. package/templates/nextblock-template/app/cms/blocks/editors/PostsGridBlockEditor.tsx +79 -0
  49. package/templates/nextblock-template/app/cms/blocks/editors/SectionBlockEditor.tsx +337 -0
  50. package/templates/nextblock-template/app/cms/blocks/editors/TextBlockEditor.tsx +81 -0
  51. package/templates/nextblock-template/app/cms/blocks/editors/VideoEmbedBlockEditor.tsx +64 -0
  52. package/templates/nextblock-template/app/cms/components/ConfirmationModal.tsx +51 -0
  53. package/templates/nextblock-template/app/cms/components/ContentLanguageSwitcher.tsx +145 -0
  54. package/templates/nextblock-template/app/cms/components/CopyContentFromLanguage.tsx +203 -0
  55. package/templates/nextblock-template/app/cms/components/LanguageFilterSelect.tsx +69 -0
  56. package/templates/nextblock-template/app/cms/dashboard/page.tsx +247 -0
  57. package/templates/nextblock-template/app/cms/layout.tsx +10 -0
  58. package/templates/nextblock-template/app/cms/media/UploadFolderContext.tsx +22 -0
  59. package/templates/nextblock-template/app/cms/media/[id]/edit/page.tsx +80 -0
  60. package/templates/nextblock-template/app/cms/media/actions.ts +577 -0
  61. package/templates/nextblock-template/app/cms/media/components/DeleteMediaButtonClient.tsx +53 -0
  62. package/templates/nextblock-template/app/cms/media/components/FolderNavigator.tsx +273 -0
  63. package/templates/nextblock-template/app/cms/media/components/FolderTree.tsx +122 -0
  64. package/templates/nextblock-template/app/cms/media/components/MediaEditForm.tsx +157 -0
  65. package/templates/nextblock-template/app/cms/media/components/MediaGridClient.tsx +275 -0
  66. package/templates/nextblock-template/app/cms/media/components/MediaImage.tsx +70 -0
  67. package/templates/nextblock-template/app/cms/media/components/MediaPickerDialog.tsx +195 -0
  68. package/templates/nextblock-template/app/cms/media/components/MediaUploadForm.tsx +362 -0
  69. package/templates/nextblock-template/app/cms/media/page.tsx +120 -0
  70. package/templates/nextblock-template/app/cms/navigation/[id]/edit/page.tsx +101 -0
  71. package/templates/nextblock-template/app/cms/navigation/actions.ts +358 -0
  72. package/templates/nextblock-template/app/cms/navigation/components/DeleteNavItemButton.tsx +52 -0
  73. package/templates/nextblock-template/app/cms/navigation/components/NavigationItemForm.tsx +248 -0
  74. package/templates/nextblock-template/app/cms/navigation/components/NavigationLanguageSwitcher.tsx +132 -0
  75. package/templates/nextblock-template/app/cms/navigation/components/NavigationMenuDnd.tsx +701 -0
  76. package/templates/nextblock-template/app/cms/navigation/components/SortableNavItem.tsx +98 -0
  77. package/templates/nextblock-template/app/cms/navigation/new/page.tsx +26 -0
  78. package/templates/nextblock-template/app/cms/navigation/page.tsx +102 -0
  79. package/templates/nextblock-template/app/cms/navigation/utils.ts +51 -0
  80. package/templates/nextblock-template/app/cms/pages/[id]/edit/EditPageClient.tsx +121 -0
  81. package/templates/nextblock-template/app/cms/pages/[id]/edit/page.tsx +79 -0
  82. package/templates/nextblock-template/app/cms/pages/actions.ts +241 -0
  83. package/templates/nextblock-template/app/cms/pages/components/DeletePageButtonClient.tsx +47 -0
  84. package/templates/nextblock-template/app/cms/pages/components/PageForm.tsx +253 -0
  85. package/templates/nextblock-template/app/cms/pages/new/page.tsx +52 -0
  86. package/templates/nextblock-template/app/cms/pages/page.tsx +232 -0
  87. package/templates/nextblock-template/app/cms/posts/[id]/edit/page.tsx +183 -0
  88. package/templates/nextblock-template/app/cms/posts/actions.ts +309 -0
  89. package/templates/nextblock-template/app/cms/posts/components/DeletePostButtonClient.tsx +55 -0
  90. package/templates/nextblock-template/app/cms/posts/components/PostForm.tsx +419 -0
  91. package/templates/nextblock-template/app/cms/posts/new/page.tsx +21 -0
  92. package/templates/nextblock-template/app/cms/posts/page.tsx +192 -0
  93. package/templates/nextblock-template/app/cms/revisions/JsonDiffView.tsx +86 -0
  94. package/templates/nextblock-template/app/cms/revisions/RevisionHistoryButton.tsx +201 -0
  95. package/templates/nextblock-template/app/cms/revisions/actions.ts +84 -0
  96. package/templates/nextblock-template/app/cms/revisions/service.ts +344 -0
  97. package/templates/nextblock-template/app/cms/revisions/utils.ts +127 -0
  98. package/templates/nextblock-template/app/cms/settings/copyright/actions.ts +68 -0
  99. package/templates/nextblock-template/app/cms/settings/copyright/components/CopyrightForm.tsx +78 -0
  100. package/templates/nextblock-template/app/cms/settings/copyright/page.tsx +32 -0
  101. package/templates/nextblock-template/app/cms/settings/extra-translations/actions.ts +117 -0
  102. package/templates/nextblock-template/app/cms/settings/extra-translations/page.tsx +216 -0
  103. package/templates/nextblock-template/app/cms/settings/languages/[id]/edit/page.tsx +77 -0
  104. package/templates/nextblock-template/app/cms/settings/languages/actions.ts +261 -0
  105. package/templates/nextblock-template/app/cms/settings/languages/components/DeleteLanguageButton.tsx +76 -0
  106. package/templates/nextblock-template/app/cms/settings/languages/components/LanguageForm.tsx +167 -0
  107. package/templates/nextblock-template/app/cms/settings/languages/new/page.tsx +34 -0
  108. package/templates/nextblock-template/app/cms/settings/languages/page.tsx +156 -0
  109. package/templates/nextblock-template/app/cms/settings/logos/[id]/edit/page.tsx +19 -0
  110. package/templates/nextblock-template/app/cms/settings/logos/actions.ts +114 -0
  111. package/templates/nextblock-template/app/cms/settings/logos/components/LogoForm.tsx +177 -0
  112. package/templates/nextblock-template/app/cms/settings/logos/new/page.tsx +11 -0
  113. package/templates/nextblock-template/app/cms/settings/logos/page.tsx +118 -0
  114. package/templates/nextblock-template/app/cms/settings/logos/types.ts +8 -0
  115. package/templates/nextblock-template/app/cms/users/[id]/edit/page.tsx +91 -0
  116. package/templates/nextblock-template/app/cms/users/actions.ts +156 -0
  117. package/templates/nextblock-template/app/cms/users/components/DeleteUserButton.tsx +71 -0
  118. package/templates/nextblock-template/app/cms/users/components/UserForm.tsx +138 -0
  119. package/templates/nextblock-template/app/cms/users/page.tsx +183 -0
  120. package/templates/nextblock-template/app/favicon.ico +0 -0
  121. package/templates/nextblock-template/app/globals.css +401 -0
  122. package/templates/nextblock-template/app/layout.tsx +191 -0
  123. package/templates/nextblock-template/app/lib/sitemap-utils.ts +68 -0
  124. package/templates/nextblock-template/app/page.tsx +109 -0
  125. package/templates/nextblock-template/app/providers.tsx +43 -0
  126. package/templates/nextblock-template/app/robots.txt/route.ts +19 -0
  127. package/templates/nextblock-template/app/sitemap.xml/route.ts +63 -0
  128. package/templates/nextblock-template/app/unauthorized/page.tsx +27 -0
  129. package/templates/nextblock-template/backup/backup_2025-06-19.sql +8057 -0
  130. package/templates/nextblock-template/backup/backup_2025-06-20.sql +8159 -0
  131. package/templates/nextblock-template/backup/backup_2025-07-08.sql +8411 -0
  132. package/templates/nextblock-template/backup/backup_2025-07-09.sql +8442 -0
  133. package/templates/nextblock-template/backup/backup_2025-07-10.sql +8442 -0
  134. package/templates/nextblock-template/backup/backup_2025-10-01.sql +8803 -0
  135. package/templates/nextblock-template/backup/backup_2025-10-02.sql +9749 -0
  136. package/templates/nextblock-template/components/BlockRenderer.tsx +119 -0
  137. package/templates/nextblock-template/components/FooterNavigation.tsx +33 -0
  138. package/templates/nextblock-template/components/Header.tsx +42 -0
  139. package/templates/nextblock-template/components/HtmlScriptExecutor.tsx +47 -0
  140. package/templates/nextblock-template/components/LanguageSwitcher.tsx +103 -0
  141. package/templates/nextblock-template/components/ResponsiveNav.tsx +372 -0
  142. package/templates/nextblock-template/components/blocks/PostCardSkeleton.tsx +17 -0
  143. package/templates/nextblock-template/components/blocks/PostsGridBlock.tsx +93 -0
  144. package/templates/nextblock-template/components/blocks/PostsGridClient.tsx +180 -0
  145. package/templates/nextblock-template/components/blocks/renderers/ButtonBlockRenderer.tsx +92 -0
  146. package/templates/nextblock-template/components/blocks/renderers/ClientTextBlockRenderer.tsx +69 -0
  147. package/templates/nextblock-template/components/blocks/renderers/FormBlockRenderer.tsx +98 -0
  148. package/templates/nextblock-template/components/blocks/renderers/HeadingBlockRenderer.tsx +41 -0
  149. package/templates/nextblock-template/components/blocks/renderers/HeroBlockRenderer.tsx +240 -0
  150. package/templates/nextblock-template/components/blocks/renderers/ImageBlockRenderer.tsx +79 -0
  151. package/templates/nextblock-template/components/blocks/renderers/PostsGridBlockRenderer.tsx +33 -0
  152. package/templates/nextblock-template/components/blocks/renderers/SectionBlockRenderer.tsx +189 -0
  153. package/templates/nextblock-template/components/blocks/renderers/TextBlockRenderer.tsx +31 -0
  154. package/templates/nextblock-template/components/blocks/renderers/VideoEmbedBlockRenderer.tsx +59 -0
  155. package/templates/nextblock-template/components/blocks/renderers/inline/AlertWidgetRenderer.tsx +51 -0
  156. package/templates/nextblock-template/components/blocks/renderers/inline/CtaWidgetRenderer.tsx +40 -0
  157. package/templates/nextblock-template/components/blocks/types.ts +8 -0
  158. package/templates/nextblock-template/components/env-var-warning.tsx +33 -0
  159. package/templates/nextblock-template/components/form-message.tsx +26 -0
  160. package/templates/nextblock-template/components/header-auth.tsx +71 -0
  161. package/templates/nextblock-template/components/submit-button.tsx +23 -0
  162. package/templates/nextblock-template/components/theme-switcher.tsx +78 -0
  163. package/templates/nextblock-template/context/AuthContext.tsx +138 -0
  164. package/templates/nextblock-template/context/CurrentContentContext.tsx +42 -0
  165. package/templates/nextblock-template/context/LanguageContext.tsx +206 -0
  166. package/templates/nextblock-template/docs/cms-application-overview.md +56 -0
  167. package/templates/nextblock-template/docs/cms-architecture-overview.md +73 -0
  168. package/templates/nextblock-template/docs/files-structure.md +426 -0
  169. package/templates/nextblock-template/docs/tiptap-bundle-optimization-summary.md +174 -0
  170. package/templates/nextblock-template/eslint.config.mjs +28 -0
  171. package/templates/nextblock-template/index.d.ts +5 -0
  172. package/templates/nextblock-template/lib/blocks/README.md +670 -0
  173. package/templates/nextblock-template/lib/blocks/blockRegistry.ts +1001 -0
  174. package/templates/nextblock-template/lib/ui/ColorPicker.ts +1 -0
  175. package/templates/nextblock-template/lib/ui/ConfirmationDialog.ts +1 -0
  176. package/templates/nextblock-template/lib/ui/CustomSelectWithInput.ts +1 -0
  177. package/templates/nextblock-template/lib/ui/Skeleton.ts +1 -0
  178. package/templates/nextblock-template/lib/ui/avatar.ts +1 -0
  179. package/templates/nextblock-template/lib/ui/badge.ts +1 -0
  180. package/templates/nextblock-template/lib/ui/button.ts +1 -0
  181. package/templates/nextblock-template/lib/ui/card.ts +1 -0
  182. package/templates/nextblock-template/lib/ui/checkbox.ts +1 -0
  183. package/templates/nextblock-template/lib/ui/dialog.ts +1 -0
  184. package/templates/nextblock-template/lib/ui/dropdown-menu.ts +1 -0
  185. package/templates/nextblock-template/lib/ui/input.ts +1 -0
  186. package/templates/nextblock-template/lib/ui/label.ts +1 -0
  187. package/templates/nextblock-template/lib/ui/popover.ts +1 -0
  188. package/templates/nextblock-template/lib/ui/progress.ts +1 -0
  189. package/templates/nextblock-template/lib/ui/select.ts +1 -0
  190. package/templates/nextblock-template/lib/ui/separator.ts +1 -0
  191. package/templates/nextblock-template/lib/ui/table.ts +1 -0
  192. package/templates/nextblock-template/lib/ui/textarea.ts +1 -0
  193. package/templates/nextblock-template/lib/ui/tooltip.ts +1 -0
  194. package/templates/nextblock-template/lib/ui/ui.ts +1 -0
  195. package/templates/nextblock-template/middleware.ts +206 -0
  196. package/templates/nextblock-template/next-env.d.ts +6 -0
  197. package/templates/nextblock-template/next.config.js +99 -0
  198. package/templates/nextblock-template/package.json +52 -0
  199. package/templates/nextblock-template/postcss.config.js +6 -0
  200. package/templates/nextblock-template/project.json +7 -0
  201. package/templates/nextblock-template/public/.gitkeep +0 -0
  202. package/templates/nextblock-template/scripts/backfill-image-meta.ts +149 -0
  203. package/templates/nextblock-template/scripts/backup.js +53 -0
  204. package/templates/nextblock-template/scripts/test-bundle-optimization.js +114 -0
  205. package/templates/nextblock-template/tailwind.config.ts +19 -0
  206. package/templates/nextblock-template/tsconfig.json +62 -0
@@ -0,0 +1,273 @@
1
+ "use client";
2
+
3
+ import React, { useEffect, useMemo, useRef, useState } from "react";
4
+ import { useRouter, useSearchParams } from "next/navigation";
5
+ import { Button, Badge, Input } from "@nextblock-cms/ui";
6
+ import { Folder as FolderIcon, Search as SearchIcon, ChevronLeft, ChevronRight } from "lucide-react";
7
+
8
+ interface FolderNavigatorProps {
9
+ folders: string[];
10
+ basePath: string;
11
+ selectedFolder?: string;
12
+ selectedPrefix?: string;
13
+ counts?: Record<string, number>;
14
+ searchTerm?: string;
15
+ }
16
+
17
+ export default function FolderNavigator({ folders, basePath, selectedFolder, selectedPrefix, counts = {}, searchTerm = "" }: FolderNavigatorProps) {
18
+ const router = useRouter();
19
+ const searchParams = useSearchParams();
20
+
21
+ // Build top-level groups and counts
22
+ const groups = useMemo(() => {
23
+ const map = new Map<string, Set<string>>();
24
+ folders.forEach((f) => {
25
+ const parts = f.replace(/^\/+/, "").split("/").filter(Boolean);
26
+ const top = (parts[0] || "").toLowerCase();
27
+ const child = parts.length >= 2 ? `${top}/${parts[1]}/` : `${top}/`;
28
+ let set = map.get(top);
29
+ if (!set) {
30
+ set = new Set<string>();
31
+ map.set(top, set);
32
+ }
33
+ set.add(child);
34
+ });
35
+ return Array.from(map.entries())
36
+ .map(([name, children]) => ({ name, count: children.size }))
37
+ .sort((a, b) => a.name.localeCompare(b.name));
38
+ }, [folders]);
39
+
40
+ const lastPushedRef = useRef<string | null>(null);
41
+
42
+ const apply = (params: { folder?: string | null; folderPrefix?: string | null }, clearSearch?: boolean) => {
43
+ const current = new URLSearchParams(Array.from(searchParams.entries()));
44
+ if (params.folder !== undefined) {
45
+ if (params.folder) current.set("folder", params.folder); else current.delete("folder");
46
+ }
47
+ if (params.folderPrefix !== undefined) {
48
+ if (params.folderPrefix) current.set("folderPrefix", params.folderPrefix); else current.delete("folderPrefix");
49
+ }
50
+ if (clearSearch) {
51
+ current.delete('q');
52
+ if (query) setQuery("");
53
+ }
54
+ const nextUrl = `${basePath}${current.toString() ? `?${current.toString()}` : ""}`;
55
+ const currentUrl = `${basePath}${searchParams.toString() ? `?${searchParams.toString()}` : ""}`;
56
+ if (nextUrl === currentUrl || nextUrl === lastPushedRef.current) return;
57
+ lastPushedRef.current = nextUrl;
58
+ router.push(nextUrl);
59
+ };
60
+
61
+ const activeGroup = useMemo(() => {
62
+ const firstSeg = (p: string) => p.replace(/^\/+/, '').split('/').filter(Boolean)[0] || '';
63
+ if (selectedPrefix) return firstSeg(selectedPrefix);
64
+ if (selectedFolder) return firstSeg(selectedFolder);
65
+ return '';
66
+ }, [selectedPrefix, selectedFolder]);
67
+
68
+ const topTabs = ["logos", "pages", "posts", "uploads"].filter((t) => groups.find((g) => g.name === t));
69
+
70
+ const getImmediateChildren = (basePrefix: string): string[] => {
71
+ if (!basePrefix) return [];
72
+ const base = basePrefix.replace(/^\/+/, "");
73
+ const set = new Set<string>();
74
+ folders.forEach((f) => {
75
+ const norm = f.replace(/^\/+/, "");
76
+ if (!norm.startsWith(base)) return;
77
+ const rest = norm.slice(base.length);
78
+ if (!rest) return;
79
+ const seg = rest.split("/").filter(Boolean)[0];
80
+ if (!seg) return;
81
+ set.add(`${base}${seg}/`);
82
+ });
83
+ return Array.from(set).sort((a, b) => a.localeCompare(b));
84
+ };
85
+
86
+ // Build all level prefixes from the selected prefix to render rows recursively
87
+ const levelPrefixes = useMemo(() => {
88
+ const arr: string[] = [];
89
+ if (!activeGroup) return arr;
90
+ const top = `${activeGroup}/`;
91
+ arr.push(top);
92
+ const current = (selectedPrefix || selectedFolder || '').replace(/^\/+/, '');
93
+ if (current && current.startsWith(top)) {
94
+ const parts = current.replace(/\/$/, '').split('/').filter(Boolean);
95
+ // parts includes activeGroup as first element; start after it
96
+ for (let i = 1; i < parts.length; i++) {
97
+ const prefix = `${activeGroup}/${parts.slice(1, i + 1).join('/')}/`;
98
+ arr.push(prefix);
99
+ }
100
+ }
101
+ return arr;
102
+ }, [activeGroup, selectedPrefix, selectedFolder]);
103
+
104
+ // --- Search as filter: update query param 'q' to filter server results ---
105
+ const [query, setQuery] = useState(searchTerm);
106
+ useEffect(() => setQuery(searchTerm), [searchTerm]);
107
+ useEffect(() => {
108
+ const handler = setTimeout(() => {
109
+ const current = new URLSearchParams(Array.from(searchParams.entries()));
110
+ const term = query.trim();
111
+ if (term) {
112
+ // Auto-select best matching folder path when searching folders
113
+ // Heuristic scoring so partial matches on the last segment still work well.
114
+ const t = term.toLowerCase();
115
+ const minLen = 2; // avoid over-eager matching for 1-char inputs
116
+ const candidates = folders.filter((f) => f.toLowerCase().includes(t));
117
+
118
+ // Compute a score for each candidate
119
+ let best: { key: string; score: number } | null = null;
120
+ for (const f of candidates) {
121
+ const full = f.toLowerCase();
122
+ const last = f.replace(/\/$/, '').split('/').filter(Boolean).pop()?.toLowerCase() ?? '';
123
+ let score = 0;
124
+ if (last === t) score = 100; // exact last segment
125
+ else if (last.startsWith(t)) score = 85;
126
+ else if (last.includes(t)) score = 70;
127
+ else if (full.includes(t)) score = 50;
128
+ // shorter last segment gets a small boost (more likely precise)
129
+ score -= Math.min(last.length, 20) * 0.2;
130
+ if (!best || score > best.score) best = { key: f, score };
131
+ }
132
+
133
+ // If a strong folder match is found (score threshold) and term length is reasonable,
134
+ // auto-select that folder and clear q so files in it are visible.
135
+ if (best && best.score >= 50 && t.length >= minLen) {
136
+ const norm = best.key.endsWith('/') ? best.key : `${best.key}/`;
137
+ current.set('folderPrefix', norm);
138
+ current.delete('folder');
139
+ current.delete('q');
140
+ } else {
141
+ // Otherwise treat the term as a content filter for files only
142
+ current.set('q', term);
143
+ }
144
+ } else {
145
+ current.delete('q');
146
+ }
147
+ const nextUrl = `${basePath}${current.toString() ? `?${current.toString()}` : ''}`;
148
+ const currentUrl = `${basePath}${searchParams.toString() ? `?${searchParams.toString()}` : ''}`;
149
+ if (nextUrl === currentUrl || nextUrl === lastPushedRef.current) return;
150
+ lastPushedRef.current = nextUrl;
151
+ router.push(nextUrl);
152
+ }, 300);
153
+ return () => clearTimeout(handler);
154
+ }, [query, router, basePath, folders, searchParams]);
155
+
156
+ // Compute prefixes/rows to render
157
+
158
+ const prefixesToRender = useMemo(() => {
159
+ if (activeGroup) return levelPrefixes;
160
+ // When "All" is selected, do not render any subfolder rows
161
+ return [] as string[];
162
+ }, [activeGroup, levelPrefixes]);
163
+
164
+ // RowScroller: keep a row on one line with arrows + drag scroll
165
+ const RowScroller: React.FC<{ children: React.ReactNode; ariaLabel?: string }> = ({ children, ariaLabel }) => {
166
+ const ref = useRef<HTMLDivElement>(null);
167
+ const [canLeft, setCanLeft] = useState(false);
168
+ const [canRight, setCanRight] = useState(false);
169
+ const isDown = useRef(false);
170
+ const startX = useRef(0);
171
+ const startScroll = useRef(0);
172
+
173
+ const refresh = () => {
174
+ const el = ref.current; if (!el) return;
175
+ setCanLeft(el.scrollLeft > 0);
176
+ setCanRight(el.scrollLeft < el.scrollWidth - el.clientWidth - 1);
177
+ };
178
+ useEffect(() => { refresh(); const onResize = () => refresh(); window.addEventListener('resize', onResize); return () => window.removeEventListener('resize', onResize); }, []);
179
+
180
+ const onMD = (e: React.MouseEvent) => { const el = ref.current; if (!el) return; isDown.current = true; startX.current = e.pageX; startScroll.current = el.scrollLeft; el.style.cursor = 'grabbing'; };
181
+ const onMM = (e: React.MouseEvent) => { const el = ref.current; if (!el || !isDown.current) return; el.scrollLeft = startScroll.current - (e.pageX - startX.current); refresh(); };
182
+ const onMU = () => { const el = ref.current; if (!el) return; isDown.current = false; el.style.cursor = ''; };
183
+ const scrollBy = (dx: number) => { const el = ref.current; if (!el) return; el.scrollBy({ left: dx, behavior: 'smooth' }); setTimeout(refresh, 200); };
184
+
185
+ return (
186
+ <div className="relative w-full max-w-full overflow-x-hidden">
187
+ {canLeft && (
188
+ <div className="absolute inset-y-0 left-0 flex items-center pl-1 z-10">
189
+ <Button size="icon" variant="outline" onClick={() => scrollBy(-260)} aria-label="Scroll left"><ChevronLeft className="h-4 w-4" /></Button>
190
+ </div>
191
+ )}
192
+ {canRight && (
193
+ <div className="absolute inset-y-0 right-0 flex items-center pr-1 z-10">
194
+ <Button size="icon" variant="outline" onClick={() => scrollBy(260)} aria-label="Scroll right"><ChevronRight className="h-4 w-4" /></Button>
195
+ </div>
196
+ )}
197
+ <div
198
+ ref={ref}
199
+ aria-label={ariaLabel}
200
+ className={`overflow-hidden w-full ${canLeft ? 'pl-16' : ''} ${canRight ? 'pr-12' : 'pr-4'}`}
201
+ onMouseDown={onMD}
202
+ onMouseMove={onMM}
203
+ onMouseUp={onMU}
204
+ onMouseLeave={onMU}
205
+ onScroll={refresh}
206
+ >
207
+ <div className="flex gap-2 items-center py-1 whitespace-nowrap">
208
+ {children}
209
+ </div>
210
+ </div>
211
+ </div>
212
+ );
213
+ };
214
+
215
+ return (
216
+ <div className="flex flex-col gap-3">
217
+ {/* Top-level tabs + search */}
218
+ <div className="flex items-center gap-2 justify-between">
219
+ <div className="flex-1 min-w-0">
220
+ <RowScroller ariaLabel="Top folders">
221
+ <Button size="sm" variant={!selectedFolder && !selectedPrefix ? "default" : "outline"} onClick={() => apply({ folder: null, folderPrefix: null })}>
222
+ All
223
+ </Button>
224
+ {topTabs.map((t) => {
225
+ const isActive = selectedPrefix === `${t}/` || (!!selectedFolder && selectedFolder.startsWith(`${t}/`));
226
+ const count = counts[`${t}/`] ?? 0;
227
+ return (
228
+ <Button key={t} size="sm" variant={isActive ? "default" : "outline"} onClick={() => apply({ folder: null, folderPrefix: `${t}/` }, true)} className="flex items-center gap-1">
229
+ <FolderIcon className="h-3.5 w-3.5" aria-hidden />
230
+ <span className="capitalize">{t}</span>
231
+ <Badge variant={isActive ? "secondary" : "outline"} className="ml-1 px-1.5 py-1 text-[10px] leading-none">{count}</Badge>
232
+ </Button>
233
+ );
234
+ })}
235
+ </RowScroller>
236
+ </div>
237
+ <div className="relative min-w-[260px]">
238
+ <SearchIcon className="h-4 w-4 absolute left-2 top-1/2 -translate-y-1/2 text-muted-foreground" />
239
+ <Input value={query} onChange={(e) => setQuery(e.target.value)} placeholder="Filter files and folders..." className="pl-8 h-9" />
240
+ </div>
241
+ </div>
242
+
243
+ {/* Recursive rows: for each level prefix, render base + its children */}
244
+ {prefixesToRender.map((prefix) => {
245
+ const children = getImmediateChildren(prefix);
246
+ if (children.length === 0) return null;
247
+ const baseLabel = prefix.replace(/\/$/, '').split('/').pop();
248
+ const isBaseActive = selectedPrefix === prefix;
249
+ return (
250
+ <RowScroller key={`row-${prefix}`} ariaLabel={`Folders under ${baseLabel}`}>
251
+ <Button size="sm" variant={isBaseActive ? "default" : "outline"} onClick={() => apply({ folder: null, folderPrefix: prefix }, true)} className="flex items-center gap-1">
252
+ <FolderIcon className="h-3.5 w-3.5" aria-hidden />
253
+ <span className="capitalize">{baseLabel}</span>
254
+ <Badge variant={isBaseActive ? "secondary" : "outline"} className="ml-1 px-1.5 py-1 text-[10px] leading-none">{counts[prefix] ?? 0}</Badge>
255
+ </Button>
256
+ {children.map((child) => {
257
+ const label = child.replace(/\/$/, '').split('/').pop();
258
+ const isSel = (selectedPrefix === child || selectedFolder === child);
259
+ const c = counts[child] ?? 0;
260
+ return (
261
+ <Button key={child} size="sm" variant={isSel ? "default" : "outline"} onClick={() => apply({ folder: null, folderPrefix: child }, true)} className="flex items-center gap-1" title={child}>
262
+ <FolderIcon className="h-3.5 w-3.5" aria-hidden />
263
+ {label}
264
+ <Badge variant={isSel ? "secondary" : "outline"} className="ml-1 px-1.5 py-1 text-[10px] leading-none">{c}</Badge>
265
+ </Button>
266
+ );
267
+ })}
268
+ </RowScroller>
269
+ );
270
+ })}
271
+ </div>
272
+ );
273
+ }
@@ -0,0 +1,122 @@
1
+ "use client";
2
+
3
+ import React, { useMemo, useState } from "react";
4
+ import { useRouter, useSearchParams } from "next/navigation";
5
+ import { Button } from "@nextblock-cms/ui";
6
+
7
+ interface FolderTreeProps {
8
+ folders: string[];
9
+ basePath: string;
10
+ selectedFolder?: string; // exact folder filter
11
+ selectedPrefix?: string; // prefix filter (e.g., pages/)
12
+ }
13
+
14
+ type Group = {
15
+ name: string; // top-level segment, e.g., 'pages'
16
+ count: number; // number of folders under this group
17
+ children: string[]; // immediate subfolders like 'pages/slug/'
18
+ };
19
+
20
+ export default function FolderTree({ folders, basePath, selectedFolder, selectedPrefix }: FolderTreeProps) {
21
+ const router = useRouter();
22
+ const searchParams = useSearchParams();
23
+ const [expanded, setExpanded] = useState<Record<string, boolean>>({});
24
+
25
+ const groups = useMemo<Group[]>(() => {
26
+ const map = new Map<string, Set<string>>();
27
+ folders.forEach((f) => {
28
+ const parts = f.replace(/^\/+/, '').split('/').filter(Boolean);
29
+ const top = parts[0] || '';
30
+ const childPrefix = parts.length >= 2 ? `${top}/${parts[1]}/` : `${top}/`;
31
+ if (!map.has(top)) {
32
+ map.set(top, new Set<string>());
33
+ }
34
+ const childSet = map.get(top);
35
+ if (childSet) {
36
+ childSet.add(childPrefix);
37
+ }
38
+ });
39
+ return Array.from(map.entries()).map(([name, children]) => ({
40
+ name,
41
+ count: children.size,
42
+ children: Array.from(children).sort((a, b) => a.localeCompare(b)),
43
+ })).sort((a, b) => a.name.localeCompare(b.name));
44
+ }, [folders]);
45
+
46
+ const apply = (params: Record<string, string | undefined>) => {
47
+ const current = new URLSearchParams(Array.from(searchParams.entries()));
48
+ if (params.folder !== undefined) {
49
+ if (params.folder) current.set('folder', params.folder); else current.delete('folder');
50
+ }
51
+ if (params.folderPrefix !== undefined) {
52
+ if (params.folderPrefix) current.set('folderPrefix', params.folderPrefix); else current.delete('folderPrefix');
53
+ }
54
+ const query = current.toString();
55
+ router.push(`${basePath}${query ? `?${query}` : ''}`);
56
+ };
57
+
58
+ const isPrefixActive = (prefix: string) => selectedPrefix === prefix && !selectedFolder;
59
+ const isFolderActive = (folder: string) => selectedFolder === folder;
60
+
61
+ const topLevel = groups.map((g) => `${g.name}/`);
62
+
63
+ return (
64
+ <div className="flex flex-col gap-3">
65
+ <div className="flex flex-wrap gap-2">
66
+ <Button size="sm" variant={!selectedFolder && !selectedPrefix ? 'default' : 'outline'} onClick={() => apply({ folder: undefined, folderPrefix: undefined })}>All</Button>
67
+ {topLevel.map((prefix) => (
68
+ <Button
69
+ key={prefix}
70
+ size="sm"
71
+ variant={isPrefixActive(prefix) ? 'default' : 'outline'}
72
+ onClick={() => apply({ folder: undefined, folderPrefix: prefix })}
73
+ title={`${prefix}*`}
74
+ >
75
+ {prefix}*
76
+ </Button>
77
+ ))}
78
+ </div>
79
+
80
+ {/* Collapsible children (first N by default) */}
81
+ {groups.map((g) => {
82
+ const prefix = `${g.name}/`;
83
+ const open = !!expanded[prefix];
84
+ const children = g.children;
85
+ const limit = 10;
86
+ const visible = open ? children : children.slice(0, limit);
87
+ return (
88
+ <div key={g.name} className="border rounded-md p-2">
89
+ <div className="flex items-center justify-between">
90
+ <div className="flex items-center gap-2">
91
+ <span className="text-sm font-medium">{prefix}</span>
92
+ <span className="text-xs text-muted-foreground">{g.count} subfolder(s)</span>
93
+ </div>
94
+ {g.count > limit && (
95
+ <Button size="sm" variant="outline" onClick={() => setExpanded((e) => ({ ...e, [prefix]: !open }))}>
96
+ {open ? 'Show less' : `Show all ${g.count}`}
97
+ </Button>
98
+ )}
99
+ </div>
100
+ <div className="mt-2 flex flex-wrap gap-2">
101
+ {visible.map((child) => (
102
+ <Button
103
+ key={child}
104
+ size="sm"
105
+ variant={isFolderActive(child) ? 'default' : 'outline'}
106
+ onClick={() => apply({ folder: child, folderPrefix: undefined })}
107
+ title={child}
108
+ >
109
+ {child}
110
+ </Button>
111
+ ))}
112
+ {visible.length === 0 && (
113
+ <span className="text-xs text-muted-foreground">No subfolders yet</span>
114
+ )}
115
+ </div>
116
+ </div>
117
+ );
118
+ })}
119
+ </div>
120
+ );
121
+ }
122
+
@@ -0,0 +1,157 @@
1
+ // app/cms/media/components/MediaEditForm.tsx
2
+ "use client";
3
+
4
+ import React, { useState, useTransition, useEffect } from 'react';
5
+ import { useRouter, useSearchParams } from 'next/navigation';
6
+ import Image from 'next/image';
7
+ import { Button } from '@nextblock-cms/ui';
8
+ import { Input } from '@nextblock-cms/ui';
9
+ import { Label } from '@nextblock-cms/ui';
10
+ import { Textarea } from '@nextblock-cms/ui';
11
+ import type { Database } from '@nextblock-cms/db';
12
+ import { useAuth } from '@/context/AuthContext';
13
+
14
+ type Media = Database['public']['Tables']['media']['Row'];
15
+ import { FileText } from 'lucide-react';
16
+
17
+ interface MediaEditFormProps {
18
+ mediaItem: Media;
19
+ // The formAction will be updateMediaItem bound with the mediaItem.id
20
+ formAction: (formData: FormData) => Promise<{ error?: string; success?: boolean; media?: Media } | void>;
21
+ }
22
+
23
+ const R2_BASE_URL = process.env.NEXT_PUBLIC_R2_BASE_URL || "";
24
+
25
+ export default function MediaEditForm({ mediaItem, formAction }: MediaEditFormProps) {
26
+ const router = useRouter();
27
+ const searchParams = useSearchParams();
28
+ const [isPending, startTransition] = useTransition();
29
+ const { user, isLoading: authLoading, isAdmin, isWriter } = useAuth();
30
+
31
+ const [fileName, setFileName] = useState(mediaItem.file_name);
32
+ const [description, setDescription] = useState(mediaItem.description || "");
33
+ const [formMessage, setFormMessage] = useState<{ type: 'success' | 'error', text: string } | null>(null);
34
+
35
+ useEffect(() => {
36
+ const successMessage = searchParams.get('success');
37
+ const errorMessage = searchParams.get('error');
38
+ if (successMessage) {
39
+ setFormMessage({ type: 'success', text: successMessage });
40
+ } else if (errorMessage) {
41
+ setFormMessage({ type: 'error', text: errorMessage });
42
+ }
43
+ }, [searchParams]);
44
+
45
+ const handleSubmit = async (event: React.FormEvent<HTMLFormElement>) => {
46
+ event.preventDefault();
47
+ setFormMessage(null);
48
+ const formData = new FormData(event.currentTarget);
49
+ // Ensure current values are on the formData if not explicitly set by controlled inputs
50
+ formData.set('file_name', fileName);
51
+ formData.set('description', description);
52
+
53
+
54
+ startTransition(async () => {
55
+ const result = await formAction(formData);
56
+ if (result?.error) {
57
+ setFormMessage({ type: 'error', text: result.error });
58
+ } else if (result?.success) {
59
+ setFormMessage({ type: 'success', text: "Media item updated successfully!" });
60
+ // Optionally, update local state if the server returns the updated media item
61
+ if (result.media) {
62
+ setFileName(result.media.file_name);
63
+ setDescription(result.media.description || "");
64
+ }
65
+ router.refresh(); // Refresh server components on the page
66
+ }
67
+ });
68
+ };
69
+
70
+ if (authLoading) {
71
+ return <div>Loading form...</div>;
72
+ }
73
+ if (!user || (!isAdmin && !isWriter)) {
74
+ return <div>Access Denied. You do not have permission to edit media.</div>;
75
+ }
76
+
77
+ return (
78
+ <div className="grid grid-cols-1 md:grid-cols-3 gap-6">
79
+ <div className="md:col-span-1 space-y-4">
80
+ <h2 className="text-lg font-semibold">Media Preview</h2>
81
+ {mediaItem.file_type?.startsWith("image/") ? (
82
+ <Image
83
+ src={`${R2_BASE_URL}/${mediaItem.object_key}`}
84
+ alt={description || fileName}
85
+ width={400}
86
+ height={400}
87
+ className="rounded-lg border object-contain aspect-square w-full max-w-sm mx-auto"
88
+ />
89
+ ) : (
90
+ <div className="aspect-square w-full max-w-sm mx-auto bg-muted rounded-lg flex flex-col items-center justify-center p-4 text-center">
91
+ <FileText className="h-16 w-16 text-muted-foreground mb-2" /> {/* Using FileText as a generic icon */}
92
+ <p className="text-sm text-muted-foreground">No preview available for this file type.</p>
93
+ <p className="text-xs text-muted-foreground mt-1">({mediaItem.file_type})</p>
94
+ </div>
95
+ )}
96
+ <div className="text-xs text-muted-foreground space-y-1">
97
+ <p><strong>Object Key:</strong> <span className="font-mono break-all">{mediaItem.object_key}</span></p>
98
+ <p><strong>File Type:</strong> {mediaItem.file_type}</p>
99
+ <p><strong>Size:</strong> {typeof mediaItem.size_bytes === 'number' ? (mediaItem.size_bytes / 1024 / 1024).toFixed(2) + ' MB' : 'Unknown'}</p>
100
+ <p><strong>Uploaded:</strong> {new Date(mediaItem.created_at).toLocaleString()}</p>
101
+ </div>
102
+ </div>
103
+
104
+ <form onSubmit={handleSubmit} className="md:col-span-2 space-y-6">
105
+ {formMessage && (
106
+ <div
107
+ className={`p-3 rounded-md text-sm ${
108
+ formMessage.type === 'success'
109
+ ? 'bg-green-100 text-green-700 border border-green-200'
110
+ : 'bg-red-100 text-red-700 border border-red-200'
111
+ }`}
112
+ >
113
+ {formMessage.text}
114
+ </div>
115
+ )}
116
+ <div>
117
+ <Label htmlFor="file_name">Display Name</Label>
118
+ <Input
119
+ id="file_name"
120
+ name="file_name"
121
+ value={fileName}
122
+ onChange={(e) => setFileName(e.target.value)}
123
+ required
124
+ className="mt-1"
125
+ />
126
+ </div>
127
+
128
+ <div>
129
+ <Label htmlFor="description">Description (Alt Text for Images)</Label>
130
+ <Textarea
131
+ id="description"
132
+ name="description"
133
+ value={description}
134
+ onChange={(e) => setDescription(e.target.value)}
135
+ className="mt-1"
136
+ rows={4}
137
+ placeholder="e.g., A vibrant sunset over a mountain range"
138
+ />
139
+ </div>
140
+
141
+ <div className="flex justify-end space-x-3 pt-4">
142
+ <Button
143
+ type="button"
144
+ variant="outline"
145
+ onClick={() => router.push("/cms/media")}
146
+ disabled={isPending}
147
+ >
148
+ Cancel
149
+ </Button>
150
+ <Button type="submit" disabled={isPending || authLoading}>
151
+ {isPending ? "Saving..." : "Update Media Info"}
152
+ </Button>
153
+ </div>
154
+ </form>
155
+ </div>
156
+ );
157
+ }