Vue组件开发“干~倒”尤雨溪

作为一个有梦想的前端。我们的目标是:
向尤雨溪学习,“干~倒”尤雨溪。

“尤雨溪”将会出一系列的内容。对于vue相关的思想会较为深度的体验。

又又又发现了一种开发组件的方式,虽然很多人已经这样开发了。但是作为菜鸡的发现者,依然觉得很香,之前说过通过遍历solt。然后劫持监听事件。然后通过ref进行组件的控制。这种方式,代码复杂,灵活性不高。

今天使用bus来封装组件。对你没听错,就是耳熟能详的bus(对于不知道bus的可以自己百度)。

其实使用bus是对于很多需求,很多联动组件。就需要使用bus来控制。本次就拿比较常见的。开始日期与结束日期来举例。

“干倒”尤雨溪 – 懋官的个人博客 (hengweishiye.top)

用法

    <StartDate v-model="startDate1" />
    <EndDate v-model="endDate1" />

效果图

是不是用起来极其的简单?是不是觉得还能这样?

没错的,这只是其中一种的用法,为你打开组件开发的大门。

StartDate组件

<template>
  <a-date-picker :placeholder="placeholder" :value="value" @change="change" valueFormat="YYYY-MM-DD" :disabled-date="disabledStartDate"
                 :disabled="disabled" :allowClear="allowClear"/>
</template>

<script lang="ts">
import {defineComponent, onUnmounted, ref, unref, watch} from "@vue/composition-api";
import moment from "moment";
import {$bus, START_VALUE, END_VALUE} from "./hooks/bus";

export default defineComponent({
  name: "StartDate",
  model: {
    prop: 'value',
    event: 'change'
  },
  props: {
    placeholder: String,
    value: {
      type: String,
      required: true
    },
    /**
     * 开始日期的监听symbol 同组件中使用多个 开始日期组件。需要用传入不同的symbol
     */
    startSymbol: {
      type: Symbol,
      default: START_VALUE
    },
    /**
     * 结束日期的监听symbol 同组件中使用多个 结束日期组件。需要用传入不同的symbol
     */
    endSymbol: {
      type: Symbol,
      default: END_VALUE
    },
    disabled: {
      type: Boolean,
      default: false
    },
    allowClear: {
      type: Boolean,
      default: false
    },
    /**
     * 是否可以等于
     */
    eq: {
      type: Boolean,
      default: false
    }
  },
  setup(props, {emit}) {
    const endValue = ref('')

    const change = (value: string) => {
      emit('change', value)
    }

    watch(() => props.value, (value: string) => {
      setTimeout(() => $bus.$emit(props.startSymbol, value))
    }, {
      immediate: true
    })

    $bus.$on(props.endSymbol, (value: string) => {
      endValue.value = value;
    })

    const disabledStartDate = (startValue: string) => {
      if (!startValue || !unref(endValue)) {
        return false;
      }
      if(props.eq) {
        return moment(startValue).valueOf() > moment(unref(endValue)).valueOf();
      }
      return moment(startValue).endOf('day').valueOf() >= moment(unref(endValue)).endOf('day').valueOf();
    }

    onUnmounted(() => {
      $bus.$off(props.startSymbol)
      $bus.$off(props.endSymbol)
    })

    return {
      change,
      disabledStartDate,
      endValue
    }
  }
})
</script>

EndDate组件

<template>
  <a-date-picker :placeholder="placeholder" :value="value" @change="change" valueFormat="YYYY-MM-DD" :disabled-date="disabledEndDate"
                 :disabled="disabled" :allowClear="allowClear"/>
</template>

<script lang="ts">
import {defineComponent, onUnmounted, ref, unref, watch} from "@vue/composition-api";
import moment from "moment";
import {$bus, START_VALUE, END_VALUE} from "./hooks/bus";

export default defineComponent({
  name: "StartDate",
  model: {
    prop: 'value',
    event: 'change'
  },
  props: {
    placeholder: String,
    value: {
      type: String,
      required: true
    },
    startSymbol: {
      type: Symbol,
      default: START_VALUE
    },
    endSymbol: {
      type: Symbol,
      default: END_VALUE
    },
    disabled: {
      type: Boolean,
      default: false
    },
    allowClear: {
      type: Boolean,
      default: false
    },
    /**
     * 是否可以等于
     */
    eq: {
      type: Boolean,
      default: false
    }
  },
  setup(props, {emit}) {
    const startValue = ref('');

    const change = (value: string) => {
      emit('change', value)
    }

    watch(() => props.value, (value: string) => {
      setTimeout(() => $bus.$emit(props.endSymbol, value))
    }, {
      immediate: true
    })

    $bus.$on(props.startSymbol, (value: string) => {
      startValue.value = value;
    })

    onUnmounted(() => {
      $bus.$off(props.startSymbol)
      $bus.$off(props.endSymbol)
    })

    const disabledEndDate = (endValue: string) => {
      if (!endValue || !unref(startValue)) {
        return false;
      }
      if(props.eq) {
        return moment(unref(startValue)).endOf('day').valueOf() > moment(endValue).endOf('day').valueOf();
      }
      return moment(unref(startValue)).endOf('day').valueOf() >= moment(endValue).endOf('day').valueOf();
    }

    return {
      change,
      disabledEndDate,
      startValue
    }
  }
})
</script>

bus

// @ts-nocheck 为保持和vue2版本中使用bus一致,emit,on,off前面都加了$
export type EventType = string | symbol;
export type Handler<T = any> = (event: T) => void;
export type EventMap = { [key: EventType]: Array<Handler> };
class Bus {
    private readonly list: EventMap;
    constructor() {
        // 收集订阅信息,调度中心
        this.list = {} as EventMap;
    }

    // 订阅
    $on(name: EventType, fn: Handler) {
        this.list[name] = this.list[name] || [];
        this.list[name].push(fn);
    }

    // 发布
    $emit(name: EventType, data?: any) {
        if (this.list[name]) {
            this.list[name].forEach((fn: Handler) => {
                fn(data);
            });
        }
    }

    // 取消订阅
    $off(name: EventType) {
        if (this.list[name]) {
            delete this.list[name];
        }
    }
}
export default Bus;

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值