VUE.js
基本命令
v-if || v-for : 条件与循环
v-model: 为页面输入框进行数据双向绑定
v-on || @click : 为页面元素绑定各种事件
v-bind:class || :class :给页面标签的一个属性"绑定"一个属性class 然后设置
v-if || v-else-if || v-else: if else语句
v-show : 显示或隐藏但dom元素还存在 其他的是虚拟dom所以false会不存在
.lazy || .number || .trim: ||multiple 分别对应 [懒] [数值] [字符串两端空格删除] [可以多选]
v-once 只会展示第一次更新的数据 以后不会更改
v-html <h2 v-html='url'></h2> 把data里的url变量作为html展示出来
v-text <h2 v-text="message"></h2> data里的数据作为文本展示
v-pre <h2 v-pre="message">{{message}}</h2> 把变量原封不动的展示出来
基本形式
let myApp = new Vue ({
el: '#myApp',
data: {
数据与变量 或函数(组件当中data必须是一个函数)
},
methods: {
if 事件 函数 和各种方法函数
},
filters: {
过滤器方法 格式化变量内容的输出
},
computed: {
计算属性方法,
方法名: {
get: function () {
设置计算属性
},
set: function () {
设置计算属性
}
}
},
watch: {
观察属性方法
},
components : {
局部组件的注册
},
beforeCreate: function(){
各种生命周期钩子函数
}
})
组件内容:
v-on 侦听组件事件 当组件触发事件后进行事件处理 用在父组件当中
$emit 触发提交事件 并将数据提交给事件侦听者 用在子组件当中
如果想在表格里使用组件 需要使用is属性
Vue.component('组件标签名',{
props: {
name:{ //组件创建设置属性
//组件参数的验证方法
type: String,
required: true, // required必须输入项
validator: function(){}, //validator验证器
default: function(){} // 默认
}
},
template:'组件模板如 <div>今天下雨 出不去啦 干什么呢?看Youtube吧!</div>>'
<slot></slot> //是父组件与子组件的通讯方式 可以将父组件的内容显示在子组件当中
<div>小前:<slot name="e"></slot></div> //组合slot插槽
,
data: function () {
组件数据函数
},
computed: {
组件计算函数
}
})
1、引用vue文件:<script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script>
2、声明一个带有id的div
3、在script中 引用带有id 的div 初始化一个对象 el选择对象 data是数据 message是变量
4、data数据这里 在实际里都是在服务器取得 然后再渲染到页面当中
1、开始
<div id="myApp">
{{message}}
</div>
const myApp = new Vue({
el: '#myApp',
data: {
message: Hello Vue.js Wrold!
}
})
2、条件与循环 v-if v-for
<div id="myApp">
<h3>游戏列表</h3>
<div v-if="seen">2017最新发售</div> //如果这个是true就显示 false就隐藏
<ol>
<li v-for="game in games">{{game.title}} / {{game.price}}元</li> //遍历games in game
</ol>
</div>
<script>
var myApp = new Vue({
el: '#myApp',
data:{
seen: false, //这里false 2017最新发售在dom里面也不会有 因为是虚拟dom
games: [
{title: 勇者斗恶龙
,price:400},
{title: 超级马里奥
,price:380},
{title: 我的世界
,price:99}
]
}
})
</script>
3、处理用户输入 v-model
为页面输入框进行数据双向绑定:
input
select
textarea
components 组件
这里用v-model绑定了一个input输入框 在输入框输入的值会保存在data的mygame值里 会动态显示在p标签上
v-model
<div id="myApp">
<p>你最喜欢的游戏是:{{mygame}}</p>
<input v-model="mygame">
</div>
<script>
let myApp = new Vue({
el:'#myApp',
data:{
mygame: '超级马里奥'
}
})
</script>
4、按钮事件 v-on @click
为页面元素绑定各种事件(keydown,keyup,click,dbclick,load,etd)
methods定义各种方法 和data平级也是vue的保留词
这里默认值 超级马里奥 当点击下面的按钮时 methods里的方法执行 this指向当前myApp实例
传入一个参数 把这个参数赋值给 this.mygame值里 "btnClick('我的世界')" 我的世界是传入的值
当我点击哪个 哪个的值就能显示在页面上
<div class="myApp">
<p>您最喜欢的游戏是:{{mygame}}</p>
<button v-on:click="btnClick('我的世界')">我的世界</button>
<button v-on:click="btnClick('勇者斗恶龙')">勇者斗恶龙</button>
<button v-on:click="btnClick('塞尔达传说')">塞尔达传说</button>
<button @click="btnClick('魔界战记')">魔界战记</button>
</div>
<script>
var myApp = new Vue({
el: '.myApp',
data: {
mygame: '超级马里奥'
},
methods: {
btnClick: function (pname){
this.mygame = pname;
}
}
})
</script>
5、组件 component
定义页面的局部区域块 完成单独的页面小功能
//定义1geme-item组件 等于创建了一个html组件表标签
2props属性 这里games每次循环后的item 赋值绑定给了game属性(v-bind:game="item")
3template是这个组件里的内容 也就是网页该显示什么内容
<div class="myApp">
<ol>
<geme-item v-for="item in games" v-bind:game="item"></geme-item>
</ol>
</div>
<script>
Vue.component('geme-item',{
props: ['game'],
template: `<li>{{game.title}}</li>>`
})
var myApp = new Vue({
el: '.myApp',
data: {
games: [
{title:'斗地主'},
{title:'打麻将'},
{title:'UNO'}
]
}
})
</script>
6、过滤器 filters关键字 格式化变量内容的输出
比如日期格式化 字母大小写 数字再计算等等
如有一个日期[2019,06,10] 从数据库读取出来这个数据 然后是加 年月日 还是以/为分隔 叫作格式化内容
数据是0.3 表现出来的形式是30% 这就是数形分离 非常经典的mvc 设计方法
<div class="myApp">
<p>{{message}}</p>
<p>{{message | toupper}}</p>
<p>现在的学习进度是{{num}}({{num | topercentage}}).</p>
</div>
<script>
var myApp = new Vue({
el: '.myApp',
data: {
message: 'hello vue.js world',
num: 0.3
},
filters: {
toupper: (value) => return value.toUpperCase(),
topercentage: (value) => return value * 100 + '%'
}
})
</script>
7、计算属性 computed关键词
处理元数据(数据库中取出的还没加工的数据) 便于进行二次利用 (比如:消费税自动计算功能)
<div class="myApp">
今年3月3日发售的任天堂新一代主机的价格是:{{price}}日元,含税价格为:{{priceInTax}}日元,折合人民币为:{{priceChinaRMB}}元
</div>
<script>
var myApp = new Vue ({
el: '.myApp',
data: {
price: 29980
},
computed: {
priceInTax: function () {
return this.price * 1.08
},
priceChinaRMB: function () {
return Math.round(this.priceInTax / 16.75)
}
}
})
</script>
8、观察-属性 watch关键词 angular而来
监视变量 当这个属性或者是变量 改变的时候 的相应处理 如这个值加了1000日元后
类似事件 当点击时做什么 这个是当变量改变时做什么
function(newVal,oldVal){} newVal,oldVal 这两个参数是默认属性
<div class="myApp">
<p>今年3月3日发售的任天堂新一代主机的价格是:{{price}}日元,含税价格为:{{priceInTax}}日元,折合人民币为:{{priceChinaRMB}}元</p>
<button @click="btnClick(10000)">加价10000日元</button>
</div>
<script>
var myApp = new Vue({
el: '.myApp',
data: {
price: 0,
priceInTax: 0,
priceChinaRMB: 0
},
watch:{
price: function(newVal,oldVal){
console.log(newVal,oldVal)
this.priceInTax = this.price * 1.08
this.priceChinaRMB = this.priceInTax / 16.75
}
},
methods: {
btnClick: function (newprice){
this.price = this.price + newprice
}
}
})
myApp.price = 29980
</script>
9、设置计算属性 set get
设计计算属性 同步更新元数据的值
设置计算后的值 推想回原来的值 如现在含税价格是10800 那么没含税价是多少
当要取得priceInTax的值时走get逻辑 当要去设置priceInTax时走set逻辑
<div class="myApp">
<p>今年3月3日发售的任天堂新一代主机的价格是:{{price}}日元,含税价格为:{{priceInTax}}日元,折合人民币为:{{priceChinaRMB}}元</p>
<button @click="btnClick(10800 )">把含税改价为10800日元</button>
</div>
<script>
var myApp = new Vue({
el: '.myApp',
data: {
price : 29980
},
computed: {
priceInTax: {
get : function(){
return this.price * 1.08
},
set : function (value){
this.price = value / 1.08
}
},
priceChinaRMB: function (){
return Math.round(this.priceInTax / 16.75)
}
},
methods: {
btnClick: function(newPrice){
this.priceInTax = newPrice
}
}
})
</script>
10、Class绑定 v-bind:class
为这个div 的class属性 绑定(v-bind)一个active的class 但是这个active是什么时候生效 看isActive的变量
为html标记绑定样式单class属性
可以用来切换class 做一个样式切换
正常:v-bind:class
简写::class
<div class="myApp">
<div v-bind:class="{active:isActive}">红色文本</div> 为这个div的class属性绑定上active这个css样式 什么时候绑定看isActive这个Vue的变量
<div :class="{active:isActive}">红色文本</div>
<button @click="btnClick">改变class</button>
</div>
<script>
var myApp = new Vue({
el: '.myApp',
data:{
isActive: true
},
methods:{
btnClick: function(){
this.isActive = !this.isActive
}
}
})
</script>
11、class对象绑定 :class
为html标记绑定样式单class对象
可以用来切换class 做一个样式切换
<div class="myApp">
<div :class="myClass">红色文本</div>
<button @click="btnClick">改变class</button>
</div>
<script>
var myApp = new Vue({
el: '.myApp',
data:{
myClass:{
active: true,
big:true
}
},
methods:{
btnClick: function(){
this.myClass.active = !this.myClass.active
this.myClass.big = !this.myClass.big
}
}
})
</script>
12、条件渲染 v-if v-else-if v-else
判断vue.js的变量的值然后执行页面渲染逻辑 (if-elseif-else)
<div class="myApp">
<h1 v-if="result == 0">成绩未公布</h1>
<h1 v-else-if="result < 60">{{result}}分 - 考试不及格</h1>
<h1 v-else-if="result < 80">{{result}}分 - 还需努力</h1>
<h1 v-else>{{result}}分 - 考的不错 玩游戏吧</h1>
<button>考试成绩</button>
</div>
<script>
var myApp = new Vue({
el: '.myApp',
data: {
result: 0
},
methods: {
btnClick: function(){
this.result = Math.round(Math.random() * 100)
}
}
})
</script>
13、元素显示 v-show
标记是否显示html元素 (注意 v-show设置的标记在html dom中不会消失)
这个不是虚拟dom 所以show就是显示和不显示 但是dom还在 相等于display:none
<div class="myApp">
<h1 v-show="result">任天堂新一代主机switch</h1>
<button @click="btnClick">显示按钮</button>
</div>
<script>
var myApp = new Vue({
el: '.myApp',
data:{
result: true
},
methods:{
btnClick:function(){
this.result = !this.result
}
}
})
</script>
14、循环数组 v-for
循环数组元素 整理内容显示到页面上
循环 index是数组的下标
<div class="myApp">
<ul>
<li v-for="(game,index) in games">{{index + 1}} {{game.title}} /售价: {{game.price}}元</li> 循环games数组里的数据包含两个参数和下标
</ul>
</div>
<script>
var myApp = new Vue({
el: '.myApp',
data: {
games:[
{title:'勇者斗恶龙',price:500},
{title:'泡泡卡丁车',price:400},
{title:'马里奥',price:550}
]
}
})
</script>
15、对象循环 v-for
循环js对象 把对象内容循环显示到页面上
要先写value 再写key 顺序不能搞反
<div class="myApp">
<h1>js对象迭代</h1>
<div v-for="(value,key) in mygame">
{{key}}:{{value}}
</div>
</div>
<script>
var myApp = new Vue({
el:'.myApp',
data:{
mygame:{
title: '乌贼娘二代',
price:350,
pubdate: '2017年夏季',
category:'射击',
agerange:'全年龄'
}
}
})
</script>
16、事件处理器 @click
页面元素的事件绑定 (click keyup load等等)
keyup 按下键盘的某一个按键时
$event 事件对象
获得用户的输入信息
<div class="myApp">
<h1>事件处理器</h1>
<input id="txtName" v-on:keyup="txtKeyup(event)">OK</button>
</div>
<script>
var myApp = new Vue({
el: '.myApp',
data:{},
methods:{
txtKeyup:function (event) {
this.debugLog(event)
},
btnClick:function (event) {
this.debugLog(event)
},
debugLog:function (event) {
console.log(
event.srcElement.tagName,
event.srcElement.id,
event.srcElement.innerHTML,
event.key?event.key:""
)
}
}
})
</script>
【表单系列】
1、表单控件绑定 v-model input[type='text']
为表单控件元素创建双向数据绑定 (将js变量的值 快速 设定到控件上 然后将用户输入的内容 快速 设置回js变量)
<div class="myApp">
<h1>表单控件绑定</h1>
<input type="text" v-model="message" placeholder="来呀 编辑我吧!">
<p>Message is :{{message}}</p>
</div>
<script>
var myApp = new Vue({
el:'.myApp',
data:{
message:'超级马里奥'
}
})
</script>
2、表单复选框 v-model input[type='checkbox']
<div class="myApp">
<h1>表单复选框</h1>
<input type="checkbox" id="生化危机" value="生化危机" v-model="checkedGames">
<label for="生化危机">生化危机</label>
<input type="checkbox" id="模拟飞行" value="模拟飞行" v-model="checkedGames">
<label for="模拟飞行">模拟飞行</label>
<input type="checkbox" id="塞尔达传说" value="塞尔达传说" v-model="checkedGames">
<label for="塞尔达传说">塞尔达传说</label>
<p>您选择的游戏是:{{checkedGames}}</p>
</div>
<script>
var myApp = new Vue({
el:'.myApp',
data:{
checkedGames:[]
}
})
</script>
3、表单单选按钮 v-model input[type='radio']
<div class="myApp">
<h1>表单单选按钮</h1>
<h3>选择性别</h3>
<input type="radio" id="male" value="男" v-model="pickedSex">
<label for="male">男</label>
<br>
<input type="radio" id="female" value="女" v-model="pickedSex">
<label for="female">女</label>
<h3>选择爱好</h3>
<input type="radio" id="game" value="玩游戏" v-model="pickedHobby">
<label for="game">玩游戏</label>
<br>
<input type="radio" id="movie" value="看电影" v-model="pickedHobby">
<label for="movie">看电影</label>
<h3>选择结果</h3>
<p>性别:{{pickedSex}}</p>
<p>爱好:{{pickedHobby}}</p>
</div>
<script>
var myApp = new Vue({
el:'.myApp',
data:{
pickedSex:'',
pickedHobby:''
},
methods:{
}
})
</script>
4、表单下拉框 v-model select
multiple 代表可以多选
<div class="myApp">
<h3>你最喜欢的NBA明星是:</h3>
<select v-model="likedNBAStar" style="width: 210px;">
<option>科比</option>
<option>詹姆斯</option>
<option>哈登</option>
<option>库里</option>
<option>杜兰特</option>
</select>
<h3>我的全明星:</h3>
<select v-model="likedNBAStars" multiple style="width: 210px;height: 210px;">
<option>科比</option>
<option>詹姆斯</option>
<option>哈登</option>
<option>库里</option>
<option>杜兰特</option>
<option>科比1</option>
<option>詹姆斯2</option>
<option>哈登3</option>
<option>库里4</option>
<option>杜兰特5</option>
</select>
<h3>选择结果</h3>
<p>我最喜欢:{{likedNBAStar}}</p>
<p>我的全明星:{{likedNBAStars}}</p>
</div>
<script>
var myApp = new Vue({
el:'.myApp',
data:{
likedNBAStar:null,
likedNBAStars:[]
},
methods:{
}
})
</script>
5、表单修饰符 .lazy .number .trim
.lazy 懒加载 用户的输入内容时不做绑定数据的更新处理 在控件的on change事件中更新绑定的变量 就是等用户输入完之后 鼠标点出输入框才会更新 并不会实时更新
.number 数字 将用户输入的内容转换为数值类型 如果用户输入非数值的时候 则返回NaN
.trim 修剪 自动去掉用户输入内容两端的空格
<div class="myApp">
<h1>用户注册</h1>
<div>
<label for="username">用户:</label>
<input type="text" id="username" v-model.lazy="username" @change="checkUsername">
<span v-if="checkUsernameOK">可注册</span>
</div>
<div>
<label for="age">年龄:</label>
<input type="number" id="age" v-model.number="age">
</div>
<div>
<label for="content">个人简介:</label><br>
<textarea id="content" v-model.trim="content" cols="55" rows="8"></textarea>
</div>
<h4>信息区</h4>
<p>{{username}}</p>
<p>{{age}}</p>
<p><pre>{{content}}</pre></p>
</div>
<script>
var myApp = new Vue({
el:'.myApp',
data:{
username:'',
checkUsernameOK:false,
age:'',
content:''
},
methods:{
checkUsername:function () {
if (this.username.lenght > 0) this.checkUsernameOK = true;
else this.checkUsernameOK = false;
}
}
})
</script>
【组件系列】
1、组件:基础的基础 Component,portlet
组件就是页面上的一小块区域内容 完成一个小的页面功能 视频第六课 比如天气预报
注册了一个组件 但是这个组件一定是要在实例 new myApp之前才行
<div class="myApp">
<today-weather></today-weather>
</div>
<script>
Vue.component('today-weather',{
template:'<div>今天下雨 出不去啦 干什么呢?看Youtube吧!</div>>' //这里是模板内容
})
var myApp = new Vue({
el:'.myApp'
})
</script>
2、组件:局部的组件
Vue.js的组件不仅可以单独声明注册使用 还可以在Vue实例中进行局部注册使用
<div class="myApp">
<my-weather></my-weather>
</div>
<script>
var WeatherComponent = {
template:'<div>今天下雨 随他去吧</div>'
}
var myApp = new Vue({
el:'.myApp',
data:{
},
components: {
'my-weather': WeatherComponent
}
})
</script>
3、组件:表格组件
制作表格组件 为自己的页面表格编写表格组件
<tr is="my-row1"></tr> 需要把组件注册到tr标签里 不然自己定义的组件不能在表格里显示
如果想在表格里使用组件 需要使用is属性
<div class="myApp">
<h1>2017年最期待的游戏是</h1>
<table border="1">
<tr>
<td>编号</td>
<td>游戏名称</td>
</tr>
<tr is="my-row1"></tr>
<tr is="my-row2"></tr>
<tr is="my-row3"></tr>
</table>
</div>
<script>
Vue.component('my-row1',{
template: '<tr><td>(1)</td><td>塞尔达传说:荒野之息(3/3)</td></tr>'
})
Vue.component('my-row2',{
template: '<tr><td>(2)</td><td>新马里奥赛车(4/28)</td></tr>'
})
Vue.component('my-row3',{
template: '<tr><td>(3)</td><td>喷射乌贼娘二代</td></tr>'
})
var myApp = new Vue({
el:'.myApp'
})
</script>
4、组件:数据函数
为vue.js组件添加数据 使组件可以动态显示各种数据 注意是数据函数 不是数据属性
数据函数 返回一个属性值对象
<div class="myApp">
<div>今天的天气是<today-weather/></div>
</div>
<script>
Vue.component('today-weather',{
template: '<strong>{{todayWeather}}</strong>',
data:function(){
return {todayWeather: '雨夹雪'}
}
})
var myApp = new Vue({
el:'.myApp'
})
</script>
5、组件:传递数据
为组件传递数据 制作课接受参数的组件
props:['score'] 为这个组件创建了一个score的属性
:score="50"给这个属性绑定了50这个值
<div class="myApp">
<h1>你的成绩</h1>
<test-result :score="50"></test-result>
<test-result :score="65"></test-result>
<test-result :score="90"></test-result>
<test-result :score="100"></test-result>
</div>
<script>
Vue.component('test-result',{
props:['score'],
template: '<div><strong>{{score}}分 {{testResult}}</strong></div>',
computed: {
testResult: function(){
var strResult = ''
if(this.score < 60)
strResult = '不及格'
else if(this.score < 90)
strResult = '中等生'
else if(this.score <= 100)
strResult = '优秀生'
return strResult
}
}
})
var myApp = new Vue({
el:'.myApp'
})
</script>
6、组件:传递变量
为组件传递变量数据 制作可接受变量参数的组件
<div class="myApp">
<div>请输入您的名字:<input v-model="myname"></div>
<say-hello :pname="myname" />
</div>
<script>
Vue.component('say-hello',{
props:['pname'],
template: '<div>您好,<strong>{{pname}}</strong></div>'
})
var myApp = new Vue({
el:'.myApp',
data:{
myname: 'Xiaolin'
}
})
</script>
7、组件:参数验证
proos:组件参数验证语法 为组件中接受到的变量进行逻辑验证
只有传递string时不采用v:bind形式 其余情况均采用v-bind形式传递
<div class="myApp">
<h1>身世之谜</h1>
<show-member-info name="koma" :age="25" :detail="{address:'china',language:'中文'}"></show-member-info>
</div>
<script>
Vue.component('show-member-info',{
props: {
name:{
type: String,
required: true // required必须输入项
},
age: {
type: Number,
validator: function(value){ //validator验证器
return value >= 0 && value <= 130
}
},
detail:{
type: Object,
default: function(){
return {
address: 'US',
language: 'English'
}
}
}
},
template:
<div>姓名:{{this.name}}<br/> 年龄:{{this.age}}岁<br/> 地址:{{this.detail.address}}<br/> 语言: {{this.detail.language}}</div>
})
var myApp = new Vue({
el:'.myApp'
})
</script>
8、组件:事件传递
父组件myApp 子组件 <add-method></add-method>
子组件通过事件给父组件传递参数
v-on 侦听组件事件 当组件触发事件后进行事件处理 用在父组件当中
$emit 触发提交事件 并将数据提交给事件侦听者 用在子组件当中
$emit触发add_event 子组件事件 把参数传给 getAddResult父组件
父组件中getAddResult:function (pval)接收到的参数就是子组件this.$emit的第二个参数[result:value]
父组件中的pval.result 也就是result:value
<div class="myApp">
<h1>人生加法</h1>
<add-method :a="6" :b="12" v-on:add_event="getAddResult"></add-method>
<h3>{{result}}</h3>
</div>
<script>
Vue.component('add-method',{
props:['a','b'],
template:'<div><button v-on:click="add">加吧</button></div>',
methods:{
add:function () {
var value = 0
value = this.a + this.b
this.$emit('add_event',{
//这里result 等于新声明了一个变量 和下面的pval.result对应
result:value
})
}
}
})
var myApp = new Vue({
el:'.myApp',
data:{
result: 0
},
methods:{
getAddResult:function (pval) {
this.result = pval.result
}
}
})
</script>
9、组件:slot插槽
slot是父组件与子组件的通讯方式 可以将父组件的内容显示在子组件当中
slot 内容有[你的视频做的太差了,千万不要学koma,你教教他们两个吧]这些内容显示在父组件的后面
你好 koma! 你的视频做的太差了
你好 mike! 千万不要学koma
你好 jojo! 你教教他们两个吧
<div class="myApp">
<say-to pname="koma">
你的视频做的太差了
</say-to>
<say-to pname="mike">
千万不要学koma
</say-to>
<say-to pname="jojo">
你教教他们两个吧
</say-to>
</div>
<script>
Vue.component('say-to',{
props:['pname'],
template: <div> 你好 <strong>{{pname}}</strong>! <slot></slot> </div>
})
var myApp = new Vue({
el:'.myApp',
})
</script>
10、组件:组合slot
slot命名 在子组件中通过为多个slot进行命名 来接受父组件的不同内容的数据
<div class="myApp">
<nba-all-stars c="奥尼尔" d="加内特">
<span slot="e">皮尔斯</span>
<span slot="f">杜兰特</span>
<span slot="g">隆多</span>
</nba-all-stars>
</div>
<script>
Vue.component('nba-all-stars',{
props: ['c','d'],
template:<div> <div>中锋:{{c}}</div> <div>大前:{{d}}</div> <div>小前:<slot name="e"></slot></div> <div>分卫:<slot name="f"></slot></div> <div>控卫:<slot name="g"></slot></div> </div>
})
var myApp = new Vue({
el:'.myApp'
})
</script>
以上入门
VUE进阶
渐进式框架 Vue的核心库只关注视图层 不仅易于上手 还便于与第三方库或既有项目整合 另一方面
当与现代化的工具链以及各种支持类库结合使用时 Vue 也完全能够为复杂的单页应用提供驱动
cli:vue.js命令行方式的使用和开发
vue-router: 路由框架
vuex:状态管理
transition: 动画效果
vue安装命令行
最新版本确认: npm show vue-cli
[sudo] npm install -g vue-cli@2.9.6
安装版本确认: vue -V
命令行帮助: vue -h 使用如:vue help init
1、init 初始化
2、list 模板列表
3、build 发布
4、help[cmd] 帮助
vue命令行
vue init webpack 项目名称 推荐使用
vue list 就是显示全部模板
vue init webpack my-web [使用webpack模板初始化了一个叫my-web的工程]
vue init username/repo my-project [使用自己的gitHub的 用户名/子项目名 工程名 初始化了一个工程]
wabpack 模板工程
创建vue工程 cd到所需要创建工程的目录
vue init webpack myweb
创建工程选项:
1、工程名 Yes
2、确认Vue工程 Yes
3、自己的账户 邮箱 Yes
4、(1)编译器 (2)运行时 选择1
5、路由 Yes
6、ESlint 语法检测器 no
7、单元测试 no
8、e2e 测试 no
9、(1)npm包管理 (2)yarn包管理 选1
目录结构
build/ 编译
config/ 设置
src/ 源代码 重要
static/ 静态目录 放静态文件 比如图片 css
test/ 测试目录
e2e/ 基于浏览器测试
babelrc/ 基于es6语法设置
index.js 初始化的文件
packge.json node.js的包管理文件
pre-processors 预处理 sass less
packge.json文件里的"vue": "^2.5.2" 版本太老 去github里去看修改 目前是2.6.10
"2.5.2"这里的 删除了就是精确指定版本 推荐删除
cd myweb 进入项目
npm run dev 开启本地服务器
npm run build 打包加密发布 会创建dist文件夹
引入bootstrap4框架
cd到 myweb
npm install bootstrap --save --save-exact 精确安装版本
编辑 main.js 应用程序入口文件 入口文件 就是首页 一个网页的入口 全局变量在这里声明
编辑 main.js 入口文件js 导入bootstrap的CSS 和相关依赖
import Vue from 'vue'
import App from './App'
import router from './router'
import 'bootstrap/dist/css/bootstrap.min.css'
编辑 app.vue 入口组件文件html 添加三个按钮 :
<button class="btn btn-primary">确定</button>
<button class="btn btn-success">使用</button>
<button class="btn btn-danger">注意</button>
这里入口文件可以想成本来一个文件 有html 有 js 但是分成了两个文件 都是入口文件
入口html文件是 App.vue
入口js文件是main.js
npm run dev
【可选知识】 Bootstrap + Vue
https:// bootstrap-vue.js.org/
使用ajax库 axios
为网页增加远程数据存取能力ajax 或 http库axios(用于 浏览器 或 node.js)
导入axios库 从服务器端取得数据
cd myweb
npm install --save --save-exact axios vue-axios
vue-axios是对axios进行封装的vue组件 装上之后能够简单的使用axios库
注册组件 在main.js中添加
import axios from 'axios'
import VueAxios from 'vue-axios'
Vue.use(VueAxios,axios) 在vue的全局区域中注册这两个组件 等于就是给vue安装组件 注册完之后就能然所有的组件访问axios库 而上面的是安装导入文件
编辑组件 components组件文件夹下的 helloworld.vue文件 删除div内所有内容 style所有内容
在template中编辑
<pre>{{content}}</pre>
在script中编辑
export default {
name: 'HelloWorld',
data () {
return {
content:''
}
},
//mounted 安装
mounted() {
this.axios.post("http://api.komavideo.com/news/list").then(body => {
this.content = body.data
})
}
}
使用这个axios库post提交给这个API("http://api.komavideo.com/news/list")获得一段数据body
然后把数据的值赋值给content变量 在pre标签里显示出来
从这个网址中取数据 返回一个body 然后这个body.data属性 赋值给 content内容
自定义css样式
定义属于自己的css样式文件 my.css 入口文件的css
入口文件app.vue 写html
入口js文件 main.js 写js
此次课程是指入口文件的css
进入 assets目录 新建my.css文件 添加
.myclass1{color: red;
border: 1px solid green
}
app.vue入口文件 script里import css 相对路径(必须)
import './assets/my.css'
在template添加html
<div class="myclass1">您好!</div>
开启服务器后 修改代码 能自动刷新页面 不用修改后再关闭开启 = 热更新 或 实时更新
组件的结构
vue命令行开发模式为我们带来的组件开发方式
开发各种各样的vue组件 拼装组件生成页面 命令行开发特点
所有的组件都被统一放在工程中的组件文件夹中
{myproject}/src/components/
组件格式
template: 组件html内容
script: 组件js脚本(es6)
style: 组件css样式单
html部分
<div>
<pre>{{msg}}</pre>
</div>
js部分
improt...各种
输出这个组件:
export default{
name: 'HelloWorld', //组件名
data() { //组件的数据 不是属性 是方法
return{
msg: 'hello' //return java脚本的对象 一个变量名
}
}
}
css部分
scoped代表这些css只应用到自己这个组件 推荐添加
<style scoped>
</style>
一个组件
<template>
<div class="hello">
<pre class="class1">{{msg}}</pre>
</div>
</template>
<script>
export default {
name: 'HelloWorld',
data () {
return {
msg: 'hello World'
}
}
}
</script>
<style scoped>
.class1 {
color: red;
}
</style>
工程的结构
webpack模板生成的工程结构
工程目录
build = webpack设置文件 : 作webpack打包工具的设置 几乎不用改
config = 工程设置文件 :设置工程 如端口号8080换成8081 打包时这个css文件想使用sass还是less该怎么做 插件 在这里设置
src = 源代码文件 : assets资源文件 图片 css components组件文件夹 router路由文件夹 vuex状态管理
static = 静态文件夹(编译时直接拷贝至发布文件夹) :比如放一些谷歌的html 一些万年不变的文件放在这里
index.html = 网页启动入口文件 :单页应用程序 单就这一个页面index.html body不要改 head可以改
1、路由 vue-router
路由组件vue-router
安装
npm i -E vue-router
但是在创建工程之初已经安装过vue-router了
导入 //在 ./touter/index.js导入 文件里已经默认导入
import Vue from 'vue'
import Router from 'vue-router'
注册到全局 ////在 ./touter/index.js导入 文件里已经默认注册
Vue.use(Router)
输出一个Router类 这个Router是定义各种url的映射
export default new Router({
routes: [
{
path: '/', //url /根目录
name: 'HelloWorld', //根目录后面的url /HelloWorld
component: HelloWorld //绑定导入的 HelloWorld组件
}
]
})
流程
1、组件写好
2、路由文件导入组件路径 import HelloWorld from '@/components/HelloWorld'
3、把这个导入的组件 绑定给某一个url 这样完成url对于组件的映射
{
path: '/',
name: 'HelloWorld',
component: HelloWorld
}
使用路由
1、在main.js 里导入相对路径 import router from './router' //这里已经默认导入
2、下面是整个程序的入口组件 把刚刚导入的 router 在入口组件里注册
意味着今后所有的子组件 都能使用到 或 访问到 这个router对象 //这里已经默认导入
new Vue({
el: '#app',
router, // 注册router 已经默认注册
components: { App },
template: '<App/>'
})
2、静态路由
做一个简单的路由
1、组件文件夹创建文件
HelloWorld.vue
News.vue
About.vue
每个组件文件添加<h1>News Page</h1> 一个html
2、编辑路由文件index.js 导入每个组件
import About from '@/components/About'
import News from '@/components/News'
3、然后把导入的组件绑定到url上
{
path: '/',
name: 'HelloWorld',
component: HelloWorld
},
{
path: '/about',
name: 'About',
component: About
},
{
path: '/news',
name: 'News',
component: News
}
4、编辑app.vue 总入口组件 添加html
此时使用<router-link></router-link>标签 此标签展示一个a链接 to 是链接指向哪个组件路由地址
<router-link to="/">Home</router-link>
<router-link to="/news">新闻</router-link>
<router-link to="/about">关于</router-link>
5、<router-view/> 是路由出口 路由匹配到的组件将渲染在这里 就是说上面的所有内容会显示在这个里面 这个就是一个模板
3、动态路由
动态路由是指在URL路径中含有动态参数的路由 比如说 /player/1 /player/2 /player/3
url是同一个 但是后面的参数不一样 所以结果也不一样
1、创建组件 Player.vue
//这里必须用div包裹 不包裹会编译出错
<template>
<div>
<h1>球员页面</h1>
<p>{{detail}}</p>
</div>
</template>
<script>
export default {
name: "Player",
data() {
return{
detail: {}
}
},
//组件初始化 页面装载渲染
mounted() {
//接受url参数id
this.detail = this.getPlayer(this.$route.params.uid)
},
//在路由更新前 需要调用的方法 点库里和哈登两个能动态显示不用回home
beforeRouteUpdate(to,from,next){
this.detail = this.getPlayer(to.params.uid)
//程序提交给系统
next()
},
//接受到的参数 这里做演示写死 实际从服务器获取
methods:{
getPlayer(uid){
switch (uid) {
case '1':
return {uid:1,name:'库里',point:26}
case '2':
return {uid:2,name:'哈登',point:30}
default:
return {uid:-1}
}
}
}
}
</script>
2、设置路由
import Player from '@/components/Player'
{
path: '/player/:uid',
name: 'Player',
component: Player
}
3、app.vue设置渲染
<router-link to="/player/1">库里</router-link>
<router-link to="/player/2">哈登</router-link>
4、嵌套路由
指在动态路由的基础上再加上附加的嵌套url (就是加上 子组件) 比如说(/player/:uid/*)
/player/1/profile 或 /player/1/stats 等等
1、创建对于Pleyer.vue组件的子文件夹 Player文件夹
然后创建文件 Profile.vue + Stats.vue
2、编辑路由文件
import PlayerProfile from '@/components/Player/Profile'
import PlayerStats from '@/components/Player/Stats'
{
path: '/player/:uid',
name: 'Player',
component: Player,
//添加孩子 子组件 这里['/profile']会出错
children: [
{
path: 'profile',
component: PlayerProfile
}, {
path: 'stats',
component: PlayerStats
}
]
}
3、编辑父组件Player
<template>
<div>
<h1>球员页面</h1>
<ul>
<li>编号:{{detail.uid}}</li>
<li>名字:{{detail.name}}</li>
<li>得分:{{detail.point}}</li>
</ul>
<router-link :to="profile">简介</router-link>
<router-link :to="stats">数据</router-link>
<router-view></router-view>
</div>
</template>
<script>
export default {
name: "Player",
data() {
return{
//这里添加了属性
detail: {},
profile: '',
stats: ''
}
},
//组件初始化 页面装载渲染
mounted() {
//接受url参数id
this.detail = this.getPlayer(this.route.params.uid + '/profile'
this.stats = '/player/' + this.$route.params.uid + '/stats'
},
//在路由更新前 需要调用的方法 点库里和哈登两个能动态显示不用回home
beforeRouteUpdate(to,from,next){
this.detail = this.getPlayer(to.params.uid)
//添加路由方法
this.profile = '/player/' + to.params.uid + '/profile'
this.stats = '/player/' + to.params.uid + '/stats'
//程序提交给系统 交接给下一个事件
next()
},
//接受到的参数 这里做演示写死 实际从服务器获取
methods:{
getPlayer(uid){
switch (uid) {
case '1':
return {uid:1,name:'库里',point:26}
case '2':
return {uid:2,name:'哈登',point:30}
default:
return {uid:-1}
}
}
}
}
</script>
5、路由编程
在前面的讲解中是通过router-link标签来生成页面的a标签 然后进行url转向的 其实我们不仅可以通过标签来实现
还可以通过编程的方式来实现
router-link
router.push(location,onComplete,onAbort?)
1、编辑文件 App.vue
添加html
:to="{}" 这个:to是后面需要一个对象 和下面click方法里的 所有push 完全一样
可以省略click事件写法 直接写在标签里
<button @click="btnClick(1)">库里2</button>
<button @click="btnClick(2)">哈登2</button>
<router-link :to="{name:'Player',params:{uid:1}}">库里3</router-link>
<router-link :to="{path:'/Player/2/stats'}">哈登3</router-link>
<script>
import './assets/my.css'
export default {
name: 'App',
methods:{
//上面的@Click方法
btnClick(uid){
click事件链接到库里1页面 或者 stats页面
//this.{uid}}) //this.$router.push({path:
/player/${uid}/stats`})
会重定向到player组件里面去 同时传参数uid player组件接受的参数.toString()
//this.$router.push({name:'player', params:{uid:uid}})
url-get参数写法 player?uid=1 player?uid=2
//this.$router.push({path: '/player',query:{uid:uid}})
url履历 历史 控制 (-1)往后走 ro (1)往前走
//this.$router.go(1)
}
}
}
</script>
$router是 router路由对象
6、多路由拼图
为一个路由(画面)组合多个组件 完成整个页面 用组件进行页面内容的组合
router-view[name] 用于绑定某一个路由中的某一个组件
components
1、创建组件 创建文件夹setting
Detail.vue
Header.vue
Sidebar.vue
2、路由导入组件 index.js
import SettingHeader from '@/components/setting/Header'
import SettingDetail from '@/components/setting/Detail'
import SettingSidebar from '@/components/setting/Sidebar'
修改根路径component为components设置多个组件{} 起一个名字mymyHeader后面是导入的组件名
export default new Router({
routes: [
{
path: '/',
name: 'HelloWorld',
components: {
myHeader: SettingHeader,
mySidebar: SettingSidebar,
myDetail: SettingDetail
}
3、App.vue 使用渲染组件 这里使用<router-view name="myHeader">
<table width="100%">
<tr>
<td colspan="2" style="background-color: darkgoldenrod;">
<router-view name="myHeader"></router-view>
</td>
</tr>
<tr>
<td width="20%" style="background-color: thistle;">
<router-view name="mySidebar"></router-view>
</td>
<td width="80%" style="background-color: aquamarine;">
<router-view name="myDetail"></router-view>
</td>
</tr>
</table>
7、URL重定向
重定向指令:redirect 设置一个路由 给这个路由重新指定一个地址('/player/1') 或 组件{name: 'About'}
路由别名:alias 就是给一个路由地址设置多个名字 都指向一个地址
1、创建组件文件About1.vue
<template>
<div>
<h1>About page</h1>
</div>
</template>
2、路由设置
import About1 from '@/components/About1'
{
path: '/about1',
name: 'About1',
component: About1,
//alias别名 about1能访问 同时 /aboutme也能访问
alias: '/aboutme'
},
{
//创建了一个路由内容 路由敲上/curry 路由就能重定向(redirect)到'/player/1' 这里
path: '/curry',
redirect: '/player/1'
这里是重定向到About1组件 两种都可以
//redirect: {name: 'About1'}
//redirect: '/about1'
}
3、App.js入口文件调用
<div id="app">
<router-link to="/">Home</router-link>
<router-link to="/aboutme">About</router-link>
<router-link to="/player/1">Player1</router-link>
<router-link to="/curry">Curry</router-link>
<hr>
<router-view></router-view>
</div>
8、多参数路由
路由属性props
$route.params.uid = 路由对象的.参数属性的.uid 取到这个uid的值
用这种方式 能够为一个url传递多个参数
1、创建User.vue 组件
<template>
<div>
<h1>User</h1>
<p>uid={{uid}},{{nationality}}</p>
<p>route.params.uid}}</p>
<p>route.params.nationality}}</p>
</div>
</template>
<script>
export default {
name: "User",
props: ['uid','nationality']
}
</script>
2、编辑路由 index.js
import User from '@/components/User'
{
//接收两个参数
path: '/user/:uid/:nationality',
name: 'User',
component: User,
//这里属性设置true
props: true
}
3、编辑 App.vue
<div class="App2">
<router-link to="/">Home</router-link>
<router-link to="/User/1/usa">User1</router-link>
<router-link to="/User/2/china">User2</router-link>
<router-link to="/User/3/korea">User3</router-link>
</div>
Vue.cli 3
1、推荐配置
推荐node.js 8.11.0+以上
node.js -v 查看版本命令
2、如果安装了1.x或者2.x版本的需要先卸载
npm uninstall vue-cli -g 卸载命令
3、安装
npm install -g @vue/cli 安装命令
vue -V 查看版本
4、vue-cli 3 创建项目
vue create hello-world(项目名) 初始化一个项目
5、安装选项
(1)默认或者自定义 选2
(2)自定义各种插件 空格选中 自选
(3)类风格语法 yes
(4)babel typeScript polyfills Yes
(5)选packge.json
(6)no
6、启动新项目
cd hello-world
npm run serve
7重点
(1)学习 typescript
(2)尽量使用vue 3
(3)typescript 三个流行框架都可以开发
VUE生命周期钩子函数 钩子hook = 回调
vue实例 创建、运行、到销毁期间的事件 这些事件统称为生命周期 钩子就是生命周期事件的别名
创建期间的生命周期函数:
1 beforeCreate() 实例刚在内存中被创建出来 此时 data 和 methods中的数据都还没有初始化
2重要 created() 实列在内存中创建OK data 和methods 被初始化好了 开始编译模板 一般会做一些网络请求
3 beforeMount() 模板已经在内存中编辑完成了 但是还没有把模板渲染到页面中
4重要 mounted() 此时 已经将编译好的模板 挂载到了页面指定的容器中显示 可以看到渲染好的页面
运行期间的生命周期函数:
5 beforeUpdate() data改变时触发 页面中显示的数据还没有更新 但是data数据被更新了
6 updated() 触发时 页面和data数据都已经保持同步最新
销毁期间的生命周期函数:
beforeDestroy() 触发时 vue实例就已经从运行阶段 进入到销毁阶段 实例上 所有的 data和methods和 过滤器 和指令...都处于可用状态 还没有真正执行销毁的过程
destroyed() 触发时 组件已经被完全的销毁了 此时组件中所有的data和methods 指令 过滤器...都已经不可用
Vue路由全局守卫
由一个路由跳转到另一个路由 监听这个过程 在跳转期间做一件事
前置钩子(回调) 跳转前回调
router.beforeEach((to,form,next) => {
//从from跳转到to
document.title = to.metched[0].meta.title
next() //必须先调用一下 可以传入false不跳转 或者 '/'跳转到某一个路由
})
后置钩子(hook) 跳转后回调
router.afterEach((to,from) => {
})
路由独享的守卫
只有我进入某一个路由内 才会进行一个回调
const router = new VueRouter({
touters: [
{
path: 'foo',
component: Foo,
beforeEnter: (to,from,next) => {
//...
}
}
]
})
组件内的守卫
Vuex 多页面状态管理 所有组件的共享变量 而且还是响应式的 是插件 需要安装
需要多个组件共享的变量 使用一个外置的公共储存状态的实例
变量是响应式的 数据一旦改变 界面自动更新
管理什么状态? 多个用户界面都需要用到的一些状态 如用户名称 登录状态 商品收藏类
用户登录 用户名称 头像 地理位置等等
商品的收藏 购物车中的物品等等
单页界面的状态管理
state data数据 也就是状态
view 可以针对state的变化 显示不同的信息
actions 各种操作 点击 输入 等等行为 能改变状态state
开始
npm install vuex --save
1在src文件夹目录下 创建一个文件夹 store(仓库) 作为vuex的文件夹
2创建index.js文件
import Vue from 'vue'
import Vuex from 'vuex'
1/安装插件 底层会执行 Vuex.install = '' vue.prototype.$store = store
Vue.use(Vuex)
2.创建对象
const store = new Vux.store({
state: {
const: 1000 保存状态 单一状态树概念(单一数据源)
},
mutations: {
方法
使用mutations修改状态 会有一个浏览器插件(Devtools) 去谷歌商店搜索安装
会记录哪个组件 是谁修改的 以便于更好的跟踪 查错 这里是同步操作
increment(默认会有一个参数state ) {
state.const++
},
decrement() {
state.const--
},
updateinfo(){
this.store.dispatch('aUpdateinfo')
},1000)
}
},
getters: {
类似组件的计算属性 从state里拿到的数据 这个数据是需要经过变化的 再给别人用的 这种情况用getters
powerCounter(默认会有一个参数state ) {
return state.counter * state.counter
}
},
modules: {
划分模块 针对不同的模块 在这里面再做一些相关数据的保存 可以把这个a模块抽离出来成一个对象变量 这个变量里有这些对象 这里a使用这个外面写的对象变量
使用的时候$store.state.a commit只会调用自己模块的东西 context也是
a: {
state:{},
mutations{},
actions{},
getters{}
} ,
b: {
state:{},
mutations{},
actions{},
getters{}
}
}
})
3.导出store独享
export default store
回到main.js文件 挂载 import store from './store'
把store放到 el下面
然后需要用到的状态就可以在组件里使用 {{ store.commit('increment'.count)这里第二个参数 是指的app.vue传到vuex的值 参数被称为是mutation的payload 负载的意思 提交的时候顺带一个参数
mutations提交风格
mutations响应规则 只有在vuex里已经定义好的数据才能有响应式 其他操作添加的数据没有响应式 但是可以用push/pop/splice()/set()
Vue.set(state.info,'address','洛杉矶') 响应式添加 界面刷新
Vue.delete(state.info, 'age') 响应式删除
mutations常量类型
getters状态的获取
$store.getters.powerCounter
Vue性能优化9法
函数型组件
优点:普通写法会有vue生命周期的各种处理 函数型组件会作为一个函数处理 使页面效率提高
<template functional>
<div class='cell'>
<div v-if='proos.value' class='on'></div>
<section v-else class='off'>
</div>
</template>
<script>
export defailt {
props:['value']
}
</script>
子组件拆分
优点:Vue生命周期 重处理给拆分成子组件
局部变量
优点:把计算属性在 方法函数里赋值给常量 引用常量提高效率
活用v-show,减少v-if
优点:v-show是不操作dom树 性能比较高 v-if是动态改变dom树的内容 渲染的性能会比较低
使用keep-alive 保持活动
优点:把一些不动的如<router-view> 给包裹住 保持 提高页面渲染的效率
活用延迟加载(Defer)
优点:把一部分逻辑先执行 让一部分逻辑后执行 顺序执行
分批处理(Time slicing)
优点:把传入的所有未处理数据 分成几次 几批处理 提高画面渲染性能
非响应式模式(非观察模式,non-reactive)
优点:因为vue会实时动态观察一个双向属性 所以把一个数据的观察者模式关闭
仅渲染可视化部分
优点:先显示用户能看到的数据 再加载用户看不到的数据
优化心得
1.简单的单页应用可以忽略优化细节,重点可放在功能的实现上
2.但是对于上百个对象的协作交互渲染的应用时 就该考虑对于每个细节部分 来考虑如何进行优化
3.同时 如果影响页面显示 渲染的操作 比如 帧数>60帧 为了防止影响用户的操作体验 必须考虑到优化细节
4.对于拥有大量计算或者大量渲染的耗时操作,应该随时计算测试处理的时间 积极思考优化策略 哪怕只能提高1秒
监听事件
在外面写组件的 <Cell @click='onClickCell'>
methods:{
onClickCell(){
console.log('某个cell被点了')
}
}
在组件里面写<div class='cell' @click='y'>
methods: {
y() {
this.a = true;
this.$emit('click',this.text)
}
}
外面接收的话要在相应的地方写$event 就能收到里面的这个this.text 只能传事件 不能传值
计算偶数是什么 奇数是什么
this.text = this.n % 2 === 0 ? 'x' : 'o'
如果点击之后 不等于空的话 那么再点击就什么都不做
if(this.text !== ''){
return
}
MVVM
什么是mvvm Model View ViewModel
{{表达式 + " " + 相加}}
{{数值 * 2 }}