>웹 프론트엔드 >JS 튜토리얼 >Vue Mixin 기능을 사용하는 단계에 대한 자세한 설명

Vue Mixin 기능을 사용하는 단계에 대한 자세한 설명

php中世界最好的语言
php中世界最好的语言원래의
2018-04-28 11:47:162874검색

이번에는 Vue Mixin 기능을 사용하는 단계에 대해 자세히 설명하겠습니다. Vue Mixin 기능을 사용할 때 주의 사항은 무엇입니까?

Vue 애플리케이션을 작성하기 위해 Typescript를 사용하도록 전환한 후 일련의 도구 체인과 종속성을 거쳐 마침내 비틀거리기 시작했습니다. 그러나 매우 일반적으로 사용되는 함수 mixin이 있으며 아직 공식적인 솔루션은 없는 것 같습니다.

믹스인의 유연성과 편리함을 즐기고 싶지만, TS 유형 시스템이 제공하는 개발 중 IntelliSense 사용의 보안보증과 원활한 경험도 얻고 싶습니다.

Vuejs 공식 조직에는 'vue-class-comComponent'와 권장 'vue-property- decorator'가 있지만 해당 구현이 없습니다. 이전 이슈를 살펴보니 오랫동안 보류되었던 기능이 있는데 바로 믹스인 지원입니다. '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-comComponent 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에서 확장 메서드를 호출하면 새로운 하위 클래스가 생성됩니다. 하위 클래스 생성자.

rrreee

ABC의 경우 이것은 순전히 장르 선언의 수동 연습입니다.

사용법
실제 사용:

rrreee

Vue를 직접 확장하는 DisposableMixin은 유효한 Vue 구성 요소가 아니며 mixins 옵션에서 직접 사용할 수 없습니다. 자체 정의된 구성 요소를 확장하려면 확장 메서드를 사용하여 이를 마무리해야 합니다. rrreeeAbstract class

비즈니스 시스템에서 사용되는 Mixin은 실제로 대부분의 경우 더 복잡하고 몇 가지 기본 기능을 제공하지만, 일부 부분은 후임자가 직접 구현하도록 남겨야 합니다. 이 경우 추상 수업🎜이 매우 적합합니다. 🎜rrreee🎜그러나 추상 클래스는 인스턴스화할 수 없고 { new(): T }의 요구 사항을 충족하지 않으므로 상속만 가능하고 혼합될 수 없습니다. 같은 이유로 추상 클래스는 'vue-class-comComponent'의 Component 함수로 클래스를 장식할 수 없습니다. 🎜🎜이때 구현된 함수를 Mixin에 작성하고, 구현하려는 함수를 인터페이스에 넣어서 특정 클래스에서 구현하도록 해야 합니다. 🎜rrreee🎜컴파일러를 속이는 이 방법은 실제로 매우 서투릅니다. 특정 클래스가 PlayerMixin을 상속하지만 IPlayerImplementation을 구현하도록 명시적으로 선언하지 않으면 컴파일러는 이 오류를 알려줄 수 없습니다. 우리는 코드에 주석을 주의 깊게 작성할 수 있으며 사용자가 이를 잊지 않기를 바랍니다. 🎜🎜이 기사의 사례를 읽은 후 방법을 마스터했다고 생각합니다. 더 흥미로운 정보를 보려면 PHP 중국어 웹사이트의 다른 관련 기사를 주목하세요! 🎜🎜추천도서: 🎜🎜🎜페이지 공유 후 홈페이지로 리디렉션🎜🎜🎜🎜🎜Vue에서 watch 방식 사용에 대한 자세한 설명🎜🎜🎜

위 내용은 Vue Mixin 기능을 사용하는 단계에 대한 자세한 설명의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.