基于Vue的图片剪切插件vue-cropper的使用

基本需求:

  • 对上传图片有大小、比例的要求,在上传之前可以进行裁剪
  • 先上传到本地,再上传至服务器,要求bsae64格式

剪裁效果图如下

屏幕快照 2021-11-13 下午2.45.03.png

vue-cropper的使用

安装

npm install vue-cropper --save

在main.js中引入

import VueCropper from 'vue-cropper'
Vue.use(VueCropper)

前端代码 这里使用ant-design-vue组件

代码如下
<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' }">
                <vue-cropper
                    ref="cropper"
                    :img="options.img"
                    :info="true"
                    :auto-crop="options.autoCrop"
                    :auto-crop-width="options.autoCropWidth"
                    :auto-crop-height="options.autoCropHeight"
                    :fixed-box="options.fixedBox"
                    @realTime="realTime"
                >
                </vue-cropper>
            </a-col>
            <a-col :xs="24" :md="12" :style="{ height: '350px' }">
                <div :style="previews.div" class="avatar-upload-preview">
                    <img :src="previews.url" :style="previews.img" />
                </div>
            </a-col>
        </a-row>
        <a-row style="margin-top: 20px;">
            <a-col :lg="2" :md="2" style="margin-right: 20px;">
                <a-upload
                    name="file"
                    accept="image/*"
                    :before-upload="beforeUpload"
                    :show-upload-list="false">
                    <a-button icon="upload">选择图片</a-button>
                </a-upload>
            </a-col>
            <a-col :lg="{ span: 1, offset: 2 }" :md="2">
                <a-button icon="plus" @click="changeScale(1)" />
            </a-col>
            <a-col :lg="{ span: 1, offset: 1 }" :md="2">
                <a-button icon="minus" @click="changeScale(-1)" />
            </a-col>
            <a-col :lg="{ span: 1, offset: 1 }" :md="2">
                <a-button icon="undo" @click="rotateLeft" />
            </a-col>
            <a-col :lg="{ span: 1, offset: 1 }" :md="2">
                <a-button icon="redo" @click="rotateRight" />
            </a-col>
            <a-col :lg="{ span: 2, offset: 6 }" :md="2">
                <a-button type="primary" @click="finish('blob')">保存</a-button>
            </a-col>
        </a-row>
    </a-modal>

这里用到了a-upload组件,因为是上传到本地,且不需要展示列表,所以需要将show-upload-list设为falsebefore-upload也需要返回false,阻止直接上传至服务器。

js代码如下

<script>
    export default {
        name: 'avatarModal',
        data () {
            return {
                visible: false,
                id: null,
                confirmLoading: false,
                options: {
                    img: '',  // 需要剪裁的图片
                    autoCrop: true, // 是否默认生成截图框
                    autoCropWidth: 150, // 默认生成截图框的宽度
                    autoCropHeight: 150, // 默认生成截图框的长度
                    fixedBox: true // 是否固定截图框的大小 不允许改变
                },
                previews: {} // 裁剪之后的结果
            }
        },
        methods: {
            edit (id) {
                this.visible = true
                this.id = id
            },
            close () {
                this.id = null
                this.visible = false
            },
            cancelHandel () {
                this.close()
            },
            // 图片放大/缩小
            changeScale (num) {
                num = num || 1
                this.$refs.cropper.changeScale(num)
            },
            // 左旋转
            rotateLeft () {
                this.$refs.cropper.rotateLeft()
            },
            // 右旋转
            rotateRight () {
                this.$refs.cropper.rotateRight()
            },
            beforeUpload (file) {
                const reader = new FileReader()
                // 转化为base64
                reader.readAsDataURL(file)
                reader.onload = () => {
                    // 获取到需要剪裁的图片 展示到剪裁框中
                    this.options.img = reader.result
                }
                return false
            },
            // 上传图片(点击保存按钮)
            finish (type) {
                this.$refs.cropper.getCropData((data) => {
                    this.visible = false
                    console.log(data)
                    // data为base64图片,供接口使用
                    this.$emit('save', data)
                })
            },
            // 裁剪之后的数据
            realTime (data) {
                this.previews = data
            }
        }
    }
</script>

裁剪之后的数据格式

{
           div: {
                height: "150px",
                width: "150px"
           },
           img: {
                height: "640px",
                transform: "scale(0.509375)translate3d(-422.0858895705521px, -361.22699386503064px, 0px)rotateZ(0deg)",
                width: "640px"
           },
           h: 150,
           w: 150,
           url: "blob:http://dev.paas.25.honops.com:8001/fe5d60f5-ce30-40e6-83a7-2a0d9ccb1c68",
            html: "
                  <div class="show-preview" style="width: 150px; height: 150px,; overflow: hidden">
                      <div style="width: 150px; height: 150px">
                          <img src=blob:http://dev.paas.25.honops.com:8001/fe5d60f5-ce30-40e6-83a7-2a0d9ccb1c68 style="width: 640px; height: 640px; transform:
                          scale(0.509375)translate3d(-422.0858895705521px, -361.22699386503064px, 0px)rotateZ(0deg)">
                      </div>
                  </div>"
}

主要用于右侧的图片裁剪预览展示。

vue-cropper的其他属性,仅供参考

option: {
        img: '', // 裁剪图片的地址
        info: true, // 裁剪框的大小信息
        outputSize: 0.8, // 裁剪生成图片的质量 [1至0.1]
        outputType: 'jpeg', // 裁剪生成图片的格式
        canScale: false, // 图片是否允许滚轮缩放
        autoCrop: true, // 是否默认生成截图框
        autoCropWidth: 300, // 默认生成截图框宽度
        autoCropHeight: 200, // 默认生成截图框高度
        fixedBox: true, // 固定截图框大小 不允许改变
        fixed: true, // 是否开启截图框宽高固定比例
        fixedNumber: [7, 5], // 截图框的宽高比例
        full: true, // 是否输出原图比例的截图
        canMoveBox: false, // 截图框能否拖动
        original: false, // 上传图片按照原始比例渲染
        centerBox: false, // 截图框是否被限制在图片里面
        infoTrue: true // true 为展示真实输出图片宽高 false 展示看到的截图框宽高
 }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,542评论 6 504
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,822评论 3 394
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,912评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,449评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,500评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,370评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,193评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,074评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,505评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,722评论 3 335
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,841评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,569评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,168评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,783评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,918评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,962评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,781评论 2 354

推荐阅读更多精彩内容