index.vue 6.01 KB
<script lang="ts" setup>
  import { ref } from 'vue';
  import { useGenDynamicForm } from './useGenDynamicForm';
  import { ModalParamsType } from '/#/utils';
  import { DeviceModelOfMatterAttrs } from '/@/api/device/model/deviceModel';
  import { StructJSON } from '/@/api/device/model/modelOfMatterModel';
  import { BasicForm, useForm } from '/@/components/Form';
  import { BasicModal, useModalInner } from '/@/components/Modal';
  import { DataTypeEnum } from '/@/views/device/profiles/step/cpns/physical/cpns/config';
  import { sendCommandOneway } from '/@/api/dataBoard';
  import { useMessage } from '/@/hooks/web/useMessage';
  import { unref } from 'vue';
  import { genModbusCommand } from '/@/api/task';
  import { TaskTypeEnum } from '/@/views/task/center/config';
  import { SingleToHex, formSchemasConfig } from './config';

  defineEmits(['register']);
  const props = defineProps<{ deviceId: string; deviceName: string }>();

  const [registerForm, { setProps, getFieldsValue, resetFields, validate }] = useForm({
    schemas: [],
    showActionButtonGroup: false,
    layout: 'vertical',
  });

  const { genForm } = useGenDynamicForm();

  const keys = ref<string[]>([]);

  const modBUSForm = ref<any>({});
  const isShowModBUS = ref<Boolean>(false); //用于判断标识符类型是否时自定义还是modBUS
  const isShowActionType = ref<Boolean>(true); //判断设备属性标识符为modBus时没有填写扩展描述
  const formField = ref(''); //存一个表单取值的field
  const zoomFactorValue = ref<number>(1); //缩放因子
  const isShowMultiply = ref<Boolean>(false); // 只有tcp --> int和double类型才相乘缩放因子

  const [register] = useModalInner(async (params: ModalParamsType<DeviceModelOfMatterAttrs>) => {
    const { record } = params;
    const { name, detail, identifier, deviceDetail, extensionDesc } = record;
    const { dataType } = detail;
    const { type } = dataType || {};
    const { codeType, deviceProfile } = deviceDetail || {};
    const { transportType } = deviceProfile || {};
    const { registerAddress, actionType, zoomFactor } = extensionDesc || {}; //获取扩展描述内容

    formField.value = identifier;
    zoomFactorValue.value = zoomFactor ? Number(zoomFactor) : 1;
    isShowMultiply.value = type == 'INT' || type == 'DOUBLE' ? true : false;

    let schemas = [{ dataType: dataType, identifier, functionName: name } as StructJSON];

    if (type === DataTypeEnum.IS_STRUCT) {
      schemas = dataType?.specs as StructJSON[];
    }

    keys.value = schemas.map((item) => {
      return item.identifier!;
    });
    isShowActionType.value = actionType ? true : false; //判断modBUS类型时 物模型是否填写扩展描述

    //是modBUS类型的就用另外的表单
    //判断是否是TCP ==> modBus的下发命令
    if (codeType == TaskTypeEnum.MODBUS && transportType == 'TCP') {
      isShowModBUS.value = true;
      modBUSForm.value = {
        crc: 'CRC_16_LOWER',
        deviceCode: '01',
        method: actionType == '16' ? '10' : actionType,
        registerAddress,
        registerNumber: 1,
        registerValues: [],
      };
      setProps({ schemas: formSchemasConfig(schemas[0], actionType) });
    } else {
      isShowModBUS.value = false;
      const formSchemas = genForm(schemas);
      setProps({ schemas: formSchemas });
    }

    resetFields();
  });

  const getArray = (values) => {
    const str = values.replace(/\s+/g, '');
    const array: any = [];

    for (let i = 0; i < str.length; i += 4) {
      const chunk = parseInt(str.substring(i, i + 4), 16);
      array.push(chunk);
    }
    return array;
  };

  const { createMessage } = useMessage();
  const loading = ref(false);
  const handleSend = async () => {
    try {
      loading.value = true;
      if (!props.deviceId) return;

      const sendValue = ref({});
      //判断tcp类型 标识符是自定义还是ModBus
      if (unref(isShowModBUS)) {
        if (!unref(isShowActionType)) {
          createMessage.warning('当前物模型扩展描述没有填写');
          return;
        }
        const flag = await validate();
        if (!flag) return;

        const oldValue = getFieldsValue()[unref(formField)];
        modBUSForm.value.registerNumber = 1;
        modBUSForm.value.registerValues = [oldValue];
        if (unref(isShowMultiply) && unref(modBUSForm).method == '06') {
          //bool类型的就不用去乘缩放因子了
          modBUSForm.value.registerValues = [oldValue * unref(zoomFactorValue)];
        }

        if (unref(modBUSForm).method == '16' || unref(modBUSForm).method == '10') {
          const newValue = getArray(
            SingleToHex(unref(isShowMultiply) ? oldValue * unref(zoomFactorValue) : oldValue)
          );
          modBUSForm.value.registerValues = newValue;
          modBUSForm.value.registerNumber = 2;
          modBUSForm.value.method = '10';
        }

        sendValue.value = await genModbusCommand(unref(modBUSForm));
      } else {
        const _value = getFieldsValue();

        sendValue.value = unref(keys).reduce((prev, next) => {
          return { ...prev, [next]: _value[next] };
        }, {});
      }

      await sendCommandOneway({
        deviceId: props.deviceId,
        value: {
          persistent: true,
          method: 'methodThingskit',
          params: unref(isShowModBUS)
            ? unref(sendValue)
            : {
                ...unref(sendValue),
              },
        },
      });
      createMessage.success('属性下发成功~');
    } catch (error) {
      throw error;
    } finally {
      loading.value = false;
    }
  };
</script>

<template>
  <BasicModal
    title="属性下发"
    @register="register"
    :min-height="60"
    ok-text="属性下发"
    wrap-class-name="model-of-matter-send-command-modal"
    @ok="handleSend"
    :ok-button-props="{ loading }"
  >
    <BasicForm @register="registerForm" class="dynamic-form" />
  </BasicModal>
</template>

<style lang="less">
  .model-of-matter-send-command-modal {
    .dynamic-form {
      .ant-input-number {
        @apply !w-full;
      }
    }
  }
</style>