CardMode.vue 7.84 KB
<script lang="ts" setup>
  import { Button, Tooltip, Card, Image, Popconfirm } from 'ant-design-vue';
  import { AuthIcon, EnumTableCardMode, ModeSwitchButton } from '/@/components/Widget';
  import { Authority } from '/@/components/Authority';
  import {
    deviceConfigDelete,
    deviceConfigGetQuery,
    setDeviceProfileIsDefaultApi,
  } from '/@/api/device/deviceConfigApi';
  import { ProfileRecord } from '/@/api/device/model/deviceConfigModel';
  import {
    defaultObj,
    searchFormSchema,
    DeviceTypeName,
    ProductPermission,
  } from './device.profile.data';
  import { useMessage } from '/@/hooks/web/useMessage';
  import DeviceProfileModal from './DeviceProfileModal.vue';
  import DeviceProfileDrawer from './DeviceProfileDrawer.vue';
  import { useModal } from '/@/components/Modal';
  import { useDrawer } from '/@/components/Drawer';
  import productDefault from '/@/assets/icons/product-default.svg';
  import AuthDropDown from '/@/components/Widget/AuthDropDown.vue';
  import { BasicCardList, useCardList } from '/@/components/CardList';
  import { useRoute } from 'vue-router';
  import { ref } from 'vue';
  import edgefornt from '/@/assets/icons/edgefornt.svg';

  defineProps<{
    mode: EnumTableCardMode;
  }>();

  const emit = defineEmits(['changeMode']);

  enum DropMenuEvent {
    SET_DEFAULT = 'setDefault',
    DELETE = 'delete',
  }

  const IMAGE_FALLBACK = productDefault;

  const { createMessage } = useMessage();

  const { query } = useRoute();

  const disabledDeleteFlag = ref(true);

  const [registerCardList, { reload, getSelectedKeys, clearSelectedKeys }] = useCardList({
    api: deviceConfigGetQuery,
    useSearchForm: true,
    gutter: 4,
    rowKey: 'id',
    formConfig: {
      schemas: searchFormSchema,
      labelWidth: 100,
      model: {
        name: (query as Recordable)?.name ? decodeURIComponent((query as Recordable)?.name) : null,
      },
    },
    selections: {
      beforeSelectValidate: (record: ProfileRecord) => {
        return !record.default && record?.name !== 'default';
      },
      onSelect: (_record, _flag, allSelecteds) => {
        disabledDeleteFlag.value = !allSelecteds.length;
      },
      onSelectAll: () => {
        // 全选事件
        disabledDeleteFlag.value = false;
      },
      onUnSelectAll: () => {
        // 反选事件
        disabledDeleteFlag.value = true;
      },
      onSelectToggle: (status: boolean) => {
        // 全选是false,反选是true
        if (!status) disabledDeleteFlag.value = false;
        else disabledDeleteFlag.value = true;
      },
    },
  });

  const [registerModal, { openModal }] = useModal();
  const [registerDrawer, { openDrawer }] = useDrawer();

  const handleModeChange = (mode: EnumTableCardMode) => {
    emit('changeMode', mode);
  };

  const handleCreate = () => {
    openModal(true, {
      isUpdate: false,
    });
  };

  const handleShowDetail = (record: ProfileRecord) => {
    openDrawer(true, { record });
  };

  const handleUpdate = (record: ProfileRecord) => {
    openModal(true, {
      record,
      isUpdate: true,
    });
  };

  const handleDelete = async (ids?: string[]) => {
    try {
      ids = ids || getSelectedKeys();
      await deviceConfigDelete(ids);
      createMessage.success('删除成功');
      clearSelectedKeys();
      disabledDeleteFlag.value = true;
      await reload();
    } catch (error) {
      throw error;
    }
  };

  const handleSetDefault = async (record: ProfileRecord) => {
    try {
      const { tbProfileId } = record;
      const data = await setDeviceProfileIsDefaultApi(tbProfileId, 'default', defaultObj);
      if (!data) return createMessage.error('设置该产品为默认失败');
      createMessage.success('设置该产品为默认成功');
      await reload();
    } catch (error) {
      throw error;
    }
  };
</script>

<template>
  <section>
    <BasicCardList @register="registerCardList">
      <template #toolbar>
        <Authority :value="ProductPermission.CREATE">
          <Button type="primary" @click="handleCreate">新增产品</Button>
        </Authority>

        <ModeSwitchButton :value="$props.mode" @change="handleModeChange" />

        <Authority :value="ProductPermission.DELETE">
          <Popconfirm
            title="您确定要批量删除数据"
            ok-text="确定"
            cancel-text="取消"
            @confirm="handleDelete()"
            :disabled="disabledDeleteFlag"
          >
            <Button type="primary" danger :disabled="disabledDeleteFlag"> 批量删除 </Button>
          </Popconfirm>
        </Authority>
      </template>
      <template #renderItem="{ item }: BasicCardListRenderItem<ProfileRecord>">
        <Card hoverable>
          <template #cover>
            <div class="h-full w-full !flex justify-center items-center text-center p-1 relative">
              <Image
                @click.stop
                wrapper-class-name="!w-32 !h-32 !flex !items-center overflow-hidden"
                :src="item.image || IMAGE_FALLBACK"
                placeholder
                :fallback="IMAGE_FALLBACK"
              />
              <div v-if="item.isEdge" class="absolute top-0 left-0">
                <img :src="edgefornt" />
                <span
                  class="absolute top-0 left-0 text-light-50 transform -rotate-45 translate-y-1 translate-x-0.5"
                  >边</span
                >
              </div>
            </div>
          </template>
          <template class="ant-card-actions" #actions>
            <Tooltip title="详情">
              <AuthIcon
                :auth="ProductPermission.DETAIL"
                class="!text-lg"
                icon="ant-design:eye-outlined"
                @click.stop="handleShowDetail(item)"
              />
            </Tooltip>
            <Tooltip title="编辑">
              <AuthIcon
                :auth="ProductPermission.UPDATE"
                class="!text-lg"
                icon="ant-design:form-outlined"
                @click.stop="handleUpdate(item)"
                :disabled="item.name == 'default'"
              />
            </Tooltip>
            <AuthDropDown
              @click.stop
              :trigger="['hover']"
              :drop-menu-list="[
                {
                  text: '默认',
                  event: DropMenuEvent.SET_DEFAULT,
                  icon: 'ant-design:unordered-list-outlined',
                  onClick: handleSetDefault.bind(null, item),
                  disabled: item.default,
                },
                {
                  text: '删除',
                  event: DropMenuEvent.DELETE,
                  auth: ProductPermission.DELETE,
                  icon: 'ant-design:delete-outlined',
                  popconfirm: {
                    title: !!item.isEdge ? '此产品来自边端,请谨慎删除' : '是否确认删除',
                    onConfirm: handleDelete.bind(null, [item.id]),
                    disabled: item.default || item.name == 'default',
                  },
                  disabled: item.default || item.name == 'default',
                },
              ]"
            />
          </template>
          <Card.Meta>
            <template #title>
              <span class="truncate"> {{ item.name }} </span>
            </template>
            <template #description>
              <div class="truncate h-11">
                <div class="truncate">{{ DeviceTypeName[item.deviceType] }} </div>
                <div class="truncate">{{ item.transportType }} </div>
              </div>
            </template>
          </Card.Meta>
        </Card>
      </template>
    </BasicCardList>
    <DeviceProfileModal @register="registerModal" @success="reload" />
    <DeviceProfileDrawer @register="registerDrawer" />
  </section>
</template>

<style lang="less" scoped>
  .profile-list:deep(.ant-image-img) {
    @apply !w-full !h-full;
  }

  .profile-list:deep(.ant-card-body) {
    @apply !p-4;
  }
</style>