feat: add cropper comp
							parent
							
								
									189d509075
								
							
						
					
					
						commit
						7d4eec1ced
					
				|  | @ -0,0 +1,159 @@ | ||||||
|  | <script lang="ts" setup> | ||||||
|  | import type { CSSProperties } from 'vue'; | ||||||
|  | 
 | ||||||
|  | import type { CropperAvatarProps } from './typing'; | ||||||
|  | 
 | ||||||
|  | import { computed, ref, unref, watch, watchEffect } from 'vue'; | ||||||
|  | 
 | ||||||
|  | import { useVbenModal } from '@vben/common-ui'; | ||||||
|  | import { $t } from '@vben/locales'; | ||||||
|  | 
 | ||||||
|  | import { NButton } from 'naive-ui'; | ||||||
|  | 
 | ||||||
|  | import { message } from '#/adapter/naive'; | ||||||
|  | 
 | ||||||
|  | import cropperModal from './cropper-modal.vue'; | ||||||
|  | 
 | ||||||
|  | defineOptions({ name: 'CropperAvatar' }); | ||||||
|  | 
 | ||||||
|  | const props = withDefaults(defineProps<CropperAvatarProps>(), { | ||||||
|  |   width: 200, | ||||||
|  |   value: '', | ||||||
|  |   showBtn: true, | ||||||
|  |   btnProps: () => ({}), | ||||||
|  |   btnText: '', | ||||||
|  |   uploadApi: () => Promise.resolve(), | ||||||
|  |   size: 5, | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | const emit = defineEmits(['update:value', 'change']); | ||||||
|  | 
 | ||||||
|  | const sourceValue = ref(props.value || ''); | ||||||
|  | const prefixCls = 'cropper-avatar'; | ||||||
|  | const [CropperModal, modalApi] = useVbenModal({ | ||||||
|  |   connectedComponent: cropperModal, | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | const getClass = computed(() => [prefixCls]); | ||||||
|  | 
 | ||||||
|  | const getWidth = computed(() => `${`${props.width}`.replace(/px/, '')}px`); | ||||||
|  | 
 | ||||||
|  | const getIconWidth = computed( | ||||||
|  |   () => `${Number.parseInt(`${props.width}`.replace(/px/, '')) / 2}px`, | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | const getStyle = computed((): CSSProperties => ({ width: unref(getWidth) })); | ||||||
|  | 
 | ||||||
|  | const getImageWrapperStyle = computed( | ||||||
|  |   (): CSSProperties => ({ height: unref(getWidth), width: unref(getWidth) }), | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | watchEffect(() => { | ||||||
|  |   sourceValue.value = props.value || ''; | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | watch( | ||||||
|  |   () => sourceValue.value, | ||||||
|  |   (v: string) => { | ||||||
|  |     emit('update:value', v); | ||||||
|  |   }, | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | function handleUploadSuccess({ data, source }: any) { | ||||||
|  |   sourceValue.value = source; | ||||||
|  |   emit('change', { data, source }); | ||||||
|  |   message.success($t('ui.cropper.uploadSuccess')); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const closeModal = () => modalApi.close(); | ||||||
|  | const openModal = () => modalApi.open(); | ||||||
|  | 
 | ||||||
|  | defineExpose({ | ||||||
|  |   closeModal, | ||||||
|  |   openModal, | ||||||
|  | }); | ||||||
|  | </script> | ||||||
|  | 
 | ||||||
|  | <template> | ||||||
|  |   <div :class="getClass" :style="getStyle"> | ||||||
|  |     <div | ||||||
|  |       :class="`${prefixCls}-image-wrapper`" | ||||||
|  |       :style="getImageWrapperStyle" | ||||||
|  |       @click="openModal" | ||||||
|  |     > | ||||||
|  |       <div :class="`${prefixCls}-image-mask`" :style="getImageWrapperStyle"> | ||||||
|  |         <span | ||||||
|  |           :style="{ | ||||||
|  |             ...getImageWrapperStyle, | ||||||
|  |             width: `${getIconWidth}`, | ||||||
|  |             height: `${getIconWidth}`, | ||||||
|  |             lineHeight: `${getIconWidth}`, | ||||||
|  |           }" | ||||||
|  |           class="icon-[ant-design--cloud-upload-outlined] text-[#d6d6d6]" | ||||||
|  |         ></span> | ||||||
|  |       </div> | ||||||
|  |       <img v-if="sourceValue" :src="sourceValue" alt="avatar" /> | ||||||
|  |     </div> | ||||||
|  |     <NButton | ||||||
|  |       v-if="showBtn" | ||||||
|  |       :class="`${prefixCls}-upload-btn`" | ||||||
|  |       @click="openModal" | ||||||
|  |       v-bind="btnProps" | ||||||
|  |     > | ||||||
|  |       {{ btnText ? btnText : $t('ui.cropper.selectImage') }} | ||||||
|  |     </NButton> | ||||||
|  | 
 | ||||||
|  |     <CropperModal | ||||||
|  |       :size="size" | ||||||
|  |       :src="sourceValue" | ||||||
|  |       :upload-api="uploadApi" | ||||||
|  |       @upload-success="handleUploadSuccess" | ||||||
|  |     /> | ||||||
|  |   </div> | ||||||
|  | </template> | ||||||
|  | 
 | ||||||
|  | <style lang="scss" scoped> | ||||||
|  | .cropper-avatar { | ||||||
|  |   display: inline-block; | ||||||
|  |   text-align: center; | ||||||
|  | 
 | ||||||
|  |   &-image-wrapper { | ||||||
|  |     overflow: hidden; | ||||||
|  |     cursor: pointer; | ||||||
|  |     background: #fff; | ||||||
|  |     border: 1px solid #eee; | ||||||
|  |     border-radius: 50%; | ||||||
|  | 
 | ||||||
|  |     img { | ||||||
|  |       width: 100%; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   &-image-mask { | ||||||
|  |     position: absolute; | ||||||
|  |     display: flex; | ||||||
|  |     align-items: center; | ||||||
|  |     justify-content: center; | ||||||
|  |     width: inherit; | ||||||
|  |     height: inherit; | ||||||
|  |     cursor: pointer; | ||||||
|  |     background: rgb(0 0 0 / 40%); | ||||||
|  |     border: inherit; | ||||||
|  |     border-radius: inherit; | ||||||
|  |     opacity: 0; | ||||||
|  |     transition: opacity 0.4s; | ||||||
|  | 
 | ||||||
|  |     ::v-deep(svg) { | ||||||
|  |       margin: auto; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   &-image-mask:hover { | ||||||
|  |     opacity: 40; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   &-upload-btn { | ||||||
|  |     margin: 10px auto; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | </style> | ||||||
|  | @ -0,0 +1,350 @@ | ||||||
|  | <script lang="ts" setup> | ||||||
|  | import type { CropendResult, CropperModalProps, CropperType } from './typing'; | ||||||
|  | 
 | ||||||
|  | import { ref } from 'vue'; | ||||||
|  | 
 | ||||||
|  | import { useVbenModal } from '@vben/common-ui'; | ||||||
|  | import { $t } from '@vben/locales'; | ||||||
|  | import { dataURLtoBlob, isFunction } from '@vben/utils'; | ||||||
|  | 
 | ||||||
|  | import { NAvatar, NButton, NSpace, NTooltip, NUpload } from 'naive-ui'; | ||||||
|  | 
 | ||||||
|  | import CropperImage from './cropper.vue'; | ||||||
|  | 
 | ||||||
|  | defineOptions({ name: 'CropperModal' }); | ||||||
|  | 
 | ||||||
|  | const props = withDefaults(defineProps<CropperModalProps>(), { | ||||||
|  |   circled: true, | ||||||
|  |   size: 0, | ||||||
|  |   src: '', | ||||||
|  |   uploadApi: () => Promise.resolve(), | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | const emit = defineEmits(['uploadSuccess', 'uploadError', 'register']); | ||||||
|  | 
 | ||||||
|  | let filename = ''; | ||||||
|  | const src = ref(props.src || ''); | ||||||
|  | const previewSource = ref(''); | ||||||
|  | const cropper = ref<CropperType>(); | ||||||
|  | let scaleX = 1; | ||||||
|  | let scaleY = 1; | ||||||
|  | 
 | ||||||
|  | const prefixCls = 'cropper-am'; | ||||||
|  | const [Modal, modalApi] = useVbenModal({ | ||||||
|  |   onConfirm: handleOk, | ||||||
|  |   onOpenChange(isOpen) { | ||||||
|  |     if (isOpen) { | ||||||
|  |       // 打开时,进行 loading 加载。后续 CropperImage 组件加载完毕,会自动关闭 loading(通过 handleReady) | ||||||
|  |       modalLoading(true); | ||||||
|  |     } else { | ||||||
|  |       // 关闭时,清空右侧预览 | ||||||
|  |       previewSource.value = ''; | ||||||
|  |       modalLoading(false); | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | function modalLoading(loading: boolean) { | ||||||
|  |   modalApi.setState({ confirmLoading: loading, loading }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Block upload | ||||||
|  | function handleBeforeUpload(file: File) { | ||||||
|  |   if (props.size > 0 && file.size > 1024 * 1024 * props.size) { | ||||||
|  |     emit('uploadError', { msg: $t('ui.cropper.imageTooBig') }); | ||||||
|  |     return false; | ||||||
|  |   } | ||||||
|  |   const reader = new FileReader(); | ||||||
|  |   reader.readAsDataURL(file); | ||||||
|  |   src.value = ''; | ||||||
|  |   previewSource.value = ''; | ||||||
|  |   reader.addEventListener('load', (e) => { | ||||||
|  |     src.value = (e.target?.result as string) ?? ''; | ||||||
|  |     filename = file.name; | ||||||
|  |   }); | ||||||
|  |   return false; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function handleCropend({ imgBase64 }: CropendResult) { | ||||||
|  |   previewSource.value = imgBase64; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function handleReady(cropperInstance: CropperType) { | ||||||
|  |   cropper.value = cropperInstance; | ||||||
|  |   // 画布加载完毕 关闭 loading | ||||||
|  |   modalLoading(false); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function handlerToolbar(event: string, arg?: number) { | ||||||
|  |   if (event === 'scaleX') { | ||||||
|  |     scaleX = arg = scaleX === -1 ? 1 : -1; | ||||||
|  |   } | ||||||
|  |   if (event === 'scaleY') { | ||||||
|  |     scaleY = arg = scaleY === -1 ? 1 : -1; | ||||||
|  |   } | ||||||
|  |   (cropper?.value as any)?.[event]?.(arg); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | async function handleOk() { | ||||||
|  |   const uploadApi = props.uploadApi; | ||||||
|  |   if (uploadApi && isFunction(uploadApi)) { | ||||||
|  |     if (!previewSource.value) { | ||||||
|  |       message.warn('未选择图片'); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     const blob = dataURLtoBlob(previewSource.value); | ||||||
|  |     try { | ||||||
|  |       modalLoading(true); | ||||||
|  |       const url = await uploadApi({ file: blob, filename, name: 'file' }); | ||||||
|  |       emit('uploadSuccess', { data: url, source: previewSource.value }); | ||||||
|  |       await modalApi.close(); | ||||||
|  |     } finally { | ||||||
|  |       modalLoading(false); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | </script> | ||||||
|  | 
 | ||||||
|  | <template> | ||||||
|  |   <Modal | ||||||
|  |     v-bind="$attrs" | ||||||
|  |     :confirm-text="$t('ui.cropper.okText')" | ||||||
|  |     :fullscreen-button="false" | ||||||
|  |     :title="$t('ui.cropper.modalTitle')" | ||||||
|  |     class="w-[800px]" | ||||||
|  |   > | ||||||
|  |     <div :class="prefixCls"> | ||||||
|  |       <div :class="`${prefixCls}-left`" class="w-full"> | ||||||
|  |         <div :class="`${prefixCls}-cropper`"> | ||||||
|  |           <CropperImage | ||||||
|  |             v-if="src" | ||||||
|  |             :circled="circled" | ||||||
|  |             :src="src" | ||||||
|  |             height="300px" | ||||||
|  |             @cropend="handleCropend" | ||||||
|  |             @ready="handleReady" | ||||||
|  |           /> | ||||||
|  |         </div> | ||||||
|  | 
 | ||||||
|  |         <div :class="`${prefixCls}-toolbar`"> | ||||||
|  |           <NUpload | ||||||
|  |             :before-upload="handleBeforeUpload" | ||||||
|  |             :file-list="[]" | ||||||
|  |             accept="image/*" | ||||||
|  |           > | ||||||
|  |             <NTooltip :title="$t('ui.cropper.selectImage')" placement="bottom"> | ||||||
|  |               <NButton size="small" type="primary"> | ||||||
|  |                 <template #icon> | ||||||
|  |                   <div class="flex items-center justify-center"> | ||||||
|  |                     <span class="icon-[ant-design--upload-outlined]"></span> | ||||||
|  |                   </div> | ||||||
|  |                 </template> | ||||||
|  |               </NButton> | ||||||
|  |             </NTooltip> | ||||||
|  |           </NUpload> | ||||||
|  |           <NSpace> | ||||||
|  |             <NTooltip :title="$t('ui.cropper.btn_reset')" placement="bottom"> | ||||||
|  |               <NButton | ||||||
|  |                 :disabled="!src" | ||||||
|  |                 size="small" | ||||||
|  |                 type="primary" | ||||||
|  |                 @click="handlerToolbar('reset')" | ||||||
|  |               > | ||||||
|  |                 <template #icon> | ||||||
|  |                   <div class="flex items-center justify-center"> | ||||||
|  |                     <span class="icon-[ant-design--reload-outlined]"></span> | ||||||
|  |                   </div> | ||||||
|  |                 </template> | ||||||
|  |               </NButton> | ||||||
|  |             </NTooltip> | ||||||
|  |             <NTooltip | ||||||
|  |               :title="$t('ui.cropper.btn_rotate_left')" | ||||||
|  |               placement="bottom" | ||||||
|  |             > | ||||||
|  |               <NButton | ||||||
|  |                 :disabled="!src" | ||||||
|  |                 size="small" | ||||||
|  |                 type="primary" | ||||||
|  |                 @click="handlerToolbar('rotate', -45)" | ||||||
|  |               > | ||||||
|  |                 <template #icon> | ||||||
|  |                   <div class="flex items-center justify-center"> | ||||||
|  |                     <span | ||||||
|  |                       class="icon-[ant-design--rotate-left-outlined]" | ||||||
|  |                     ></span> | ||||||
|  |                   </div> | ||||||
|  |                 </template> | ||||||
|  |               </NButton> | ||||||
|  |             </NTooltip> | ||||||
|  |             <NTooltip | ||||||
|  |               :title="$t('ui.cropper.btn_rotate_right')" | ||||||
|  |               placement="bottom" | ||||||
|  |             > | ||||||
|  |               <NButton | ||||||
|  |                 :disabled="!src" | ||||||
|  |                 pre-icon="ant-design:rotate-right-outlined" | ||||||
|  |                 size="small" | ||||||
|  |                 type="primary" | ||||||
|  |                 @click="handlerToolbar('rotate', 45)" | ||||||
|  |               > | ||||||
|  |                 <template #icon> | ||||||
|  |                   <div class="flex items-center justify-center"> | ||||||
|  |                     <span | ||||||
|  |                       class="icon-[ant-design--rotate-right-outlined]" | ||||||
|  |                     ></span> | ||||||
|  |                   </div> | ||||||
|  |                 </template> | ||||||
|  |               </NButton> | ||||||
|  |             </NTooltip> | ||||||
|  |             <NTooltip :title="$t('ui.cropper.btn_scale_x')" placement="bottom"> | ||||||
|  |               <NButton | ||||||
|  |                 :disabled="!src" | ||||||
|  |                 size="small" | ||||||
|  |                 type="primary" | ||||||
|  |                 @click="handlerToolbar('scaleX')" | ||||||
|  |               > | ||||||
|  |                 <template #icon> | ||||||
|  |                   <div class="flex items-center justify-center"> | ||||||
|  |                     <span class="icon-[vaadin--arrows-long-h]"></span> | ||||||
|  |                   </div> | ||||||
|  |                 </template> | ||||||
|  |               </NButton> | ||||||
|  |             </NTooltip> | ||||||
|  |             <NTooltip :title="$t('ui.cropper.btn_scale_y')" placement="bottom"> | ||||||
|  |               <NButton | ||||||
|  |                 :disabled="!src" | ||||||
|  |                 size="small" | ||||||
|  |                 type="primary" | ||||||
|  |                 @click="handlerToolbar('scaleY')" | ||||||
|  |               > | ||||||
|  |                 <template #icon> | ||||||
|  |                   <div class="flex items-center justify-center"> | ||||||
|  |                     <span class="icon-[vaadin--arrows-long-v]"></span> | ||||||
|  |                   </div> | ||||||
|  |                 </template> | ||||||
|  |               </NButton> | ||||||
|  |             </NTooltip> | ||||||
|  |             <NTooltip :title="$t('ui.cropper.btn_zoom_in')" placement="bottom"> | ||||||
|  |               <NButton | ||||||
|  |                 :disabled="!src" | ||||||
|  |                 size="small" | ||||||
|  |                 type="primary" | ||||||
|  |                 @click="handlerToolbar('zoom', 0.1)" | ||||||
|  |               > | ||||||
|  |                 <template #icon> | ||||||
|  |                   <div class="flex items-center justify-center"> | ||||||
|  |                     <span class="icon-[ant-design--zoom-in-outlined]"></span> | ||||||
|  |                   </div> | ||||||
|  |                 </template> | ||||||
|  |               </NButton> | ||||||
|  |             </NTooltip> | ||||||
|  |             <NTooltip :title="$t('ui.cropper.btn_zoom_out')" placement="bottom"> | ||||||
|  |               <NButton | ||||||
|  |                 :disabled="!src" | ||||||
|  |                 size="small" | ||||||
|  |                 type="primary" | ||||||
|  |                 @click="handlerToolbar('zoom', -0.1)" | ||||||
|  |               > | ||||||
|  |                 <template #icon> | ||||||
|  |                   <div class="flex items-center justify-center"> | ||||||
|  |                     <span class="icon-[ant-design--zoom-out-outlined]"></span> | ||||||
|  |                   </div> | ||||||
|  |                 </template> | ||||||
|  |               </NButton> | ||||||
|  |             </NTooltip> | ||||||
|  |           </NSpace> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |       <div :class="`${prefixCls}-right`"> | ||||||
|  |         <div :class="`${prefixCls}-preview`"> | ||||||
|  |           <img | ||||||
|  |             v-if="previewSource" | ||||||
|  |             :alt="$t('ui.cropper.preview')" | ||||||
|  |             :src="previewSource" | ||||||
|  |           /> | ||||||
|  |         </div> | ||||||
|  |         <template v-if="previewSource"> | ||||||
|  |           <div :class="`${prefixCls}-group`"> | ||||||
|  |             <NAvatar :src="previewSource" size="large" /> | ||||||
|  |             <NAvatar :size="48" :src="previewSource" /> | ||||||
|  |             <NAvatar :size="64" :src="previewSource" /> | ||||||
|  |             <NAvatar :size="80" :src="previewSource" /> | ||||||
|  |           </div> | ||||||
|  |         </template> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   </Modal> | ||||||
|  | </template> | ||||||
|  | 
 | ||||||
|  | <style lang="scss"> | ||||||
|  | .cropper-am { | ||||||
|  |   display: flex; | ||||||
|  | 
 | ||||||
|  |   &-left, | ||||||
|  |   &-right { | ||||||
|  |     height: 340px; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   &-left { | ||||||
|  |     width: 55%; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   &-right { | ||||||
|  |     width: 45%; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   &-cropper { | ||||||
|  |     height: 300px; | ||||||
|  |     background: #eee; | ||||||
|  |     background-image: | ||||||
|  |       linear-gradient( | ||||||
|  |         45deg, | ||||||
|  |         rgb(0 0 0 / 25%) 25%, | ||||||
|  |         transparent 0, | ||||||
|  |         transparent 75%, | ||||||
|  |         rgb(0 0 0 / 25%) 0 | ||||||
|  |       ), | ||||||
|  |       linear-gradient( | ||||||
|  |         45deg, | ||||||
|  |         rgb(0 0 0 / 25%) 25%, | ||||||
|  |         transparent 0, | ||||||
|  |         transparent 75%, | ||||||
|  |         rgb(0 0 0 / 25%) 0 | ||||||
|  |       ); | ||||||
|  |     background-position: | ||||||
|  |       0 0, | ||||||
|  |       12px 12px; | ||||||
|  |     background-size: 24px 24px; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   &-toolbar { | ||||||
|  |     display: flex; | ||||||
|  |     align-items: center; | ||||||
|  |     justify-content: space-between; | ||||||
|  |     margin-top: 10px; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   &-preview { | ||||||
|  |     width: 220px; | ||||||
|  |     height: 220px; | ||||||
|  |     margin: 0 auto; | ||||||
|  |     overflow: hidden; | ||||||
|  |     border: 1px solid #eee; | ||||||
|  |     border-radius: 50%; | ||||||
|  | 
 | ||||||
|  |     img { | ||||||
|  |       width: 100%; | ||||||
|  |       height: 100%; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   &-group { | ||||||
|  |     display: flex; | ||||||
|  |     align-items: center; | ||||||
|  |     justify-content: space-around; | ||||||
|  |     padding-top: 8px; | ||||||
|  |     margin-top: 8px; | ||||||
|  |     border-top: 1px solid #eee; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | </style> | ||||||
|  | @ -0,0 +1,173 @@ | ||||||
|  | <script lang="ts" setup> | ||||||
|  | import type { CSSProperties } from 'vue'; | ||||||
|  | 
 | ||||||
|  | import type { CropperProps } from './typing'; | ||||||
|  | 
 | ||||||
|  | import { computed, onMounted, onUnmounted, ref, unref, useAttrs } from 'vue'; | ||||||
|  | 
 | ||||||
|  | import { useDebounceFn } from '@vueuse/core'; | ||||||
|  | import Cropper from 'cropperjs'; | ||||||
|  | 
 | ||||||
|  | import { defaultOptions } from './typing'; | ||||||
|  | 
 | ||||||
|  | import 'cropperjs/dist/cropper.css'; | ||||||
|  | 
 | ||||||
|  | defineOptions({ name: 'CropperImage' }); | ||||||
|  | 
 | ||||||
|  | const props = withDefaults(defineProps<CropperProps>(), { | ||||||
|  |   src: '', | ||||||
|  |   alt: '', | ||||||
|  |   circled: false, | ||||||
|  |   realTimePreview: true, | ||||||
|  |   height: '360px', | ||||||
|  |   crossorigin: undefined, | ||||||
|  |   imageStyle: () => ({}), | ||||||
|  |   options: () => ({}), | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | const emit = defineEmits(['cropend', 'ready', 'cropendError']); | ||||||
|  | const attrs = useAttrs(); | ||||||
|  | 
 | ||||||
|  | type ElRef<T extends HTMLElement = HTMLDivElement> = null | T; | ||||||
|  | const imgElRef = ref<ElRef<HTMLImageElement>>(); | ||||||
|  | const cropper = ref<Cropper | null>(); | ||||||
|  | const isReady = ref(false); | ||||||
|  | 
 | ||||||
|  | const prefixCls = 'cropper-image'; | ||||||
|  | const debounceRealTimeCropped = useDebounceFn(realTimeCropped, 80); | ||||||
|  | 
 | ||||||
|  | const getImageStyle = computed((): CSSProperties => { | ||||||
|  |   return { | ||||||
|  |     height: props.height, | ||||||
|  |     maxWidth: '100%', | ||||||
|  |     ...props.imageStyle, | ||||||
|  |   }; | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | const getClass = computed(() => { | ||||||
|  |   return [ | ||||||
|  |     prefixCls, | ||||||
|  |     attrs.class, | ||||||
|  |     { | ||||||
|  |       [`${prefixCls}--circled`]: props.circled, | ||||||
|  |     }, | ||||||
|  |   ]; | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | const getWrapperStyle = computed((): CSSProperties => { | ||||||
|  |   return { height: `${`${props.height}`.replace(/px/, '')}px` }; | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | onMounted(init); | ||||||
|  | 
 | ||||||
|  | onUnmounted(() => { | ||||||
|  |   cropper.value?.destroy(); | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | async function init() { | ||||||
|  |   const imgEl = unref(imgElRef); | ||||||
|  |   if (!imgEl) { | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |   cropper.value = new Cropper(imgEl, { | ||||||
|  |     ...defaultOptions, | ||||||
|  |     ready: () => { | ||||||
|  |       isReady.value = true; | ||||||
|  |       realTimeCropped(); | ||||||
|  |       emit('ready', cropper.value); | ||||||
|  |     }, | ||||||
|  |     crop() { | ||||||
|  |       debounceRealTimeCropped(); | ||||||
|  |     }, | ||||||
|  |     zoom() { | ||||||
|  |       debounceRealTimeCropped(); | ||||||
|  |     }, | ||||||
|  |     cropmove() { | ||||||
|  |       debounceRealTimeCropped(); | ||||||
|  |     }, | ||||||
|  |     ...props.options, | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Real-time display preview | ||||||
|  | function realTimeCropped() { | ||||||
|  |   props.realTimePreview && cropped(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // event: return base64 and width and height information after cropping | ||||||
|  | function cropped() { | ||||||
|  |   if (!cropper.value) { | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |   const imgInfo = cropper.value.getData(); | ||||||
|  |   const canvas = props.circled | ||||||
|  |     ? getRoundedCanvas() | ||||||
|  |     : cropper.value.getCroppedCanvas(); | ||||||
|  |   canvas.toBlob((blob) => { | ||||||
|  |     if (!blob) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     const fileReader: FileReader = new FileReader(); | ||||||
|  |     fileReader.readAsDataURL(blob); | ||||||
|  |     fileReader.onloadend = (e) => { | ||||||
|  |       emit('cropend', { | ||||||
|  |         imgBase64: e.target?.result ?? '', | ||||||
|  |         imgInfo, | ||||||
|  |       }); | ||||||
|  |     }; | ||||||
|  |     // eslint-disable-next-line unicorn/prefer-add-event-listener | ||||||
|  |     fileReader.onerror = () => { | ||||||
|  |       emit('cropendError'); | ||||||
|  |     }; | ||||||
|  |   }, 'image/png'); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Get a circular picture canvas | ||||||
|  | function getRoundedCanvas() { | ||||||
|  |   const sourceCanvas = cropper.value!.getCroppedCanvas(); | ||||||
|  |   const canvas = document.createElement('canvas'); | ||||||
|  |   const context = canvas.getContext('2d')!; | ||||||
|  |   const width = sourceCanvas.width; | ||||||
|  |   const height = sourceCanvas.height; | ||||||
|  |   canvas.width = width; | ||||||
|  |   canvas.height = height; | ||||||
|  |   context.imageSmoothingEnabled = true; | ||||||
|  |   context.drawImage(sourceCanvas, 0, 0, width, height); | ||||||
|  |   context.globalCompositeOperation = 'destination-in'; | ||||||
|  |   context.beginPath(); | ||||||
|  |   context.arc( | ||||||
|  |     width / 2, | ||||||
|  |     height / 2, | ||||||
|  |     Math.min(width, height) / 2, | ||||||
|  |     0, | ||||||
|  |     2 * Math.PI, | ||||||
|  |     true, | ||||||
|  |   ); | ||||||
|  |   context.fill(); | ||||||
|  |   return canvas; | ||||||
|  | } | ||||||
|  | </script> | ||||||
|  | 
 | ||||||
|  | <template> | ||||||
|  |   <div :class="getClass" :style="getWrapperStyle"> | ||||||
|  |     <img | ||||||
|  |       v-show="isReady" | ||||||
|  |       ref="imgElRef" | ||||||
|  |       :alt="alt" | ||||||
|  |       :crossorigin="crossorigin" | ||||||
|  |       :src="src" | ||||||
|  |       :style="getImageStyle" | ||||||
|  |     /> | ||||||
|  |   </div> | ||||||
|  | </template> | ||||||
|  | 
 | ||||||
|  | <style lang="scss"> | ||||||
|  | .cropper-image { | ||||||
|  |   &--circled { | ||||||
|  |     .cropper-view-box, | ||||||
|  |     .cropper-face { | ||||||
|  |       border-radius: 50%; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | </style> | ||||||
|  | @ -0,0 +1,3 @@ | ||||||
|  | export { default as CropperAvatar } from './cropper-avatar.vue'; | ||||||
|  | export { default as CropperImage } from './cropper.vue'; | ||||||
|  | export type { CropperType } from './typing'; | ||||||
|  | @ -0,0 +1,68 @@ | ||||||
|  | import type Cropper from 'cropperjs'; | ||||||
|  | import type { ButtonProps } from 'naive-ui'; | ||||||
|  | 
 | ||||||
|  | import type { CSSProperties } from 'vue'; | ||||||
|  | 
 | ||||||
|  | export interface apiFunParams { | ||||||
|  |   file: Blob; | ||||||
|  |   filename: string; | ||||||
|  |   name: string; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export interface CropendResult { | ||||||
|  |   imgBase64: string; | ||||||
|  |   imgInfo: Cropper.Data; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export interface CropperProps { | ||||||
|  |   src?: string; | ||||||
|  |   alt?: string; | ||||||
|  |   circled?: boolean; | ||||||
|  |   realTimePreview?: boolean; | ||||||
|  |   height?: number | string; | ||||||
|  |   crossorigin?: '' | 'anonymous' | 'use-credentials' | undefined; | ||||||
|  |   imageStyle?: CSSProperties; | ||||||
|  |   options?: Cropper.Options; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export interface CropperAvatarProps { | ||||||
|  |   width?: number | string; | ||||||
|  |   value?: string; | ||||||
|  |   showBtn?: boolean; | ||||||
|  |   btnProps?: ButtonProps; | ||||||
|  |   btnText?: string; | ||||||
|  |   uploadApi?: (params: apiFunParams) => Promise<any>; | ||||||
|  |   size?: number; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export interface CropperModalProps { | ||||||
|  |   circled?: boolean; | ||||||
|  |   uploadApi?: (params: apiFunParams) => Promise<any>; | ||||||
|  |   src?: string; | ||||||
|  |   size?: number; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export const defaultOptions: Cropper.Options = { | ||||||
|  |   aspectRatio: 1, | ||||||
|  |   zoomable: true, | ||||||
|  |   zoomOnTouch: true, | ||||||
|  |   zoomOnWheel: true, | ||||||
|  |   cropBoxMovable: true, | ||||||
|  |   cropBoxResizable: true, | ||||||
|  |   toggleDragModeOnDblclick: true, | ||||||
|  |   autoCrop: true, | ||||||
|  |   background: true, | ||||||
|  |   highlight: true, | ||||||
|  |   center: true, | ||||||
|  |   responsive: true, | ||||||
|  |   restore: true, | ||||||
|  |   checkCrossOrigin: true, | ||||||
|  |   checkOrientation: true, | ||||||
|  |   scalable: true, | ||||||
|  |   modal: true, | ||||||
|  |   guides: true, | ||||||
|  |   movable: true, | ||||||
|  |   rotatable: true, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export type { Cropper as CropperType }; | ||||||
		Loading…
	
		Reference in New Issue
	
	 xingyu4j
						xingyu4j