Skip to content

vue-cropper

下载 npm install vue-cropper

引入

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

封装

components文件夹新建文件 cropper.vue
vue
<template>
  <a-modal v-model="visible" title="裁剪" :width="1000">
    <template slot="footer">
      <a-button @click="handleCancel"> 取消 </a-button>
      <a-button type="primary" @click="handleOk"> 确认 </a-button>
    </template>
    <div class="cropper">
      <div>
        <vueCropper
          ref="cropper"
          :img="option.img"
          :outputSize="option.size"
          :outputType="option.outputType"
          :info="true"
          :full="option.full"
          :canMove="option.canMove"
          :canMoveBox="option.canMoveBox"
          :original="option.original"
          :autoCrop="option.autoCrop"
          :fixed="option.fixed"
          :fixedNumber="option.fixedNumber"
          :centerBox="option.centerBox"
          :infoTrue="option.infoTrue"
          :fixedBox="option.fixedBox"
          :autoCropWidth="option.autoCropWidth"
          :autoCropHeight="option.autoCropHeight"
          @realTime="realTime"
        ></vueCropper>
      </div>
      <div v-html="previews"></div>//预览
    </div>
  </a-modal>
</template>
<script>
import BMF from 'browser-md5-file'
const bmf = new BMF()

export default {
  props: {
    fixedNumber: {//自定义截图框比例
      type: Array,
      default:()=>[1,1]
    },
  },
  data() {
    return {
      visible: false,
      option: {
        img: '', // 裁剪图片的地址
        info: true, // 裁剪框的大小信息
        outputSize: 1, // 裁剪生成图片的质量
        outputType: 'jpeg', // 裁剪生成图片的格式
        canScale: false, // 图片是否允许滚轮缩放
        canMove: true,
        autoCrop: true, // 是否默认生成截图框
        autoCropWidth: this.fixedNumber[0] * 100, // 默认生成截图框宽度
        autoCropHeight: this.fixedNumber[1] * 100, // 默认生成截图框高度
        fixedBox: false, // 固定截图框大小 不允许改变
        fixed: true, // 是否开启截图框宽高固定比例
        fixedNumber: this.fixedNumber, // 截图框的宽高比例
        full: true, // 是否输出原图比例的截图
        canMoveBox: true, // 截图框能否拖动
        original: false, // 上传图片按照原始比例渲染
        centerBox: true, // 截图框是否被限制在图片里面
        infoTrue: true, // true 为展示真实输出图片宽高 false 展示看到的截图框宽高
      },
      fileName: null,
      previews: null,
    }
  },
  methods: {
    realTime(data) {
      this.previews = data.html
    },
    openModal(file) {
      //file: 父组件传入的文件对象
      this.visible = true
      let _this = this
      let temp = window.URL.createObjectURL(new Blob([file]))
      // get filename suffix
      const _arr = file.name.split('.')
      const suffix = _arr[_arr.length - 1]
      this.option.img = temp
      bmf.md5(file, (err, md5) => {
        _this.fileName = `${md5}.${suffix}`
      })
    },
    handleOk() {
      let _this = this
      this.$refs.cropper.getCropBlob(data => {//内置方法,获取截图的 blob 数据
        _this.$emit("updateImg",{ fileName: _this.fileName, blob: data })//点击确认后将文件名和图片的blob数据传入父组件中
      })
      this.visible = false
    },
    handleCancel() {
      this.visible = false
    },
  },
}
</script>
<style scoped lang='less'>
.cropper {
  display: flex;
  justify-content: space-around;
  div {
    width: 400px;
    height: 400px;
  }
  & > :last-child {
    margin: 0 auto;
    background-color: #666666;
    position: relative;
    ::v-deep .show-preview { /*这里style是scope模式 要用v-deep样式穿透样式才会生效*/
      position: absolute;
      top: 50%;
      left: 50%;
      transform: translate(-50%, -50%);
    }
  }
}
</style>

使用

vue
<template>
  <div class="upload-wrapper">
    <div class="upload-container">
      <div class="preview-img">
        <a-spin tip="正在上传..." :spinning="loading">
          <img :src="logoUrl" v-if="logoUrl" />
        </a-spin>
      </div>
      <div class="mask">
        <div @click="showModal">
          <a-icon class="pre-icon" type="eye"></a-icon>
          <span class="pre-text">预览</span>
          <a-modal v-model="visible" title="预览" :footer="null" :centered="true" @ok="showModal">
            <img :src="logoUrl" v-if="logoUrl" width="100%" />
          </a-modal>
        </div>
        <div @click="$refs.inputFile.click()">
          <a-icon class="icon" type="plus" />
          <span class="text">点击上传</span>
        </div>
      </div>
    </div>
    <div class="name">
      <span>园区LOGO - 小</span>
      <span style="font-size: 12px">推荐尺寸:32 x 32</span>
    </div>

    <!-- upload input -->
    <input ref="inputFile" type="file" @change="cropperUpload" style="display: none" />
    <cropper-modal ref="cropperModal" @updateImg="updateImg" :fixedNumber="[1, 1]" />
  </div>
</template>

<script>
import { getOssSignature } from '@/api/upload'
import { editClientInfo } from '@/api/user'
import cropperModal from '@/components/cropper'

export default {
  name: 'UploadLogoMini',

  components: {
    cropperModal,
  },

  data() {
    return {
      logoUrl: '',
      loading: false,
      visible: false,
    }
  },

  computed: {
    userInfo() {
      return this.$store.getters.userInfo
    },
  },

  methods: {
    startUpload(file, fileName) {//开始上传
      this.loading = true
      const _this = this
      getOssSignature().then(res => {
        const obj = res.result
        const formData = new FormData()
        formData.append('OSSAccessKeyId', obj.accessId)
        formData.append('policy', obj.policy)
        formData.append('signature', obj.signature)
        formData.append('key', `${obj.dir}${fileName}`)
        formData.append('success_action_status', '200') // 指定返回的状态码
        formData.append('file', file)
        const url = `${obj.maintainHost}/${obj.dir}${random}${fileName}`
        axios
          .post(obj.host, formData, {
            headers: { 'Content-Type': 'multipart/form-data' },
          })
          .then(res => {
            editClientInfo({ clientId: _this.$ls.get('clientId'), appMiNiImg: url })
            _this.logoUrl = url
            _this.$store.commit('UPDATE_APP_MINI_IMG', url)
          })
          .finally(() => {
            _this.loading = false
          })
      })
    },
    cropperUpload(e) {
      const file = e.target.files[0]
      const isJpgOrPng = file.type === 'image/jpeg' || file.type === 'image/png' || file.type === 'image/jpg'
      if (!isJpgOrPng) {
        this.$message.error('图片格式不正确,请上传格式为 .jpg/.jpeg/.png 的图片!')
      }
      const isLt5M = file.size / 1024 / 1024 < 5
      if (!isLt5M) {
        this.$message.error('图片大小超过5MB!')
      }
      let temp = window.URL.createObjectURL(new Blob([file]))
      this.src = temp
      this.$refs.cropperModal.openModal(file)//将文件对象传入cropper组件中
      this.$refs.inputFile.value = null;
    },
    showModal() {
      this.visible = !this.visible
    },
    updateImg(fileObj) {
      //接收子组件传入的参数,执行上传
      this.startUpload(fileObj.blob, fileObj.fileName)
    },
  }
}
</script>

props

名称功能默认值可选值
img裁剪图片的地址url 地址, base64, blob
outputSize裁剪生成图片的质量10.1 ~ 1
outputType裁剪生成图片的格式jpg (jpg 需要传入jpeg)jpeg, png, webp
info裁剪框的大小信息truetrue, false
canScale图片是否允许滚轮缩放truetrue, false
autoCrop是否默认生成截图框falsetrue, false
autoCropWidth默认生成截图框宽度容器的 80%0 ~ max
autoCropHeight默认生成截图框高度容器的 80%0 ~ max
fixed是否开启截图框宽高固定比例falsetrue, false
fixedNumber截图框的宽高比例[1, 1][ 宽度 , 高度 ]
full是否输出原图比例的截图falsetrue, false
fixedBox固定截图框大小不允许改变false
canMove上传图片是否可以移动truetrue, false
canMoveBox截图框能否拖动truetrue, false
original上传图片按照原始比例渲染falsetrue, false
centerBox截图框是否被限制在图片里面falsetrue, false
high是否按照设备的dpr 输出等比例图片truetrue, false
infoTruetrue 为展示真实输出图片宽高 false 展示看到的截图框宽高falsetrue, false
maxImgSize限制图片最大宽度和高度20000 ~ max
enlarge图片根据截图框输出比例倍数10 ~ max
mode图片默认渲染方式containcontain , cover, 100px, 100% auto

内置方法 和 属性

通过 this.$refs.cropper 调用

属性

属性说明
this.$refs.cropper.cropW截图框宽度
this.$refs.cropper.cropH截图框高度

方法

方法说明
this.$refs.cropper.startCrop()开始截图
this.$refs.cropper.stopCrop()停止截图
this.$refs.cropper.clearCrop()清除截图
this.$refs.cropper.changeScale()修改图片大小 正数为变大 负数变小
this.$refs.cropper.getImgAxis()获取图片基于容器的坐标点
this.$refs.cropper.getCropAxis()获取截图框基于容器的坐标点
this.$refs.cropper.goAutoCrop自动生成截图框函数
this.$refs.cropper.rotateRight()向右边旋转90度
this.$refs.cropper.rotateLeft()向左边旋转90度

可用回调方法

  • @realTime 实时预览事件
  • @imgMoving 图片移动回调函数
  • @cropMoving 截图框移动回调函数
  • @imgLoad 图片加载的回调, 返回结果 success, error

获取截图内容

获取截图的 base64 数据

this.$refs.cropper.getCropData(data => {
  // do something
  console.log(data)  
})

获取截图的 blob 数据

this.$refs.cropper.getCropBlob(data => {
  // do something
  console.log(data)  
})