@sigmaott/base-library-next 2.1.9

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 (71) hide show
  1. package/README.md +1 -0
  2. package/locales/en.yaml +289 -0
  3. package/locales/vi.yaml +294 -0
  4. package/nuxt.config.ts +18 -0
  5. package/package.json +33 -0
  6. package/public/routes.json +34 -0
  7. package/src/api/axios.ts +3 -0
  8. package/src/api/index.ts +86 -0
  9. package/src/api-client-library/.openapi-generator/FILES +20 -0
  10. package/src/api-client-library/.openapi-generator/VERSION +1 -0
  11. package/src/api-client-library/.openapi-generator-ignore +23 -0
  12. package/src/api-client-library/api/health-api.ts +119 -0
  13. package/src/api-client-library/api/presets-api.ts +599 -0
  14. package/src/api-client-library/api/profiles-api.ts +676 -0
  15. package/src/api-client-library/api.ts +20 -0
  16. package/src/api-client-library/base.ts +72 -0
  17. package/src/api-client-library/common.ts +150 -0
  18. package/src/api-client-library/configuration.ts +101 -0
  19. package/src/api-client-library/git_push.sh +57 -0
  20. package/src/api-client-library/index.ts +18 -0
  21. package/src/api-client-library/models/create-preset-dto.ts +223 -0
  22. package/src/api-client-library/models/create-profile-dto.ts +45 -0
  23. package/src/api-client-library/models/health-controller-get-health200-response-info-value.ts +32 -0
  24. package/src/api-client-library/models/health-controller-get-health200-response.ts +51 -0
  25. package/src/api-client-library/models/health-controller-get-health503-response.ts +51 -0
  26. package/src/api-client-library/models/index.ts +7 -0
  27. package/src/api-client-library/models/update-preset-dto.ts +223 -0
  28. package/src/api-client-library/models/update-profile-dto.ts +45 -0
  29. package/src/components/MediaSelection.vue +40 -0
  30. package/src/components/PresetModify.vue +154 -0
  31. package/src/components/PresetTable.vue +114 -0
  32. package/src/components/ProfileAllList.vue +137 -0
  33. package/src/components/ProfileFormModal.vue +79 -0
  34. package/src/components/ProfileModify.vue +152 -0
  35. package/src/components/ProfileTable.vue +68 -0
  36. package/src/components/WatermarkDraggableItem.vue +88 -0
  37. package/src/components/channel/ConfigWatermarkItem.vue +239 -0
  38. package/src/components/channel/WatermarkPreview.vue +19 -0
  39. package/src/components/common/Vue3DraggableResizable/Container.vue +71 -0
  40. package/src/components/common/Vue3DraggableResizable/index.vue +1327 -0
  41. package/src/components/common/Vue3DraggableResizable/utils/dom.js +63 -0
  42. package/src/components/common/Vue3DraggableResizable/utils/fns.js +37 -0
  43. package/src/components/common/VueDraggableResizable/dom.js +63 -0
  44. package/src/components/common/VueDraggableResizable/fns.js +37 -0
  45. package/src/components/common/VueDraggableResizable/index.vue +958 -0
  46. package/src/components/preset/ConfigItem.vue +956 -0
  47. package/src/components/profile/ConfigItem.vue +765 -0
  48. package/src/components/profile/TableColumns.vue +137 -0
  49. package/src/components/shared/AudioInfoViewer.vue +101 -0
  50. package/src/components/shared/MediaInfoViewer.vue +249 -0
  51. package/src/components/shared/MediaInfoViewerSmall.vue +105 -0
  52. package/src/components/shared/PopoverProfile.vue +17 -0
  53. package/src/components/shared/VideoInfoViewer.vue +136 -0
  54. package/src/components/shared/fileSizeFilter.ts +26 -0
  55. package/src/composables/preset.ts +141 -0
  56. package/src/public/apple-touch-icon-180x180.png +0 -0
  57. package/src/public/build-time.json +1 -0
  58. package/src/public/favicon.ico +0 -0
  59. package/src/public/favicon.svg +15 -0
  60. package/src/public/logo.png +0 -0
  61. package/src/public/logo.svg +9 -0
  62. package/src/public/maskable-icon-512x512.png +0 -0
  63. package/src/public/pwa-192x192.png +0 -0
  64. package/src/public/pwa-512x512.png +0 -0
  65. package/src/public/pwa-64x64.png +0 -0
  66. package/src/public/routes.json +87 -0
  67. package/src/utils/common.ts +175 -0
  68. package/src/utils/config.ts +19 -0
  69. package/src/utils/preset.ts +353 -0
  70. package/src/utils/profile.ts +30 -0
  71. package/tsconfig.json +3 -0
@@ -0,0 +1,136 @@
1
+ <script setup lang="ts">
2
+ import { fileSizeFilter } from './fileSizeFilter'
3
+
4
+ defineProps<{ video: any }>()
5
+ </script>
6
+
7
+ <template>
8
+ <div class="item-list-row flex justify-center">
9
+ <ul class="flex flex-wrap justify-center">
10
+ <li title="Name">
11
+ <el-icon v-if="video.codec !== 'copy'" class="text-primary" name="video-camera-solid">
12
+ <div class="i-ep:camera-filled text-primary" />
13
+ </el-icon>&nbsp;
14
+ <el-icon v-if="video.codec === 'copy'" class="text-indigo-500">
15
+ <div class="i-ep:document-copy text-indigo-500" />
16
+ </el-icon>
17
+ <span v-if="video.codec === 'copy'">&nbsp;</span>
18
+ <span class="item__name" :title="video.name">{{
19
+ (video && video.name) || '-'
20
+ }}</span>
21
+ </li>
22
+ <span class="text-gray-400">&nbsp;|&nbsp;</span>
23
+ <li title="Codec">
24
+ <el-icon>
25
+ <div class="i-ep:grid text-indigo-500" />
26
+ </el-icon>&nbsp;
27
+ <span>{{ video.codec }}</span>
28
+ </li>
29
+ <span class="text-gray-400">&nbsp;|&nbsp;</span>
30
+ <li title="Resolution">
31
+ <el-icon name="crop">
32
+ <div class="i-ep:crop text-pink-600" />
33
+ </el-icon>&nbsp;
34
+ <span>{{ video.width }}x{{ video.height }}</span>
35
+ </li>
36
+ <span class="text-gray-400">&nbsp;|&nbsp;</span>
37
+ <li title="Bitrate">
38
+ <el-icon>
39
+ <div class="i-ep:scale-to-original text-teal-600" />
40
+ </el-icon>&nbsp;
41
+ <span>{{ video.bitrate ? fileSizeFilter(video.bitrate) : fileSizeFilter(video.videoBitrate) }}/s</span>
42
+ </li>
43
+ <span class="text-gray-400">&nbsp;|&nbsp;</span>
44
+ <li title="FPS">
45
+ <el-icon>
46
+ <div class="i-ep:sort text-green-600" />
47
+ </el-icon>&nbsp;
48
+ <span>{{ video.fps }} fps</span>
49
+ </li>
50
+ <span class="text-gray-400">&nbsp;|&nbsp;</span>
51
+ <li title="CBR">
52
+ <el-icon>
53
+ <div class="i-ep:switch-button" :class="!!video.cbr ? 'text-success' : 'text-gray-400'" />
54
+ </el-icon>
55
+ &nbsp;
56
+ <span> cbr </span>
57
+ </li>
58
+ <span class="text-gray-400">&nbsp;|&nbsp;</span>
59
+ <li title="HDR">
60
+ <el-icon>
61
+ <div class="i-carbon:distribute-horizontal-right text-success" :class="video.hdr ? 'text-success' : 'text-gray-400'" />
62
+ </el-icon>
63
+ &nbsp;
64
+ <span> hdr </span>
65
+ </li>
66
+ <span class="text-gray-400">&nbsp;|&nbsp;</span>
67
+ <li title="Pixel format">
68
+ <el-icon>
69
+ <div class="i-ep:picture" :class="!video.pixelFormat || 'text-indigo-500'" />
70
+ </el-icon>&nbsp;
71
+ <span>{{ video.pixelFormat || '-' }}</span>
72
+ </li>
73
+ <span class="text-gray-400">&nbsp;|&nbsp;</span>
74
+ <li title="Scale type">
75
+ <el-icon>
76
+ <div class="i-ep:scale-to-original" :class="!video.scaleType || 'text-pink-600'" />
77
+ </el-icon>&nbsp;
78
+ <span>{{ video.scaleType || '-' }}</span>
79
+ </li>
80
+ <span class="text-gray-400">&nbsp;|&nbsp;</span>
81
+ <li title="Constant quality">
82
+ <el-icon>
83
+ <div class="i-ep:star-filled" :class="!video.cq || 'text-teal-600'" />
84
+ </el-icon>&nbsp;
85
+ <span>{{ video.cq || '-' }}</span>
86
+ </li>
87
+ <span class="text-gray-400">&nbsp;|&nbsp;</span>
88
+ <li title="Interlaced mode">
89
+ <el-icon>
90
+ <div class="i-ep:switch-button" :class="video.interlaced ? 'text-success' : 'text-gray-400'" />
91
+ </el-icon>&nbsp;
92
+ <span> interlaced </span>
93
+ </li>
94
+ </ul>
95
+ <div v-if="$slots && $slots.anchor" class="anchor">
96
+ <slot name="anchor" />
97
+ </div>
98
+ </div>
99
+ </template>
100
+
101
+ <style lang="scss" scoped>
102
+ .item-list-row {
103
+ margin: 5px;
104
+ padding: 5px;
105
+ position: relative;
106
+ border: 1px solid rgba(64, 158, 255, 0.4);
107
+ border-radius: 4px;
108
+
109
+ ul {
110
+ li {
111
+ white-space: nowrap;
112
+ display: flex;
113
+ align-items: center;
114
+ }
115
+
116
+ .item__name {
117
+ max-width: 85px;
118
+ overflow: hidden;
119
+ white-space: nowrap;
120
+ text-overflow: ellipsis;
121
+ }
122
+ }
123
+
124
+ .anchor {
125
+ padding: 5px;
126
+ margin: -5px -5px -5px 5px;
127
+ display: flex;
128
+ flex-direction: column;
129
+ border-left: 1px solid;
130
+ border-color: inherit;
131
+ background-color: rgba(64, 158, 255, 0.2);
132
+ border-top-right-radius: inherit;
133
+ border-bottom-right-radius: inherit;
134
+ }
135
+ }
136
+ </style>
@@ -0,0 +1,26 @@
1
+ export function fileSizeFilter(size: number) {
2
+ if (Number.isNaN(size))
3
+ size = 0
4
+
5
+ if (size < 1024)
6
+ return `${size} Kb`
7
+
8
+ size /= 1024
9
+
10
+ if (size < 1024)
11
+ return `${size.toFixed(2)} KB`
12
+
13
+ size /= 1024
14
+
15
+ if (size < 1024)
16
+ return `${size.toFixed(2)} MB`
17
+
18
+ size /= 1024
19
+
20
+ if (size < 1024)
21
+ return `${size.toFixed(2)} GB`
22
+
23
+ size /= 1024
24
+
25
+ return `${size.toFixed(2)} TB`
26
+ }
@@ -0,0 +1,141 @@
1
+ import { presetsApi, profilesApi } from '../api'
2
+ import type { CreatePresetDto, CreateProfileDto, UpdatePresetDto, UpdateProfileDto } from '../api-client-library'
3
+
4
+ export function usePresetMutation() {
5
+ const { t } = useI18n()
6
+ const queryClient = useQueryClient()
7
+
8
+ const createPresetMutation = useMutation(
9
+ async (createPresetDto: CreatePresetDto) =>
10
+ (await presetsApi.presetsControllerCreate({ createPresetDto })).data,
11
+ {
12
+ onSettled() {
13
+ queryClient.invalidateQueries([QueryKeys.LibraryPreset])
14
+ },
15
+ },
16
+ )
17
+
18
+ const updatePresetMutation = useMutation(
19
+ async ({ id, updatePresetDto }: { id: string; updatePresetDto: UpdatePresetDto }) =>
20
+ (await presetsApi.presetsControllerUpdate({ id, updatePresetDto })).data,
21
+ {
22
+ onSettled() {
23
+ queryClient.invalidateQueries([QueryKeys.LibraryPreset])
24
+ },
25
+ },
26
+ )
27
+
28
+ // const deleteSession = useMutation(
29
+ // async (id: string) => (await sessionsApi.sessionControllerDelete({ id })).data,
30
+ // {
31
+ // onSettled: () => {
32
+ // queryClient.invalidateQueries([QueryKeys.SESSION])
33
+ // },
34
+ // },
35
+ // )
36
+
37
+ // const renameSession = useMutation(
38
+ // async ({ id, renameSessionDto }: { id: string; renameSessionDto: RenameSessionDto }) => (await sessionsApi.sessionControllerRename({ id, renameSessionDto })).data,
39
+ // {
40
+ // onSettled: () => {
41
+ // queryClient.invalidateQueries([QueryKeys.SESSION])
42
+ // },
43
+ // onSuccess() {
44
+ // ElMessage({
45
+ // type: 'success',
46
+ // message: t('LiveChannel.message.sessionSuccessfullyRename'),
47
+ // })
48
+ // },
49
+ // },
50
+ // )
51
+
52
+ // function handleDeleteSession(id: string) {
53
+ // ElMessageBox.confirm(
54
+ // t('LiveChannel.message.confirmDeleteSession'),
55
+ // `${t('LiveChannel.label.deleteSession')}?`,
56
+ // {
57
+ // confirmButtonText: t('Actions.delete'),
58
+ // cancelButtonText: t('Actions.cancel'),
59
+ // type: 'warning',
60
+ // },
61
+ // )
62
+ // .then(async () => {
63
+ // await deleteSession.mutateAsync(id)
64
+ // ElMessage({
65
+ // message: t('LiveChannel.message.successDelete'),
66
+ // type: 'success',
67
+ // })
68
+ // })
69
+ // }
70
+
71
+ // function handleFinishSession(id: string, cb?: () => void) {
72
+ // ElMessageBox.confirm(
73
+ // t('LiveChannel.message.confirmFinishSession'),
74
+ // t('LiveChannel.message.warning'),
75
+ // {
76
+ // confirmButtonText: t('Actions.ok'),
77
+ // cancelButtonText: t('Actions.cancel'),
78
+ // type: 'warning',
79
+ // },
80
+ // )
81
+ // .then(async () => {
82
+ // await finishSession.mutateAsync(id)
83
+ // if (cb)
84
+ // cb()
85
+ // ElMessage({
86
+ // type: 'success',
87
+ // message: t('LiveChannel.message.sessionSuccessfullyTerminated'),
88
+ // })
89
+ // })
90
+ // .catch(() => {})
91
+ // }
92
+
93
+ const isMutationLoading = computed(() =>
94
+ createPresetMutation.isLoading.value
95
+ || updatePresetMutation.isLoading.value,
96
+ )
97
+
98
+ return {
99
+ createPresetMutation,
100
+ updatePresetMutation,
101
+ isMutationLoading,
102
+ }
103
+ }
104
+
105
+ export function useProfileMutation() {
106
+ const { t } = useI18n()
107
+ const queryClient = useQueryClient()
108
+
109
+ const createProfileMutation
110
+ = useMutation(
111
+ async (createProfileDto: CreateProfileDto) =>
112
+ (await profilesApi.httpProfilesControllerCreate({ createProfileDto })).data,
113
+ {
114
+ onSettled() {
115
+ queryClient.invalidateQueries([QueryKeys.LibraryProfile])
116
+ },
117
+ },
118
+ )
119
+
120
+ const updateProfileMutation
121
+ = useMutation(
122
+ async ({ id, updateProfileDto }: { id: string; updateProfileDto: UpdateProfileDto }) =>
123
+ (await profilesApi.httpProfilesControllerUpdate({ updateProfileDto, id })).data,
124
+ {
125
+ onSettled() {
126
+ queryClient.invalidateQueries([QueryKeys.LibraryProfile])
127
+ },
128
+ },
129
+ )
130
+
131
+ const isMutationLoading = computed(() =>
132
+ createProfileMutation.isLoading.value
133
+ || updateProfileMutation.isLoading.value,
134
+ )
135
+
136
+ return {
137
+ createProfileMutation,
138
+ updateProfileMutation,
139
+ isMutationLoading,
140
+ }
141
+ }
@@ -0,0 +1 @@
1
+ 1752806726088
Binary file
@@ -0,0 +1,15 @@
1
+ <svg version="1.1" id="Group_1_copy_11" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px"
2
+ y="0px" viewBox="0 0 286.3 286.3" style="enable-background:new 0 0 286.3 286.3;" xml:space="preserve">
3
+ <style type="text/css">
4
+ .st0{fill:#D71921;}
5
+ </style>
6
+ <g>
7
+ <path class="st0" d="M171.6,139.6l-35.9-23.7c-5.2-3.4-12,0.3-12,6.5v47.4c0,6.2,6.9,9.9,12,6.5l35.9-23.7
8
+ C176.3,149.5,176.3,142.7,171.6,139.6z"/>
9
+ <path class="st0" d="M282.5,126.4h-12.4h-32.4h-23C206,95,177.4,71.9,143.3,71.9S80.6,95,72.1,126.4H48.8c8-45,47.1-79.1,94.4-79.1
10
+ c35.6,0,66.7,19.4,83.3,48.3h48.8c-19.5-54-71.3-92.6-132-92.6C65.6,2.8,2.8,65.7,2.8,143.2c0,7,0.5,13.9,1.5,20.6h11.9h33.2h22.3
11
+ c8.1,32.1,37.1,56,71.6,56s63.6-23.9,71.6-56h21.9c-9.4,43.1-47.8,75.4-93.7,75.4c-34,0-63.9-17.7-81-44.5H12.6
12
+ c20.5,52,71.2,88.9,130.6,88.9c77.5,0,140.3-62.8,140.3-140.3C283.5,137.5,283.1,131.9,282.5,126.4z M143.3,197.4
13
+ c-28.4,0-51.6-23.1-51.6-51.6c0-28.4,23.1-51.6,51.6-51.6s51.6,23.1,51.6,51.6C194.9,174.2,171.8,197.4,143.3,197.4z"/>
14
+ </g>
15
+ </svg>
Binary file
@@ -0,0 +1,9 @@
1
+ <svg width="192" height="192" viewBox="0 0 192 192" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
2
+ <rect width="192" height="192" fill="url(#pattern0)"/>
3
+ <defs>
4
+ <pattern id="pattern0" patternContentUnits="objectBoundingBox" width="1" height="1">
5
+ <use xlink:href="#image0_8_2" transform="scale(0.0034965)"/>
6
+ </pattern>
7
+ <image id="image0_8_2" width="286" height="286" xlink:href=""/>
8
+ </defs>
9
+ </svg>
Binary file
Binary file
Binary file
@@ -0,0 +1,87 @@
1
+ [
2
+ {
3
+ "name": "channels",
4
+ "path": "/channels",
5
+ "meta": {
6
+ "title": "Channels",
7
+ "key": "SSAI.channels",
8
+ "icon": "i-carbon:airplay",
9
+ "requiresAuth": false,
10
+ "keepAlive": true,
11
+ "notAsMenu": false,
12
+ "order": 1
13
+ }
14
+ },
15
+ {
16
+ "name": "ads",
17
+ "path": "/ads",
18
+ "meta": {
19
+ "title": "Ads List",
20
+ "key": "SSAI.adsList",
21
+ "icon": "i-carbon:layers",
22
+ "requiresAuth": false,
23
+ "keepAlive": true,
24
+ "notAsMenu": false,
25
+ "order": 2
26
+ }
27
+ },
28
+ {
29
+ "name": "statistic",
30
+ "path": "/statistic",
31
+ "meta": {
32
+ "title": "Statistic",
33
+ "key": "SSAI.Statistic",
34
+ "icon": "i-carbon:analytics",
35
+ "requiresAuth": false,
36
+ "keepAlive": false,
37
+ "notAsMenu": false,
38
+ "order": 3
39
+ },
40
+ "children": [
41
+ {
42
+ "name": "statistic-index-monitor-system",
43
+ "path": "/statistic/monitor-system",
44
+ "meta": {
45
+ "title": "Monitor System",
46
+ "key": "SSAI.statistic.monitorSystem",
47
+ "icon": "i-carbon:ibm-cloud-pak-system",
48
+ "requiresAuth": false,
49
+ "keepAlive": true
50
+ }
51
+ },
52
+ {
53
+ "name": "statistic-index-fillrate",
54
+ "path": "/statistic/fillrate",
55
+ "meta": {
56
+ "title": "Fill Rate",
57
+ "key": "SSAI.statistic.fillRate",
58
+ "icon": "i-carbon:bare-metal-server",
59
+ "requiresAuth": false,
60
+ "keepAlive": true
61
+ }
62
+ },
63
+ {
64
+ "name": "statistic-index-event-chart",
65
+ "path": "/statistic/event-chart",
66
+ "meta": {
67
+ "title": "Event Chart",
68
+ "key": "SSAI.statistic.eventChart",
69
+ "icon": "i-carbon:chart-column",
70
+ "requiresAuth": false,
71
+ "keepAlive": true
72
+ }
73
+ },
74
+ {
75
+ "name": "statistic-index-data-report",
76
+ "path": "/statistic/data-report",
77
+ "meta": {
78
+ "title": "Data Report",
79
+ "key": "SSAI.statistic.dataReport",
80
+ "icon": "i-carbon:report-data",
81
+ "requiresAuth": false,
82
+ "keepAlive": true
83
+ }
84
+ }
85
+ ]
86
+ }
87
+ ]
@@ -0,0 +1,175 @@
1
+ import { objectMap } from '@antfu/utils'
2
+
3
+ export function emptyStr2Undefined(obj: Record<string, any>): Record<string, any> {
4
+ return objectMap(obj, (k, v) => !v ? undefined : [k, v])
5
+ }
6
+
7
+ /**
8
+ * Generate rules for validation form that contain fields are sent to api
9
+ * @param {Object} laws
10
+ */
11
+ export function genApiValidationRules(laws) {
12
+ const required = laws?.required || []
13
+ const props = laws?.properties || {}
14
+ const rules = {}
15
+
16
+ function generateRequiredRule(desc) {
17
+ return {
18
+ required: required.includes(desc.field),
19
+ message: 'Shouldn\'t be empty',
20
+ trigger: ['change', 'blur'],
21
+ }
22
+ }
23
+
24
+ function rulesForNumber(rule) {
25
+ const rules = []
26
+
27
+ const required = generateRequiredRule(rule)
28
+ rules.push(required)
29
+
30
+ function isNumber(value) {
31
+ return typeof value === 'number'
32
+ }
33
+
34
+ function isValidMin(value, min) {
35
+ return value >= min
36
+ }
37
+
38
+ function isValidMax(value, max) {
39
+ return value <= max
40
+ }
41
+
42
+ function validate(currentRule, value, callback) {
43
+ if (!isNumber(value)) {
44
+ if (value)
45
+ callback(new Error('Should be a number'))
46
+
47
+ else
48
+ callback()
49
+
50
+ return
51
+ }
52
+
53
+ if ((rule.minimum || rule.minimum === 0) && !isValidMin(value, rule.minimum)) {
54
+ callback(new Error(`Should be greater than or equal ${rule.minimum}`))
55
+ return
56
+ }
57
+ if (rule.maximum && !isValidMax(value, rule.maximum)) {
58
+ callback(new Error(`Should be less than or equal ${rule.maximum}`))
59
+ return
60
+ }
61
+ if (rule.enum && Array.isArray(rule.enum) && !rule.enum.includes(value)) {
62
+ callback(new Error(`Must be one of values: ${rule.num.join(', ')}`))
63
+ return
64
+ }
65
+ callback()
66
+ }
67
+
68
+ const tmpRule = {
69
+ validator: validate,
70
+ trigger: ['blur', 'change'],
71
+ }
72
+ rules.push(tmpRule)
73
+
74
+ return rules
75
+ }
76
+
77
+ function rulesForString(rule) {
78
+ const rules = []
79
+
80
+ const required = generateRequiredRule(rule)
81
+ rules.push(required)
82
+
83
+ function validate(currentRule, value, callback) {
84
+ if (rule.length && !value.length === rule.length)
85
+ return callback(new Error(`${rule.label} must be ${rule.length} characters`))
86
+
87
+ if (rule.minLength && value.length < rule.minLength)
88
+ return callback(new Error(`Min ${rule.minLength} character(s)`))
89
+
90
+ if (rule.maxLength && value.length > rule.maxLength)
91
+ return callback(new Error(`Max ${rule.maxLength} character(s)`))
92
+
93
+ if (rule.pattern) {
94
+ const pattern = rule.pattern
95
+ const patternStr = pattern.slice(1, -1)
96
+ const regex = new RegExp(patternStr)
97
+ if (!regex.test(value))
98
+ return callback(new Error(`Invalid format ${rule.pattern}`))
99
+ }
100
+ if (rule.enum && Array.isArray(rule.enum)) {
101
+ if (value && !rule.enum.includes(value))
102
+ return callback(new Error(`Must be one of values: ${rule.enum.join(', ')}`))
103
+ }
104
+ return callback()
105
+ }
106
+
107
+ rules.push({
108
+ validator: validate,
109
+ trigger: ['change', 'blur'],
110
+ })
111
+
112
+ return rules
113
+ }
114
+
115
+ function rulesForBoolean(rule) {
116
+ const rules = []
117
+
118
+ const required = generateRequiredRule(rule)
119
+ rules.push(required)
120
+
121
+ function isBoolean(value) {
122
+ return typeof value === 'boolean'
123
+ }
124
+
125
+ function validate(currentRule, value, callback) {
126
+ if (!isBoolean(value))
127
+ callback(new Error('Should be a boolean value'))
128
+
129
+ else
130
+ callback()
131
+ }
132
+
133
+ const tmpRule = {
134
+ validator: validate,
135
+ trigger: ['change', 'blur'],
136
+ }
137
+ rules.push(tmpRule)
138
+
139
+ return rules
140
+ }
141
+
142
+ for (const field in props) {
143
+ const desc = props[field]
144
+ if (desc.$ref) {
145
+ // Check if rule is a reference
146
+ continue
147
+ }
148
+ else {
149
+ desc.field = field
150
+ if (desc.title)
151
+ desc.title = field
152
+
153
+ const fieldType = desc.type
154
+
155
+ switch (fieldType) {
156
+ case 'string': {
157
+ const rule = rulesForString(desc)
158
+ rules[field] = rule
159
+ break
160
+ }
161
+ case 'number': {
162
+ const rule = rulesForNumber(desc)
163
+ rules[field] = rule
164
+ break
165
+ }
166
+ case 'boolean': {
167
+ const rule = rulesForBoolean(desc)
168
+ rules[field] = rule
169
+ break
170
+ }
171
+ }
172
+ }
173
+ }
174
+ return rules
175
+ }
@@ -0,0 +1,19 @@
1
+ export const VIDEO_CODECS = {
2
+ H264: 'h264',
3
+ HEVC: 'hevc',
4
+ COPY: 'copy',
5
+ }
6
+
7
+ export const AUDIO_CODECS = {
8
+ AAC: 'aac',
9
+ AC3: 'ac3',
10
+ EAC3: 'eac3',
11
+ MP2: 'mp2',
12
+ COPY: 'copy',
13
+ }
14
+
15
+ export const DATA_CODECS = {
16
+ SCTE35: 'scte35',
17
+ ID3: 'id3',
18
+ COPY: 'copy',
19
+ }