浏览代码

Merge branch 'iteration-20231012'

lex 1 年之前
父节点
当前提交
fd01d0712d

+ 53 - 23
src/tenant/exercise-record/exercis-detail.tsx

@@ -125,12 +125,40 @@ export default defineComponent({
         membershipEndTime: users?.membershipEndTime
       }
     })
+
+    // 初始化周一
+    const formatWeekDays = (time?: any) => {
+      const week = dayjs(time).day()
+      let startTime = ''
+      let endTime = ''
+      if (week === 0) {
+        // 星期天
+        startTime = dayjs(time).subtract(6, 'day').format('YYYY-MM-DD')
+        endTime = dayjs(time).format('YYYY-MM-DD')
+      } else if (week === 1) {
+        // 星期一
+        startTime = dayjs(time).format('YYYY-MM-DD')
+        endTime = dayjs(time).add(6, 'day').format('YYYY-MM-DD')
+      } else {
+        startTime = dayjs(time)
+          .subtract(week - 1, 'day')
+          .format('YYYY-MM-DD')
+        endTime = dayjs(time)
+          .add(7 - week, 'day')
+          .format('YYYY-MM-DD')
+      }
+      return {
+        startTime,
+        endTime
+      }
+    }
+
     const forms = reactive({
-      startTime: dayjs().day(1).format('YYYY-MM-DD'),
-      endTime: dayjs().day(7).format('YYYY-MM-DD'),
+      ...formatWeekDays(),
       page: 1,
       rows: 20
     })
+
     const refreshing = ref(false)
     const loading = ref(false)
     const finished = ref(false)
@@ -400,27 +428,29 @@ export default defineComponent({
                 dayjs(forms.endTime)
               )
             ) {
-              const week = dayjs(item[0]).day()
-              if (week === 0) {
-                // 星期天
-                forms.startTime = dayjs(item[0])
-                  .subtract(6, 'day')
-                  .format('YYYY-MM-DD')
-                forms.endTime = dayjs(item[0]).format('YYYY-MM-DD')
-              } else if (week === 1) {
-                // 星期一
-                forms.startTime = dayjs(item[0]).format('YYYY-MM-DD')
-                forms.endTime = dayjs(item[0])
-                  .add(6, 'day')
-                  .format('YYYY-MM-DD')
-              } else {
-                forms.startTime = dayjs(item[0])
-                  .subtract(week - 1, 'day')
-                  .format('YYYY-MM-DD')
-                forms.endTime = dayjs(item[0])
-                  .add(7 - week, 'day')
-                  .format('YYYY-MM-DD')
-              }
+              const times = formatWeekDays(item[0])
+              forms.startTime = times.startTime
+              forms.endTime = times.endTime
+              // if (week === 0) {
+              //   // 星期天
+              //   forms.startTime = dayjs(item[0])
+              //     .subtract(6, 'day')
+              //     .format('YYYY-MM-DD')
+              //   forms.endTime = dayjs(item[0]).format('YYYY-MM-DD')
+              // } else if (week === 1) {
+              //   // 星期一
+              //   forms.startTime = dayjs(item[0]).format('YYYY-MM-DD')
+              //   forms.endTime = dayjs(item[0])
+              //     .add(6, 'day')
+              //     .format('YYYY-MM-DD')
+              // } else {
+              //   forms.startTime = dayjs(item[0])
+              //     .subtract(week - 1, 'day')
+              //     .format('YYYY-MM-DD')
+              //   forms.endTime = dayjs(item[0])
+              //     .add(7 - week, 'day')
+              //     .format('YYYY-MM-DD')
+              // }
             }
             state.showPopoverTime = false
             refreshing.value = true

+ 1 - 0
src/tenant/member-center/index.module.less

@@ -149,6 +149,7 @@
     padding-left: 26px;
     font-size: 13px;
     color: #742626;
+    position: relative;
 
     &::after {
       content: ' ';

+ 3 - 1
src/tenant/member-center/index.tsx

@@ -280,8 +280,10 @@ export default defineComponent({
                 </span>
               </div>
+            ) : !this.userInfo.isVip && this.userInfo.membershipEndTime ? (
+              <div>您的会员已过期,续费后会员权益可继续使用</div>
             ) : (
-              <div>亲,您还不是平台会员哦</div>
+              <div>亲,您还不是会员哦~</div>
             )}
           </div>
         </div>

二进制
src/tenant/ranking-list/images/day-bg.png


二进制
src/tenant/ranking-list/images/day-title.png


二进制
src/tenant/ranking-list/images/empty-logo.png


二进制
src/tenant/ranking-list/images/icon-arrow.png


二进制
src/tenant/ranking-list/images/level-1-bg.png


二进制
src/tenant/ranking-list/images/level-1-header.png


二进制
src/tenant/ranking-list/images/level-2-bg.png


二进制
src/tenant/ranking-list/images/level-2-header.png


二进制
src/tenant/ranking-list/images/level-3-bg.png


二进制
src/tenant/ranking-list/images/level-3-header.png


二进制
src/tenant/ranking-list/images/time-bg.png


二进制
src/tenant/ranking-list/images/time-title.png


+ 360 - 5
src/tenant/ranking-list/index.module.less

@@ -1,20 +1,375 @@
+.rankListPage {
+  background: linear-gradient(270deg, #FBE3F6 0%, #FFDFD0 100%);
+  min-height: 100vh;
+
+  :global {
+    .van-tabs__nav {
+      background-color: transparent !important;
+      justify-content: center;
+    }
+
+    .van-tab {
+      font-size: 16px;
+      font-weight: bold;
+      color: rgba(0, 0, 0, 0.5);
+      line-height: 22px;
+    }
+
+    .van-tab--active {
+      color: rgba(19, 20, 21, 1);
+    }
+
+    .van-tabs__line {
+      background: #2B2A2F;
+      border-radius: 2px !important;
+      bottom: 20px;
+      width: 22px;
+    }
+
+    .van-tab--shrink {
+      padding: 0 25px;
+    }
+
+    .van-calendar__day--end,
+    .van-calendar__day--start,
+    .van-calendar__day--start-end,
+    .van-calendar__day--multiple-middle,
+    .van-calendar__day--multiple-selected {
+      background: #FF5074;
+    }
+
+    .van-overlay,
+    .van-popup {
+      z-index: 99999999 !important;
+    }
+  }
+
+  .timeRange {
+    display: flex;
+    align-items: center;
+    font-size: 14px;
+    color: rgba(19, 20, 21, 0.6);
+
+    .iconArrow {
+      display: inline-block;
+      width: 9px;
+      height: 5px;
+      margin-left: 3px;
+      background: url('./images/icon-arrow.png') no-repeat center;
+      background-size: contain;
+    }
+  }
+}
+
 .bgImg {
   position: fixed;
   top: 0;
   left: 0;
   width: 100%;
-  height: 214px;
+  height: 397px;
   // object-fit: cover;
-  z-index: -1;
+  z-index: 0;
 }
 
 .sticky {
   :global {
     .van-sticky {
-      background: url('../images/bg-image.png') no-repeat top center;
-      background-size: 100% 214px;
+      transition: none;
+      background: url('./images/time-bg.png') no-repeat top center;
+      background-size: 100% 397px;
+    }
+  }
+
+  .backArrow {
+    position: absolute;
+    z-index: 10;
+    left: 13px;
+    top: 14px;
+    display: inline-block;
+    font: 0.37333rem/1 vant-icon;
+    font-size: inherit;
+    text-rendering: auto;
+    -webkit-font-smoothing: antialiased;
+    margin-right: var(--van-padding-base);
+    font-size: var(--van-nav-bar-arrow-size);
+  }
+}
+
+.rankContainer {
+  position: relative;
+}
+
+.rankLevel {
+  padding-top: 10px;
+  display: flex;
+  align-items: center;
+  justify-content: space-between;
+  margin: 0 13px;
+
+  .levelItem {
+    width: 112px;
+    text-align: center;
+
+    &::after {
+      content: ' ';
+      display: block;
+      width: 112px;
+      height: 124px;
+    }
+
+    .levelUserImg {
+      position: relative;
+      width: 86px;
+      height: 77px;
+      margin: 0 auto;
+      display: flex;
+      align-items: center;
+      justify-content: center;
+
+      &::after {
+        content: '';
+        position: absolute;
+        left: 0;
+        right: 0;
+        top: 0;
+        bottom: 0;
+        z-index: 1;
+      }
+
+      .img {
+        width: 50px;
+        height: 50px;
+        border-radius: 50%;
+        overflow: hidden;
+      }
+    }
+
+    h2 {
+      padding-top: 5px;
+      font-size: 16px;
+      font-weight: 600;
+      color: #131415;
+      line-height: 22px;
+    }
+
+    .levelTime {
+      font-size: 12px;
+      line-height: 17px;
+      padding: 2px 0 8px;
+      color: rgba(0, 0, 0, 0.6);
+    }
+  }
+
+  .level1 {
+    &::after {
+      height: 142px;
+      background: url('./images/level-1-bg.png') no-repeat center;
+      background-size: contain;
+    }
+
+    .levelUserImg {
+      width: 92px !important;
+      height: 88px !important;
+
+      &::after {
+        background: url('./images/level-1-header.png') no-repeat center;
+        background-size: contain;
+      }
+    }
+
+    .img {
+      width: 58px !important;
+      height: 58px !important;
+    }
+  }
+
+  .level2 {
+    margin-top: 33px;
+
+    &::after {
+      background: url('./images/level-2-bg.png') no-repeat center;
+      background-size: contain;
+    }
+
+    .levelUserImg {
+      &::after {
+        background: url('./images/level-2-header.png') no-repeat center;
+        background-size: contain;
+      }
     }
   }
 
-  // --van-nav-bar-height: 0px;
+  .level3 {
+    margin-top: 30px;
+
+    &::after {
+      height: 118px;
+      background: url('./images/level-3-bg.png') no-repeat center;
+      background-size: contain;
+    }
+
+    .levelUserImg {
+      &::after {
+        background: url('./images/level-3-header.png') no-repeat center;
+        background-size: contain;
+      }
+    }
+  }
+}
+
+.rankList {
+  background: rgba(255, 255, 255, 0.7);
+  border-radius: 20px 20px 0px 0px;
+  border: 2px solid #FFFFFF;
+  backdrop-filter: blur(6px);
+  min-height: 40vh;
+  margin: -62px 13px 0;
+
+  .rankTitle {
+    display: flex;
+    align-items: center;
+    justify-content: space-between;
+    padding: 19px 12px 4px 0;
+
+    .titleName {
+      display: flex;
+      align-items: center;
+
+      &::before {
+        content: '';
+        display: flex;
+        width: 4px;
+        height: 14px;
+        background-color: #FFA2A0;
+        border-radius: 0px 3px 3px 0px;
+        margin-right: 8px;
+      }
+
+      img {
+        width: 80px;
+        height: 16px;
+      }
+    }
+  }
+}
+
+.rankContainer {
+
+  .rankItem,
+  .selfInfo {
+    position: relative;
+    background-color: transparent;
+    padding: 6px 10px 6px 30px;
+    border-radius: 12px;
+    margin: 0 6px 4px;
+    width: auto;
+
+
+
+    &.active {
+      background-color: #FFE5EA;
+    }
+
+
+    .num {
+      position: absolute;
+      left: 8px;
+      top: 20px;
+      font-size: 20px;
+      font-family: DINAlternate-Bold, DINAlternate;
+      font-weight: bold;
+      color: #AAAAAA;
+      line-height: 24px;
+    }
+
+    .userImg {
+      width: 48px;
+      height: 48px;
+      border-radius: 50%;
+      overflow: hidden;
+      margin-right: 12px;
+      flex-shrink: 0;
+    }
+
+    .userName {
+      font-size: 16px;
+      font-weight: 600;
+      color: #131415;
+      line-height: 22px;
+    }
+
+    .subjectName {
+      font-size: 12px;
+      color: #777777;
+      line-height: 17px;
+    }
+
+    .times {
+      font-size: 14px;
+      color: #777777;
+      line-height: 20px;
+
+      span {
+        font-size: 18px;
+        font-family: DINAlternate-Bold, DINAlternate;
+        font-weight: bold;
+        color: #333333;
+        line-height: 21px;
+        padding-right: 4px;
+      }
+    }
+  }
+
+  .selfInfo {
+    background: #FFFFFF;
+    box-shadow: 0px 1px 12px 0px rgba(240, 192, 221, 0.43);
+    border-radius: 20px 20px 0px 0px;
+    margin: 0;
+    padding: 14px 25px calc(14px + env(safe-area-inset-bottom)) 56px;
+
+    .num {
+      position: absolute;
+      left: 30px;
+      top: 26px;
+    }
+  }
+}
+
+.rankListDay {
+  background: linear-gradient(180deg, #C1F2F9 0%, #D3FFFF 29%, #FFEBCE 100%);
+
+
+  .sticky {
+    :global {
+      .van-sticky {
+        background: url('./images/day-bg.png') no-repeat top center #C1F2F9;
+        background-size: 100% 397px;
+      }
+    }
+  }
+
+  .rankItem,
+  .selfInfo {
+    &.active {
+      background-color: rgba(228, 247, 247, 1);
+    }
+  }
+
+  .selfInfo {
+    box-shadow: 0px 1px 12px 0px rgba(236, 219, 196, 0.34);
+
+    .num {
+      position: absolute;
+      left: 30px;
+      top: 26px;
+    }
+  }
+}
+
+.emptyResult {
+  position: absolute;
+  width: 100%;
+  height: calc(100vh - var(--header-height));
+  display: flex;
+  align-items: center;
 }

+ 358 - 14
src/tenant/ranking-list/index.tsx

@@ -1,14 +1,121 @@
 import TheSticky from '@/components/the-sticky'
-import { defineComponent } from 'vue'
+import { computed, defineComponent, onMounted, reactive } from 'vue'
+import { useRoute } from 'vue-router'
 import styles from './index.module.less'
 import ColHeader from '@/components/col-header'
-import bgImg from '../images/bg-image.png'
-
+import bgImg from './images/time-bg.png'
+import bgImg2 from './images/day-bg.png'
+import timeTitle from './images/time-title.png'
+import dayTitle from './images/day-title.png'
+import { Tabs, Tab, Image, Icon, Calendar, Cell } from 'vant'
+import dayjs from 'dayjs'
+import isBetween from 'dayjs/plugin/isBetween'
+dayjs.extend(isBetween)
+import request from '@/helpers/request'
+import { state as baseState } from '@/state'
+import studentLogo from '@/common/images/icon_student.png'
+import emptyLogo from './images/empty-logo.png'
+import ColResult from '@/components/col-result'
 export default defineComponent({
   name: 'ranking-list',
   setup() {
+    const route = useRoute()
+    const state = reactive({
+      showPopoverTime: false,
+      tabActive: 'timeBill',
+      firstInfo: {} as any,
+      secondInfo: {} as any,
+      threeInfo: {} as any,
+      myInfo: {} as any,
+      list: [] as any,
+      loading: false,
+      beforListShow: true,
+      listShow: true // 是否显示
+    })
+
+    const isDayRank = computed(() => {
+      return state.tabActive === 'dayBill'
+    })
+
+    // 初始化周一
+    const formatWeekDays = (time?: any) => {
+      const week = dayjs(time).day()
+      let startTime = ''
+      let endTime = ''
+      if (week === 0) {
+        // 星期天
+        startTime = dayjs(time).subtract(6, 'day').format('YYYY-MM-DD')
+        endTime = dayjs(time).format('YYYY-MM-DD')
+      } else if (week === 1) {
+        // 星期一
+        startTime = dayjs(time).format('YYYY-MM-DD')
+        endTime = dayjs(time).add(6, 'day').format('YYYY-MM-DD')
+      } else {
+        startTime = dayjs(time)
+          .subtract(week - 1, 'day')
+          .format('YYYY-MM-DD')
+        endTime = dayjs(time)
+          .add(7 - week, 'day')
+          .format('YYYY-MM-DD')
+      }
+      return {
+        startTime,
+        endTime
+      }
+    }
+
+    const forms = reactive({
+      ...formatWeekDays(),
+      studentIds: baseState.user.data?.userId,
+      name: 0,
+      page: 1,
+      rows: 20
+    })
+
+    const getList = async () => {
+      state.loading = true
+      try {
+        const { data } = await request.post(
+          '/api-tenant/studentMusicCompareRecord/rankingList',
+          {
+            data: {
+              ...forms
+            }
+          }
+        )
+        state.beforListShow = data.detail && data.detail.length > 0
+        state.myInfo = data.head || {}
+        state.myInfo.trainTime = state.myInfo.trainTime
+          ? Math.floor(state.myInfo.trainTime / 60)
+          : 0
+        state.list = data.detail || []
+        state.list.forEach((item: any) => {
+          item.trainTime = item.trainTime ? Math.floor(item.trainTime / 60) : 0
+        })
+        state.firstInfo =
+          state.list.length > 0 ? state.list.splice(0, 1)[0] : {}
+        state.secondInfo =
+          state.list.length > 0 ? state.list.splice(0, 1)[0] : {}
+        state.threeInfo =
+          state.list.length > 0 ? state.list.splice(0, 1)[0] : {}
+
+        state.listShow = state.list.length > 0
+      } catch {
+        //
+      }
+      state.loading = false
+    }
+
+    onMounted(() => {
+      getList()
+    })
     return () => (
-      <div class={styles.activationCode}>
+      <div
+        class={[
+          styles.rankListPage,
+          isDayRank.value ? styles.rankListDay : styles.rankListTime
+        ]}
+      >
         <div class={styles.sticky}>
           <TheSticky position="top">
             <ColHeader
@@ -17,18 +124,255 @@ export default defineComponent({
               border={false}
               title=" "
               color="#131415"
-            />
-
-            {/* <van-tabs v-model:active="active">
-              <van-tab title="标签 1">内容 1</van-tab>
-              <van-tab title="标签 2" disabled>
-                内容 2
-              </van-tab>
-              <van-tab title="标签 3">内容 3</van-tab>
-            </van-tabs> */}
+            >
+              {{
+                content: () => (
+                  <>
+                    <Icon name="arrow-left" class={styles.backArrow} />
+                    <Tabs
+                      lineWidth={22}
+                      lineHeight={4}
+                      shrink
+                      v-model:active={state.tabActive}
+                      onChange={(val: string) => {
+                        if (val === 'dayBill') {
+                          forms.name = 2
+                        } else {
+                          forms.name = 0
+                        }
+
+                        getList()
+                      }}
+                    >
+                      <Tab title="时长榜" name="timeBill"></Tab>
+                      <Tab title="天数榜" name="dayBill"></Tab>
+                    </Tabs>
+                  </>
+                )
+              }}
+            </ColHeader>
           </TheSticky>
-          <img class={styles.bgImg} src={bgImg} />
+          <img class={styles.bgImg} src={isDayRank.value ? bgImg2 : bgImg} />
         </div>
+
+        {/* {state.beforListShow && ( */}
+          <div class={styles.rankContainer}>
+            <div class={styles.rankLevel}>
+              <div class={[styles.levelItem, styles.level2]}>
+                {state.secondInfo.userId ? (
+                  <>
+                    <div class={styles.levelUserImg}>
+                      <Image
+                        class={styles.img}
+                        src={state.secondInfo.avatar || studentLogo}
+                      />
+                    </div>
+                    <h2 class={'van-ellipsis'}>
+                      {state.secondInfo.studentName}
+                    </h2>
+                    <p class={[styles.levelTime, 'van-ellipsis']}>
+                      {state.secondInfo.subjectName}-
+                      {state.secondInfo.trainTime}
+                      分钟
+                    </p>
+                  </>
+                ) : (
+                  <>
+                    <div class={styles.levelUserImg}>
+                      <Image class={styles.img} src={emptyLogo} />
+                    </div>
+                    <h2 class={'van-ellipsis'}>虚位以待</h2>
+                    <p class={[styles.levelTime, 'van-ellipsis']}>&nbsp;</p>
+                  </>
+                )}
+              </div>
+              <div class={[styles.levelItem, styles.level1]}>
+                {state.firstInfo.userId ? (
+                  <>
+                    <div class={styles.levelUserImg}>
+                      <Image
+                        class={styles.img}
+                        src={state.firstInfo.avatar || studentLogo}
+                      />
+                    </div>
+                    <h2 class={'van-ellipsis'}>{state.firstInfo.studentName}</h2>
+                    <p class={[styles.levelTime, 'van-ellipsis']}>
+                      {state.firstInfo.subjectName}-{state.firstInfo.trainTime}分钟
+                    </p>
+                  </>
+                ) : (
+                  <>
+                    <div class={styles.levelUserImg}>
+                      <Image class={styles.img} src={emptyLogo} />
+                    </div>
+                    <h2 class={'van-ellipsis'}>虚位以待</h2>
+                    <p class={[styles.levelTime, 'van-ellipsis']}>&nbsp;</p>
+                  </>
+                )}
+              </div>
+              <div class={[styles.levelItem, styles.level3]}>
+                {state.threeInfo.userId ? (
+                  <>
+                    <div class={styles.levelUserImg}>
+                      <Image
+                        class={styles.img}
+                        src={state.threeInfo.avatar || studentLogo}
+                      />
+                    </div>
+                    <h2 class={'van-ellipsis'}>
+                      {state.threeInfo.studentName}
+                    </h2>
+                    <p class={[styles.levelTime, 'van-ellipsis']}>
+                      {state.threeInfo.subjectName}-{state.threeInfo.trainTime}
+                      分钟
+                    </p>
+                  </>
+                ) : (
+                  <>
+                    <div class={styles.levelUserImg}>
+                      <Image class={styles.img} src={emptyLogo} />
+                    </div>
+                    <h2 class={'van-ellipsis'}>虚位以待</h2>
+                    <p class={[styles.levelTime, 'van-ellipsis']}>&nbsp;</p>
+                  </>
+                )}
+              </div>
+            </div>
+
+            <div class={styles.rankList}>
+              <div class={styles.rankTitle}>
+                <div class={styles.titleName}>
+                  <img src={isDayRank.value ? dayTitle : timeTitle} />
+                </div>
+                <span
+                  class={styles.timeRange}
+                  onClick={() => (state.showPopoverTime = true)}
+                >
+                  {dayjs(forms.startTime).format('YYYY-MM-DD')}至
+                  {dayjs(forms.endTime).format('YYYY-MM-DD')}
+                  <i class={styles.iconArrow}></i>
+                </span>
+              </div>
+
+              {state.list.map((item: any) => (
+                // i == 4 && styles.active
+                <Cell
+                  class={[
+                    styles.rankItem,
+                    state.myInfo.userId === item.userId ? styles.active : ''
+                  ]}
+                  border={false}
+                  center
+                >
+                  {{
+                    icon: () => (
+                      <>
+                        <span class={styles.num}>{item.rankNum}</span>
+                        <Image
+                          class={styles.userImg}
+                          src={item.avatar || studentLogo}
+                        />
+                      </>
+                    ),
+                    title: () => (
+                      <div class={styles.userInfo}>
+                        <p class={styles.userName}>{item.studentName}</p>
+                        <p class={styles.subjectName}>{item.subjectName}</p>
+                      </div>
+                    ),
+                    value: () => (
+                      <div class={styles.times}>
+                        <span>{item.trainTime}</span>分钟
+                      </div>
+                    )
+                  }}
+                </Cell>
+              ))}
+
+              {!state.loading && !state.listShow && (
+                <ColResult tips="暂无数据" btnStatus={false} />
+              )}
+            </div>
+            {state.myInfo?.userId ? (
+              <TheSticky position="bottom">
+                <Cell class={styles.selfInfo} border={false} center>
+                  {{
+                    icon: () => (
+                      <>
+                        <span class={styles.num}>
+                          {state.myInfo.rankNum == -1
+                            ? '-'
+                            : state.myInfo.rankNum}
+                        </span>
+                        <Image
+                          src={state.myInfo.avatar || studentLogo}
+                          class={styles.userImg}
+                        />
+                      </>
+                    ),
+                    title: () => (
+                      <div class={styles.userInfo}>
+                        <p class={styles.userName}>
+                          {state.myInfo.studentName}
+                        </p>
+                        <p class={styles.subjectName}>
+                          {state.myInfo.subjectName}
+                        </p>
+                      </div>
+                    ),
+                    value: () => (
+                      <div class={styles.times}>
+                        <span>{state.myInfo.trainTime}</span>分钟
+                      </div>
+                    )
+                  }}
+                </Cell>
+              </TheSticky>
+            ) : (
+              ''
+            )}
+          </div>
+        // )}
+
+        {/* {!state.loading && !state.beforListShow && (
+          <div class={styles.emptyResult}>
+            <ColResult tips="暂无数据" btnStatus={false} />
+          </div>
+        )} */}
+
+        <Calendar
+          v-model:show={state.showPopoverTime}
+          firstDayOfWeek={1}
+          showConfirm={false}
+          type="range"
+          title="周期选择"
+          maxRange={7}
+          minDate={new Date('2023-02-27')}
+          defaultDate={[
+            dayjs(forms.startTime).toDate(),
+            dayjs(forms.endTime).toDate()
+          ]}
+          style={{
+            height: '70%'
+          }}
+          onSelect={(item: any) => {
+            forms.startTime = ''
+            forms.endTime = ''
+            if (
+              !dayjs(item[0]).isBetween(
+                dayjs(forms.startTime),
+                dayjs(forms.endTime)
+              )
+            ) {
+              const times = formatWeekDays(item[0])
+              forms.startTime = times.startTime
+              forms.endTime = times.endTime
+            }
+            state.showPopoverTime = false
+
+            getList()
+          }}
+        />
       </div>
     )
   }

+ 206 - 0
src/tenant/ranking-list/list.tsx

@@ -0,0 +1,206 @@
+import { defineComponent, reactive } from 'vue'
+import styles from './index.module.less'
+import { Image, Calendar, Cell } from 'vant'
+import timeTitle from './images/time-title.png'
+import dayTitle from './images/day-title.png'
+import { useRoute } from 'vue-router'
+import dayjs from 'dayjs'
+import isBetween from 'dayjs/plugin/isBetween'
+dayjs.extend(isBetween)
+export default defineComponent({
+  name: 'rank-list',
+  setup() {
+    const route = useRoute()
+    const state = reactive({
+      showPopoverTime: false,
+      currentDate: [dayjs().format('YYYY'), dayjs().format('MM')],
+      isClick: false,
+      practiceMonthName: route.query.practiceMonthName
+        ? route.query.practiceMonthName
+        : dayjs().format('YYYY') + '年' + dayjs().format('MM') + '月',
+      userTrainOverView: {
+        trainDays: 0,
+        trainNum: 0,
+        trainTime: 0,
+        avgTrainTime: 0
+      },
+      userTrainChartData: [] as any,
+      myChart: null as any
+    })
+
+    // 初始化周一
+    const formatWeekDays = (time?: any) => {
+      const week = dayjs(time).day()
+      let startTime = ''
+      let endTime = ''
+      if (week === 0) {
+        // 星期天
+        startTime = dayjs(time).subtract(6, 'day').format('YYYY-MM-DD')
+        endTime = dayjs(time).format('YYYY-MM-DD')
+      } else if (week === 1) {
+        // 星期一
+        startTime = dayjs(time).format('YYYY-MM-DD')
+        endTime = dayjs(time).add(6, 'day').format('YYYY-MM-DD')
+      } else {
+        startTime = dayjs(time)
+          .subtract(week - 1, 'day')
+          .format('YYYY-MM-DD')
+        endTime = dayjs(time)
+          .add(7 - week, 'day')
+          .format('YYYY-MM-DD')
+      }
+      return {
+        startTime,
+        endTime
+      }
+    }
+
+    const forms = reactive({
+      ...formatWeekDays(),
+      page: 1,
+      rows: 20
+    })
+    return () => (
+      <>
+        <div class={styles.rankContainer}>
+          <div class={styles.rankLevel}>
+            <div class={[styles.levelItem, styles.level2]}>
+              <div class={styles.levelUserImg}>
+                <Image
+                  class={styles.img}
+                  src={
+                    'https://ks3-cn-beijing.ksyuncs.com/daya/16644378620311664437783089.jpg'
+                  }
+                />
+              </div>
+              <h2 class={'van-ellipsis'}>夏可可夏可可夏可可夏可可</h2>
+              <p class={[styles.levelTime, 'van-ellipsis']}>
+                夏可可夏可可竖笛-240分钟
+              </p>
+            </div>
+            <div class={[styles.levelItem, styles.level1]}>
+              <div class={styles.levelUserImg}>
+                <Image
+                  class={styles.img}
+                  src={
+                    'https://ks3-cn-beijing.ksyuncs.com/daya/16644378620311664437783089.jpg'
+                  }
+                />
+              </div>
+              <h2 class={'van-ellipsis'}>夏可可</h2>
+              <p class={[styles.levelTime, 'van-ellipsis']}>竖笛-240分钟</p>
+            </div>
+            <div class={[styles.levelItem, styles.level3]}>
+              <div class={styles.levelUserImg}>
+                <Image
+                  class={styles.img}
+                  src={
+                    'https://ks3-cn-beijing.ksyuncs.com/daya/16644378620311664437783089.jpg'
+                  }
+                />
+              </div>
+              <h2 class={'van-ellipsis'}>夏可可</h2>
+              <p class={[styles.levelTime, 'van-ellipsis']}>竖笛-240分钟</p>
+            </div>
+          </div>
+
+          <div class={styles.rankList}>
+            <div class={styles.rankTitle}>
+              <div class={styles.titleName}>
+                <img src={timeTitle} />
+              </div>
+              <span
+                class={styles.timeRange}
+                onClick={() => (state.showPopoverTime = true)}
+              >
+                {dayjs(forms.startTime).format('YYYY-MM-DD')}至
+                {dayjs(forms.endTime).format('YYYY-MM-DD')}
+                <i class={styles.iconArrow}></i>
+              </span>
+            </div>
+
+            {[1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15].map(() => (
+              <Cell class={styles.rankItem} border={false} center>
+                {{
+                  icon: () => (
+                    <>
+                      <span class={styles.num}>4</span>
+                      <Image class={styles.userImg} />
+                    </>
+                  ),
+                  title: () => (
+                    <div class={styles.userInfo}>
+                      <p class={styles.userName}>李同学</p>
+                      <p class={styles.subjectName}>竖笛</p>
+                    </div>
+                  ),
+                  value: () => (
+                    <div class={styles.times}>
+                      <span>229</span>分钟
+                    </div>
+                  )
+                }}
+              </Cell>
+            ))}
+          </div>
+
+          <TheSticky position="bottom">
+            <Cell class={styles.selfInfo} border={false} center>
+              {{
+                icon: () => (
+                  <>
+                    <span class={styles.num}>4</span>
+                    <Image class={styles.userImg} />
+                  </>
+                ),
+                title: () => (
+                  <div class={styles.userInfo}>
+                    <p class={styles.userName}>李同学</p>
+                    <p class={styles.subjectName}>竖笛</p>
+                  </div>
+                ),
+                value: () => (
+                  <div class={styles.times}>
+                    <span>229</span>分钟
+                  </div>
+                )
+              }}
+            </Cell>
+          </TheSticky>
+        </div>
+
+        <Calendar
+          v-model:show={state.showPopoverTime}
+          firstDayOfWeek={1}
+          showConfirm={false}
+          type="range"
+          title="周期选择"
+          maxRange={7}
+          minDate={new Date('2023-02-27')}
+          defaultDate={[
+            dayjs(forms.startTime).toDate(),
+            dayjs(forms.endTime).toDate()
+          ]}
+          style={{
+            height: '70%'
+          }}
+          onSelect={(item: any) => {
+            forms.startTime = ''
+            forms.endTime = ''
+            if (
+              !dayjs(item[0]).isBetween(
+                dayjs(forms.startTime),
+                dayjs(forms.endTime)
+              )
+            ) {
+              const times = formatWeekDays(item[0])
+              forms.startTime = times.startTime
+              forms.endTime = times.endTime
+            }
+            state.showPopoverTime = false
+          }}
+        />
+      </>
+    )
+  }
+})

+ 409 - 406
src/views/member-center/index.tsx

@@ -1,406 +1,409 @@
-import ColHeader from '@/components/col-header'
-import { Button, Cell, Icon, Image, Popup, Toast } from 'vant'
-import { defineComponent } from 'vue'
-import styles from './index.module.less'
-import request from '@/helpers/request'
-import { setLogin, state } from '@/state'
-import iconStudent from '@common/images/icon_student.png'
-import iconTeacher from '@common/images/icon_teacher.png'
-import iconGift from './images/icon_gift.png'
-import iconShare from '../music/album/icon_share2.svg'
-import iconDiscount from './images/icon_discount.png'
-import iconMemberLogo from './images/member_logo.png'
-import { orderStatus } from '@/views/order-detail/orderStatus'
-import dayjs from 'dayjs'
-import { memberType } from '@/constant'
-import { moneyFormat } from '@/helpers/utils'
-import ColShare from '@/components/col-share'
-
-export const getAssetsHomeFile = (fileName: string) => {
-  const path = `./images/${fileName}`
-  const modules = import.meta.globEager('./images/*')
-  return modules[path].default
-}
-
-export default defineComponent({
-  name: 'MemberCenter',
-  data() {
-    const query = this.$route.query
-    return {
-      activityId: query.activityId,
-      recomUserId: query.recomUserId,
-      apiSuffix:
-        state.platformType === 'STUDENT' ? '/api-student' : '/api-teacher',
-      agreeStatus: false,
-      functionList: [],
-      memberList: [],
-      selectMember: {} as any,
-      params: {
-        page: 1,
-        rows: 20
-      },
-      discountTeacher: {
-        avatar: '',
-        discount: 0,
-        username: ''
-      },
-      shareStatus: false,
-      shareUrl: '',
-      shareDiscount: 0 // 是否有优惠活动
-    }
-  },
-  computed: {
-    userInfo() {
-      const users = state.user.data
-      return {
-        username: users?.username,
-        phone: users?.phone,
-        avatar: users?.heardUrl,
-        id: users?.userId,
-        memberRankSettingId: users?.memberRankSettingId,
-        isVip: users?.isVip,
-        membershipDays: users?.membershipDays,
-        membershipEndTime: users?.membershipEndTime
-      }
-    }
-  },
-  async mounted() {
-    try {
-      const userInfo = await request.get(
-        state.platformType === 'TEACHER'
-          ? '/api-teacher/teacher/queryUserInfo'
-          : '/api-student/student/queryUserInfo'
-      )
-      setLogin(userInfo.data)
-
-      const res = await request.post(
-        `${this.apiSuffix}/memberPriceSettings/vipPermissions`
-      )
-      const result = res.data || []
-      this.functionList = result.map((item: any) => {
-        return {
-          title: item.paramName,
-          icon: getAssetsHomeFile(`${item.paramValue}.png`)
-        }
-      })
-
-      const setting = await request.post(
-        `${this.apiSuffix}/memberPriceSettings/list`,
-        {
-          data: {
-            activityId: Number(this.activityId),
-            userId: this.recomUserId
-          }
-        }
-      )
-      const { list, ...more } = setting.data
-      this.discountTeacher = {
-        ...more
-      }
-      const settingResult = list || []
-      let settingList: any = []
-      settingResult.forEach((item: any) => {
-        const tempItem = {
-          title: '',
-          salePrice: item.salePrice,
-          originalPrice: item.originalPrice,
-          period: item.period,
-          id: item.id,
-          discount: item.discount,
-          discountPrice: item.discountPrice,
-          status: false
-        }
-
-        tempItem.title = memberType[item.period]
-
-        item.period !== 'DAY' && settingList.push(tempItem)
-      })
-
-      settingList = settingList ? settingList.reverse() : []
-      if (settingList.length > 0) {
-        settingList[0].status = true
-        this.selectMember = settingList[0]
-      }
-      console.log(settingList)
-      this.memberList = settingList
-    } catch {}
-  },
-  methods: {
-    async onShare() {
-      try {
-        const res = await request.post('/api-teacher/open/vipProfit', {
-          data: {
-            userId: this.userInfo.id
-          }
-        })
-
-        this.shareUrl = `${location.origin}/teacher#/shareVip?recomUserId=${this.userInfo.id}&userType=${state.platformType}`
-        // 判断是否有我分享的编号
-        if (res.data && res.data.activityId) {
-          this.shareUrl = this.shareUrl + '&activityId=' + res.data.activityId
-        }
-        this.shareStatus = true
-        this.shareDiscount = res.data.discount || 0
-        // let url =
-        //   location.origin +
-        //   `/accompany/colexiu-share.html?id=${moreData.value.id}&recomUserId=${state.user.data?.userId}`
-        // // 判断是否有活动
-        // if (res.data.discount === 1) {
-        //   url += `&activityId=${res.data.activityId}`
-        // }
-        // shareDiscount.value = res.data.discount || 0
-        // shareUrl.value = url
-        // isMore.value = false
-        // shareStatus.value = true
-        return
-      } catch {}
-    },
-    calcSalePrice(item: any) {
-      // discount
-      if (item.discount === 1) {
-        const tempPrice = Number(
-          (item.salePrice - item.discountPrice).toFixed(2)
-        )
-        return tempPrice >= 0 ? tempPrice : 0
-      }
-      return item.salePrice
-    },
-    onSubmit() {
-      const member: any = this.selectMember
-      // 判断是否有会员
-      const startTime = this.userInfo.isVip
-        ? dayjs(this.userInfo.membershipEndTime).toDate()
-        : new Date()
-      let endTime = new Date()
-      if (member.period === 'MONTH') {
-        endTime = dayjs(startTime).add(1, 'month').toDate()
-      } else if (member.period === 'QUARTERLY') {
-        endTime = dayjs(startTime).add(3, 'month').toDate()
-      } else if (member.period === 'YEAR_HALF') {
-        endTime = dayjs(startTime).add(6, 'month').toDate()
-      } else if (member.period === 'YEAR') {
-        endTime = dayjs(startTime).add(1, 'year').toDate()
-      }
-
-      orderStatus.orderObject.orderType = 'VIP'
-      orderStatus.orderObject.orderName = '小酷Ai' + member.title
-      orderStatus.orderObject.orderDesc = '小酷Ai' + member.title
-      orderStatus.orderObject.actualPrice = this.calcSalePrice(member)
-      orderStatus.orderObject.recomUserId = this.recomUserId
-      orderStatus.orderObject.activityId = this.activityId
-      orderStatus.orderObject.orderNo = ''
-      orderStatus.orderObject.orderList = [
-        {
-          orderType: 'VIP',
-          goodsName: '小酷Ai' + member.title,
-          id: member.id,
-          title: member.title,
-          price: this.calcSalePrice(member),
-          startTime: dayjs(startTime).format('YYYY-MM-DD'),
-          endTime: dayjs(endTime).format('YYYY-MM-DD'),
-          recomUserId: this.recomUserId
-        }
-      ]
-      this.$router.push({
-        path: '/orderDetail',
-        query: {
-          orderType: 'VIP'
-        }
-      })
-    }
-  },
-  render() {
-    return (
-      <div class={styles['member-center']}>
-        <ColHeader
-          background="#ffe5cc"
-          v-slots={{
-            right: () => (
-              <div class={styles.shareBtn} onClick={this.onShare}>
-                <Image src={iconShare} />
-                分享
-              </div>
-            )
-          }}
-        />
-        <div class={styles.member_container}>
-          <Cell
-            class={styles.userMember}
-            labelClass={styles.timeRemaining}
-            v-slots={{
-              icon: () => (
-                <div class={styles.userImgSection}>
-                  <Image
-                    class={styles.userImg}
-                    src={this.userInfo.avatar || iconStudent}
-                    fit="cover"
-                  />
-                </div>
-              ),
-              title: () => (
-                <div class={styles.userInfo}>
-                  <span class={styles.name}>{this.userInfo.username}</span>
-                  {!!this.userInfo.isVip && (
-                    <Image
-                      class={styles.level}
-                      src="https://daya.ks3-cn-beijing.ksyun.com/202107/ScSTL1D.png"
-                    />
-                  )}
-                  <span
-                    class={styles.phone}
-                    v-html={`(${this.userInfo.phone})`}
-                  ></span>
-                </div>
-              ),
-              label: () => (
-                <div class={styles.member_time}>
-                  {this.userInfo.isVip ? (
-                    <div>
-                      会员权益有效期剩余
-                      <span class={styles.remaining}>
-                        {this.userInfo.membershipDays}
-                      </span>
-                      天
-                    </div>
-                  ) : (
-                    <div>亲,您还不是会员哟</div>
-                  )}
-                </div>
-              )
-            }}
-          ></Cell>
-        </div>
-
-        <div class={styles.memberContainer}>
-          <div class={styles.memberItem}>
-            <div class={styles.title}>
-              会员<span>VIP</span>
-            </div>
-
-            {/* 判断是否有推荐老师 */}
-            {this.discountTeacher.discount == 1 && (
-              <div class={styles.memberDiscount}>
-                <Image
-                  src={this.discountTeacher.avatar || iconTeacher}
-                  class={styles.discountAvatar}
-                />
-
-                <span class={styles.discountName}>
-                  {this.discountTeacher.username}老师的专属优惠~
-                </span>
-
-                <Image src={iconGift} class={styles.discountGift} />
-              </div>
-            )}
-
-            <div class={styles['system-list']}>
-              {this.memberList.map((item: any) => (
-                <div
-                  class={[styles['system-item'], item.status && styles.active]}
-                  onClick={() => {
-                    this.memberList.forEach((item: any) => {
-                      item.status = false
-                    })
-                    item.status = true
-                    this.selectMember = item
-                  }}
-                >
-                  <div class={styles.discountItem}>
-                    {item.discount == 1 && <img src={iconDiscount} />}
-                  </div>
-                  <p class={styles.title}>{item.title}</p>
-                  <p class={styles.price}>
-                    <span>¥</span>
-                    {moneyFormat(this.calcSalePrice(item), '0,0[.]00')}
-                  </p>
-                  <del class={styles.originalPrice}>
-                    ¥{moneyFormat(item.originalPrice, '0,0[.]00')}
-                  </del>
-                </div>
-              ))}
-            </div>
-          </div>
-
-          <div class={[styles.intro]}>
-            <p>
-              酷乐秀会员可使用包括平台提供的所有训练乐谱,并专享“
-              <b>小酷Ai</b>
-              ”八大核心功能,孩子在家就能轻松完成乐器自主规范练习。
-            </p>
-          </div>
-
-          {this.functionList.length > 0 && (
-            <div class={styles.memberItem}>
-              <div class={styles.title}>会员功能</div>
-
-              <div class={styles.member_function}>
-                {this.functionList.map((item: any) => (
-                  <div class={styles.function_item}>
-                    <Icon name={item.icon} size={34} />
-                    <div class={styles.function_text} v-html={item.title}></div>
-                  </div>
-                ))}
-              </div>
-            </div>
-          )}
-
-          {/* <ColProtocol
-            v-model={this.agreeStatus}
-            showHeader
-            style={{ paddingLeft: 0, paddingRight: 0, marginBottom: '64px' }}
-          /> */}
-        </div>
-        <div class={styles.btnGroup}>
-          <div class={styles.priceSection}>
-            支付金额:
-            <div class={styles.price}>
-              <span class={styles.priceUnit}>¥</span>
-              <span class={styles.priceNum}>
-                {(this as any).$filters.moneyFormat(
-                  this.calcSalePrice(this.selectMember) || 0
-                )}
-              </span>
-            </div>
-            {this.selectMember?.discount == 1 && (
-              <div class={[styles.discountItem, styles.discountBuy]}>
-                <img src={iconDiscount} />
-              </div>
-            )}
-          </div>
-          <Button
-            color="linear-gradient(220deg, #DFA164 0%, #FAC87E 100%)"
-            round
-            class={styles.btn}
-            onClick={this.onSubmit}
-          >
-            立即支付
-          </Button>
-        </div>
-
-        <Popup
-          v-model:show={this.shareStatus}
-          style={{ background: 'transparent' }}
-        >
-          <ColShare
-            teacherId={this.userInfo.id}
-            shareUrl={this.shareUrl}
-            shareType="vip"
-            shareLength={2}
-          >
-            <div class={styles.shareVip}>
-              {this.shareDiscount === 1 && (
-                <div class={styles.tagDiscount}>专属优惠</div>
-              )}
-
-              <img class={styles.icon} src={iconMemberLogo} />
-              <div class={styles.info}>
-                <h4 class="van-multi-ellipsis--l2">小酷Ai会员</h4>
-                <p>海量曲谱、智能评测,专为器乐学习者量身打造</p>
-              </div>
-            </div>
-          </ColShare>
-        </Popup>
-      </div>
-    )
-  }
-})
+import ColHeader from '@/components/col-header'
+import { Button, Cell, Icon, Image, Popup, Toast } from 'vant'
+import { defineComponent } from 'vue'
+import styles from './index.module.less'
+import request from '@/helpers/request'
+import { setLogin, state } from '@/state'
+import iconStudent from '@common/images/icon_student.png'
+import iconTeacher from '@common/images/icon_teacher.png'
+import iconGift from './images/icon_gift.png'
+import iconShare from '../music/album/icon_share2.svg'
+import iconDiscount from './images/icon_discount.png'
+import iconMemberLogo from './images/member_logo.png'
+import { orderStatus } from '@/views/order-detail/orderStatus'
+import dayjs from 'dayjs'
+import { memberType } from '@/constant'
+import { moneyFormat } from '@/helpers/utils'
+import ColShare from '@/components/col-share'
+
+export const getAssetsHomeFile = (fileName: string) => {
+  const path = `./images/${fileName}`
+  const modules = import.meta.globEager('./images/*')
+  return modules[path].default
+}
+
+export default defineComponent({
+  name: 'MemberCenter',
+  data() {
+    const query = this.$route.query
+    return {
+      activityId: query.activityId,
+      recomUserId: query.recomUserId,
+      apiSuffix:
+        state.platformType === 'STUDENT' ? '/api-student' : '/api-teacher',
+      agreeStatus: false,
+      functionList: [],
+      memberList: [],
+      selectMember: {} as any,
+      params: {
+        page: 1,
+        rows: 20
+      },
+      discountTeacher: {
+        avatar: '',
+        discount: 0,
+        username: ''
+      },
+      shareStatus: false,
+      shareUrl: '',
+      shareDiscount: 0 // 是否有优惠活动
+    }
+  },
+  computed: {
+    userInfo() {
+      const users = state.user.data
+      return {
+        username: users?.username,
+        phone: users?.phone,
+        avatar: users?.heardUrl,
+        id: users?.userId,
+        memberRankSettingId: users?.memberRankSettingId,
+        isVip: users?.isVip,
+        membershipDays: users?.membershipDays,
+        membershipEndTime: users?.membershipEndTime
+      }
+    }
+  },
+  async mounted() {
+    try {
+      const userInfo = await request.get(
+        state.platformType === 'TEACHER'
+          ? '/api-teacher/teacher/queryUserInfo'
+          : '/api-student/student/queryUserInfo'
+      )
+      setLogin(userInfo.data)
+
+      const res = await request.post(
+        `${this.apiSuffix}/memberPriceSettings/vipPermissions`
+      )
+      const result = res.data || []
+      this.functionList = result.map((item: any) => {
+        return {
+          title: item.paramName,
+          icon: getAssetsHomeFile(`${item.paramValue}.png`)
+        }
+      })
+
+      const setting = await request.post(
+        `${this.apiSuffix}/memberPriceSettings/list`,
+        {
+          data: {
+            activityId: Number(this.activityId),
+            userId: this.recomUserId
+          }
+        }
+      )
+      const { list, ...more } = setting.data
+      this.discountTeacher = {
+        ...more
+      }
+      const settingResult = list || []
+      let settingList: any = []
+      settingResult.forEach((item: any) => {
+        const tempItem = {
+          title: '',
+          salePrice: item.salePrice,
+          originalPrice: item.originalPrice,
+          period: item.period,
+          id: item.id,
+          discount: item.discount,
+          discountPrice: item.discountPrice,
+          status: false
+        }
+
+        tempItem.title = memberType[item.period]
+
+        item.period !== 'DAY' && settingList.push(tempItem)
+      })
+
+      settingList = settingList ? settingList.reverse() : []
+      if (settingList.length > 0) {
+        settingList[0].status = true
+        this.selectMember = settingList[0]
+      }
+      console.log(settingList)
+      this.memberList = settingList
+    } catch {}
+  },
+  methods: {
+    async onShare() {
+      try {
+        const res = await request.post('/api-teacher/open/vipProfit', {
+          data: {
+            userId: this.userInfo.id
+          }
+        })
+
+        this.shareUrl = `${location.origin}/teacher#/shareVip?recomUserId=${this.userInfo.id}&userType=${state.platformType}`
+        // 判断是否有我分享的编号
+        if (res.data && res.data.activityId) {
+          this.shareUrl = this.shareUrl + '&activityId=' + res.data.activityId
+        }
+        this.shareStatus = true
+        this.shareDiscount = res.data.discount || 0
+        // let url =
+        //   location.origin +
+        //   `/accompany/colexiu-share.html?id=${moreData.value.id}&recomUserId=${state.user.data?.userId}`
+        // // 判断是否有活动
+        // if (res.data.discount === 1) {
+        //   url += `&activityId=${res.data.activityId}`
+        // }
+        // shareDiscount.value = res.data.discount || 0
+        // shareUrl.value = url
+        // isMore.value = false
+        // shareStatus.value = true
+        return
+      } catch {}
+    },
+    calcSalePrice(item: any) {
+      // discount
+      if (item.discount === 1) {
+        const tempPrice = Number(
+          (item.salePrice - item.discountPrice).toFixed(2)
+        )
+        return tempPrice >= 0 ? tempPrice : 0
+      }
+      return item.salePrice
+    },
+    onSubmit() {
+      const member: any = this.selectMember
+      // 判断是否有会员
+      const startTime = this.userInfo.isVip
+        ? dayjs(this.userInfo.membershipEndTime).toDate()
+        : new Date()
+      let endTime = new Date()
+      if (member.period === 'MONTH') {
+        endTime = dayjs(startTime).add(1, 'month').toDate()
+      } else if (member.period === 'QUARTERLY') {
+        endTime = dayjs(startTime).add(3, 'month').toDate()
+      } else if (member.period === 'YEAR_HALF') {
+        endTime = dayjs(startTime).add(6, 'month').toDate()
+      } else if (member.period === 'YEAR') {
+        endTime = dayjs(startTime).add(1, 'year').toDate()
+      }
+
+      orderStatus.orderObject.orderType = 'VIP'
+      orderStatus.orderObject.orderName = '小酷Ai' + member.title
+      orderStatus.orderObject.orderDesc = '小酷Ai' + member.title
+      orderStatus.orderObject.actualPrice = this.calcSalePrice(member)
+      orderStatus.orderObject.recomUserId = this.recomUserId
+      orderStatus.orderObject.activityId = this.activityId
+      orderStatus.orderObject.orderNo = ''
+      orderStatus.orderObject.orderList = [
+        {
+          orderType: 'VIP',
+          goodsName: '小酷Ai' + member.title,
+          id: member.id,
+          title: member.title,
+          price: this.calcSalePrice(member),
+          startTime: dayjs(startTime).format('YYYY-MM-DD'),
+          endTime: dayjs(endTime).format('YYYY-MM-DD'),
+          recomUserId: this.recomUserId
+        }
+      ]
+      this.$router.push({
+        path: '/orderDetail',
+        query: {
+          orderType: 'VIP'
+        }
+      })
+    }
+  },
+  render() {
+    return (
+      <div class={styles['member-center']}>
+        <ColHeader
+          background="#ffe5cc"
+          v-slots={{
+            right: () => (
+              <div class={styles.shareBtn} onClick={this.onShare}>
+                <Image src={iconShare} />
+                分享
+              </div>
+            )
+          }}
+        />
+        <div class={styles.member_container}>
+          <Cell
+            class={styles.userMember}
+            labelClass={styles.timeRemaining}
+            v-slots={{
+              icon: () => (
+                <div class={styles.userImgSection}>
+                  <Image
+                    class={styles.userImg}
+                    src={this.userInfo.avatar || iconStudent}
+                    fit="cover"
+                  />
+                </div>
+              ),
+              title: () => (
+                <div class={styles.userInfo}>
+                  <span class={styles.name}>{this.userInfo.username}</span>
+                  {!!this.userInfo.isVip && (
+                    <Image
+                      class={styles.level}
+                      src="https://daya.ks3-cn-beijing.ksyun.com/202107/ScSTL1D.png"
+                    />
+                  )}
+                  <span
+                    class={styles.phone}
+                    v-html={`(${this.userInfo.phone})`}
+                  ></span>
+                </div>
+              ),
+              label: () => (
+                <div class={styles.member_time}>
+                  {this.userInfo.isVip ? (
+                    <div>
+                      会员权益有效期剩余
+                      <span class={styles.remaining}>
+                        {this.userInfo.membershipDays}
+                      </span>
+                      天
+                    </div>
+                  ) : !this.userInfo.isVip &&
+                    this.userInfo.membershipEndTime ? (
+                    <div>您的会员已过期,续费后会员权益可继续使用</div>
+                  ) : (
+                    <div>亲,您还不是会员哦~</div>
+                  )}
+                </div>
+              )
+            }}
+          ></Cell>
+        </div>
+
+        <div class={styles.memberContainer}>
+          <div class={styles.memberItem}>
+            <div class={styles.title}>
+              会员<span>VIP</span>
+            </div>
+
+            {/* 判断是否有推荐老师 */}
+            {this.discountTeacher.discount == 1 && (
+              <div class={styles.memberDiscount}>
+                <Image
+                  src={this.discountTeacher.avatar || iconTeacher}
+                  class={styles.discountAvatar}
+                />
+
+                <span class={styles.discountName}>
+                  {this.discountTeacher.username}老师的专属优惠~
+                </span>
+
+                <Image src={iconGift} class={styles.discountGift} />
+              </div>
+            )}
+
+            <div class={styles['system-list']}>
+              {this.memberList.map((item: any) => (
+                <div
+                  class={[styles['system-item'], item.status && styles.active]}
+                  onClick={() => {
+                    this.memberList.forEach((item: any) => {
+                      item.status = false
+                    })
+                    item.status = true
+                    this.selectMember = item
+                  }}
+                >
+                  <div class={styles.discountItem}>
+                    {item.discount == 1 && <img src={iconDiscount} />}
+                  </div>
+                  <p class={styles.title}>{item.title}</p>
+                  <p class={styles.price}>
+                    <span>¥</span>
+                    {moneyFormat(this.calcSalePrice(item), '0,0[.]00')}
+                  </p>
+                  <del class={styles.originalPrice}>
+                    ¥{moneyFormat(item.originalPrice, '0,0[.]00')}
+                  </del>
+                </div>
+              ))}
+            </div>
+          </div>
+
+          <div class={[styles.intro]}>
+            <p>
+              酷乐秀会员可使用包括平台提供的所有训练乐谱,并专享“
+              <b>小酷Ai</b>
+              ”八大核心功能,孩子在家就能轻松完成乐器自主规范练习。
+            </p>
+          </div>
+
+          {this.functionList.length > 0 && (
+            <div class={styles.memberItem}>
+              <div class={styles.title}>会员功能</div>
+
+              <div class={styles.member_function}>
+                {this.functionList.map((item: any) => (
+                  <div class={styles.function_item}>
+                    <Icon name={item.icon} size={34} />
+                    <div class={styles.function_text} v-html={item.title}></div>
+                  </div>
+                ))}
+              </div>
+            </div>
+          )}
+
+          {/* <ColProtocol
+            v-model={this.agreeStatus}
+            showHeader
+            style={{ paddingLeft: 0, paddingRight: 0, marginBottom: '64px' }}
+          /> */}
+        </div>
+        <div class={styles.btnGroup}>
+          <div class={styles.priceSection}>
+            支付金额:
+            <div class={styles.price}>
+              <span class={styles.priceUnit}>¥</span>
+              <span class={styles.priceNum}>
+                {(this as any).$filters.moneyFormat(
+                  this.calcSalePrice(this.selectMember) || 0
+                )}
+              </span>
+            </div>
+            {this.selectMember?.discount == 1 && (
+              <div class={[styles.discountItem, styles.discountBuy]}>
+                <img src={iconDiscount} />
+              </div>
+            )}
+          </div>
+          <Button
+            color="linear-gradient(220deg, #DFA164 0%, #FAC87E 100%)"
+            round
+            class={styles.btn}
+            onClick={this.onSubmit}
+          >
+            立即支付
+          </Button>
+        </div>
+
+        <Popup
+          v-model:show={this.shareStatus}
+          style={{ background: 'transparent' }}
+        >
+          <ColShare
+            teacherId={this.userInfo.id}
+            shareUrl={this.shareUrl}
+            shareType="vip"
+            shareLength={2}
+          >
+            <div class={styles.shareVip}>
+              {this.shareDiscount === 1 && (
+                <div class={styles.tagDiscount}>专属优惠</div>
+              )}
+
+              <img class={styles.icon} src={iconMemberLogo} />
+              <div class={styles.info}>
+                <h4 class="van-multi-ellipsis--l2">小酷Ai会员</h4>
+                <p>海量曲谱、智能评测,专为器乐学习者量身打造</p>
+              </div>
+            </div>
+          </ColShare>
+        </Popup>
+      </div>
+    )
+  }
+})