123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221 |
- <!-- 使用 type="home" 属性设置首页,其他页面不需要设置,默认为page;推荐使用json5,更强大,且允许注释 -->
- <route lang="json5" type="home">
- {
- layout: 'tabbar',
- style: {
- navigationStyle: 'custom',
- navigationBarTitleText: '首页',
- },
- }
- </route>
- <script lang="ts" setup>
- import Card from '@/components/card.vue'
- import HotActivity from '@/components/hot-activity.vue'
- import MomentItem from '@/components/moment-item.vue'
- import HomeBanner from './components/home-banner.vue'
- import useRequest from '../../hooks/useRequest'
- import Menus from './components/menus.vue'
- import {
- getActivities,
- getCircles,
- getMyStudyTours,
- getSetIndexConfigs,
- getStudyTours,
- shareCircle,
- updateSetIndexConfig,
- } from '../../core/libs/requests'
- import { logo } from '../../core/libs/svgs'
- import { ComponentExposed } from 'vue-component-type-helpers'
- import { usePermissions } from '../../composables/permissions'
- import { storeToRefs } from 'pinia'
- import { messages } from '../../core/libs/messages'
- import { handleUpvoteClick, handleShareClick } from '../../core/libs/actions'
- import { useUserStore } from '../../store'
- import ScheduleCard from './components/schedule-card.vue'
- import dayjs from 'dayjs'
- import { pick, sort } from 'radash'
- import { Activity, StudyTour } from '../../core/libs/models'
- import PageHelperEvo from '@/components/page-helper-evo.vue'
- import { useMessage } from 'wot-design-uni'
- import { useHonorDialog } from '../../composables/honor-dialog'
- defineOptions({
- name: 'Home',
- })
- useMessage()
- const { show } = useHonorDialog()
- const userStore = useUserStore()
- const { userInfo } = storeToRefs(userStore)
- const { features, isLogined } = usePermissions()
- const pageHelperRef = ref<ComponentExposed<typeof PageHelperEvo>>()
- const { data: indexConfigsData, run: setIndexConfigsData } = useRequest(
- () => getSetIndexConfigs(),
- { initialData: { list: [] } },
- )
- const { data: studyTours, run: setStudyTours } = useRequest(() => getMyStudyTours(), {
- initialData: [],
- })
- const swiperData = ref<{ data: any }[]>()
- const swiperCurrent = ref(0)
- const autoplay = ref(true)
- const homeBannerRef = ref<ComponentExposed<typeof HomeBanner>[]>()
- const hotActivities =
- ref<{ type: 'studyTour' | 'activity'; data: StudyTour & Activity; startAt: string | number }[]>()
- const currentStudyTour = computed(() =>
- studyTours.value.find(
- (it) => dayjs(it.studyStartTime).isBefore(dayjs()) && dayjs(it.studyEndTime).isAfter(dayjs()),
- ),
- )
- const toAbout = () => {
- uni.navigateTo({ url: '/pages/home/about/index' })
- }
- const handleSwiperChange = ({ detail: { current, source } }) => {
- // console.log('current', current)
- console.log(current, source, swiperCurrent.value)
- homeBannerRef.value?.[swiperCurrent.value]?.videoContext.pause()
- swiperCurrent.value = current
- }
- const handleLike = async (options) => {
- await handleUpvoteClick({
- ...options,
- userId: userInfo.value.userId,
- userName: userInfo.value.nickname,
- })
- await pageHelperRef.value?.refresh()
- }
- const setHotActivities = async () => {
- const res = await Promise.all([
- getStudyTours({ headRecommend: 1 }).then((res) =>
- res.data.list.map((it) => ({ type: 'studyTour', data: it, startAt: it.applyStartTime })),
- ),
- getActivities({ headRecommend: 1 }).then((res) =>
- res.data.list.map((it) => ({ type: 'activity', data: it, startAt: it.applyStartTime })),
- ),
- ])
- console.log(sort(res.flat(), (it) => it.startAt))
- hotActivities.value = sort(res.flat(), (it) => it.startAt) as any
- }
- const handlePlay = async (id) => {
- const body = pick(swiperData.value?.find((it) => it.data.id === id).data, ['id', 'status'])
- autoplay.value = false
- await updateSetIndexConfig(body)
- }
- onShow(async () => {
- await pageHelperRef.value?.reload()
- const reqs = [setHotActivities()]
- if (isLogined.value) {
- reqs.push(setStudyTours())
- }
- await Promise.all(reqs)
- })
- onLoad(async () => {
- await Promise.all([setIndexConfigsData()])
- swiperData.value = indexConfigsData.value.list.map((it) => ({
- data: it,
- }))
- show.value({ title: '看不见', content: '看不见', path: '', image: '' })
- })
- onHide(() => {
- // autoplay.value = true
- homeBannerRef.value?.[swiperCurrent.value]?.videoContext.pause()
- })
- onShareAppMessage(async ({ from, target }) => {
- console.log('from', from)
- console.log('target', target)
- const res: Page.CustomShareContent = {}
- if (from === 'button') {
- await shareCircle(target.dataset.options.id)
- res.path = `/pages/home/moment/index?id=${target.dataset.options.id}&isShared=true`
- res.imageUrl = target.dataset.options.bannerUrls[0]
- res.title = `${target.dataset.options.stylistName}: ${target.dataset.options.circleDesc}`
- }
- if (from === 'menu') {
- res.title = messages.home.shareTitle
- }
- return res
- })
- </script>
- <template>
- <view class="">
- <view class="bg-black w-full relative aspect-[1.26/1]">
- <swiper :autoplay="autoplay" @change="handleSwiperChange">
- <template
- v-for="{ data: { id, coverVideoImage, indexPromotionalVideoImage } } of swiperData"
- :key="id"
- >
- <swiper-item>
- <HomeBanner
- ref="homeBannerRef"
- :id="id"
- :url="indexPromotionalVideoImage"
- :cover="coverVideoImage"
- @play="handlePlay"
- @ended="autoplay = true"
- />
- </swiper-item>
- </template>
- </swiper>
- </view>
- <view class="bg-[#f6f6f6] relative bottom-4 rounded-t-2xl py-1">
- <template v-if="currentStudyTour">
- <ScheduleCard
- custom-class="my-6 mx-3.5"
- :items="currentStudyTour.studyTravelDOList"
- ></ScheduleCard>
- </template>
- <!-- <SectionHeading title="按钮调试"></SectionHeading> -->
- <!-- <div class="flex">
- <TiltedButton size="large" custom-class="w-30"></TiltedButton>
- <ButtonEvo>神奇按钮</ButtonEvo>
- <ImgBtnEvo></ImgBtnEvo>
- </div> -->
- <!-- <div class="flex flex-col gap-1">
- <button>咲</button>
- <ButtonEvo custom-class="w-auto!">咲</ButtonEvo>
- <ButtonEvo custom-class="">遥星</ButtonEvo>
- <ButtonEvo custom-class="">血不染</ButtonEvo>
- <ButtonEvo custom-class="">天地不容</ButtonEvo>
- <ButtonEvo custom-class="">飒沓如流星</ButtonEvo>
- <ButtonEvo custom-class="">君有才能纵捭阖</ButtonEvo>
- <div class="w-full">
- <ButtonEvo custom-class="" block>诗仙纵横,刀剑茫茫去不还</ButtonEvo>
- </div>
- <ButtonEvo color="red" size="lg">敌无命</ButtonEvo>
- </div> -->
- <menus></menus>
- <view v-if="hotActivities?.length" class="my-6 mx-3.5">
- <HotActivity :items="hotActivities"></HotActivity>
- </view>
- <view v-if="features.about" class="my-6 mx-3.5" @click="toAbout()">
- <Card>
- <div class="flex items-center gap-2">
- <wd-img width="28" height="28" :src="logo"></wd-img>
- <div class="text-black/90 text-base font-normal font-['PingFang_SC'] leading-[10.18px]">
- 1分钟快速了解筑巢荟
- </div>
- <div class="flex-1"></div>
- <wd-icon name="help-circle" size="22px" custom-class="text-black/60"></wd-icon>
- </div>
- </Card>
- </view>
- <view class="mx-3.5 text-5 font-400">设计圈</view>
- <view class="mx-3.5">
- <PageHelperEvo ref="pageHelperRef" :request="getCircles" class="">
- <template #default="{ source }">
- <template v-for="it of source.list" :key="it.id">
- <view class="my-3">
- <MomentItem :options="it" @like="handleLike"></MomentItem>
- </view>
- </template>
- </template>
- </PageHelperEvo>
- </view>
- </view>
- </view>
- </template>
- <style></style>
|