Files
aida_front/src/component/Detail/canvas/index.vue

607 lines
17 KiB
Vue
Raw Normal View History

2025-02-17 15:24:01 +08:00
<template>
<div class="canvasBox">
<!-- designDetailShow -->
<!-- :class="[driver__.driver?'hideEvents':'']" -->
2025-06-23 09:27:29 +08:00
<div class="canvasContent" ref="canvasContent">
<div class="content-bottom" ref="canvasContent">
2025-02-25 15:32:55 +08:00
<div class="contet">
<div class="canvas" :class="{'active': currentView === 'canvasEditor'}"@click.stop>
2026-01-19 11:07:04 +08:00
<!-- :clothingMinIOPath="selectDetail.minIOPath" 部件选取 -->
2025-07-19 14:04:48 +08:00
<editCanvas v-if="canvasLoad" :config="canvasConfig"
2026-01-15 17:14:11 +08:00
@canvasInit="editSketchCanvasInit"
is-edit
2026-01-02 11:24:11 +08:00
:clothingImageUrl="selectDetail.path"
2026-01-15 17:14:11 +08:00
:clothingImageUrl2="selectDetail.layersObject[0].maskUrl"
2025-07-19 14:04:48 +08:00
showFixedLayer
2025-08-22 10:27:48 +08:00
:canvasJSON="canvasJSON"
@canvasLoadJsonSuccess="canvasLoadJsonSuccess"
2025-06-23 09:27:29 +08:00
:clothing-image-opts="{
imageMode:'contains',
}"
2026-01-21 10:09:29 +08:00
:hideCanvas="hideCanvas || !isEditPattern"
2025-06-30 10:53:25 +08:00
ref="editCanvas">
</editCanvas>
2025-06-23 09:27:29 +08:00
<!-- <canvasContent ref="canvasContent"></canvasContent> -->
2025-02-25 15:32:55 +08:00
</div>
2025-07-19 14:04:48 +08:00
<div class="editFrontBack" v-if="currentView === 'redGreenExample'" @click.stop>
<editCanvas v-if="canvasLoad" :config="canvasConfig"
2026-01-15 17:14:11 +08:00
@canvasInit="editFrontBackCanvasInit"
2025-06-23 09:27:29 +08:00
:enabledRedGreenMode="true"
:clothingImageUrl="selectDetail.path"
2025-06-23 09:27:29 +08:00
:redGreenImageUrl="frontBack.front[imgDomIndex].maskUrl"
@trigger-red-green-mouseup="frontBackChange"
is-edit
2025-06-23 09:27:29 +08:00
:clothing-image-opts="{
imageMode:'contains',
}"
2026-01-21 10:09:29 +08:00
:hideCanvas="hideCanvas || !isEditPattern"
2025-06-30 10:53:25 +08:00
ref="editCanvasBackFront">
</editCanvas>
2025-02-25 15:32:55 +08:00
</div>
<div class="editSketch" v-if="currentView === 'editSketch'" @click.stop>
<generalMiniCanvas ref="generalMiniCanvas" :btnShow="false" :imgUrl="selectDetail.sketchString || selectDetail.path"></generalMiniCanvas>
</div>
2025-02-25 15:32:55 +08:00
</div>
<!-- <div class="Finish">
<div class="gallery_btn" @click="privewDetail">Finish</div>
</div> -->
2025-02-17 15:24:01 +08:00
</div>
</div>
2025-08-22 10:27:48 +08:00
<div class="mark_loading" v-show="isShowMark">
2025-02-17 15:24:01 +08:00
<a-spin size="large" />
2025-08-22 10:27:48 +08:00
</div>
2025-02-17 15:24:01 +08:00
</div>
</template>
<script lang="ts">
2025-02-25 15:32:55 +08:00
import { defineComponent,computed,onBeforeUnmount,provide,nextTick,watch,toRefs, reactive} from 'vue'
2025-02-17 15:24:01 +08:00
import { Https } from "@/tool/https";
import { Modal,message } from 'ant-design-vue';
import {getUploadUrl,isMoible,setGradual} from '@/tool/util'
import { useStore } from "vuex";
import { useI18n } from 'vue-i18n'
2025-06-23 09:27:29 +08:00
import editCanvas from "@/component/Canvas/CanvasEditor/index.vue";
2025-08-22 10:27:48 +08:00
import { formatTime,segmentImage,getMinioUrl } from "@/tool/util";
2025-11-14 11:19:18 +08:00
import { useRouter, useRoute } from 'vue-router'
import generalMiniCanvas from "../../modules/generalMiniCanvas.vue";
2025-02-17 15:24:01 +08:00
export default defineComponent({
components:{
editCanvas,generalMiniCanvas
2025-02-17 15:24:01 +08:00
},
2025-08-22 10:27:48 +08:00
props:{
2025-09-15 13:33:42 +08:00
isEditPattern:{
type:String,
default:''
},
updateOtherLayers:{
type:Function,
default:()=>{}
},
sketchSize:{
type:Object,
default:()=>{}
},
2025-08-22 10:27:48 +08:00
},
emits:['update:loadingShow'],
2025-02-17 15:24:01 +08:00
setup(props,{emit}) {
const store = useStore();
2025-02-25 15:32:55 +08:00
const {t} = useI18n();
2025-11-14 11:19:18 +08:00
const route = useRoute()
2025-02-17 15:24:01 +08:00
const detailDom = reactive({
2025-02-25 15:32:55 +08:00
editFrontBack:null as any,
model:null,
2025-06-23 09:27:29 +08:00
editCanvas:null as any,
2025-06-26 15:41:08 +08:00
editCanvasBackFront:null as any,
2025-06-23 09:27:29 +08:00
canvasContent:null as any,
generalMiniCanvas:null as any,
2025-02-17 15:24:01 +08:00
})
const userDetail = computed(()=>{
return store.state.UserHabit.userDetail
})
const detailData = reactive({
2025-02-25 15:32:55 +08:00
isShowMark:false,
liquefactionData:null as any,
liquefaction:null as any,
canvasType:'export',
imgDomIndex:-1,
2025-08-22 10:27:48 +08:00
selectDetail:computed(()=>store.state.DesignDetail.selectDetail),
designDetail:computed(()=>store.state.DesignDetail.designDetail),
frontBack:computed(()=>store.state.DesignDetail.frontBack),
2025-06-23 09:27:29 +08:00
canvasLoad:false,
canvasConfig:{
} as any,
2025-09-15 13:33:42 +08:00
currentView:props.isEditPattern,
2025-07-19 14:04:48 +08:00
getCanvasIfEdit:inject('getCanvasIfEdit')as any,
canvasInstance:null as any,
2025-08-22 10:27:48 +08:00
canvasJSON:'',
2026-01-21 09:43:26 +08:00
hideCanvas: computed(()=>(store.state.Workspace.projectPath !== route.fullPath)),
2026-01-02 11:24:11 +08:00
otherData:computed(()=>({
canvasId: store.state.DesignDetail.selectDetail.canvasId,
color: store.state.DesignDetail.selectDetail.color,
printObject: store.state.DesignDetail.selectDetail.printObject,
trims: store.state.DesignDetail.selectDetail.trims,
})),
changeSketchUpdateFrontBack:null as any,//切换sketch后是否需要合成图片进行风格
2025-02-17 15:24:01 +08:00
})
2025-02-25 15:32:55 +08:00
watch(()=>detailData.selectDetail,(newValue,oldValue)=>{
detailData.imgDomIndex = detailData.frontBack.front.findIndex((item:any)=>item.id == newValue.id)
2025-04-09 14:09:19 +08:00
// privewDetail(oldValue)
2025-02-25 15:32:55 +08:00
},{immediate: true})
provide('isShowMark',detailData.isShowMark)
provide('canvasType',detailData.canvasType)
2025-09-12 14:58:07 +08:00
2025-07-19 14:04:48 +08:00
const editFront = (str:any)=>{//编辑前后片
2025-06-26 15:41:08 +08:00
let canvasJSON = '' as any
2025-07-19 14:04:48 +08:00
if(detailData.currentView === 'canvasEditor'){
2025-06-26 15:41:08 +08:00
sessionStorage.setItem('sketchEdit',detailDom.editCanvas.getJSON())
canvasJSON = sessionStorage.getItem('frontBackEdit');
}else if(detailData.currentView === 'redGreenExample'){
2025-06-26 15:41:08 +08:00
sessionStorage.setItem('frontBackEdit',detailDom.editCanvasBackFront.getJSON())
canvasJSON = sessionStorage.getItem('sketchEdit');
}
// detailData.canvasLoad = false
2025-07-19 14:04:48 +08:00
detailData.currentView = str
2025-06-26 15:41:08 +08:00
if(canvasJSON){
// detailData.canvasLoad = true
2025-02-25 15:32:55 +08:00
nextTick(()=>{
2025-07-19 14:04:48 +08:00
if(detailData.currentView === 'redGreenExample'){
2025-06-26 15:41:08 +08:00
detailDom.editCanvas.loadJSON(canvasJSON)
}else{
detailDom.editCanvasBackFront.loadJSON(canvasJSON)
}
2025-02-25 15:32:55 +08:00
})
}else{
2025-07-19 14:04:48 +08:00
if(detailData.currentView === 'redGreenExample'){
2025-06-26 15:41:08 +08:00
nextTick(()=>{
2026-01-15 17:14:11 +08:00
setCanvas(detailData.selectDetail.path).then(()=>{
// detailData.canvasLoad = true
2025-06-26 15:41:08 +08:00
})
})
}else{
nextTick(()=>{
setCanvas(detailData.frontBack.front[detailData.imgDomIndex].maskUrl).then(()=>{
// detailData.canvasLoad = true
2025-06-26 15:41:08 +08:00
})
})
}
2025-02-25 15:32:55 +08:00
}
}
const updateOtherLayers = (obj:any)=>{
2025-07-19 14:04:48 +08:00
if(!detailDom.editCanvas)return
return new Promise(async (res,reject)=>{
await detailDom?.editCanvas.updateOtherLayers(obj)
res('')
})
}
const privewDetail = async (oldSelectDetail = detailData.selectDetail)=>{
// if(!detailDom.editCanvas)return
2026-01-15 17:14:11 +08:00
return new Promise(async (res,reject)=>{
console.log(detailDom.editCanvas)
await detailDom.editCanvas.exportImage({
isContainFixed:true,
width:props.sketchSize.width,
height:props.sketchSize.height,
}).then((rv)=>{
2025-08-22 10:27:48 +08:00
if(oldSelectDetail?.partialDesign)oldSelectDetail.partialDesign.partialDesignBase64 = rv
2025-06-26 15:41:08 +08:00
})
await setUndivideLayer()
2026-01-15 17:14:11 +08:00
res('')
2025-06-26 15:41:08 +08:00
})
}
const getCanvasElement = ()=>{
if(!detailDom?.editCanvas)return ''
return detailDom?.editCanvas.exportExtraInfo();
2025-02-25 15:32:55 +08:00
}
const setFrontBackColor = (data:any)=>{
detailDom.editFrontBack.setBackground(data)
}
2025-06-23 09:27:29 +08:00
const setCanvas = (url:any)=>{
return new Promise((res,rev)=>{
let img = new Image()
img.onload = ()=>{
let wH = [1,1]
let domHeight = detailDom.canvasContent?.offsetHeight - 200
2025-06-23 09:27:29 +08:00
let imgHeight = img.height
wH = [1,domHeight/imgHeight]
// detailData.canvasConfig.width = img.width * wH[1]
// detailData.canvasConfig.height = domHeight
detailData.canvasConfig.width = img.width
detailData.canvasConfig.height = img.height
2026-01-21 10:06:11 +08:00
detailData.canvasConfig.initZoom = true
2025-06-23 09:27:29 +08:00
res('')
}
img.src = url
})
}
const resizeImageWithNativeCanvas = async (image1Url, imageBUrl)=>{
try {
// 加载第一张图片获取尺寸
const img1 = await loadImage(image1Url);
const targetWidth = img1.naturalWidth;
const targetHeight = img1.naturalHeight;
// 加载第二张图片
const imgB = await loadImage(imageBUrl);
// 创建canvas元素
const canvas = document.createElement('canvas');
canvas.width = targetWidth;
canvas.height = targetHeight;
const ctx = canvas.getContext('2d');
// 绘制调整尺寸后的图片
ctx.drawImage(imgB, 0, 0, targetWidth, targetHeight);
// 导出base64
const base64 = canvas.toDataURL('image/png', 1);
return base64;
} catch (error) {
console.error('处理图片时出错:', error);
throw error;
}
}
// 图片加载辅助函数
const loadImage = (url)=>{
return new Promise((resolve, reject) => {
const img = new Image();
img.crossOrigin = 'anonymous';
img.onload = () => resolve(img);
img.onerror = reject;
img.src = url;
});
}
2025-06-23 09:27:29 +08:00
const frontBackChange = (value:any)=>{
2026-01-15 17:14:11 +08:00
let full = detailData.selectDetail.partialDesign.partialDesignBase64 || detailData.selectDetail.path
2025-06-23 09:27:29 +08:00
let size = {
...detailData.canvasConfig,
}
store.commit('DesignDetail/updataDetailItem',{maskUrl:value})
segmentImage(value,full,size).then(async (rv)=>{
2025-07-19 14:04:48 +08:00
let front = detailData.frontBack.front[detailData.imgDomIndex]
let back = detailData.frontBack.back[detailData.imgDomIndex]
if(!front?.oldImageUrl)front.oldImageUrl = front.imageUrl
2025-09-15 13:33:42 +08:00
if(!front?.oldMaskUrl)front.oldMaskUrl = front.maskUrl
2025-07-19 14:04:48 +08:00
if(!back?.oldImageUrl)back.oldImageUrl = back.imageUrl
2025-08-22 10:27:48 +08:00
if(!front?.oldMaskUrl)store.commit('DesignDetail/updataDetailItem',{maskUrl:front.oldMaskUrl})
2025-07-19 14:04:48 +08:00
front.imageUrl = rv.targetFrontUrl
let base64 = await resizeImageWithNativeCanvas(front.oldMaskUrl,value)
front.maskUrl = base64
2025-07-19 14:04:48 +08:00
back.imageUrl = rv.targetBackUrl
// store.commit('DesignDetail/updataDetailItem',{maskUrl:value})
2025-06-23 09:27:29 +08:00
})
}
const editSketchCanvasInit = async (value:any)=>{
2026-01-15 17:14:11 +08:00
detailData.canvasInstance = value
detailData.getCanvasIfEdit.fun = getCanvasLength
detailData.isShowMark = false
console.log('初始化完成')
}
const editFrontBackCanvasInit = (value:any)=>{
2025-07-19 14:04:48 +08:00
detailData.canvasInstance = value
detailData.getCanvasIfEdit.fun = getCanvasLength
detailData.isShowMark = false
console.log('初始化完成')
2025-07-19 14:04:48 +08:00
}
const getCanvasLength = ()=>{
return detailData.canvasInstance?.commandManager?.undoStack?.length
2025-06-30 10:53:25 +08:00
}
2026-01-15 17:14:11 +08:00
const getCanvasJSON = ()=>{
if(!detailDom?.editCanvas)return ''
return detailDom?.editCanvas?.getJSON()
}
const saveCanvas = async (canvasData:any)=>{
const index = detailData.designDetail.clothes.findIndex(item => item.id === canvasData.id);
2025-08-22 10:27:48 +08:00
await new Promise<void>((resolve, reject) => {
2026-01-15 17:14:11 +08:00
let canvasJSON = JSON.parse(canvasData.canvasJSON)
if(!canvasJSON)return resolve()
// canvasData.canvas.objects.forEach((objectsItem:any) => {
// if(objectsItem.type == 'image')objectsItem.minioUrl = getMinioUrl(objectsItem.src)
// });
let blob = new Blob([JSON.stringify(canvasJSON)], { type: "application/json" });
2025-08-22 10:27:48 +08:00
let formData = new FormData();
formData.append("file", blob, "data.json");
formData.append("designItemDetailId", detailData.selectDetail.id);
formData.append("module", "designItemDetail");
let config = {
headers: { "Content-Type": "multipart/form-data", Accept: "*/*" },
};
Https.axiosPost(Https.httpUrls.exportSave, formData, config).then(
(rv) => {
if(index>-1)detailData.designDetail.clothes[index].canvasId = rv
// detailData.selectDetail.canvasId = rv
detailData.isShowMark = false
resolve()
}
).catch(()=>{
detailData.isShowMark = false
resolve()
});
})
}
// let time = null as any
// const changeCanvas = ()=>{
// clearTimeout(time)
// time = setTimeout(()=>{
// saveCanvas('auto')
// },3000)
// }
const canvasLoadJsonSuccess = async ()=>{
let otherData = await props.updateOtherLayers()
await updateOtherLayers(otherData)
if(detailData.changeSketchUpdateFrontBack){
await detailData.changeSketchUpdateFrontBack()
detailData.changeSketchUpdateFrontBack = null
}
setUndivideLayer()
emit('update:loadingShow',false)
}
const setUndivideLayer = async ()=>{
await new Promise<void>(async (resolve, reject) => {
// if(!detailData.selectDetail.undividedLayerColor){
await detailDom.editCanvas.exportImage({
isContainFixed:true,
isPrintTrimsRepeat:false,
isPrintTrimsNoRepeat:false,
isContainNormalLayer:false,
width:props.sketchSize.width,
height:props.sketchSize.height}).then((rv)=>{
detailData.selectDetail.undividedLayerColor = rv
})
// }
// if(!detailData.selectDetail.undividedLayer){
await detailDom.editCanvas.exportImage({
isContainFixed:true,
isPrintTrimsNoRepeat:false,
isPrintTrimsRepeat:true,
isContainNormalLayer:false,
width:props.sketchSize.width,
height:props.sketchSize.height,
}).then((rv)=>{
detailData.selectDetail.undividedLayer = rv
})
// }
resolve()
})
2025-08-22 10:27:48 +08:00
}
const submitBase64Data = ()=>{
return detailDom.generalMiniCanvas.submitBase64Data()
}
2025-02-17 15:24:01 +08:00
onBeforeUnmount(()=>{
2025-07-19 14:04:48 +08:00
let front = detailData.frontBack.front[detailData.imgDomIndex]
let back = detailData.frontBack.back[detailData.imgDomIndex]
2025-09-15 13:33:42 +08:00
2025-07-19 14:04:48 +08:00
if(front?.oldImageUrl)front.imageUrl = front.oldImageUrl
if(front?.oldMaskUrl)front.maskUrl = front.oldMaskUrl
if(back?.oldImageUrl)back.imageUrl = back.oldImageUrl
if(front?.oldMaskUrl)store.commit('DesignDetail/updataDetailItem',{maskUrl:front.maskUrl})
2025-07-19 14:04:48 +08:00
2025-06-26 15:41:08 +08:00
sessionStorage.removeItem('frontBackEdit');
sessionStorage.removeItem('sketchEdit');
2025-06-23 09:27:29 +08:00
detailData.canvasLoad = false
2026-01-02 11:24:11 +08:00
// privewDetail()
2025-02-17 15:24:01 +08:00
})
2025-06-23 09:27:29 +08:00
onMounted(()=>{
2025-08-22 10:27:48 +08:00
nextTick(async ()=>{
2025-09-15 13:33:42 +08:00
// detailData.currentView = 'canvasEditor'
2026-01-15 17:14:11 +08:00
const sessionCanvasList = sessionStorage.getItem('canvasList');
const canvasList = sessionCanvasList ? JSON.parse(sessionCanvasList) : []
let canvasIndex = canvasList.findIndex(item => item.id === detailData.selectDetail.id);
if(canvasIndex>-1){
2026-01-15 17:14:11 +08:00
detailData.canvasJSON = canvasList[canvasIndex].canvasJSON
}else{
if(detailData.selectDetail.canvasId){
detailData.isShowMark = true
await new Promise((resolve, reject) => {
let value = {
module:'designItemDetail',
id:detailData.selectDetail.canvasId,
}
Https.axiosPost(Https.httpUrls.exportSearch, value)
.then((rv) => {
detailData.canvasJSON = rv
resolve('')
})
.catch((rv) => {
resolve(null)
});
2025-08-22 10:27:48 +08:00
})
}
2025-08-22 10:27:48 +08:00
}
2026-01-15 17:14:11 +08:00
setCanvas(detailData.selectDetail.path).then(()=>{
2025-06-23 09:27:29 +08:00
detailData.canvasLoad = true
})
})
})
2025-02-17 15:24:01 +08:00
return{
...toRefs(detailDom),
...toRefs(detailData),
2025-02-25 15:32:55 +08:00
editFront,
privewDetail,
setFrontBackColor,
2025-06-23 09:27:29 +08:00
frontBackChange,
2026-01-15 17:14:11 +08:00
editSketchCanvasInit,
editFrontBackCanvasInit,
2025-08-22 10:27:48 +08:00
saveCanvas,
getCanvasElement,
updateOtherLayers,
canvasLoadJsonSuccess,
submitBase64Data,
2026-01-15 17:14:11 +08:00
getCanvasJSON,
2025-02-17 15:24:01 +08:00
}
},
provide() {
return {
}
},
mounted(){
},
})
</script>
<style lang="less" scoped>
.canvasBox{
flex: 1;
overflow: hidden;
// top: -100%;
display: flex;
align-items: center;
2025-02-25 15:32:55 +08:00
.tool-box{
width: 4rem;
height: 100%;
display: flex;
flex-direction: column;
justify-content: center;
}
.Finish{
margin-top: auto;
> .gallery_btn{
cursor: pointer;
}
}
.content-bottom{
:deep(.tools-sidebar){
i{
font-size: 2.5rem;
cursor: pointer;
width: 3.5rem;
height: 3.5rem;
display: flex;
align-items: center;
justify-content: center;
flex-shrink: 0;
transition: all .3s;
margin-bottom: .5rem;
&.active{
border: 1px solid;
border-radius: .4rem;
}
&.icon-xiala{
transform: rotate(-90deg);
}
&.icon-xialaActive{
transform: rotate(90deg);
}
&.eventNone{
cursor: no-drop;
border: none;
opacity: .5;
}
2025-02-25 15:32:55 +08:00
}
2025-02-17 15:24:01 +08:00
}
}
.canvasContent{
2025-03-03 14:52:05 +08:00
height: 100%;
// height: 70rem;
2025-02-17 15:24:01 +08:00
width: 100%;
border: 2px solid #000;
border-radius: 3rem;
padding: 4rem;
2025-02-25 15:32:55 +08:00
display: flex;
flex-direction: column;
}
.content-bottom{
flex: 1;
overflow: hidden;
display: flex;
> .contet{
flex: 1;
overflow: hidden;
position: relative;
.canvas{
opacity: 0;
position: absolute;
&.active{
opacity: 1;
position: relative;
}
}
.canvas,.editFrontBack,.editSketch{
2025-02-25 15:32:55 +08:00
width: 100%;
height: 100%;
}
.editFrontBack{
z-index: 2;
position: absolute;
top: 0;
left: 0;
}
}
2025-02-17 15:24:01 +08:00
}
}
2025-06-30 10:53:25 +08:00
.custom-tool-btn {
position: relative;
2025-07-24 20:15:39 +08:00
width: 3.6rem;
height: 3.6rem;
2025-06-30 10:53:25 +08:00
display: flex;
align-items: center;
justify-content: center;
background: none;
border: none;
border-radius: 4px;
cursor: pointer;
2025-07-24 20:15:39 +08:00
font-size: 1.6rem;
2025-06-30 10:53:25 +08:00
color: #333;
transition: all 0.2s ease;
2025-07-19 14:04:48 +08:00
&.active{
background-color: #e6f7ff;
color: #1890ff;
}
2025-06-30 10:53:25 +08:00
}
.custom-tool-btn:hover {
background-color: #f0f0f0;
2025-07-19 14:04:48 +08:00
&.active{
background-color: #e6f7ff;
color: #1890ff;
}
2025-06-30 10:53:25 +08:00
}
.custom-tool-btn:hover .tool-tooltip {
display: block;
}
.tool-tooltip {
display: none;
position: absolute;
left: 100%;
top: 50%;
transform: translateY(-50%);
background-color: rgba(0, 0, 0, 0.7);
color: white;
2025-07-24 20:15:39 +08:00
padding: .4rem .8rem;
border-radius: .4rem;
margin-left: .8rem;
2025-06-30 10:53:25 +08:00
white-space: nowrap;
2025-07-24 20:15:39 +08:00
font-size: 1.2rem;
2025-06-30 10:53:25 +08:00
z-index: 10;
}
.tool-tooltip:before {
content: "";
position: absolute;
top: 50%;
right: 100%;
2025-07-24 20:15:39 +08:00
margin-top: -.5rem;
border-width: .5rem;
2025-06-30 10:53:25 +08:00
border-style: solid;
border-color: transparent rgba(0, 0, 0, 0.7) transparent transparent;
}
2025-02-17 15:24:01 +08:00
</style>