今回は、Vue Mixin 機能を使用する手順について詳しく説明します。Vue Mixin 機能を使用する際の 注意事項 について、実際のケースを見てみましょう。
Typescript を使用して Vue アプリケーションを作成することに切り替えた後、ツールチェーンと依存関係を一巡した後、ついにつまずき始めました。しかし、非常に一般的に使用される関数 mixin があり、正式な解決策はまだないようです。 ミックスインの柔軟性と利便性を享受したいだけでなく、ts 型システムによってもたらされる開発中の IntelliSense の使用のセキュリティ保証とスムーズなエクスペリエンスも享受したいと考えています。 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
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'
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' }
rrreee
ABC に関しては、これは純粋に手動によるジャンル宣言の練習です。 使用法 実際の使用方法: rrreee
Vue を直接拡張する DisposableMixin は有効な Vue コンポーネントではなく、また、使用する場合には mixins オプションで直接使用することもできないことに注意してください。 Vue として使用されます。自己定義コンポーネントを拡張するには、必ず Component を使用してください。
抽象クラス ビジネス システムで使用されるミックスインは、ほとんどの場合、実際にはより複雑で、いくつかの基本的な機能を提供しますが、一部の部分は後継者に任せて実装する必要があります。この場合、 抽象クラス
は非常に適しています。
{ new(): T }
の要件を満たしていないため、継承のみが可能で、混合することはできません。同じ理由で、抽象クラスも'vue-class-component'
の Component 関数でクラスを装飾することはできません。 🎜🎜この時、実装する関数をMixinに記述し、実装したい関数をインターフェースに入れて特定のクラスに実装させる必要があります。 🎜rrreee🎜このコンパイラを欺く方法は、実際には非常に不格好です。特定のクラスが PlayerMixin を継承していても、IPlayerImplementation の実装を明示的に宣言していない場合、コンパイラはこのエラーを通知できません。私たちができることは、コード内にコメントを慎重に記述することだけであり、ユーザーがこれを忘れないことを願っています。 🎜🎜この記事の事例を読んだ後は、この方法を習得したと思います。さらに興味深い情報については、php 中国語 Web サイトの他の関連記事に注目してください。 🎜🎜推奨読書: 🎜🎜🎜ページを共有した後にホームページにリダイレクト🎜🎜🎜🎜🎜Vueでのwatchメソッドの使用方法の詳細な説明🎜🎜🎜以上がVue Mixin機能の利用手順を詳しく解説の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。