list.vue 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. <!-- 商品列表 -->
  2. <template>
  3. <view class="list-box">
  4. <view class="head-box">
  5. <!-- 标题栏 -->
  6. <shopro-navbar>
  7. <view class="u-flex-1 u-flex u-col-center u-m-x-20" slot="content">
  8. <u-search placeholder="请输入关键字" @change="onSearch" @search="onSearch" @clear="clearSearch" v-model="searchVal" :showAction="false" height="60"></u-search>
  9. </view>
  10. </shopro-navbar>
  11. <!-- 筛选栏 -->
  12. <sh-filter @change="onFilter"></sh-filter>
  13. </view>
  14. <view class="u-waterfall u-p-16" v-if="!isEmpty">
  15. <view id="u-left-column" class="u-column">
  16. <view class="goods-item u-m-b-16 u-flex u-row-center u-col-center" v-for="leftGoods in leftList" :key="leftGoods.id">
  17. <shopro-goods-card
  18. :detail="leftGoods"
  19. :type="leftGoods.activity_type"
  20. :image="leftGoods.image"
  21. :title="leftGoods.title"
  22. :subtitle="leftGoods.subtitle"
  23. :price="leftGoods.price"
  24. :originPrice="leftGoods.original_price"
  25. :sales="leftGoods.sales"
  26. :tagTextList="leftGoods.activity_discounts_tags"
  27. @click="$Router.push({ path: '/pages/goods/detail', query: { id: leftGoods.id } })"
  28. ></shopro-goods-card>
  29. </view>
  30. </view>
  31. <view id="u-right-column" class="u-column">
  32. <view class="goods-item u-m-b-16 u-flex u-row-center u-col-center" v-for="rightGoods in rightList" :key="rightGoods.id">
  33. <shopro-goods-card
  34. :detail="rightGoods"
  35. :type="rightGoods.activity_type"
  36. :image="rightGoods.image"
  37. :title="rightGoods.title"
  38. :subtitle="rightGoods.subtitle"
  39. :price="rightGoods.price"
  40. :originPrice="rightGoods.original_price"
  41. :sales="rightGoods.sales"
  42. :tagTextList="rightGoods.activity_discounts_tags"
  43. @click="$Router.push({ path: '/pages/goods/detail', query: { id: rightGoods.id } })"
  44. ></shopro-goods-card>
  45. </view>
  46. </view>
  47. </view>
  48. <!-- 缺省页 -->
  49. <shopro-empty v-if="isEmpty" :image="$IMG_URL + '/imgs/empty/empty_goods.png'" tipText="暂无该商品,还有更多好货等着你噢~"></shopro-empty>
  50. <!-- 加载更多 -->
  51. <u-loadmore v-show="!isEmpty" height="80rpx" :status="loadStatus" icon-type="flower" color="#ccc" />
  52. <!-- 登录弹窗 -->
  53. <shopro-auth-modal></shopro-auth-modal>
  54. </view>
  55. </template>
  56. <script>
  57. import shFilter from './components/sh-filter.vue';
  58. import { mapMutations, mapActions, mapState } from 'vuex';
  59. let systemInfo = uni.getSystemInfoSync();
  60. let historyTag = uni.getStorageSync('searchHistoryArr') ? JSON.parse(uni.getStorageSync('searchHistoryArr')) : [];
  61. export default {
  62. components: {
  63. shFilter
  64. },
  65. data() {
  66. return {
  67. isEmpty: false,
  68. goodsList: [],
  69. searchVal: '',
  70. listParams: {
  71. category_id: 0,
  72. keywords: '',
  73. page: 1
  74. },
  75. loadStatus: 'loadmore', //loadmore-加载前的状态,loading-加载中的状态,nomore-没有更多的状态
  76. lastPage: 1,
  77. // 瀑布流 350-330
  78. addTime: 100, //排序间隙时间
  79. leftHeight: 0,
  80. rightHeight: 0,
  81. leftList: [],
  82. rightList: [],
  83. tempList: []
  84. };
  85. },
  86. // 触底加载更多
  87. onReachBottom() {
  88. if (this.listParams.page < this.lastPage) {
  89. this.listParams.page += 1;
  90. this.getGoodsList();
  91. }
  92. },
  93. onLoad() {
  94. if (this.$Route.query.id) {
  95. this.listParams.category_id = this.$Route.query.id;
  96. this.getGoodsList();
  97. } else if (this.$Route.query.hasOwnProperty('keywords')) {
  98. this.listParams.keywords = this.$Route.query.keywords;
  99. this.searchVal = this.$Route.query.keywords;
  100. !this.$Route.query.keywords && this.getGoodsList();
  101. } else {
  102. this.getGoodsList();
  103. }
  104. },
  105. methods: {
  106. // 瀑布流相关
  107. async splitData() {
  108. if (!this.tempList.length) return;
  109. let item = this.tempList[0];
  110. if (!item) return;
  111. // 分左右
  112. if (this.leftHeight < this.rightHeight) {
  113. this.leftHeight += item.activity_discounts_tags.length ? 350 : 330;
  114. this.leftList.push(item);
  115. } else if (this.leftHeight > this.rightHeight) {
  116. this.rightHeight += item.activity_discounts_tags.length ? 350 : 330;
  117. this.rightList.push(item);
  118. } else {
  119. this.leftHeight += item.activity_discounts_tags.length ? 350 : 330;
  120. this.leftList.push(item);
  121. }
  122. // 移除临时列表的第一项,如果临时数组还有数据,继续循环
  123. this.tempList.splice(0, 1);
  124. if (this.tempList.length) {
  125. setTimeout(() => {
  126. this.splitData();
  127. }, this.addTime);
  128. }
  129. },
  130. clear() {
  131. this.leftList = [];
  132. this.rightList = [];
  133. this.leftHeight = 0;
  134. this.rightHeight = 0;
  135. this.tempList = [];
  136. },
  137. onFilter(e) {
  138. this.listParams.order = e;
  139. this.goodsList = [];
  140. this.listParams.page = 1;
  141. this.lastPage = 1;
  142. this.clear();
  143. this.$u.debounce(this.getGoodsList);
  144. },
  145. // 键盘搜索,输入搜索
  146. onSearch() {
  147. this.goodsList = [];
  148. this.listParams.page = 1;
  149. this.lastPage = 1;
  150. this.listParams.keywords = this.searchVal;
  151. this.clear();
  152. this.$u.debounce(this.getGoodsList);
  153. },
  154. // 队列
  155. getArr(list, item) {
  156. let arr = list;
  157. let length = 10; //队列长度
  158. arr.length < length ? arr.unshift(item) : arr.pop();
  159. return arr;
  160. },
  161. // 清除搜索框
  162. clearSearch() {
  163. this.searchVal = '';
  164. this.clear();
  165. },
  166. // 商品列表
  167. getGoodsList() {
  168. let that = this;
  169. that.loadStatus = 'loading';
  170. that.$http('goods.lists', that.listParams, '加载中...').then(res => {
  171. if (this.searchVal && !historyTag.includes(this.searchVal)) {
  172. let searchHistoryArr = JSON.stringify(this.getArr(historyTag, this.searchVal));
  173. uni.setStorageSync('searchHistoryArr', searchHistoryArr);
  174. }
  175. if (res.code === 1) {
  176. that.goodsList = [...that.goodsList, ...res.data.data];
  177. that.isEmpty = !that.goodsList.length;
  178. that.lastPage = res.data.last_page;
  179. that.loadStatus = that.listParams.page < res.data.last_page ? 'loadmore' : 'nomore';
  180. that.tempList = res.data.data;
  181. that.splitData();
  182. }
  183. });
  184. }
  185. }
  186. };
  187. </script>
  188. <style lang="scss">
  189. @mixin vue-flex($direction: row) {
  190. /* #ifndef APP-NVUE */
  191. display: flex;
  192. flex-direction: $direction;
  193. /* #endif */
  194. }
  195. .u-waterfall {
  196. @include vue-flex;
  197. flex-direction: row;
  198. align-items: flex-start;
  199. }
  200. .u-column {
  201. @include vue-flex;
  202. flex: 1;
  203. flex-direction: column;
  204. height: auto;
  205. }
  206. .head-box {
  207. position: -webkit-sticky;
  208. position: sticky;
  209. top: 0;
  210. z-index: 998;
  211. background: #fff;
  212. }
  213. </style>