generated from duanshuwen/webapp-vue-frontend
346 lines
9.1 KiB
Vue
346 lines
9.1 KiB
Vue
<template>
|
|
<div class="app-container">
|
|
<transition name="fade" mode="out-in">
|
|
<img :key="activeScene.bg" :src="activeScene.bg" class="bg-image" />
|
|
</transition>
|
|
|
|
<div class="gradient-overlay"></div>
|
|
|
|
<TopNavBar title="AI生成合影" color="white" :showHistory="true" @back="onBack" @history="onHistory" />
|
|
|
|
<div class="content-layer">
|
|
<div class="style-tabs">
|
|
<div v-for="tab in styles" :key="tab.id" :class="['tab-item', { active: activeStyleId === tab.id }]"
|
|
@click="handleStyleChange(tab.id)">
|
|
{{ tab.name }}
|
|
</div>
|
|
</div>
|
|
|
|
<div class="scene-list-container">
|
|
<div v-for="(item, index) in currentScenes" :key="`${activeStyleId}-${index}`"
|
|
:class="['scene-card', { active: activeSceneIndex === index }]" @click="activeSceneIndex = index">
|
|
<img :src="item.thumb" class="thumb-img" />
|
|
<div class="scene-name">{{ item.name }}</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="active-title">{{ activeScene.name }}</div>
|
|
|
|
<div class="quota-text">✨ 今日还可生成 <span>2</span> 张</div>
|
|
|
|
<div class="footer-section">
|
|
<button class="generate-btn" @click="generateAction">立即生成</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- 协议提示 -->
|
|
<van-popup v-model:show="showAgree" round :close-on-click-overlay="false"
|
|
:style="{ padding: '30px 24px', width: '80%' }">
|
|
<AgreementTip @cancel="showAgree = false" @confirm="onAgree" @view-rule="onViewRule" />
|
|
</van-popup>
|
|
|
|
<!-- 选图说明 -->
|
|
<van-popup v-model:show="showGuide" round position="bottom" closeable close-icon-position="top-left">
|
|
<PhotoGuide @start="onStartSelect" />
|
|
</van-popup>
|
|
|
|
</template>
|
|
|
|
<script setup>
|
|
import { ref, computed, onMounted } from 'vue';
|
|
import { useRouter } from 'vue-router';
|
|
import TopNavBar from '../components/TopNavBar.vue';
|
|
import PhotoGuide from '../components/PhotoGuide.vue';
|
|
import AgreementTip from '../components/AgreementTip.vue';
|
|
import { getAigcSceneList } from '@api';
|
|
|
|
// --- 测试数据 ---
|
|
const styles = [
|
|
{ id: 'real', name: '真实风格' },
|
|
{ id: 'comic', name: '漫画风格' }
|
|
];
|
|
|
|
const mockData = {
|
|
real: [
|
|
{ name: '无边戏水', thumb: 'https://pic.rmb.bdstatic.com/bjh/news/1660587f950922830e980e276b00a912.jpeg', bg: 'https://pic.rmb.bdstatic.com/bjh/news/1660587f950922830e980e276b00a912.jpeg' },
|
|
{ name: '花瓣赏樱', thumb: 'https://pic.rmb.bdstatic.com/bjh/a5138770aac4303a36122ba78b9bb0243471.jpeg@h_1280', bg: 'https://pic.rmb.bdstatic.com/bjh/a5138770aac4303a36122ba78b9bb0243471.jpeg@h_1280' },
|
|
{ name: '洞穴泡池', thumb: 'https://img2.baidu.com/it/u=2049087723,586351684&fm=253&app=138&f=JPEG?w=800&h=1400', bg: 'https://img2.baidu.com/it/u=2049087723,586351684&fm=253&app=138&f=JPEG?w=800&h=1400' },
|
|
{ name: '萌鸡小队', thumb: 'https://img2.baidu.com/it/u=2558178253,1119347742&fm=253&app=138&f=JPEG?w=800&h=1400', bg: 'https://img2.baidu.com/it/u=2558178253,1119347742&fm=253&app=138&f=JPEG?w=800&h=1400' },
|
|
{ name: '图腾戏水', thumb: 'https://img2.baidu.com/it/u=3580358401,288479606&fm=253&app=138&f=JPEG?w=800&h=1400', bg: 'https://img2.baidu.com/it/u=3580358401,288479606&fm=253&app=138&f=JPEG?w=800&h=1400' },
|
|
],
|
|
comic: [
|
|
{ name: '二次元泳池', thumb: 'https://img2.baidu.com/it/u=3357105687,2201196310&fm=253&app=138&f=JPEG?w=800&h=1400', bg: 'https://img2.baidu.com/it/u=3357105687,2201196310&fm=253&app=138&f=JPEG?w=800&h=1400' },
|
|
{ name: '赛博樱花', thumb: 'https://img0.baidu.com/it/u=790408018,2989683687&fm=253&app=138&f=JPEG?w=800&h=1400', bg: 'https://img0.baidu.com/it/u=790408018,2989683687&fm=253&app=138&f=JPEG?w=800&h=1400' },
|
|
{ name: '梦幻山洞', thumb: 'https://img2.baidu.com/it/u=2720967234,3522597250&fm=253&app=138&f=JPEG?w=800&h=1400', bg: 'https://img2.baidu.com/it/u=2720967234,3522597250&fm=253&app=138&f=JPEG?w=800&h=1400' },
|
|
{ name: '卡通乐园', thumb: 'https://img1.baidu.com/it/u=3958396600,3215281876&fm=253&app=138&f=JPEG?w=500&h=889', bg: 'https://img1.baidu.com/it/u=3958396600,3215281876&fm=253&app=138&f=JPEG?w=500&h=889' },
|
|
]
|
|
};
|
|
|
|
const router = useRouter();
|
|
|
|
// --- 状态 ---
|
|
const activeStyleId = ref('real');
|
|
const activeSceneIndex = ref(0);
|
|
const showGuide = ref(false);
|
|
|
|
const showAgree = ref(true);
|
|
|
|
const onAgree = () => {
|
|
console.log("用户同意了协议");
|
|
showAgree.value = false;
|
|
// 执行下一步选图逻辑
|
|
};
|
|
|
|
const onViewRule = () => {
|
|
console.log("跳转到规则详情页");
|
|
// 可以是打开另一个 popup 或者 router.push
|
|
};
|
|
|
|
|
|
const onBack = () => {
|
|
router.back();
|
|
};
|
|
|
|
const onHistory = () => {
|
|
router.push('/history');
|
|
};
|
|
|
|
const generateAction = () => {
|
|
showGuide.value = true;
|
|
}
|
|
|
|
const onStartSelect = () => {
|
|
console.log("用户已阅读说明,开始打开相册逻辑...");
|
|
showGuide.value = false;
|
|
// 这里写调用系统相册的代码
|
|
router.push('/generate');
|
|
};
|
|
|
|
|
|
// --- 计算属性 ---
|
|
const currentScenes = computed(() => mockData[activeStyleId.value] || []);
|
|
const activeScene = computed(() => currentScenes.value[activeSceneIndex.value] || {});
|
|
|
|
// --- 方法 ---
|
|
const handleStyleChange = (id) => {
|
|
activeStyleId.value = id;
|
|
activeSceneIndex.value = 0;
|
|
};
|
|
|
|
// 数据请求
|
|
const fetchSceneList = async () => {
|
|
// 这里可以替换成真实的 API 请求
|
|
const response = await getAigcSceneList();
|
|
// mockData[activeStyleId.value] = response.data.scenes;
|
|
};
|
|
|
|
onMounted(() => {
|
|
fetchSceneList();
|
|
});
|
|
|
|
</script>
|
|
|
|
<style scoped>
|
|
.app-container {
|
|
position: relative;
|
|
width: 100vw;
|
|
height: 100vh;
|
|
background-color: #000;
|
|
overflow: hidden;
|
|
color: #fff;
|
|
font-family: sans-serif;
|
|
}
|
|
|
|
/* 背景图片 */
|
|
.bg-image {
|
|
position: absolute;
|
|
inset: 0;
|
|
width: 100%;
|
|
height: 60%;
|
|
object-fit: cover;
|
|
z-index: 0;
|
|
}
|
|
|
|
/* 黑色渐变浮层 */
|
|
.gradient-overlay {
|
|
position: absolute;
|
|
inset: 0;
|
|
height: 60%;
|
|
z-index: 1;
|
|
background: linear-gradient(to bottom, rgba(0, 0, 0, 0) 0%, rgba(0, 0, 0, 0.1) 70%, rgba(0, 0, 0, 0.7) 85%, #000 100%);
|
|
}
|
|
|
|
|
|
|
|
/* 内容布局层 */
|
|
.content-layer {
|
|
position: relative;
|
|
z-index: 2;
|
|
height: 100%;
|
|
display: flex;
|
|
flex-direction: column;
|
|
justify-content: flex-end;
|
|
}
|
|
|
|
/* 风格切换 Tab */
|
|
.style-tabs {
|
|
display: flex;
|
|
background: rgba(255, 255, 255, 0.2);
|
|
backdrop-filter: blur(15px);
|
|
margin: 0 auto 16px;
|
|
border-radius: 30px;
|
|
padding: 4px;
|
|
}
|
|
|
|
.tab-item {
|
|
padding: 8px 22px;
|
|
border-radius: 26px;
|
|
font-size: 14px;
|
|
color: #ddd;
|
|
transition: all 0.2s;
|
|
cursor: pointer;
|
|
}
|
|
|
|
.tab-item.active {
|
|
background: #fff;
|
|
color: #000;
|
|
font-weight: bold;
|
|
}
|
|
|
|
/* 横向场景列表 */
|
|
.scene-list-container {
|
|
display: flex;
|
|
overflow-x: auto;
|
|
padding: 20px;
|
|
gap: 16px;
|
|
scrollbar-width: none;
|
|
align-items: flex-end;
|
|
}
|
|
|
|
.scene-list-container::-webkit-scrollbar {
|
|
display: none;
|
|
}
|
|
|
|
/* Item 基础样式 */
|
|
.scene-card {
|
|
flex: 0 0 90px;
|
|
position: relative;
|
|
height: 120px;
|
|
border-radius: 14px;
|
|
transition: all 0.3s cubic-bezier(0.34, 1.56, 0.64, 1);
|
|
background: #1a1a1a;
|
|
overflow: visible;
|
|
/* 必须 visible 才能让三角形溢出 */
|
|
}
|
|
|
|
.thumb-img {
|
|
width: 100%;
|
|
height: 100%;
|
|
object-fit: cover;
|
|
border-radius: 12px;
|
|
opacity: 0.5;
|
|
display: block;
|
|
}
|
|
|
|
/* 3. 选中时放大并显示 border */
|
|
.scene-card.active {
|
|
border: 2px solid #fff;
|
|
transform: scale(1.15);
|
|
z-index: 5;
|
|
}
|
|
|
|
.scene-card.active .thumb-img {
|
|
opacity: 1;
|
|
}
|
|
|
|
/* 2. 选中时的倒三角:无间隔紧贴 border */
|
|
.scene-card.active::before {
|
|
content: '';
|
|
position: absolute;
|
|
/* top = -(三角形高度) 确保严丝合缝 */
|
|
top: -8px;
|
|
left: 50%;
|
|
transform: translateX(-50%);
|
|
width: 0;
|
|
height: 0;
|
|
border-left: 7px solid transparent;
|
|
border-right: 7px solid transparent;
|
|
border-bottom: 8px solid #fff;
|
|
}
|
|
|
|
/* item 底部文字半透明背景 */
|
|
.scene-name {
|
|
position: absolute;
|
|
bottom: 0;
|
|
left: 0;
|
|
width: 100%;
|
|
background: rgba(0, 0, 0, 0.5);
|
|
color: #fff;
|
|
font-size: 11px;
|
|
padding: 4px 0;
|
|
text-align: center;
|
|
border-bottom-left-radius: 12px;
|
|
border-bottom-right-radius: 12px;
|
|
backdrop-filter: blur(2px);
|
|
}
|
|
|
|
/* 大标题样式 */
|
|
.active-title {
|
|
text-align: center;
|
|
font-size: 44px;
|
|
margin: 12px 0 32px;
|
|
background: linear-gradient(to bottom, #FFFFFF 0%, #B9E9FF 100%);
|
|
-webkit-background-clip: text;
|
|
-webkit-text-fill-color: transparent;
|
|
font-weight: 900;
|
|
letter-spacing: 2px;
|
|
font-style: italic;
|
|
}
|
|
|
|
/* 底部操作区 */
|
|
.footer-section {
|
|
padding: 0 23px 32px;
|
|
text-align: center;
|
|
}
|
|
|
|
.quota-text {
|
|
text-align: center;
|
|
font-size: 13px;
|
|
color: #bbb;
|
|
padding: 12px;
|
|
margin: 0 102px 15px;
|
|
background: rgba(255, 255, 255, 0.1);
|
|
border-radius: 999px 999px 999px 999px;
|
|
border: 1px solid rgba(255, 255, 255, 0.2);
|
|
}
|
|
|
|
.quota-text span {
|
|
color: #5ef9ff;
|
|
font-weight: bold;
|
|
}
|
|
|
|
.generate-btn {
|
|
width: 100%;
|
|
height: 60px;
|
|
border-radius: 30px;
|
|
border: none;
|
|
background: linear-gradient(90deg, #D9FBC8 0%, #8BE2FF 100%);
|
|
box-shadow: 0px 0px 20px 0px rgba(217, 251, 200, 0.3);
|
|
font-weight: bold;
|
|
font-size: 20px;
|
|
color: #000;
|
|
cursor: pointer;
|
|
}
|
|
|
|
/* 背景切换过渡动画 */
|
|
.fade-enter-active,
|
|
.fade-leave-active {
|
|
transition: opacity 0.5s ease;
|
|
}
|
|
|
|
.fade-enter-from,
|
|
.fade-leave-to {
|
|
opacity: 0;
|
|
}
|
|
</style> |