category.vue 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354
  1. <template>
  2. <div class="cate-list">
  3. <div class="cate-tag">
  4. <div class="tag-kinds">类型:</div>
  5. <div v-for="item in typeArr"
  6. :key="item.id"
  7. class="tags"
  8. :class="{ active: typeIndex === item.id }"
  9. @click="selectType('type', item.id)">
  10. {{ item.name }}
  11. </div>
  12. </div>
  13. <div class="cate-tag">
  14. <div class="tag-kinds">标签:</div>
  15. <div v-for="item in tagArr"
  16. :key="item.id"
  17. class="tags"
  18. :class="{ active: tag_idIndex === item.id }"
  19. @click="selectType('tag_id', item.id)">
  20. {{ item.name }}
  21. </div>
  22. </div>
  23. <div class="drop-tag">
  24. <van-dropdown-menu ref="menuRef">
  25. <van-dropdown-item title="类型" >
  26. <van-cell
  27. v-for="item in typeArr"
  28. :key="item.id"
  29. :title="item.name"
  30. :class="{ active: typeIndex === item.id }"
  31. @click="selectType('type', item.id)"/>
  32. </van-dropdown-item>
  33. <van-dropdown-item title="标签" ref="tag">
  34. <van-cell
  35. v-for="item in tagArr"
  36. :key="item.id"
  37. :title="item.name"
  38. :class="{ active: tag_idIndex === item.id }"
  39. @click="selectType('tag_id', item.id)"/>
  40. </van-dropdown-item>
  41. </van-dropdown-menu>
  42. </div>
  43. <el-divider class="divide" />
  44. <ul v-if="gameLis.length">
  45. <li class="hot-item" v-for="(item,index) in gameLis" :key="index">
  46. <img class="game-logo" :src="prefix + item.logopic" :onerror="logoErrorFun" alt=""/>
  47. <van-button type="primary" class="down-btn" :disabled="item.download_url === ''" size="small" round color="#14b9c7" @click="downGame(item.download_url, item.download_url === '')">下载</van-button>
  48. <div class="detail">
  49. <div class="top">
  50. <span class="game_title ellip">{{ item.screen_name }}</span>
  51. <el-tag size="small" type="success">{{ (item.game_score / 10) > 5 ? '5.0' : (item.game_score /
  52. 10).toFixed(1)
  53. }}</el-tag>
  54. </div>
  55. <div class="desc ellip">{{ item.sketch || '暂无简介' }}</div>
  56. <el-tag
  57. size="small"
  58. class="game_tag"
  59. type="warning"
  60. round
  61. v-for="(tag, tIndex) in item.tags"
  62. :key="tIndex"
  63. >
  64. {{ tagArr.findIndex((i:any)=>i.id === tag)>=0 ? tagArr.find((i:any)=>i.id === tag).name :'' }}
  65. </el-tag>
  66. </div>
  67. <img :src="prefix + item.screenshot" class="game-poster" :onerror="posterErrorFun" alt=""/>
  68. <div class="action" @click="downGame(item.download_url, item.download_url === '')" :style="item.download_url === ''? 'cursor:not-allowed;':'cursor: pointer;'">
  69. <img :src="img('download.png')" alt=""/>
  70. <div class="size">{{ bytesChange(item.size) }}</div>
  71. </div>
  72. </li>
  73. </ul>
  74. <el-empty v-else :image-size="200" />
  75. <div class="more-action">
  76. <el-button v-if="params.page*params.pagesize < total" type="info" @click="loadMore">更多</el-button>
  77. <el-divider v-else>没有更多了</el-divider>
  78. </div>
  79. </div>
  80. </template>
  81. <script setup lang="ts">
  82. import { ref, reactive, onMounted, inject } from 'vue'
  83. import { getGameType, getGameTag, getGameList } from '@/api/index'
  84. import Message from '@/utils/Message'
  85. import { bytesChange } from '@/utils/bytesFormatter'
  86. import type { DropdownMenuInstance } from 'vant'
  87. import { useRoute } from 'vue-router'
  88. const route = useRoute()
  89. const img:any = inject('img')
  90. const typeIndex = ref<number>(0)
  91. const tag_idIndex = ref<number>(0)
  92. const gameLis: any = ref([])
  93. const typeArr: any = ref([])
  94. const tagArr: any = ref([])
  95. const prefix = ref<string>('')
  96. const isLoading = ref<boolean>(true)
  97. const total = ref<number>(0)
  98. interface Params {
  99. type: number,
  100. tag_id: number,
  101. page: number,
  102. pagesize: number
  103. }
  104. const params = reactive<Params>({
  105. type: 0,
  106. tag_id: 0,
  107. page: 1,
  108. pagesize: 15
  109. })
  110. onMounted(async() => {
  111. await getGameType().then(res => {
  112. // console.log('类型', res);
  113. typeArr.value = res.data.data
  114. typeArr.value.unshift({ id: 0, name: '全部' })
  115. })
  116. await getGameTag().then(res => {
  117. // console.log('标签', res);
  118. tagArr.value = res.data.data
  119. tagArr.value.unshift({ id: 0, name: '全部' })
  120. })
  121. if (route.query.tag_id) {
  122. const id = parseInt(route.query.tag_id as string)
  123. selectType('tag_id', id)
  124. } else {
  125. getGameLists(params)
  126. }
  127. })
  128. // 初始化列表数据
  129. const getGameLists = async(params:any) => {
  130. // console.log('参数', params);
  131. await getGameList(params).then(res => {
  132. // console.log('数据', res);
  133. if (res.data.code === 200 && res.data.data) {
  134. gameLis.value = gameLis.value.concat(res.data.data.lists)
  135. total.value = res.data.data.total
  136. prefix.value = res.data.data.prefix
  137. isLoading.value = false
  138. }
  139. }).catch(err => {
  140. Message.error(err.data.msg)
  141. })
  142. }
  143. const menuRef = ref<DropdownMenuInstance>()
  144. // 点击类型、标签、排序
  145. const selectType = (type: 'type' | 'tag_id', index: any) => {
  146. console.log(index)
  147. gameLis.value = []
  148. params.page = 1
  149. if (type === 'type') {
  150. typeIndex.value = index
  151. params.type = index
  152. }
  153. if (type === 'tag_id') {
  154. tag_idIndex.value = index
  155. params.tag_id = index
  156. }
  157. if (menuRef.value) {
  158. menuRef.value.close()
  159. }
  160. // 如果都为 全部 则发起请求 页面显示15条数据
  161. if (params.type === 0 && params.tag_id === 0) {
  162. params.pagesize = 15
  163. getGameLists(params)
  164. } else {
  165. allData()
  166. }
  167. }
  168. // 全部数据
  169. const allData = async() => {
  170. params.pagesize = total.value
  171. // console.log('params', params);
  172. await getGameList(params).then(resp => {
  173. // console.log('resp===>', resp);
  174. if (resp.data.code === 200 && resp.data.data) {
  175. gameLis.value = resp.data.data.lists.filter((item:any) => {
  176. // 如果标签为全部 类型不为全部 返回匹配的类型数据
  177. if (params.tag_id === 0) {
  178. return item.type === params.type
  179. } else if (params.type === 0) {
  180. // 如果类型为全部 标签不为全部 返回匹配的标签数据
  181. return item.tags.includes(params.tag_id)
  182. } else return item.type === params.type && item.tags.includes(params.tag_id) // 否则返回匹配的标签和类型数据
  183. })
  184. }
  185. }).catch(err => {
  186. Message.error(err.data.msg)
  187. })
  188. }
  189. const downGame = (url: string, disabled:boolean) => {
  190. if (disabled) return
  191. // 下载游戏
  192. window.open(url)
  193. // window.location.href = url;
  194. }
  195. // 加载更多
  196. const loadMore = () => {
  197. params.page += 1
  198. getGameLists(params)
  199. }
  200. const logoErrorFun = (event:any) => {
  201. var imgDom = event.srcElement
  202. imgDom.src = img('default.png')
  203. imgDom.onerror = null
  204. }
  205. const posterErrorFun = (event:any) => {
  206. var imgDom = event.srcElement
  207. imgDom.src = img('defaultPoster.png')
  208. imgDom.onerror = null
  209. }
  210. </script>
  211. <style scoped lang="scss">
  212. .cate-list{
  213. background-color: #fff;
  214. padding: 1.1rem 1rem;
  215. box-sizing: border-box;
  216. width: 100%;
  217. >div{
  218. font-size: .8rem;
  219. }
  220. .tag-kinds{
  221. margin-top: 1rem;
  222. }
  223. .cate-tag{
  224. display: flex;
  225. .tags{
  226. margin: 1rem;
  227. cursor: pointer;
  228. }
  229. }
  230. .drop-tag{
  231. display: none;
  232. }
  233. .hot-item{
  234. display: flex;
  235. align-items: center;
  236. justify-content: space-between;
  237. position: relative;
  238. margin-bottom: .5rem;
  239. border-bottom: .5px solid #ddd;
  240. height: 10rem;
  241. .down-btn{
  242. display: none;
  243. }
  244. .game-logo{
  245. width: 6rem;
  246. height: 6rem;
  247. border-radius: .2rem;
  248. background-color: #f7f7f7;
  249. }
  250. .game-poster{
  251. width: 12.1rem;
  252. height: 7rem;
  253. }
  254. .detail{
  255. width: 30rem;
  256. .top{
  257. display: flex;
  258. align-items: center;
  259. .game_title{
  260. margin-right: .3rem;
  261. font-size: 1rem;
  262. }
  263. }
  264. .desc{
  265. font-size: .8rem;
  266. margin: .5rem 0;
  267. }
  268. }
  269. &:hover .action{
  270. transform: translateX(-5rem);
  271. }
  272. .action{
  273. position: absolute;
  274. right: -6rem;
  275. // top: 50%;
  276. width: 5rem;
  277. height: 7rem;
  278. background-color: rgba($color: #e6a23c, $alpha: .5);
  279. // transform: translateY(-50%);
  280. display: flex;
  281. align-items: center;
  282. justify-content: center;
  283. flex-direction: column;
  284. transition: all 200ms ease;
  285. img{
  286. width: 2.5rem;
  287. }
  288. .size{
  289. font-size: .5rem;
  290. margin-top: .5rem;
  291. color: #fff;
  292. }
  293. }
  294. }
  295. .more-action{
  296. text-align: center;
  297. }
  298. }
  299. // .header_list {
  300. // .top_type {
  301. // margin: 15px 0 10px;
  302. // background-color: #fff;
  303. // padding: 20px 50px 20px 20px;
  304. // box-sizing: border-box;
  305. // .left_title {
  306. // width: 50px;
  307. // font-size: 14px;
  308. // color: #666;
  309. // }
  310. // .right_type {
  311. // // background-color: #ed8c0f;
  312. // .cont {
  313. // margin-right: 20px;
  314. // padding: 3px 5px;
  315. // cursor: pointer;
  316. // &:hover {
  317. // color: #ed8c0f;
  318. // }
  319. // }
  320. // }
  321. // .type {
  322. // margin-bottom: 20px;
  323. // }
  324. // }
  325. // .el-button {
  326. // margin: 5px 20px 15px 0;
  327. // padding: 0 20px;
  328. // border: 0;
  329. // color: #999;
  330. // }
  331. // }
  332. .active {
  333. color: #ed8c0f
  334. }
  335. </style>