create-nextjs-cms 0.7.0 → 0.7.2

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 (184) hide show
  1. package/LICENSE +21 -21
  2. package/README.md +71 -71
  3. package/dist/helpers/utils.js +16 -16
  4. package/dist/lib/section-creators.js +166 -166
  5. package/package.json +3 -3
  6. package/templates/default/.eslintrc.json +5 -5
  7. package/templates/default/.prettierignore +7 -7
  8. package/templates/default/.prettierrc.json +27 -27
  9. package/templates/default/CHANGELOG.md +140 -140
  10. package/templates/default/_gitignore +57 -57
  11. package/templates/default/app/(auth)/auth/login/LoginPage.tsx +192 -192
  12. package/templates/default/app/(auth)/auth/login/page.tsx +11 -11
  13. package/templates/default/app/(auth)/auth-locale-provider.tsx +34 -34
  14. package/templates/default/app/(auth)/layout.tsx +81 -81
  15. package/templates/default/app/(rootLayout)/(plugins)/[...slug]/page.tsx +40 -40
  16. package/templates/default/app/(rootLayout)/(plugins)/[...slug]/plugin-server-registry.ts +22 -22
  17. package/templates/default/app/(rootLayout)/admins/page.tsx +10 -10
  18. package/templates/default/app/(rootLayout)/browse/[section]/[page]/page.tsx +22 -22
  19. package/templates/default/app/(rootLayout)/categorized/[section]/page.tsx +15 -15
  20. package/templates/default/app/(rootLayout)/dashboard/page.tsx +63 -63
  21. package/templates/default/app/(rootLayout)/dashboard-new/page.tsx +7 -7
  22. package/templates/default/app/(rootLayout)/edit/[section]/[itemId]/page.tsx +17 -17
  23. package/templates/default/app/(rootLayout)/layout.tsx +81 -81
  24. package/templates/default/app/(rootLayout)/loading.tsx +10 -10
  25. package/templates/default/app/(rootLayout)/log/page.tsx +7 -7
  26. package/templates/default/app/(rootLayout)/new/[section]/page.tsx +15 -15
  27. package/templates/default/app/(rootLayout)/section/[section]/page.tsx +16 -16
  28. package/templates/default/app/(rootLayout)/settings/page.tsx +13 -13
  29. package/templates/default/app/_trpc/client.ts +3 -3
  30. package/templates/default/app/api/auth/csrf/route.ts +25 -25
  31. package/templates/default/app/api/auth/refresh/route.ts +10 -10
  32. package/templates/default/app/api/auth/session/route.ts +20 -20
  33. package/templates/default/app/api/editor/photo/route.ts +49 -49
  34. package/templates/default/app/api/photo/route.ts +27 -27
  35. package/templates/default/app/api/submit/section/item/[slug]/route.ts +66 -66
  36. package/templates/default/app/api/submit/section/item/route.ts +56 -56
  37. package/templates/default/app/api/submit/section/simple/route.ts +57 -57
  38. package/templates/default/app/api/trpc/[trpc]/route.ts +33 -33
  39. package/templates/default/app/api/video/route.ts +174 -174
  40. package/templates/default/app/globals.css +219 -219
  41. package/templates/default/app/providers.tsx +152 -152
  42. package/templates/default/cms.config.ts +49 -52
  43. package/templates/default/components/AdminCard.tsx +166 -166
  44. package/templates/default/components/AdminEditPage.tsx +124 -124
  45. package/templates/default/components/AdminPrivilegeCard.tsx +185 -185
  46. package/templates/default/components/AdminsPage.tsx +43 -43
  47. package/templates/default/components/AnalyticsPage.tsx +128 -128
  48. package/templates/default/components/BarChartBox.tsx +42 -42
  49. package/templates/default/components/BrowsePage.tsx +106 -106
  50. package/templates/default/components/CategorizedSectionPage.tsx +31 -31
  51. package/templates/default/components/CategoryDeleteConfirmPage.tsx +130 -130
  52. package/templates/default/components/CategorySectionSelectInput.tsx +140 -140
  53. package/templates/default/components/ConditionalFields.tsx +49 -49
  54. package/templates/default/components/ContainerBox.tsx +24 -24
  55. package/templates/default/components/DashboardNewPage.tsx +253 -253
  56. package/templates/default/components/DashboardPage.tsx +188 -188
  57. package/templates/default/components/DashboardPageAlt.tsx +45 -45
  58. package/templates/default/components/DefaultNavItems.tsx +3 -3
  59. package/templates/default/components/Dropzone.tsx +154 -154
  60. package/templates/default/components/EmailCard.tsx +138 -138
  61. package/templates/default/components/EmailPasswordForm.tsx +85 -85
  62. package/templates/default/components/EmailQuotaForm.tsx +73 -73
  63. package/templates/default/components/EmailsPage.tsx +49 -49
  64. package/templates/default/components/ErrorComponent.tsx +16 -16
  65. package/templates/default/components/GalleryPhoto.tsx +93 -93
  66. package/templates/default/components/InfoCard.tsx +93 -93
  67. package/templates/default/components/ItemEditPage.tsx +214 -214
  68. package/templates/default/components/Layout.tsx +84 -84
  69. package/templates/default/components/LoadingSpinners.tsx +67 -67
  70. package/templates/default/components/LogPage.tsx +107 -107
  71. package/templates/default/components/Modal.tsx +166 -166
  72. package/templates/default/components/Navbar.tsx +258 -258
  73. package/templates/default/components/NewAdminForm.tsx +173 -173
  74. package/templates/default/components/NewEmailForm.tsx +132 -132
  75. package/templates/default/components/NewPage.tsx +205 -205
  76. package/templates/default/components/NewVariantComponent.tsx +229 -229
  77. package/templates/default/components/PhotoGallery.tsx +35 -35
  78. package/templates/default/components/PieChartBox.tsx +101 -101
  79. package/templates/default/components/ProgressBar.tsx +48 -48
  80. package/templates/default/components/ProtectedDocument.tsx +78 -78
  81. package/templates/default/components/ProtectedImage.tsx +143 -143
  82. package/templates/default/components/ProtectedVideo.tsx +76 -76
  83. package/templates/default/components/SectionItemCard.tsx +144 -144
  84. package/templates/default/components/SectionItemStatusBadge.tsx +17 -17
  85. package/templates/default/components/SectionPage.tsx +125 -125
  86. package/templates/default/components/SelectBox.tsx +98 -98
  87. package/templates/default/components/SelectInputButtons.tsx +125 -125
  88. package/templates/default/components/SettingsPage.tsx +232 -232
  89. package/templates/default/components/Sidebar.tsx +201 -201
  90. package/templates/default/components/SidebarDropdownItem.tsx +80 -80
  91. package/templates/default/components/SidebarItem.tsx +20 -20
  92. package/templates/default/components/ThemeProvider.tsx +8 -8
  93. package/templates/default/components/TooltipComponent.tsx +27 -27
  94. package/templates/default/components/VariantCard.tsx +124 -124
  95. package/templates/default/components/VariantEditPage.tsx +230 -230
  96. package/templates/default/components/analytics/BounceRate.tsx +70 -70
  97. package/templates/default/components/analytics/LivePageViews.tsx +55 -55
  98. package/templates/default/components/analytics/LiveUsersCount.tsx +33 -33
  99. package/templates/default/components/analytics/MonthlyPageViews.tsx +42 -42
  100. package/templates/default/components/analytics/TopCountries.tsx +52 -52
  101. package/templates/default/components/analytics/TopDevices.tsx +46 -46
  102. package/templates/default/components/analytics/TopMediums.tsx +58 -58
  103. package/templates/default/components/analytics/TopSources.tsx +45 -45
  104. package/templates/default/components/analytics/TotalPageViews.tsx +41 -41
  105. package/templates/default/components/analytics/TotalSessions.tsx +41 -41
  106. package/templates/default/components/analytics/TotalUniqueUsers.tsx +41 -41
  107. package/templates/default/components/custom/RightHomeRoomVariantCard.tsx +138 -138
  108. package/templates/default/components/dndKit/Draggable.tsx +21 -21
  109. package/templates/default/components/dndKit/Droppable.tsx +20 -20
  110. package/templates/default/components/dndKit/SortableItem.tsx +18 -18
  111. package/templates/default/components/form/DateRangeFormInput.tsx +57 -57
  112. package/templates/default/components/form/Form.tsx +317 -317
  113. package/templates/default/components/form/FormInputElement.tsx +70 -70
  114. package/templates/default/components/form/FormInputs.tsx +112 -112
  115. package/templates/default/components/form/helpers/_section-hot-reload.js +1 -1
  116. package/templates/default/components/form/helpers/util.ts +17 -17
  117. package/templates/default/components/form/inputs/CheckboxFormInput.tsx +33 -33
  118. package/templates/default/components/form/inputs/ColorFormInput.tsx +44 -44
  119. package/templates/default/components/form/inputs/DateFormInput.tsx +156 -156
  120. package/templates/default/components/form/inputs/DocumentFormInput.tsx +222 -222
  121. package/templates/default/components/form/inputs/MapFormInput.tsx +140 -140
  122. package/templates/default/components/form/inputs/MultipleSelectFormInput.tsx +83 -83
  123. package/templates/default/components/form/inputs/NumberFormInput.tsx +42 -42
  124. package/templates/default/components/form/inputs/PasswordFormInput.tsx +47 -47
  125. package/templates/default/components/form/inputs/PhotoFormInput.tsx +219 -219
  126. package/templates/default/components/form/inputs/RichTextFormInput.tsx +135 -135
  127. package/templates/default/components/form/inputs/SelectFormInput.tsx +175 -175
  128. package/templates/default/components/form/inputs/SlugFormInput.tsx +129 -129
  129. package/templates/default/components/form/inputs/TagsFormInput.tsx +154 -154
  130. package/templates/default/components/form/inputs/TextFormInput.tsx +48 -48
  131. package/templates/default/components/form/inputs/TextareaFormInput.tsx +47 -47
  132. package/templates/default/components/form/inputs/VideoFormInput.tsx +118 -118
  133. package/templates/default/components/locale-dropdown.tsx +74 -74
  134. package/templates/default/components/locale-picker.tsx +85 -85
  135. package/templates/default/components/login-locale-dropdown.tsx +46 -46
  136. package/templates/default/components/multi-select.tsx +1144 -1144
  137. package/templates/default/components/pagination/Pagination.tsx +36 -36
  138. package/templates/default/components/pagination/PaginationButtons.tsx +147 -147
  139. package/templates/default/components/theme-toggle.tsx +37 -37
  140. package/templates/default/components/ui/accordion.tsx +53 -53
  141. package/templates/default/components/ui/alert-dialog.tsx +157 -157
  142. package/templates/default/components/ui/alert.tsx +46 -46
  143. package/templates/default/components/ui/badge.tsx +38 -38
  144. package/templates/default/components/ui/button.tsx +62 -62
  145. package/templates/default/components/ui/calendar.tsx +166 -166
  146. package/templates/default/components/ui/card.tsx +43 -43
  147. package/templates/default/components/ui/checkbox.tsx +29 -29
  148. package/templates/default/components/ui/command.tsx +137 -137
  149. package/templates/default/components/ui/custom-alert-dialog.tsx +113 -113
  150. package/templates/default/components/ui/custom-dialog.tsx +123 -123
  151. package/templates/default/components/ui/dialog.tsx +123 -123
  152. package/templates/default/components/ui/dropdown-menu.tsx +182 -182
  153. package/templates/default/components/ui/input-group.tsx +54 -54
  154. package/templates/default/components/ui/input.tsx +22 -22
  155. package/templates/default/components/ui/label.tsx +19 -19
  156. package/templates/default/components/ui/popover.tsx +42 -42
  157. package/templates/default/components/ui/progress.tsx +31 -31
  158. package/templates/default/components/ui/scroll-area.tsx +42 -42
  159. package/templates/default/components/ui/select.tsx +165 -165
  160. package/templates/default/components/ui/separator.tsx +28 -28
  161. package/templates/default/components/ui/sheet.tsx +103 -103
  162. package/templates/default/components/ui/switch.tsx +29 -29
  163. package/templates/default/components/ui/table.tsx +83 -83
  164. package/templates/default/components/ui/tabs.tsx +55 -55
  165. package/templates/default/components/ui/toast.tsx +113 -113
  166. package/templates/default/components/ui/toaster.tsx +35 -35
  167. package/templates/default/components/ui/tooltip.tsx +30 -30
  168. package/templates/default/components/ui/use-toast.ts +188 -188
  169. package/templates/default/components.json +21 -21
  170. package/templates/default/context/ModalProvider.tsx +53 -53
  171. package/templates/default/drizzle.config.ts +4 -4
  172. package/templates/default/dynamic-schemas/schema.ts +10 -0
  173. package/templates/default/env/env.js +130 -130
  174. package/templates/default/envConfig.ts +4 -4
  175. package/templates/default/hooks/useModal.ts +8 -8
  176. package/templates/default/lib/apiHelpers.ts +92 -92
  177. package/templates/default/lib/postinstall.js +14 -14
  178. package/templates/default/lib/utils.ts +6 -6
  179. package/templates/default/next-env.d.ts +6 -6
  180. package/templates/default/next.config.ts +23 -23
  181. package/templates/default/package.json +2 -4
  182. package/templates/default/postcss.config.mjs +6 -6
  183. package/templates/default/proxy.ts +32 -32
  184. package/templates/default/tsconfig.json +48 -48
@@ -1,143 +1,143 @@
1
- import Image from 'next/image'
2
- import React, { useEffect } from 'react'
3
- import { useAxiosPrivate } from 'nextjs-cms/auth/hooks'
4
- import { AxiosInstance } from 'axios'
5
- import { base64ToBlob } from 'nextjs-cms/utils'
6
-
7
- const defaultDataPlaceholder: string =
8
- 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mPkY2Q8AgABCwDWK6njxAAAAABJRU5ErkJggg=='
9
-
10
- async function getProtectedPhoto({
11
- src,
12
- axiosPrivate,
13
- }: {
14
- src: string
15
- axiosPrivate: AxiosInstance
16
- }): Promise<string | null> {
17
- try {
18
- const res = await axiosPrivate.get(src)
19
- return res.data
20
- } catch (error) {
21
- return null
22
- }
23
- }
24
-
25
- const ProtectedImage = ({
26
- section,
27
- photo,
28
- isThumb = false,
29
- alt,
30
- width,
31
- height,
32
- fill = false,
33
- className = 'rounded-3xl object-cover',
34
- displayAs = 'blob',
35
- }: {
36
- section: string
37
- photo: string
38
- isThumb?: boolean
39
- alt: string
40
- width: number
41
- height: number
42
- fill?: boolean
43
- className?: string
44
- displayAs?: 'blob' | 'base64'
45
- }) => {
46
- const axiosPrivate = useAxiosPrivate()
47
- const [srcUrl, setSrcUrl] = React.useState<string | null>(null)
48
-
49
- // Use trpc?
50
- /*
51
- const { data, isLoading } = trpc.files.getPhoto.useQuery({
52
- name: photo,
53
- folder: section,
54
- isThumb: isThumb,
55
- })
56
-
57
- useEffect(() => {
58
- if (!data) return
59
- switch (displayAs) {
60
- case 'blob':
61
- setSrcUrl(
62
- URL.createObjectURL(
63
- base64ToBlob({
64
- base64: data,
65
- contentType: 'image/webp',
66
- stripHeader: true,
67
- }),
68
- ),
69
- )
70
- break
71
- case 'base64':
72
- setSrcUrl(data)
73
- break
74
- }
75
-
76
- return () => {
77
- setSrcUrl(null)
78
- if (displayAs === 'blob' && srcUrl) {
79
- URL.revokeObjectURL(srcUrl)
80
- }
81
- }
82
- }, [data])
83
- */
84
- useEffect(() => {
85
- if (!photo || !section) return
86
- getProtectedPhoto({
87
- src: `/photo?name=${photo}&folder=${section}&isThumb=${isThumb}`,
88
- axiosPrivate: axiosPrivate,
89
- }).then((res) => {
90
- if (!res) return
91
- switch (displayAs) {
92
- case 'blob':
93
- setSrcUrl(
94
- URL.createObjectURL(
95
- base64ToBlob({
96
- base64: res,
97
- contentType: 'image/webp',
98
- stripHeader: true,
99
- }),
100
- ),
101
- )
102
- break
103
- case 'base64':
104
- setSrcUrl(res)
105
- break
106
- }
107
- })
108
-
109
- return () => {
110
- setSrcUrl(null)
111
- if (displayAs === 'blob' && srcUrl) {
112
- URL.revokeObjectURL(srcUrl)
113
- }
114
- }
115
- }, [])
116
-
117
- return (
118
- <Image
119
- alt={alt}
120
- className={className}
121
- src={srcUrl ? srcUrl : defaultDataPlaceholder}
122
- width={fill ? undefined : width}
123
- height={fill ? undefined : height}
124
- style={
125
- !fill
126
- ? {
127
- width: `${width}px`,
128
- height: 'auto',
129
- }
130
- : undefined
131
- }
132
- fill={fill ? true : undefined}
133
- onLoad={(e) => {
134
- // Revoke the blob url after the image is loaded
135
- if (displayAs === 'blob' && srcUrl) {
136
- URL.revokeObjectURL(srcUrl)
137
- }
138
- }}
139
- />
140
- )
141
- }
142
-
143
- export default ProtectedImage
1
+ import Image from 'next/image'
2
+ import React, { useEffect } from 'react'
3
+ import { useAxiosPrivate } from 'nextjs-cms/auth/hooks'
4
+ import { AxiosInstance } from 'axios'
5
+ import { base64ToBlob } from 'nextjs-cms/utils'
6
+
7
+ const defaultDataPlaceholder: string =
8
+ 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mPkY2Q8AgABCwDWK6njxAAAAABJRU5ErkJggg=='
9
+
10
+ async function getProtectedPhoto({
11
+ src,
12
+ axiosPrivate,
13
+ }: {
14
+ src: string
15
+ axiosPrivate: AxiosInstance
16
+ }): Promise<string | null> {
17
+ try {
18
+ const res = await axiosPrivate.get(src)
19
+ return res.data
20
+ } catch (error) {
21
+ return null
22
+ }
23
+ }
24
+
25
+ const ProtectedImage = ({
26
+ section,
27
+ photo,
28
+ isThumb = false,
29
+ alt,
30
+ width,
31
+ height,
32
+ fill = false,
33
+ className = 'rounded-3xl object-cover',
34
+ displayAs = 'blob',
35
+ }: {
36
+ section: string
37
+ photo: string
38
+ isThumb?: boolean
39
+ alt: string
40
+ width: number
41
+ height: number
42
+ fill?: boolean
43
+ className?: string
44
+ displayAs?: 'blob' | 'base64'
45
+ }) => {
46
+ const axiosPrivate = useAxiosPrivate()
47
+ const [srcUrl, setSrcUrl] = React.useState<string | null>(null)
48
+
49
+ // Use trpc?
50
+ /*
51
+ const { data, isLoading } = trpc.files.getPhoto.useQuery({
52
+ name: photo,
53
+ folder: section,
54
+ isThumb: isThumb,
55
+ })
56
+
57
+ useEffect(() => {
58
+ if (!data) return
59
+ switch (displayAs) {
60
+ case 'blob':
61
+ setSrcUrl(
62
+ URL.createObjectURL(
63
+ base64ToBlob({
64
+ base64: data,
65
+ contentType: 'image/webp',
66
+ stripHeader: true,
67
+ }),
68
+ ),
69
+ )
70
+ break
71
+ case 'base64':
72
+ setSrcUrl(data)
73
+ break
74
+ }
75
+
76
+ return () => {
77
+ setSrcUrl(null)
78
+ if (displayAs === 'blob' && srcUrl) {
79
+ URL.revokeObjectURL(srcUrl)
80
+ }
81
+ }
82
+ }, [data])
83
+ */
84
+ useEffect(() => {
85
+ if (!photo || !section) return
86
+ getProtectedPhoto({
87
+ src: `/photo?name=${photo}&folder=${section}&isThumb=${isThumb}`,
88
+ axiosPrivate: axiosPrivate,
89
+ }).then((res) => {
90
+ if (!res) return
91
+ switch (displayAs) {
92
+ case 'blob':
93
+ setSrcUrl(
94
+ URL.createObjectURL(
95
+ base64ToBlob({
96
+ base64: res,
97
+ contentType: 'image/webp',
98
+ stripHeader: true,
99
+ }),
100
+ ),
101
+ )
102
+ break
103
+ case 'base64':
104
+ setSrcUrl(res)
105
+ break
106
+ }
107
+ })
108
+
109
+ return () => {
110
+ setSrcUrl(null)
111
+ if (displayAs === 'blob' && srcUrl) {
112
+ URL.revokeObjectURL(srcUrl)
113
+ }
114
+ }
115
+ }, [])
116
+
117
+ return (
118
+ <Image
119
+ alt={alt}
120
+ className={className}
121
+ src={srcUrl ? srcUrl : defaultDataPlaceholder}
122
+ width={fill ? undefined : width}
123
+ height={fill ? undefined : height}
124
+ style={
125
+ !fill
126
+ ? {
127
+ width: `${width}px`,
128
+ height: 'auto',
129
+ }
130
+ : undefined
131
+ }
132
+ fill={fill ? true : undefined}
133
+ onLoad={(e) => {
134
+ // Revoke the blob url after the image is loaded
135
+ if (displayAs === 'blob' && srcUrl) {
136
+ URL.revokeObjectURL(srcUrl)
137
+ }
138
+ }}
139
+ />
140
+ )
141
+ }
142
+
143
+ export default ProtectedImage
@@ -1,76 +1,76 @@
1
- import React, { useEffect } from 'react'
2
- import { useRefreshToken } from 'nextjs-cms/auth/hooks'
3
- import { useSession } from 'nextjs-cms/auth/react'
4
-
5
- const ProtectedVideo = ({
6
- section,
7
- fieldName,
8
- file,
9
- width,
10
- height,
11
- className = 'rounded-3xl object-cover',
12
- }: {
13
- section: string
14
- fieldName: string
15
- file: string
16
- width: number
17
- height: number
18
- className?: string
19
- }) => {
20
- const refresh = useRefreshToken()
21
- const ref = React.useRef<HTMLVideoElement>(null)
22
- const [currentTime, setCurrentTime] = React.useState(0)
23
- const [isSeeking, setIsSeeking] = React.useState(false)
24
-
25
- const session = useSession()
26
- useEffect(() => {
27
- if (ref.current && currentTime) {
28
- ref.current.currentTime = currentTime
29
- ref.current.play().then(() => {})
30
- }
31
- }, [session])
32
-
33
- return (
34
- <div className={className}>
35
- {/**
36
- * The video will stream as long as the access token is valid
37
- * If the access token expires, the video will pause
38
- * A refresh token request will be sent to `/api/auth/refresh`
39
- * If the refresh token is valid, the video will resume playing
40
- * If the refresh token is invalid, the user will be logged out
41
- */}
42
- <video
43
- ref={ref}
44
- src={`/api/video?sectionName=${section}&fieldName=${fieldName}&name=${file}`}
45
- className='max-w-full'
46
- width={width}
47
- controls={true}
48
- height={height}
49
- onErrorCapture={async (e) => {
50
- if (isSeeking) return
51
- setIsSeeking(true)
52
- // console.log('seeked')
53
- /**
54
- * Pause the video
55
- */
56
- e.currentTarget.pause()
57
- /**
58
- * Get the current time of the video and save it in the state
59
- */
60
- setCurrentTime(e.currentTarget.currentTime)
61
- /**
62
- * Refresh the access token
63
- * This will trigger the `useEffect` hook
64
- * and set the current time of the video
65
- * to the time it was paused at
66
- * and play the video
67
- */
68
- await refresh()
69
- setIsSeeking(false)
70
- }}
71
- />
72
- </div>
73
- )
74
- }
75
-
76
- export default ProtectedVideo
1
+ import React, { useEffect } from 'react'
2
+ import { useRefreshToken } from 'nextjs-cms/auth/hooks'
3
+ import { useSession } from 'nextjs-cms/auth/react'
4
+
5
+ const ProtectedVideo = ({
6
+ section,
7
+ fieldName,
8
+ file,
9
+ width,
10
+ height,
11
+ className = 'rounded-3xl object-cover',
12
+ }: {
13
+ section: string
14
+ fieldName: string
15
+ file: string
16
+ width: number
17
+ height: number
18
+ className?: string
19
+ }) => {
20
+ const refresh = useRefreshToken()
21
+ const ref = React.useRef<HTMLVideoElement>(null)
22
+ const [currentTime, setCurrentTime] = React.useState(0)
23
+ const [isSeeking, setIsSeeking] = React.useState(false)
24
+
25
+ const session = useSession()
26
+ useEffect(() => {
27
+ if (ref.current && currentTime) {
28
+ ref.current.currentTime = currentTime
29
+ ref.current.play().then(() => {})
30
+ }
31
+ }, [session])
32
+
33
+ return (
34
+ <div className={className}>
35
+ {/**
36
+ * The video will stream as long as the access token is valid
37
+ * If the access token expires, the video will pause
38
+ * A refresh token request will be sent to `/api/auth/refresh`
39
+ * If the refresh token is valid, the video will resume playing
40
+ * If the refresh token is invalid, the user will be logged out
41
+ */}
42
+ <video
43
+ ref={ref}
44
+ src={`/api/video?sectionName=${section}&fieldName=${fieldName}&name=${file}`}
45
+ className='max-w-full'
46
+ width={width}
47
+ controls={true}
48
+ height={height}
49
+ onErrorCapture={async (e) => {
50
+ if (isSeeking) return
51
+ setIsSeeking(true)
52
+ // console.log('seeked')
53
+ /**
54
+ * Pause the video
55
+ */
56
+ e.currentTarget.pause()
57
+ /**
58
+ * Get the current time of the video and save it in the state
59
+ */
60
+ setCurrentTime(e.currentTarget.currentTime)
61
+ /**
62
+ * Refresh the access token
63
+ * This will trigger the `useEffect` hook
64
+ * and set the current time of the video
65
+ * to the time it was paused at
66
+ * and play the video
67
+ */
68
+ await refresh()
69
+ setIsSeeking(false)
70
+ }}
71
+ />
72
+ </div>
73
+ )
74
+ }
75
+
76
+ export default ProtectedVideo