Vue2实现子组件改变父子组件的数据

简介: 制作组件的时候大部分情况需要在子组件内部改变父组件的数据,但是由于vue的限制,在子组件改变父组件的state会报错,本文介绍如何利用v-model实现父子组件的数据双向绑定。

制作组件的时候大部分情况需要在子组件内部改变父组件的数据,但是由于vue的限制,在子组件改变父组件的state会报错,本文介绍如何利用v-model实现父子组件的数据双向绑定。

本文环境
- vue2(2.5+)
- vue-class-component
- vue-property-decorator

大致思路是用input的value承载父组件的数据,然后使用v-model实现双向绑定数据。

首先把父组件声传入的值声明为prop

// vue-class-component 写法
@Component({
  model: {
    prop: 'val'
  }
})
export default class Slidr extends Vue {
  // ...
  @Prop()
  val: number
  // ...
}

然后在子组件添加一个input标签,然后将props绑定上去。

// 将val绑定在input的value上,并给input事件添加函数
<input style="display: none;"
  :value="val"
  @input="changeVal"/>

然后要改变父组件绑定的值的时候,用自定义事件去改变

// 当input的值改变时触发
@Emit('input')
  changeVal(val: number) { }
// 要改变父组件的值时使用自定义事件函数,这样不会报错
this.changeVal(5)

还可以封装当值改变时的change事件,绑定在父组件上就可在val改变时触发

// 自定义事件change
@Emit('change')
  outVal(val: number) { }
// 改写事件触发时执行的函数
@Emit('input')
  changeVal(val: number) { this.outVal(val) }

然后就可以自己封装组件了


下面是一个数据双向绑定的slider组件,可以参考一下

<template>
<section class="slider-box"
ref="slider"
:style="{
  width: wid,
  opacity: disabled ? 0.5 : 1}">

  <div class="slider-line"
  ref="line"
  :style="lineSty"></div>

  <div class="slider-line_choose"
  :style="lineChooseSty"></div>

  <transition name="tag">
    <div class="tag"
    v-if="showTag && isShowTag"
    :style="tagStyle">{{tagPrompt || val}}</div>
  </transition>


  <div
  class="slider-btn"
  @touchstart="touchStart"
  @touchend="touchEnd"
  :class="{
    enter: isTouch,
    out: !isTouch
  }"
  :style="[{
    borderColor: chooseColor},
    btnPosit]"></div>

  <!-- 使用input做到prop数据的双向绑定,在子组件改变父组件传入的prop的值不报错 -->
  <input class="slider-val"
  :value="val"
  @input="changeVal"/>
</section>
</template>

<script lang="ts">
import Vue from 'vue'
import Component from 'vue-class-component'
import { Prop, Emit } from 'vue-property-decorator'
import { throttle } from 'lodash'
import $ from 'jquery'

@Component({
  components: {
  },
  model: {
    prop: 'val'
  },
  filters: {
  }
})

export default class Slidr extends Vue {
  isTouch: boolean = false // 是否触摸选中
  onTouch :any = throttle(this.onTouchFn, 50) // 节流触摸事件
  showTag: boolean = false
  startValue: number = 0 // 检测触摸开始和结束的值是否一样
  sliderInfo: any = {
    wid: 0, // slider的长度
    hei: 0, // line的高度
    left: 0, // slider在文档中的位置
    right: 0, // slider在文档中的位置
  }
  @Prop({default: 2})
  digit: number // 数据精度,小数点后几位
  @Prop({default: '100%'})
  wid: String // slider长度
  @Prop({default: '2px'})
  hei: String // slider高度
  @Prop({default: 'rgba(223,228,237,1)'})
  color: String // 未选中line背景色
  @Prop({default: 'rgba(40,164,186,1)'})
  chooseColor: String
  @Prop()
  val: number // btn所在位置的值,用v-modal绑定
  @Prop({default: 0})
  minVal: number
  @Prop({default: 1})
  maxVal: number
  @Prop({default: () => { return{backgroundColor:'rgba(0,0,0,0.8)'}}})
  tagStyle: String // tag背景色
  @Prop({default: false})
  tagPrompt: any // tag显示内容
  @Prop({default: true})
  isShowTag: Boolean
  @Prop({default: false})
  disabled: Boolean // 是否禁用
  @Emit('input')
  changeVal(val: number) { this.outVal(val) }
  @Emit('change')
  outVal(val: number) { }
  @Emit('changeEnd') // 结束事件返回两个参数,结束时的值,位置是否改变
  endVal(val: number, sta: boolean) { }
  @Emit('changeStart')
  startVal(val: number) { }
  created () {
  }
  mounted () {
    // 存储slider在页面的位置及大小信息
    const slider: any = $(this.$refs.slider),
      line: any = $(this.$refs.line)

    this.sliderInfo = {
      wid: slider.width(),
      hei: line.height(),
      left: slider.offset().left,
      right: slider.width() + slider.offset().left
    }
  }
  destroyed () {
  }
  get lineSty() {
    return {
      height: this.hei,
      backgroundColor: this.color,
      borderRadius: this.hei
    }
  }
  get btnPosit() {
    return {
      left: ((this.val / (this.maxVal - this.minVal)) * 100) + '%'
    }
  }
  get lineChooseSty() {
    let wid = ((this.val / (this.maxVal - this.minVal)) * 100) + '%'

    return {
      width: wid,
      height: this.hei,
      backgroundColor: this.chooseColor
    }
  }
  touchStart() {
    if(this.disabled){ return }
    window.addEventListener('touchmove', this.onTouch)
    this.startValue = this.val
    this.startVal(this.val)
    this.isTouch = true
    this.showTag = true }
  touchEnd() {
    if(this.disabled){ return }
    window.removeEventListener('touchmove', this.onTouch)
    const sta = this.startValue === this.val

    this.endVal(this.val, sta)
    this.isTouch = false
    this.showTag = false }
  onTouchFn(e) :void{
    if(!this.isTouch){ return }
    const pos: number = e.targetTouches[0].pageX

    if(pos >= this.sliderInfo.right) {
      this.changeVal(this.maxVal)
      return }
    else if(pos <= this.sliderInfo.left) {
      this.changeVal(this.minVal)
      return }
    else {
      const ratio: number = (pos - this.sliderInfo.left) / this.sliderInfo.wid,
        val = this.maxVal * ratio

      this.changeVal(Number(Number(val).toFixed(this.digit)))
    }
  }
}
</script>

<style lang="stylus" scoped>
@import '../../static/css/common.styl'

// tag动画
.tag-enter-active, .tag-leave-active
  transition opacity .5s
.tag-enter, .tag-leave-to
  opacity 0


// 按钮动画
.enter
  animation sliderEnter .5s
  animation-fill-mode forwards
.out
  animation sliderOut .5s
  animation-fill-mode forwards
@keyframes sliderEnter
  0% 
    transform scale(1) translate(-50%, -50%)
  100%
    transform scale(1.2) translate(-50%, -50%)
@keyframes sliderOut
  0% 
    transform scale(1.2) translate(-50%, -50%)
  100%
    transform scale(1) translate(-50%, -50%)

.slider-box
  position relative
  display flex
  align-items center
  height 50px
  .slider-line
    position absolute
    top 50%
    transform translateY(-50%)
    width 100%
  .slider-line_choose
    position absolute
    top 50%
    transform translateY(-50%)
    width 100%
    transition all .2s ease
  .slider-btn
    position absolute
    top 50%
    vertical-align middle
    transition all .05s ease
    width 13px
    height 13px
    border 1.5px solid
    border-radius 50%
    background #fff
  .slider-val
    display none
  .tag
    position absolute
    left 50%
    transform translate(-50%, -60px)
    padding 5px 10px
    border-radius 5px
</style>

文章原址:http://blog.csdn.net/qq_25243451/article/details/78664354

目录
相关文章
|
2月前
|
缓存 JavaScript UED
Vue3中v-model在处理自定义组件双向数据绑定时有哪些注意事项?
在使用`v-model`处理自定义组件双向数据绑定时,要仔细考虑各种因素,确保数据的准确传递和更新,同时提供良好的用户体验和代码可维护性。通过合理的设计和注意事项的遵循,能够更好地发挥`v-model`的优势,实现高效的双向数据绑定效果。
152 64
|
2月前
|
监控 JavaScript 算法
深度剖析 Vue.js 响应式原理:从数据劫持到视图更新的全流程详解
本文深入解析Vue.js的响应式机制,从数据劫持到视图更新的全过程,详细讲解了其实现原理和运作流程。
|
4月前
|
JavaScript
Vue组件传值异步问题--子组件拿到数据较慢
Vue组件传值异步问题--子组件拿到数据较慢
263 58
|
2月前
|
JavaScript 数据管理 Java
在 Vue 3 中使用 Proxy 实现数据双向绑定的性能如何?
【10月更文挑战第23天】Vue 3中使用Proxy实现数据双向绑定在多个方面都带来了性能的提升,从更高效的响应式追踪、更好的初始化性能、对数组操作的优化到更优的内存管理等,使得Vue 3在处理复杂的应用场景和大量数据时能够更加高效和稳定地运行。
66 1
|
2月前
|
JavaScript 开发者
在 Vue 3 中使用 Proxy 实现数据的双向绑定
【10月更文挑战第23天】Vue 3利用 `Proxy` 实现了数据的双向绑定,无论是使用内置的指令如 `v-model`,还是通过自定义事件或自定义指令,都能够方便地实现数据与视图之间的双向交互,满足不同场景下的开发需求。
68 1
|
3月前
|
API
vue3知识点:响应式数据的判断
vue3知识点:响应式数据的判断
35 3
|
3月前
|
存储 缓存 JavaScript
vue表单案例练习:vue表单创建一行数据及删除数据的实现与理解
vue表单案例练习:vue表单创建一行数据及删除数据的实现与理解
52 2
|
4月前
|
JavaScript
Vue+element_Table树形数据与懒加载报错Error in render: “RangeError: Maximum call stack size exceeded“
本文讨论了在使用Vue和Element UI实现树形数据和懒加载时遇到的“Maximum call stack size exceeded”错误,指出问题的原因通常是因为数据中的唯一标识符`id`不唯一,导致递归渲染造成调用栈溢出。
212 1
Vue+element_Table树形数据与懒加载报错Error in render: “RangeError: Maximum call stack size exceeded“
|
4月前
|
JavaScript
关于Vue非父子组件通信遇到的细节问题
本文讨论了Vue中非父子组件通信的一个细节问题,即当使用事件总线($eventsBus)进行通信时,需要确保两个组件能够同时在页面上显示,否则可能无法正确触发和监听事件。作者通过组件A和B的例子说明了这一点,并指出解决方案是在一个共同的父组件C中监听事件。
27 1
关于Vue非父子组件通信遇到的细节问题
|
3月前
|
JavaScript
vue3,使用watch监听props中的数据
【10月更文挑战第3天】
2023 2