|
@@ -1,114 +1,187 @@
|
|
|
<template>
|
|
<template>
|
|
|
- <el-dialog v-model="visible" title="数据模拟" width="870" :close-on-click-modal="false" @closed="$emit('closed')">
|
|
|
|
|
- <el-form ref="formRef" :model="form" :rules="rules" label-width="110px">
|
|
|
|
|
|
|
+ <el-dialog v-model="visible" title="数据模拟" fullscreen :close-on-click-modal="false" @closed="$emit('closed')">
|
|
|
|
|
+ <el-tabs v-model="apiKey">
|
|
|
|
|
+ <el-tab-pane label="参数配置" name="makeData"></el-tab-pane>
|
|
|
|
|
+ <el-tab-pane label="数据复制" name="copyData" disabled></el-tab-pane>
|
|
|
|
|
+ </el-tabs>
|
|
|
|
|
+
|
|
|
|
|
+ <el-form ref="formRef" :model="form" :rules="rules" label-width="126">
|
|
|
<el-row>
|
|
<el-row>
|
|
|
<el-col :md="12" :xs="24">
|
|
<el-col :md="12" :xs="24">
|
|
|
- <el-form-item label="时间范围" prop="dateRange">
|
|
|
|
|
- <el-date-picker v-model="form.dateRange" type="daterange" :clearable="false" :shortcuts="shortcuts" range-separator="至" start-placeholder="开始日期" end-placeholder="结束日期"></el-date-picker>
|
|
|
|
|
|
|
+ <el-form-item label="模拟项目" prop="targetProjectId">
|
|
|
|
|
+ <el-select v-model="form.targetProjectId" filterable placeholder="请选择模拟项目" @change="form.targetMountedId = null, dataTimeRange()">
|
|
|
|
|
+ <el-option v-for="item in $TOOL.data.get('PROJECT')" :key="item.fpiId" :label="item.projectName" :value="item.fpiId"></el-option>
|
|
|
|
|
+ </el-select>
|
|
|
|
|
+ </el-form-item>
|
|
|
|
|
+ </el-col>
|
|
|
|
|
+ <el-col v-if="form.targetProjectId" :md="12" :xs="24">
|
|
|
|
|
+ <el-form-item label="数据时间范围">
|
|
|
|
|
+ <template v-if="XEUtils.isEmpty(acceptItem)">该项目未配置验收清单,<el-button type="primary" link @click="$router.push('/basic/acceptItems')">去配置</el-button></template>
|
|
|
|
|
+ <template v-else-if="acceptItem.beginTime">{{ $TOOL.dateFormat(acceptItem.beginTime, "YY.M.D") }}<span>-{{ acceptItem.endTime && $TOOL.dateFormat(acceptItem.endTime, "YY.M.D") || "至今" }}</span></template>
|
|
|
|
|
+ <template v-else>该项目未配置数据时间范围,<el-button type="primary" link @click="$router.push('/basic/project')">去配置</el-button></template>
|
|
|
</el-form-item>
|
|
</el-form-item>
|
|
|
</el-col>
|
|
</el-col>
|
|
|
<el-col :md="12" :xs="24">
|
|
<el-col :md="12" :xs="24">
|
|
|
- <el-form-item class="step-item" label="时间步长" prop="timeStep">
|
|
|
|
|
- <el-input-number v-model="form.timeStep" :min="0"
|
|
|
|
|
- :max="['second', 'minute'].includes(form.timeStepType) ? 60 : form.timeStepType == 'hour' ? 12 : Infinity"
|
|
|
|
|
- :controls="false" placeholder="时间步长">
|
|
|
|
|
- </el-input-number>
|
|
|
|
|
- <el-form-item>
|
|
|
|
|
- <el-select v-model="form.timeStepType">
|
|
|
|
|
- <el-option label="秒" value="second"></el-option>
|
|
|
|
|
- <el-option label="分钟" value="minute"></el-option>
|
|
|
|
|
- <el-option label="小时" value="hour"></el-option>
|
|
|
|
|
- <el-option label="天" value="day"></el-option>
|
|
|
|
|
- </el-select>
|
|
|
|
|
- </el-form-item>
|
|
|
|
|
|
|
+ <el-form-item label="模拟项目安装点" prop="targetMountedId">
|
|
|
|
|
+ <el-select v-model="form.targetMountedId" filterable placeholder="请选择模拟项目安装点">
|
|
|
|
|
+ <el-option v-for="item in filterTargetM" :key="item.id" :label="item.mountedName" :value="item.id"></el-option>
|
|
|
|
|
+ </el-select>
|
|
|
</el-form-item>
|
|
</el-form-item>
|
|
|
</el-col>
|
|
</el-col>
|
|
|
<el-col :md="12" :xs="24">
|
|
<el-col :md="12" :xs="24">
|
|
|
- <el-form-item label="精度偏差" prop="precision">
|
|
|
|
|
- <el-input-number v-model="form.precision" :min="0" :max="2" :precision="2" :controls="false" placeholder="请输入精度偏差">
|
|
|
|
|
- <template #prefix>±</template>
|
|
|
|
|
- </el-input-number>
|
|
|
|
|
|
|
+ <el-form-item label="模拟时间范围" prop="targetTime">
|
|
|
|
|
+ <el-date-picker v-model="form.targetTime" type="datetimerange" :clearable="false" value-format="YYYY-MM-DD HH:mm:ss" :default-time="[new Date(2000, 1, 1, 0, 0, 0), new Date(2000, 1, 1, 23, 59, 59)]" :shortcuts="shortcuts" range-separator="至" start-placeholder="开始时间" end-placeholder="结束时间"></el-date-picker>
|
|
|
</el-form-item>
|
|
</el-form-item>
|
|
|
</el-col>
|
|
</el-col>
|
|
|
|
|
+
|
|
|
<el-col :md="12" :xs="24">
|
|
<el-col :md="12" :xs="24">
|
|
|
- <el-form-item label="数据处理" prop="handler">
|
|
|
|
|
- <el-radio-group v-model="form.handler">
|
|
|
|
|
- <el-radio value="copy">重复新增</el-radio>
|
|
|
|
|
- <el-radio value="cover">数据覆盖</el-radio>
|
|
|
|
|
- <el-radio value="partly">部分覆盖</el-radio>
|
|
|
|
|
|
|
+ <el-form-item label="数据类型" prop="dataSouce">
|
|
|
|
|
+ <el-radio-group v-model="form.dataSouce">
|
|
|
|
|
+ <el-radio value="normal">标准数据</el-radio>
|
|
|
|
|
+ <el-radio value="alarm">报警数据</el-radio>
|
|
|
</el-radio-group>
|
|
</el-radio-group>
|
|
|
</el-form-item>
|
|
</el-form-item>
|
|
|
</el-col>
|
|
</el-col>
|
|
|
- </el-row>
|
|
|
|
|
|
|
+ <el-col :md="12" :xs="24">
|
|
|
|
|
+ <el-form-item label="数据处理" prop="isCover">
|
|
|
|
|
+ <el-radio-group v-model="form.isCover">
|
|
|
|
|
+ <el-radio :value="false">重复新增</el-radio>
|
|
|
|
|
+ <el-radio :value="true">数据覆盖</el-radio>
|
|
|
|
|
+ </el-radio-group>
|
|
|
|
|
+ </el-form-item>
|
|
|
|
|
+ </el-col>
|
|
|
|
|
+
|
|
|
|
|
+ <template v-if="form.dataSouce == 'normal'">
|
|
|
|
|
+ <el-col :md="12" :xs="24">
|
|
|
|
|
+ <el-form-item class="step-item" label="时间步长" prop="timeStepSec">
|
|
|
|
|
+ <el-input-number v-model="form.timeStepSec" :min="0" :controls="false" placeholder="时间步长"></el-input-number>
|
|
|
|
|
|
|
|
- <el-form-item label="数据设置">
|
|
|
|
|
- <sc-form-table v-model="form.warnList" hideSeq>
|
|
|
|
|
- <el-table-column label="报警类型" prop="field" width="100" align="center"></el-table-column>
|
|
|
|
|
- <el-table-column class-name="threshold-cell" label="阈值" prop="threshold">
|
|
|
|
|
- <template #default="scope">
|
|
|
|
|
- <el-input-number v-model="scope.row.min" :min="0" :max="scope.row.max" :precision="2" :controls="false" placeholder="最小值"></el-input-number>
|
|
|
|
|
- <div class="symbol">至</div>
|
|
|
|
|
- <el-input-number v-model="scope.row.max" :min="scope.row.min" :precision="2" :controls="false" placeholder="最大值"></el-input-number>
|
|
|
|
|
- </template>
|
|
|
|
|
- </el-table-column>
|
|
|
|
|
- <el-table-column label="报警条数" prop="count" width="120" align="center">
|
|
|
|
|
- <template #default="scope">
|
|
|
|
|
- <el-input-number v-model="scope.row.count" :min="0" :controls="false" placeholder="请输入报警条数"></el-input-number>
|
|
|
|
|
- </template>
|
|
|
|
|
- </el-table-column>
|
|
|
|
|
- </sc-form-table>
|
|
|
|
|
- </el-form-item>
|
|
|
|
|
|
|
+ <el-radio-group v-model="form.timeStepType" @change="stepTypeChange">
|
|
|
|
|
+ <el-radio value="second">秒</el-radio>
|
|
|
|
|
+ <el-radio value="minute">分钟</el-radio>
|
|
|
|
|
+ </el-radio-group>
|
|
|
|
|
+ </el-form-item>
|
|
|
|
|
+ </el-col>
|
|
|
|
|
+ <el-col :md="12" :xs="24">
|
|
|
|
|
+ <el-form-item label="数据波动" prop="floatNumber">
|
|
|
|
|
+ <el-input-number v-model="form.floatNumber" :min="0" :precision="2" :controls="false" placeholder="请输入数据波动">
|
|
|
|
|
+ <template #prefix>±</template>
|
|
|
|
|
+ </el-input-number>
|
|
|
|
|
+ </el-form-item>
|
|
|
|
|
+ </el-col>
|
|
|
|
|
+ </template>
|
|
|
|
|
+ <template v-if="form.dataSouce == 'alarm'">
|
|
|
|
|
+ <el-col :md="12" :xs="24">
|
|
|
|
|
+ <el-form-item label="温度报警数" prop="temWarnNum">
|
|
|
|
|
+ <el-input-number v-model="form.temWarnNum" :min="0" :controls="false" placeholder="请输入温度报警数"></el-input-number>
|
|
|
|
|
+ </el-form-item>
|
|
|
|
|
+ </el-col>
|
|
|
|
|
+ <el-col :md="12" :xs="24">
|
|
|
|
|
+ <el-form-item label="湿度报警数" prop="humWarnNum">
|
|
|
|
|
+ <el-input-number v-model="form.humWarnNum" :min="0" :controls="false" placeholder="请输入湿度报警数"></el-input-number>
|
|
|
|
|
+ </el-form-item>
|
|
|
|
|
+ </el-col>
|
|
|
|
|
+ </template>
|
|
|
|
|
+ </el-row>
|
|
|
</el-form>
|
|
</el-form>
|
|
|
|
|
|
|
|
<template #footer>
|
|
<template #footer>
|
|
|
- <el-button :loading="isSaving" type="primary" auto-insert-space @click="submit">保存</el-button>
|
|
|
|
|
|
|
+ <el-button :loading="isSaving" type="primary" auto-insert-space @click="submit()">提交</el-button>
|
|
|
|
|
+ <el-button v-if="form.source == 'other'" :loading="isSaving" type="primary" auto-insert-space @click="submit('template')">保存为模版</el-button>
|
|
|
<el-button auto-insert-space @click="visible = false">取消</el-button>
|
|
<el-button auto-insert-space @click="visible = false">取消</el-button>
|
|
|
</template>
|
|
</template>
|
|
|
</el-dialog>
|
|
</el-dialog>
|
|
|
</template>
|
|
</template>
|
|
|
|
|
|
|
|
<script setup>
|
|
<script setup>
|
|
|
-import moment from "moment";
|
|
|
|
|
import XEUtils from "xe-utils";
|
|
import XEUtils from "xe-utils";
|
|
|
|
|
+import API from "@/api";
|
|
|
|
|
+import TOOL from "@/utils/tool";
|
|
|
|
|
+import { rangeShortcuts } from "@/utils/shortcuts";
|
|
|
|
|
|
|
|
|
|
+const route = useRoute();
|
|
|
const $emit = defineEmits(["success", "closed"]);
|
|
const $emit = defineEmits(["success", "closed"]);
|
|
|
|
|
+const apiKey = ref("makeData");
|
|
|
const visible = ref(false);
|
|
const visible = ref(false);
|
|
|
const isSaving = ref(false);
|
|
const isSaving = ref(false);
|
|
|
|
|
|
|
|
-const shortcuts = [
|
|
|
|
|
- { text: "上周", value: () => [moment().subtract(1, "week"), moment()] },
|
|
|
|
|
- { text: "上月", value: () => [moment().subtract(1, "month"), moment()] },
|
|
|
|
|
- { text: "去年", value: () => [moment().subtract(1, "year"), moment()] }
|
|
|
|
|
-]
|
|
|
|
|
-
|
|
|
|
|
|
|
+const shortcuts = rangeShortcuts();
|
|
|
const form = ref({
|
|
const form = ref({
|
|
|
- id: null,
|
|
|
|
|
- dateRange: null,
|
|
|
|
|
|
|
+ targetProjectId: TOOL.data.get("PROJECT_ID"),
|
|
|
|
|
+ targetMountedId: null,
|
|
|
|
|
+ targetTime: ["2025-08-28 00:00:00", "2025-08-28 00:59:59"],
|
|
|
|
|
+ dataSouce: "normal",
|
|
|
|
|
+ isCover: false,
|
|
|
|
|
+ timeStepSec: 3,
|
|
|
timeStepType: "minute",
|
|
timeStepType: "minute",
|
|
|
- timeStep: 3,
|
|
|
|
|
- precision: null,
|
|
|
|
|
- handler: "copy",
|
|
|
|
|
- warnList: [
|
|
|
|
|
- { field: "温度 (℃)", min: 18, max: 22, count: 0 },
|
|
|
|
|
- { field: "湿度 (%rh)", min: 95, max: 100, count: 0 }
|
|
|
|
|
- ]
|
|
|
|
|
|
|
+ floatNumber: 1,
|
|
|
|
|
+ temWarnNum: null,
|
|
|
|
|
+ humWarnNum: null
|
|
|
});
|
|
});
|
|
|
|
|
|
|
|
const rules = reactive({
|
|
const rules = reactive({
|
|
|
- dateRange: [{ required: true, message: "请选择时间范围" }],
|
|
|
|
|
- timeStep: [{ required: true, message: "请输入时间步长" }],
|
|
|
|
|
- precision: [{ required: true, message: "请输入精度偏差" }],
|
|
|
|
|
- warnList: [{ required: true, validator: (rule, value, callback) => {
|
|
|
|
|
- callback();
|
|
|
|
|
- }}]
|
|
|
|
|
-})
|
|
|
|
|
|
|
+ targetProjectId: [{ required: true, message: "请选择模拟项目" }],
|
|
|
|
|
+ targetMountedId: [{ required: true, message: "请选择模拟项目安装点" }],
|
|
|
|
|
+ targetTime: [{ required: true, message: "请选择模拟时间范围" }],
|
|
|
|
|
+ dataSouce: [{ required: true }],
|
|
|
|
|
+ isCover: [{ required: true }],
|
|
|
|
|
+ timeStepSec: [{ required: true, message: "请输入时间步长" }],
|
|
|
|
|
+ floatNumber: [{ required: true, message: "请输入数据波动" }],
|
|
|
|
|
+ temWarnNum: [{ required: true, message: "请输入温度报警数" }],
|
|
|
|
|
+ humWarnNum: [{ required: true, message: "请输入湿度报警数" }]
|
|
|
|
|
+});
|
|
|
|
|
|
|
|
-const open = () => visible.value = true;
|
|
|
|
|
-const formRef = ref();
|
|
|
|
|
|
|
+const acceptItem = ref({});
|
|
|
|
|
+const dataTimeRange = async () => {
|
|
|
|
|
+ const query = {
|
|
|
|
|
+ projectId: form.value.targetProjectId,
|
|
|
|
|
+ itemName: XEUtils.last(route.meta.title.split("-"))
|
|
|
|
|
+ }
|
|
|
|
|
+ const res = await API.system.project.bindItem.judgment(query);
|
|
|
|
|
+ acceptItem.value = res || {};
|
|
|
|
|
+}
|
|
|
|
|
|
|
|
-const submit = () => {
|
|
|
|
|
|
|
+const mounteds = ref([]);
|
|
|
|
|
+const filterTargetM = computed(() => form.value.targetProjectId ? XEUtils.filter(mounteds.value, item => item.projectId == form.value.targetProjectId) : []);
|
|
|
|
|
+const fetchMounted = async () => {
|
|
|
|
|
+ const res = await API.scc.mounted.get();
|
|
|
|
|
+ mounteds.value = res || [];
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+const open = () => {
|
|
|
|
|
+ visible.value = true;
|
|
|
|
|
+ TOOL.data.get("PROJECT_ID") && dataTimeRange();
|
|
|
|
|
+ fetchMounted();
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+const stepTypeChange = e => {
|
|
|
|
|
+ if (e == "minute") XEUtils.set(form.value, "timeStepSec", XEUtils.floor(XEUtils.divide(form.value.timeStepSec, 60)));
|
|
|
|
|
+ if (e == "second") XEUtils.set(form.value, "timeStepSec", XEUtils.multiply(form.value.timeStepSec, 60));
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+const formRef = ref();
|
|
|
|
|
+const submit = key => {
|
|
|
formRef.value.validate(valid => {
|
|
formRef.value.validate(valid => {
|
|
|
if (valid) {
|
|
if (valid) {
|
|
|
|
|
+ const data = XEUtils.pick(form.value, "targetProjectId", "targetMountedId", "isCover");
|
|
|
|
|
+ XEUtils.set(data, "targetBeginTime", XEUtils.first(form.value.targetTime));
|
|
|
|
|
+ XEUtils.set(data, "targetEndTime", XEUtils.last(form.value.targetTime));
|
|
|
|
|
+
|
|
|
|
|
+ if (form.value.dataSouce == "normal") {
|
|
|
|
|
+ XEUtils.set(data, "floatNumber", XEUtils.multiply(form.value.floatNumber, 100));
|
|
|
|
|
+ XEUtils.set(data, "timeStepSec", form.value.timeStepType == "minute" ? XEUtils.multiply(form.value.timeStepSec, 60) : form.value.timeStepSec);
|
|
|
|
|
+ } else {
|
|
|
|
|
+ XEUtils.set(data, "temWarnNum", form.value.temWarnNum);
|
|
|
|
|
+ XEUtils.set(data, "humWarnNum", form.value.humWarnNum);
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ isSaving.value = true;
|
|
|
|
|
+ API.scc.dataMock[apiKey.value][form.value.dataSouce](data).then(() => {
|
|
|
|
|
+ isSaving.value = false;
|
|
|
|
|
+ ElMessage.success("操作成功");
|
|
|
|
|
+ visible.value = false;
|
|
|
|
|
+ $emit("success");
|
|
|
|
|
+ }).catch(() => isSaving.value = false);
|
|
|
} else {
|
|
} else {
|
|
|
return false;
|
|
return false;
|
|
|
}
|
|
}
|
|
@@ -121,20 +194,15 @@ defineExpose({
|
|
|
</script>
|
|
</script>
|
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
<style lang="scss" scoped>
|
|
|
-.el-form {padding-right: calc(var(--el-dialog-padding-primary) + var(--el-message-close-size, 16px));}
|
|
|
|
|
|
|
+.el-form {margin-top: 5px;padding-right: var(--el-message-close-size, 16px);}
|
|
|
|
|
+.el-form-item .el-radio-group {flex-wrap: nowrap;}
|
|
|
|
|
|
|
|
.el-form-item .el-input-number {width: 100%;}
|
|
.el-form-item .el-input-number {width: 100%;}
|
|
|
.el-form-item .el-input-number :deep(.el-input__prefix) {margin-right: 8px;}
|
|
.el-form-item .el-input-number :deep(.el-input__prefix) {margin-right: 8px;}
|
|
|
.el-form-item .el-input-number :deep(.el-input__inner) {text-align: unset;}
|
|
.el-form-item .el-input-number :deep(.el-input__inner) {text-align: unset;}
|
|
|
-.el-form-item .el-input-number + .el-form-item {margin-left: 20px;}
|
|
|
|
|
-
|
|
|
|
|
|
|
+
|
|
|
.el-form-item.step-item {
|
|
.el-form-item.step-item {
|
|
|
.el-input-number {flex: 1;}
|
|
.el-input-number {flex: 1;}
|
|
|
- .el-form-item {width: 100px;}
|
|
|
|
|
|
|
+ .el-radio-group {margin-left: 20px;}
|
|
|
}
|
|
}
|
|
|
-
|
|
|
|
|
-.el-form-item .el-radio-group {flex-wrap: nowrap;}
|
|
|
|
|
-
|
|
|
|
|
-.el-form-item :deep(.el-table) .threshold-cell .cell {display: flex;justify-content: center;align-items: center;}
|
|
|
|
|
-.el-form-item :deep(.el-table) .threshold-cell .cell .symbol {margin: 0 10px;}
|
|
|
|
|
</style>
|
|
</style>
|