Featured image of post 学习Vue.js笔记

学习Vue.js笔记

个人的Vue学习记录,是基于[哔哩哔哩-尚硅谷]的Vue课程记录的

Vue简单常识:

v-bind:(单项数据绑定)

用来绑定数据的,区别于{{}},v-bind多用于标签元素的attribution,而{{}}多用于标签中间的数字/数值/字符串等

v-bind可以简写为:,而且他是单项数据绑定,数据只能由data里面的变量决定,即使在表单元素里面修改,v-bind绑定的数据也不会有反应

v-model:(双向数据绑定)

虽然都是数据绑定,但是v-model只能用于表单元素,而且表单内数据的更新会带着data内的数据一起更新.

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
<div id="doubt">
	<h3>{{eerr}} {{Date.now()}} </h3>
	<a v-bind:href="urls" v-bind:title="span_title" >{{intos}}</a>
	<a :href="urls" :title="span_title" >{{intos}}</a><br>
	<input type="text" v-model:value="eerr" style="width: 100%;margin-top: 30px">
	<input type="text" v-model="intos" style="width: 100%;margin-top: 30px">
</div>
<script type="text/javascript" src="vue.js"></script>
<script>
Vue.config.productionTip=false
Vue.config.devtools=false
const vapp = new Vue({
	el: "#doubt",
	data:{
		intos:"去JokemeBlog",
		eerr: "一刀999,是兄弟就来砍我",
		urls:"https://jokeme.top",
		span_title:"JOkemeBlog官网"
	}
})
</script>

使用mount挂载

其实咱可以不用上来就指定好el的值,咱可以已mount的形式来挂在上.

1
2
3
<div id="niubi">
	{{welcome}}
</div>
1
2
3
4
5
6
const vapp = new Vue({
	data:{
		welcome:"欢迎学习Vue 2.0!"
	}
})
vapp.$mount("#niubi")

说完了el咱再来说 data

data函数式

其实这里的data还可以写为函数式的. json/对象式 & 函数式都可以正常使用,以后使用组件的时候就必须要用函数式

1
2
3
4
5
6
7
const vapp = new Vue({
	data(){
		return{
			welcome:"欢迎学习Vue 2.0!"
		}
	}
})

defineProperty

Vue的代理模式,其中先大概了解一下defineProperty的用法:defineProperty(obj,properties,options/functions)

代码我也懒得整,就截了两张图片,学过高级语言的 应该一眼就看明白了.代理也就是通过set()/get()操作的

image-20211229012235756

image-20211229015619753

v-click

就是监听点击事件的,然后执行一个函数,可以加括号,不加括号就不传参

1
2
3
4
	<div id="niubi">
		<button @click="callbaby('ss',$event)">点我!!!!</button>
		<!-- 传参的时候,这个$event就相当于关键字了,加上$才有事件,不加就是变量 -->
	</div>>
1
2
3
4
5
6
7
8
9
const vapp = new Vue({
	methods:{
		callbaby(a,b){
			console.log(a,b)
			alert("Hello啊!同学!")
		}
	}
})
vapp.$mount("#niubi")

但是! 但是! 现在Vue应该是改了,不传$event 也可以直接输出event了,如下

1
2
"callbaby('ss')"
console.log(a,event)

**Tips: **methods里面的方法也可以丢data里面哦,但是data会多一个代理,浪费电脑性能!又学到了一个离职埋坑小技巧哦!

click的事件修饰符

image-20211229033005645

反正就这几个常用的,而且还是前三个常用.

eg:

1
<button class="qq" @click.once="callbaby(12)">点我!!!!</button><br>

下面的这个就是这个click只在第一次操作触发,第二次及以后都不在响应事件了

详细解说见 哔哩哔哩-尚硅谷-事件修饰符

键盘监听

@keyup

键盘回弹触发事件

@keydown

键盘按下触发事件

下面的小代码就可以让你理解了,其中.enter是遇到回车键才执行

为了方便起见,只贴出关键代码,其余部分都没有怎么变得就省略了,上面也有.

1
<input type="text" placeholder="随便写点啥吧!" @keydown.enter=whichOneUInput>
1
2
3
whichOneUInput(e){
	console.log(e.target.value)
}

像这样的操作,Vue为我们准备了一下几个

按键 别名
回车 enter
删除(捕获『删除』&『退回』) delete
退出 esc
空格 space
换行 tab
上下 up/down
左右 left/right

@keydown.ctrl.y就可以让ctrl + y触发

更多按键见:哔哩哔哩-尚硅谷-键盘事件

计算属性

computed类似data,el,methods它也是Vue带来的全新属性,可以直接从data里面获取数据进行操作/计算

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
data(){
	return{
		welcome:"欢迎学习Vue 2.0!",
		dabo:"Jack"
	}
},
computed:{
		fullname:{
			get(){
				return this.welcome + '-' + this.dabo
			}
		}
	}
1
<p>{{fullname}}</p>

计算完成以后就可以直接调用了,有点像Linux里的 alias,给一些列操作定义一个名字,我调用这个名字就执行这个操作

监视属性

watch是一个用于监听数据变化的属性,他有一个方法handler(new,old)可以在监视的数据发生变化时做出相应的处理.其中new是改变后的数据,old是改变前的数据.

immediate:true可以让数据在初始化的时候就开始监听,就是从无到有的过程也执行一次handler()

还有一点需要注意的是,被监视的数据只能是 datacomputed里面的.

image-20211230170345170

deep监视不能监视多层数据里面改变前的数据,只能拿到改变后的数据.

1
<button @click="dbqs.tom++">加一</button>
1
2
3
4
5
6
7
8
watch:{
	temperature:{
		immediate:true,
		handler(a,b){
			console.log(a+"``"+b)
		}
	}
},

除此之外watch还有一种api调用方法

1
2
3
4
5
6
shit.$watch("temperature",{
	immediate:true,
	handler(a,b){
		console.log(a+"``"+b)
	}
})

deep的用法

1
2
3
4
5
6
dbqs:{
	deep:true,
	handler(a,b,c){
		console.log(a.tom+"----"+b.tom)
	}
}

此外当你在监视的时候,用不到set,你可以用简写

1
2
3
temperature(a,b){
	console.log(a+"----"+b)
},

直接就是你需要监视的数据变成函数名就可以了,handler有啥参数,这个简写就有啥.

绑定Class样式

多说无益,先上代码

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
.basic{
	width: 400px;
	height: 200px;
	border: 2px tomato solid;
	font-family: 'Trebuchet MS', 'Lucida Sans Unicode', 'Lucida Grande', 'Lucida Sans', Arial, sans-serif;
	margin-top: 30px;
	text-align: center;
}
.day {
	background-color: rgb(255, 255, 255);
	color: rgb(26, 26, 26);
}
.night{
	background-color: rgb(49, 49, 49);
	color: rgb(226, 226, 226);
}
1
2
3
4
5
6
<div id="bobp" class="basic" :class="color" >
    <P>今天是个好日子啊!</P>
	<p>老Tom今天在干嘛啊</p>
	<hr style="width: 80%;margin-left: auto;margin-right: auto;margin-top: 3px;margin-bottom: 3px;">
	<button @click="change">改变</button>
</div>
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
data() {
    return {
		color:"day",
},
methods: {
	change(){
		this.color == "day" ?  this.color = "night" : this.color = "day"
	}
},

原理::class绑定一个data里面的变量color,当你点击按钮的时候,就会做出判断,给原来的样式basic 后面再加上day/night样式

image-20211230195124370

下面的两种会用就可以,根据实际情况来,数组和对象一般用的不是很多

v-if

1
2
3
4
<p v-if="codelang[0] == 'Java'">{{codelang[0]}}</p>
<p v-else-if="codelang[1] == 'Python'">{{codelang[1]}}</p>
<p v-if="codelang[2] == Cpp">{{codelang[2]}}</p>
<p v-else>{{codelang[3]}}</p>
1
codelang:["Java","Python","Cpp","Golang"]

可以看的出来哈,v-if只会只要命中条件以后就会显示后面的内容而没有命中的条件都不不会出现。这里也没有啥难点,和别的语言的if else没有啥差别

<template>标签只能与v-if一块用。其作用是不破坏dom结构。

v-for

1
2
3
<ul style="list-style-type: none;margin:  auto;padding: 0;border: orange 1px solid;width: 80%;">
	<li v-for="item in codelang" :key="index" style="margin-top: 5px;margin-bottom: 5px;">{{item}}</li>
</ul>

下面就来一段具体操作

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
const shit = new Vue({
			data() {
				return {
					user:[
						{uuid:123,name:"lx",sex:0},
						{uuid:124,name:"by",sex:1},
						{uuid:163,name:"lj",sex:1},
					],
				}
			},
			methods: {
				adduser(){
					// 这个函数在调用的 时候需要加上.once,否则会一直加人
					this.user.unshift({uuid:156,name:"lx",sex:1})
					// unshift() 直接把数据干到最上面
					// push() 数据加在最后面
				}
			},
})
1
2
3
4
5
<button @click.once="adduser">click me!</button>
<ul>
	<li v-for="item in user" :key="item.uuid" >{{item.name}} --- {{item.uuid}}
    <input type="text" :placeholder="item.name"> </li>
</ul>

箭头函数

复习一下!看这里 廖雪峰的官方网站

filter

同上复习:w3school

表单收集

哔哩哔哩-尚硅谷-表单收集

image-20220108002203213

自定义Vue指令

哔哩哔哩-尚硅谷-自定义指令

生命周期

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
mounted() {
	setInterval(() => {
		this.opacity -= 0.01
		if (this.opacity <= 0) {
			a = Math.random(0, 1)
			this.opacity = a
			console.log(a)
		}
	}, 15)
},

主要的函数都在这里

image-20220108040257962

哔哩哔哩-尚硅谷-生命周期

components(组件)

Q1:什么是components(组件)

Vue里面的组件其实就是相当于一个小型的Vue实例,Vue可以干的,它也可以,但是它就是不可以直接被$mount,换句话说也就是不能有自己的el:元素

Q2:如何定义一个组件?

问得好呀!

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
//这种是完整写法
const jokeme = Vue.extend({
    template:``
    ... ...
})
// 简写形式,只需要为对象即可,以后基本上都是用这种
const goofiest = {
    template:`
    <div>
        <p>网站名称: {{webname}} </p>
        <p>网站地址: {{site}} </p>
    </div>
    `,
    data(){
        return{
            webname: "goofiest",
            site:"goofiest.top"                    
        }
    }
}

Q3:定义完了以后咋使用呀?

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
const vm = new Vue({
    components:{
        // 局部注册,Σ(っ °Д °;)っ,只有当前的Vue实例可以使用
        laozhang:goofiest,// goofiest, 如果前后名字一样,直接写一个也是可以的
        // laoli:jokeme
    },
    methods:{
        show(){
            console.log("靓仔你谁呀?")
        }
    }
})
Vue.component("ll",jokeme)//全局注册所有的Vue实例都可以使用

render&&ref

中间几节课简单就没有写代码了,链接在下面

哔哩哔哩-尚硅谷-render函数

哔哩哔哩-尚硅谷-修改默认配置

哔哩哔哩-尚硅谷-ref属性

props传参

咱们先在Windows.vue里面定义好下面的props

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
props:{
	giveup:String,
	version:Number,
},
//带有数据校验的接收
// props:{
// 	giveup:{
// 		type:String,
// 		required:true,
// 		default:'111'
// 	}
// }
// 这种方式就可以更详细的约定那些是需要的,那些不需要,默认值等等...

然后在App.vue里面的Windows标签传递参数,

1
<Windows ref="xx" giveup="No" :version="19043.1466" ></Windows>

学习课程见:哔哩哔哩-尚硅谷-props配置

mixin

咱首先需要在main.js同级目录下建一个 **mixin.js**的文件

然后文件内容就是:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
const wheseisvc = {
	data(){
		return {
			hell:"dashaz"
		}
	},
	methods:{
		sch(){
			console.log("i am here",this.hell);
			// alert("I am here")
		}
	}
}
// 在这个文件里可以定义data,methods,watch... vue允许的都可以
// 可以全局引入mixin方法就是在main.js 里面定义
// Vue.mixin(whereisvc)
// 单组件定义就在该组件里面import使用即可
export default wheseisvc;

写完了以后一定要export出去,要不然导入的时候会报错的

1
2
3
4
5
6
7
<template>
<h2 @click.once="sch">OS_Creator: {{creator}} </h2>
</template>

<script>
import whereisvc from "../mixin"
<script>

就在标签里面可以直接调用mixin里面定义好的方法了

学习课程见:哔哩哔哩-尚硅谷-mixin

plugins&&scoped

我也没有写代码,学习课程在下面

哔哩哔哩-尚硅谷-插件

哔哩哔哩-尚硅谷-scoped

简单案例

姓名案例

1
2
3
4
5
6
7
<div id="niubi">
	{{welcome}}<br>
	<input type="text" placeholder="随便写点啥吧!" v-model=welcome>
	<input type="text" placeholder="随便写点啥吧!" v-model=dabo>
	<button @click=fullname>dianwo</button>
</div>

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
<html>
    ...
	<laozhang></laozhang>
	<ll></ll>
</html>
<script>
const vapp = new Vue({
	data(){
		return{
			welcome:"欢迎学习Vue 2.0!",
			dabo:"Jack"
		}
	},
	methods:{
		fullname(){
			console.log(this.welcome + "-" + this.dabo)
		}
	}
})
vapp.$mount("#niubi")
</script>

天气案例

点击按钮切换今天的天气

1
2
3
4
<div id="bobp">
    <P>今天是个好日子啊!天气{{weatherQ}}</P>
    <button @click="turnWeather()">点我</button>
</div>
 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
const shit = new Vue({
    data() {
        return {
            temperature:true
        }
    },
    methods: {
        turnWeather(){
            this.temperature == true ? this.temperature = false :this.temperature = true
        }
    },
    computed: {
    weatherQ:{
        get(){
            return this.temperature ? "很热" : "凉爽"
        	}
    	}
	}
})
shit.$mount("#bobp")

通过计算属性拿到temperature以后在计算返回文字weatherQ,这更加人性化

名称查找

在input框内输入,下面的ul显示查询结果

Vue watch实现

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
<!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="kendaman">
		<input type="text" placeholder="输入以搜索🔍" v-model="kw">
		<ul>
			<li v-for="it in userT" :key="it.uuid"> {{it.name}} </li>
		</ul>
	</div>
	<script src="vue.js"></script>
	<script>
		Vue.config.productionTip = false
		Vue.config.devtools = false
		const vapp = new Vue({
			data() {
				return {
					kw: '',
					user: [
						{ uuid: 123, name: "托塔天王", sex: 0 },
						{ uuid: 124, name: "斗战胜佛", sex: 1 },
						{ uuid: 163, name: "净坛使者", sex: 1 },
						{ uuid: 165, name: "祥龙罗汉", sex: 1 }
					],
					userT: [],
				}
			},
			methods: {

			},
			watch: {
				kw: {
					immediate: true,
					handler(newkw) {
						this.userT = this.user.filter((u) => {
							return u.name.indexOf(newkw) !== -1
						})
					}
				}

			},
		})
		vapp.$mount("#kendaman")
	</script>
</body>
</html>

这里的一个小知识点是:immediate在初始化的时候就会执行一次,而此时,kw即我们的输入为空,newkw监视变化前的值也为空,filter在过滤的时候,匹配到空,是会返回0的而不是-1,匹配成功,因而最终返回user数组里面所有的数据给userT。众所周知js里面,任何字符串里都包含空字符串的。狗头保命!

Vue 计算属性实现

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
<!DOCTYPE html>
<html lang="en">

<head>
	<style>
		.bk{
			background-color: #2b2b2b;
			color: #dbdbdb;
		}
		.er {
			width: 300px;
			height: 200px;
			margin: 100px auto;
			border: 1px rgb(104, 104, 104) solid;
			border-radius: 10px;
			box-shadow:  1px 1px 4px 1px cornflowerblue ;
		}
		.ipt{
			background-color: #3f3f3f;
		}
		.ert{
			width: 80%;
			line-height: 30px;
			margin:0 auto;
			margin-top: 10px;
			list-style: none;
			border-radius:5px;
			text-align: center;
			padding: 0;
		}
		li{
			width: 80%;
			border-radius: 3px;
			margin:4px auto;
			background-color: #3f3f3f;
			font-size: 14px;
		}
		li:hover{
			border: 1px #caa155 solid;
			margin: -1px auto;
			font-size: 15px;
		}
	</style>
	<meta charset="UTF-8">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<title>Document</title>
</head>

<body class="bk">
	<div class="er" id="kendaman">
		<div class="ert">
			<input class="ert ipt" type="text" placeholder="输入以搜索🔍" v-model="kw">
		</div>
		<ul class="ert">
			<li v-for="it in userT" :key="it.uuid"> {{it.name}} </li>
		</ul>
	</div>
	<script src="vue.js"></script>
	<script>
		Vue.config.productionTip = false
		Vue.config.devtools = false
		const vapp = new Vue({
			data() {
				return {
					kw: '',
					user: [
						{ uuid: 123, name: "托塔天王", sex: 0 },
						{ uuid: 124, name: "斗战胜佛", sex: 1 },
						{ uuid: 163, name: "净坛使者", sex: 1 },
						{ uuid: 165, name: "祥龙罗汉", sex: 1 }
					], 
				}
			},
			methods: {

			},
			computed: {
				userT() {
					return this.user.filter((u) => {
						return u.name.indexOf(this.kw) !== -1
					})
				}
			}
		})
		vapp.$mount("#kendaman")
	</script>
</body>

</html>

computed里面有两个返回值可能比较搞心态,但是你只需要明白,第一个返回值是给filter做判断的,第二个返回值是给userT的就可以了,主要的原理和上面的一样,也是filter箭头函数

image

咱们来欣赏一下成果吧!暗黑模式真好看,我真厉害😜!

排序案例

接上面的案例继续,如果咱们有一个排序的要求并且是按照uuid或者是姓名来排序的呢?这活应该怎么整?

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
computed: {
userT() {
		let hugo = this.user.filter((u) => {
			return u.name.indexOf(this.kw) !== -1
		})
		if(this.sortby){
			hugo.sort((a,b)=>{
				if(this.sortby == 1){
					return a.uuid - b.uuid//升序前减后
				}
				if(this.sortby == 2){
					return b.uuid - a.uuid//降序后减前
				}
			})
		}
		return hugo
	},
}

废话也不多说,就这样写,杠杠的! 对搜索后的数据进行排序那是相当的easy了,只需要在查找到的符合条件的数组后面接着写一个排序即可,写完了把排序后的数组再返回出去就可以了.

npm基础

npm是node.js的一个包管理器,有点像pip,咱们在学Vue以后是需要vue-cli来操作的create,build…

首先安装node!不管是用apt、yum。还是手动下载安装包配置环境变量都可以,装上能有node环境就可以了

安装完node以后呢,由于默认的服务器在国外,咱们下载包会非常的慢,所以!换源!

1
npm config set registry https://registry.npm.taobao.org

或者直接使用**cnpm**替代npm

接着就可以安装Vue了,咱们这里全局安装:

1
 npm install -g @vue/cli

安装完Vue就可以创建咱们的第一个项目了

1
vue create xxx

创建完Vue项目以后,最后一行会提示你cd xxx ;npm run serve

按照命令来就可以了

就可以在浏览器看到项目了

todolist(一代目)

首先是项目拆分,看看todolist可以怎么样拆,

我是把项目拆分为三个模块的:input,itenlist.status

App.vue

这个组件也就是用来调度所有的VC,所以在<script>里面先导入三个子组件,并且在components里面注册三个组件

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
<script>
import ipt from "./components/input";
import todoitem from "./components/todoitem";
import status from "./components/status";

export default {
  name: "App",
  components: {
    ipt,
    todoitem,
    status
  },
}
</script>

其次就是methods这里面的方法大都是配合与子组件通信用的。做法都是把函数定义在父组件里面,通过子组件的调用来达到传参的目的,这种做法算是最最最基础原始的做法。但是缺点也很明显如果 父 -> 孙就太麻烦了,需要父 -> 子;子 -> 孙。耦合性太高了,如果需要改动代码,那就即将是一件非常痛苦的事情。但是以我目前的水平也就只能写出这样的扣脚代码!但是咱不急!等着后面二代目,三代目版本的慢慢优化吼!

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
<script>
methods: {
		//添加待办事项的
		// 配合input.vue组件的使用,在这里定义好函数,然后再在input里面调用达到input->App传递数据的目的
    adddolist(e) {
      this.dolist.unshift(e);
    },
		// 删除项目 原理同上
		delt(uid){
			for(var er = 0;er<this.dolist.length;er++){
				if (this.dolist[er].uuid == uid) {
					this.dolist.splice(er, 1);
					break;
				}
			}
		},
		// 标记项目已完成 原理同上
		setstatus(uid){
			for (let er = 0; er < this.dolist.length; er++){
				if(this.dolist[er].uuid == uid){
					this.dolist[er].done = !(this.dolist[er].done)
				}
			}
		}
  },
</script>

再来看<template>标签,这里面都是App向其子组件传递的数据、函数

1
2
3
4
5
6
7
<template>
  <div class="edge">
    <ipt :adddolist="adddolist"></ipt>
    <todoitem :itemlist="dolist" :delt="delt" :setstatus="setstatus"></todoitem>
    <status :itemlistSta="dolist"></status>
  </div>
</template>

input.vue

这个组件就是来完成输入的整体功能是相当的简单🍳,咱们先来看看<template>标签,就一个<div>包裹着<input>但是这个input有一个键盘⌨监听事件“addItem”,可在用户按下enter时执行函数。

1
2
3
4
5
<template>
	<div :class="ipt" id="commom">
		<input type="text" :placeholder="pleaseholder" @keydown.enter="addItem">
	</div>
</template>

接下来咱再来看看<script>里面写了啥

data:定义了placeholder,可以让我们自己去动态的更新!很人性!

props:接收adddolist函数。

methods:就是添加数据的时候不允许空值,然后把数据打包成数据发给父组件,最后清空一下value就大功告成了

watch:自动根据时间切换黑暗模式

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
<script>
export default {
	data() {
		return {
			ipt:"ipt-day",
			pleaseholder:"你想干啥?"
		}
	},
	props:["adddolist"],
	methods: {
		addItem(e){
			if (e.target.value == "") {
				alert("不允许输入空值哦!")				
			} else {
				const abcd = Math.floor(Math.random()*1000+1)
				const fna ={"uuid":abcd,"thing":e.target.value,"done":false}
				this.adddolist(fna)
				e.target.value = ""
			}
		},
	},
	watch:{
		ipt:{
			immediate:true,
			handler(){
				let ho = new Date().getHours()
				if (ho >= 7 && ho < 18) {
					this.ipt="ipt-day"
				} else {
					this.ipt="ipt-night"
				}
			}
		}
	}
}
</script>

todoitem.vue

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
<template>
	<div :class="sus"  id="todoit" v-show="itemlist.length != 0" >
		<div class="fir" v-for="item in itemlist" :key="item.uuid">
			<div>
				<label @click="checkstat($event,item.uuid)">
					<input type="checkbox" :checked="item.done">
					<span>{{item.thing}} </span>
				</label>
				<button @click="deletet(item.uuid)">删除</button> 
			</div>
		</div>
	</div>
</template>

首先就是v-show="itemlist.length != 0"在没有数据的情况下不会显示该项。其次v-for="item in itemlist"利用v-for渲染待办事项。然后就是利用<label>标签来提升一下用户体验,一个checkbox类型的input来标识是否完成

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
<script>
export default {
	data(){
		return {
			sus:"item-sur-day"
		}
	},
	props:['itemlist',"delt","setstatus"],
	methods:{
        //删除待办
		deletet(uuid){
			this.delt(uuid)
		},
        //完成待办,
		checkstat(ele,uid){
			if(ele.target.tagName == "SPAN"){
				return 
			}
			this.setstatus(uid)
		}
	},
	watch:{
		sus:{
			immediate:true,
			handler(){
				let ho = new Date().getHours()
				if (ho >= 7 && ho < 18) {
					this.sus="item-sur-day"
				} else {
					this.sus="item-sur-night"
				}
			}
		},
	}
}
</script>

status.vue

其实这个没有啥好说的,代码里面都是浅显易懂的东西!主要就是统计一共有几个待办事项,已完成几个。还有和父组件的参数传递


最后来看看一代目的成品吧

image-20220127014027587

Licensed under CC BY-NC-SA 4.0