品牌管理案例
添加新品牌
代码实例
add () {
//分析:
//1. 获取 id 和 name ,直接从data上获取
//2. 组织出一个对象
//3. 把这个对象,调用数组的相关方法,添加到当前的data上的list中去
//4. 注意:在Vue中,已经实现了数据的双休绑定,每当我们修改了data中的数据,
// Vue会默认监听到数据的改动,自动把最新的数据,应用到页面上
//5. 当我们意识到上面的第四步的时候,就证明大家已经入门Vue了,我们更多的是在进行VM中的Model数据的操作,
// 同时,在操作Model数据的时候,指定的业务逻辑操作
var car = {id:this.id, name:this.name, ctime:new Date()}
this.list.push(car)
this.id = this.name = ''
}
删除品牌
代码实例
del (id) {//根据id删除数据
//分析:
//1. 如何根据id找到要删除对象的索引
//2. 如果找到了索引,直接调用数组的splice方法
//方法1
// this.list.some((item, i)=>{
// if(item.id == id){
// this.list.splice(i, 1)
// //在数组的some方法中,如果return true,就会立即终止这个数组的后续循环
// return true;
// }
// })
//方法2
var index = this.list.findIndex(item => {
if(item.id == id){
this.list.splice(index, 1)
return true
}
})
}
根据条件筛选品牌
1. 1.x版本中的[fiterBy指令](https://cn.vuejs.org/api/#fiterBy),在2.x中已经被废除
2. 在2.x版本中[手动实现筛选的方式](https://cn.vuejs.org/v2/guide/list.html#显示过滤-排序结果)
过滤器
概念:Vue.js允许你自定义过滤器,可被用做一些常见的文本格式化。过滤器可以用在两个地方:mustache插值器和v-bind表达式。过滤器应该被添加在JavaScript表达式的尾部,由"管道"符知识;
私有过滤器
- HTML元素
<td>{{item.ctime | dataFormat('yyyy-mm-dd')}}</td>
- 私有
filters定义方式
代码实例
//自定义一个私有的过滤器
var vm2 = new Vue({
el: '#app2',
data () {
return {
dt: new Date()
}
},
filters:{//定义私有过滤器 过滤器有两个条件 【过滤器名称和处理函数】
//过滤器调用的时候,采用的是就近原则,如果私有过滤器和全局过滤器名称一致了,这时候,优先调用私有过滤器
dateFormat:function(dateStr, pattern = ""){
//根据给定的时间字符串,得到特定的时间
var dt = new Date(dateStr)
// yyyy-mm-dd
var y = dt.getFullYear()
var m = (dt.getMonth() + 1).toString().padStart(2, '0')
var d = (dt.getDate()).toString().padStart(2, '0')
// return y + '-' + m + '-' + d
if(pattern.toLowerCase() == 'yyyy-mm-dd'){
return `${y}-${m}-${d}`
}else{
var hh = dt.getHours()
var mm = (dt.getMinutes()).toString().padStart(2, '0')
var ss = (dt.getSeconds()).toString().padStart(2, '0')
return `${y}-${m}-${d} ${hh}:${mm}:${ss}~~~~`
}
}
}
})
使用ES6中的字符串新方法String.prototype.padStart(maxLength, fillString = ' ')或String.prototype.padEnd(maxLength, fillString = ' ')来填充字符串;
全局过滤器(代码实例)
//全局的过滤器,进行时间的格式化
//所谓的全局过滤器,就是所有的VM实例都共享
Vue.filter('dateFormat', function(dataStr, pattern = ""){
//根据给定的时间字符串,得到特定的时间
var dt = new Date(dataStr)
// yyyy-mm-dd
var y = dt.getFullYear()
var m = dt.getMonth() + 1
var d = dt.getDate()
// return y + '-' + m + '-' + d
if(pattern.toLowerCase() == 'yyyy-mm-dd'){
return `${y}-${m}-${d}`
}else{
var hh = dt.getHours()
var mm = dt.getMinutes()
var ss = dt.getSeconds()
return `${y}-${m}-${d} ${hh}:${mm}:${ss}`
}
})
注意:当有局部和全局两个名称相同的过滤器时候,会以就近原则来进行调用,即:局部过滤器优先于全局过滤器被调用
按键修饰符以及自定义键盘修饰符
- x中自定义键盘修饰符【了解即可】
Vue.directive('on').keyCode.f2 = 113;
- 通过
Vue.config.keyCodes.名称 = 按键值来自定义按键修饰符的别名:
Vue.config.keyCodes.f2 = 113;
- 使用自定义的按键修饰符:
<input type = "text" v-model = "name" @keyup.f2 = "add">
自定义指令
全局自定义指令
- 使用 Vue.directive() 定义全局的指令
- 参数1: 指令的名称,注意,在定义的时候,指令的名称前面,不需要加 v- 前缀但是,在调用的时候必须在指令名称前加上 v- 前缀
- 参数2:是一个对象,这个对象身上,有一些自定相关的函数,这些函数可以在特定的阶段,执行相关的操作
代码实例
<input type="text" class="form-control" v-model="keywords" v-focus v-color="'blue'">
Vue.directive('focus', {
bind: function (el) {//每当指令绑定到元素上的时候,会立即执行这个bind函数,只执行一次
//注意:在每个函数中,第一个参数,永远是el, 表示被绑定了指令的那个元素,这个el参数,是原生的JS对象
//在元素刚绑定了指令的时候,还没有插入到DOM中去,这时候,调用focus方法没有作用
//因为,一个元素,只有插入了DOM之后,才能获取焦点
// el.focus()
},
inserted: function (el) {//inserted表示元素 插入到DOM中的时候,会执行inserted 函数,只执行一次
el.focus()
//和JS行为有关的操作,最好在 inserted 中去执行,防止 JS行为不生效
},
updated: function (el) { //当Vnode更新的时候,会执行 updated ,可能会触发多次
}
})
自定义私有指令
语法结构和自定义私有过滤器类似
directives:{//自定义私有指令
'fontweight':{//设置字体粗细的
bind: function(el, binding){
el.style.fontWeight = binding.value
}
},
//下面这个是简写
'fontsize': function(el, binding){//注意:这个function等同于把代码写到了bind和update中去
el.style.fontSize = parseInt(binding.value) + 'px'
}
}
品牌管理案例全部代码实例
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>Document</title>
<script src='https://cdn.bootcss.com/vue/2.6.10/vue.min.js'></script>
<link href="https://cdn.bootcss.com/twitter-bootstrap/3.3.7/css/bootstrap.min.css" rel="stylesheet">
</head>
<body>
<div id="app">
<div class="panel panel-primary">
<div class="panel-heading">
<h3 class="panel-title">添加品牌</h3>
</div>
<div class="panel-body form-inline">
<label>
Id:
<input type="text" class="form-control" v-model="id">
</label>
<label>
Name:
<input type="text" class="form-control" v-model="name" @keyup.enter="add">
</label>
<!-- 在Vue中,使用事件绑定机制, 为元素指定处理函数的时候,如果加了小括号,就可以给函数传参了-->
<input type="button" value="添加" class="btn btn-primary" @click="add()">
<label>
搜索名称关键字:
<!-- 注意:Vue中所有的指令,再调用的时候,都以 v- 开头 -->
<input type="text" class="form-control" v-model="keywords" v-focus v-color="'blue'">
</label>
</div>
</div>
<table class="table table-bordered table-hover table-striped">
<thead>
<tr>
<th>Id</th>
<th>Name</th>
<th>Ctime</th>
<th>Operation</th>
</tr>
</thead>
<tbody>
<!-- 之前, v-for中的数据,都是直接从data上的list中直接渲染过来的 -->
<!-- 现在,我们自定义了一个search方法,同时,把所有的关键字,通过传参的形式,传递给了search方法 -->
<!-- 在search方法内部,通过执行for循环,把所有符合搜索关键字的数据,保存到一个新数组中,返回 -->
<tr v-for="item in search(keywords)" :key="item.id">
<td>{{item.id}}</td>
<td v-text="item.name"></td>
<td>{{item.ctime | dateFormat()}}</td>
<td><a href="" @click.prevent="del(item.id)">删除</td>
</tr>
</tbody>
</table>
</div>
<div id="app2">
<h3 v-color = "'pink'" v-fontweight = "900" v-fontsize = "50">{{dt | dateFormat}}</h3>
</div>
<script>
//全局的过滤器,进行时间的格式化
//所谓的全局过滤器,就是所有的VM实例都共享
Vue.filter('dateFormat', function (dataStr, pattern = "") {
//根据给定的时间字符串,得到特定的时间
var dt = new Date(dataStr)
// yyyy-mm-dd
var y = dt.getFullYear()
var m = dt.getMonth() + 1
var d = dt.getDate()
// return y + '-' + m + '-' + d
if (pattern.toLowerCase() == 'yyyy-mm-dd') {
return `${y}-${m}-${d}`
} else {
var hh = dt.getHours()
var mm = dt.getMinutes()
var ss = dt.getSeconds()
return `${y}-${m}-${d} ${hh}:${mm}:${ss}`
}
})
//自定义全局按键修饰符
Vue.config.keyCodes.f2 = 113
//使用 Vue.directive() 定义全局的指令
//其中:参数1 : 指令的名称,注意,在定义的时候,指令的名称前面,不需要加 v- 前缀
//但是,在调用的时候必须在指令名称前加上 v- 前缀
//参数2:是一个对象,这个对象身上,有一些自定相关的函数,这些函数可以在特定的阶段,执行相关的操作
Vue.directive('focus', {
bind: function (el) {//每当指令绑定到元素上的时候,会立即执行这个bind函数,只执行一次
//注意:在每个函数中,第一个参数,永远是el, 表示被绑定了指令的那个元素,这个el参数,是原生的JS对象
//在元素刚绑定了指令的时候,还没有插入到DOM中去,这时候,调用focus方法没有作用
//因为,一个元素,只有插入了DOM之后,才能获取焦点
// el.focus()
},
inserted: function (el) {//inserted表示元素 插入到DOM中的时候,会执行inserted 函数,只执行一次
el.focus()
//和JS行为有关的操作,最好在 inserted 中去执行,防止 JS行为不生效
},
updated: function (el) { //当Vnode更新的时候,会执行 updated ,可能会触发多次
}
})
//自定义一个设置字体颜色的指令
Vue.directive('color', {
//样式,只要通过指令绑定给了元素,不管这个元素有没有被插入到页面中去,这个元素肯定有了一个内联的样式
//将来元素肯定会显示到页面中去,这时候,浏览器的渲染引擎必然会解析样式,应用给这个元素
bind: function(el, binding){
el.style.color = binding.value
//和样式相关的操作,一般都可以在bind执行
}
})
var vm = new Vue({
el: '#app',
data() {
return {
list: [
{ id: 1, name: '奔驰', ctime: new Date() },
{ id: 2, name: '宝马', ctime: new Date() },
],
id: '',
name: '',
keywords: ''
}
},
methods: {
add() {
//分析:
//1. 获取 id 和 name ,直接从data上获取
//2. 组织出一个对象
//3. 把这个对象,调用数组的相关方法,添加到当前的data上的list中去
//4. 注意:在Vue中,已经实现了数据的双休绑定,每当我们修改了data中的数据,
// Vue会默认监听到数据的改动,自动把最新的数据,应用到页面上
//5. 当我们意识到上面的第四步的时候,就证明大家已经入门Vue了,我们更多的是在进行VM中的Model数据的操作,
// 同时,在操作Model数据的时候,指定的业务逻辑操作
var car = { id: this.id, name: this.name, ctime: new Date() }
this.list.push(car)
this.id = this.name = ''
},
del(id) {//根据id删除数据
//分析:
//1. 如何根据id找到要删除对象的索引
//2. 如果找到了索引,直接调用数组的splice方法
//方法1
// this.list.some((item, i)=>{
// if(item.id == id){
// this.list.splice(i, 1)
// //在数组的some方法中,如果return true,就会立即终止这个数组的后续循环
// return true;
// }
// })
//方法2
var index = this.list.findIndex(item => {
if (item.id == id) {
this.list.splice(index, 1)
return true
}
})
},
search(keywords) {//根据关键字进行数据的搜索
// var newList = []
// this.list.forEach(item => {
// if(item.name.indexOf(keywords) != -1){
// newList.push(item)
// }
// });
// return newList
//注意:forEach some filter findIndex 这些都属于数组的新方法
//都会对数组中的每一项进行遍历,执行相关操作
return this.list.filter(item => {
//注意: ES6中,为字符串提供了一个新方法,叫做String.prototype.includes('要包含的字符串')
//如果包含,则返回true, 否则返回false
if (item.name.includes(keywords)) {
return item
}
})
// return newList
}
}
});
//如何自定义一个私有的过滤器()
var vm2 = new Vue({
el: '#app2',
data() {
return {
dt: new Date()
}
},
filters: {//定义私有过滤器 过滤器有两个条件 【过滤器名称和处理函数】
//过滤器调用的时候,采用的是就近原则,如果私有过滤器和全局过滤器名称一致了,这时候,优先调用私有过滤器
dateFormat: function (dateStr, pattern = "") {
//根据给定的时间字符串,得到特定的时间
var dt = new Date(dateStr)
// yyyy-mm-dd
var y = dt.getFullYear()
var m = (dt.getMonth() + 1).toString().padStart(2, '0')
var d = (dt.getDate()).toString().padStart(2, '0')
// return y + '-' + m + '-' + d
if (pattern.toLowerCase() == 'yyyy-mm-dd') {
return `${y}-${m}-${d}`
} else {
var hh = dt.getHours()
var mm = (dt.getMinutes()).toString().padStart(2, '0')
var ss = (dt.getSeconds()).toString().padStart(2, '0')
return `${y}-${m}-${d} ${hh}:${mm}:${ss}~~~~`
}
}
},
directives:{//自定义私有指令
'fontweight':{//设置字体粗细的
bind: function(el, binding){
el.style.fontWeight = binding.value
}
},
'fontsize': function(el, binding){//注意:这个function等同于把代码写到了bind和update中去
el.style.fontSize = parseInt(binding.value) + 'px'
}
}
})
</script>
</body>
</html>
Vue实例的生命周期
- 什么是生命周期:从Vue实例创建、运行、到销毁期间,总是伴随着各种各样的事件,这些事件,统称为生命周期!
- 生命周期钩子:就是生命周期事件的别名而已;
- 生命周期钩子 = 生命周期函数 = 生命周期事件
- 主要的生命周期函数分类:
- 创建期间的生命周期函数:
- beforeCreate:实例刚在内存中创建出来,此时,还没有初始化好data和methods属性
- created:实例已经在内存中创建好,此时data和methods已经创建好,此时还没有开始编译模板
- beforeMount:此时已经完成了模板的编译,但是还没有挂载到页面中
- mounted:此时,已经将编译好的模板,挂载到了页面指定的容器中显示
- 运行期间的生命周期函数:
- beforeUpdate:状态更新之前执行此函数,此时data中的状态值是最新的,但是界面上显示的数据还是旧的,因为此时还没有开始重新渲染DOM节点
- updated:实例更新完毕之后调用此函数,此时data中的状态值和界面上显示的数据都已经完成了更新,界面已经被重新渲染好了
- 销毁期间的生命周期函数
- beforeDestroy:实例销毁之前调用,在这一步,实例仍然完全可用
- destoryed:Vue实例销毁后调用,调用后,Vue实例指示的所有东西都会解绑定,所有的事件监听器会被移除,所有的子实例 也会被销毁。
- 创建期间的生命周期函数:
实例代码
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>Document</title>
<script src='https://cdn.bootcss.com/vue/2.6.10/vue.min.js'></script>
</head>
<body>
<div id="app">
<input type="button" value="修改msg" @click="msg = 'No'">
<h3 id = "h3">{{msg}}</h3>
</div>
<script>
var vm=new Vue({
el:'#app',
data () {
return{
msg: 'ok'
}
},
methods:{
show () {
console.log('执行了show方法')
}
},
beforeCreate() {//这是我们遇到的第一个生命周期函数,表示实例完全被创建出来之前,会执行它
// console.log(this.msg)
// this.show()
//注意:在beforeCreate 生命周期函数执行的时候,data和methods中的数据都还没有被初始化
},
created() {//这是遇到的第二个生命周期函数
// console.log(this.msg)
// this.show()
//在created中,data和methods都已经被初始化好了
//如果要调用methods中的方法,或者操作data中的数据,最早只能在created中操作
},
beforeMount() {//这是遇到的第三个生命周期函数,表示模板已经在内存中编辑完成了,但是尚未把模板渲染到页面中
// console.log(document.getElementById('h3').innerText)
//在beforeMount执行的时候,页面中的元素还没有被真正替换过来,只是之前写的一些模板字符串
},
mounted() {//这是遇到的第四个生命周期函数,表示内存中的模板已经真实的挂载到了页面中,用户已经可以看到渲染好的页面了
// console.log(document.getElementById('h3').innerText)
//注意:mounted是实例创建期间的最后一个生命周期函数,当执行完mounted就表示,实例已经被完成创建好了,
//此时,如果没有其他操作的话,这个实例就静静的躺在我们内存中,一动不动
},
//接下来是运行中的两个事件
beforeUpdate() {//这时候,表示我们的界面还没有被更新【数据被更新了吗? 数据肯定被更新了】
// console.log('界面上元素的内容:' + document.getElementById('h3').innerText)
// console.log('data 中的msg数据是:' + this.msg)
//得出结论:当执行beforeUpdate的时候,页面中显示的数据,还是旧的,此时data数据是最新的,页面和最新的数据还未保持同步
},
updated() {
console.log('界面上元素的内容:' + document.getElementById('h3').innerText)
console.log('data 中的msg数据是:' + this.msg)
//updated事件执行的时候,页面和data数据已经保持同步了,都是最新的
},
});
</script>
</body>
</html>
Vue-resource实现get,post,jsonp请求
除了Vue-resource之外,还可以使用axios的第三方包实现数据的请求
- 之前的学习中,如何发情数据请求?
- 常见的数据请求类型?get post jsonp
- 测试的URL请求资源地址:
- JSONP的实现原理
- 由于浏览器的安全性限制,不允许AJAX访问协议不同、域名不同、端口号不同的数据接口,浏览器认为这种访问不安全;
- 可以通过动态创建script标签的形式,把script标签的属性,指向数据接口的地址,因为script标签不存在跨域限制,这种数据获取方式,称作JSONP(注意:根据JSONP的实现原理,知晓,JSONP只支持Get请求)
- 具体实现过程:
- 现在客户端定义一个回调方法,预定义对数据的操作;
- 再把这个回调方法的名称,通过URL传参的形式,提交到服务器的数据接口
- 服务器数据接口组织好要发送给客户端的数据,再拿着客户端传递过来的回调方法名称,拼接处一个调用这个方法的字符串,发送给客户端去解析执行;
- 客户端拿到服务器返回的字符串之后,当做script脚本去解析执行,这样就能够拿到JSONP的数据了
代码实例
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>Document</title>
<script src='https://cdn.bootcss.com/vue/2.6.10/vue.min.js'></script>
<!-- 注意:vue-resource依赖于 Vue ,所以先后顺序要注意 -->
<!-- this.$http -->
<script src="https://cdn.bootcss.com/vue-resource/1.5.1/vue-resource.min.js"></script>
<!-- <script src="https://cdn.jsdelivr.net/npm/vue-resource@1.5.1"></script> -->
</head>
<body>
<div id="app">
<input type="button" value="get请求" @click="getInfo">
<input type="button" value="post请求" @click="postInfo">
<input type="button" value="jsonp请求" @click="jsonpInfo">
</div>
<script>
var vm = new Vue({
el: '#app',
data() {
return {
}
},
methods: {
getInfo() {//发请get请求
//当发起get请求之后,通过.then来设置成功的回调函数
this.$http.get('http://www.liulongbin.top:3005/api/getprodlist').then(response => {
console.log(response)
//通过result.body拿到服务器返回的请求数据
})
},
postInfo() {//发起post请求 application/x-wwww-form-urlencoded
//手动发起的post请求默认没有表单格式,所以,有的服务器处理不了
//通过post方法的第三个参数,{emulateJSON : true}设置提交的内容类型为普通表单数据格式
this.$http.post('http://www.liulongbin.top:3005/api/addproduct', {}, { emulateJSON: true }).then(result => {
console.log(result.body)
})
},
jsonpInfo(){//发起jsonp请求
this.$http.jsonp('http://vue.studyit.io/api/jsonp').then(result => {
console.log(result.body)
})
}
}
});
</script>
</body>
</html>