vue3中使用vue-cropper进行图片剪裁

之前在使用vue-cropper 的时候,搜了一下资料基本都是照搬。要么不符合自己的需求,要么就是无法正常运行,要么就是无法复制,非常不符合我们搬运工的气质。于是就参考网上大佬的文章进行了修改,可以正常使用。希望对需要的同学有帮助。
首先安装vue-cropper

cnpm install vue-cropper --save

我使用的是ant-vue。下面贴出代码.
html 部分

<template>

    <a-modal
            title="图像剪裁"
            :visible="visible"
            :mask-closable="false"
            :confirm-loading="confirmLoading"
            :width="700"
            :footer="null"
            @cancel="cancelHandel">
            <a-row>
                <a-col :xs="24" :md="12" :style="{ height: '350px' }">
                    <vueCropper
                        ref="cropperRef"
                        :img="options.img"
                        :info="true"
                        :infoTrue="options.infoTrue"
                        :auto-crop="options.autoCrop"
                        :fixed-box="options.fixedBox"
                        :fixedNumber="options.fixedNumber"
                        :fixed="options.fixed"
                        :centerBox="options.centerBox"
                        @realTime="realTime"
                    >
                    </vueCropper>
                </a-col>
                <a-col :xs="24" :md="12" :style="{ height: '350px' }">
                    <div style="margin-left:10px;margin-bottom: 10px;">图片预览:</div>
                    <div :style="previews.div" style="margin:auto">
                        <!-- <img :src="previews.url" :style="previews.img" /> -->
                        <div v-html="previews.html"  ></div>
                    </div>
                </a-col>
            </a-row>
            <a-row :gutter="24" style="margin-top: 20px;">
                <a-col  :span="4" style="margin-right: 20px;">
                    <a-upload
                        name="file"
                        accept="image/*"
                        :before-upload="beforeUpload"
                        :show-upload-list="false">
                        <a-button  type="primary">选择图片</a-button>
                    </a-upload>
                </a-col>
                <a-col  :span="3">
                    <a-button type="primary"   @click="refreshCrop" >复位</a-button>
                </a-col>
                <a-col :span="3" :offset="13">
                    <a-button type="primary" @click="finish('blob')">保存</a-button>
                </a-col>
            </a-row>
    </a-modal>

    
</template>

script部分代码

<script lang="ts">
import { defineComponent, reactive, Ref, ref, toRefs, UnwrapNestedRefs, watch } from 'vue'
import VueCropper from "vue-cropper/src/vue-cropper.vue"
//import VueCropper from "vue-cropper" 注意不要使用这个
interface Options{
    img:string | ArrayBuffer | null // 裁剪图片的地址
    info: true, // 裁剪框的大小信息
    outputSize:number, // 裁剪生成图片的质量 [1至0.1]
    outputType: 'jpeg', // 裁剪生成图片的格式
    canScale: boolean, // 图片是否允许滚轮缩放
    autoCrop: boolean, // 是否默认生成截图框
    autoCropWidth:number // 默认生成截图框宽度
    autoCropHeight:number // 默认生成截图框高度
    fixedBox:boolean // 固定截图框大小 不允许改变
    fixed: boolean, // 是否开启截图框宽高固定比例
    fixedNumber: Array<number>, // 截图框的宽高比例  需要配合centerBox一起使用才能生效
    full: boolean, // 是否输出原图比例的截图
    canMoveBox: boolean, // 截图框能否拖动
    original: boolean, // 上传图片按照原始比例渲染
    centerBox: boolean, // 截图框是否被限制在图片里面
    infoTrue: boolean // true 为展示真实输出图片宽高 false 展示看到的截图框宽高
}
export default defineComponent({
    components:{
        VueCropper
    },
    props:{
            imgPath:{
                type:String,
                default:""
            }
          },
    setup(props,{expose,emit}) {
        const {imgPath} = toRefs(props)
          
         
          const  options:UnwrapNestedRefs<Options> = reactive({
                    img: '',  // 需要剪裁的图片
                    autoCrop: true, // 是否默认生成截图框
                    autoCropWidth: 150, // 默认生成截图框的宽度
                    autoCropHeight: 150, // 默认生成截图框的长度
                    fixedBox: false, // 是否固定截图框的大小 不允许改变
                    info: true, // 裁剪框的大小信息
                    outputSize: 0.5, // 裁剪生成图片的质量 [1至0.1]
                    outputType: 'jpeg', // 裁剪生成图片的格式
                    canScale: false, // 图片是否允许滚轮缩放
                    fixed: true, // 是否开启截图框宽高固定比例
                    fixedNumber: [5, 4], // 截图框的宽高比例 需要配合centerBox一起使用才能生效
                    full: true, // 是否输出原图比例的截图
                    canMoveBox: false, // 截图框能否拖动
                    original: false, // 上传图片按照原始比例渲染
                    centerBox: true, // 截图框是否被限制在图片里面
                    infoTrue: true, // true 为展示真实输出图片宽高 false 展示看到的截图框宽高
                })
          const previews:Ref<any> = ref({})
          const visible = ref(false)
          const confirmLoading = ref(false)
          const cropperRef:Ref<any> = ref(null)
          if(imgPath.value){
            options.img = imgPath.value
          }
          watch(imgPath,()=>{
            if(imgPath.value){
                options.img = imgPath.value
            }
          })
            expose({
                showModal(){
                    show()
                }
            })
           const show= ()=> {
                visible.value = true
            }
            const close= () => {
                visible.value = false
            }
            const cancelHandel = () => {
               close()
            }
            const beforeUpload = (file) => {
                const reader = new FileReader()
                // 转化为base64
                reader.readAsDataURL(file)
                reader.onload = () => {
                    // 获取到需要剪裁的图片 展示到剪裁框中
                    options.img = reader.result
                }
                return false
            }
            const refreshCrop = () => {
                cropperRef.value.refresh()
            }
            // 上传图片(点击保存按钮)
            const finish = (type) => {
              
                console.log(cropperRef.value)
               cropperRef.value.getCropData((data) => {
                   visible.value = false
                    console.log(data)
                    // data为base64图片,供接口使用
                    emit('saveResult', data)
                })
            }
            // 裁剪之后的数据
           const  realTime = (data) => {
                 previews.value = data
            }    
        return {
            visible,confirmLoading,options,previews,cropperRef,
            finish,beforeUpload,rotateLeft,rotateRight,changeScale,
            cancelHandel,realTime,refreshCrop
        }
        
    },
})
</script>
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352