虚位以待(AD)
虚位以待(AD)
首页 > 网页特效 > JavaScript > 在 Typescript 中使用可被复用的 Vue Mixin功能

在 Typescript 中使用可被复用的 Vue Mixin功能
类别:JavaScript   作者:码皇   来源:互联网   点击:

这篇文章主要介绍了在 Typescript 中使用可被复用的 Vue Mixin功能,非常不错,具有参考借鉴价值,需要的朋友可以参考下

转到用 Typescript 写 Vue 应用以后,经过一轮工具链和依赖的洗礼,总算蹒跚地能走起来了,不过有一个很常用的功能 mixin,似乎还没有官方的解决方案。

既想享受 mixin 的灵活和方便,又想收获 ts 的类型系统带来的安全保障和开发时使用 IntelliSense 的顺滑体验。

vuejs 官方组织里有一个 'vue-class-component' 以及连带推荐的 'vue-property-decorator',都没有相应实现。翻了下前者的 issue,有一条挂了好些时间的待做 feature 就是 mixin 的支持。

也不是什么复杂的事,自己写一个吧。

后注:vue-class-component 6.2.0 开始提供 mixins 方法,和本文的实现思路相似。

实现

    import Vue, {
    VueConstructor }
    from 'vue'export type VClass<T> = {
    new(): T}
    & Pick<VueConstructor, keyof VueConstructor>/** * mixins for class style vue component */function Mixins<A>(c: VClass<A>): VClass<A>function Mixins<A, B>(c: VClass<A>, c1: VClass<B>): VClass<A&B>function Mixins<A, B, C>(c: VClass<A>, c1: VClass<B>, c2: VClass<C>): VClass<A&B&C>function Mixins<T>(c: VClass<T>, ...traits: Array<VClass<T>>): VClass<T> {
    return c.extend({
    mixins: traits }
    )}

声明 VClass<T> 可作为 T 的类构造器。同时通过 Pick 拿到 Vue 的构造器上的静态方法(extend/mixin 之类),如此才能够支持下面这段中的真正实现,通过调用一个 Vue 的子类构造器上的 extend 方法生成新的子类构造器。

    function Mixins<T>(c: VClass<T>, ...traits: Array<VClass<T>>): VClass<T> {
    return c.extend({
    mixins: traits }
    )}

至于 ABC 这个纯粹是类型声明的体力活了。

使用

实际使用时:

    import {
    Component, Vue }
    from 'vue-property-decorator'import {
    Mixins }
    from '../../util/mixins'@Componentclass PageMixin extends Vue {
    title = "Test Page" redirectTo(path: string) {
    console.log('calling reidrectTo', path) this.$router.push({
    path }
    ) }
    }
    interface IDisposable {
    dispose(...args: any[]): any}
    class DisposableMixin extends Vue {
    _disposables: IDisposable[] created() {
    console.log('disposable mixin created');
    this._disposables = [] }
    beforeDestroy() {
    console.log('about to clear disposables') this._disposables.map((d) => {
    d.dispose() }
    ) delete this._disposables }
    registerDisposable(d: IDisposable) {
    this._disposables.push(d) }
    }
    @Component({
    template: ` <div> <h1>{
    {
    title }
    }
    </h1> <p>Counted: {
    {
    counter }
    }
    </p> </div> `}
    )export default class TimerPage extends Mixins(PageMixin, DisposableMixin) {
    counter = 0 mounted() {
    const timer = setInterval(() => {
    if (this.counter++ >= 3) {
    return this.redirectTo('/otherpage') }
    console.log('count to', this.counter);
    }
    , 1000) this.registerDisposable({
    dispose() {
    clearInterval(timer) }
    }
    ) }
    }
    count to 1count to 2count to 3calling reidrectTo /otherpageabout to clear disposables

注意到直接 extends Vue 的 DisposableMixin 并不是一个有效的 Vue 组件,也不可以直接在 mixins 选项里使用,如果要被以 Vue.extend 方式扩展的自定义组件使用,记住使用 Component 包装一层。

    const ExtendedComponent = Vue.extend({
    name: 'ExtendedComponent', mixins: [Component(DisposableMixin)],}
    )

Abstract class

在业务系统中会使用到的 Mixin 其实多数情况下会更复杂,提供一些基础功能,但有些部分需要留给继承者自行实现,这个时候使用抽象类就很合适。

    abstract class AbstractMusicPlayer extends Vue {
    abstract audioSrc: string playing = false togglePlay() {
    this.playing = !this.playing }
    }
    class MusicPlayerA extends AbstractMusicPlayer {
    audioSrc = '/audio-a.mp3'}
    class MusicPlayerB extends AbstractMusicPlayer {
    staticBase = '/statics' get audioSrc() {
    return `${
    this.staticBase}
    /audio-b.mp3` }
    }

但抽象类是无法被实例化的,并不满足 { new(): T } 这个要求,因此只能被继承,而不能被混入,由于同样的原因,抽象类也无法被 'vue-class-component' 的 Component 函数装饰。

这时候只好将实现了的功能写入 Mixin 中,待实现的功能放到接口里,让具体类来实现。

    interface IMusicSourceProvider {
    audioSrc: string}
    /** * @implements IPlayerImplementation */class PlayerMixin extends Vue {
    /** @abstract */ audioSrc: string logSrc() {
    console.log(this.audioSrc) }
    }
    interface IPlayerImplementation extends IMusicSourceProvider {
    }
    class RealPlayer extends Mixins(PlayerMixin) implements IPlayerImplementation {
    audioSrc = '/audio-c.mp3'}

这种欺骗编译器的方式其实还是比较拙劣的,如果一个具体类继承了 PlayerMixin,却没有显示声明实现 IPlayerImplementation ,编译器无法告诉你这个错误。我们只能在代码里小心翼翼写上注释,期待使用者不要忘了这件事。

总结

以上所述是小编给大家介绍的在 Typescript 中使用可被复用的 Vue Mixin功能,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对脚本之家网站的支持!

您可能感兴趣的文章:

  • Vue使用mixins实现压缩图片代码
  • 详解Vue 中 extend 、component 、mixins 、extends 的区别
  • 详解vue mixins和extends的巧妙用法
  • 谈谈vue中mixin的一点理解
  • 详解Vue.js Mixins 混入使用
  • 在Vue.js中使用Mixins的方法
  • 详解Vue.js组件可复用性的混合(mixin)方式和自定义指令
  • vue mixins组件复用的几种方式(小结)
相关热词搜索: vue mixin vue typescript