ホームページ  >  記事  >  ウェブフロントエンド  >  Vue Mixin機能のユースケースを詳しく解説

Vue Mixin機能のユースケースを詳しく解説

php中世界最好的语言
php中世界最好的语言オリジナル
2018-06-08 10:47:431780ブラウズ

今回は、Vue Mixin 機能の使用例について詳しく説明します。Vue Mixin 機能の使用例の注意点は次のとおりです。

vuejs 公式組織には 'vue-class-component' と推奨される 'vue-property-decorator' がありますが、対応する実装はありません。前の問題に目を通した後、長い間保留されていた機能があり、それは mixin のサポートです。 '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 的类构造器。同时通过 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'
@Component
class 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: `
 <p>
  <h1>{{ title }}</h1>
  <p>Counted: {{ counter }}</p>
 </p>
 `
})
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 1
count to 2
count to 3
calling reidrectTo /otherpage
about 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'

複雑ではありません。自分で書くだけです。

後記: vue-class-component 6.2.0 では、この記事の実装アイデアと同様の mixins メソッドが提供され始めます。

implementation

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'
}
は VClass を T のクラス コンストラクターとして宣言します。同時に、Vue のコンストラクター上の静的メソッド (extend/mixin など) が Pick を通じて取得されるため、次の段落の実際の実装がサポートされます。 Vue 上で extend メソッドを呼び出すことによって、新しいサブクラスが生成されます。サブクラスのコンストラクター。

rrreee

ABC に関しては、これは純粋に手動によるジャンル宣言の練習です。

使用法
実際の使用方法:

rrreee

Vue を直接拡張する DisposableMixin は有効な Vue コンポーネントではなく、また、使用する場合には mixins オプションで直接使用することもできないことに注意してください。 Vue として使用されます。カスタム コンポーネントを拡張するには、必ず Component でラップしてください。 rrreee抽象クラス

ビジネス システムで使用される Mixin は実際にはより複雑で、いくつかの基本的な機能を提供しますが、一部の部分は継承者に任せて実装する必要があります。この場合、抽象クラスを使用するのが適切です。 。 🎜rrreee🎜しかし、抽象クラスはインスタンス化できず、 { new(): T } の要件を満たしていないため、継承のみが可能で、混合することはできません。同じ理由で、抽象クラスも'vue-class-component' の Component 関数でクラスを装飾することはできません。 🎜🎜この時、実装する関数をMixinに記述し、実装したい関数をインターフェースに入れて特定のクラスに実装させる必要があります。 🎜rrreee🎜このコンパイラを欺く方法は、実際には非常に不格好です。特定のクラスが PlayerMixin を継承していても、IPlayerImplementation の実装を明示的に宣言していない場合、コンパイラはこのエラーを通知できません。私たちができることは、コード内にコメントを慎重に記述することだけであり、ユーザーがこれを忘れないことを願っています。 🎜🎜この記事の事例を読んだ後は、その方法を習得したと思います。さらに興味深い情報については、php 中国語 Web サイトの他の関連記事に注目してください。 🎜🎜推奨書籍: 🎜🎜🎜Progressbarコンポーネントの実践的な事例分析🎜🎜🎜🎜🎜アドレス帳の頭文字検索機能の実装🎜🎜🎜

以上がVue Mixin機能のユースケースを詳しく解説の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。