第01章-前端核心技术-VUE基础使用
学习目标
掌握VUE安装和基本使用
掌握VUE条件渲染
掌握VUE列表渲染
VUE 简介
Vue.js 是一套构建用户界面的 渐进式框架。目标是通过尽可能简单的 API 实现响应的数据绑定和组合的视图组件。
官网地址:https://v3.cn.vuejs.org/----https://cn.vuejs.org/ Vue3相比Vue2提升点
性能比Vue2.x快1.2~2倍
diff方法优化 
vue2中的虚拟dom是全量的对比(每个节点不论写死的还是动态的都会比较) vue3新增了静态标记(patchflag)与上次虚拟节点对比时,只对比带有patchflag的节点(动态数据所在的节点);可通过flag信息得知当前节点要对比的具体内容
静态提升 
vue2无论元素是否参与更新,每次都会重新创建然后再渲染 vue3对于不参与更新的元素,会做静态提升,只会被创建一次,在渲染时直接复用即可
事件侦听器缓存 
默认情况下onClick会被视为动态绑定,所以每次都会追踪它的变化 但是因为是同一个函数,所以不用追踪变化,直接缓存起来复用即可
ssr服务端渲染 
在服务器段完成渲染后再将整个html页面返回到浏览器,页面的加载时间就会加快很多
组合API(类似react hooks)
将数据和业务逻辑放在setup中一起处理,而不是像vue2,将数据放在data中,业务逻辑分别放在methods,watch,computed中等
按需编译,体积比vue2.x更小
更好的Ts支持
VUE 环境搭建
下载源码
如果仅仅只是在项目或者某个文件中简单的使用vue,就可以直接在html中引入如下链接

<script src="https://unpkg.com/vue@next"></script>
或者
<script src="https://unpkg.com/vue@3"></script>

可以复制上面的地址到浏览器打开源代码,并复制全部源代码,保存为本地文件js文件,加入到项目中就可以使用了
创建项目

<!DOCTYPE html>
<html lang="ch">
<head>
    <meta charset="UTF-8">
    <title>VUE3</title>
    <script src="vue.global.js"></script>
</head>
<body>
<div id="app">
    Counter: {{ counter }}
</div>
</body>
<script>
    const App = {
        data() {
            return {
                counter: 0
            }
        },
        mounted() {
            setInterval(() => {
                this.counter++
            }, 1000)
        }
    }
    Vue.createApp(App).mount('#app')
</script>
</html>

VUE 语法
静态数据 data
参数类型:Object | Function,组件的定义只接受 function
当一个 Vue 实例被创建时,它向 Vue 的响应式系统中加入了其 data 对象中能找到的所有的属性。当这些属性的值发生改变时,视图将会产生“响应”,即匹配更新为新的值。
案例01

<!DOCTYPE html>
<html>

<head>
	<meta charset="UTF-8" />
	<title></title>
	<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
</head>

<body>
	<div id="app">来自 data 的属性值 = {{ count }}</div>
</body>
<script type="text/javascript">
	const app = Vue.createApp({
		data() {
			return { count: 4 };
		},
	});

	const vm = app.mount("#app");

	console.log(vm.$data.count); // => 4
	console.log(vm.count); // => 4

	// 修改 vm.count 的值也会更新 $data.count
	vm.count = 5;
	console.log(vm.$data.count); // => 5

	// 反之亦然
	vm.$data.count = 6;
	console.log(vm.count); // => 6
</script>

</html>

js中的数据直接展示到页面山给,不需要再通过bom对象来获取和设置页面元素的内容,使用非常方便
当这些数据改变时,视图会进行重渲染。值得注意的是只有当实例被创建时 data 中存在的属性才是响应式的。也就是说如果你添加一个新的属性不会起作用
动态数据 computed
参数类型:{ [key: string]: Function | { get: Function, set: Function } }
和data一样,向 Vue 的响应式系统中加入了其 data 对象中能找到的所有的属性。当这些属性的值发生改变时,视图将会产生“响应”,即匹配更新为新的值。
不同的是computed允许动态属性计算,所以一般都是用方法。
computed接受一个具有 get 和 set 函数的对象,可以单独设置读写的方式。

const count = ref(1)
const plusOne = computed({
  get: () => count.value + 1,
  set: val => {
    count.value = val - 1
  }
})

plusOne.value = 1
console.log(count.value) // 0

案例02

<!DOCTYPE html>
<html>

<head>
	<meta charset="UTF-8" />
	<title></title>
	<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
</head>

<body>
	<div id="app">
		<div>从vue传过来的数据:<b>{{ count }}</b></div>
		<div>库存是否足够:<b>{{ isCount }}</b></div>
	</div>
</body>
<script type="text/javascript">
	const app = Vue.createApp({
		data() {
			return { count: 4 }
		},
		computed: {
			isCount() {
				// `this` 指向 vue 对象实例
				return this.count > 0 ? 'Yes' : 'No'
			}
		},
	});

	const vm = app.mount("#app");
</script>

</html>

输出渲染 v-text & v-html
双大括号会将数据解释为普通文本,而非 HTML 代码。类似v-text
为了输出真正的 HTML,需要使用 v-html 指令:
案例03

<!DOCTYPE html>
<html>

<head>
	<meta charset="UTF-8" />
	<title></title>
	<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
</head>

<body>
	<div id="app">
		<div>{}类似与使用v-text绑定的:<b v-text="text">{{count}}</b></div>
		<div>使用v-html绑定的可以识别html标签:<p v-html="text">{{isCount}}</p>
			<p>v-html 和 v-text 优先级高于 {} </p>
		</div>
	</div>
</body>
<script type="text/javascript">
	const app = Vue.createApp({
		data() {
			return {
				count: 4,
				text: '<span style="color: blue;">132456</span>'
			}
		},
		computed: {
			isCount() {
				// `this` 指向 vue 对象实例
				return this.count > 0 ? 'Yes' : 'No'
			}
		},
	});

	const vm = app.mount("#app");
</script>

</html>

绑定属性 v-bind
普通语法不能作用在 HTML 属性上,遇到这种情况应该使用 v-bind 指令:如:v-bind:id、v-bind:class、v-bind:src、v-bind:align等。
语法:v-bind:attribute
案例04

<!DOCTYPE html>
<html>

<head>
	<meta charset="UTF-8" />
	<title></title>
	<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
</head>
<style>
	.active {
		background-color: darkcyan;
	}

	.error {
		color: red;
	}
</style>

<body>
	<div id="app">
		<div :class="{active: clas.isActive,'error': clas.hasError}">
			v-bind:id="id"相当于setAttribute方法设置id属性值
		</div>
		<p>v-bind:id="id"可以省略为 :id</p>
	</div>
</body>
<script type="text/javascript">
	const app = Vue.createApp({
		data() {
			return {
				clas: {
					isActive: false,
					hasError: true
				}
			}
		},
	});

	const vm = app.mount("#app");
</script>

</html>

绑定事件 v-on
绑定事件监听器。事件类型由参数指定。表达式可以是一个方法的名字或一个内联语句,如果没有修饰符也可以省略。
用在普通元素上时,只能监听原生 DOM 事件。用在自定义元素组件上时,也可以监听子组件触发的自定义事件。
v-on 指令 (通常缩写为 @ 符号) 。用法为 v-on:click="methodName" 或 `@click="methodName"
语法:v-on:event.修饰符
有时也需要在内联语句处理器中访问原始的 DOM 事件。可以用特殊变量 $event 把它传入方法:
<button @click="warn('自定义参数', $event)">Submit

js部分

methods: {
  warn(message, event) {
    // 现在可以访问到原生事件
    if (event) {
      event.preventDefault()
    }
    alert(message)
  }
}

多事件绑定
事件处理程序中可以有多个方法,这些方法由逗号运算符分隔:

<!-- 这两个 one() 和 two() 将执行按钮点击事件 -->
<button @click="one($event), two($event)">
  Submit
</button>

js部分

// ...
methods: {
  one(event) {
    // 第一个事件处理器逻辑...
  },
  two(event) {
   // 第二个事件处理器逻辑...
  }
}

案例05

<!DOCTYPE html>
<html>

<head>
	<meta charset="UTF-8" />
	<title></title>
	<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
</head>
<style>
	.active {
		background-color: darkcyan;
	}

	.error {
		color: red;
	}
</style>

<body>
	<div id="app">
		<button v-on:click="doClick('按钮已经被点击')">点击事件</button>
		<br>
		<input type="text" @keyup="doKeyup" />
		<span>{{showtext}}</span><span>{{inputcount}}</span>
	</div>
</body>
<script type="text/javascript">
	const app = Vue.createApp({
		data() {
			return {
				showtext: '',
				inputcount: 0
			}
		},
		methods: {
			//event:被触发的事件对象
			doClick: function (me) {
				//this:当前Vue对象
				this.showtext = '字数:';
				event.target.innerText = me;
			},
			doKeyup: function () {
				this.inputcount = event.target.value.length;
			}
		}
	});

	const vm = app.mount("#app");
</script>

</html>

事件修饰符
在事件处理程序中调用 event.preventDefault() 或 event.stopPropagation() 是非常常见的需求。尽管我们可以在方法中轻松实现这点,但更好的方式是:方法只有纯粹的数据逻辑,而不是去处理 DOM 事件细节。
为了解决这个问题,Vue.js 为 v-on 提供了事件修饰符。之前提过,修饰符是由点开头的指令后缀来表示的。
.stop
.prevent
.capture
.self
.once
.passive
如:

<a @click.stop="doThis">

<a @click.stop.prevent="doThat">

...

...

使用修饰符时,顺序很重要;相应的代码会以同样的顺序产生。因此,用 v-on:click.prevent.self 会阻止所有的点击,而 v-on:click.self.prevent 只会阻止对元素自身的点击。
按键修饰符
在监听键盘事件时,我们经常需要检查详细的按键。Vue 允许为 v-on 或者 @ 在监听键盘事件时添加按键修饰符:
1
2

<input @keyup.enter="submit" />

你可以直接将 KeyboardEvent.key 暴露的任意有效按键名转换为 kebab-case 来作为修饰符。
<input @keyup.page-down="onPageDown" />

案件 kebab-case的参考地址:https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/key/Key_Values Vue 为最常用的键提供了别名:
.enter
.tab
.delete
 
(捕获“删除”和“退格”键)
.esc
.space
.up
.down
.left
.right
可以用如下修饰符来实现仅在按下相应按键时才触发鼠标或键盘事件的监听器。
.ctrl
.alt
.shift
.meta
如:

<input @keyup.alt.enter="clear" />

Do something

.exact 修饰符
.exact 修饰符允许你控制由精确的系统修饰符组合触发的事件。
如:

<button @click.ctrl="onClick">A

<button @click.ctrl.exact="onCtrlClick">A

<button @click.exact="onClick">A

鼠标按钮修饰符鼠标按钮修饰符
.left
.right
.middle
这些修饰符会限制处理函数仅响应特定的鼠标按钮
双向绑定 v-model
可以用 v-model 指令在表单 、 及 <select> 元素上创建双向数据绑定。它会根据控件类型自动选取正确的方法来更新元素。 它负责监听用户的输入事件以更新数据,并对一些极端场景进行一些特殊处理。

<br>
v-model 本质上是一个语法糖。<input v-model="test">本质如下:<br>
1<br>
2<br>
<input :value="test" @input="test = $event.target.value"></p>
<p>其中 $event 是触发事件的元素对象,通过$event.target.value 可以获取元素的值<br>
v-model 会忽略所有表单元素的 value、checked、selected attribute 的初始值<br>
v-model 在内部为不同的输入元素使用不同的 property 并抛出不同的事件:<br>
text 和 textarea 元素使用 value property 和 input事件;<br>
checkbox 和 radio 使用 checkedproperty 和change<br>
 <br>
事件;<br>
select 字段将<br>
 <br>
value<br>
 <br>
作为 prop 并将<br>
 <br>
change<br>
 <br>
作为事件。<br>
案例06:表单元素双向绑定</p>
<pre><code><!DOCTYPE html>
<html>

<head>
	<meta charset="UTF-8" />
	<title></title>
	<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
</head>
<style>
	.active {
		background-color: darkcyan;
	}

	.error {
		color: red;
	}
</style>

<body>
	<div id="app">
		<p>单行输入框:</p>
		<input v-model="value" />
		<p style="color: blue">{{ value }}</p>

		<p>多行输入框:</p>
		<textarea v-model="text"></textarea>
		<p style="white-space: pre-line; color: blue">{{ text }}</p>

		<p>复选框:</p>
		<input type="checkbox" id="checkbox" v-model="checked" />
		<label for="checkbox">{{ checked }}</label>

		<p>单选框:</p>
		<div>
			<input type="radio" id="one" value="One" v-model="picked" />
			<label for="one">One</label>
			<input type="radio" id="two" value="Two" v-model="picked" />
			<label for="two">Two</label>
			<p style="white-space: pre-line; color: blue">{{ picked }}</p>
		</div>

		<p>多个复选框,绑定到同一个数组:</p>
		<div>
			<input type="checkbox" value="Jack" v-model="checkedNames" />
			<label for="jack">Jack</label>
			<input type="checkbox" value="John" v-model="checkedNames" />
			<label for="john">John</label>
			<input type="checkbox" value="Mike" v-model="checkedNames" />
			<label for="mike">Mike</label>
		</div>
		<p style="white-space: pre-line; color: blue">
			选择的内容: {{ checkedNames }}
		</p>

		<p>单选下拉框框:</p>
		<div id="v-model-select">
			<select v-model="selected">
				<option disabled value="">请选择</option>
				<option>A</option>
				<option>B</option>
				<option>C</option>
			</select>
			<p style="white-space: pre-line; color: blue">
				选择的内容: {{ selected }}
			</p>
		</div>

		<p>多选下拉框框:</p>
		<div id="v-model-select">
			<select v-model="selecteds" multiple>
				<option disabled value="">请选择</option>
				<option>A</option>
				<option>B</option>
				<option>C</option>
			</select>
			<p style="white-space: pre-line; color: blue">
				选择的内容: {{ selecteds }}
			</p>
		</div>
	</div>
</body>
<script type="text/javascript">
	const app = Vue.createApp({
		data() {
			return {
				value: "",
				text: "",
				checked: true,
				picked: "",
				checkedNames: [],
				selected: "",
				selecteds: []
			}
		},
	});

	const vm = app.mount("#app");
</script>

</html>
</code></pre>
<p>修饰符<br>
.lazy<br>
在默认情况下,v-model 在每次 input 事件触发后将输入框的值与数据进行同步 (除了上述输入法组织文字时)。你可以添加 lazy 修饰符,从而转为在 change 事件_之后_进行同步:<br>
1<br>
2</p>
<!-- 在“change”时而非“input”时更新 -->
<input v-model.lazy="msg" />
<p>.number<br>
如果想自动将用户的输入值转为数值类型,可以给 v-model 添加 number 修饰符:<br>
<input v-model.number="age" type="number" /></p>
<p>这通常很有用,因为即使在 type="number" 时,HTML 输入元素的值也总会返回字符串。如果这个值无法被 parseFloat() 解析,则会返回原始的值。<br>
.trim<br>
如果要自动过滤用户输入的首尾空白字符,可以给 v-model 添加 trim 修饰符:<br>
<input v-model.trim="msg" /></p>
<p>条件渲染<br>
v-if<br>
v-if 指令用于条件性地渲染一块内容。这块内容只会在指令的表达式返回 truthy 值的时候被渲染。</p>
<h1 v-if="awesome">Vue is awesome!</h1>
<p>也可以用 v-else 添加一个“else 块”:<br>
1<br>
2</p>
<h1 v-if="awesome">Vue is awesome!</h1>
<h1 v-else>Oh no 😢</h1>
<p>v-else-if,顾名思义,充当 v-if 的“else-if 块”,并且可以连续使用:</p>
<pre><code>12
<div v-if="type === 'A'">
  A
</div>
<div v-else-if="type === 'B'">
  B
</div>
<div v-else-if="type === 'C'">
  C
</div>
<div v-else>
  Not A/B/C
</div>
</code></pre>
<p>与 v-else 的用法类似,v-else-if 也必须紧跟在带 v-if 或者 v-else-if 的元素之后。<br>
v-show<br>
另一个用于条件性展示元素的选项是 v-show 指令。用法大致一样:</p>
<h1 v-show="ok">Hello!</h1>
<p>不同的是带有 v-show 的元素始终会被渲染并保留在 DOM 中。v-show 只是简单地切换元素的 CSS property display。<br>
注意:v-show 不支持 <template> 元素,也不支持 v-else。当 v-if 与 v-for 一起使用时,v-if 具有比 v-for 更高的优先级。<br>
案例07</p>
<pre><code><!DOCTYPE html>
<html>

<head>
	<meta charset="UTF-8" />
	<title></title>
	<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
</head>
<style>
	.active {
		background-color: darkcyan;
	}

	.error {
		color: red;
	}
</style>

<body>
	<div id="app">
		<button @click="change()">切换</button>
		<div v-if="type == '0'">
			A
		</div>
		<div v-else-if="type == '1'">
			B
		</div>
		<div v-else-if="type == '2'">
			C
		</div>
		<div v-else>
			Not A/B/C
		</div>
	</div>
</body>
<script type="text/javascript">
	const app = Vue.createApp({
		data() {
			return {
				type: 0,
			}
		},
		methods: {
			//event:被触发的事件对象
			change: function () {
				//this:当前Vue对象
				this.type = ++this.type % 4;
			},
		}
	});

	const vm = app.mount("#app");
</script>

</html>
</code></pre>
<p>列表渲染<br>
用 v-for 指令基于一个数组来渲染一个列表。v-for 指令需要使用 item in items 形式的特殊语法,其中 items 是源数据数组,而 item 则是被迭代的数组元素的别名。</p>
<pre><code>5
<ul>
  <li v-for="item in items">
    {{ item.message }}
  </li>
</ul>

在 v-for 块中,我们可以访问所有父作用域的 property。v-for 还支持一个可选的第二个参数,即当前项的索引。

<ul>
  <li v-for="(item, index) in items">
    {{ parentMessage }} - {{ index }} - {{ item.message }}
  </li>
</ul>
```
还可以用第三个参数作为索引:
</code></pre>
<p>2<br>
3</p>
<li v-for="(value, name, index) in myObject">
  {{ index }}. {{ name }}: {{ value }}
</li>
````
v-for 也可以接受整数。在这种情况下,它会把模板重复对应次数。
1
2
3
<div id="range" class="demo">
  <span v-for="n in 10" :key="n">{{ n }} </span>
</div>
<p>也可以用 of 替代 in 作为分隔符,因为它更接近 JavaScript 迭代器的语法:</p>
<div v-for="item of items"></div>
<p>注意:不推荐在同一元素上使用 v-if 和 v-for。<br>
当它们处于同一节点,v-if 的优先级比 v-for 更高,这意味着 v-if 将没有权限访问 v-for 里的变量:</p>
<!-- This will throw an error because property "todo" is not defined on instance. -->
<li v-for="todo in todos" v-if="!todo.isComplete">
  {{ todo.name }}
</li>
<p>可以把 v-for 移动到 <template> 标签中来修正:</p>
<template v-for="todo in todos" :key="todo.name">
  <li v-if="!todo.isComplete">
    {{ todo.name }}
  </li>
</template>
<p>案例08</p>
<pre><code><!DOCTYPE html>
<html>

<head>
	<meta charset="UTF-8" />
	<title></title>
	<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
</head>
<style>
	.active {
		background-color: darkcyan;
	}

	.error {
		color: red;
	}
</style>

<body>
	<div id="app">
		<ul>
			<li v-for="item in items">
				{{ item }}
			</li>
		</ul>

		<ul>
			<li v-for="(item, index) in items">
				{{ index }} - {{ item }}
			</li>
		</ul>

		<ul>
			<li v-for="(value, name, index) in objs">
				{{ index }}. {{ name }}: {{ value }}
			</li>
		</ul>

		<div>
			<span v-for="n in 10" :key="n">{{ n }} </span>
		</div>

		<div v-for="item of items"></div>
	</div>
</body>
<script type="text/javascript">
	const app = Vue.createApp({
		data() {
			return {
				items: [1, 2, 3, 4, 5],
				objs: { a: 10, b: 20, c: 30 }
			}
		},
		methods: {
			//event:被触发的事件对象
			change: function () {
				//this:当前Vue对象
				this.type = ++this.type % 4;
			},
		}
	});

	const vm = app.mount("#app");
</script>

</html>
</code></pre>
<p>触发视图更新的函数<br>
Vue 将被侦听的数组的变更方法进行了包裹,所以它们也将会触发视图更新。这些被包裹过的方法包括:<br>
push()<br>
pop()<br>
shift()<br>
unshift()<br>
splice()<br>
sort()<br>
reverse()<br>
相比之下,也有非变更方法,例如 filter()、concat() 和 slice()。它们不会变更原始数组,而总是返回一个新数组。当使用非变更方法时,可以用新数组替换旧数组:<br>
example1.items = example1.items.filter(item => item.message.match(/Foo/))</p>
<p>监听数据 watch<br>
类型:<br>
{ [key: string]: string | Function | Object | Array}<br>
详细:<br>
一个对象,键是要侦听的响应式 property——包含了 data 或 computed 属性,而值是对应的回调函数。值也可以是方法名,或者包含额外选项的对象。<br>
示例:</p>
<pre><code>const app = createApp({
  data() {
    return {
      a: 1,
      b: 2,
      c: {
        d: 4
      },
      e: 5,
      f: 6
    }
  },
  watch: {
    // 侦听顶级 property
    a(val, oldVal) {
      console.log(`new: ${val}, old: ${oldVal}`)
    },
    // 字符串方法名
    b: 'someMethod',
    // 该回调会在任何被侦听的对象的 property 改变时被调用,不论其被嵌套多深
    c: {
      handler(val, oldVal) {
        console.log('c changed')
      },
      deep: true
    },
    // 侦听单个嵌套 property
    'c.d': function (val, oldVal) {
      // do something
    },
    // 该回调将会在侦听开始之后被立即调用
    e: {
      handler(val, oldVal) {
        console.log('e changed')
      },
      immediate: true
    },
    // 你可以传入回调数组,它们会被逐一调用
    f: [
      'handle1',
      function handle2(val, oldVal) {
        console.log('handle2 triggered')
      },
      {
        handler: function handle3(val, oldVal) {
          console.log('handle3 triggered')
        }
        /* ... */
      }
    ]
  },
  methods: {
    someMethod() {
      console.log('b changed')
    },
    handle1() {
      console.log('handle 1 triggered')
    }
  }
})

const vm = app.mount('#app')

vm.a = 3 // => new: 3, old: 1
</code></pre>
<p>注意,不应该使用箭头函数来定义 watcher 函数 (例如 searchQuery: newValue => this.updateAutocomplete(newValue))。<br>
理由是箭头函数绑定了父级作用域的上下文,所以 this 将不会按照期望指向组件实例,this.updateAutocomplete 将是 undefined。<br>
VUE 过度<br>
Vue 提供了一些抽象概念,可以帮助处理过渡和动画,特别是在响应某些变化时。这些抽象的概念包括:<br>
在 CSS 和 JS 中,使用内置的<br>
 <br>
<transition><br>
 <br>
组件来钩住组件中进入和离开 DOM。<br>
在处理多个元素位置更新时,使用<br>
 <br>
<transition-group><br>
 <br>
组件,通过 FLIP 技术来提高性能。<br>
自定义动画过渡<br>
尽管 <transition> 组件对于组件的进入和离开非常有用,但你也可以通过添加一个条件 class 来激活动画,而无需挂载组件。<br>
Transform 和 Opacity更改 transform 不会触发任何几何形状变化或绘制。这意味着该操作可能是由合成器线程在 GPU 的帮助下执行。<br>
因此,他们是在 web 上做元素移动的理想选择。<br>
诸如 perspective、backface-visibility 和 transform:translateZ(x) 等 property 将让浏览器知道你需要硬件加速。<br>
如果要对一个元素进行硬件加速,可以应用以下任何一个 property (并不是需要全部,任意一个就可以):<br>
1<br>
2<br>
3<br>
perspective: 1000px;<br>
backface-visibility: hidden;<br>
transform: translateZ(0);</p>
<p>案例09</p>
<pre><code><!DOCTYPE html>
<html>

	<head>
		<meta charset="UTF-8" />
		<title></title>
		<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
	</head>
	<style>
		.shake {
		  animation: shake 0.82s cubic-bezier(0.36, 0.07, 0.19, 0.97) both;
		  transform: translate3d(0, 0, 0);
		  backface-visibility: hidden;
		  perspective: 1000px;
		}
		
		@keyframes shake {
		  10%,
		  90% {
		    transform: translate3d(-1px, 0, 0);
		  }
		
		  20%,
		  80% {
		    transform: translate3d(2px, 0, 0);
		  }
		
		  30%,
		  50%,
		  70% {
		    transform: translate3d(-4px, 0, 0);
		  }
		
		  40%,
		  60% {
		    transform: translate3d(4px, 0, 0);
		  }
		}
	</style>

	<body>
		<div id="app">
			<div :class="{ shake: noActivated }">
				<button @click="noActivated = true">Click me</button>
				<span v-if="noActivated">Oh no!</span>
			</div>
		</div>
	</body>
	<script type="text/javascript">
		const app = Vue.createApp({
			data() {
				return {
					noActivated: false
				}
			},
		});

		const vm = app.mount("#app");
	</script>

</html>
</code></pre>
<p>当页面上的数据发生变化的时候也会触发过度效果<br>
案例10</p>
<pre><code><!DOCTYPE html>
<html>

	<head>
		<meta charset="UTF-8" />
		<title></title>
		<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
	</head>
	<style>
		.movearea {
			transition: 0.2s background-color ease;
		}
	</style>

	<body>
		<div id="app">
			<div @mousemove="xCoordinate" :style="{ backgroundColor: `hsl(${x}, 100%, 50%)` }" class="movearea">
				<h3>鼠标移动上来</h3>
				<p>x: {{x}}</p>
			</div>
		</div>
	</body>
	<script type="text/javascript">
		const app = Vue.createApp({
			data() {
				return {
					x: 0
				}
			},
			methods: {
				xCoordinate(e) {
					this.x = e.clientX
				}
			},
		});

		const vm = app.mount("#app");
	</script>

</html>
</code></pre>
<p>单元素过渡<br>
Vue 提供了 transition 的封装组件,在下列情形中,可以给任何元素和组件添加进入/离开过渡<br>
条件渲染 (使用 v-if)<br>
条件展示 (使用 v-show)<br>
动态组件<br>
组件根节点<br>
案例11</p>
<pre><code><!DOCTYPE html>
<html>

	<head>
		<meta charset="UTF-8" />
		<title></title>
		<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
	</head>
	<style>
		.movearea {
			transition: 0.2s background-color ease;
		}
	</style>

	<body>
		<div id="app">
			<button @click="show = !show">切换显示</button>

			<transition name="fade">
				<p v-if="show">hello</p>
			</transition>
		</div>
	</body>
	<script type="text/javascript">
		const app = Vue.createApp({
			data() {
				return {
					show: false
				}
			},
		});

		const vm = app.mount("#app");
	</script>

</html>
</code></pre>
<p>动画相关的 class<br>
在进入/离开的过渡中,会有 6 个 class 切换。<br>
v-enter-from<br>
:定义进入过渡的开始状态。在元素被插入之前生效,在元素被插入之后的下一帧移除。<br>
v-enter-active<br>
:定义进入过渡生效时的状态。在整个进入过渡的阶段中应用,在元素被插入之前生效,在过渡/动画完成之后移除。这个类可以被用来定义进入过渡的过程时间,延迟和曲线函数。<br>
v-enter-to<br>
:定义进入过渡的结束状态。在元素被插入之后下一帧生效 (与此同时<br>
 <br>
v-enter-from<br>
 <br>
被移除),在过渡/动画完成之后移除。<br>
v-leave-from<br>
:定义离开过渡的开始状态。在离开过渡被触发时立刻生效,下一帧被移除。<br>
v-leave-active<br>
:定义离开过渡生效时的状态。在整个离开过渡的阶段中应用,在离开过渡被触发时立刻生效,在过渡/动画完成之后移除。这个类可以被用来定义离开过渡的过程时间,延迟和曲线函数。<br>
v-leave-to<br>
:离开过渡的结束状态。在离开过渡被触发之后下一帧生效 (与此同时<br>
 <br>
v-leave-from<br>
 <br>
被删除),在过渡/动画完成之后移除。<br>
对于这些在过渡中切换的类名来说,如果你使用一个没有名字的 <transition>,则 v- 是这些class名的默认前缀。如果你使用了 <transition name="my-transition">,那么 v-enter-from会替换为 my-transition-enter-from。<br>
v-enter-active 和 v-leave-active 可以控制进入/离开过渡的不同的缓和曲线<br>
案例12</p>
<pre><code><!DOCTYPE html>
<html>

	<head>
		<meta charset="UTF-8" />
		<title></title>
		<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
	</head>
	<style>
		/* 可以设置不同的进入和离开动画   */
		/* 设置持续时间和动画函数        */
		.fade-enter-active {
			transition: all 0.3s ease-out;
		}

		.fade-leave-active {
			transition: all 0.8s cubic-bezier(1, 0.5, 0.8, 1);
		}

		.fade-enter-from,
		.fade-leave-to {
			transform: translateX(50px);
			opacity: 0;
		}
	</style>

	<body>
		<div id="app">
			<button @click="show = !show">切换显示</button>

			<transition name="fade">
				<p v-if="show">hello</p>
			</transition>
		</div>
	</body>
	<script type="text/javascript">
		const app = Vue.createApp({
			data() {
				return {
					show: false
				}
			},
		});

		const vm = app.mount("#app");
	</script>

</html>
</code></pre>
<p>多按钮切换状态<br>
原理:<br>
使用多个 v-if 的多个元素的过渡可以重写为绑定了动态属性的单个元素过渡。<br>
案例13</p>
<pre><code><!DOCTYPE html>
<html>

	<head>
		<meta charset="UTF-8" />
		<title></title>
		<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
	</head>
	<style>
		.v-enter-active {
			animation: bounce-in 0.5s;
		}

		.v-leave-active {
			animation: bounce-in 0.5s reverse;
		}

		@keyframes bounce-in {
			0% {
				transform: scale(0);
			}

			50% {
				transform: scale(1.25);
			}

			100% {
				transform: scale(1);
			}
		}
	</style>

	<body>
		<div id="app">
			<transition>
				<button v-if="docState == 0" key="0" @click="change">
					已保存
				</button>
				<button v-else-if="docState == 1" key="1" @click="change">
					编辑完成
				</button>
				<button v-else-if="docState == 2" key="2" @click="change">
					编辑中
				</button>
			</transition>
		</div>
	</body>
	<script type="text/javascript">
		const app = Vue.createApp({
			data() {
				return {
					docState: 1
				}
			},
			methods: {
				change() {
					this.docState = ++this.docState % 3
					console.log(this.docState)
				}
			}
		});

		const vm = app.mount("#app");
	</script>

</html>
</code></pre>
<p>也可以写为:<br>
案例14</p>
<pre><code><!DOCTYPE html>
<html>

	<head>
		<meta charset="UTF-8" />
		<title></title>
		<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
	</head>
	<style>
		.v-enter-active {
			animation: bounce-in 0.5s;
		}

		.v-leave-active {
			animation: bounce-in 0.5s reverse;
		}

		@keyframes bounce-in {
			0% {
				transform: scale(0);
			}

			50% {
				transform: scale(1.25);
			}

			100% {
				transform: scale(1);
			}
		}
	</style>

	<body>
		<div id="app">
			<transition>
				<button :key="docState" @click="change">
					{{ buttonMessage }}
				</button>
			</transition>
		</div>
	</body>
	<script type="text/javascript">
		const app = Vue.createApp({
			data() {
				return {
					docState: 1
				}
			},
			computed: {
				buttonMessage() {
					switch (this.docState) {
						case 0:
							return '已保存'
						case 1:
							return '编辑完成'
						case 2:
							return '编辑中'
					}
				}
			},
			methods: {
				change() {
					this.docState = ++this.docState % 3
					console.log(this.docState)
				}
			}
		});

		const vm = app.mount("#app");
	</script>

</html>
</code></pre>
<p>列表过渡<br>
普通过度的特征:<br>
单个节点<br>
多个节点,每次只渲染一个<br>
那么怎么同时渲染整个列表,比如使用 v-for?在这种场景下,我们会使用 <transition-group> 组件。<br>
<transition-group> 组件还有一个特殊之处。不仅可以进入和离开动画,还可以改变定位。<br>
要使用这个新功能只需了解新增的 v-move 类,它会在元素的改变定位的过程中应用。像之前的类名一样,可以通过 name attribute 来自定义前缀,也可以通过 move-class attribute 手动设置。<br>
案例15</p>
<pre><code><!DOCTYPE html>
<html>

	<head>
		<meta charset="UTF-8" />
		<title></title>
		<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
	</head>
	<style>
		.v-move {
			transition: transform 1s;
			-webkit-transition: -webkit-transform 1s;
		}
	</style>

	<body>
		<div id="app">
			<button @click="add">Add</button>
			<button @click="remove">Remove</button>
			<transition-group class="list" tag="p">
				<span v-for="item in items" :key="item" class="list-item">
					{{ item }}
				</span>
			</transition-group>
		</div>
	</body>
	<script type="text/javascript">
		const app = Vue.createApp({
			data() {
				return {
					items: [1, 2, 3, 4, 5, 6, 7, 8, 9],
					nextNum: 10
				}
			},
			methods: {
				randomIndex() {
					return Math.floor(Math.random() * this.items.length)
				},
				add() {
					this.items.splice(this.randomIndex(), 0, this.nextNum++)
				},
				remove() {
					this.items.splice(this.randomIndex(), 1)
				}
			}
		});

		const vm = app.mount("#app");
	</script>

</html>
</code></pre>
<p>动画效果也可以配合各种插件一起使用<br>
如 animate.css等<br>
案例16</p>
<pre><code><!DOCTYPE html>
<html>

	<head>
		<meta charset="UTF-8" />
		<title></title>
		<link rel="stylesheet" type="text/css" href="css/animate.min.css" />
		<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
	</head>
	<style>
		.v-move {
			transition: transform 1s;
			-webkit-transition: -webkit-transform 1s;
		}
	</style>

	<body>
		<div id="app">
			<button v-on:click="add">添加</button>
			<button v-on:click="remove">删除</button>
			<button v-on:click="shuffle">乱序</button>
			<transition-group>
				<div v-for="item in items" v-bind:key="item" class="list-item">
					{{ item }}
				</div>
			</transition-group>
		</div>
	</body>
	<script type="text/javascript">
		const app = Vue.createApp({
			data() {
				return {
					items: [1, 2, 3, 4, 5, 6, 7, 8, 9],
					nextNum: 10
				}
			},
			methods: {
				randomIndex: function() {
					return Math.floor(Math.random() * this.items.length)
				},
				add: function() {
					//splice() 方法向/从数组中添加/删除项目,然后返回被删除的项目
					//index	必需。整数,规定添加/删除项目的位置,使用负数可从数组结尾处规定位置。
					//howmany	必需。要删除的项目数量。如果设置为 0,则不会删除项目。
					//item1, ..., itemX	可选。向数组添加的新项目。
					this.items.splice(this.randomIndex(), 0, this.nextNum++)
				},
				remove: function() {
					this.items.splice(this.randomIndex(), 1)
				},
				shuffle: function() {
					this.items.sort(function() {
						return 0.5 - Math.random();
					});
				}
			}
		});

		const vm = app.mount("#app");
	</script>

</html>
</code></pre>
<p>作业<br>
1.实现列表切换效果<br>
效果:</p>
<p>参考代码:</p>
<pre><code><!DOCTYPE html>
<html>

	<head>
		<meta charset="UTF-8">
		<title></title>
		<link rel="stylesheet" type="text/css" href="css/animate.min.css" />
		<script src="../js/vue.min.js" type="text/javascript" charset="utf-8"></script>
		<style type="text/css">
			.cell {
				display: flex;
				justify-content: space-around;
				align-items: center;
				width: 25px;
				height: 25px;
				border: 1px solid #aaa;
				margin-right: -1px;
				margin-bottom: -1px;
			}
			
			.v-move {
				transition: transform 1s;
				-webkit-transition: -webkit-transform 1s;
			}
		</style>
	</head>

	<body>
		<div id="app">
			<button @click="shuffle">乱序</button>
			<transition-group
				mode="out-in"
				enter-active-class="animated bounceInDown"
				leave-active-class="animated bounceOutRight">
				<div v-for="cell in items" :key="cell.id" class="cell">
					{{ cell.number }}
				</div>
			</transition-group>
		</div>
	</body>

	<script type="text/javascript">
 const app = Vue.createApp({
			data() {
				return {
					items: Array.apply(null, {
						length: 9
					}).map(function(_, index) {
						return {
							id: index,
							number: index % 9 + 1
						}
					})
				}
			},
			methods: {
				shuffle: function() {
					this.items.sort(function() {
						return 0.5 - Math.random();
					});
				}
			}
		});

		const vm = app.mount("#app");
	</script>

</html>
```
2.实现tab切换效果
效果:


参考代码:
</code></pre>
<!DOCTYPE html>
<html>
	<head>
		<meta charset="utf-8">
		<title></title>
		<script src="./vue.global.js/vue.global.js" type="text/javascript" charset="utf-8"></script>
	</head>
<style type="text/css">
	.tab ul {
		margin: 0;
		padding: 0;
		list-style: none;
		display: flex;
		justify-content: center;
	}
<pre><code>.tab ul li {
	padding: 10px 0;
	color: #2C3E50;
	width: 100px;
	position: relative;
	text-align: center;
	user-select: none;
	cursor: pointer;
}

.tab .active {
	font-weight: 900;
	color: blue;
}

.tab .active-line {
	content: "";
	width: 100px;
	height: 3px;
	background-color: blue;
	position: absolute;
	bottom: 0;
	left: 0%;
	transition: all 0.3s;
}

.tab .active-line::before {
	content: "";
	position: absolute;
	width: 0px;
	height: 0px;
	border-top: blue 8px solid;
	border-left: transparent 8px solid;
	border-right: transparent 8px solid;
	left: 50%;
	transform: translateX(-50%);
	top: 2px;
}

.content {
	display: flex;
	width: 1000px;
	margin: auto;
	padding: 30px 0;
}

.content .left {
	width: 300px;
	margin-right: 40px;
	flex: 0 0 auto;
}

.content .left h4 {
	color: blue;
	text-align: center;
	margin: 4px;
}

.content .left p {
	color: gray;
	line-height: 26px;
	height: 52px;
	overflow: hidden;
}

.content img {
	width: 100%;
}

.content .list {
	display: flex;
	flex-wrap: wrap;
}

.content .list .item {
	flex: 0 0 300px;
	padding: 10px;
	box-sizing: border-box;
	border: 1px solid rgba(0, 0, 0, 0.2);
	display: flex;
	width: 300px;
	margin-bottom: 20px;
}

.content .list .item:nth-child(odd) {
	margin-right: 40px;
}

.content .list .item .time {
	flex: 0 0 auto;
	width: 50px;
	height: 50px;
	background-color: dimgrey;
	color: white;
	font-size: 13px;
	text-align: center;
	border-radius: 2px;
	margin-right: 14px;
}

.content .list .item .time span {
	font-size: 20px;
	line-height: 30px;
	font-weight: 900;
}

.content .list .item h4 {
	margin: 0;
	font-size: 14px;
}

.content .list .item p {
	margin: 0;
	font-size: 12px;
	color: gray;
	line-height: 14px;
	height: 28px;
	overflow: hidden;
}
</code></pre>
</style>
	<body>
	<div id="app">
			<div class="tab">
				<ul>
					<li @click="tabChange($event , index)" :class="active == index ? 'active' : ''"
						v-for="(item,index) in newsList">
						{{item.title}}
						<div class="active-line" v-if="index == 0"></div>
					</li>
				</ul>
				<div class="content">
					<div class="left">
						<img :src="newsList[activeContent].content[0].image">
						<h4>{{newsList[activeContent].content[0].title}}</h4>
						<p>{{newsList[activeContent].content[0].content}}</p>
					</div>
					<div class="list">
						<template v-for="(v,i) in newsList[activeContent].content">
							<div class="item" v-if="i > 0">
								<div class="time">
									<span>{{v.date}}</span>
									{{v.year}}
								</div>
								<div>
									<h4>{{v.title}}</h4>
									<p>{{v.content}}</p>
								</div>
							</div>
						</template>
					</div>
				</div>
			</div>
		</div>
	</body>
	<script type="text/javascript">
			const App = {
				data() {
					return {
						active: 0,
						activeContent: 0,
						newsList: [{
								title: '公司新闻',
								content: [{
										time: '2025-12-3',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2025-12-21',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2025-12-32',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2025-12-12',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2025-12-3',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2025-12-4',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2025-12-13',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									}
								]
							},
							{
								title: '行业新闻',
								content: [{
										time: '2025-12-3',
										image: './img/banner.jpg',
										title: '大厦帝国沙皇我换句话说',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2025-12-2',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2025-12-32',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2025-12-12',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2025-12-3',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2025-12-13',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2025-12-13',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									}
								]
							}, {
								title: '媒体新闻',
								content: [{
										time: '2065-12-14',
										image: './img/banner.jpg',
										title: '刚刚有个后果就是空间的卡很久',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2545-12-12',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2075-12-1',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2054-12-3',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2025-12-1',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2034-12-2',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									},
									{
										time: '2012-12-16',
										image: './img/banner.jpg',
										title: '天下数据2018年国庆节放假啊',
										content: '打击和发动机卡仕达卡仕达看见哈萨克就打算电话卡实践活动卡换卡'
									}
								]
							}
						]
					}
				},
				methods: {
					tabChange(event, i) {
						this.active = i
						line = event.target.parentElement.children[0].children[0]
						line.style.left = (i * 100) + '%'
						this.activeContent = i
					}
				},
				created() {
					// ajax 求情数据完成后修改数据
					this.newsList.forEach(function(e) {
						// forEach\filter\some\every
						e.content.map(function(ee, i) {
							index = ee.time.lastIndexOf('-')
							ee.year = ee.time.substring(0,index)
							ee.date = ee.time.substr(index+1)
							return ee
						})
					})
					console.log(this.newsList)
				}
			}
			Vue.createApp(App).mount('#app')
	</script>
</html>
````
3.使用class绑定完成以下效果
效果:
<p>代码参考</p>
<pre><code><!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <title>练习</title>
    <script src="https://cdn.jsdelivr.net/npm/vue/dist/vue.js"></script><style>
    * {
        margin: 0;
        padding: 0
    }

    .stars {
        margin: 30px auto;
        padding: 10px;
        width: 360px;
        border: 1px saddlebrown solid;
    }

    .stars span {
        float: left;
        height: 30px;
        line-height: 30px;
    }

    .stars i {
        width: 20px;
        height: 30px;
        line-height: 30px;
        float: left;
        margin-right: 30px;
        color: #bab2b2;
        text-align: center;
        cursor: default;
        font-style: normal;
    }
    .stars input {
        width: 20px;
        height: 30px;
        border:0px;
    }

    .stars .on {
        color:red;
    }


</style>
</head>
<body>
<div id="app">
    <div class="stars">
        <span>商品评价:</span>
        <i v-bind:class="stars1>=1?'on':''" v-on:click="stars1=1">★</i>
        <i v-bind:class="stars1>=2?'on':''" v-on:click="stars1=2">★</i>
        <i v-bind:class="stars1>=3?'on':''" v-on:click="stars1=3">★</i>
        <i v-bind:class="stars1>=4?'on':''" v-on:click="stars1=4">★</i>
        <i v-bind:class="stars1>=5?'on':''" v-on:click="stars1=5">★</i>
        <input type="text" v-model="stars1"/>
    </div>
</div>

<script>
    new Vue({
        el: "#app",
        data: {
            stars1: 0,

        },
        methods: {}
    })
</script>
</body>
</html>
</code></pre>