声明式渲染 => 组件系统 => 客户端路由 => 集中式状态管理 => 项目构建

防止页面加载时出现闪烁问题
<style type="text/css"> /* 1、通过属性选择器 选择到 带有属性 v-cloak的标签 让他隐藏 */ [v-cloak]{ /* 元素隐藏 */ display: none; } </style><body> <div id="app"> <!-- 2、 在插值表达式中 添加 v-cloak 属性 在 数据渲染完场之后,v-cloak 属性会被自动去除, v-cloak一旦移除也就是没有这个属性了 属性选择器就选择不到该标签 也就是对应的标签会变为可见 --> <div v-cloak >{{msg}}</div> </div> <script type="text/javascript" src="js/vue.js"></script> <script type="text/javascript"> var vm = new Vue({ // el 指定元素 id 是 app 的元素 el: '#app', // data 里面存储的是数据 data: { msg: 'Hello Vue' } });</script></body></html>x<div id="app"> <!-- 注意:在指令中不要写插值语法 直接写对应的变量名称 在 v-text 中 赋值的时候不要在写 插值语法 一般属性中不加 {{}} 直接写 对应 的数据名 --> <p v-text="msg"></p> <p> <!-- Vue 中只有在标签的 内容中 才用插值语法 --> {{msg}} </p></div><script> new Vue({ el: '#app', data: { msg: 'Hello Vue.js' } });</script>用法和v-text 相似 但是他可以将HTML片段填充到标签中
可能有安全问题, 一般只在可信任内容上使用 v-html,永不 用在用户提交的内容上
它与v-text区别在于v-text输出的是纯文本,浏览器不会对其再进行html解析,但v-html会将其当html标签解析后输出。
xxxxxxxxxx<div id="app"> <p v-html="html"></p> <!-- 输出:html标签在渲染的时候被解析 --> <p>{{message}}</p> <!-- 输出:<span>通过双括号绑定</span> --> <p v-text="text"></p> <!-- 输出:<span>html标签在渲染的时候被源码输出</span> --></div><script> let app = new Vue({ el: "#app", data: { message: "<span>通过双括号绑定</span>", html: "<span>html标签在渲染的时候被解析</span>", text: "<span>html标签在渲染的时候被源码输出</span>", } });</script>xxxxxxxxxx <span v-pre>{{ this will not be compiled }}</span> <!-- 显示的是{{ this will not be compiled }} --> <span v-pre>{{msg}}</span> <!-- 即使data里面定义了msg这里仍然是显示的{{msg}} --><script> new Vue({ el: '#app', data: { msg: 'Hello Vue.js' } });</script>如何理解响应式
什么是数据绑定
xxxxxxxxxx <!-- 即使data里面定义了info 后期我们修改了 仍然显示的是第一次data里面存储的数据即 '你好' --> <!-- v-once的应用场景: 如果显示的信息后续不需要再修改, 那么可以使用 v-once, 这样可以提高性能 --><div id= "app"> <div>{{ msg }}</div> <div v-once>{{info}}</div></div><script> new Vue({ el: '#app', data: { msg: 'Hello Vue.js', info: '你好' } });</script>
<input>、<select>、<textarea>、components中使用xxxxxxxxxx <div id="app"> <div>{{msg}}</div> <div> <!-- 当输入框中内容改变的时候, 页面上的msg 会自动更新 --> <input type="text" v-model='msg'> </div> </div><script type="text/javascript"> /* 双向数据绑定 1、从页面到数据 2、从数据到页面 */ var vm = new Vue({ el: '#app', data: { msg: 'Hello Vue' } }); </script>MVVM是前端视图层的概念,也就是分而治之,也就是说:MVVM把前端的视图层,分为了 三部分 Model, View , ViewModel
m model
v view 视图
vm (view-model) 控制器 将数据和视图层建立联系

xxxxxxxxxx<div id="app"> <div>{{num}}</div> <div> <!-- 1 如果事件直接绑定函数名称,那么默认会传递事件对象作为事件函数的第一个参数 --> <button v-on:click='handle1'>点击1</button> <!-- 2、如果事件绑定函数调用,那么事件对象必须作为最后一个参数显示传递, 并且事件对象的名称必须是$event --> <button v-on:click='handle2(123, 456, $event)'>点击2</button> </div></div><script type="text/javascript" src="js/vue.js"></script><script type="text/javascript"> var vm = new Vue({ el: '#app', data: { num: 0 }, methods: { handle1: function(event) { console.log(event.target.innerHTML) }, handle2: function(p, p1, event) { console.log(p, p1) console.log(event.target.innerHTML) this.num++; } } });</script>event.preventDefault() 或 event.stopPropagation() 是非常常见的需求。v-on 提供了事件修饰符xxxxxxxxxx<div v-on:click='handle0'> <button v-on:click.stop='handle1'>点击1</button></div><div> <a href="http://www.baidu.com" v-on:click.prevent='handle2'>百度</a></div><script type="text/javascript"> /* 事件绑定-事件修饰符 */ var vm = new Vue({ el: '#app', data: { num: 0 }, methods: { handle0: function(){ this.num++; }, handle1: function(event){ // 阻止冒泡 // event.stopPropagation(); }, handle2: function(event){ // 阻止默认行为 // event.preventDefault(); } } });</script>
v-on 在监听键盘事件时添加按键修饰符xxxxxxxxxx <form action=""> <div> 用户名:<input type="text" v-on:keyup.delete='clearContent' v-model='uname'> </div> <div> 密码:<input type="text" v-on:keyup.enter='handleSubmit' v-model='pwd'> </div> <div> <input type="button" v-on:click='handleSubmit' value="提交"> </div> </form> <script type="text/javascript"> /* 事件绑定-按键修饰符 */ var vm = new Vue({ el: '#app', data: { uname: '', pwd: '', age: 0 }, methods: { clearContent:function(){ // 按delete键的时候,清空用户名 this.uname = ''; }, handleSubmit: function(){ console.log(this.uname,this.pwd) } } });</script><!--当点击enter或者space时 时调用 `vm.alertMe()` --><input type="text" v-on:keyup.enter.space="alertMe" >常用的按键修饰符.enter => enter键.tab => tab键.delete (捕获“删除”和“退格”按键) => 删除键.esc => 取消键.space => 空格键.up => 上.down => 下.left => 左.right => 右config.keyCodes自定义按键修饰符别名xxxxxxxxxx<div id="app"> <input type="text" v-on:keyup.65='handle' > <input type="text" v-on:keyup.aaa='handle'> </div> <script type="text/javascript"> /* 事件绑定-自定义按键修饰符 规则:自定义按键修饰符名字是自定义的,但是对应的值必须是按键对应event.keyCode值 */ Vue.config.keyCodes.aaa = 65 var vm = new Vue({ el: '#app', data: { info: '' }, methods: { handle: function(event){ console.log(event.keyCode) } } }); </script>
xxxxxxxxxx<div id="app"> <h1>简单计算器</h1> <div> <span>数值A:</span> <span> <input type="text" > </span> </div> <div> <span>数值B:</span> <span> <input type="text"> </span> </div> <div> <button >计算</button> </div> <div> <span>计算结果:</span> <span ></span> </div></div><script type="text/javascript"> /* 简单计算器案例 */ var vm = new Vue({ el: '#app', data: { }, methods: { } });</script>
xxxxxxxxxx<div id="app"> <h1>简单计算器</h1> <div> <span>数值A:</span> <span> <input type="text" v-model='a'> </span> </div> <div> <span>数值B:</span> <span> <input type="text" v-model='b'> </span> </div> <div> <button v-on:click='handle'>计算</button> </div> <div> <span>计算结果:</span> <span v-text='result'></span> </div></div><script type="text/javascript"> /* 简单计算器案例 */ var vm = new Vue({ el: '#app', data: { a: '', b: '', result: '' }, methods: { handle: function () { // 实现计算逻辑 this.result = parseInt(this.a) + parseInt(this.b); } } });</script>
xxxxxxxxxx<!-- 绑定一个属性 --><img v-bind:src="imageSrc"><!-- 缩写 --><img :src="imageSrc"><!-- 例子 --><div id="app"> <a v-bind:href="url">百度</a> <a :href="url">百度1</a> <button v-on:click='handle'>切换</button></div><script type="text/javascript"> /* 属性绑定 */ var vm = new Vue({ el: '#app', data: { url: 'http://www.baidu.com' }, methods: { handle: function(){ // 修改URL地址 this.url = 'http://itcast.cn'; } } });</script>xxxxxxxxxx<div id="app"> <div>{{msg}}</div> <!-- 分别通过v-bind 绑定这个值 通过input事件改变这个值 --> <input type="text" v-bind:value="msg" v-on:input='handle'> <!-- 简化写法 --> <input type="text" v-bind:value="msg" v-on:input='msg=$event.target.value'> <!-- 双向绑定写法 --> <input type="text" v-model='msg'></div><script type="text/javascript"> /* v-model指令的本质 */ var vm = new Vue({ el: '#app', data: { msg: 'hello' }, methods: { handle: function(event){ // 使用输入域中的最新的数据覆盖原来的数据 this.msg = event.target.value; } } });</script>xxxxxxxxxx1、 v-bind 中支持绑定一个对象 如果绑定的是一个对象 则 键为 对应的类名 值 为对应data中的数据 <!-- HTML最终渲染为 <div class="active error"></div> 注意: active,error 对应的渲染到页面上的CSS类名 isActive,isSisErrorize 对应vue data中的数据 如果为true 则对应的类名 渲染到页面上 当 isActive 和 isError 变化时,class列表将相应的更新, 例如,isActive,变成 false class列表将变为 <ul class="error"></ul>--><style> .active { border: 1px solid red; width: 100px; height: 100px; } .error { background-color: orange; }</style><div id="app"> <div v-bind:class="{active: isActive ,error: isError}"> 测试样式 </div> <button v-on:click='handle'>切换</button> </div><script> var vm = new Vue({ el: '#app', data: { isActive: true, isError: true }, methods: { handle: function(){ // 控制isActive的值在true和false之间进行切换 this.isActive = !this.isActive; this.isError = !this.isError; } }</sript>xxxxxxxxxx2、 v-bind 中支持绑定一个数组 数组中 activeClass 和 errorClass 对应为data中的数据 这里的 activeClass 对应 data 中的 active 这里的 errorClass 对应 data 中的 error<div id="app"> <div v-bind:class='[activeClass, errorClass]'>测试样式</div> <button v-on:click='handle'>切换</button> </div><script>var vm = new Vue({ el: '#app', data: { activeClass: 'active', errorClass: 'error' }, methods: { handle: function(){ this.activeClass = ''; this.errorClass = ''; } } });</script><style> .active { border: 1px solid red; width: 100px; height: 100px; } .error { background-color: orange; }</style>xxxxxxxxxx<div id="app"> <!-- 对象绑定和数组绑定可以结合使用 --> <div v-bind:class='[activeClass, errorClass, {test: isTest}]'>测试样式</div> <!-- 可以在data中声明数组用于存放样式 方便后期维护 --> <div v-bind:class='arrClasses'></div> <!-- 对象绑定 也可以在 data中声明数据来存放样式 方便后期维护 --> <div v-bind:class='objClasses'></div> <!-- 默认的 class 样式 会与绑定的 class 一同被渲染出来 --> <div class="base" v-bind:class='objClasses'></div> <button v-on:click='handle'>切换</button> </div><script type="text/javascript"> var vm = new Vue({ el: '#app', data: { activeClass: 'active', errorClass: 'error', isTest: true, arrClasses: ['active','error'], objClasses: { active: true, error: true } }, methods: { handle: function(){ // this.isTest = false; this.objClasses.error = false; } } }); </script> <style type="text/css"> .active { border: 1px solid red; width: 100px; height: 100px; } .error { background-color: orange; } .test { color: blue; } .base { font-size: 28px; } </style>
xxxxxxxxxx<div id="app"> <!-- CSS 属性名可以用驼峰式 (borderStyle) --> <div v-bind:style='{border: borderStyle, width: widthStyle, height: heightStyle}'></div> <!-- 也可以直接绑定一个对象 --> <div v-bind:style='objStyles'></div> <!-- 也可以直接绑定一个数组 --> <div v-bind:style='[objStyles, overrideStyles]'></div> <button v-on:click='handle'>切换</button> </div><script> var vm = new Vue({ el: '#app', data: { borderStyle: '1px solid blue', widthStyle: '100px', heightStyle: '200px', objStyles: { border: '1px solid green', width: '200px', height: '100px' }, overrideStyles: { border: '5px solid orange', backgroundColor: 'blue' } }, methods: { handle: function(){ this.heightStyle = '100px'; this.objStyles.width = '100px'; } }</script>xxxxxxxxxx<div id="app"> <!-- 判断是否加载,如果为真,就加载,否则不加载--> <span v-if="flag"> 如果flag为true则显示,false不显示! </span></div><script> var vm = new Vue({ el:"#app", data:{ flag:true } })</script>---------------------------------------------------------- <div id="app"> <div v-if='score>=90'>优秀</div> <div v-else-if='score<90&&score>=80'>良好</div> <div v-else-if='score<80&&score>60'>一般</div> <div v-else>比较差</div> <div v-show='flag'>测试v-show</div> <button v-on:click='handle'>点击</button> </div><script type="text/javascript"> var vm = new Vue({ el: '#app', data: { score: 10, flag: false }, methods: { handle: function(){ this.flag = !this.flag; } } }); </script>v-show本质就是标签display样式设置为none,控制隐藏
v-if是动态的向DOM树内添加或者删除DOM元素
xxxxxxxxxx<div id="app"> <div>水果列表</div> <ul> <!-- 循环结构-遍历数组 item 是我们自己定义的一个名字 代表数组里面的每一项 items对应的是 data中的数组--> <li v-for='item in fruits'>{{item}}</li> <li v-for='(item, index) in fruits'>{{item + '---' + index}}</li> <li :key='item.id' v-for='item in myFruits'> <span>{{item.ename}}</span> <span>-----</span> <span>{{item.cname}}</span> </li> </ul></div><script type="text/javascript"> /* 循环结构-遍历数组 */ var vm = new Vue({ el: '#app', data: { fruits: ['apple', 'orange', 'banana'], myFruits: [{ id: 1, ename: 'apple', cname: '苹果' },{ id: 2, ename: 'orange', cname: '橘子' },{ id: 3, ename: 'banana', cname: '香蕉' }] } }); </script>xxxxxxxxxx <!-- 循环结构-遍历对象 v 代表 对象的value k 代表对象的 键 i 代表索引 ---> <div v-if='v==13' v-for='(value,key,index) in obj'> {{value + '---' + key + '---' + index}}</div><script> new Vue({ el: '#app', data: { obj: { uname: 'zhangsan', age: 13, gender: 'female' } }})</script>key 的作用
xxxxxxxxxx<ul> <li v-for="item in items" :key="item.id">...</li></ul>
xxxxxxxxxx <div id="app"> <div class="tab"> <!-- tab栏 --> <ul> <li class="active">apple</li> <li class="">orange</li> <li class="">lemon</li> </ul> <!-- 对应显示的图片 --> <div class="current"><img src="img/apple.png"></div> <div class=""><img src="img/orange.png"></div> <div class=""><img src="img/lemon.png"></div> </div> </div> <script type="text/javascript"> var vm = new Vue({ el: '#app', data: {}, methods: {} }); </script> <style type="text/css"> .tab ul { overflow: hidden; padding: 0; margin: 0; } .tab ul li { box-sizing: border-box; padding: 0; float: left; width: 100px; height: 45px; line-height: 45px; list-style: none; text-align: center; border-top: 1px solid blue; border-right: 1px solid blue; cursor: pointer; } .tab ul li:first-child { border-left: 1px solid blue; } .tab ul li.active { background-color: orange; } .tab div { width: 500px; height: 300px; display: none; text-align: center; font-size: 30px; line-height: 300px; border: 1px solid blue; border-top: 0px; } .tab div.current { display: block; } </style>xxxxxxxxxx list: [{ id: 1, title: 'apple', path: 'img/apple.png' }, { id: 2, title: 'orange', path: 'img/orange.png' }, { id: 3, title: 'lemon', path: 'img/lemon.png' }]
把tab栏 中的数替换到页面上
xxxxxxxxxx <div id="app"> <div class="tab"> <ul> <!-- 1、绑定key的作用 提高Vue的性能 2、 key 需要是唯一的标识 所以需要使用id, 也可以使用index , index 也是唯一的 3、 item 是 数组中对应的每一项 4、 index 是 每一项的 索引 --> <li :key='item.id' v-for='(item,index) in list'>{{item.title}}</li> </ul> <div :key='item.id' v-for='(item, index) in list'> <!-- : 是 v-bind 的简写 绑定属性使用 v-bind --> <img :src="item.path"> </div> </div> </div><script>var vm = new Vue({ // 指定 操作元素 是 id 为app 的 el: '#app', data: { list: [{ id: 1, title: 'apple', path: 'img/apple.png' }, { id: 2, title: 'orange', path: 'img/orange.png' }, { id: 3, title: 'lemon', path: 'img/lemon.png' }] } })</script>4.1 、让默认的第一项tab栏高亮
tab栏高亮 通过添加类名active 来实现 (CSS active 的样式已经提前写好)
在data 中定义一个 默认的 索引 currentIndex 为 0
给第一个li 添加 active 的类名
4.2 、让默认的第一项tab栏对应的div 显示
xxxxxxxxxx <ul> <!-- 动态绑定class 有 active 类名高亮 无 active 不高亮--> <li :class='currentIndex==index?"active":""' :key='item.id' v-for='(item,index) in list' >{{item.title}}</li> </ul> <!-- 动态绑定class 有 current 类名显示 无 current 隐藏--> <div :class='currentIndex==index?"current":""' :key='item.id' v-for='(item, index) in list'> <!-- : 是 v-bind 的简写 绑定属性使用 v-bind --> <img :src="item.path"> </div><script> var vm = new Vue({ el: '#app', data: { currentIndex: 0, // 选项卡当前的索引 默认为 0 list: [{ id: 1, title: 'apple', path: 'img/apple.png' }, { id: 2, title: 'orange', path: 'img/orange.png' }, { id: 3, title: 'lemon', path: 'img/lemon.png' }] } })</script>4.3 、点击每一个tab栏 当前的高亮 其他的取消高亮
给每一个li添加点击事件
让当前的索引 index 和 当前 currentIndex 的 值 进项比较
如果相等 则当前li 添加active 类名 当前的 li 高亮 当前对应索引的 div 添加 current 当前div 显示 其他隐藏
xxxxxxxxxx <div id="app"> <div class="tab"> <ul> <!-- 通过v-on 添加点击事件 需要把当前li 的索引传过去 --> <li v-on:click='change(index)' :class='currentIndex==index?"active":""' :key='item.id' v-for='(item,index) in list'>{{item.title}}</li> </ul> <div :class='currentIndex==index?"current":""' :key='item.id' v-for='(item, index) in list'> <img :src="item.path"> </div> </div> </div><script> var vm = new Vue({ el: '#app', data: { currentIndex: 0, // 选项卡当前的索引 默认为 0 list: [{ id: 1, title: 'apple', path: 'img/apple.png' }, { id: 2, title: 'orange', path: 'img/orange.png' }, { id: 3, title: 'lemon', path: 'img/lemon.png' }] }, methods: { change: function(index) { // 通过传入过来的索引来让当前的 currentIndex 和点击的index 值 相等 // 从而实现 控制类名 this.currentIndex = index; } } })</script>