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