@hed-hog/lms 0.0.350 → 0.0.353

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 (160) hide show
  1. package/dist/certificate/certificate.controller.d.ts +2 -2
  2. package/dist/certificate/certificate.controller.d.ts.map +1 -1
  3. package/dist/certificate/certificate.controller.js +8 -6
  4. package/dist/certificate/certificate.controller.js.map +1 -1
  5. package/dist/certificate/certificate.service.d.ts +5 -2
  6. package/dist/certificate/certificate.service.d.ts.map +1 -1
  7. package/dist/certificate/certificate.service.js +70 -6
  8. package/dist/certificate/certificate.service.js.map +1 -1
  9. package/dist/course/course-structure.controller.d.ts +24 -10
  10. package/dist/course/course-structure.controller.d.ts.map +1 -1
  11. package/dist/course/course-structure.controller.js +23 -2
  12. package/dist/course/course-structure.controller.js.map +1 -1
  13. package/dist/course/course-structure.service.d.ts +16 -8
  14. package/dist/course/course-structure.service.d.ts.map +1 -1
  15. package/dist/course/course-structure.service.js +61 -30
  16. package/dist/course/course-structure.service.js.map +1 -1
  17. package/dist/course/course-video-conversion.service.d.ts +37 -0
  18. package/dist/course/course-video-conversion.service.d.ts.map +1 -0
  19. package/dist/course/course-video-conversion.service.js +308 -0
  20. package/dist/course/course-video-conversion.service.js.map +1 -0
  21. package/dist/course/course.controller.d.ts +17 -0
  22. package/dist/course/course.controller.d.ts.map +1 -1
  23. package/dist/course/course.controller.js +23 -0
  24. package/dist/course/course.controller.js.map +1 -1
  25. package/dist/course/course.module.d.ts.map +1 -1
  26. package/dist/course/course.module.js +15 -2
  27. package/dist/course/course.module.js.map +1 -1
  28. package/dist/course/course.service.d.ts +15 -0
  29. package/dist/course/course.service.d.ts.map +1 -1
  30. package/dist/course/course.service.js +103 -49
  31. package/dist/course/course.service.js.map +1 -1
  32. package/dist/course/dto/create-course-structure-lesson.dto.d.ts +5 -1
  33. package/dist/course/dto/create-course-structure-lesson.dto.d.ts.map +1 -1
  34. package/dist/course/dto/create-course-structure-lesson.dto.js +16 -2
  35. package/dist/course/dto/create-course-structure-lesson.dto.js.map +1 -1
  36. package/dist/course/dto/create-course.dto.d.ts +1 -0
  37. package/dist/course/dto/create-course.dto.d.ts.map +1 -1
  38. package/dist/course/dto/create-course.dto.js +9 -0
  39. package/dist/course/dto/create-course.dto.js.map +1 -1
  40. package/dist/enterprise/enterprise.controller.d.ts +3 -3
  41. package/dist/enterprise/enterprise.controller.d.ts.map +1 -1
  42. package/dist/enterprise/enterprise.controller.js +0 -1
  43. package/dist/enterprise/enterprise.controller.js.map +1 -1
  44. package/dist/enterprise/enterprise.service.d.ts +3 -3
  45. package/dist/evaluation/evaluation.service.d.ts.map +1 -1
  46. package/dist/evaluation/evaluation.service.js +9 -2
  47. package/dist/evaluation/evaluation.service.js.map +1 -1
  48. package/dist/index.d.ts +1 -0
  49. package/dist/index.d.ts.map +1 -1
  50. package/dist/index.js +1 -0
  51. package/dist/index.js.map +1 -1
  52. package/dist/lms.module.d.ts.map +1 -1
  53. package/dist/lms.module.js +3 -0
  54. package/dist/lms.module.js.map +1 -1
  55. package/dist/video-resolution-profile/dto/create-video-resolution-profile.dto.d.ts +6 -0
  56. package/dist/video-resolution-profile/dto/create-video-resolution-profile.dto.d.ts.map +1 -0
  57. package/dist/video-resolution-profile/dto/create-video-resolution-profile.dto.js +33 -0
  58. package/dist/video-resolution-profile/dto/create-video-resolution-profile.dto.js.map +1 -0
  59. package/dist/video-resolution-profile/dto/update-video-resolution-profile.dto.d.ts +6 -0
  60. package/dist/video-resolution-profile/dto/update-video-resolution-profile.dto.d.ts.map +1 -0
  61. package/dist/video-resolution-profile/dto/update-video-resolution-profile.dto.js +33 -0
  62. package/dist/video-resolution-profile/dto/update-video-resolution-profile.dto.js.map +1 -0
  63. package/dist/video-resolution-profile/video-resolution-profile.controller.d.ts +38 -0
  64. package/dist/video-resolution-profile/video-resolution-profile.controller.d.ts.map +1 -0
  65. package/dist/video-resolution-profile/video-resolution-profile.controller.js +89 -0
  66. package/dist/video-resolution-profile/video-resolution-profile.controller.js.map +1 -0
  67. package/dist/video-resolution-profile/video-resolution-profile.mcp-tools.d.ts +26 -0
  68. package/dist/video-resolution-profile/video-resolution-profile.mcp-tools.d.ts.map +1 -0
  69. package/dist/video-resolution-profile/video-resolution-profile.mcp-tools.js +160 -0
  70. package/dist/video-resolution-profile/video-resolution-profile.mcp-tools.js.map +1 -0
  71. package/dist/video-resolution-profile/video-resolution-profile.module.d.ts +3 -0
  72. package/dist/video-resolution-profile/video-resolution-profile.module.d.ts.map +1 -0
  73. package/dist/video-resolution-profile/video-resolution-profile.module.js +26 -0
  74. package/dist/video-resolution-profile/video-resolution-profile.module.js.map +1 -0
  75. package/dist/video-resolution-profile/video-resolution-profile.service.d.ts +45 -0
  76. package/dist/video-resolution-profile/video-resolution-profile.service.d.ts.map +1 -0
  77. package/dist/video-resolution-profile/video-resolution-profile.service.js +117 -0
  78. package/dist/video-resolution-profile/video-resolution-profile.service.js.map +1 -0
  79. package/hedhog/data/menu.yaml +17 -0
  80. package/hedhog/data/route.yaml +133 -0
  81. package/hedhog/data/video_resolution_profile.yaml +7 -0
  82. package/hedhog/frontend/app/_components/class-form-sheet.tsx.ejs +269 -324
  83. package/hedhog/frontend/app/_components/course-form-sheet.tsx.ejs +124 -70
  84. package/hedhog/frontend/app/_components/create-lms-instructor-sheet.tsx.ejs +7 -4
  85. package/hedhog/frontend/app/_components/create-lms-person-sheet.tsx.ejs +2 -2
  86. package/hedhog/frontend/app/_components/create-lms-student-person-sheet.tsx.ejs +2 -2
  87. package/hedhog/frontend/app/_lib/editor/templateSerializer.ts.ejs +34 -4
  88. package/hedhog/frontend/app/_lib/editor/types.ts.ejs +28 -3
  89. package/hedhog/frontend/app/achievements/page.tsx.ejs +9 -3
  90. package/hedhog/frontend/app/bitcodes/page.tsx.ejs +9 -3
  91. package/hedhog/frontend/app/certificates/issued/page.tsx.ejs +7 -3
  92. package/hedhog/frontend/app/certificates/models/CanvasStage.tsx.ejs +29 -8
  93. package/hedhog/frontend/app/certificates/models/LeftPanel.tsx.ejs +14 -0
  94. package/hedhog/frontend/app/certificates/models/RightPanel.tsx.ejs +194 -9
  95. package/hedhog/frontend/app/certificates/models/page.tsx.ejs +15 -5
  96. package/hedhog/frontend/app/classes/[id]/page.tsx.ejs +9 -5
  97. package/hedhog/frontend/app/classes/page.tsx.ejs +73 -47
  98. package/hedhog/frontend/app/courses/[id]/_components/CourseCertificateCard.tsx.ejs +19 -9
  99. package/hedhog/frontend/app/courses/[id]/_components/CourseClassificationCard.tsx.ejs +24 -1
  100. package/hedhog/frontend/app/courses/[id]/_components/CourseContentCard.tsx.ejs +1 -1
  101. package/hedhog/frontend/app/courses/[id]/_components/CourseMainInfoCard.tsx.ejs +1 -1
  102. package/hedhog/frontend/app/courses/[id]/_components/CourseRelationsCard.tsx.ejs +28 -16
  103. package/hedhog/frontend/app/courses/[id]/_components/CourseSectionCard.tsx.ejs +11 -6
  104. package/hedhog/frontend/app/courses/[id]/_components/CourseSummaryCard.tsx.ejs +7 -4
  105. package/hedhog/frontend/app/courses/[id]/_components/course-edit-types.ts.ejs +1 -0
  106. package/hedhog/frontend/app/courses/[id]/page.tsx.ejs +24 -87
  107. package/hedhog/frontend/app/courses/[id]/structure/_components/editor-course.tsx.ejs +892 -411
  108. package/hedhog/frontend/app/courses/[id]/structure/_components/editor-lesson.tsx.ejs +1004 -293
  109. package/hedhog/frontend/app/courses/[id]/structure/_components/editor-session.tsx.ejs +11 -11
  110. package/hedhog/frontend/app/courses/[id]/structure/_components/shortcuts-help.tsx.ejs +62 -52
  111. package/hedhog/frontend/app/courses/[id]/structure/_components/types.ts.ejs +2 -0
  112. package/hedhog/frontend/app/courses/[id]/structure/_data/adapters/course-structure.adapter.ts.ejs +19 -6
  113. package/hedhog/frontend/app/courses/[id]/structure/_data/services/course-structure.service.ts.ejs +86 -1
  114. package/hedhog/frontend/app/courses/[id]/structure/_data/types/api-course.types.ts.ejs +3 -0
  115. package/hedhog/frontend/app/courses/[id]/structure/_data/use-course-structure-mutations.ts.ejs +1 -0
  116. package/hedhog/frontend/app/courses/page.tsx.ejs +112 -89
  117. package/hedhog/frontend/app/enterprise/[id]/page.tsx.ejs +1 -1
  118. package/hedhog/frontend/app/enterprise/_components/enterprise-admin-create-sheet.tsx.ejs +10 -3
  119. package/hedhog/frontend/app/enterprise/_components/enterprise-detail-sheet.tsx.ejs +8 -4
  120. package/hedhog/frontend/app/enterprise/_components/enterprise-person-edit-sheet.tsx.ejs +2 -2
  121. package/hedhog/frontend/app/enterprise/_components/enterprise-sheet.tsx.ejs +10 -4
  122. package/hedhog/frontend/app/enterprise/_components/enterprise-student-create-sheet.tsx.ejs +10 -3
  123. package/hedhog/frontend/app/enterprise/_components/enterprise-user-create-sheet.tsx.ejs +10 -3
  124. package/hedhog/frontend/app/evaluations/_components/evaluation-topic-form-sheet.tsx.ejs +10 -3
  125. package/hedhog/frontend/app/exams/[id]/questions/page.tsx.ejs +23 -9
  126. package/hedhog/frontend/app/exams/page.tsx.ejs +14 -6
  127. package/hedhog/frontend/app/instructor-skills/page.tsx.ejs +9 -3
  128. package/hedhog/frontend/app/instructors/_components/instructor-form-sheet.tsx.ejs +190 -17
  129. package/hedhog/frontend/app/layout.tsx.ejs +5 -1
  130. package/hedhog/frontend/app/paths/page.tsx.ejs +13 -5
  131. package/hedhog/frontend/app/reports/evaluations/page.tsx.ejs +10 -10
  132. package/hedhog/frontend/app/training/page.tsx.ejs +13 -5
  133. package/hedhog/frontend/app/video-resolution-profiles/page.tsx.ejs +607 -0
  134. package/hedhog/frontend/messages/en.json +250 -9
  135. package/hedhog/frontend/messages/pt.json +250 -9
  136. package/hedhog/table/course.yaml +4 -0
  137. package/hedhog/table/course_lesson_file.yaml +8 -0
  138. package/hedhog/table/course_video_resolution_profile.yaml +22 -0
  139. package/hedhog/table/video_resolution_profile.yaml +18 -0
  140. package/package.json +9 -8
  141. package/src/certificate/certificate.controller.ts +19 -14
  142. package/src/certificate/certificate.service.ts +106 -11
  143. package/src/course/course-structure.controller.ts +24 -2
  144. package/src/course/course-structure.service.ts +21 -4
  145. package/src/course/course-video-conversion.service.ts +415 -0
  146. package/src/course/course.controller.ts +18 -0
  147. package/src/course/course.module.ts +15 -2
  148. package/src/course/course.service.ts +72 -2
  149. package/src/course/dto/create-course-structure-lesson.dto.ts +13 -2
  150. package/src/course/dto/create-course.dto.ts +8 -0
  151. package/src/enterprise/enterprise.controller.ts +0 -1
  152. package/src/evaluation/evaluation.service.ts +9 -2
  153. package/src/index.ts +1 -0
  154. package/src/lms.module.ts +3 -0
  155. package/src/video-resolution-profile/dto/create-video-resolution-profile.dto.ts +16 -0
  156. package/src/video-resolution-profile/dto/update-video-resolution-profile.dto.ts +16 -0
  157. package/src/video-resolution-profile/video-resolution-profile.controller.ts +62 -0
  158. package/src/video-resolution-profile/video-resolution-profile.mcp-tools.ts +128 -0
  159. package/src/video-resolution-profile/video-resolution-profile.module.ts +13 -0
  160. package/src/video-resolution-profile/video-resolution-profile.service.ts +117 -0
@@ -0,0 +1,607 @@
1
+ 'use client';
2
+
3
+ import {
4
+ EmptyState,
5
+ Page,
6
+ PageHeader,
7
+ PaginationFooter,
8
+ SearchBar,
9
+ } from '@/components/entity-list';
10
+ import { FfmpegParamsEditor } from '@/components/ffmpeg-params-editor';
11
+ import {
12
+ AlertDialog,
13
+ AlertDialogAction,
14
+ AlertDialogCancel,
15
+ AlertDialogContent,
16
+ AlertDialogDescription,
17
+ AlertDialogHeader,
18
+ AlertDialogTitle,
19
+ } from '@/components/ui/alert-dialog';
20
+ import { Badge } from '@/components/ui/badge';
21
+ import { Button } from '@/components/ui/button';
22
+ import {
23
+ Form,
24
+ FormControl,
25
+ FormField,
26
+ FormItem,
27
+ FormLabel,
28
+ FormMessage,
29
+ } from '@/components/ui/form';
30
+ import { Input } from '@/components/ui/input';
31
+ import {
32
+ Select,
33
+ SelectContent,
34
+ SelectItem,
35
+ SelectTrigger,
36
+ SelectValue,
37
+ } from '@/components/ui/select';
38
+ import {
39
+ Sheet,
40
+ SheetHeader,
41
+ SheetTitle,
42
+ } from '@/components/ui/sheet';
43
+ import { ResizableSheetContent } from '@/components/ui/resizable-sheet-content';
44
+ import { Skeleton } from '@/components/ui/skeleton';
45
+ import {
46
+ Table,
47
+ TableBody,
48
+ TableCell,
49
+ TableHead,
50
+ TableHeader,
51
+ TableRow,
52
+ } from '@/components/ui/table';
53
+ import {
54
+ Tooltip,
55
+ TooltipContent,
56
+ TooltipTrigger,
57
+ } from '@/components/ui/tooltip';
58
+ import { usePersistedPageSize } from '@/hooks/use-persisted-page-size';
59
+ import { cn } from '@/lib/utils';
60
+ import { useApp, useQuery } from '@hed-hog/next-app-provider';
61
+ import { zodResolver } from '@hookform/resolvers/zod';
62
+ import {
63
+ Film,
64
+ PauseCircle,
65
+ Pencil,
66
+ PlayCircle,
67
+ Plus,
68
+ Trash2,
69
+ } from 'lucide-react';
70
+ import { useTranslations } from 'next-intl';
71
+ import { useEffect, useState } from 'react';
72
+ import { useForm } from 'react-hook-form';
73
+ import { toast } from 'sonner';
74
+ import * as z from 'zod';
75
+
76
+ // ─── Types ────────────────────────────────────────────────────────────────────
77
+
78
+ type VideoResolutionProfile = {
79
+ id: number;
80
+ name: string;
81
+ ffmpeg_params: string;
82
+ status: 'active' | 'inactive';
83
+ };
84
+
85
+ type ProfilePaginatedResult = {
86
+ data: VideoResolutionProfile[];
87
+ total: number;
88
+ page: number;
89
+ pageSize: number;
90
+ lastPage?: number;
91
+ };
92
+
93
+ // ─── Zod schema ───────────────────────────────────────────────────────────────
94
+
95
+ const createSchema = (t: ReturnType<typeof useTranslations>) =>
96
+ z.object({
97
+ name: z
98
+ .string()
99
+ .min(1, t('form.validation.required'))
100
+ .max(100, t('form.validation.max100')),
101
+ ffmpeg_params: z.string().min(1, t('form.validation.required')),
102
+ status: z.enum(['active', 'inactive']),
103
+ });
104
+
105
+ type ProfileFormValues = z.infer<ReturnType<typeof createSchema>>;
106
+
107
+ // ─── ProfileFormSheet ──────────────────────────────────────────────────────────
108
+
109
+ interface ProfileFormSheetProps {
110
+ open: boolean;
111
+ onOpenChange: (open: boolean) => void;
112
+ profileToEdit: VideoResolutionProfile | null;
113
+ onSaved: () => void;
114
+ }
115
+
116
+ function ProfileFormSheet({
117
+ open,
118
+ onOpenChange,
119
+ profileToEdit,
120
+ onSaved,
121
+ }: ProfileFormSheetProps) {
122
+ const { request } = useApp();
123
+ const t = useTranslations('lms.VideoResolutionProfilesPage');
124
+ const [isSaving, setIsSaving] = useState(false);
125
+
126
+ const form = useForm<ProfileFormValues>({
127
+ resolver: zodResolver(createSchema(t)),
128
+ defaultValues: { name: '', ffmpeg_params: '', status: 'active' },
129
+ });
130
+
131
+ useEffect(() => {
132
+ if (open) {
133
+ if (profileToEdit) {
134
+ form.reset({
135
+ name: profileToEdit.name,
136
+ ffmpeg_params: profileToEdit.ffmpeg_params,
137
+ status: profileToEdit.status,
138
+ });
139
+ } else {
140
+ form.reset({ name: '', ffmpeg_params: '', status: 'active' });
141
+ }
142
+ }
143
+ }, [open, profileToEdit, form]);
144
+
145
+ const onSubmit = async (values: ProfileFormValues) => {
146
+ try {
147
+ setIsSaving(true);
148
+ if (profileToEdit) {
149
+ await request({
150
+ url: `/lms/video-resolution-profiles/${profileToEdit.id}`,
151
+ method: 'PATCH',
152
+ data: values,
153
+ });
154
+ toast.success(t('messages.updateSuccess'));
155
+ } else {
156
+ await request({
157
+ url: '/lms/video-resolution-profiles',
158
+ method: 'POST',
159
+ data: values,
160
+ });
161
+ toast.success(t('messages.createSuccess'));
162
+ }
163
+ onSaved();
164
+ onOpenChange(false);
165
+ } catch {
166
+ toast.error(t('messages.saveError'));
167
+ } finally {
168
+ setIsSaving(false);
169
+ }
170
+ };
171
+
172
+ return (
173
+ <Sheet open={open} onOpenChange={onOpenChange}>
174
+ <ResizableSheetContent
175
+ sheetId="lms-video-resolution-profiles-form-sheet"
176
+ defaultWidth={560}
177
+ minWidth={420}
178
+ maxWidth={920}
179
+ className="w-full overflow-y-auto sm:max-w-lg"
180
+ >
181
+ <SheetHeader>
182
+ <SheetTitle>
183
+ {profileToEdit ? t('sheet.editTitle') : t('sheet.createTitle')}
184
+ </SheetTitle>
185
+ </SheetHeader>
186
+
187
+ <Form {...form}>
188
+ <form
189
+ onSubmit={form.handleSubmit(onSubmit)}
190
+ className="flex flex-col gap-4 px-4 py-4"
191
+ >
192
+ <div className="grid grid-cols-2 gap-4">
193
+ <FormField
194
+ control={form.control}
195
+ name="name"
196
+ render={({ field }) => (
197
+ <FormItem>
198
+ <FormLabel>{t('form.name')}</FormLabel>
199
+ <FormControl>
200
+ <Input
201
+ placeholder={t('form.namePlaceholder')}
202
+ {...field}
203
+ />
204
+ </FormControl>
205
+ <FormMessage />
206
+ </FormItem>
207
+ )}
208
+ />
209
+
210
+ <FormField
211
+ control={form.control}
212
+ name="status"
213
+ render={({ field }) => (
214
+ <FormItem>
215
+ <FormLabel>{t('form.status')}</FormLabel>
216
+ <Select onValueChange={field.onChange} value={field.value}>
217
+ <FormControl>
218
+ <SelectTrigger className="w-full">
219
+ <SelectValue
220
+ placeholder={t('form.statusPlaceholder')}
221
+ />
222
+ </SelectTrigger>
223
+ </FormControl>
224
+ <SelectContent>
225
+ <SelectItem value="active">
226
+ {t('status.active')}
227
+ </SelectItem>
228
+ <SelectItem value="inactive">
229
+ {t('status.inactive')}
230
+ </SelectItem>
231
+ </SelectContent>
232
+ </Select>
233
+ <FormMessage />
234
+ </FormItem>
235
+ )}
236
+ />
237
+ </div>
238
+
239
+ <FormField
240
+ control={form.control}
241
+ name="ffmpeg_params"
242
+ render={({ field }) => (
243
+ <FormItem>
244
+ <FormLabel>{t('form.ffmpegParams')}</FormLabel>
245
+ <FormControl>
246
+ <FfmpegParamsEditor
247
+ value={field.value}
248
+ onChange={field.onChange}
249
+ />
250
+ </FormControl>
251
+ <FormMessage />
252
+ </FormItem>
253
+ )}
254
+ />
255
+
256
+ <div className="flex justify-end gap-2 pt-2">
257
+ <Button
258
+ type="button"
259
+ variant="outline"
260
+ onClick={() => onOpenChange(false)}
261
+ disabled={isSaving}
262
+ >
263
+ {t('actions.cancel')}
264
+ </Button>
265
+ <Button type="submit" disabled={isSaving}>
266
+ {isSaving ? t('actions.saving') : t('actions.save')}
267
+ </Button>
268
+ </div>
269
+ </form>
270
+ </Form>
271
+ </ResizableSheetContent>
272
+ </Sheet>
273
+ );
274
+ }
275
+
276
+ // ─── Page ─────────────────────────────────────────────────────────────────────
277
+
278
+ export default function VideoResolutionProfilesPage() {
279
+ const { request } = useApp();
280
+ const t = useTranslations('lms.VideoResolutionProfilesPage');
281
+
282
+ const [page, setPage] = useState(1);
283
+ const [pageSize, setPageSize] = usePersistedPageSize({
284
+ storageKey: 'pagination:global:pageSize',
285
+ defaultValue: 6,
286
+ allowedValues: [6, 12, 24, 48],
287
+ });
288
+ const [searchInput, setSearchInput] = useState('');
289
+ const [debouncedSearch, setDebouncedSearch] = useState('');
290
+ const [sheetOpen, setSheetOpen] = useState(false);
291
+ const [profileToEdit, setProfileToEdit] =
292
+ useState<VideoResolutionProfile | null>(null);
293
+ const [deleteDialogOpen, setDeleteDialogOpen] = useState(false);
294
+ const [profileToDelete, setProfileToDelete] =
295
+ useState<VideoResolutionProfile | null>(null);
296
+ const [isDeleting, setIsDeleting] = useState(false);
297
+ const [togglingProfileId, setTogglingProfileId] = useState<number | null>(
298
+ null
299
+ );
300
+
301
+ useEffect(() => {
302
+ const timeout = setTimeout(() => {
303
+ setDebouncedSearch(searchInput.trim());
304
+ setPage(1);
305
+ }, 300);
306
+ return () => clearTimeout(timeout);
307
+ }, [searchInput]);
308
+
309
+ const {
310
+ data: paginate = { data: [], total: 0, page: 1, pageSize, lastPage: 1 },
311
+ isLoading,
312
+ refetch: refetchList,
313
+ } = useQuery<ProfilePaginatedResult>({
314
+ queryKey: [
315
+ 'lms-video-resolution-profiles',
316
+ page,
317
+ pageSize,
318
+ debouncedSearch,
319
+ ],
320
+ queryFn: async () => {
321
+ const params = new URLSearchParams({
322
+ page: String(page),
323
+ pageSize: String(pageSize),
324
+ });
325
+ if (debouncedSearch) params.set('search', debouncedSearch);
326
+ const response = await request<ProfilePaginatedResult>({
327
+ url: `/lms/video-resolution-profiles?${params.toString()}`,
328
+ method: 'GET',
329
+ });
330
+ return response.data;
331
+ },
332
+ placeholderData: (prev) =>
333
+ prev ?? { data: [], total: 0, page: 1, pageSize, lastPage: 1 },
334
+ });
335
+
336
+ const totalPages = Math.max(
337
+ 1,
338
+ (paginate.lastPage ?? Math.ceil((paginate.total || 0) / pageSize)) || 1
339
+ );
340
+
341
+ useEffect(() => {
342
+ if (page > totalPages) setPage(totalPages);
343
+ }, [page, totalPages]);
344
+
345
+ const openCreateSheet = () => {
346
+ setProfileToEdit(null);
347
+ setSheetOpen(true);
348
+ };
349
+
350
+ const openEditSheet = (profile: VideoResolutionProfile) => {
351
+ setProfileToEdit(profile);
352
+ setSheetOpen(true);
353
+ };
354
+
355
+ const handleSaved = async () => {
356
+ await refetchList();
357
+ };
358
+
359
+ const handleDeleteConfirm = async () => {
360
+ if (!profileToDelete) return;
361
+ try {
362
+ setIsDeleting(true);
363
+ await request({
364
+ url: `/lms/video-resolution-profiles/${profileToDelete.id}`,
365
+ method: 'DELETE',
366
+ });
367
+ toast.success(t('messages.deleteSuccess'));
368
+ setDeleteDialogOpen(false);
369
+ setProfileToDelete(null);
370
+ await refetchList();
371
+ } catch {
372
+ toast.error(t('messages.deleteError'));
373
+ } finally {
374
+ setIsDeleting(false);
375
+ }
376
+ };
377
+
378
+ const handleQuickStatusToggle = async (profile: VideoResolutionProfile) => {
379
+ const nextStatus = profile.status === 'active' ? 'inactive' : 'active';
380
+
381
+ try {
382
+ setTogglingProfileId(profile.id);
383
+ await request({
384
+ url: `/lms/video-resolution-profiles/${profile.id}`,
385
+ method: 'PATCH',
386
+ data: { status: nextStatus },
387
+ });
388
+ toast.success(t('messages.updateSuccess'));
389
+ await refetchList();
390
+ } catch {
391
+ toast.error(t('messages.saveError'));
392
+ } finally {
393
+ setTogglingProfileId(null);
394
+ }
395
+ };
396
+
397
+ return (
398
+ <Page>
399
+ <PageHeader
400
+ breadcrumbs={[
401
+ { label: t('breadcrumbs.home'), href: '/' },
402
+ { label: t('breadcrumbs.lms'), href: '/lms' },
403
+ { label: t('breadcrumbs.current') },
404
+ ]}
405
+ title={t('title')}
406
+ description={t('description')}
407
+ actions={[
408
+ {
409
+ label: t('actions.create'),
410
+ onClick: openCreateSheet,
411
+ icon: <Plus className="h-4 w-4" />,
412
+ },
413
+ ]}
414
+ />
415
+
416
+ <SearchBar
417
+ searchQuery={searchInput}
418
+ onSearchChange={(value) => setSearchInput(value)}
419
+ onSearch={() => setPage(1)}
420
+ placeholder={t('filters.searchPlaceholder')}
421
+ />
422
+
423
+ {isLoading ? (
424
+ <div className="space-y-3 p-4">
425
+ {Array.from({ length: 5 }).map((_, i) => (
426
+ <Skeleton key={i} className="h-14 w-full" />
427
+ ))}
428
+ </div>
429
+ ) : paginate.data.length === 0 ? (
430
+ <EmptyState
431
+ icon={<Film className="h-12 w-12" />}
432
+ title={t('empty.title')}
433
+ description={t('empty.description')}
434
+ actionLabel={t('actions.create')}
435
+ actionIcon={<Plus className="mr-2 h-4 w-4" />}
436
+ onAction={openCreateSheet}
437
+ />
438
+ ) : (
439
+ <div className="overflow-x-auto">
440
+ <Table>
441
+ <TableHeader>
442
+ <TableRow>
443
+ <TableHead>{t('table.name')}</TableHead>
444
+ <TableHead>{t('table.ffmpegParams')}</TableHead>
445
+ <TableHead>{t('table.status')}</TableHead>
446
+ <TableHead className="w-10" />
447
+ </TableRow>
448
+ </TableHeader>
449
+ <TableBody>
450
+ {paginate.data.map((profile) => (
451
+ <TableRow
452
+ key={profile.id}
453
+ className="cursor-pointer"
454
+ onDoubleClick={() => openEditSheet(profile)}
455
+ >
456
+ <TableCell>
457
+ <span className="font-medium text-sm">{profile.name}</span>
458
+ </TableCell>
459
+ <TableCell className="max-w-xs">
460
+ <span className="font-mono text-xs text-muted-foreground truncate block max-w-xs">
461
+ {profile.ffmpeg_params}
462
+ </span>
463
+ </TableCell>
464
+ <TableCell>
465
+ <Badge
466
+ variant="outline"
467
+ className={cn(
468
+ 'border px-2.5 py-1 text-xs font-medium',
469
+ profile.status === 'active'
470
+ ? 'border-green-500/20 bg-green-500/10 text-green-600'
471
+ : 'border-gray-500/20 bg-gray-500/10 text-gray-600'
472
+ )}
473
+ >
474
+ {profile.status === 'active'
475
+ ? t('status.active')
476
+ : t('status.inactive')}
477
+ </Badge>
478
+ </TableCell>
479
+ <TableCell className="text-right">
480
+ <div className="flex items-center justify-end gap-1">
481
+ <Tooltip>
482
+ <TooltipTrigger asChild>
483
+ <Button
484
+ variant="ghost"
485
+ size="icon"
486
+ className="h-8 w-8 cursor-pointer"
487
+ onClick={(event) => {
488
+ event.stopPropagation();
489
+ openEditSheet(profile);
490
+ }}
491
+ aria-label={t('actions.edit')}
492
+ >
493
+ <Pencil className="h-4 w-4" />
494
+ </Button>
495
+ </TooltipTrigger>
496
+ <TooltipContent>{t('actions.edit')}</TooltipContent>
497
+ </Tooltip>
498
+
499
+ <Tooltip>
500
+ <TooltipTrigger asChild>
501
+ <Button
502
+ variant="ghost"
503
+ size="icon"
504
+ className={cn(
505
+ 'h-8 w-8 cursor-pointer',
506
+ profile.status === 'active'
507
+ ? 'text-amber-600 hover:text-amber-600'
508
+ : 'text-emerald-600 hover:text-emerald-600'
509
+ )}
510
+ onClick={(event) => {
511
+ event.stopPropagation();
512
+ void handleQuickStatusToggle(profile);
513
+ }}
514
+ aria-label={
515
+ profile.status === 'active'
516
+ ? t('status.inactive')
517
+ : t('status.active')
518
+ }
519
+ disabled={togglingProfileId === profile.id}
520
+ >
521
+ {profile.status === 'active' ? (
522
+ <PauseCircle className="h-4 w-4" />
523
+ ) : (
524
+ <PlayCircle className="h-4 w-4" />
525
+ )}
526
+ </Button>
527
+ </TooltipTrigger>
528
+ <TooltipContent>
529
+ {profile.status === 'active'
530
+ ? t('status.inactive')
531
+ : t('status.active')}
532
+ </TooltipContent>
533
+ </Tooltip>
534
+
535
+ <Tooltip>
536
+ <TooltipTrigger asChild>
537
+ <Button
538
+ variant="ghost"
539
+ size="icon"
540
+ className="h-8 w-8 cursor-pointer text-destructive hover:text-destructive"
541
+ onClick={(event) => {
542
+ event.stopPropagation();
543
+ setProfileToDelete(profile);
544
+ setDeleteDialogOpen(true);
545
+ }}
546
+ aria-label={t('actions.delete')}
547
+ >
548
+ <Trash2 className="h-4 w-4" />
549
+ </Button>
550
+ </TooltipTrigger>
551
+ <TooltipContent>{t('actions.delete')}</TooltipContent>
552
+ </Tooltip>
553
+ </div>
554
+ </TableCell>
555
+ </TableRow>
556
+ ))}
557
+ </TableBody>
558
+ </Table>
559
+ </div>
560
+ )}
561
+
562
+ <PaginationFooter
563
+ currentPage={page}
564
+ pageSize={pageSize}
565
+ totalItems={paginate.total}
566
+ onPageChange={setPage}
567
+ onPageSizeChange={(nextPageSize) => {
568
+ setPageSize(nextPageSize);
569
+ setPage(1);
570
+ }}
571
+ pageSizeOptions={[6, 12, 24]}
572
+ />
573
+
574
+ <ProfileFormSheet
575
+ open={sheetOpen}
576
+ onOpenChange={setSheetOpen}
577
+ profileToEdit={profileToEdit}
578
+ onSaved={handleSaved}
579
+ />
580
+
581
+ <AlertDialog open={deleteDialogOpen} onOpenChange={setDeleteDialogOpen}>
582
+ <AlertDialogContent>
583
+ <AlertDialogHeader>
584
+ <AlertDialogTitle>{t('deleteDialog.title')}</AlertDialogTitle>
585
+ <AlertDialogDescription>
586
+ {t('deleteDialog.description', {
587
+ name: profileToDelete?.name ?? '',
588
+ })}
589
+ </AlertDialogDescription>
590
+ </AlertDialogHeader>
591
+ <div className="flex justify-end gap-2">
592
+ <AlertDialogCancel disabled={isDeleting}>
593
+ {t('actions.cancel')}
594
+ </AlertDialogCancel>
595
+ <AlertDialogAction
596
+ onClick={handleDeleteConfirm}
597
+ disabled={isDeleting}
598
+ className="bg-red-600 hover:bg-red-700"
599
+ >
600
+ {isDeleting ? t('actions.deleting') : t('actions.delete')}
601
+ </AlertDialogAction>
602
+ </div>
603
+ </AlertDialogContent>
604
+ </AlertDialog>
605
+ </Page>
606
+ );
607
+ }