Vue生命周期:就是一个Vue实例从创建 到 销毁 的整个过程。 生命周期四个阶段:① 创建 ② 挂载 ③ 更新 ④ 销毁
<div id="app">
<h3>{{ title }}</h3>
<div>
<button @click="count--">-</button>
<span>{{ count }}</span>
<button @click="count++">+</button>
</div>
</div>
<script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
<script>
const app = new Vue({
el: '#app',
data: {
count: 100,
title: '计数器'
},
beforeCreate(){ }
// 1. 创建阶段(准备数据)
//进行创建响应式数据阶段created(){ }
beforeMount( ){ }
// 2. 挂载阶段(渲染模板)
//进行挂载数据阶段 mounted(){ }
beforeUpdate(){ }
// 3. 更新阶段(修改数据 → 更新视图)
//进行update阶段 updated(){ }
beforeDestroy(){ }
// 4. 卸载阶段
// destroyed(){}
})
</script>
Vue生命周期过程中,会自动运行一些函数,被称为【生命周期钩子】→ 让开发者可以在【特定阶段】运行自己的代码
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
</head>
<body>
<div id="app">
<h3>{{ title }}</h3>
<div>
<button @click="count--">-</button>
<span>{{ count }}</span>
<button @click="count++">+</button>
</div>
</div>
<script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
<script>
const app = new Vue({
el: '#app',
data: {
count: 100,
title: '计数器'
},
//TODO: 和生命周期图一同观察每个阶段的console的内容
// 1. 创建阶段(准备数据)
beforeCreate () {
console.log('beforeCreate 响应式数据准备好之前', this.count)
},
created () {
console.log('created 响应式数据准备好之后', this.count)
// this.数据名 = 请求回来的数据
// 可以开始发送初始化渲染的请求了
},
// 2. 挂载阶段(渲染模板)
beforeMount () {
console.log('beforeMount 模板渲染之前', document.querySelector('h3').innerHTML)
},
mounted () {
console.log('mounted 模板渲染之后', document.querySelector('h3').innerHTML)
// 可以开始操作dom了
},
// 3. 更新阶段(修改数据 → 更新视图)
beforeUpdate () {
console.log('beforeUpdate 数据修改了,视图还没更新', document.querySelector('span').innerHTML)
},
updated () {
console.log('updated 数据修改了,视图已经更新', document.querySelector('span').innerHTML)
},
// 4. 卸载阶段
beforeDestroy () {
console.log('beforeDestroy, 卸载前')
console.log('清除掉一些Vue以外的资源占用,定时器,延时器...')
},
destroyed () {
console.log('destroyed,卸载后')
}
})
</script>
</body>
</html>
1.基本渲染 2.添加功能 3.删除功能 4.饼图渲染
js
/**
* 接口文档地址:
* https://www.apifox.cn/apidoc/shared-24459455-ebb1-4fdc-8df8-0aff8dc317a8/api-53371058
*
* 功能需求:
* 1. 基本渲染
* 2. 添加功能
* 3. 删除功能
* 4. 饼图渲染
*/
const app = new Vue({
el: '#app',
data: {
// 对象数组
list: [
{
id: 0,
name: '',
price: 0,
creator: ''
}
],
//需要添加的单个对象内容
itemName: '',
itemPrice: ''
},
computed: {
totalCount (){
return this.list.reduce((sum, item)=>sum += item.price,0)
}
},
methods: {
// 删除数据
del(id) {
this.list = this.list.filter(item => item.id !== id)
},
add(){
console.log("商品名称:"+ this.itemName)
this.list.unshift({
id: +new Date(),
name: this.itemName,
price: this.itemPrice,
creator: 'rayce'
})
//todo: 更新试图
this.updateView()
// 清空表单
this.itemName= ''
this.itemPrice=''
},
updateView(){
// 更新图表
this.myCharts.setOption({
// 数据项
series: [
{
// data: [
// { value: 1048, name: '球鞋' },
// { value: 735, name: '防晒霜' }
// ]
data: this.list.map(item => ({ value: item.price, name: item.name}))
}
]
})
}
},
async created(){
//发送请求拿到数据, 然后进行渲染
console.log("生命周期的准备响应式数据阶段")
const res = await axios({
url: 'https://applet-base-api-t.itheima.net/bill',
params: {
creator: 'rayce'
}
})
// console.log(res.data.data)
this.list = res.data.data
this.updateView()
},
beforeCreated(){
console.log("生命周期的创建数据阶段。 发送初始化渲染请求")
},
mounted(){
console.log("生命周期的挂载数据阶段(操作dom阶段)")
//进行柄图渲染
//初始化一个饼图 echarts.init(dom) mounted钩子中渲染
//根据数据试试更新饼图 echarts.setOptions({...})
this.myCharts = echarts.init(document.querySelector('#main'))
this.myCharts.setOption({
// 大标题
title: {
text: '消费账单列表',
left: 'center'
},
// 提示框
tooltip: {
trigger: 'item'
},
// 图例
legend: {
orient: 'vertical',
left: 'left'
},
// 数据项
series: [
{
name: '消费账单',
type: 'pie',
radius: '50%', // 半径
data: [
// { value: 1048, name: '球鞋' },
// { value: 735, name: '防晒霜' }
],
emphasis: {
itemStyle: {
shadowBlur: 10,
shadowOffsetX: 0,
shadowColor: 'rgba(0, 0, 0, 0.5)'
}
}
}
]
})
}
})
/**
* TODO:
3.思路分析
1.基本渲染
立刻发送请求获取数据 created
拿到数据,存到data的响应式数据中
结合数据,进行渲染 v-for
消费统计 —> 计算属性
2.添加功能
收集表单数据 v-model,使用指令修饰符处理数据
给添加按钮注册点击事件,对输入的内容做非空判断,发送请求
请求成功后,对文本框内容进行清空
重新渲染列表
3.删除功能
注册点击事件,获取当前行的id
根据id发送删除请求
需要重新渲染
4.饼图渲染
*/
html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
</head>
<body>
<!-- CSS only -->
<link
rel="stylesheet"
href="https://cdn.jsdelivr.net/npm/bootstrap@5.1.3/dist/css/bootstrap.min.css"
/>
<style>
.red {
color: red!important;
}
.search {
width: 300px;
margin: 20px 0;
}
.my-form {
display: flex;
margin: 20px 0;
}
.my-form input {
flex: 1;
margin-right: 20px;
}
.table > :not(:first-child) {
border-top: none;
}
.contain {
display: flex;
padding: 10px;
}
.list-box {
flex: 1;
padding: 0 30px;
}
.list-box a {
text-decoration: none;
}
.echarts-box {
width: 600px;
height: 400px;
padding: 30px;
margin: 0 auto;
border: 1px solid #ccc;
}
tfoot {
font-weight: bold;
}
@media screen and (max-width: 1000px) {
.contain {
flex-wrap: wrap;
}
.list-box {
width: 100%;
}
.echarts-box {
margin-top: 30px;
}
}
</style>
<div id="app">
<div class="contain">
<!-- 左侧列表 -->
<div class="list-box">
<!-- 添加资产 -->
<form class="my-form">
<input v-model="itemName" type="text" class="form-control" placeholder="消费名称" />
<input v-model="itemPrice" type="text" class="form-control" placeholder="消费价格" />
<button @click="add()" type="button" class="btn btn-primary">添加账单</button>
</form>
<table class="table table-hover">
<thead>
<tr>
<th>编号</th>
<th>消费名称</th>
<th>消费价格</th>
<th>操作</th>
</tr>
</thead>
<tbody v-if="list.length > 0">
<tr v-for="(item, index) in list">
<td>{{item.id}}</td>
<td>{{ item.name }}</td>
<td>{{ item.price }}</td>
<td @click="del(item.id)"><a>删除</a></td>
</tr>
<!-- <tr>
<td>2</td>
<td>大衣</td>
<td class="red">199.00</td>
<td><a href="javascript:;">删除</a></td>
</tr> -->
</tbody>
<tfoot>
<tr>
<td colspan="4">消费总计: {{ totalCount }}</td>
</tr>
</tfoot>
</table>
</div>
<!-- 右侧图表 -->
<div class="echarts-box" id="main"></div>
</div>
</div>
<script src="https://cdn.jsdelivr.net/npm/echarts@5.4.0/dist/echarts.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script>
<script src="https://cdn.jsdelivr.net/npm/axios/dist/axios.min.js"></script>
<script src="./zhangdan.js"></script>
</body>
</html>
重点关注图表内容的实现 通过这个案例最好学会echarts的使用。
工程化开发模式:基于构建工具(例如:webpack)的环境中开发Vue。
工程化开发模式优点: 提高编码效率,比如使用JS新语法、Less/Sass、Typescript等通过webpack都可以编译成浏览器识别的ES3/ES5/CSS等 工程化开发模式问题:
为了解决以上问题,所以我们需要一个工具,生成标准化的配置
Vue CLI 是Vue官方提供的一个全局命令工具 可以帮助我们快速创建一个开发Vue项目的标准化基础架子。【集成了webpack配置】
项目目录:
main.js
文件核心作用: 导入
App.vue
,基于App.vue
创建结构渲染index.html
Vue.config.productionTip = false
提示当前处于什么环境
main.js
的内容
// 导入vue核心包
import Vue from 'vue'
import App from './App.vue'
//2. 提示当前的生产环境
Vue.config.productionTip = false
//3. 创建vue实例
new Vue({
// render: h => h(App),
//上述的完整的写法
render: (createElement) =>{
//基于App构建元素结构
return createElement(App)
}
}).$mount('#app')// 和el: '#app'一样, 都是指定管理的容器
比如:下面这个页面,可以把所有的代码都写在一个页面中,但是这样显得代码比较混乱,难易维护。咱们可以按模块进行组件划分
只能在注册的组件内使用
当成html标签使用即可 <组件名></组件名>
组件名规范 —> 大驼峰命名法, 如 Header
在App.vue
中 使用
<script>
//导入组件
import Head from "./components/Head.vue"
import Main from './components/Main.vue'
import Foot from './components/Foot.vue'
export default {
// 注册组件 (对于导入的组件名和组件对象一样时, 我们可以直接使用组件名)
components: {
// 组件名: 组件对象
Head,
Main,
Foot
}
}
</script>
<template>
//使用组件
<!-- 类名·最好和当前组件名同名 -->
<div class="App">
<Head>这里是Head</Head>
<Main>这里是main</Main>
<Foot>这里是Foot</Foot>
</div>
</template>
全局注册的组件,在项目的任何组件中都能使用(相当于工具组件)
当成HTML标签直接使用 <组件名></组件名>
组件名规范 —> 大驼峰命名法, 如 HmHeader
注册:
Vue.component('组件名', 组件对象)
//进行全局组件的导入
import Button1 from './components/Button1.vue'
//全局注册组件
Vue.component('Button1', Button1)
<template>
<div class="hm-header">
我是hm-header
<Button1>按钮</Button1>
</div>
</template>
// 项目快捷键
//1. 所有都折叠 ctrl + k , ctrl + 0
//2. 所有都展开 ctrl + k , ctrl + j
**实现思路: **