PictureComponent.vue 5.61 KB
<script lang="ts">
  export default {
    inheritAttrs: false,
  };
</script>
<script lang="ts" setup>
  import { computed, ref, watch } from 'vue';
  import { Tooltip, Image as AntImage } from 'ant-design-vue';
  import {
    getUpdateTime,
    DEFAULT_RADIO_RECORD,
    fontSize,
    RadioRecord,
  } from '../../detail/config/util';
  import { PictureComponentValue } from './pictureComponent.config';

  const props = defineProps<{
    layout?: Recordable;
    value?: PictureComponentValue;
    radio?: RadioRecord;
  }>();

  const fallback =
    '';

  const getImagBase64 = ref(fallback);

  const getRadio = computed(() => {
    return props.radio || DEFAULT_RADIO_RECORD;
  });

  const getWidth = computed(() => {
    const marign = 5;
    const offsetHight = 62 + (props.radio ? 62 : 0);
    const { width = 240, height = 200 } = props.radio || {};
    return width > height - offsetHight ? height - offsetHight - marign : width - marign;
  });

  watch(
    () => props.value?.value,
    () => {
      if (props.value?.value) {
        getBase64Image(props.value.value);
      }
    }
  );

  const getBase64Image = (url: string) => {
    let canvas: Nullable<HTMLCanvasElement> = document.createElement('canvas');
    const ctx = canvas.getContext('2d');
    let image: Nullable<HTMLImageElement> = new Image();

    image.onload = function () {
      canvas!.height = image!.height;
      canvas!.width = image!.width;
      ctx?.drawImage(image!, 0, 0);
      const dataUrl = canvas!.toDataURL('image/png');
      getImagBase64.value = dataUrl;
      image = null;
      canvas = null;
    };
    image.setAttribute('crossOrigin', 'Anonymous');
    image.src = url;
  };
</script>

<template>
  <section
    class="w-full h-full flex flex-col justify-center items-center justify-between widget-picture"
  >
    <AntImage :width="getWidth" :src="getImagBase64" :fallback="fallback" />
    <div
      class="w-full text-center truncate p-5"
      :style="{ fontSize: fontSize({ radioRecord: getRadio, basic: 12, max: 12 }), color: '#999' }"
    >
      <Tooltip placement="top" :title="getUpdateTime(props.value?.updateTime)">
        <span class="mr-1">更新时间:</span>
        <span class="truncate">
          {{ getUpdateTime(props.value?.updateTime) }}
        </span>
      </Tooltip>
    </div>
  </section>
</template>

<style scoped lang="less">
  .widget-picture:deep(.ant-image) {
    flex: auto;
    display: flex;
    justify-content: center;
    align-items: center;
  }
</style>