vue3中使用tui-image-editor对图片进行处理,效果如图
1、第一步安装
npm i tui-image-editor
yarn add tui-image-editor
2、第二部封装组件
在src》components 下新建SignImage文件夹,文件夹中新建index.vue,index.vue代码如下
<template>
<el-dialog
:title="title"
:modelValue="dialogVisible"
:show-close="true"
:close-on-click-modal="false"
:close-on-press-escape="false"
:before-close="closeDialog"
destroy-on-close
width="60%"
>
<div class="drawing-container">
<!-- 绘图组件容器DOM -->
<div id="tui-image-editor"></div>
<el-button class="save" type="primary" @click="save">保存</el-button>
</div>
</el-dialog>
</template>
<script setup>
import 'tui-image-editor/dist/tui-image-editor.css'
import 'tui-color-picker/dist/tui-color-picker.css'
import ImageEditor from 'tui-image-editor'
import { importNewImg } from "@/api/back/home.js";
import { uploadAvatar } from "@/api/system/user";
// 中文菜单
const locale_zh = {
ZoomIn: '放大',
ZoomOut: '缩小',
Hand: '手掌',
History: '历史',
Resize: '调整宽高',
Crop: '裁剪',
DeleteAll: '全部删除',
Delete: '删除',
Undo: '撤销',
Redo: '反撤销',
Reset: '重置',
Flip: '镜像',
Rotate: '旋转',
Draw: '画',
Shape: '形状标注',
Icon: '图标标注',
Text: '文字标注',
Mask: '遮罩',
Filter: '滤镜',
Bold: '加粗',
Italic: '斜体',
Underline: '下划线',
Left: '左对齐',
Center: '居中',
Right: '右对齐',
Color: '颜色',
'Text size': '字体大小',
Custom: '自定义',
Square: '正方形',
Apply: '应用',
Cancel: '取消',
'Flip X': 'X 轴',
'Flip Y': 'Y 轴',
Range: '区间',
Stroke: '描边',
Fill: '填充',
Circle: '圆',
Triangle: '三角',
Rectangle: '矩形',
Free: '曲线',
Straight: '直线',
Arrow: '箭头',
'Arrow-2': '箭头2',
'Arrow-3': '箭头3',
'Star-1': '星星1',
'Star-2': '星星2',
Polygon: '多边形',
Location: '定位',
Heart: '心形',
Bubble: '气泡',
'Custom icon': '自定义图标',
'Load Mask Image': '加载蒙层图片',
Grayscale: '灰度',
Blur: '模糊',
Sharpen: '锐化',
Emboss: '浮雕',
'Remove White': '除去白色',
Distance: '距离',
Brightness: '亮度',
Noise: '噪音',
'Color Filter': '彩色滤镜',
Sepia: '棕色',
Sepia2: '棕色2',
Invert: '负片',
Pixelate: '像素化',
Threshold: '阈值',
Tint: '色调',
Multiply: '正片叠底',
Blend: '混合色',
Width: '宽度',
Height: '高度',
'Lock Aspect Ratio': '锁定宽高比例'
}
// 画布组件自定义样式
const customTheme = {
"common.bi.image": "", // 左上角logo图片
"common.bisize.width": "0px",
"common.bisize.height": "0px",
"common.backgroundImage": "none",
"common.backgroundColor": "#f3f4f6",
"common.border": "1px solid #333",
// header
"header.backgroundImage": "none",
"header.backgroundColor": "#f3f4f6",
"header.border": "0px",
// load button
"loadButton.backgroundColor": "#fff",
"loadButton.border": "1px solid #ddd",
"loadButton.color": "#222",
"loadButton.fontFamily": "NotoSans, sans-serif",
"loadButton.fontSize": "12px",
"loadButton.display": "none", // 可以直接隐藏掉
// download button
"downloadButton.backgroundColor": "#fdba3b",
"downloadButton.border": "1px solid #fdba3b",
"downloadButton.color": "#fff",
"downloadButton.fontFamily": "NotoSans, sans-serif",
"downloadButton.fontSize": "12px",
"downloadButton.display": "none", // 可以直接隐藏掉
// icons default
"menu.normalIcon.color": "#8a8a8a",
"menu.activeIcon.color": "#555555",
"menu.disabledIcon.color": "#ccc",
"menu.hoverIcon.color": "#e9e9e9",
"submenu.normalIcon.color": "#8a8a8a",
"submenu.activeIcon.color": "#e9e9e9",
"menu.iconSize.width": "24px",
"menu.iconSize.height": "24px",
"submenu.iconSize.width": "32px",
"submenu.iconSize.height": "32px",
// submenu primary color
"submenu.backgroundColor": "#1e1e1e",
"submenu.partition.color": "#858585",
// submenu labels
"submenu.normalLabel.color": "#858585",
"submenu.normalLabel.fontWeight": "lighter",
"submenu.activeLabel.color": "#fff",
"submenu.activeLabel.fontWeight": "lighter",
// checkbox style
"checkbox.border": "1px solid #ccc",
"checkbox.backgroundColor": "#fff",
// rango style
"range.pointer.color": "#fff",
"range.bar.color": "#666",
"range.subbar.color": "#d1d1d1",
"range.disabledPointer.color": "#414141",
"range.disabledBar.color": "#282828",
"range.disabledSubbar.color": "#414141",
"range.value.color": "#fff",
"range.value.fontWeight": "lighter",
"range.value.fontSize": "11px",
"range.value.border": "1px solid #353535",
"range.value.backgroundColor": "#151515",
"range.title.color": "#fff",
"range.title.fontWeight": "lighter",
// colorpicker style
"colorpicker.button.border": "1px solid #1e1e1e",
"colorpicker.title.color": "#fff",
};
// props
const props = defineProps({
dialogVisible: {
type: Boolean,
default: () => {
return false;
},
},
title: {
type: String,
default: "",
},
imgUrl: {
type: String,
default: "",
},
});
const emit = defineEmits()
const instance = ref(null)
onMounted(() => {
nextTick(() => {
init() // 页面创建完成后调用
})
})
// 关闭弹框
const closeDialog = ()=> {
emit('closeCropperDialog')
// options.img = ''
}
const init = ()=> {
instance.value = new ImageEditor(document.querySelector('#tui-image-editor'), {
includeUI: {
loadImage: {
path: props.imgUrl,
name: 'image'
},
menu: ['resize', 'crop', 'rotate', 'draw', 'shape', 'icon', 'text', 'filter'], // 底部菜单按钮列表 隐藏镜像flip和遮罩mask
initMenu: 'draw', // 默认打开的菜单项
menuBarPosition: 'bottom', // 菜单所在的位置
locale: locale_zh, // 本地化语言为中文
theme: customTheme // 自定义样式
},
cssMaxWidth: 400, // canvas 最大宽度
cssMaxHeight: 500 // canvas 最大高度
})
document.getElementsByClassName('tui-image-editor-main')[0].style.top = '45px' // 调整图片显示位置
document.getElementsByClassName(
'tie-btn-reset tui-image-editor-item help'
)[0].style.display = 'none' // 隐藏顶部重置按钮
}
// 保存图片,并上传
const save = ()=> {
const base64String = instance.value.toDataURL() // base64 文件
const data = window.atob(base64String.split(',')[1])
const ia = new Uint8Array(data.length)
for (let i = 0; i < data.length; i++) {
ia[i] = data.charCodeAt(i)
}
const blob = new Blob([ia], { type: 'image/png' }) // blob 文件
const form = new FormData()
form.append("avatarfile", blob);
uploadAvatar(form).then(res=>{
emit('getNewImg',res.imgUrl);
closeDialog()
})
}
</script>
<style lang="scss" scoped>
.drawing-container {
width: 100%;
height: 80vh;
position: relative;
.save {
position: absolute;
right: 50px;
top: 15px;
}
}
</style>
3、第三步使用组件
<template>
<el-dialog
:title="dialogTitle"
:modelValue="dialogVisible"
width="55%"
top="4vh"
:before-close="closeDialog"
:close-on-click-modal="false"
destroy-on-close
>
<div class="newBox">
<!-- 处理完图片回显 -->
<el-image :src="newImg" fit="cover"/><br/>
<!-- 点击弹出图片处理框 -->
<el-button type="primary" plain icon="Plus" @click="cropperObj.openCropperView">点击弹出图片处理框</el-button>
<!-- 图片处理框 -->
<SignImage
v-if="cropperObj.cVisible"
:dialogVisible.sync="cropperObj.cVisible"
:title="cropperObj.ctitle"
:imgUrl="cropperObj.previewsImgUrl"
@getNewImg="cropperObj.getNewImg"
@closeCropperDialog="cropperObj.closeCropperView"
></SignImage>
</div>
<template #footer>
<div class="dialog-footer">
<!-- <el-button class="btn-sure" type="primary" @click="closeDialog">确 定</el-button> -->
<el-button @click="closeDialog">取 消</el-button>
</div>
</template>
</el-dialog>
</template>
<script setup>
import SignImage from '@/components/SignImage/index.vue'
import { ref, reactive, watch } from "vue";
const props = defineProps({
dialogVisible: {
type: Boolean,
default: false,
},
dialogTitle: {
type: String,
default: "",
},
});
// 关闭弹窗
const emits = defineEmits(["closeDialog"]);
const closeDialog = () => {
emits("closeDialog");
};
// 图片处理
const newImg = ref("")
const cropperObj = reactive({
cVisible:false, // 显示切图弹框
ctitle:"", // 弹框标题
previewsImgUrl:"http://xxx.xxx.xx.xxx:7788/profile/upload/2023/09/08/chris_20230908164948A038.jpg", //图片地址
// 开启剪切弹框
openCropperView: () => {
cropperObj.ctitle="图片处理"
cropperObj.cVisible = true
},
// 关闭弹框所触发的事件
closeCropperView: (data)=> {
cropperObj.cVisible = false
},
// 获取处理完的图片
getNewImg:(val) =>{
newImg.value = window.serverUrl.IMG_SERVER + val
}
})
// watch(
// () => props.rowObj,
// (newVal) => {
// if (newVal.id) {
// formData.id = newVal.id ? newVal.id : "";
// }
// },
// { immediate: true }
// );
</script>
<style lang="scss" scoped>
</style>