123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591 |
- <template>
- <div
- class="video_container"
- :class="largeScreen ? 'large_screen' : 'normal_screen'"
- >
- <!-- 弹窗内容 -->
- <div class="dialog_box">
- <!-- 头部 -->
- <div class="dialog_header_box">
- <div class="dialog_header_title">
- {{ cameraData ? cameraData.name : "监控视频" }}
- </div>
- </div>
- <!-- 内容 -->
- <div
- class="dialog_content_box"
- :class="{ large_content: largeScreen }"
- v-show="!!cameraId"
- >
- <div class="content_box">
- <div
- class="img_box"
- :class="{ comm_ptz: largeScreen && cameraData.ptzEnable }"
- >
- <img :src="url" v-if="url" />
- <div class="image_slot" v-else>
- <i class="el-icon-video-camera"></i>
- <span>无信号</span>
- </div>
- </div>
- <div class="control_box" v-if="largeScreen && cameraData.ptzEnable">
- <div class="video-opt">
- <div class="comm-panel">
- <div
- class="btn-box"
- v-for="item of commandConfig.direction"
- :key="item.type"
- >
- <el-button
- type="primary"
- :icon="`el-icon-${item.type}`"
- size="mini"
- @mousedown.native="changePtzCommParam(item.command, false)"
- @mouseup.native="changePtzCommParam(item.command, true)"
- ></el-button>
- </div>
- </div>
- <div class="zoom-panel">
- <div
- class="btn-box"
- v-for="item of commandConfig.zoom"
- :key="item.type"
- >
- <el-button
- type="primary"
- :icon="`el-icon-${item.type}`"
- size="mini"
- @mousedown.native="changePtzCommParam(item.command, false)"
- @mouseup.native="changePtzCommParam(item.command, true)"
- ></el-button>
- </div>
- </div>
- <div class="speed-panel">
- <span class="form-label">速度:</span>
- <el-select
- class="myInput"
- size="mini"
- v-model="speed"
- placeholder="请选择速度"
- >
- <el-option
- v-for="item in Array.from(
- { length: 7 },
- (item, index) => index + 1
- )"
- :key="item"
- :label="item"
- :value="item"
- ></el-option>
- </el-select>
- </div>
- <div class="preset-panel">
- <span class="form-label">预置点:</span>
- <el-select
- class="myInput"
- size="mini"
- v-model="presetIndex"
- placeholder="请选择预置点"
- >
- <el-option
- v-for="item in Array.from(
- { length: 255 },
- (item, index) => index + 1
- )"
- :key="item"
- :label="item"
- :value="item"
- ></el-option>
- </el-select>
- <el-button
- type="primary"
- size="mini"
- @click="changePtzPresetParam(8)"
- >设置</el-button
- >
- <el-button
- type="primary"
- size="mini"
- @click="changePtzPresetParam(39)"
- >转到</el-button
- >
- </div>
- <div class="playback-panel">
- <span class="form-label">时间:</span>
- <el-date-picker
- style="max-width: 170px"
- class="myInput"
- v-model="startTime"
- type="datetime"
- placeholder="选择回放开始时间"
- size="mini"
- ></el-date-picker>
- <el-button
- type="success"
- size="mini"
- @click="changePlayTypeParam(true)"
- >回放</el-button
- >
- <el-button
- type="primary"
- size="mini"
- @click="changePlayTypeParam(false)"
- >实时</el-button
- >
- </div>
- </div>
- </div>
- </div>
- </div>
- <div
- class="dialog_content_box"
- :class="{ large_content: largeScreen }"
- v-show="cameraData && !cameraId"
- style="color: #fff"
- >
- <!-- 暂无数据 -->
- <!-- <img :src="require(`@/assets/imgs/maps/video_img_${cameraData.id}.png`)" alt="" /> -->
- </div>
- <div class="info_box">
- <span class="label">浓度</span>
- <span class="value">{{ ndValue || 0 }}</span>
- <span class="unit">ppm.m</span>
- </div>
- </div>
- </div>
- </template>
- <script>
- import Dayjs from "dayjs";
- import { mapGetters } from "vuex";
- export default {
- name: "VideoImageWindow",
- props: {
- cameraId: {
- type: String,
- require: true,
- },
- cameraData: {
- type: Object,
- default: () => ({}),
- },
- },
- computed: {
- ...mapGetters(["rtData"]),
- },
- watch: {
- rtData: {
- handler(newVal) {
- console.log(newVal);
- if (!!this.cameraData.tags) {
- if (!!this.cameraData.dataList) {
- this.cameraData.dataList.forEach((item) => {
- if (!!item.tag && !!newVal[item.tag]) {
- item.value = newVal[item.tag].value + " " + item.unit;
- }
- if (item.isTag && !!newVal[item.value]) {
- }
- });
- }
- }
- },
- deep: true,
- },
- },
- data() {
- return {
- largeScreen: false,
- ws: null,
- url: "",
- lastUrl: "",
- // 视频控制相关 start
- commandConfig: {
- //视频控制开关配置
- direction: [
- //方向
- { type: "top-left", name: "左上", command: 25 },
- { type: "top", name: "上", command: 21 },
- { type: "top-right", name: "右上", command: 26 },
- { type: "back", name: "左", command: 23 },
- { type: "rank", name: "中", command: 0 },
- { type: "right", name: "右", command: 24 },
- { type: "bottom-left", name: "左下", command: 27 },
- { type: "bottom", name: "下", command: 22 },
- { type: "bottom-right", name: "右下", command: 28 },
- ],
- zoom: [
- //放大缩小
- { type: "zoom-in", name: "放大", command: 11 },
- { type: "zoom-out", name: "缩小", command: 12 },
- ],
- },
- startTime: null,
- speed: 4,
- presetIndex: 1,
- // 视频控制 end
- ndValue: "0",
- };
- },
- beforeDestroy() {
- this.closeHandler();
- },
- created() {
- this.init(JSON.stringify({ id: this.cameraId }));
- },
- mounted() {
- const ndTopic = "DataCommunication/TagData/" + this.cameraData.tags.ndTag;
- const _this = this;
- this.$store.dispatch("mqtt/subscribe", {
- topic: [ndTopic],
- onMessage: (topic, message, packet) => {
- // console.log(topic);
- const content = message.toString();
- if (topic === ndTopic) {
- //console.log(new Date(), content);
- const data = JSON.parse(content);
- _this.ndValue = data.Value;
- }
- },
- });
- this.$once("hook:beforeDestroy", () => {
- this.$store.dispatch("mqtt/unsubscribe", [ndTopic]);
- });
- },
- methods: {
- getTagValue(tag) {
- return this.rtData[this.cameraData.tags[tag]]?.Value || 0;
- },
- // 放大缩小
- showVideoPopup(data) {
- this.largeScreen = data;
- this.$nextTick(() => {
- if (data) {
- this.baseDialogConfig.width = 937;
- this.baseDialogConfig.height = 598;
- this.baseDialogConfig.center = true;
- } else {
- this.baseDialogConfig.width = 420;
- this.baseDialogConfig.height = 252;
- this.baseDialogConfig.center = false;
- }
- });
- },
- init(str) {
- // this.socketList[`ws_${this.cameraId}`];
- this.ws = new WebSocket(
- VUE_APP_BASE_WS() + "/VideoOverWebSocket"
- // "wss://" + window.location.host + BASE_URL + "/VideoOverWebSocket"
- );
- this.ws.onmessage = (evt) => {
- if (typeof evt.data === "string") {
- console.log(JSON.parse(evt.data).Msg);
- } else {
- let _this = this;
- // const updateUrl = _.debounce(function () {
- let currUrl = URL.createObjectURL(evt.data);
- if (_this.lastUrl) URL.revokeObjectURL(_this.lastUrl);
- _this.lastUrl = currUrl;
- _this.url = currUrl;
- // }, 500);
- // updateUrl();
- }
- };
- this.handleSend(str);
- // console.log(window);
- },
- handleSend(str) {
- if (this.ws.readyState === WebSocket.OPEN) {
- this.ws.send(str);
- } else if (this.ws.readyState == WebSocket.CONNECTING) {
- // Wait for the open event, maybe do something with promises
- // depending on your use case. I believe in you developer!
- this.ws.addEventListener("open", () => this.handleSend(str));
- } else {
- // etc.
- }
- },
- updateEvent(data, state) {
- const param = JSON.stringify(data);
- if (this.ws && this.ws.readyState == 1) {
- this.handleSend(param);
- } else {
- this.init(param);
- }
- if (state) {
- setTimeout(() => {
- this.url = "";
- if (this.lastUrl) URL.revokeObjectURL(this.lastUrl);
- this.lastUrl = "";
- }, 500);
- }
- },
- closeHandler() {
- this.ws && this.ws.close();
- setTimeout(() => {
- if (this.url) URL.revokeObjectURL(this.url);
- this.url = "";
- this.lastUrl = "";
- this.ws &&
- (this.ws.onclose = (evt) => {
- // this.ws = null;
- console.log("websocket已关闭");
- });
- }, 500);
- },
- changePtzCommParam(comm, stop) {
- if (!comm) return;
- let param = {
- id: this.cameraId,
- ptzCommand: comm,
- ptzStop: stop,
- ptzSpeed: this.speed,
- };
- this.updateEvent(param);
- },
- changePtzPresetParam(comm) {
- if (!comm) return;
- let param = {
- id: this.cameraId,
- ptzPreset: comm,
- ptzPresetIndex: this.presetIndex,
- };
- this.updateEvent(param);
- },
- /**
- * state true代表回放模式
- */
- changePlayTypeParam(state) {
- let param = { id: this.cameraId };
- if (state) {
- if (!this.startTime) {
- this.$message.warning("请选择回放开始时间!");
- return;
- }
- param["startTime"] = Dayjs(this.startTime).format(
- "YYYY-MM-DD HH:mm:ss"
- );
- }
- this.updateEvent(param, true);
- },
- },
- };
- </script>
- <style lang="less" scoped>
- .normal_screen {
- background: url("~@/assets/imgs/maps/img_SPJK_bg@2x.png") center center
- no-repeat;
- background-size: 100% 100%;
- width: 1.984375rem /* 381/192 */;
- height: 1.385417rem /* 266/192 */;
- }
- .large_screen {
- background: url("~@/assets/imgs/dialog/pop_enlarge_TW@2x.png") center center
- no-repeat;
- background-size: 100% 100%;
- width: 4.880208rem /* 937/192 */;
- height: 3.114583rem /* 598/192 */;
- }
- .video_container {
- pointer-events: auto;
- .dialog_box {
- width: 100%;
- height: 100%;
- .dialog_header_box {
- position: relative;
- .dialog_header_title {
- width: 100%;
- height: 35px;
- line-height: 40px;
- // font-family: 时尚中黑简体;
- font-size: 16px;
- padding-left: 24px;
- color: #fff;
- }
- .dialog_header_retract {
- // line-height: 40px;
- position: absolute;
- top: -0.041667rem /* 8/192 */;
- right: 0.182292rem /* 35/192 */;
- cursor: pointer;
- img {
- width: 20px;
- height: 20px;
- }
- }
- .dialog_header_close {
- // line-height: 40px;
- position: absolute;
- top: -0.041667rem /* 8/192 */;
- right: 0.052083rem /* 10/192 */;
- img {
- width: 20px;
- height: 20px;
- }
- cursor: pointer;
- }
- }
- .dialog_content_box {
- box-sizing: border-box;
- width: 342px;
- height: 173px;
- margin: 0 auto;
- display: flex;
- justify-content: center;
- align-items: center;
- .content_box {
- width: 100%;
- height: 100%;
- display: flex;
- .img_box {
- width: 100%;
- height: 100%;
- &.comm_ptz {
- width: 75%;
- }
- }
- .control_box {
- width: 25%;
- height: 100%;
- border-left: 1px solid #7bacaa;
- }
- }
- img {
- width: 100%;
- height: 100%;
- object-fit: fill;
- // display: block;
- }
- .image_slot {
- width: 100%;
- height: 100%;
- // background-color: #f5f7fa;
- color: #f5f7fa;
- font-size: 24px;
- display: flex;
- flex-direction: column;
- justify-content: center;
- align-items: center;
- font-weight: 500;
- span {
- font-size: 16px;
- }
- }
- &.large_content {
- width: 4.630208rem /* 889/192 */;
- height: 2.770833rem /* 532/192 */;
- }
- }
- .info_box {
- width: 100%;
- height: 32px;
- margin-top: 6px;
- display: flex;
- justify-content: center;
- align-items: center;
- font-size: 14px;
- color: #fff;
- .value {
- padding: 0 8px;
- color: #ffc800;
- }
- }
- }
- }
- .video-opt {
- display: flex;
- flex-direction: column;
- align-items: center;
- padding: 10px;
- .comm-panel {
- width: 180px;
- display: flex;
- flex-wrap: wrap;
- .el-button {
- background: #51c6c1;
- border-color: #51c6c1;
- }
- }
- .zoom-panel {
- width: 120px;
- display: flex;
- .el-button {
- background: #51c6c1;
- border-color: #51c6c1;
- }
- }
- .speed-panel {
- margin-top: 10px;
- display: flex;
- align-items: center;
- }
- .preset-panel {
- margin-top: 10px;
- display: flex;
- align-items: center;
- flex-wrap: wrap;
- .el-select {
- width: 150px;
- }
- .el-button {
- margin-top: 10px;
- background: rgba(0, 60, 57, 0.3);
- border: 1px solid #65f1ea;
- color: #7cd6dd;
- }
- }
- .playback-panel {
- margin-top: 10px;
- display: flex;
- align-items: center;
- flex-wrap: wrap;
- .form-label {
- margin-bottom: 10px;
- }
- .el-button {
- margin-top: 15px;
- background: rgba(0, 60, 57, 0.3);
- border: 1px solid #65f1ea;
- color: #7cd6dd;
- }
- }
- .btn-box {
- flex-shrink: 0;
- width: 60px;
- height: 40px;
- display: flex;
- justify-content: center;
- align-items: center;
- }
- .form-label {
- color: #7cd6dd;
- display: inline-block;
- width: 50px;
- flex-shrink: 0;
- // text-align: right;
- }
- .myInput {
- ::v-deep .el-input__inner {
- // width: 110px;
- height: 30px;
- border-radius: 0;
- background: rgba(0, 20, 23, 0.6);
- border: 0.5px solid #51c6c1;
- color: #7cd6dd;
- // padding-right: 35px;
- }
- ::v-deep .el-input__inner:focus {
- border: 0.5px solid #51c6c1 !important;
- }
- ::v-deep .el-input.is-focus .el-input__inner {
- border: 0.5px solid #51c6c1 !important;
- }
- ::v-deep .el-input__suffix {
- right: 5px;
- }
- ::v-deep .el-input__icon {
- line-height: 30px;
- }
- }
- }
- </style>
|