Files
aida_front/src/views/SellerDashboard/MyListings/EditDetail/index.vue

1087 lines
26 KiB
Vue
Raw Normal View History

2026-04-09 14:23:52 +08:00
<template>
2026-04-15 10:54:21 +08:00
<div class="edit-detail-wrapper flex-1">
2026-04-09 14:23:52 +08:00
<seller-header
2026-04-15 10:54:21 +08:00
class="edit-detail-header"
2026-04-09 14:23:52 +08:00
title="Edit Listing Details"
:breadcrumbs="[
{ title: 'My Listings', name: 'myListingsIndex' },
{ title: 'Select Collection', name: 'myListingsSelect' },
{ title: 'Select Sketch', name: 'myListingsSelectItem' },
{ title: 'Edit Listing Details', name: 'EditDetail' }
]"
>
<template #right>
<div class="operate-menu flex">
2026-04-16 17:31:31 +08:00
<div class="menu-btn flex align-center save" @click="handleClickMenu('draft')">
2026-04-15 10:54:21 +08:00
<span>{{ $t("SellerListEdit.saveDraft") }}</span>
2026-04-16 14:03:05 +08:00
<SvgIcon name="CSave" size="16" />
2026-04-09 14:23:52 +08:00
</div>
2026-04-16 17:31:31 +08:00
<div
class="menu-btn flex align-center publish"
@click="handleClickMenu('publish')"
>
2026-04-15 10:54:21 +08:00
<span>{{ $t("SellerListEdit.publish") }}</span>
2026-04-16 14:03:05 +08:00
<SvgIcon name="CPublish" size="16" />
2026-04-09 14:23:52 +08:00
</div>
</div>
</template>
</seller-header>
2026-04-16 14:03:05 +08:00
<div class="edit-detail-content flex">
2026-04-09 14:23:52 +08:00
<div class="left">
<div class="main-image-container flex">
2026-04-15 10:54:21 +08:00
<div
v-for="type in topImageList"
:key="type"
:class="`main-image-item flex flex-col align-center ${type}`"
>
<div class="title" :class="{ required: type !== 'mainProductImage' }">
2026-04-27 14:39:59 +08:00
{{ $t(topImageTitleMap[type]) }}
2026-04-15 10:54:21 +08:00
</div>
<div class="sketch-item flex flex-center" :class="type">
2026-04-17 17:59:51 +08:00
<div
v-if="previewImageMap[type]"
class="crop-tool flex flex-center"
2026-04-21 13:47:23 +08:00
@click="handleClickCrop(previewImageMap[type], type)"
2026-04-17 17:59:51 +08:00
>
2026-04-15 10:54:21 +08:00
<SvgIcon name="CCrop" color="#fff" size="12" />
</div>
<img
2026-04-16 14:03:05 +08:00
v-if="previewImageMap[type]"
:src="previewImageMap[type]"
2026-04-15 10:54:21 +08:00
class="sketch-img"
2026-04-27 14:39:59 +08:00
:class="type"
2026-04-15 10:54:21 +08:00
alt=""
/>
2026-04-16 14:03:05 +08:00
<div v-else class="trigger flex flex-col align-center">
2026-04-27 14:39:59 +08:00
<div
v-if="type === 'cover'"
class="cover-trigger flex flex-col align-center"
@click="handleClickCrop(null, 'cover')"
>
2026-04-15 10:54:21 +08:00
<SvgIcon
class="trigger-icon"
name="CCrop"
color="#585858"
size="24"
/>
<div class="trigger-tips">
{{ $t("SellerListEdit.cropDesc") }}
</div>
2026-04-27 14:39:59 +08:00
</div>
2026-04-15 10:54:21 +08:00
<template v-else>
2026-04-16 14:03:05 +08:00
<div class="trigger-img placeholder"></div>
2026-04-15 10:54:21 +08:00
<div class="trigger-tips">
{{ $t("SellerListEdit.productImageDesc") }}
</div>
</template>
</div>
</div>
</div>
</div>
<div class="product-image-list-container">
<div class="title flex align-center space-between">
<div class="title-left">
<span class="main-title">{{
$t("SellerListEdit.productImageMainTitle")
}}</span>
<span class="sub-title">{{
$t("SellerListEdit.productImageSubTitle")
}}</span>
</div>
<div class="title-right">
{{ selectedProdImgs }}/{{ prodImgList.length }} selected
</div>
</div>
<div class="product-image-list flex">
<div
v-for="(item, index) in prodImgList"
2026-04-16 14:03:05 +08:00
:key="index"
2026-04-15 10:54:21 +08:00
class="product-image-item flex flex-center"
:class="{ selected: item.selected }"
@click="handleSelectProdImg(index)"
>
<img
2026-04-16 14:03:05 +08:00
v-if="item.selected"
2026-04-15 10:54:21 +08:00
src="@/assets/images/seller/checked.png"
class="checked"
2026-04-16 14:03:05 +08:00
alt=""
2026-04-15 10:54:21 +08:00
/>
2026-04-16 14:03:05 +08:00
<img class="img-src" :src="item.url" alt="" />
2026-04-15 10:54:21 +08:00
<div
v-if="item.selected && index === firstSelectedIndex"
class="main-pic"
>
main
</div>
</div>
</div>
</div>
<div class="apparel-container">
<div class="title">
<span class="main-title">{{
$t("SellerListEdit.apparelSketchTitle")
}}</span>
<span class="sub-title">
{{ $t("SellerListEdit.apparelSketchSubTitle") }}</span
>
</div>
<div class="sketch-list-container flex">
<div
2026-04-17 17:59:51 +08:00
v-for="(item, index) in selectList[currentIndex].sketchList"
2026-04-15 10:54:21 +08:00
:key="index"
class="sketch-element flex flex-center"
>
2026-04-16 14:03:05 +08:00
<img class="img-src" :src="item.url" alt="" />
2026-04-21 13:47:23 +08:00
<div
class="crop-tool flex flex-center"
@click="handleClickCrop(item.url, 'apparel')"
>
2026-04-15 10:54:21 +08:00
<SvgIcon name="CCrop" color="#fff" size="12" />
</div>
</div>
</div>
</div>
</div>
<div class="right">
<div class="form-container flex flex-col">
<div class="form-item">
2026-04-16 17:31:31 +08:00
<div class="form-item-label required">
{{ $t("SellerListEdit.productName") }}
</div>
2026-04-15 10:54:21 +08:00
<div class="form-item-value product-name">
<a-input
2026-04-16 14:03:05 +08:00
v-model:value="currentListing.productName"
2026-04-15 10:54:21 +08:00
show-count
placeholder="Enter product name"
:bordered="false"
:maxlength="60"
/>
</div>
</div>
<div class="form-item">
2026-04-16 17:31:31 +08:00
<div class="form-item-label required">{{ $t("SellerListEdit.price") }}</div>
2026-04-15 10:54:21 +08:00
<div class="form-item-value price flex align-center">
<span>HK$</span>
<a-input
2026-04-16 14:03:05 +08:00
v-model:value="currentListing.price"
2026-04-15 10:54:21 +08:00
placeholder="0.00"
:bordered="false"
/>
</div>
</div>
2026-04-16 14:03:05 +08:00
<div class="form-item">
2026-04-16 17:31:31 +08:00
<div class="form-item-label required">
{{ $t("SellerListEdit.productDescription") }}
</div>
2026-04-16 14:03:05 +08:00
<div class="form-item-value desc">
<a-textarea
v-model:value="currentListing.desc"
show-count
:rows="4"
placeholder="Enter product description"
:bordered="false"
:maxlength="500"
/>
</div>
</div>
<div class="form-item">
2026-04-16 17:31:31 +08:00
<div class="form-item-label required">
{{ $t("SellerListEdit.designFor") }}
</div>
2026-04-16 14:03:05 +08:00
<div class="form-item-value no-border">
<Radio :options="genderOptions" v-model="currentListing.gender" />
</div>
</div>
<div class="form-item">
<div class="form-item-label with-tip">
2026-04-16 17:31:31 +08:00
<span class="required">{{ $t("SellerListEdit.productCategory") }}</span>
<span class="help-text">{{ $t("SellerListEdit.categoryTips") }}</span>
2026-04-16 14:03:05 +08:00
</div>
<div class="form-item-value no-border">
<Radio
multiple
:options="categoryOptions"
v-model="currentListing.category"
/>
2026-04-16 14:03:05 +08:00
</div>
</div>
<div class="license-note flex align-center">
<img src="@/assets/images/seller/tips.png" class="info-icon" />
<div class="note-copy">
2026-04-16 17:31:31 +08:00
{{ $t("SellerListEdit.policy") }}
<a href="javascript:void(0)">{{ $t("SellerListEdit.learnMore") }}</a>
2026-04-16 14:03:05 +08:00
</div>
</div>
</div>
2026-04-27 14:39:59 +08:00
<div class="page-control flex align-center" v-if="selectList.length > 1">
2026-04-16 14:03:05 +08:00
<a-pagination
v-model:current="currentPage"
:total="selectList.length"
:page-size="1"
showQuickJumper
showLessItems
responsive
:showSizeChanger="false"
/>
2026-04-09 14:23:52 +08:00
</div>
</div>
</div>
</div>
2026-04-17 17:59:51 +08:00
<ImageClipDialog
ref="imageClipDialogRef"
fixedBox
isProduct
:info="false"
2026-04-21 13:47:23 +08:00
v-bind="$attrs"
:type="cropType"
2026-04-17 17:59:51 +08:00
/>
2026-04-09 14:23:52 +08:00
</template>
<script setup lang="ts">
import { computed, ref, watch, onMounted } from "vue"
import { useRouter } from "vue-router"
import { useI18n } from "vue-i18n"
import { message } from "ant-design-vue"
import SellerHeader from "../../seller-header.vue"
import Radio from "./components/Radio.vue"
import ImageClipDialog from "../../BrandProfile/image-clip-dialog.vue"
import { useStore } from "vuex"
import {
fetchSketchDetail,
uploadFile,
fetchListingDetailById,
fetchUpdateListing
} from "./api"
const ROUTER = useRouter()
const { t } = useI18n()
const imageClipDialogRef = ref<InstanceType<typeof ImageClipDialog> | null>(null)
defineOptions({
name: "EditDetail"
})
const STORE = useStore()
2026-04-15 10:54:21 +08:00
type CategoryOption = {
label: string
value: string
2026-04-16 14:03:05 +08:00
}
type ListingItem = {
designItemId: number | string | null
sketch: string | null
mainProductImage: string
cover: string
productImage: string[]
apparelSketch: string[]
productName: string
price: string
desc: string
gender: string
2026-04-28 11:43:12 +08:00
category: string[] | null
prodImageList: Array<{
url: string
selected?: boolean
}>
sketchList: Array<{ url: string | null }>
2026-04-27 14:39:59 +08:00
}
2026-04-28 16:03:52 +08:00
type ListingDetailImage = {
category?: string | null
imageUrl?: string | null
isSelected?: boolean | number | string | null
sortOrder?: number | null
}
type ListingDetailResponse = {
id?: number | string | null
title?: string | null
description?: string | null
price?: number | string | null
designFor?: string | null
productCategory?: string | string[] | null
images?: ListingDetailImage[] | null
}
type StatusType = "draft" | "publish"
const createListingItem = (
sketch: string | null = null,
designItemId: number | string | null = null
): ListingItem => ({
designItemId,
sketch,
2026-04-09 14:23:52 +08:00
mainProductImage: "",
cover: "",
productImage: [],
apparelSketch: [],
productName: "",
price: "",
2026-04-15 10:54:21 +08:00
desc: "",
2026-04-16 14:03:05 +08:00
gender: "FEMALE",
category: null,
prodImageList: [],
sketchList: []
})
const topImageList = ["sketch", "mainProductImage", "cover"] as const
const topImageTitleMap: Record<(typeof topImageList)[number], string> = {
sketch: "SellerListEdit.sketch",
mainProductImage: "SellerListEdit.mainProductImage",
cover: "SellerListEdit.cover"
2026-04-27 14:39:59 +08:00
}
const genderOptions = STORE.state.UserHabit?.sex.value || []
const fallbackCategoryOptions: Record<string, CategoryOption[]> = {
MALE: STORE.state.UserHabit?.MalePosition || [],
FEMALE: STORE.state.UserHabit?.FemalePosition || []
2026-04-21 13:47:23 +08:00
}
2026-04-15 10:54:21 +08:00
const currentPage = ref(1)
const currentIndex = computed(() => currentPage.value - 1)
2026-04-16 14:03:05 +08:00
const itemId = ref("")
const selectList = ref<ListingItem[]>([createListingItem()])
2026-04-16 14:03:05 +08:00
const prodImgList = computed(() => currentListing.value.prodImageList || [])
2026-04-16 14:03:05 +08:00
const categoryOptions = computed(() => {
const gender = selectList.value[currentIndex.value].gender
return fallbackCategoryOptions[gender] || []
})
const currentListing = computed(() => selectList.value[currentIndex.value])
2026-04-16 14:03:05 +08:00
const previewImageMap = computed(() => ({
sketch: currentListing.value.sketch,
mainProductImage: currentListing.value.mainProductImage,
cover: currentListing.value.cover
}))
2026-04-28 16:03:52 +08:00
const firstSelectedIndex = ref<number | null>(null) //显示main标签的图片索引
const selectedProdImgs = computed(() => {
return prodImgList.value.filter((item) => item.selected).length
})
2026-04-15 10:54:21 +08:00
2026-04-28 16:03:52 +08:00
const getSortedDetailImages = (images: ListingDetailImage[] = []) => {
return [...images].sort((prev, next) => (prev.sortOrder ?? 0) - (next.sortOrder ?? 0))
}
const getImageSelected = (value: ListingDetailImage["isSelected"]) =>
value === true || value === 1 || value === "1"
const normalizeDetailGender = (value: ListingDetailResponse["designFor"]) => {
const gender = String(value || "").toUpperCase()
return gender === "MALE" || gender === "FEMALE" ? gender : "FEMALE"
}
const normalizeDetailCategory = (
value: ListingDetailResponse["productCategory"]
): ListingItem["category"] => {
const categories = Array.isArray(value) ? value : value ? [value] : []
const normalized = categories
.filter((category) => category !== null && typeof category !== "undefined")
.map((category) => String(category).toLowerCase())
return normalized.length ? normalized : null
}
const createListingItemFromDetail = (detail: ListingDetailResponse): ListingItem => {
const listing = createListingItem()
listing.productName = detail.title || ""
listing.price =
detail.price === null || typeof detail.price === "undefined" ? "" : String(detail.price)
listing.desc = detail.description || ""
listing.gender = normalizeDetailGender(detail.designFor)
listing.category = normalizeDetailCategory(detail.productCategory)
getSortedDetailImages(detail.images || []).forEach((image) => {
const imageUrl = image.imageUrl || ""
if (!imageUrl) return
if (image.category === "cover") {
listing.cover = imageUrl
return
}
if (image.category === "sketch") {
listing.sketch = imageUrl
return
}
if (image.category === "mainProductImage") {
listing.mainProductImage = imageUrl
return
}
if (image.category === "main_product" || image.category === "product") {
listing.prodImageList.push({
url: imageUrl,
selected: getImageSelected(image.isSelected)
})
return
}
if (image.category === "apparel") {
listing.sketchList.push({ url: imageUrl })
}
})
if (!listing.mainProductImage) {
listing.mainProductImage =
listing.prodImageList.find((item) => item.selected)?.url || ""
}
listing.productImage = listing.prodImageList.map((item) => item.url)
listing.apparelSketch = listing.sketchList
.map((item) => item.url)
.filter((url): url is string => Boolean(url))
return listing
}
const handleSelectProdImg = (index: number) => {
const target = prodImgList.value[index]
const willSelect = !target.selected
2026-04-16 14:03:05 +08:00
target.selected = willSelect
2026-04-16 14:03:05 +08:00
2026-04-28 16:03:52 +08:00
if (willSelect && firstSelectedIndex.value === null) {
currentListing.value.mainProductImage = target.url
firstSelectedIndex.value = index
2026-04-28 16:03:52 +08:00
return
}
if (!willSelect && currentListing.value.mainProductImage === target.url) {
firstSelectedIndex.value = null
currentListing.value.mainProductImage = ""
}
2026-04-27 14:39:59 +08:00
}
const cropType = ref("")
const handleClickCrop = (data, type, list = []) => {
// console.log(data, type)
// console.log(selectList.value[currentIndex.value])
let origin = []
const currentItem = selectList.value[currentIndex.value]
if (currentItem.sketch) {
origin.push({ type: "sketch", url: currentItem.sketch })
}
if (currentItem.mainProductImage) {
origin.push({ type: "mainProductImage", url: currentItem.mainProductImage })
}
if (type !== "cover") origin = []
const titleList = {
sketch: "Crop Sketch",
mainProductImage: "Crop Main Product Image",
cover: "Crop Cover",
apparel: "Crop Apparel Sketch"
}
const ratio = type === "cover" ? [4, 5] : [9, 16]
cropType.value = type
imageClipDialogRef.value.open(
data,
(file) => {
// console.log(file)
uploadFile(file).then((res) => {
selectList.value[currentIndex.value][type] = res
})
},
{ ratio, isPreview: true, title: titleList[type], isProduct: true },
origin
2026-04-27 14:39:59 +08:00
)
}
const hasValue = (value: unknown) =>
value !== null && value !== undefined && String(value).trim() !== ""
const getMissingRequiredField = (item: ListingItem) => {
const cover = item.cover || item.mainProductImage || item.sketch
const requiredFields = [
{ value: item.sketch, label: t("SellerListEdit.sketch") },
{ value: cover, label: t("SellerListEdit.cover") },
{ value: item.productName, label: t("SellerListEdit.productName") },
{ value: item.price, label: t("SellerListEdit.price") },
{ value: item.desc, label: t("SellerListEdit.productDescription") },
{ value: item.gender, label: t("SellerListEdit.designFor") },
{ value: item.category, label: t("SellerListEdit.productCategory") }
]
const missingField = requiredFields.find((field) => !hasValue(field.value))
if (missingField) return missingField.label
const missingSketchIndex = item.sketchList.findIndex((sketch) => !hasValue(sketch.url))
if (item.sketchList.length === 0 || missingSketchIndex !== -1) {
return `${t("SellerListEdit.apparelSketchTitle").trim()} ${
missingSketchIndex === -1 ? 1 : missingSketchIndex + 1
}`
2026-04-27 16:58:45 +08:00
}
return ""
2026-04-21 13:47:23 +08:00
}
const validatePublishRequired = () => {
for (let index = 0; index < selectList.value.length; index += 1) {
const field = getMissingRequiredField(selectList.value[index])
if (!field) continue
currentPage.value = index + 1
message.warning(
t(
selectList.value.length > 1
? "SellerListEdit.requiredFieldTipsWithPage"
: "SellerListEdit.requiredFieldTips",
{ index: index + 1, field }
)
)
return false
}
return true
}
const handleSaveForm = async (type: StatusType) => {
const paramsList = []
2026-04-28 16:03:52 +08:00
selectList.value.forEach((item: ListingItem) => {
const params = {
id: null,
2026-04-28 16:03:52 +08:00
title: item.productName,
description: item.desc,
price: item.price,
status: type === "draft" ? 0 : 1,
images: [],
2026-04-28 16:03:52 +08:00
designFor: item.gender.toLowerCase,
productCategory: item.category
}
2026-04-28 16:03:52 +08:00
;["sketch", "cover"].forEach((el) => {
params.images.push({
category: el,
2026-04-28 16:03:52 +08:00
imageUrl: item[el],
isSelected: 1
})
2026-04-27 16:58:45 +08:00
})
2026-04-28 16:03:52 +08:00
if (item.mainProductImage) {
params.images.push({
category: 'main_product',
imageUrl: item.mainProductImage,
isSeleted:1
})
}
item.prodImageList.forEach((item) => {
params.images.push({
category: "product",
imageUrl: item.url,
isSelected: Number(!!item.selected)
})
})
2026-04-28 16:03:52 +08:00
item.sketchList.forEach((item) => {
2026-04-27 16:58:45 +08:00
params.images.push({
category: "apparel",
2026-04-27 16:58:45 +08:00
imageUrl: item.url,
isSelected: 1
2026-04-27 16:58:45 +08:00
})
})
paramsList.push(params)
2026-04-27 16:58:45 +08:00
})
console.log(paramsList)
2026-04-28 16:03:52 +08:00
debugger
await fetchUpdateListing(paramsList)
}
const handleClickMenu = async (status: StatusType) => {
2026-04-28 11:43:12 +08:00
if (status === "draft" && !selectList.value[currentIndex.value].cover) {
message.error("请先完成封面制作")
return
}
if (status === "publish" && !validatePublishRequired()) return
await handleSaveForm(status)
if (status === "draft") {
// save draft logic
// console.log("Saving draft...", currentListing.value)
ROUTER.push({ name: "Status", params: { status: "draft" } })
} else if (status === "publish") {
// publish logic
// console.log("Publishing...", currentListing.value)
ROUTER.push({ name: "Status", params: { status: "publish" } })
}
2026-04-16 17:31:31 +08:00
}
2026-04-27 09:40:00 +08:00
const handleFetchItemDetial = (list) => {
fetchSketchDetail(list).then((res) => {
console.log(res)
res.forEach((item, index) => {
if (!selectList.value[index]) return
selectList.value[index].sketchList = item.clothes.map((el) => ({ url: el }))
selectList.value[index].prodImageList = item.toProductImageUrls.map((el) => ({
url: el
}))
})
2026-04-27 14:39:59 +08:00
})
}
2026-04-27 09:40:00 +08:00
2026-04-28 16:03:52 +08:00
const handleGetDetailById = () => {
fetchListingDetailById(itemId.value).then((res: ListingDetailResponse) => {
const listing = createListingItemFromDetail(res)
const selectedIndex = listing.prodImageList.findIndex((item) => item.selected)
2026-04-28 16:03:52 +08:00
currentPage.value = 1
selectList.value = [listing]
firstSelectedIndex.value = selectedIndex === -1 ? null : selectedIndex
})
}
onMounted(() => {
const data = history.state
if (data?.type === "edit") {
itemId.value = history.state?.id || ""
handleGetDetailById()
} else {
const designItemIds = history.state?.designItemIds || []
if (!designItemIds.length) return
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)
handleFetchItemDetial(list)
}
2026-04-27 09:40:00 +08:00
})
2026-04-09 14:23:52 +08:00
</script>
<style lang="less" scoped>
.c-svg {
width: initial;
height: initial;
2026-04-15 10:54:21 +08:00
}
.edit-detail-wrapper {
display: flex;
flex-direction: column;
height: 100%;
min-height: 0;
margin-top: -1rem;
overflow: hidden;
2026-04-16 14:03:05 +08:00
&::-webkit-scrollbar {
display: none;
}
2026-04-09 14:23:52 +08:00
.menu-btn {
height: 6rem;
border: 0.15rem solid #000000;
border-radius: 4rem;
text-align: center;
line-height: 6rem;
padding: 0 2rem;
font-size: 1.6rem;
column-gap: 0.8rem;
cursor: pointer;
transition: all 0.25s ease;
&:hover {
background: #000;
color: #fff;
}
2026-04-16 14:03:05 +08:00
// &.publish:hover {
// background: #fff;
// color: #000;
// }
}
2026-04-09 14:23:52 +08:00
.edit-detail-header {
margin-bottom: 2rem;
}
2026-04-16 14:03:05 +08:00
.operate-menu {
column-gap: 2rem;
2026-04-16 14:03:05 +08:00
// .publish {
// background-color: #000000;
// color: #ffffff;
// }
2026-04-15 10:54:21 +08:00
}
2026-04-16 14:03:05 +08:00
}
.edit-detail-content {
flex: 1;
min-height: 0;
overflow-y: auto;
justify-content: space-between;
2026-04-16 14:03:05 +08:00
&::-webkit-scrollbar {
width: 0;
height: 0;
2026-04-15 10:54:21 +08:00
}
2026-04-16 14:03:05 +08:00
.required {
&::after {
content: "*";
color: #df2b2c;
margin-left: 0.4rem;
}
}
2026-04-16 14:03:05 +08:00
.left {
// flex: 1;
// min-width: 0;
2026-04-16 14:03:05 +08:00
.main-image-container {
// max-width: 80.2rem;
column-gap: 3.5rem;
.main-image-item {
flex-shrink: 0;
2026-04-16 14:03:05 +08:00
.title {
font-size: 1.4rem;
margin-bottom: 0.8rem;
text-align: center;
2026-04-15 10:54:21 +08:00
}
2026-04-16 14:03:05 +08:00
.sketch-item {
width: 11.6rem;
height: 20.4rem;
border: 0.15rem solid #d1d1d1;
border-radius: 1rem;
position: relative;
background-color: #f6f6f6;
overflow: hidden;
2026-04-16 14:03:05 +08:00
&.cover {
width: 16.2rem;
background-image: url("data:image/svg+xml,%3csvg width='100%25' height='100%25' xmlns='http://www.w3.org/2000/svg'%3e%3crect width='100%25' height='100%25' rx='11' ry='11' fill='none' stroke='%23D1D1D1' stroke-width='1.5' stroke-dasharray='8%2c 5' stroke-linecap='square'/%3e%3c/svg%3e");
border: none;
2026-04-27 14:39:59 +08:00
}
2026-04-16 14:03:05 +08:00
.crop-tool {
position: absolute;
top: 0.8rem;
right: 0.8rem;
width: 2rem;
height: 2rem;
border-radius: 50%;
background-color: #000000;
z-index: 1;
cursor: pointer;
2026-04-27 14:39:59 +08:00
}
2026-04-16 14:03:05 +08:00
.sketch-img {
height: 100%;
&.sketch {
height: initial;
width: 100%;
}
2026-04-15 10:54:21 +08:00
}
2026-04-16 14:03:05 +08:00
.trigger {
cursor: pointer;
height: 100%;
padding: 6rem 2rem 0;
&,
.cover-trigger {
row-gap: 1.2rem;
}
.placeholder {
width: 2.4rem;
height: 2.4rem;
border-radius: 0.6rem;
background: linear-gradient(135deg, #efefef 0%, #cdcdcd 100%);
}
.trigger-tips {
font-size: 1.2rem;
text-align: center;
color: #585858;
line-height: 1.3;
}
2026-04-15 10:54:21 +08:00
}
}
}
}
2026-04-16 14:03:05 +08:00
.product-image-list-container {
margin-top: 3rem;
2026-04-16 14:03:05 +08:00
.title {
2026-04-15 10:54:21 +08:00
font-size: 1.4rem;
margin-bottom: 1.2rem;
2026-04-16 14:03:05 +08:00
.main-title {
font-weight: 400;
font-style: bold;
}
2026-04-15 10:54:21 +08:00
.sub-title {
font-size: 1.2rem;
color: #999;
}
2026-04-16 14:03:05 +08:00
.title-right {
color: #585858;
font-size: 1.4rem;
}
2026-04-15 10:54:21 +08:00
}
2026-04-16 14:03:05 +08:00
.product-image-list {
overflow-x: auto;
overflow-y: hidden;
column-gap: 0.8rem;
max-width: 80.2rem;
padding-bottom: 1.2rem;
2026-04-09 14:23:52 +08:00
&::-webkit-scrollbar {
height: 0.8rem;
}
2026-04-16 14:03:05 +08:00
&::-webkit-scrollbar-track {
background: #d9d9d9;
border-radius: 0.8rem;
2026-04-15 10:54:21 +08:00
}
2026-04-16 14:03:05 +08:00
&::-webkit-scrollbar-thumb {
background: #000000;
border-radius: 0.8rem;
2026-04-15 10:54:21 +08:00
}
2026-04-16 14:03:05 +08:00
.product-image-item {
width: 11.6rem;
height: 20.6rem;
border-radius: 1rem;
border: 0.15rem solid #c7c7c7;
position: relative;
cursor: pointer;
overflow: hidden;
flex-shrink: 0;
2026-04-16 14:03:05 +08:00
2026-04-15 10:54:21 +08:00
&::after {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
content: "";
background-color: #fcfcfc;
opacity: 0.7;
border-radius: 1rem;
2026-04-15 10:54:21 +08:00
}
2026-04-16 14:03:05 +08:00
&.selected {
border-color: #000;
2026-04-16 14:03:05 +08:00
&::after {
display: none;
}
}
2026-04-16 14:03:05 +08:00
.checked {
width: 2rem;
height: 2rem;
position: absolute;
top: 0.8rem;
right: 0.8rem;
z-index: 1;
}
.img-src {
height: 100%;
}
.main-pic {
position: absolute;
height: 2.4rem;
line-height: 2.4rem;
left: 0.8rem;
right: 0.8rem;
bottom: 0.8rem;
z-index: 1;
background: rgba(0, 0, 0, 0.8);
color: #fff;
font-size: 1.4rem;
border-radius: 1.2rem;
text-align: center;
}
2026-04-15 10:54:21 +08:00
}
}
}
2026-04-16 14:03:05 +08:00
.apparel-container {
margin-top: 3rem;
2026-04-16 14:03:05 +08:00
.title {
font-size: 1.4rem;
margin-bottom: 0.8rem;
2026-04-16 14:03:05 +08:00
.main-title {
font-weight: 400;
font-style: bold;
2026-04-16 14:03:05 +08:00
&::after {
content: "*";
color: #df2b2c;
}
2026-04-15 10:54:21 +08:00
}
2026-04-16 14:03:05 +08:00
.sub-title {
font-size: 1.2rem;
color: #999;
}
2026-04-15 10:54:21 +08:00
}
2026-04-16 14:03:05 +08:00
.sketch-list-container {
column-gap: 1rem;
flex-wrap: wrap;
2026-04-16 14:03:05 +08:00
.sketch-element {
width: 10rem;
height: 14.6rem;
border: 0.15rem solid #c7c7c7;
border-radius: 1.2rem;
position: relative;
overflow: hidden;
2026-04-16 14:03:05 +08:00
.img-src {
// height: 100%;
width: 100%;
}
2026-04-16 14:03:05 +08:00
.crop-tool {
position: absolute;
top: 0.4rem;
right: 0.4rem;
width: 2rem;
height: 2rem;
border-radius: 50%;
background-color: #000000;
}
2026-04-15 10:54:21 +08:00
}
}
}
}
2026-04-16 14:03:05 +08:00
.right {
width: 55.2rem;
flex-shrink: 0;
2026-04-16 14:03:05 +08:00
.form-container {
row-gap: 3rem;
2026-04-16 14:03:05 +08:00
.form-item {
.form-item-label {
font-size: 1.4rem;
font-weight: 400;
font-style: bold;
margin-bottom: 0.6rem;
line-height: 1.5;
2026-04-16 14:03:05 +08:00
&.with-tip {
display: flex;
align-items: center;
column-gap: 0.8rem;
}
2026-04-16 14:03:05 +08:00
.help-text {
font-size: 1rem;
color: #999;
}
2026-04-15 10:54:21 +08:00
}
.form-item-value {
border: 0.16rem solid #d1d1d1;
border-radius: 1.2rem;
position: relative;
padding: 1.6rem;
box-sizing: border-box;
2026-04-15 10:54:21 +08:00
font-size: 1.2rem;
2026-04-16 14:03:05 +08:00
color: #000;
&.no-border {
border: none;
padding: 0;
}
2026-04-16 14:03:05 +08:00
&.price {
column-gap: 0.6rem;
}
2026-04-16 14:03:05 +08:00
:deep(.ant-input),
:deep(.ant-input-affix-wrapper),
:deep(.ant-input-textarea textarea) {
border: none;
padding: 0;
font-size: 1.2rem;
color: #000;
box-shadow: none;
}
2026-04-16 14:03:05 +08:00
:deep(.ant-input) {
line-height: 1.5;
}
:deep(.ant-input-show-count-suffix) {
2026-04-16 14:03:05 +08:00
color: #df2c2c;
font-size: 1rem;
}
:deep(textarea.ant-input) {
resize: none;
min-height: 5.4rem;
padding-bottom: 1.8rem;
line-height: 1.5;
}
:deep(.ant-input-textarea-show-count) {
position: relative;
&::after {
float: none;
position: absolute;
color: #df2c2c;
bottom: 0;
right: 1.6rem;
}
2026-04-16 14:03:05 +08:00
}
}
2026-04-15 10:54:21 +08:00
}
}
2026-04-16 14:03:05 +08:00
.license-note {
padding: 1.6rem;
column-gap: 1.6rem;
background: #f7f7f7;
border-radius: 0.8rem;
2026-04-16 14:03:05 +08:00
.info-icon {
width: 2.4rem;
height: 2.4rem;
flex-shrink: 0;
}
.note-copy {
font-size: 1.2rem;
line-height: 1.5;
color: #000;
font-weight: 400;
font-style: medium;
2026-04-16 14:03:05 +08:00
a {
color: #0080ed;
text-decoration: underline;
margin-left: 0.4rem;
}
2026-04-16 14:03:05 +08:00
}
}
.page-control {
justify-content: flex-end;
margin-top: 4rem;
}
2026-04-16 14:03:05 +08:00
}
2026-04-09 14:23:52 +08:00
}
</style>