waterfalls-flow.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368
  1. <template>
  2. <view class="waterfalls-flow">
  3. <view v-for="(item,index) in data.column" :key="index" class="waterfalls-flow-column" :id="`waterfalls_flow_column_${index+1}`"
  4. :msg="msg" :style="{'width':w,'margin-left':index==0?0:m}">
  5. <view class="cell" v-for="(item2,index2) in columnValue(index)" :key="index2"
  6. :class="['column-value',{'column-value-show':item2.o}]" :style="[s1]" @click.stop="wapperClick(item2)">
  7. <view class="inner" v-if="data.seat==1">
  8. <!-- #ifdef MP-WEIXIN -->
  9. <!-- #ifdef VUE2 -->
  10. <slot name="slot{{item2.index}}"></slot>
  11. <!-- #endif -->
  12. <!-- #ifdef VUE3 -->
  13. <slot :name="`slot${item2.index}`"></slot>
  14. <!-- #endif -->
  15. <!-- #endif -->
  16. <!-- #ifndef MP-WEIXIN -->
  17. <slot v-bind="item2"></slot>
  18. <!-- #endif -->
  19. </view>
  20. <image :class="['img',{'img-hide':item2[hideImageKey]==true||item2[hideImageKey]==1},{'img-error':!item2[data.imageKey]}]"
  21. :src="item2[data.imageKey]" mode="widthFix" @load="imgLoad(item2,index+1)" @error="imgError(item2,index+1)"
  22. @click.stop="imageClick(item2)"></image>
  23. <view class="inner" v-if="data.seat==2">
  24. <!-- #ifdef MP-WEIXIN -->
  25. <!-- #ifdef VUE2 -->
  26. <slot name="slot{{item2.index}}"></slot>
  27. <!-- #endif -->
  28. <!-- #ifdef VUE3 -->
  29. <slot :name="`slot${item2.index}`"></slot>
  30. <!-- #endif -->
  31. <!-- #endif -->
  32. <!-- #ifndef MP-WEIXIN -->
  33. <slot v-bind="item2"></slot>
  34. <!-- #endif -->
  35. </view>
  36. </view>
  37. </view>
  38. </view>
  39. </template>
  40. <script>
  41. export default {
  42. props: {
  43. value: Array,
  44. column: {
  45. // 列的数量
  46. type: [String, Number],
  47. default: 2
  48. },
  49. maxColumn: {
  50. // 最大列数
  51. type: [String, Number],
  52. default: 5
  53. },
  54. columnSpace: {
  55. // 列之间的间距 百分比
  56. type: [String, Number],
  57. default: 2
  58. },
  59. imageKey: {
  60. // 图片key
  61. type: [String],
  62. default: 'image'
  63. },
  64. hideImageKey: {
  65. // 隐藏图片key
  66. type: [String],
  67. default: 'hide'
  68. },
  69. seat: {
  70. // 文本的位置,1图片之上 2图片之下
  71. type: [String, Number],
  72. default: 2
  73. },
  74. listStyle: {
  75. // 单个展示项的样式:eg:{'background':'red'}
  76. type: Object
  77. }
  78. },
  79. data() {
  80. return {
  81. data: {
  82. list: this.value ? this.value : [],
  83. column: this.column < 2 ? 2 : this.column,
  84. columnSpace: this.columnSpace <= 5 ? this.columnSpace : 5,
  85. imageKey: this.imageKey,
  86. seat: this.seat
  87. },
  88. msg: 0,
  89. listInitStyle: {
  90. 'border-radius': '16rpx',
  91. 'margin-bottom': '24rpx'
  92. },
  93. adds: [], //预置数据
  94. isLoaded: true,
  95. curIndex: 0,
  96. isRefresh: true,
  97. flag: false,
  98. refreshDatas: []
  99. }
  100. },
  101. computed: {
  102. // 计算列宽
  103. w() {
  104. const column_rate = `${100 / this.data.column - +this.data.columnSpace}%`
  105. return column_rate
  106. },
  107. // 计算margin
  108. m() {
  109. const column_margin = `${
  110. (100 -
  111. (100 / this.data.column - +this.data.columnSpace).toFixed(5) *
  112. this.data.column) /
  113. (this.data.column - 1)
  114. }%`
  115. return column_margin
  116. },
  117. // list样式
  118. s1() {
  119. return { ...this.listInitStyle, ...this.listStyle }
  120. }
  121. },
  122. created() {
  123. // 初始化
  124. this.refresh()
  125. },
  126. methods: {
  127. // 预加载图片
  128. loadImages(idx = 0) {
  129. let count = 0
  130. const newList = this.data.list.filter((item, index) => index >= idx)
  131. for (let i = 0; i < newList.length; i++) {
  132. // #ifndef APP-PLUS
  133. uni.getImageInfo({
  134. src: `${newList[i][this.imageKey]}.jpg`,
  135. complete: (res) => {
  136. count++
  137. if (count == newList.length) this.initValue(idx)
  138. }
  139. })
  140. // #endif
  141. // #ifdef APP-PLUS
  142. plus.io.getImageInfo({
  143. src: `${newList[i][this.imageKey]}.jpg`,
  144. complete: (res) => {
  145. count++
  146. if (count == newList.length) this.initValue(idx)
  147. }
  148. })
  149. // #endif
  150. }
  151. },
  152. // 刷新
  153. refresh() {
  154. if (!this.isLoaded) {
  155. this.refreshDatas = this.value
  156. return false
  157. }
  158. setTimeout(() => {
  159. this.refreshDatas = []
  160. this.isRefresh = true
  161. this.adds = []
  162. this.data.list = this.value ? this.value : []
  163. this.data.column =
  164. this.column < 2
  165. ? 2
  166. : this.column >= this.maxColumn
  167. ? this.maxColumn
  168. : this.column
  169. this.data.columnSpace = this.columnSpace <= 5 ? this.columnSpace : 5
  170. this.data.imageKey = this.imageKey
  171. this.data.seat = this.seat
  172. this.curIndex = 0
  173. // 每列的数据初始化
  174. for (let i = 1; i <= this.data.column; i++) {
  175. this.data[`column_${i}_values`] = []
  176. this.msg++
  177. }
  178. this.$nextTick(() => {
  179. this.initValue(this.curIndex, 'refresh==>')
  180. })
  181. }, 1)
  182. },
  183. columnValue(index) {
  184. return this.data[`column_${index + 1}_values`]
  185. },
  186. change(newValue) {
  187. for (let i = 0; i < this.data.list.length; i++) {
  188. const cv = this.data[`column_${this.data.list[i].column}_values`]
  189. for (let j = 0; j < cv.length; j++) {
  190. if (newValue[i] && i === cv[j].index) {
  191. this.data[`column_${this.data.list[i].column}_values`][j] = Object.assign(
  192. cv[j],
  193. newValue[i]
  194. )
  195. this.msg++
  196. break
  197. }
  198. }
  199. }
  200. },
  201. getMin(a, s) {
  202. let m = a[0][s]
  203. let mo = a[0]
  204. for (var i = a.length - 1; i >= 0; i--) {
  205. if (a[i][s] < m) {
  206. m = a[i][s]
  207. }
  208. }
  209. mo = a.filter((i) => i[s] == m)
  210. return mo[0]
  211. },
  212. // 计算每列的高度
  213. getMinColumnHeight() {
  214. return new Promise((resolve) => {
  215. const heightArr = []
  216. for (let i = 1; i <= this.data.column; i++) {
  217. const query = uni.createSelectorQuery().in(this)
  218. query
  219. .select(`#waterfalls_flow_column_${i}`)
  220. .boundingClientRect((data) => {
  221. heightArr.push({ column: i, height: data.height })
  222. })
  223. .exec(() => {
  224. if (this.data.column <= heightArr.length) {
  225. resolve(this.getMin(heightArr, 'height'))
  226. }
  227. })
  228. }
  229. })
  230. },
  231. async initValue(i, from) {
  232. this.isLoaded = false
  233. if (i >= this.data.list.length || this.refreshDatas.length) {
  234. this.msg++
  235. this.loaded()
  236. return false
  237. }
  238. const minHeightRes = await this.getMinColumnHeight()
  239. const c = this.data[`column_${minHeightRes.column}_values`]
  240. this.data.list[i].column = minHeightRes.column
  241. c.push({ ...this.data.list[i], cIndex: c.length, index: i, o: 0 })
  242. this.msg++
  243. },
  244. // 图片加载完成
  245. imgLoad(item, c) {
  246. const i = item.index
  247. item.o = 1
  248. this.$set(
  249. this.data[`column_${c}_values`],
  250. item.cIndex,
  251. JSON.parse(JSON.stringify(item))
  252. )
  253. this.initValue(i + 1)
  254. },
  255. // 图片加载失败
  256. imgError(item, c) {
  257. const i = item.index
  258. item.o = 1
  259. item[this.data.imageKey] = null
  260. this.$set(
  261. this.data[`column_${c}_values`],
  262. item.cIndex,
  263. JSON.parse(JSON.stringify(item))
  264. )
  265. this.initValue(i + 1)
  266. },
  267. // 渲染结束
  268. loaded() {
  269. if (this.refreshDatas.length) {
  270. this.isLoaded = true
  271. this.refresh()
  272. return false
  273. }
  274. this.curIndex = this.data.list.length
  275. if (this.adds.length) {
  276. this.data.list = this.adds[0]
  277. this.adds.splice(0, 1)
  278. this.initValue(this.curIndex)
  279. } else {
  280. if (this.data.list.length) this.$emit('loaded')
  281. this.isLoaded = true
  282. this.isRefresh = false
  283. }
  284. },
  285. // 单项点击事件
  286. wapperClick(item) {
  287. this.$emit('wapperClick', item)
  288. },
  289. // 图片点击事件
  290. imageClick(item) {
  291. this.$emit('imageClick', item)
  292. }
  293. },
  294. watch: {
  295. value: {
  296. deep: true,
  297. handler(newValue, oldValue) {
  298. setTimeout(() => {
  299. this.$nextTick(() => {
  300. if (this.isRefresh) return false
  301. if (this.isLoaded) {
  302. // if (newValue.length <= this.curIndex) return this.refresh();
  303. if (newValue.length <= this.curIndex) return this.change(newValue)
  304. this.data.list = newValue
  305. this.$nextTick(() => {
  306. this.initValue(this.curIndex, 'watch==>')
  307. })
  308. } else {
  309. this.adds.push(newValue)
  310. }
  311. })
  312. }, 10)
  313. }
  314. },
  315. column(newValue) {
  316. this.refresh()
  317. }
  318. }
  319. }
  320. </script>
  321. <style lang="scss" scoped>
  322. .waterfalls-flow {
  323. overflow: hidden;
  324. &-column {
  325. float: left;
  326. }
  327. }
  328. .waterfalls-flow-column {
  329. border-radius: 16rpx;
  330. overflow: hidden;
  331. }
  332. .column-value {
  333. width: 100%;
  334. font-size: 0;
  335. overflow: hidden;
  336. transition: opacity 0.4s;
  337. opacity: 0;
  338. &-show {
  339. opacity: 1;
  340. }
  341. .inner {
  342. font-size: 30rpx;
  343. }
  344. .img {
  345. width: 100%;
  346. &-hide {
  347. display: none;
  348. }
  349. &-error {
  350. background: #f2f2f2
  351. url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAC4AAAAiAQMAAAAatXkPAAAABlBMVEUAAADMzMzIT8AyAAAAAXRSTlMAQObYZgAAAIZJREFUCNdlzjEKwkAUBNAfEGyCuYBkLyLuxRYW2SKlV1JSeA2tUiZg4YrLjv9PGsHqNTPMSAQuyAJgRDHSyvBPwtZoSJXakeJI9iuRLGDygdl6V0yKDtyMAeMPZySj8yfD+UapvRPj2JOwkyAooSV5IwdDjPdCPspe8LyTl9IKJvDETKKRv6vnlUasgg0fAAAAAElFTkSuQmCC)
  352. no-repeat center center;
  353. }
  354. }
  355. }
  356. .cell {
  357. background: rgba($color: #000000, $alpha: 0.5);
  358. box-shadow: 0px 0px 4px 0px rgba(167, 110, 244, 0.5);
  359. border-radius: 16rpx;
  360. }
  361. </style>