Vue mixin (应用篇)

Vue混入详解
本文深入讲解Vue中的混入(mixin)概念,展示了如何利用混入实现组件间的代码复用,特别是在组件拖拽功能上的应用。并通过实例说明了局部混入的使用方法,以及混入与Vuex在变量共享方面的区别。

vue官方文档中是这么解释的:

  混入 (mixin) 提供了一种非常灵活的方式,来分发 Vue 组件中的可复用功能。一个混入对象可以包含任意组件选项。当组件使用混入对象时,所有混入对象的选项将被“混合”进入该组件本身的选项。

人话:

       Vue.mixin给我们提供了一种混入Vue实例的方法,创建了混入对象之后,我们自定义的方法或者变量可以很轻松的挂载在Vue实例上。

Vue.mixin为我们提供了两种混入方式:局部混入和全局混入;

      这里只讲局部混入

首先我们在src目录下新建一个名为mixin的文件夹并在mixin文件中创建一个drag.js文件:

这个文件用来做组件拖拽

// dom.js

// 获取元素的位置

export function getOffset (curEle) {

  let totalLeft = null

  let totalTop = null

  let par = curEle.offsetParent

  // 首先把自己本身的进行累加

  totalLeft += curEle.offsetLeft

  totalTop += curEle.offsetTop

 

  // 只要没有找到body,我们就把父级参照物的边框和偏移量累加

  while (par) {

    if (navigator.userAgent.indexOf('MSIE 8.0') === -1) {

      // 不是标准的ie8浏览器,才进行边框累加

      // 累加父级参照物边框

      totalLeft += par.clientLeft

      totalTop += par.clientTop

    }

    // 累加父级参照物本身的偏移

    totalLeft += par.offsetLeft

    totalTop += par.offsetTop

    par = par.offsetParent

  }

  return { left: totalLeft, top: totalTop }

}

 

// drag.js

import { getOffset } from '@/utils/dom'

// let distenceX, distenceY, clickCicleTime

 

// 抛出混入对象,方便外部访问

export const mixin = {

  data() {

    return {

      number: 1,

      distenceX: '',

      distenceY: '',

      clickCicleTime: ''

    }

  },

  methods: {

    dragMouseDown(e, clickTarget) {

      if (clickTarget) {

        this.clickCicleTime = +new Date()

      }

      var positionDiv = getOffset(this.$el)

      this.distenceX = e.pageX - positionDiv.left

      this.distenceY = e.pageY - positionDiv.top

      document.addEventListener('mousemove', this.dragMouseMove)

      document.addEventListener('mouseup', this.dragMouseup)

    },

    dragMouseup(e, clickTarget) {

      if (clickTarget) {

        let _clickCicleTime = +new Date()

        let differTime = _clickCicleTime - this.clickCicleTime

        if (differTime < 200) {

          this.isZoomOut = false

        }

      }

      document.removeEventListener('mousemove', this.dragMouseMove)

    },

    dragMouseMove(e) {

      let $outWrap = this.$el

      var r =

        document.documentElement.clientWidth -

        e.clientX -

        ($outWrap.offsetWidth - this.distenceX) -

        document.documentElement.scrollLeft

      var y = e.pageY - this.distenceY

      if (r < 0) {

        r = 0

      } else if (

        r >

        document.documentElement.clientWidth - $outWrap.offsetWidth

      ) {

        r = document.documentElement.clientWidth - $outWrap.offsetWidth

      }

      if (y < 0) {

        y = 0

      } else if (

        y >

        document.documentElement.clientHeight - $outWrap.offsetHeight

      ) {

        y = document.documentElement.clientHeight - $outWrap.offsetHeight

      }

      $outWrap.style.right = r + 'px'

      $outWrap.style.top = y + 'px'

    }

  }

}

前置文件准备好后,下面是mixin在vue的使用

//  xxx.vue

import { mixin } from '@/mixin/drag'

export default {

  name: 'tip-pl',

  components: { swiper, swiperSlide },

  mixins: [mixin],

  data() {

    return {

      BEM_NAMESPACE: 'tip-pl',

      specialData: [],

      TOY_MODES_ICO_LIST,

      isShowSwiper: true,

      isClosePanel: false,

      panelSetting: this.$$lvsGlobelData.panelSetting,

      normalSettingData: {

        itemListNum: 5

      },

      swiperOption: {

        loop: true,

        speed: 1000,

        autoplay: autoplay

      },

      basicData: []

    }

  },

.......

这样混入mixin就完成了。至于变量是否公用,mixin与vuex的区别,这里有几个关键点

1.mixin混入对象的变量是不会共享的

2.例如methods,components等如果变量名和mixin混入对象的变量名发生冲突,将会以组件优先并进行递归合并,相当于组件数据直接覆盖了mixin中的同名数据;

3.

mixin混入对象和Vuex的区别:

    Vuex是状态共享管理,所以Vuex中的所有变量和方法都是可以读取和更改并相互影响的;

    mixin可以定义公用的变量或方法,但是mixin中的数据是不共享的,也就是每个组件中的mixin实例都是不一样的,都是单独存在的个体,不存在相互影响的;

    mixin混入对象值为函数的同名函数选项将会进行递归合并为数组,两个函数都会执行,只不过先执行mixin中的同名函数;

    mixin混入对象值为对象的同名对象将会进行替换,都优先执行组件内的同名对象,也就是组件内的同名对象将mixin混入对象的同名对象进行覆盖;

参考资料:

https://www.cnblogs.com/dengyao-blogs/p/11589962.html

<think>我们正在解决Vuemixin无效的问题。根据引用[1],有时需要扩展Vue的功能或将选项应用应用程序中所有Vue组件。在这种情况下,mixin可以全局应用以影响Vue中的所有组件,这些mixin称为全局mixin。但是,如果mixin没有按预期工作,我们需要检查几个方面。首先,确保mixin的注册方式正确:1.对于局部mixin,在组件选项中使用`mixins`数组引入。2.对于全局mixin,使用`Vue.mixin`方法(注意:全局mixin会影响每一个之后创建的Vue实例,包括第三方组件,因此要谨慎使用)。其次,检查mixin和组件选项的合并策略。当组件和mixin包含相同的选项(如`data`、`methods`等)时,Vue有特定的合并策略:-`data`函数返回的对象会进行递归合并,并在发生冲突时以组件数据优先。-生命周期钩子函数会被合并到一个数组,并且mixin的钩子将在组件自身钩子之前调用。-值为对象的选项(如`methods`、`components`等)将被合并为同一个对象,如果键名冲突,则组件的选项会覆盖mixin的选项。因此,如果mixin中的某个方法或数据没有被调用,可能是因为组件中定义了同名的属性或方法,导致被覆盖。另外,注意mixin中的生命周期钩子会在组件的钩子之前执行,所以如果mixin的钩子依赖于组件的状态,可能那时组件还未初始化。最后,检查mixin的代码是否有错误,比如拼写错误等。下面是一个示例,展示如何正确使用局部mixin:```javascript//定义一个mixinvarmyMixin={created:function(){this.hello()},methods:{hello:function(){console.log('hellofrommixin!')}}}//在组件中使用newVue({mixins:[myMixin],created:function(){console.log('组件钩子被调用')},methods:{//如果这里也有一个hello方法,则会覆盖mixin中的hello}})//输出顺序://hellofrommixin!//组件钩子被调用```如果问题仍然存在,请提供更多的代码细节以便进一步排查。§§相关问题§§1.Vuemixin和组件的选项合并规则是什么?2.如何避免全局mixin带来的潜在问题?3.Vue3中,mixin的使用有什么变化?</think>### Vue Mixin 无效的解决方案 在 Vue 中遇到 mixin 无效的问题,通常由以下原因导致,下面是系统排查步骤和解决方案: #### 1. **注册方式错误** - **局部 mixin** 必须在组件中显式声明: ```javascript import myMixin from './myMixin' export default { mixins: [myMixin], // ✅ 正确注册 // ... } ``` - **全局 mixin** 需在 `main.js` 中注册: ```javascript import Vue from 'vue' import globalMixin from './globalMixin' Vue.mixin(globalMixin) // ✅ 全局生效 ``` > 错误示例:直接导入但未在 `mixins` 数组中声明会导致无效[^1]。 #### 2. **命名冲突** Vue 的选项合并策略优先级:**组件选项 > 局部 mixin > 全局 mixin** - 当组件和 mixin 有同名属性时,组件属性会覆盖 mixin - 解决方案: ```javascript // mixin.js export default { methods: { // 添加唯一前缀避免冲突 __mixin_uniqueMethod() { ... } } } ``` #### 3. **生命周期钩子执行顺序** mixin 的生命周期钩子会**先于**组件自身的钩子执行: ```javascript // mixin.js export default { mounted() { console.log('Mixin mounted') // 先执行 } } // Component.vue mounted() { console.log('Component mounted') // 后执行 } ``` > 若依赖组件数据,需在 `nextTick` 中处理异步逻辑。 #### 4. **响应式数据未正确声明** - 在 mixin 中定义响应式数据需使用 **函数返回对象**: ```javascript // ✅ 正确 data() { return { mixinData: 'value' } } // ❌ 错误(非响应式) data: { mixinData: 'value' } ``` #### 5. **作用域限制** - **全局 mixin** 会影响所有组件(慎用),引用[1]指出它适用于"extend functionality to all components"[^1] - **局部 mixin** 需确保导入路径正确: ```javascript // ❌ 路径错误导致无效 import myMixin from './wrongPath/mixin' ``` #### 6. Vue 3 Composition API 替代方案 在 Vue 3 中推荐使用 `composables` 替代 mixin: ```javascript // useFeature.js import { ref } from 'vue' export function useFeature() { const count = ref(0) const increment = () => count.value++ return { count, increment } } // Component.vue import { useFeature } from './useFeature' setup() { const { count, increment } = useFeature() return { count, increment } } ``` > 避免命名冲突且类型推断更友好[^3]。 ### 调试建议 1. 检查浏览器控制台是否有"Property/method not defined"警告 2. 使用 Vue Devtools 查看组件的合并后选项 3. 简化 mixin 到最小可复现代码排查 > 通过以上步骤,90% 的 mixin 失效问题可解决。若涉及特定 UI 库(如引用[3]提到的表单组件[^2]),需检查库的扩展机制是否支持 mixin
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值