feat: 提交页面
This commit is contained in:
@@ -15,14 +15,19 @@
|
|||||||
<div class="header" :class="{ 'is-product': data.isProduct }">
|
<div class="header" :class="{ 'is-product': data.isProduct }">
|
||||||
<div class="title">{{ data.title }}</div>
|
<div class="title">{{ data.title }}</div>
|
||||||
<div class="right flex">
|
<div class="right flex">
|
||||||
<div v-if="coverOrigin.length" class="origin-container flex align-center">
|
<div v-if="coverOrigin.length > 1" class="origin-container flex align-center">
|
||||||
<span>Crop from: </span>
|
<span>Crop from: </span>
|
||||||
<div class="origin-select flex align-center">
|
<div class="origin-select flex align-center">
|
||||||
<div class="origin-item sketch" @click="handleChangeOrigin('sketch')">
|
<div
|
||||||
|
class="origin-item sketch"
|
||||||
|
:class="{ selected: currentOrigin === 'sketch' }"
|
||||||
|
@click="handleChangeOrigin('sketch')"
|
||||||
|
>
|
||||||
Sketch
|
Sketch
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
class="origin-item product selected"
|
class="origin-item product"
|
||||||
|
:class="{ selected: currentOrigin === 'mainProducImage' }"
|
||||||
@click="handleChangeOrigin('mainProductImage')"
|
@click="handleChangeOrigin('mainProductImage')"
|
||||||
>
|
>
|
||||||
Main product image
|
Main product image
|
||||||
@@ -42,7 +47,7 @@
|
|||||||
ref="imageClipRef"
|
ref="imageClipRef"
|
||||||
v-bind="$attrs"
|
v-bind="$attrs"
|
||||||
:ratio="data.ratio"
|
:ratio="data.ratio"
|
||||||
:fixedBox="data.isProduct ? type !== 'apparel' : false"
|
:isProduct="isProduct"
|
||||||
:url="data.url"
|
:url="data.url"
|
||||||
:type="type"
|
:type="type"
|
||||||
@change="(v) => (data.preview_url = v)"
|
@change="(v) => (data.preview_url = v)"
|
||||||
@@ -112,8 +117,11 @@ const data = reactive({
|
|||||||
callback: null,
|
callback: null,
|
||||||
isProduct: false // 是否商品编辑
|
isProduct: false // 是否商品编辑
|
||||||
})
|
})
|
||||||
|
|
||||||
|
const currentOrigin = ref("sketch")
|
||||||
const coverOrigin = ref([])
|
const coverOrigin = ref([])
|
||||||
const handleChangeOrigin = (type) => {
|
const handleChangeOrigin = (type) => {
|
||||||
|
currentOrigin.value = type
|
||||||
data.url = coverOrigin.value.filter((el) => el.type === type)[0].url
|
data.url = coverOrigin.value.filter((el) => el.type === type)[0].url
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -132,7 +140,10 @@ const open = (url, callback, options, origin) => {
|
|||||||
if (options.hasOwnProperty("isPreview")) data.isPreview = options.isPreview
|
if (options.hasOwnProperty("isPreview")) data.isPreview = options.isPreview
|
||||||
data.isProduct = options.isProduct
|
data.isProduct = options.isProduct
|
||||||
}
|
}
|
||||||
if (origin) coverOrigin.value = origin
|
if (origin?.length) {
|
||||||
|
coverOrigin.value = origin
|
||||||
|
data.url = origin[0].url
|
||||||
|
}
|
||||||
console.log("-------", origin)
|
console.log("-------", origin)
|
||||||
show.value = true
|
show.value = true
|
||||||
}
|
}
|
||||||
@@ -303,7 +314,7 @@ defineExpose({
|
|||||||
}
|
}
|
||||||
&.is-cover {
|
&.is-cover {
|
||||||
> .preview-image > img {
|
> .preview-image > img {
|
||||||
// width: 29.7rem;
|
width: 29.7rem;
|
||||||
height: 37.5rem;
|
height: 37.5rem;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,10 +7,9 @@
|
|||||||
crossOrigin="Anonymous"
|
crossOrigin="Anonymous"
|
||||||
:autoCrop="true"
|
:autoCrop="true"
|
||||||
:fixedNumber="ratio"
|
:fixedNumber="ratio"
|
||||||
:fixed="type !== 'apparel' && isProduct"
|
:fixed="isProduct ? type !== 'apparel' : true"
|
||||||
movable
|
movable
|
||||||
centerBox
|
centerBox
|
||||||
:fixedBox="fixedBox"
|
|
||||||
@realTime="onChange"
|
@realTime="onChange"
|
||||||
outputType="png"
|
outputType="png"
|
||||||
v-bind="bindProps"
|
v-bind="bindProps"
|
||||||
@@ -52,10 +51,6 @@ const props = defineProps({
|
|||||||
type: Boolean,
|
type: Boolean,
|
||||||
default: false
|
default: false
|
||||||
},
|
},
|
||||||
fixedBox: {
|
|
||||||
type: Boolean,
|
|
||||||
default: true
|
|
||||||
},
|
|
||||||
type: {
|
type: {
|
||||||
type: String,
|
type: String,
|
||||||
default: () => ""
|
default: () => ""
|
||||||
@@ -73,12 +68,12 @@ const autoCropHeight = computed(() => {
|
|||||||
const bindProps = computed(() => {
|
const bindProps = computed(() => {
|
||||||
// :autoCropWidth="isProduct ? undefined : type === 'cover' ? 297 : 242"
|
// :autoCropWidth="isProduct ? undefined : type === 'cover' ? 297 : 242"
|
||||||
// :autoCropHeight="isProduct ? undefined : autoCropHeight"
|
// :autoCropHeight="isProduct ? undefined : autoCropHeight"
|
||||||
if (props.isProduct) {
|
// if (props.isProduct) {
|
||||||
return {
|
// return {
|
||||||
autoCropHeight: autoCropHeight.value,
|
// autoCropHeight: autoCropHeight.value,
|
||||||
autoCropWidth: props.type === "cover" ? 297 : 242
|
// autoCropWidth: props.type === "cover" ? 297 : 242
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
})
|
})
|
||||||
|
|
||||||
const onChange = (data) => {
|
const onChange = (data) => {
|
||||||
|
|||||||
@@ -58,7 +58,7 @@ const selectOption = (value: any) => {
|
|||||||
if (index >= 0) {
|
if (index >= 0) {
|
||||||
current.splice(index, 1)
|
current.splice(index, 1)
|
||||||
} else {
|
} else {
|
||||||
current.push(value)
|
current.push(value.toLocaleLowerCase)
|
||||||
}
|
}
|
||||||
emit("update:modelValue", current)
|
emit("update:modelValue", current)
|
||||||
return
|
return
|
||||||
|
|||||||
@@ -198,7 +198,7 @@
|
|||||||
<span class="help-text">{{ $t("SellerListEdit.categoryTips") }}</span>
|
<span class="help-text">{{ $t("SellerListEdit.categoryTips") }}</span>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-item-value no-border">
|
<div class="form-item-value no-border">
|
||||||
<Radio :options="categoryOptions" v-model="currentListing.category" />
|
<Radio multiple :options="categoryOptions" v-model="currentListing.category" />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="license-note flex align-center">
|
<div class="license-note flex align-center">
|
||||||
@@ -242,7 +242,7 @@ import SellerHeader from "../../seller-header.vue"
|
|||||||
import Radio from "./components/Radio.vue"
|
import Radio from "./components/Radio.vue"
|
||||||
import ImageClipDialog from "../../BrandProfile/image-clip-dialog.vue"
|
import ImageClipDialog from "../../BrandProfile/image-clip-dialog.vue"
|
||||||
import { useStore } from "vuex"
|
import { useStore } from "vuex"
|
||||||
import { fetchSketchDetail, uploadFile, fetchListingDetailById, fetchChangeStatus } from "./api"
|
import { fetchSketchDetail, uploadFile, fetchListingDetailById, fetchUpdateListing } from "./api"
|
||||||
|
|
||||||
const ROUTER = useRouter()
|
const ROUTER = useRouter()
|
||||||
const { t } = useI18n()
|
const { t } = useI18n()
|
||||||
@@ -261,6 +261,7 @@ type CategoryOption = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type ListingItem = {
|
type ListingItem = {
|
||||||
|
designItemId: number | string | null
|
||||||
sketch: string | null
|
sketch: string | null
|
||||||
mainProductImage: string
|
mainProductImage: string
|
||||||
cover: string
|
cover: string
|
||||||
@@ -270,7 +271,7 @@ type ListingItem = {
|
|||||||
price: string
|
price: string
|
||||||
desc: string
|
desc: string
|
||||||
gender: string
|
gender: string
|
||||||
category: string
|
category: string[]
|
||||||
prodImageList: Array<{
|
prodImageList: Array<{
|
||||||
url: string
|
url: string
|
||||||
selected?: boolean
|
selected?: boolean
|
||||||
@@ -279,6 +280,25 @@ type ListingItem = {
|
|||||||
}
|
}
|
||||||
type StatusType = "draft" | "publish"
|
type StatusType = "draft" | "publish"
|
||||||
|
|
||||||
|
const createListingItem = (
|
||||||
|
sketch: string | null = null,
|
||||||
|
designItemId: number | string | null = null
|
||||||
|
): ListingItem => ({
|
||||||
|
designItemId,
|
||||||
|
sketch,
|
||||||
|
mainProductImage: "",
|
||||||
|
cover: "",
|
||||||
|
productImage: [],
|
||||||
|
apparelSketch: [],
|
||||||
|
productName: "",
|
||||||
|
price: "",
|
||||||
|
desc: "",
|
||||||
|
gender: "FEMALE",
|
||||||
|
category: null,
|
||||||
|
prodImageList: [],
|
||||||
|
sketchList: []
|
||||||
|
})
|
||||||
|
|
||||||
const topImageList = ["sketch", "mainProductImage", "cover"] as const
|
const topImageList = ["sketch", "mainProductImage", "cover"] as const
|
||||||
const topImageTitleMap: Record<(typeof topImageList)[number], string> = {
|
const topImageTitleMap: Record<(typeof topImageList)[number], string> = {
|
||||||
sketch: "SellerListEdit.sketch",
|
sketch: "SellerListEdit.sketch",
|
||||||
@@ -297,22 +317,7 @@ const currentPage = ref(1)
|
|||||||
const currentIndex = computed(() => currentPage.value - 1)
|
const currentIndex = computed(() => currentPage.value - 1)
|
||||||
|
|
||||||
const itemId = ref("")
|
const itemId = ref("")
|
||||||
const selectList = ref<ListingItem[]>([
|
const selectList = ref<ListingItem[]>([createListingItem()])
|
||||||
{
|
|
||||||
sketch: null,
|
|
||||||
mainProductImage: "",
|
|
||||||
cover: "",
|
|
||||||
productImage: [],
|
|
||||||
apparelSketch: [],
|
|
||||||
productName: "",
|
|
||||||
price: "",
|
|
||||||
desc: "",
|
|
||||||
gender: "FEMALE",
|
|
||||||
category: "",
|
|
||||||
prodImageList: [],
|
|
||||||
sketchList: [{ url: null }]
|
|
||||||
}
|
|
||||||
])
|
|
||||||
|
|
||||||
const prodImgList = computed(() => currentListing.value.prodImageList || [])
|
const prodImgList = computed(() => currentListing.value.prodImageList || [])
|
||||||
|
|
||||||
@@ -355,7 +360,7 @@ const handleSelectProdImg = (index: number) => {
|
|||||||
const cropType = ref("")
|
const cropType = ref("")
|
||||||
const handleClickCrop = (data, type, list = []) => {
|
const handleClickCrop = (data, type, list = []) => {
|
||||||
// console.log(data, type)
|
// console.log(data, type)
|
||||||
console.log(selectList.value[currentIndex.value])
|
// console.log(selectList.value[currentIndex.value])
|
||||||
let origin = []
|
let origin = []
|
||||||
const currentItem = selectList.value[currentIndex.value]
|
const currentItem = selectList.value[currentIndex.value]
|
||||||
if (currentItem.sketch) {
|
if (currentItem.sketch) {
|
||||||
@@ -371,7 +376,7 @@ const handleClickCrop = (data, type, list = []) => {
|
|||||||
cover: "Crop Cover",
|
cover: "Crop Cover",
|
||||||
apparel: "Crop Apparel Sketch"
|
apparel: "Crop Apparel Sketch"
|
||||||
}
|
}
|
||||||
const ratio = type === "apparel" ? [4, 5] : [9, 16]
|
const ratio = type === "cover" ? [4, 5] : [9, 16]
|
||||||
cropType.value = type
|
cropType.value = type
|
||||||
imageClipDialogRef.value.open(
|
imageClipDialogRef.value.open(
|
||||||
data,
|
data,
|
||||||
@@ -379,7 +384,7 @@ const handleClickCrop = (data, type, list = []) => {
|
|||||||
// console.log(file)
|
// console.log(file)
|
||||||
uploadFile(file).then((res) => {
|
uploadFile(file).then((res) => {
|
||||||
console.log(res)
|
console.log(res)
|
||||||
selectList.value[currentIndex.value].sketch = res
|
selectList.value[currentIndex.value][type] = res
|
||||||
})
|
})
|
||||||
},
|
},
|
||||||
{ ratio, isPreview: true, title: titleList[type], isProduct: true },
|
{ ratio, isPreview: true, title: titleList[type], isProduct: true },
|
||||||
@@ -387,10 +392,6 @@ const handleClickCrop = (data, type, list = []) => {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
const handleSetStatus = (type: StatusType) => {
|
|
||||||
// fetchChangeStatus()
|
|
||||||
}
|
|
||||||
|
|
||||||
const hasValue = (value: unknown) =>
|
const hasValue = (value: unknown) =>
|
||||||
value !== null && value !== undefined && String(value).trim() !== ""
|
value !== null && value !== undefined && String(value).trim() !== ""
|
||||||
|
|
||||||
@@ -438,17 +439,59 @@ const validatePublishRequired = () => {
|
|||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const handleSaveForm = async (type: StatusType) => {
|
||||||
|
const paramsList = []
|
||||||
|
selectList.value.forEach((item) => {
|
||||||
|
const params = {
|
||||||
|
id: null,
|
||||||
|
title: selectList.value[currentIndex.value].productName,
|
||||||
|
description: selectList.value[currentIndex.value].desc,
|
||||||
|
price: selectList.value[currentIndex.value].price,
|
||||||
|
status: type === "draft" ? 0 : 1,
|
||||||
|
images: [],
|
||||||
|
designFor: selectList.value[currentIndex.value].gender.toLowerCase,
|
||||||
|
productCategory: selectList.value[currentIndex.value].category
|
||||||
|
}
|
||||||
|
//
|
||||||
|
topImageList.forEach((el) => {
|
||||||
|
params.images.push({
|
||||||
|
category: el,
|
||||||
|
imageUrl: selectList.value[currentIndex.value][el],
|
||||||
|
isSelected: 1
|
||||||
|
})
|
||||||
|
})
|
||||||
|
selectList.value[currentIndex.value].prodImageList.forEach((item) => {
|
||||||
|
if (item.selected) {
|
||||||
|
params.images.push({
|
||||||
|
category: "main_product",
|
||||||
|
imageUrl: item.url,
|
||||||
|
isSelected: Number(item.selected)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
selectList.value[currentIndex.value].sketchList.forEach((item) => {
|
||||||
|
params.images.push({
|
||||||
|
category: "apparel",
|
||||||
|
imageUrl: item.url,
|
||||||
|
isSelected: 1
|
||||||
|
})
|
||||||
|
})
|
||||||
|
paramsList.push(params)
|
||||||
|
})
|
||||||
|
console.log(paramsList)
|
||||||
|
fetchUpdateListing(paramsList)
|
||||||
|
}
|
||||||
const handleClickMenu = async (status: StatusType) => {
|
const handleClickMenu = async (status: StatusType) => {
|
||||||
if (status === "publish" && !validatePublishRequired()) return
|
if (status === "publish" && !validatePublishRequired()) return
|
||||||
|
|
||||||
await handleSetStatus(status)
|
await handleSaveForm(status)
|
||||||
if (status === "draft") {
|
if (status === "draft") {
|
||||||
// save draft logic
|
// save draft logic
|
||||||
console.log("Saving draft...", currentListing.value)
|
// console.log("Saving draft...", currentListing.value)
|
||||||
ROUTER.push({ name: "Status", params: { status: "draft" } })
|
ROUTER.push({ name: "Status", params: { status: "draft" } })
|
||||||
} else if (status === "publish") {
|
} else if (status === "publish") {
|
||||||
// publish logic
|
// publish logic
|
||||||
console.log("Publishing...", currentListing.value)
|
// console.log("Publishing...", currentListing.value)
|
||||||
ROUTER.push({ name: "Status", params: { status: "publish" } })
|
ROUTER.push({ name: "Status", params: { status: "publish" } })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -457,6 +500,7 @@ const handleFetchItemDetial = (list) => {
|
|||||||
fetchSketchDetail(list).then((res) => {
|
fetchSketchDetail(list).then((res) => {
|
||||||
console.log(res)
|
console.log(res)
|
||||||
res.forEach((item, index) => {
|
res.forEach((item, index) => {
|
||||||
|
if (!selectList.value[index]) return
|
||||||
selectList.value[index].sketchList = item.clothes.map((el) => ({ url: el }))
|
selectList.value[index].sketchList = item.clothes.map((el) => ({ url: el }))
|
||||||
selectList.value[index].prodImageList = item.toProductImageUrls.map((el) => ({
|
selectList.value[index].prodImageList = item.toProductImageUrls.map((el) => ({
|
||||||
url: el
|
url: el
|
||||||
@@ -469,11 +513,15 @@ const handleFetchItemDetial = (list) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
itemId.value = history.state.id
|
const designItemIds = history.state?.designItemIds || []
|
||||||
history.state.designItemIds.forEach((item, index) => {
|
itemId.value = history.state?.id || ""
|
||||||
selectList.value[index].sketch = item.designOutfitUrl
|
if (!designItemIds.length) return
|
||||||
})
|
|
||||||
const list = history.state.designItemIds.map((el) => el.designItemId)
|
currentPage.value = 1
|
||||||
|
selectList.value = designItemIds.map((item) =>
|
||||||
|
createListingItem(item.designOutfitUrl, item.designItemId)
|
||||||
|
)
|
||||||
|
const list = designItemIds.map((el) => el.designItemId)
|
||||||
console.log("list", list.length, list)
|
console.log("list", list.length, list)
|
||||||
handleFetchItemDetial(list)
|
handleFetchItemDetial(list)
|
||||||
})
|
})
|
||||||
|
|||||||
Reference in New Issue
Block a user