water-fall.vue 6.74 KB
<template>
  <div class="wf-list">
    <div v-for="(i, index) in viewList" :key="index"
      class="wf-item"
      :class="{'wf-item-default': i._default, 'wf-item-temp': i._temporary}"
      :style="{width: 100 / cols + '%', left: i.left + '%', top: i.top + 'px'}">
      <div class="wf-item-mid">
        <router-link to="/ufo/order/1">
          <div class="layer-image" :style="{height: i.coverHeight + 'px'}">
            <img v-if="!i._temporary" :src="i[srcKey]" :alt="i.content">
          </div>
          <div class="description">{{i.content}}</div>
        </router-link>

        <div class="attribution">
          <router-link to="/ufo/order/1" class="auther">
            <span class="avatar">
              <img v-if="!i._temporary" :src="i.authorHeadIco">
            </span>
            <span class="name">{{i.authorName}}</span>
          </router-link>

          <div class="fav">21321</div>
        </div>
      </div>
    </div>
  </div>
</template>

<script>
import {assign} from 'lodash';

export default {
  data() {
    return {
      viewList: [{
        _default: true
      }],
      colWidthPer: 0,
      calcIndex: 0,
      loadedIndex: 0
    }
  },
  props: {
    pos: {
      type: Number,
      default: 0
    },
    list: {
      type: Array,
      default: []
    },
    srcKey: {
      type: String,
      default: 'coverImage'
    },
    cols: {
      type: Number,
      default: 2
    },
    space: {
      type: Number,
      default: 14
    }
  },
  mounted() {
    this.$on('calced', (nlist) => {
      this.viewList = this.viewList.concat(nlist);
      this.$nextTick(() => {
        this.calcLayout();
      })
    });

    this.reset();
    this.clacCoverSize();
  },
  watch: {
    pos() {
      this.timer && clearTimeout(this.timer);
      this.timer = setTimeout(this.resetViewList, 0);
    },
    list(newList, oldList) {
      if (this.viewList.length > newList.length) {
        this.reset();
      }

      this.clacCoverSize();
    }
  },
  computed: {
    // isViewItem() {
    //   console.log(arguments);
    //   return true;
    // },
  },
  methods: {
    clacCoverSize() {
      let nlist = [];

      for (let i = this.calcIndex; i < this.list.length; i++) {
        let item = this.list[i];

        item.coverHeight = item.imageHeight / item.imageWidth * this.coverImageWidth;
console.log(item.imageHeight / item.imageWidth * this.coverImageWidth);
console.log(item.imageHeight, item.imageWidth, this.coverImageWidth);
        nlist.push(assign({_temporary: true}, item));
      };

      this.$emit('calced', nlist);
    },
    calcCoverImgHeight(w, h) {
      return h / w * this.coverImageWidth;
    },
    calcLayout() {
      let $item = this.$el.getElementsByClassName('wf-item-temp');

      if (!$item || !$item.length) {
        return;
      }

      if (!this.loadedIndex || !this.colsHeight) {
        this.colsHeight = [];
      }

      for (let i = this.loadedIndex; i < this.list.length; i++) {
        let $elem = $item[i - this.loadedIndex];

        if (!$elem) {
          return;
        }

        let height = $elem.offsetHeight;
        let top, left;
        let item = this.list[i];

        item.height = height;

        if (i < this.cols) {
          this.colsHeight[i] = height;
          top = 0
          left = i * this.colWidthPer
        } else {
          let minHeight = Math.min.apply(null, this.colsHeight);
          let minIndex = this.colsHeight.indexOf(minHeight);

          top = minHeight;
          left = minIndex * this.colWidthPer;

          this.colsHeight[minIndex] = minHeight + height;
        }

        item.left = left;
        item.top = top;
      }

      this.loadedIndex = this.list.length;

      this.resetViewList();
    },
    setImgWidth() {
      let imgWidth = this.$el.offsetWidth / this.cols;
      let $item = this.$el.getElementsByClassName('wf-item');

      if ($item && $item.length) {
        let _w = $item[0].offsetWidth;
        let $img = $item[0].getElementsByClassName('layer-image');

        (_w > 0) && (imgWidth = _w);

        if ($img && $img.length) {
          _w = $img[0].offsetWidth;
          (_w > 0) && (imgWidth = _w);
        }
      }

      this.coverImageWidth = imgWidth;
    },
    reset() {
      this.offsetTop = this.$el.offsetTop;
      this.clientHeight = document.body.clientHeight;
      this.colWidthPer = 100 / this.cols;
      this.loadedIndex = 0;
      this.calcIndex = 0;

      this.setImgWidth();
    },
    resetViewList() {
      this.viewIndex = this.viewIndex || {};

      let i, step;
      let startPos = this.pos - this.clientHeight * 2;
      let endPos = this.pos + this.clientHeight * 2;
      let list = [];
      let indexArr = [];

      if (this.pos < this.lastPos) {
        i = this.viewIndex['end'] || 0;
        step = -1;
      } else {
        i = this.viewIndex['start'] || 0;
        step = 1;
      }

      let loop = true;

      while (loop)
      {
        let item = this.list[i];

        if (!item) {
          loop = false;
          continue;
        }

        if (item.top > endPos - this.offsetTop) {
          (step > 0) && (loop = false);
        } else if (item.top < startPos - this.offsetTop) {
          (step < 0) && (loop = false);
        } else {
          indexArr.push(i);
          list.push(item);
        }

        i += step;
      }

      if (!indexArr.length) {
        return;
      }

      indexArr = indexArr.sort();

      let viewIndex = {
        start: indexArr[0],
        end: indexArr[indexArr.length - 1]
      };

      if (this.viewIndex.start !== viewIndex.start || this.viewIndex.end !== viewIndex.end) {
        this.viewList = list;
        this.lastPos = this.pos;
        this.viewIndex = viewIndex;
      }
    }
  }
};
</script>

<style lang="css">
  .wf-list {
    margin: 0 20px;
    font-size: 0;
    position: relative;
  }

  .wf-item {
    width: 50%;
    padding: 10px;
    font-size: 24px;
    overflow: hidden;
    position: absolute;

    .wf-item-mid {
      border-radius: 4px;
      box-shadow: 0 4px 16px 0 rgba(0, 0, 0, 0.15);
      overflow: hidden;
    }

    .layer-image {
      background-color: #f4f4f4;
      min-height: 100px;

      > img {
        width: 100%;
        height: 100%;
        display: block;
      }
    }

    .description {
      line-height: 1.5;
      padding: 10px 20px;
    }

    .attribution {
      display: flex;
      justify-content: space-between;
      padding: 20px;
    }

    .avatar {
      width: 60px;
      height: 60px;
      border-radius: 50%;
      display: inline-block;
      vertical-align: middle;
      overflow: hidden;

      > img {
        width: 100%;
        height: 100%;
      }
    }

    .name {
      display: inline-block;
      vertical-align: middle;
    }
  }

  .wf-item-default,
  .wf-item-temp {
    opacity: 0;
    margin-left: -100%;
  }
</style>