Showing
9 changed files
with
251 additions
and
23 deletions
... | ... | @@ -13,11 +13,6 @@ |
13 | 13 | </n-button> |
14 | 14 | </setting-item> |
15 | 15 | </setting-item-box> |
16 | - <setting-item-box name="排布"> | |
17 | - <setting-item> | |
18 | - <n-select v-model:value="optionData.displayMode" size="small" :options="displayModeOptions"></n-select> | |
19 | - </setting-item> | |
20 | - </setting-item-box> | |
21 | 16 | </CollapseItem> |
22 | 17 | </template> |
23 | 18 | |
... | ... | @@ -32,20 +27,4 @@ defineProps({ |
32 | 27 | required: true |
33 | 28 | } |
34 | 29 | }) |
35 | - | |
36 | -// 旋转方式 | |
37 | -const displayModeOptions = [ | |
38 | - { | |
39 | - value: 'singleGrid', | |
40 | - label: '默认' | |
41 | - }, | |
42 | - { | |
43 | - value: 'fourGrid', | |
44 | - label: '四宫格' | |
45 | - }, | |
46 | - { | |
47 | - value: 'nineGrid', | |
48 | - label: '九宫格' | |
49 | - } | |
50 | -] | |
51 | 30 | </script> | ... | ... |
1 | +<template> | |
2 | + <video | |
3 | + crossOrigin="anonymous" | |
4 | + :id="`my-player${index}`" | |
5 | + ref="videoRef" | |
6 | + class="video-js my-video vjs-theme-city vjs-big-play-centered" | |
7 | + > | |
8 | + <source :src="sourceSrc" /> | |
9 | + </video> | |
10 | +</template> | |
11 | +<script setup lang="ts"> | |
12 | +import { onMounted, ref, onUnmounted, watch } from 'vue' | |
13 | +import videojs from 'video.js' | |
14 | +import type { VideoJsPlayerOptions } from 'video.js' | |
15 | +import 'video.js/dist/video-js.min.css' | |
16 | + | |
17 | +const props = defineProps({ | |
18 | + sourceSrc: { | |
19 | + type: String | |
20 | + }, | |
21 | + name: { | |
22 | + type: String | |
23 | + }, | |
24 | + avatar: { | |
25 | + type: String | |
26 | + }, | |
27 | + index: { | |
28 | + type: Number | |
29 | + } | |
30 | +}) | |
31 | + | |
32 | +// video标签 | |
33 | +const videoRef = ref<HTMLElement | null>(null) | |
34 | + | |
35 | +// video实例对象 | |
36 | +let videoPlayer: videojs.Player | null = null | |
37 | + | |
38 | +//options配置 | |
39 | +const options: VideoJsPlayerOptions = { | |
40 | + language: 'zh-CN', // 设置语言 | |
41 | + controls: true, // 是否显示控制条 | |
42 | + preload: 'auto', // 预加载 | |
43 | + autoplay: true, // 是否自动播放 | |
44 | + fluid: false, // 自适应宽高 | |
45 | + poster: props?.avatar || '', | |
46 | + src: props?.sourceSrc || '', // 要嵌入的视频源的源 URL | |
47 | + muted: true, | |
48 | + userActions: { | |
49 | + hotkeys: true | |
50 | + } | |
51 | +} | |
52 | + | |
53 | +// 初始化videojs | |
54 | +const initVideo = () => { | |
55 | + if (videoRef.value) { | |
56 | + // 创建 video 实例 | |
57 | + videoPlayer = videojs(videoRef.value, options) | |
58 | + } | |
59 | +} | |
60 | + | |
61 | +watch( | |
62 | + () => props.sourceSrc, | |
63 | + (newData: any) => { | |
64 | + // props.sourceSrc = newData | |
65 | + videoPlayer?.src(newData) as any | |
66 | + videoPlayer?.play() | |
67 | + }, | |
68 | + { | |
69 | + immediate: true | |
70 | + } | |
71 | +) | |
72 | + | |
73 | +onMounted(() => { | |
74 | + initVideo() | |
75 | +}) | |
76 | + | |
77 | +onUnmounted(() => { | |
78 | + handleVideoDispose() | |
79 | +}) | |
80 | + | |
81 | +//播放 | |
82 | +const handleVideoPlay = () => videoPlayer?.play() | |
83 | + | |
84 | +const handleVideoDispose = () => videoPlayer?.dispose() && videoPlayer?.pause() | |
85 | +//暂停 | |
86 | +defineExpose({ | |
87 | + handleVideoPlay, | |
88 | + handleVideoDispose | |
89 | +}) | |
90 | +</script> | |
91 | + | |
92 | +<style lang="scss" scoped> | |
93 | +.my-video { | |
94 | + width: 100%; | |
95 | + height: 100%; | |
96 | +} | |
97 | +</style> | ... | ... |
1 | +import { PublicConfigClass } from '@/packages/public' | |
2 | +import { CreateComponentType } from '@/packages/index.d' | |
3 | +import { SingleCameraConfig } from './index' | |
4 | +import cloneDeep from 'lodash/cloneDeep' | |
5 | + | |
6 | +export const option = { | |
7 | + dataset: [ | |
8 | + { | |
9 | + url: '' | |
10 | + } | |
11 | + ] as any, | |
12 | + // 自动播放的间隔(ms) | |
13 | + interval: 5000, | |
14 | + autoplay: true, | |
15 | + effect: 'slide', | |
16 | + displayMode: 'singleGrid' | |
17 | +} | |
18 | + | |
19 | +export default class Config extends PublicConfigClass implements CreateComponentType { | |
20 | + public key = SingleCameraConfig.key | |
21 | + public chartConfig = cloneDeep(SingleCameraConfig) | |
22 | + public option = cloneDeep(option) | |
23 | +} | ... | ... |
1 | +<template> | |
2 | + <CollapseItem name="播放器配置" :expanded="true"> | |
3 | + <setting-item-box name="源地址" :alone="true"> | |
4 | + <setting-item v-for="(item, index) in optionData.dataset" :key="index"> | |
5 | + <n-input-group> | |
6 | + <n-input v-model:value="item.url" size="small" placeholder="请输入源地址"></n-input> | |
7 | + </n-input-group> | |
8 | + </setting-item> | |
9 | + </setting-item-box> | |
10 | + </CollapseItem> | |
11 | +</template> | |
12 | + | |
13 | +<script setup lang="ts"> | |
14 | +import { PropType } from 'vue' | |
15 | +import { option } from './config' | |
16 | +import { CollapseItem, SettingItemBox, SettingItem } from '@/components/Pages/ChartItemSetting' | |
17 | + | |
18 | +defineProps({ | |
19 | + optionData: { | |
20 | + type: Object as PropType<typeof option>, | |
21 | + required: true | |
22 | + } | |
23 | +}) | |
24 | + | |
25 | +</script> | ... | ... |
1 | +import { ChartFrameEnum, ConfigType } from '@/packages/index.d' | |
2 | +import { EPackagesCategoryEnum } from '@/packages/components/external/types' | |
3 | +import { ChatCategoryEnum, ChatCategoryEnumName } from '../../index.d' | |
4 | +import { useWidgetKey } from '@/packages/external/useWidgetKey' | |
5 | + | |
6 | +const { key, chartKey, conKey } = useWidgetKey('SingleCamera') | |
7 | +export const SingleCameraConfig: ConfigType = { | |
8 | + key, | |
9 | + chartKey, | |
10 | + conKey, | |
11 | + title: '单个摄像头', | |
12 | + category: ChatCategoryEnum.MORE, | |
13 | + categoryName: ChatCategoryEnumName.MORE, | |
14 | + package: EPackagesCategoryEnum.COMPOSES, | |
15 | + chartFrame: ChartFrameEnum.NAIVE_UI, | |
16 | + image: 'camera.png' | |
17 | +} | ... | ... |
1 | +<template> | |
2 | + <div class="banner-box" ref="root"> | |
3 | + <n-grid x-gap="12" :y-gap="12" :cols="computedCols"> | |
4 | + <n-gi v-for="(item, index) in option.dataset" :key="index + item"> | |
5 | + <div class="camera-container"> | |
6 | + <CameraItem | |
7 | + ref="cameraRef" | |
8 | + :name="item.name" | |
9 | + :avatar="item.avatar" | |
10 | + :key="item + index" | |
11 | + :sourceSrc="item.url" | |
12 | + :index="index" | |
13 | + /> | |
14 | + </div> | |
15 | + </n-gi> | |
16 | + </n-grid> | |
17 | + </div> | |
18 | +</template> | |
19 | +<script setup lang="ts"> | |
20 | +import { PropType, toRefs, watch, shallowReactive, ref, computed } from 'vue' | |
21 | +import { CreateComponentType } from '@/packages/index.d' | |
22 | +import 'video.js/dist/video-js.min.css' | |
23 | +import { option as configOption } from './config' | |
24 | +import { CameraItem } from './components' | |
25 | + | |
26 | +const props = defineProps({ | |
27 | + chartConfig: { | |
28 | + type: Object as PropType<CreateComponentType>, | |
29 | + required: true | |
30 | + } | |
31 | +}) | |
32 | + | |
33 | +const { h } = toRefs(props.chartConfig.attr) | |
34 | + | |
35 | +const responsiveComputeValue = ref(0) | |
36 | + | |
37 | +const option = shallowReactive({ | |
38 | + dataset: configOption.dataset | |
39 | +}) | |
40 | + | |
41 | +const computedCols = computed(() => { | |
42 | + if (option.dataset.length <= 1) return 1 | |
43 | + if (option.dataset.length <= 4) return 2 | |
44 | + return 3 | |
45 | +}) | |
46 | + | |
47 | +const cameraRef = ref<InstanceType<typeof CameraItem>>() | |
48 | + | |
49 | +const responsive = (value: number) => { | |
50 | + responsiveComputeValue.value = value | |
51 | + if (option.dataset.length <= 2) responsiveComputeValue.value = value | |
52 | + if (option.dataset.length > 2 && option.dataset.length <= 4) responsiveComputeValue.value = value / 2.03 | |
53 | + if (option.dataset.length > 4 && option.dataset.length <= 9) responsiveComputeValue.value = value / 3.1 | |
54 | +} | |
55 | + | |
56 | +watch( | |
57 | + () => props.chartConfig.option.dataset, | |
58 | + newData => { | |
59 | + option.dataset = newData | |
60 | + responsive(h.value) | |
61 | + }, | |
62 | + { | |
63 | + immediate: true, | |
64 | + deep: true | |
65 | + } | |
66 | +) | |
67 | + | |
68 | +watch( | |
69 | + () => h.value, | |
70 | + newData => responsive(newData), | |
71 | + { | |
72 | + immediate: true | |
73 | + } | |
74 | +) | |
75 | +</script> | |
76 | + | |
77 | +<style lang="scss" scoped> | |
78 | +.banner-box { | |
79 | + .camera-container { | |
80 | + height: v-bind('`${responsiveComputeValue}px`'); | |
81 | + } | |
82 | +} | |
83 | +</style> | ... | ... |
... | ... | @@ -2,6 +2,7 @@ import { Title1Config } from './Title1/index' |
2 | 2 | import { Title2Config } from './Title2/index' |
3 | 3 | import { Title3Config } from './Title3/index' |
4 | 4 | import { CameraConfig } from './Camera/index' |
5 | +import { SingleCameraConfig } from './SingleCamera/index' | |
5 | 6 | import { ThreeDimensionalConfig } from './ThreeDimensional/index' |
6 | 7 | |
7 | -export default [Title1Config, Title2Config, Title3Config, CameraConfig, ThreeDimensionalConfig] | |
8 | +export default [Title1Config, Title2Config, Title3Config, CameraConfig, SingleCameraConfig, ThreeDimensionalConfig] | ... | ... |