<script lang="ts" setup>
import type { Boundaries } from 'vue-advanced-cropper'
import { Cropper } from 'vue-advanced-cropper'
import 'vue-advanced-cropper/dist/style.css'

export interface Props {
  /** Crop frame aspect ratio (width/height), default 1/1 */
  stencilAspectRatio?: number
  /** The ratio of the longest edge of the cut box to the length of the cut screen, default 0.9, not more than 1 */
  stencilSizePercentage?: number
}
const props = withDefaults(defineProps<Props>(), {
  stencilAspectRatio: 1 / 1,
  stencilSizePercentage: 0.9,
})

const { modelValue: file } = defineModel<{
  /** Images to be cropped */
  modelValue: File | null
}>()

const cropperDialog = ref(false)

const cropper = ref<InstanceType<typeof Cropper>>()

const cropperFlag = ref(false)

const cropperImage = reactive({
  src: '',
  type: 'image/jpg',
})

const stencilSize = ({ boundaries }: { boundaries: Boundaries }) => {
  return {
    width: boundaries.width * props.stencilSizePercentage,
    height: boundaries.height * props.stencilSizePercentage,
  }
}

watch(file, (file, _, onCleanup) => {
  let expired = false
  onCleanup(() => expired = true)

  if (file && !cropperFlag.value) {
    cropperDialog.value = true
    const reader = new FileReader()
    reader.readAsDataURL(file)
    reader.onload = (e) => {
      if (expired)
        return
      cropperImage.src = e.target?.result as string
      cropperImage.type = file.type
    }
  }
  cropperFlag.value = false
})

const cropImage = () => {
  if (cropper.value && file.value) {
    cropperFlag.value = true
    cropperDialog.value = false
    const { canvas } = cropper.value.getResult()
    canvas?.toBlob((blob) => {
      file.value = new File([blob as any], `cropped${file.value?.name}` as string, { type: blob?.type })
    }, cropperImage.type)
  }
}
</script>

<template>
  <ModalDialog v-model="cropperDialog" :use-v-if="false" max-w-500px flex>
    <div flex-1 w-0>
      <div text-lg text-center my2 px3>
        <h1>
          {{ $t('action.edit') }}
        </h1>
      </div>
      <div aspect-ratio-1>
        <Cropper
          ref="cropper"
          class="overflow-hidden w-full h-full"
          :src="cropperImage.src"
          :resize-image="{
            adjustStencil: false,
          }"
          :stencil-size="stencilSize"
          :stencil-props="{
            aspectRatio: props.stencilAspectRatio,
            movable: false,
            resizable: false,
            handlers: {},
          }"
          image-restriction="stencil"
        />
      </div>
      <div m-4>
        <button
          btn-solid w-full rounded text-sm
          @click="cropImage()"
        >
          {{ $t('action.confirm') }}
        </button>
      </div>
    </div>
  </ModalDialog>
</template>