目录
1.事件
事件:事件是在编程时系统内发生的动作或者发生的事情
比如用户在网页上单击一个按钮
1.1事件监听
事件监听:让程序检测是否有事件发生,一旦有事件发生,就立即调用一个函数做出响应,也称为注册事件
语法:
元素.addEventListener('事件', 要执行的函数)
事件监听三要素:
- 事件源:哪个DOM元素被事件触发了,要获取DOM元素
- 事件:用什么方式触发,比如鼠标单击click,鼠标经过mouseover等
- 事件调用的函数:要去做什么事
📖Note:
- 事件类型要加引号
- 函数是点击之后再去执行,每次点击都会执行一次
案例:二维码显示
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<style>
* {
margin: 0;
padding: 0;
}
.erweima {
position: relative;
width: 160px;
height: 160px;
margin: 100px auto;
border: 1px solid #ccc;
}
.erweima i {
position: absolute;
left: -13px;
top: 0;
width: 10px;
height: 10px;
border: 1px solid #ccc;
font-size: 12px;
line-height: 10px;
color: #ccc;
font-style: normal;
cursor: pointer;
}
</style>
</head>
<body>
<div class="erweima">
<img src="./pictures/code.png" alt="">
<i class="close_btn">x</i>
</div>
<script>
// 获取元素
let close_btn = document.querySelector('.close_btn')
let erweima = document.querySelector('.erweima')
// 事件监听
close_btn.addEventListener('click', function() {
// 二维码关闭
erweima.style.display = 'none'
})
</script>
</body>
</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>
<style>
* {
margin: 0;
padding: 0;
}
h2 {
text-align: center;
}
.box {
width: 600px;
margin: 50px auto;
display: flex;
font-size: 25px;
line-height: 40px;
}
.qs {
width: 450px;
height: 40px;
color: red;
}
.btns {
text-align: center;
}
.btns button {
width: 120px;
height: 35px;
margin: 0 50px;
}
</style>
</head>
<body>
<h2>随机点名</h2>
<div class="box">
<span>名字是:</span>
<div class="qs">这里显示姓名</div>
</div>
<div class="btns">
<button class="start">开始</button>
<button class="end">结束</button>
</div>
<script>
// 数据数组
let arr = ['张三', '李四', '王五', '赵六']
function getRandom(min, max) {
return Math.floor(Math.random() * (max - min + 1)) + min
}
// 获取元素
let start = document.querySelector('.start')
let end = document.querySelector('.end')
let qs = document.querySelector('.qs')
// timer要是个全局变量结束时才能访问
let timer = 0
let random = 0
// 开始按钮 监听事件
start.addEventListener('click', function() {
// 随机抽数 快速抽取
timer = setInterval(function() {
random = getRandom(0, arr.length-1)
qs.innerHTML = arr[random]
}, 50)
// 抽到最后一个 禁用按钮
if(arr.length === 1) {
start.disabled = true
end.disabled = true
}
})
// 结束按钮 监听事件
end.addEventListener('click', function() {
// 清除定时器
clearInterval(timer)
// 删除已经抽取的元素
arr.splice(random, 1)
})
</script>
</body>
</html>
1.2事件类型
鼠标事件:鼠标触发
- click:鼠标点击
- mouseenter:鼠标经过
- mouseleave:鼠标离开
焦点事件:表单获得光标
- focus:获得焦点
- blur:失去焦点
键盘事件:键盘触发
- Keydown:键盘按下触发
- Keyup:键盘抬起触发
文本事件:表单输入触发
- input:用户输入事件
案例:搜索框
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<style>
* {
margin: 0;
padding: 0;
box-sizing: border-box;
}
ul {
list-style: none;
}
.mi {
position: relative;
width: 223px;
margin: 100px auto;
}
.mi input {
width: 223px;
height: 48px;
padding: 0 10px;
font-size: 14px;
line-height: 48px;
border: 1px solid #e0e0e0;
outline: none;
}
.mi .search {
border: 1px solid #ff6700;
}
.result-list {
display: none;
position: absolute;
left: 0;
top: 48px;
width: 223px;
border: 1px solid #ff6700;
border-top: 0;
background: #fff;
}
.result-list a {
display: block;
padding: 6px 15px;
font-size: 12px;
color: #424242;
text-decoration: none;
}
.result-list a:hover {
background-color: #eee;
}
</style>
</head>
<body>
<div class="mi">
<input type="search" placeholder="珂拉琪">
<ul class="result-list">
<li><a href="#">珂拉琪唇釉</a></li>
<li><a href="#">珂拉琪眉笔</a></li>
<li><a href="#">珂拉琪眼影</a></li>
<li><a href="#">珂拉琪素颜霜</a></li>
<li><a href="#">珂拉琪粉底液</a></li>
<li><a href="#">珂拉琪礼盒套装</a></li>
<li><a href="#">珂拉琪卸妆湿巾</a></li>
</ul>
</div>
<script>
// 获取元素 input
let search = document.querySelector('input[type=search]')
let list = document.querySelector('.result-list')
// 监听事件 获得光标事件 focus
search.addEventListener('focus', function() {
// 显示下拉菜单
list.style.display = 'block'
// 文本框变色
search.classList.add('search')
})
// 监听事件 失去光标事件 focus
search.addEventListener('blur', function() {
// 显示下拉菜单
list.style.display = 'none'
// 文本框变色
search.classList.remove('search')
})
</script>
</body>
</html>
案例:全选框
<!DOCTYPE html>
<html>
<head lang="en">
<meta charset="UTF-8">
<title></title>
<style>
* {
margin: 0;
padding: 0;
}
table {
border-collapse: collapse;
border-spacing: 0;
border: 1px solid #c0c0c0;
width: 500px;
margin: 100px auto;
text-align: center;
}
th {
background-color: #09c;
font: bold 16px "微软雅黑";
color: #fff;
height: 24px;
}
td {
border: 1px solid #d0d0d0;
color: #404060;
padding: 10px;
}
.allCheck {
width: 80px;
}
</style>
</head>
<body>
<table>
<tr>
<th class="allCheck">
<input type="checkbox" name="" id="checkAll"> <span class="all">全选</span>
</th>
<th>商品</th>
<th>商家</th>
<th>价格</th>
</tr>
<tr>
<td>
<input type="checkbox" name="check" class="ck">
</td>
<td>珂拉琪唇釉</td>
<td>珂拉琪</td>
<td>¥49</td>
</tr>
<tr>
<td>
<input type="checkbox" name="check" class="ck">
</td>
<td>珂拉琪眉笔</td>
<td>珂拉琪</td>
<td>¥19.9</td>
</tr>
<tr>
<td>
<input type="checkbox" name="check" class="ck">
</td>
<td>珂拉琪眼影</td>
<td>珂拉琪</td>
<td>¥59.9</td>
</tr>
</table>
<script>
// 获取元素 全选和单选框
let all = document.querySelector('#checkAll')
let cks = document.querySelectorAll('.ck')
let span = document.querySelector('span')
// 监听事件 全选框
all.addEventListener('click', function() {
// all.checked是一个boolean类型的值
// 将这个值赋值给所有的单选框
for(let i = 0; i < cks.length; i++) {
cks[i].checked = all.checked
}
// 全选状态时,文本改为取消全选
if(all.checked === true) {
span.innerHTML = '取消'
} else {
span.innerHTML = '全选'
}
})
// 每一个单选框的点击事件
for(let i = 0; i < cks.length; i++) {
cks[i],addEventListener('click', function() {
// 遍历查看是否有未被选中的
for(let j = 0; j < cks.length; j++) {
if(cks[j].checked === false) {
all.checked = false
span.innerHTML = '全选'
// 退出函数
return
}
}
// 所有单选框均已选择
all.checked = true
span.innerHTML = '取消'
})
}
</script>
</body>
</html>
案例:购物车
<!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>
<style>
div {
width: 80px;
}
input[type=text] {
width: 50px;
height: 44px;
outline: none;
border: 1px solid #ccc;
text-align: center;
border-right: 0;
}
input[type=button] {
height: 24px;
width: 22px;
cursor: pointer;
}
input {
float: left;
border: 1px solid #ccc;
}
</style>
</head>
<body>
<div>
<input type="text" id="total" value="1" readonly>
<input type="button" value="+" id="add">
<input type="button" value="-" id="reduce" disabled>
<script>
// 获取元素
let add = document.querySelector('#add')
let reduce = document.querySelector('#reduce')
let num = document.querySelector('#total')
// 加号监听事件
add.addEventListener('click', function() {
// input中的数据为字符串类型 不能直接加1
// 但++中存在隐式类型转换
num.value++
reduce.disabled = false
})
// 减号监听事件
reduce.addEventListener('click', function() {
num.value--
// 比较运算符也存在隐式类型转换
if(num.value <= 1) {
reduce.disabled = true
}
})
</script>
</div>
</body>
</html>
2.高阶函数
高阶函数可以被简单理解为函数的高级应用,JavaScript中函数可以被当成值来对待,基于这个特性实现函数的高级应用
值就是JavaScript中的数据,如数值,字符串,布尔,对象等
2.1函数表达式
2.2回调函数
如果将函数A作为参数传递给函数B时,我们称A为回调函数
📖Note:
- 回调函数的本质还是函数,只是把它当成参数使用
- 使用匿名函数作为回调函数比较常见
3.环境对象
环境对象指的是函数内部特殊的变量this,它代表函数运行时所处的环境
📖Note:
- 函数的调用方式不同,this指代的对象也不同(谁调用,this就是谁)
- 直接调用函数,其实是相当于windows.函数,所以this指代window
4.编程思想
排他思想:
当前元素为A状态,其他元素为B状态
使用:
- 使用for循环修改其他元素的状态
- 通过this或者下标找到自己或者对应的元素
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<style>
.pink {
background-color: pink;
}
</style>
</head>
<body>
<button>按钮1</button>
<button>按钮2</button>
<button>按钮3</button>
<button>按钮4</button>
<button>按钮5</button>
<script>
let btns = document.querySelectorAll('button')
for(let i = 0; i < btns.length; i++) {
btns[i].addEventListener('click', function() {
// 修改其他元素
for(let j = 0; j < btns.length; j++) {
btns[j].classList.remove('pink')
}
// 修改自己
this.classList.add('pink')
})
}
</script>
</body>
</html>
改进:document.querySelector('.pink'),可以找到目前第一个调用pink类的按钮,清除该按钮的CSS效果
<body>
<button class="pink">按钮1</button>
<button>按钮2</button>
<button>按钮3</button>
<button>按钮4</button>
<button>按钮5</button>
<script>
let btns = document.querySelectorAll('button')
for(let i = 0; i < btns.length; i++) {
btns[i].addEventListener('click', function() {
// 修改其他元素
// for(let j = 0; j < btns.length; j++) {
// btns[j].classList.remove('pink')
// }
// 找到目前调用pink类的按钮
document.querySelector('.pink').classList.remove('pink')
// 修改自己
this.classList.add('pink')
})
}
</script>
</body>
</html>
5.Tab栏切换案例
思路:
- 点击当前选项卡,当前添加类,其余兄弟移除类,排他思想
- 下面模块盒子全部隐藏,当前模块显示
<!DOCTYPE html>
<html>
<head lang="en">
<meta charset="UTF-8" />
<title></title>
<style type="text/css">
* {
margin: 0;
padding: 0;
}
ul {
list-style: none;
}
.wrapper {
width: 1000px;
height: 475px;
margin: 0 auto;
margin-top: 100px;
}
.tab {
border: 1px solid #ddd;
border-bottom: 0;
height: 36px;
width: 320px;
}
.tab li {
position: relative;
float: left;
width: 80px;
height: 34px;
line-height: 34px;
text-align: center;
cursor: pointer;
border-top: 4px solid #fff;
}
.tab span {
position: absolute;
right: 0;
top: 10px;
background: #ddd;
width: 1px;
height: 14px;
overflow: hidden;
}
.products {
width: 1002px;
border: 1px solid #ddd;
height: 476px;
}
.products .main {
float: left;
display: none;
}
.products .main.active {
display: block;
}
.tab li.active {
border-color: red;
border-bottom: 0;
}
</style>
</head>
<body>
<div class="wrapper">
<ul class="tab">
<li class="tab-item active">国际大牌<span>◆</span></li>
<li class="tab-item">国妆名牌<span>◆</span></li>
<li class="tab-item">清洁用品<span>◆</span></li>
<li class="tab-item">男士精品</li>
</ul>
<div class="products">
<div class="main active">
<a href="###"><img src="imgs/guojidapai.jpg" alt="" /></a>
</div>
<div class="main">
<a href="###"><img src="imgs/guozhuangmingpin.jpg" alt="" /></a>
</div>
<div class="main">
<a href="###"><img src="imgs/qingjieyongpin.jpg" alt="" /></a>
</div>
<div class="main">
<a href="###"><img src="imgs/nanshijingpin.jpg" alt="" /></a>
</div>
</div>
</div>
<script>
// 获取元素
let items = document.querySelectorAll('.tab .tab-item')
let mains = document.querySelectorAll('.products .main')
// 头部Tab栏切换
for(let i = 0; i < items.length; i++) {
items[i].addEventListener('click', function() {
// 清除其他元素的点击事件
// 移除已经调用active类的按钮
document.querySelector('.tab .active').classList.remove('active')
// 设置自己的点击事件
this.classList.add('active')
// 底部内容区域 是点击事件的响应 写在点击事件的里面
document.querySelector('.products .active').classList.remove('active')
// 显示对应的内容
mains[i].classList.add('active')
})
}
</script>
</body>
</html>