v-model封装组件(定义 model 属性)

一、vue2.0

1、父级组件调用子组件counterComponents

ini 复制代码
<counterComponents v-model="item.totalNumber"/>

2、子组件counterComponents.vue

需求:计数器的输入值在1-30的范围的正整数 主要代码 子组件绑定值::value="innerValue" 组件更新数据: this.$emit('change',newInnerValue + 1)

js 复制代码
export default {
  // 使用 model 配置,支持 v-model
  model:{
    prop: 'value',
    event:'change'
  },
  props: {
    value: {
      type: Number,
      default: 1
    },
  },
  data() {
    return {
      innerValue: this.value
    }
  },
  watch: {
    // 监听外部 value 变化,同步到内部值
    value: {
      immediate: true,
      handler(newVal) {
        this.innerValue = newVal
      }
    }
  }
}

完整代码

js 复制代码
<template>
  <div
    class="pi-align-center"
    @click.stop="()=>{}">
    <span class="pi-pd-right-20">货物件数 </span>
    <div class="counter-containers pi-align-center pi-justify-between">
      <div
        class="pi-pd-lr-10 btn"
        @click.stop="handleCount('subtract')">
        <pi-icon
          name="subtract"
          size="30" />
      </div>
      <pi-input
        :value="innerValue"
        placeholder=""
        :customStyle="{
          background: '#ffffff',
          borderLeft: '1rpx solid #d7d7d7',
          borderRight: '1rpx solid #d7d7d7',
        }"
        customClass="pi-text-center"
        @input="handleInput"
      />
      <div
        class="pi-pd-lr-10 btn"
        @click.stop="handleCount('add')">
        <pi-icon
          name="add"
          size="30" />
      </div>
    </div>
  </div>
</template>
<script>
export default {
  // 使用 model 配置,支持 v-model
  model:{
    prop: 'value',
    event:'change'
  },
  props: {
    value: {
      type: Number,
      default: 1
    },
    min: {
      type: Number,
      default: 1,
    },
    max: {
      type: Number,
      default: 30,
    }
  },
  data() {
    return {
      innerValue: this.value
    }
  },
  computed: {

  },
  watch: {
    // 监听外部 value 变化,同步到内部值
    value: {
      immediate: true,
      handler(newVal) {
        this.innerValue = newVal
      }
    }
  },
  methods: {
    handleCount(sign) {
      const newInnerValue = this.innerValue
      if (sign === 'add') {
        if (newInnerValue + 1 > this.max) return
        this.$emit('change',newInnerValue + 1)
      } else {
        if (newInnerValue - 1 < this.min) return
        this.$emit('change',newInnerValue - 1)
      }
    },
    handleInput(e) {
      const num = e.replace(/[^0-9]/g, '')
      const number = num ? num * 1 : 0
      if (number) {
        const maxNumber = number > this.max ? this.max : number
        this.$emit('change',maxNumber)
      } else {
        this.$emit('change',1)
      }
    },
  },
}
</script>
<style lang="scss" scoped>
.counter-containers {
  width: 200rpx;
  background-color: #f2f2f2;
  border: 1rpx solid #d7d7d7;
  border-radius: 10rpx;
  overflow: hidden;
  box-sizing: border-box;
  .btn {
    cursor: pointer;
  }
}
</style>

二、vue3.4前

父组件代码

xml 复制代码
<script setup lang="ts">
import { ref } from 'vue';
import dialogComponent from '@/components/dialogComponent.vue';
let isShow = ref(false)
</script>
<template>
    <div>
        <button @click="isShow = true ">打开</button>
        <dialogComponent v-model="isShow"/>
    </div>
</template>

子组件代码

xml 复制代码
<script setup lang="ts">
    const props = defineProps(['modelValue'])
    const emit  =  defineEmits(['update:modelValue'])
    
</script>
<template>
    <div v-if="props.modelValue">
        弹框
        <button @click="emit('update:modelValue', false)">关闭</button>
    </div>
</template>

三、vue3.4后

父组件代码

xml 复制代码
<script setup lang="ts">
import CounterComponent from '@/components/CounterComponent.vue';
import { ref } from 'vue';
let count = ref(0)
const add = () =>{
    count.value = Number(count.value)+1
}
</script>
<template>
    <div>
        <button @click="add">+1</button>
        <CounterComponent v-model="count"/>
        <div>{{ count }}</div>
    </div>
</template>

子组件代码

xml 复制代码
<script setup lang="ts">
    // 声明一个双向绑定 prop
    const model = defineModel();
</script>
<template>
    <div> 
        <input type="number"v-model="model">
    </div>
</template>
相关推荐
Irene19917 分钟前
Vue3 规范推荐的 <script setup> 中书写顺序(附:如何响应路由参数变化)
vue.js·路由
pusheng202511 分钟前
普晟传感2026年新春年会总结与分析
前端·javascript·html
谢尔登13 分钟前
React19事件调度的设计思路
前端·javascript·react.js
Emma_Maria23 分钟前
本地项目html和jquery,访问地址报跨域解决
前端·html·jquery
奋斗吧程序媛28 分钟前
常用且好用的命令
前端·编辑器
2301_7965125231 分钟前
【精通篇】打造React Native鸿蒙跨平台开发高级复合组件库开发系列:Lazyload 懒加载(懒加载的图片)
前端·javascript·react native·react.js·ecmascript·harmonyos
敲敲了个代码37 分钟前
从N倍人力到1次修改:Vite Plugin Modular 如何拯救多产品前端维护困境
前端·javascript·面试·职场和发展·typescript·vite
Yff_world1 小时前
网络安全与 Web 基础笔记
前端·笔记·web安全
Sapphire~1 小时前
Vue3-19 hooks 前端数据和方法的封装
前端·vue3
浩宇软件开发1 小时前
基于OpenHarmony鸿蒙开发医院预约挂号系统(前端后端分离)
前端·华为·harmonyos