2.双向数据绑定
Vue的原理
Vue简单的使用场景
<body> <!-- 希望Vue实例管理的区域,帮我们把数据填充到该区域里 --> <div id="root">{{username}}</div> <script src="./lib/vue-2.6.12.js"></script> <script> // vue实例对象负责处理View, Model之间的双向流动。 // vue实例的入参是它要监听的两头,el: 使用选择器寻找它要监听管理的View容器, data: 数据源与view做双向绑定。 const vm = new Vue({ el: "#root", data: { username: 'Frog' } }); </script> </body>
npm install -g @vue/cli
vue-cli版本降级
从vue-cli v4的版本 降回v3
npm uninstall -g @vue/cli //安装指定版本 npm install -g @vue/cli@版本号
使用vue -V验证是否安装成功
zhoufeideMacBook-Pro-2:lession0 zhoufei$ vue -V @vue/cli 5.0.8
vue create 项目名称
SPA 单页面应用:所有的功能都在一个页面中实现,一个项目就一个页面。
并且vue编写的js代码编译打包后,会嵌入到div#app的下面
main.js是项目的入口,webpack打包的入口在这里
Vue的运行流程
// 导入vue框架,在js全局当如Vue构造方法 import Vue from 'vue' // 导入App.vue根组件,将根组件替换到index.html的#app标签 import App from './App.vue' Vue.config.productionTip = false new Vue({ el: '#app', //#app标签,占位标签 render: h => h(App), // 返回要替换#app占位标签的根组件 })
App.vue根据组件
<template> <div> <p>这是第1篇文章</p> <p>这是第2篇文章</p> <p>这是第3篇文章</p> <p>这是第4篇文章</p> <p>这是第5篇文章</p> <p>这是第6篇文章</p> <p>这是第7篇文章</p> <p>这是第8篇文章</p> </div> </template>
Vue挂载方式,下面2种是一样的。
new Vue({ render: h => h(App), }).$mount('#app') new Vue({ el: '#app', //#app标签,占位标签 render: h => h(App), // 返回要替换#app占位标签的根组件 })
Vue组件化开发
<template> <!-- template模板是组件的结构,下面的style和script都是模板的修饰 --> <div id="app"> 自定义组件 --- {{ username }} <button @click="changeName">修改名字</button> </div> </template> <script> export default { // 组件中的data定义和vue实例对象中data的定义不一样 // 组件中的data定义是一个函数,在函数的return里返回一个对象 data: function (params) { return { username: 'Lilei' } }, // 除了data需要使用函数,其他定义都和Vue实例的传参一样。 methods: { changeName () { this.username = '娃哈哈' } }, watch: {}, filters: {}, computed: {} } </script> <!-- 修改css的预处理器使用lang来设置 --> <style lang="less"> .app { background-color: antiquewhite; button { font-size: 20; } } </style>
Vue指令
<!-- 内容渲染指令 --> <p v-text="username">姓名:</p> <p v-text="gender">性别:</p> <hr> <p>姓名: {{ username }}</p> <p>性别: {{ gender }}</p> <hr> <p v-html="info">内容:</p>
属性绑定指令
<input type="text" v-bind:placeholder="placeholder"> <input type="text" :placeholder="placeholder">
其中属性绑定v-bind:和差值表达式里面都可以写一些简单的js语句
<p>1+2: {{ 1+2 }}</p> <input type="text" :placeholder=" '1+2:'+ 1+2 ">
事件绑定指令
<!-- 事件绑定指令 --> <p>统计结果: {{ count }}</p> <!-- v-on:可以简写成@, @click; 如果方法要传参,可以直接在方法的”“里面使用add(1)进行表示 --> <button v-on:click="add(2)" >加二</button> <!-- 如果方法sub没有传参,系统会默认传递一个event参数, 如果sub(1)传参,那么这个默认事件event参数就会被覆盖--> <button v-on:click="sub" >减一</button> <!-- 如果即要传参又要传递事件event, 可以使用Vue提供的系统事件$event当做普通参数传递sub(1, $event) --> <button v-on:click="sub2(2, $event)" >减二</button>
处理方法的添加是放在Vue({methods})项上
<script> const vm = new Vue({ el: '#app', data: { username: 'Frog', }, methods: { add: function (n) { vm.count += n }, sub(e) { //推荐这样简写 //vm就等于this, 是一个实例对象 this.count -= 1 console.log(e); if (this.count % 2 === 0) { // e.target是事件的响应对象 e.target.style.backgroundColor = 'red'; } else { e.target.style.backgroundColor = ''; } }, sub2(n, e) { console.log(n); console.log(e); } } }) </script>
<div id="app"> <!-- 事件修饰符:阻止默认行为,停止冒泡 --> <a href="http://www.baidu.com" @click.prevent="goBaidu">跳转到百度</a> </div>
<script> const vm = new Vue({ el: '#app', data: { username: 'Frog', }, methods: { goBaidu(e) { e.preventDefault() console.log('跳转到baidu'); } } }) </script>
按键修饰符
<div id="app"> <!-- 按键修饰符 --> <input type="text" @keyup.esc="cleanHandler" @keyup.enter="commitHandler"> </div>
<script> const vm = new Vue({ el: '#app', data: { username: 'Frog', }, methods: { cleanHandler(e){ //当esc键 弹出时回调 console.log(e.target); }, commitHandler(e){ console.log('提交'); } } }) </script>
双向数据绑定指令
<div id="app"> <p>用户名:{{ username }}</p> <input type="text" v-model="username"> <hr> <input type="text" :value="username"> <hr> <select v-model="city"> <option value="0">请选择城市</option> <option value="1">北京</option> <option value="2">上海</option> <option value="3">广州</option> </select> </div>
<script> const vm = new Vue({ el: '#app', data: { username: '', city: '0' } }) </script>
v-model指令修饰符
<!-- v-model双向绑定修饰符 --> <input type="text" v-model="num1"> + <input type="text" v-model ="num2"> = <span>{{ num1 + num2}}</span> <hr> <input type="text" v-model.number="num1"> + <input type="text" v-model.number ="num2"> = <span>{{ num1 + num2}}</span> <hr> <input type="text" v-model.trim="username"> <hr> <!-- 当点击enter时才更新,而不每次修改都更新 --> <input type="text" v-model.lazy="username">
const vm = new Vue({ el: '#app', data: { username: '', city: '0', num1: 1, num2: 1 } })
条件渲染指令
<div id="app"> <!-- v-if隐藏时,p标签会从Dom中移除 --> <p v-if="flag">这是用v-if渲染的</p> <!-- v-show隐藏时,只是display:none --> <p v-show="flag">这是用v-show渲染的</p> <div v-if="scoe === 'A'">优秀</div> <div v-else-if="scoe === 'B'">良好</div> <div v-else-if="scoe === 'C'">及格</div> <div v-else>不及格</div> </div>
<script> vm = new Vue({ el: '#app', data: { flag: true, scoe: 'A' } }) </script>
列表渲染指令
<div id="app"> <table class="table table-bordered table-hover table-striped"> <thead> <th>索引</th> <th>id</th> <th>姓名</th> </thead> <tbody> <!-- 列表渲染,需要展示哪种列表,就在那个上面写v-for --> <!-- 写了v-for,同时要写一个key作为唯一标识,这个key只能是数字或字符串,并且拥有有效的唯一性 --> <tr v-for="(item, index) in list" :key="item.id" :title="item.name + index"> <td>{{ index }}</td> <td>{{ item.id }}</td> <td>{{ item.name }}</td> </tr> </tbody> </table> </div>
<script> vm = new Vue({ el: '#app', data: { list: [ {id:1, name:'zhangsan'}, {id:2, name:'lisi'}, {id:3, name:'wangwu'} ] } }) </script>
过滤器
<div id="app"> <!-- 过滤器:管道符,和linux系统提供的管道操作相似,过滤器是一个函数,定义在filters下面 --> <p>{{ message | capi }}</p> <hr> <p>时间:{{ new Date() | dateFormate }}</p> <!-- 过滤器可以连续调用多次,并且过滤器可以传参,但是传递的参数在接收时是从第2个开始的,第1个默认是管道传参。 过滤在Vue3被剔除了 --> <!-- <p>时间:{{ new Date() | dateFormate1(param1, param2) | dateFormate }</p> --> </div>
// 添加全局过滤器 Vue.filter('dateFormate', function (time) { const dtstr = dayjs(time).format('YYYY-HH-DD HH:mm:ss') return dtstr }) vm = new Vue({ el: '#app', data: { message: 'hello vue.js' }, // 过滤器是一个函数,它定义在filters下面 filters: { capi(val) { // val: 标识通过管道符传过来的值 var one = val.charAt(0).toUpperCase() var two = val.slice(1) return one + two } } })
侦听器
<div id="app"> <input type="text" v-model="username"> <hr> <input type="text" v-model="userBigName"> <hr> <input type="text" v-model="userInfo.city"> </div>
侦听器有2种:函数监听器,对象监听器
var vm = new Vue({ el: '#app', data: { username: '', userBigName: 'admin', userInfo: { city: '' } }, watch: { // 函数监听器,监听哪个属性,方法名就是那个属性名 // 第一个是新值,第二个是旧值 // 缺点1:页面初始化时没法自动触发 // 缺点2:当属性是一个对象时,修改对象中的属性,函数监听器无法监听到 username(newV, oldV) { console.log(newV, oldV); $.get('https://www.escook.cn/api/finduser/'+newV, function(res){ console.log(res); console.log($); }) }, // 对象侦听器,侦听哪个属性,key值就设置成哪个属性名 // 优点1:通过设置immediate为true, 可以在页面加载时,默认执行一次调用 userBigName: { handler(newV, oldV) { console.log(newV, oldV); $.get('https://www.escook.cn/api/finduser/'+newV, function(res){ console.log(res); console.log($); }) }, immediate: true }, // 对象侦听器,可以监听对象属性中的子属性 // 优点2:对象侦听器可以开启深度监听,修改属性对象中的一个子属性,收到监听回调 userInfo: { handler(newV) { console.log(newV); }, deep: true }, // 子属性方法监听器,解决缺点2 // 可以使用下面这个'属性.子属性'的方法,做方法监听。可以做到直接监听到子属性,并且newV打印的也是子属性的值 'userInfo.city'(newV) { console.log(newV); } } })
计算属性
<div id="app"> <input type="text" v-model.number="r"> <input type="text" v-model.number="g"> <input type="text" v-model.number="b"> <div class="box" v-bind:style="{backgroundColor: rgb }"> {{ rgb }} </div> </div>
var vm = new Vue({ el: '#app', data: { r: 0, g: 0, b: 0 }, // 计算属性 // 定义时用方法定义,使用时按属性使用,在内存中保存的也是属性形式 // 优点:1.代码复用,2.data属性修改,计算属性会被同步修改 computed: { rgb() { return `rgb(${this.r}, ${this.g}, ${this.b})` } } })