u-steps-item.vue 8.99 KB
<template>
  <view class="u-steps-item" ref="u-steps-item" :class="[`u-steps-item--${parentData.direction}`]">
    <view class="u-steps-item__line" v-if="index + 1 < childLength"
          :class="[`u-steps-item__line--${parentData.direction}`]" :style="[lineStyle]"></view>
    <view class="u-steps-item__wrapper"
          :class="[`u-steps-item__wrapper--${parentData.direction}`, parentData.dot && `u-steps-item__wrapper--${parentData.direction}--dot`]"
          :style="[itemStyleInner]">
      <slot name="icon">
        <view class="u-steps-item__wrapper__dot" v-if="parentData.dot" :style="{
						backgroundColor: statusColor
					}">

        </view>
        <view class="u-steps-item__wrapper__icon" v-else-if="parentData.activeIcon || parentData.inactiveIcon">
          <up-icon :name="index <= parentData.current ? parentData.activeIcon : parentData.inactiveIcon"
                   :size="iconSize"
                   :color="index <= parentData.current ? parentData.activeColor : parentData.inactiveColor">
          </up-icon>
        </view>
        <view v-else :style="{
						backgroundColor: statusClass === 'process' ? parentData.activeColor : 'transparent',
						borderColor: statusColor
					}" class="u-steps-item__wrapper__circle">
          <text v-if="statusClass === 'process' || statusClass === 'wait'"
                class="u-steps-item__wrapper__circle__text" :style="{
							color: index == parentData.current ? '#ffffff' : parentData.inactiveColor
						}">{{ index + 1}}</text>
          <up-icon v-else :color="statusClass === 'error' ? 'error' : parentData.activeColor" size="12"
                   :name="statusClass === 'error' ? 'close' : 'checkmark'"></up-icon>
        </view>
      </slot>
    </view>
    <view class="u-steps-item__content" :class="[`u-steps-item__content--${parentData.direction}`,
			parentData.current == index ? 'u-steps-item__content--current' : '']"
          :style="[contentStyle]">
      <slot name="content" :index="index">
      </slot>
      <template v-if="!$slots['content']">
        <view class="u-steps-item__content__title">
          <slot name="title">
          </slot>
          <up-text v-if="!$slots['title']" :text="title" lineHeight="20px"
                   :type="parentData.current == index ? 'main' : 'content'"
                   :size="parentData.current == index ? 14 : 13"></up-text>
        </view>
        <view class="u-steps-item__content__desc">
          <slot name="desc">
          </slot>
          <up-text v-if="!$slots['desc']" :text="desc" type="tips" size="12"></up-text>
        </view>
      </template>
    </view>
    <!-- <view
        class="u-steps-item__line"
        v-if="showLine && parentData.direction === 'column'"
      :class="[`u-steps-item__line--${parentData.direction}`]"
        :style="[lineStyle]"
    ></view> -->
  </view>
</template>

<script>
import { props } from './props';
import { mpMixin } from '../../libs/mixin/mpMixin';
import { mixin } from '../../libs/mixin/mixin';
import { sleep, error } from '../../libs/function/index';
import color from '../../libs/config/color';
// #ifdef APP-NVUE
const dom = uni.requireNativePlugin('dom')
// #endif
/**
 * StepsItem 步骤条的子组件
 * @description 本组件需要和u-steps配合使用
 * @tutorial https://uview-plus.jiangruyi.com/components/steps.html
 * @property {String}			title			标题文字
 * @property {String}			current			描述文本
 * @property {String | Number}	iconSize		图标大小  (默认 17 )
 * @property {Boolean}			error			当前步骤是否处于失败状态  (默认 false )
 * @example <u-steps current="0"><u-steps-item title="已出库" desc="10:35" ></u-steps-item></u-steps>
 */
export default {
  name: 'u-steps-item',
  mixins: [mpMixin, mixin, props],
  data() {
    return {
      index: 0,
      childLength: 0,
      showLine: false,
      size: {
        height: 0,
        width: 0
      },
      parentData: {
        direction: 'row',
        current: 0,
        activeColor: '',
        inactiveColor: '',
        activeIcon: '',
        inactiveIcon: '',
        dot: false
      }
    }
  },
  watch: {
    'parentData'(newValue, oldValue) {
    }
  },
  created() {
    this.init()
  },
  // #ifdef MP-TOUTIAO
  options: {
    virtualHost: false
  },
  // #endif
  computed: {
    lineStyle() {
      const style = {}
      if (this.parentData.direction === 'row') {
        style.width = this.size.width + 'px'
        style.left = this.size.width / 2 + 'px'
      } else {
        style.height = this.size.height + 'px'
        // style.top = this.size.height / 2 + 'px'
      }
      style.backgroundColor = this.parent.children?.[this.index + 1]?.error ? color.error : this.index <
      this
          .parentData
          .current ? this.parentData.activeColor : this.parentData.inactiveColor
      return style
    },
    itemStyleInner() {
      return {
        ...this.itemStyle
      }
    },
    statusClass() {
      const {
        index,
        error
      } = this
      const {
        current
      } = this.parentData
      if (current == index) {
        return error === true ? 'error' : 'process'
      } else if (error) {
        return 'error'
      } else if (current > index) {
        return 'finish'
      } else {
        return 'wait'
      }
    },
    statusColor() {
      let colorTmp = ''
      switch (this.statusClass) {
        case 'finish':
          colorTmp = this.parentData.activeColor
          break
        case 'error':
          colorTmp = color.error
          break
        case 'process':
          colorTmp = this.parentData.dot ? this.parentData.activeColor : 'transparent'
          break
        default:
          colorTmp = this.parentData.inactiveColor
          break
      }
      return colorTmp
    },
    contentStyle() {
      const style = {}
      if (this.parentData.direction === 'column') {
        style.marginLeft = this.parentData.dot ? '2px' : '6px'
        style.marginTop = this.parentData.dot ? '0px' : '0px'
      } else {
        style.marginTop = this.parentData.dot ? '2px' : '0px'
        style.marginLeft = this.parentData.dot ? '2px' : '6px'
      }

      return style
    }
  },
  mounted() {
    this.parent && this.parent.updateFromChild()
    sleep().then(() => {
      this.getStepsItemRect()
    })
  },
  methods: {
    init() {
      // 初始化数据
      this.updateParentData()
      if (!this.parent) {
        return error('u-steps-item必须要搭配u-steps组件使用')
      }
      this.index = this.parent.children.indexOf(this)
      this.childLength = this.parent.children.length
    },
    updateParentData() {
      // 此方法在mixin中
      this.getParentData('u-steps')
    },
    // 父组件数据发生变化
    updateFromParent() {
      this.init()
    },
    // 获取组件的尺寸,用于设置横线的位置
    getStepsItemRect() {
      // #ifndef APP-NVUE
      this.$uGetRect('.u-steps-item').then(size => {
        this.size = size
      })
      // #endif

      // #ifdef APP-NVUE
      dom.getComponentRect(this.$refs['u-steps-item'], res => {
        const {
          size
        } = res
        this.size = size
      })
      // #endif
    }
  }
}
</script>

<style lang="scss" scoped>

.u-steps-item {
  flex: 1;
  @include flex;

  &--row {
    flex-direction: column;
    align-items: center;
    position: relative;
  }

  &--column {
    position: relative;
    flex-direction: row;
    justify-content: flex-start;
    padding-bottom: 5px;
  }

  &__wrapper {
    @include flex;
    justify-content: center;
    align-items: center;
    position: relative;
    background-color: #fff;
    border-radius: 50px;

    &--column {
      width: 20px;
      height: 20px;

      &--dot {
        height: 20px;
        width: 20px;
      }
    }

    &--row {
      width: 20px;
      height: 20px;

      &--dot {
        width: 20px;
        height: 20px;
      }
    }

    &__circle {
      width: 20px;
      height: 20px;
      /* #ifndef APP-NVUE */
      box-sizing: border-box;
      flex-shrink: 0;
      /* #endif */
      border-radius: 100px;
      border-width: 1px;
      border-color: $u-tips-color;
      border-style: solid;
      @include flex(row);
      align-items: center;
      justify-content: center;
      transition: background-color 0.3s;

      &__text {
        color: $u-tips-color;
        font-size: 11px;
        @include flex(row);
        align-items: center;
        justify-content: center;
        text-align: center;
        line-height: 11px;
      }
    }

    &__dot {
      width: 10px;
      height: 10px;
      border-radius: 100px;
      background-color: $u-content-color;
    }
  }

  &__content {
    @include flex;
    flex: 1;

    &__title {
      // #ifdef H5
      cursor: pointer;
      // #endif
    }

    &--row {
      flex-direction: column;
      align-items: center;
    }

    &--column {
      flex-direction: column;
      margin-left: 6px;
    }
  }

  &__line {
    position: absolute;
    background: $u-tips-color;

    &--row {
      top: 10px;
      height: 1px;
    }

    &--column {
      width: 1px;
      left: 10px;
    }
  }
}
</style>