Web APIs 和 JS
-
API
-
API(Application Programming Interface,应用程序编程接口)是一些预先定义的函数,目的是提供应用程序与开发人员基于某软件或硬件得以访问一组例程的能力,而又无需访问源码,或理解内部工作机制的细节
-
是给程序员提供的一种工具,以便能更轻松的实现想要完成的功能
-
-
Web API
-
是浏览器提供的一套操作浏览器功能和页面元素的 API ( BOM 和 DOM )
-
主要是针对于浏览器提供的接口,主要针对于浏览器做交互效果
-
一般都有输入和输出(函数的传参和返回值),Web API 很多都是方法(函数)
-
MDN 详细 API : Web API 接口参考 | MDN
-
DOM简介
-
DOM:文档对象模型(Document Object Model,简称 DOM),是 W3C 组织推荐的处理可扩展标记语言(HTML或者XML)的标准编程接口
-
W3C 已经定义了一系列的 DOM 接口,通过这些 DOM 接口可以改变网页的内容、结构和样式。
-
-
DOM树
-
文档:一个页面就是一个文档,DOM中使用doucument来表示
-
元素:页面中的所有标签都是元素,DOM中使用 element 表示
-
节点:网页中的所有内容都是节点(标签,属性,文本,注释等),DOM中使用node表示
-
DOM 把以上内容都看做是对象
-
获取元素
DOM在我们实际开发中主要用来操作元素。
获取页面中的元素可以使用以下几种方式:
-
根据 ID 获取
-
根据标签名获取
-
通过 HTML5 新增的方法获取
-
特殊元素获取
根据ID获取
使用 getElementByld()
方法可以获取带ID的元素对象
使用 console.dir()
可以打印我们获取的元素对象,更好的查看对象里面的属性和方法。
<div id="time">2019-9-9</div> <script> // 1.因为我们文档页面从上往下加载,所以得先有标签,所以script写在标签下面 // 2.get 获得 element 元素 by 通过 驼峰命名法 // 3.参数 id是大小写敏感的字符串 // 4.返回的是一个元素对象 var timer = document.getElementById('time'); console.log(timer); console.log(typeof timer); // object // 5. console.dir 打印我们的元素对象,更好的查看里面的属性和方法 console.dir(timer); </script>
根据标签名获取
根据标签名获取,使用 getElementByTagName()
方法可以返回带有指定标签名的对象的集合,以伪数组的形式存储
doucument.getElementsByTagName('标签名');
-
因为得到的是一个对象的集合,所以我们想要操作里面的元素就需要遍历
-
得到元素对象是动态的,即元素内容变化,下面js也会跟着变
-
返回的是获取过来元素对象的集合,以伪数组的形式存储
-
如果页面中只有一个元素,返回的也是伪数组的形式
-
如果获取不到元素,则返回为空的伪数组(因为获取不到对象)
<ul> <li>知否知否,应是等你好久</li> <li>知否知否,应是等你好久</li> <li>知否知否,应是等你好久</li> <li>知否知否,应是等你好久</li> <li>知否知否,应是等你好久</li> </ul> <script> // 1.返回的是获取过来元素对象的集合(获取的是页面里所有的li) 以伪数组的形式存储 var lis = document.getElementsByTagName('li'); console.log(lis); console.log(lis[0]); // 2.依次打印,遍历 for (var i = 0; i < lis.length; i++) { console.log(lis[i]); } // 3.如果页面中只有 1 个 li,返回的还是伪数组的形式 // 4.如果页面中没有这个元素,返回的是空伪数组 </script>
还可以获取某个元素(父元素)内部所有指定标签名的子元素
//element.getElementsByTagName('标签名'); 父元素必须是指定的单个元素 var ol = document.getElementById('ol'); // 获取所有的ol console.log(ol[0].getElementsByTagName('li')); // 获取ol第一个ol中所有的li
<ul id="list"> <li></li> <li></li> <li></li> </ul> <script> var ollist = document.getElementById('list') console.log(ollist.getElementsByClassName('li')); //获取id为list的ul里面所有的li </script>
通过HTML5新增的方法获取
根据类名返回元素对象集合document.getElementsByClassName('类名')
根据指定选择器返回第一个元素对象 document.querySelector('选择器');
根据指定选择器返回所有元素对象 document.querySelectorAll('选择器');
集合
-
注意:querySelector和 querySelectorAll里面的选择器需要加符号(类选择器.box,id选择器 #nav),比如: document.querySelector('#nav');
// 1. getElementsByClassName 根据类名获得某些元素集合 var boxs = document.getElementsByClassName('box'); console.log(boxs); // 2. querySelector 返回指定选择器的第一个元素对象 切记 里面的选择器需要加符号 .box #nav var firstBox = document.querySelector('.box'); console.log(firstBox); var nav = document.querySelector('#nav'); console.log(nav); var li = document.querySelector('li'); console.log(li); // 3. querySelectorAll()返回指定选择器的所有元素对象集合 var allBox = document.querySelectorAll('.box'); console.log(allBox); var lis = document.querySelectorAll('li'); console.log(lis);
获取body元素
返回body元素对象 document.body;
获取html元素
返回html元素对象 document.documentElement;
事件基础
JavaScript 使我们有能力创建动态页面,而事件是可以被 JavaScript 侦测到的行为。
简单理解: 触发— 响应机制。
网页中的每个元素都可以产生某些可以触发 JavaScript 的事件,例如,我们可以在用户点击某按钮时产生一个事件,然后去执行某些操作
事件三要素
-
事件源(谁)
-
事件类型(什么事件)
-
事件处理程序(做啥)
// 点击一个按钮,弹出对话框 // 1. 事件是有三部分组成 事件源 事件类型 事件处理程序 也称为事件三要素 //(1) 事件源 事件被触发的对象 谁 按钮 var btn = document.getElementById('btn'); //(2) 事件类型 如何触发 什么事件 比如鼠标点击(onclick) 还是鼠标经过 还是键盘按下 //(3) 事件处理程序 通过一个函数赋值的方式 完成 btn.onclick = function() { alert('点秋香'); }
执行事件的步骤
-
获取事件源
-
注册事件(绑定事件)
-
添加事件处理程序(通过匿名函数赋值的方法来实现)
// 执行事件步骤 // 点击div 控制台输出 我被选中了 // 1. 获取事件源 var div = document.querySelector('div'); // 2.绑定事件 注册事件 // div.onclick // 3.添加事件处理程序 div.onclick = function() { console.log('我被选中了'); }
鼠标事件
鼠标事件 | 触发条件 |
---|---|
onclick | 鼠标点击左键触发 |
onmouseover | 鼠标经过触发 |
onmouseout | 鼠标离开触发 |
onfocus | 获得鼠标焦点触发 |
onblur | 失去鼠标焦点触发 |
onmousemove | 鼠标移动触发 |
onmouseup | 鼠标弹起触发 |
onmousedown | 鼠标按下触发 |
操作元素
JavaScript 的 DOM 操作可以改变网页内容、结构和样式,我们可以利用 DOM 操作元素来改变元素里面的内容 、属性等。注意以下都是属性
改变元素内容
element.innerText
从起始位置到终止位置的内容,但它去除html标签、空格和换行。
element.innerHTML
起始位置到终止位置的全部内容,包括HTML标签,同时保留空格和换行
<div></div> <p> 我是文字 <span>123</span> </p> //1.innerText 不识别html标签;非标准;取出空格和换行 var div = document.querySelector('div'); div.innerText = '<strong>今天是:</strong> 2019'; //<strong>今天是:</strong> 2019 //2.innerHTML 识别html标签;W3C标准;保留空格和换行 div.innerHTML = '<strong>今天是:</strong> 2019'; //今天是:(加粗)2019 //这两个属性是可读写的,可以获取元素里面的内容 var p = document.querySelector('p'); console.log(p.innerText); //我是文字 123 console.log(p.innerHTML); // 我是文字 换行 <span>123</span>
改变元素属性
-
表单元素的属性操作
-
type、value、checked、selected、disabled
-
-
图片元素的属性操作
-
src、alt、title
-
-
a链接的属性操作
-
href
-
-
所有元素通用
-
id
-
img.src = '新链接'; img.title = '新标题'; img.alt = '新文字'; a.href = '新链接' input.value = '新内容'; input.type = '新input类型'; input.checked = true / false; input.selected = true / false; input.disabled = true / false;
显示密码
点击按钮将密码框切换为文本框,并可以查看密码明文
<!DOCTYPE html> <html lang="zh-CN"> <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> .box { width: 400px; border-bottom: 1px solid #ccc; margin: 0 auto; position: relative; margin: 20px auto 0; } .box input { width: 300px; height: 30px; border: 0; /*去除表单选中后的轮廓 */ outline: none; } .box img { position: absolute; top: 2px; right: 75px; width: 24px; } .box a { font-size: 14px; position: absolute; top: 10px; right: 2px; line-height: 1; text-decoration: none; border-left: 2px solid #ccc; padding-left: 10px; color: black; } .box a:hover { color: rgb(117, 117, 117); } h2 { display: block; width: 100px; text-align: center; font-size: 16px; font-weight: 400; margin: 100px auto 0; } .username { width: 400px; border-bottom: 1px solid #ccc; margin: 30px auto 0; } .username input { outline: none; width: 370px; height: 30px; border: 0; } button { display: block; width: 400px; height: 30px; border-radius: 15px; background-color: red; border: 0; color: white; margin: 20px auto 0; } .otherlogin { width: 400px; height: 30px; margin: 15px auto 0; } .otherlogin a { display: block; float: left; font-size: 14px; color: rgb(117, 117, 117); line-height: 30px; text-decoration: none; } .otherlogin a:last-child { float: right; } .otherlogin a:hover { color: black; } </style> </head> <body> <h2>京东登录</h2> <div class="username"> <input type="text" placeholder="用户名/邮箱/已验证手机"> </div> <div class="box"> <!-- label经常和input搭配一起使用 --> <label> <img src="../java-script-master/JavaScript APIs 素材/jsapis_material/第一天/images/close.png" alt="" id="eye"> </label> <input type="password" id="pwd" placeholder="请输入密码"> <a href="#">忘记密码</a> </div> <button>登录</button> <div class="otherlogin"> <a href="#">短信验证码登录</a> <a href="#">手机快速注册</a> </div> <script> //1.获取元素 var eye = document.getElementById('eye'); var pwd = document.getElementById('pwd'); //2.注册事件 处理程序 var flag = 0; eye.onclick = function() { //按钮算法:点击一次之后flag要变化 if (flag == 0) { pwd.type = 'text'; flag = 1; eye.src = '../java-script-master/JavaScript APIs 素材/jsapis_material/第一天/images/open.png' } else { pwd.type = 'password'; flag = 0; eye.src = '../java-script-master/JavaScript APIs 素材/jsapis_material/第一天/images/close.png' } } </script> </body> </html>
表单全选&取消全选
<!DOCTYPE html> <html lang="zh-CN"> <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> * { padding: 0; margin: 0; } .wrap { width: 300px; margin: 100px auto 0; } table { border-collapse: collapse; border-spacing: 0; border: 1px solid #c0c0c0; width: 300px; } th, td { border: 1px solid #d0d0d0; color: #404060; padding: 10px; } th { background-color: #09c; font: bold 16px "微软雅黑"; color: #fff; } td { font: 14px "微软雅黑"; } tbody tr { background-color: #f0f0f0; } tbody tr:hover { cursor: pointer; background-color: #fafafa; } </style> </head> <body> <div class="wrap"> <table> <thead> <tr> <th> <input type="checkbox" id="j_cbAll" /> </th> <th>商品</th> <th>价钱</th> </tr> </thead> <tbody id="j_tb"> <tr> <td> <input type="checkbox" /> </td> <td>iPhone8</td> <td>8000</td> </tr> <tr> <td> <input type="checkbox" /> </td> <td>iPad Pro</td> <td>5000</td> </tr> <tr> <td> <input type="checkbox" /> </td> <td>iPad Air</td> <td>2000</td> </tr> <tr> <td> <input type="checkbox" /> </td> <td>Apple Watch</td> <td>2000</td> </tr> </tbody> </table> </div> <script> // 1. 全选和取消全选做法: 让下面所有复选框的checked属性(选中状态) 跟随 全选按钮即可 // 获取元素 var j_cbAll = document.getElementById('j_cbAll'); // 全选按钮 var j_tbs = document.getElementById('j_tb').getElementsByTagName('input'); // 下面所有的复选框 // 注册事件 j_cbAll.onclick = function() { // this.checked 它可以得到当前复选框的选中状态如果是true 就是选中,如果是false 就是未选中 console.log(this.checked); for (var i = 0; i < j_tbs.length; i++) { j_tbs[i].checked = this.checked; //让下面所有的复选框与上面的一样 } } // 2. 下面复选框需要全部选中, 上面全选才能选中做法: 给下面所有复选框绑定点击事件,每次点击,都要循环查看下面所有的复选框是否有没选中的,如果有一个没选中的, 上面全选就不选中。 for (var i = 0; i < j_tbs.length; i++) { j_tbs[i].onclick = function() { // flag 控制全选按钮是否选中 var flag = true; // 每次点击下面的复选框都要循环检查者4个小按钮是否全被选中 for (var i = 0; i < j_tbs.length; i++) { //此for循环包在另外一个函数里面,是不同的作用域,不会影响,不用换成j if (!j_tbs[i].checked) { //取反:没有被选中 flag = false; break; // 退出for循环 这样可以提高执行效率 因为只要有一个没有选中,剩下的就无需循环判断了 } } j_cbAll.checked = flag; } } //或建立一个flag计被选中的数量: // for (var i = 0; i < j_tbs.length; i++) { // j_tbs[i].onclick = function() { // // flag统计被选中的个数,如果全都被选中,则全选框也被选中 // var flag = 0; // for (var i = 0; i < j_tbs.length; i++) { // if (j_tbs[i].checked) { // flag++; //每有一个被选中,flag加1 // } // } // if (flag === j_tbs.length) { //全被选中 // j_cbAll.checked = true; //或='checked' // } // } // } </script> </body> </html>
改变样式属性
通过 JS 修改元素的大小、颜色、位置等样式
-
行内样式操作
element.style
-
需要修改的样式比较少的时候
-
div.style.backgroundColor = 'pink'; div.style.width = '250px';
-
类名样式操作
element.className
-
注意:
-
JS里面的样式采取驼峰命名法,比如 fontSize ,backgroundColor
-
JS 修改 style 样式操作 ,产生的是行内样式,CSS权重比较高
-
如果样式修改较多,可以采取操作类名方式更改元素样式
-
class 因为是个保留字,因此使用className来操作元素类名属性
-
className 会直接更改元素的类名,会覆盖原先的类名(想要保留原类名需要
element.className = '原类名 新类名';
)
-
<body> <div class="first">文本</div> <script> // 1. 使用 element.style 获得修改元素样式 如果样式比较少 或者 功能简单的情况下使用 var test = document.querySelector('div'); test.onclick = function() { // this.style.backgroundColor = 'purple'; // this.style.color = '#fff'; // this.style.fontSize = '25px'; // this.style.marginTop = '100px'; // 让当前元素的类名改为了 change // 2. 可以通过 修改元素的className更改元素的样式 适合于样式较多或者功能复杂的情况 // 3. 如果想要保留原先的类名,可以用多类名选择器 // this.className = 'change'; this.className = 'first change'; } </script> </body>
关闭广告
<style> .box { position: relative; width: 74px; height: 88px; border: 1px solid #ccc; margin: 100px auto; font-size: 12px; text-align: center; color: #f40; /* display: block; */ } .box img { width: 60px; margin-top: 5px; } .close-btn { position: absolute; top: -1px; left: -16px; width: 14px; height: 14px; border: 1px solid #ccc; line-height: 14px; font-family: Arial, Helvetica, sans-serif; cursor: pointer; } </style> </head> <body> <div class="box"> 淘宝二维码 <img src="../java-script-master/JavaScript APIs 素材/jsapis_material/第一天/images/tao.png" alt=""> <i class="close-btn">×</i> </div> <script> // 1. 获取元素 var btn = document.querySelector('.close-btn'); var box = document.querySelector('.box'); // 2.注册事件 程序处理 btn.onclick = function() { box.style.display = 'none'; } </script> </body>
循环精灵图背景
<style> * { margin: 0; padding: 0; } li { list-style-type: none; } .box { width: 250px; margin: 100px auto; } .box li { float: left; width: 24px; height: 24px; background-color: pink; margin: 15px; background: url(images/sprite.png) no-repeat; } </style> <body> <div class="box"> <ul> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> <li></li> </ul> </div> <script> // 1. 获取元素 所有的小li var lis = document.querySelectorAll('li'); for (var i = 0; i < lis.length; i++) { // 让索引号 乘以 44 就是每个li 的背景y坐标 声明一个index就是我们的y坐标 var index = i * 44; lis[i].style.backgroundPosition = '0 -' + index + 'px'; } </script> </body>
显示隐藏文本框内容
用value做可以直接点搜索,placeholder不行
<!DOCTYPE html> <html lang="zh-CN"> <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; } label { display: block; width: 500px; height: 30px; margin: 100px auto; /* background-color: pink; */ } input { color: #999; border: 2px solid rgb(9, 147, 254); outline: none; vertical-align: middle; height: 30px; width: 420px; box-sizing: border-box; } a { text-decoration: none; color: white; vertical-align: middle; background-color: rgb(9, 147, 254); display: inline-block; width: 80px; height: 30px; text-align: center; line-height: 30px; } </style> </head> <body> <label> <input type="text" value="手机"><a href="#">搜索</a> </label> <script> var text = document.querySelector('input'); text.onfocus = function() { if (this.value === '手机') { this.value = ''; } this.style.color = 'black'; } text.onblur = function() { if (this.value !== '手机') { this.value = '手机'; } this.style.color = '#999' } </script> </body> </html>
密码框提示正确/错误信息
<!DOCTYPE html> <html lang="zh-CN"> <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> div { width: 600px; margin: 100px auto; } .message { display: inline-block; font-size: 12px; color: #999; background: url(images/mess.png) no-repeat left center; padding-left: 20px; } .wrong { color: red; background-image: url(images/wrong.png); } .right { color: green; background-image: url(images/right.png); } </style> </head> <body> <div class="register"> <input type="password" class="ipt"> <p class="message">请输入6~16位密码</p> </div> <script> // 1.获取元素 var ipt = document.querySelector('.ipt'); var message = document.querySelector('.message'); //2. 注册事件 失去焦点 ipt.onblur = function() { // 根据表单里面值的长度 ipt.value.length if (this.value.length < 6 || this.value.length > 16) { // console.log('错误'); message.className = 'message wrong'; message.innerHTML = '请输入6~16位的密码'; } else { message.className = 'message right'; message.innerHTML = '您输入的正确'; } } </script> </body> </html>
百度换肤
<!DOCTYPE html> <html lang="zh-CN"> <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; } body { background: url(images/1.jpg) no-repeat center top; } li { list-style: none; } .baidu { overflow: hidden; margin: 100px auto; background-color: #fff; width: 410px; padding-top: 3px; } .baidu li { float: left; margin: 0 1px; cursor: pointer; } .baidu img { width: 100px; } </style> </head> <body> <ul class="baidu"> <li><img src="images/1.jpg"></li> <li><img src="images/2.jpg"></li> <li><img src="images/3.jpg"></li> <li><img src="images/4.jpg"></li> </ul> <script> // 1. 获取元素 var imgs = document.querySelector('.baidu').querySelectorAll('img'); // console.log(imgs); // 2. 循环注册事件 for (var i = 0; i < imgs.length; i++) { imgs[i].onclick = function() { // this.src 就是我们点击图片的路径 images/2.jpg // console.log(this.src); // 把这个路径 this.src 给body 就可以了 document.body.style.backgroundImage = 'url(' + this.src + ')'; //this.src是个属性值,需要引引加加 } } </script> </body> </html>
表格隔行变色
<!DOCTYPE html> <html lang="zh-CN"> <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> table { width: 800px; margin: 100px auto; text-align: center; border-collapse: collapse; font-size: 14px; } thead tr { height: 30px; background-color: skyblue; } tbody tr { height: 30px; } tbody td { border-bottom: 1px solid #d7d7d7; font-size: 12px; color: blue; } .bg { background-color: pink; } </style> </head> <body> <table> <thead> <tr> <th>代码</th> <th>名称</th> <th>最新公布净值</th> <th>累计净值</th> <th>前单位净值</th> <th>净值增长率</th> </tr> </thead> <tbody> <tr> <td>003526</td> <td>农银金穗3个月定期开放债券</td> <td>1.075</td> <td>1.079</td> <td>1.074</td> <td>+0.047%</td> </tr> <tr> <td>003526</td> <td>农银金穗3个月定期开放债券</td> <td>1.075</td> <td>1.079</td> <td>1.074</td> <td>+0.047%</td> </tr> <tr> <td>003526</td> <td>农银金穗3个月定期开放债券</td> <td>1.075</td> <td>1.079</td> <td>1.074</td> <td>+0.047%</td> </tr> <tr> <td>003526</td> <td>农银金穗3个月定期开放债券</td> <td>1.075</td> <td>1.079</td> <td>1.074</td> <td>+0.047%</td> </tr> <tr> <td>003526</td> <td>农银金穗3个月定期开放债券</td> <td>1.075</td> <td>1.079</td> <td>1.074</td> <td>+0.047%</td> </tr> </tbody> </table> <script> // 1.获取元素 获取的是 tbody 里面所有的行 var trs = document.querySelector('tbody').querySelectorAll('tr'); // 2. 利用循环绑定注册事件 for (var i = 0; i < trs.length; i++) { // 3. 鼠标经过事件 onmouseover trs[i].onmouseover = function() { // console.log(11); this.className = 'bg'; } // 4. 鼠标离开事件 onmouseout trs[i].onmouseout = function() { this.className = ''; } } </script> </body> </html>
排他思想
当前的变,其余的不变
如果有同一组元素,我们相要某一个元素实现某种样式,需要用到循环的排他思想算法:
-
所有元素全部清除样式(干掉其他人)
-
给当前元素设置样式 (留下我自己)
-
注意顺序不能颠倒,首先干掉其他人,再设置自己
<body> <button>按钮1</button> <button>按钮2</button> <button>按钮3</button> <button>按钮4</button> <button>按钮5</button> <script> // 1. 获取所有按钮元素 var btns = document.getElementsByTagName('button'); // btns得到的是伪数组 里面的每一个元素 btns[i] for (var i = 0; i < btns.length; i++) { btns[i].onclick = function() { // (1) 先把所有的按钮背景颜色去掉 干掉所有人 for (var i = 0; i < btns.length; i++) { btns[i].style.backgroundColor = ''; } // (2) 然后才让当前的元素背景颜色为pink 留下自己 this.style.backgroundColor = 'pink'; } } //2. 排他思想:首先先排除其他人,然后才设置自己的样式 </script> </body>
自定义属性
获取属性值
-
获取内置属性值(元素本身自带的属性)
element.属性;
-
获取自定义的属性(主要获得自定义的属性 (标准) 我们程序员自定义的属性,如index,data-index等等)
element.getAttribute('属性');
设置属性值
-
设置内置属性值
element.属性 = '值';
-
主要设置自定义的属性(标准)
element.setAttribute('属性');
-
calss特殊,里面写的就是class不是className
-
//element.属性 = '值'; div.id = 'list'; div.classname = 'navs'; //element.setAttribute(属性, 值); 主要针对自定义属性 div.setAttribute('index', 2); div.setAttribute('class', 'footer'); //写class而不是classNAme;给div设置了一个名叫footer的类
移除属性
element.removeAttribute('属性');
tab 栏切换
<!DOCTYPE html> <html lang="zh-CN"> <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; } li { list-style-type: none; } .tab { width: 978px; margin: 100px auto; } .tab_list { height: 39px; border: 1px solid #ccc; background-color: #f1f1f1; } .tab_list li { float: left; height: 39px; line-height: 39px; padding: 0 20px; text-align: center; cursor: pointer; } .tab_list .current { background-color: #c81623; color: #fff; } .item_info { padding: 20px 0 0 20px; } .item { display: none; } </style> </head> <body> <div class="tab"> <div class="tab_list"> <ul> <li class="current">商品介绍</li> <li>规格与包装</li> <li>售后保障</li> <li>商品评价(50000)</li> <li>手机社区</li> </ul> </div> <div class="tab_con"> <div class="item" style="display: block;"> 商品介绍模块内容 </div> <div class="item"> 规格与包装模块内容 </div> <div class="item"> 售后保障模块内容 </div> <div class="item"> 商品评价(50000)模块内容 </div> <div class="item"> 手机社区模块内容 </div> </div> </div> <script> // 获取元素 var tab_list = document.querySelector('.tab_list'); var lis = tab_list.querySelectorAll('li'); var items = document.querySelectorAll('.item'); // for循环绑定点击事件 for (var i = 0; i < lis.length; i++) { console.log(i); // 0 1 2 3 4 在这一步的时候,还没点击,但是这个大的给每个li设置点击事件的for循环已经循环完了,i已经变成lis.length-1了;点击的时候这个小for循环里面的itmes[i]已经用不了了,所以必须设置一个index来控制item的变化 // 开始给5个小li 设置索引号 lis[i].setAttribute('index', i);// 索引号不管点不点击都在 lis[i].onclick = function() { // 1. 上的模块选项卡,点击某一个,当前这一个底色会是红色,其余不变(排他思想) 修改类名的方式 // 干掉所有人 其余的li清除 class 这个类 for (var i = 0; i < lis.length; i++) { lis[i].className = ''; } // 留下我自己 this.className = 'current'; // 2. 下面的显示内容模块 var index = this.getAttribute('index'); // 点哪个就得到哪个的index console.log(index); // 干掉所有人 让其余的item 这些div 隐藏 for (var i = 0; i < items.length; i++) { items[i].style.display = 'none'; } // 留下我自己 让对应的item 显示出来 items[index].style.display = 'block'; //点了第几个li就让第几个div显示出来 } } </script> </body> </html>
H5自定义属性
自定义属性目的:
-
保存并使用数据,有些数据可以保存到页面中而不用保存到数据库中
-
有些自定义属性很容易引起歧义,不容易判断到底是内置属性还是自定义的,所以H5有了规定
-
自定义属性获取是通过getAttribute(‘属性’) 获取
设置H5自定义属性
H5规定自定义属性 data-
开头作为属性名并赋值
<div data-index=“1”></div> // 或者使用 JS 设置 div.setAttribute(‘data-index’, 2)
获取H5自定义属性
-
兼容性获取
element.getAttribute('data-index')
-
H5新增的:
element.dataset.index
或element.dataset['index']
IE11才开始支持-
dataset 是一个集合,里面存放了所有以data开头的自定义属性;获取data-index写div.dataset.index
-
如果自定义属性里面有多个-链接的单词,获取的时候采取 驼峰命名法;如获取data-list-name写div.dataset.listName
-
<body> <div getTime="20" data-index="2" data-list-name="andy"></div> <script> var div = document.querySelector('div'); console.log(div.getAttribute('getTime')); // 20 div.setAttribute('data-time', 20); console.log(div.getAttribute('data-index')); // 2 console.log(div.getAttribute('data-list-name')); // andy // h5新增的获取自定义属性的方法 它只能获取data-开头的 // dataset 是一个集合里面存放了所有以data开头的自定义属性 console.log(div.dataset); console.log(div.dataset.index); console.log(div.dataset['index']); // 如果自定义属性里面有多个-链接的单词,获取的时候采取 驼峰命名法 console.log(div.dataset.listName); console.log(div.dataset['listName']); </script> </body>
节点操作
获取元素通常使用两种方式:
1.利用DOM提供的方法获取元素 | 2.利用节点层级关系获取元素 |
---|---|
document.getElementById() | 利用父子兄节点关系获取元素 |
document.getElementsByTagName() | 逻辑性强,但是兼容性较差 |
document.querySelector 等 | |
逻辑性不强,繁琐 |
节点概述
网页中的所有内容都是节点(标签、属性、文本、注释等),在DOM 中,节点使用 node 来表示。
-
HTML DOM 树中的所有节点均可通过 JavaScript 进行访问,所有 HTML 元素(节点)均可被修改,也可以创建或删除。
-
一般地,节点至少拥有nodeType(节点类型)、nodeName(节点名称)和nodeValue(节点值)这三个基本属性。
-
元素节点:nodeType 为1
-
属性节点:nodeType 为2
-
文本节点:nodeType 为3(文本节点包括文字、空格、换行等)
-
-
我们在实际开发中,节点操作主要操作的是元素节点
-
利用 DOM 树可以把节点划分为不同的层级关系,常见的是父子兄层级关系。
节点层级
父级节点
node.parentNode
-
parentNode
属性可以返回某节点的父结点,注意是最近的一个父结点 -
如果指定的节点没有父结点则返回null
<body> <div class="demo"> <div class="box"> <span class="erweima">×</span> </div> </div> <script> // 1. 父节点 parentNode var erweima = document.querySelector('.erweima'); // var box = document.querySelector('.box'); // 得到的是离元素最近的父级节点(亲爸爸) 如果找不到父节点就返回为 null console.log(erweima.parentNode); //.box </script> </body>
子节点
-
parentNode.childNodes
(标准)-
parentNode.childNodes
返回包含指定节点的子节点的集合,该集合为即时更新的集合 -
返回值包含了所有的子结点,包括元素节点,文本节点等
-
如果只想要获得里面的元素节点,则需要专门处理。所以我们一般不提倡使用
childNodes
-
// 想仅获得元素节点 var ul = document.querySelector(‘ul’); for (var i = 0; i < ul.childNodes.length; i++) { if (ul.childNodes[i].nodeType == 1) { // ul.childNodes[i] 是元素节点 console.log(ul.childNodes[i]); } }
-
parentNode.children
(非标准)-
parentNode.children
是一个只读属性,返回所有的子元素节点 -
它只返回子元素节点,其余节点不返回 (这个是我们重点掌握的)
-
虽然 children 是一个非标准,但是得到了各个浏览器的支持,因此我们可以放心使用
-
<body> <ul> <li>我是li</li> <li>我是li</li> <li>我是li</li> <li>我是li</li> </ul> <ol> <li>我是li</li> <li>我是li</li> <li>我是li</li> <li>我是li</li> </ol> <script> // DOM 提供的方法(API)获取 var ul = document.querySelector('ul'); var lis = ul.querySelectorAll('li'); // 1. 子节点 childNodes 所有的子节点 包含 元素节点 文本节点等等 console.log(ul.childNodes); console.log(ul.childNodes[0].nodeType); console.log(ul.childNodes[1].nodeType); // 2. children 获取所有的子元素节点 也是我们实际开发常用的 console.log(ul.children); console.log(ul.children[0]); // 获取ul第一个子元素 </script> </body>
-
parentNode.firstChild
-
firstChild
返回第一个子节点,找不到则返回null -
同样,也是包含所有的节点
-
-
parentNode.lastChild
-
lastChild
返回最后一个子节点,找不到则返回null -
同样,也是包含所有的节点
-
-
parentNode.firstElementChild
(兼容性)-
firstElementChild
返回第一个子元素节点,找不到则返回null -
有兼容性问题,IE9以上才支持
-
-
parentNode.lastElementChild
(兼容性)-
lastElementChild
返回最后一个子节点,找不到则返回null -
有兼容性问题,IE9以上才支持
-
-
实际开发中,没有兼容性问题
-
如果想要第一个子元素节点,可以使用
parentNode.chilren[0]
-
如果想要最后一个子元素节点,可以使用
parentNode.chilren[parentNode.chilren.length - 1]
-
<body> <ol> <li>我是li1</li> <li>我是li2</li> <li>我是li3</li> <li>我是li4</li> <li>我是li5</li> </ol> <script> var ol = document.querySelector('ol'); // 1. firstChild 第一个子节点 不管是文本节点还是元素节点 console.log(ol.firstChild); console.log(ol.lastChild); // 2. firstElementChild 返回第一个子元素节点 ie9才支持 console.log(ol.firstElementChild); console.log(ol.lastElementChild); // 3. 实际开发的写法 既没有兼容性问题又返回第一个子元素 console.log(ol.children[0]); //第一个子元素节点 console.log(ol.children[ol.children.length - 1]);//最后一个子元素节点 </script> </body>
<body> <ol> <li>我是li1</li> <li>我是li2</li> <li>我是li3</li> <li>我是li4</li> </ol> <script> var ol = document.querySelector('ol'); // 1.firstChild 获取第一个子结点的,包含文本结点和元素结点 console.log(ol.firstChild); // 返回的是文本结点 #text(第一个换行结点) console.log(ol.lastChild); // 返回的是文本结点 #text(最后一个换行结点) // 2. firstElementChild 返回第一个子元素结点 console.log(ol.firstElementChild); // <li>我是li1</li> // 第2个方法有兼容性问题,需要IE9以上才支持 // 3.实际开发中,既没有兼容性问题,又返回第一个子元素 console.log(ol.children[0]); // <li>我是li1</li> console.log(ol.children[3]); // <li>我是li4</li> // 当里面li个数不唯一时候,需要取到最后一个结点时这么写 console.log(ol.children[ol.children.length - 1]); </script> </body>
下拉菜单
大ul里面有四个标题小li,每一个标题li里有一个ul,里面有若干选项小li
<!DOCTYPE html> <html lang="zh-CN"> <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; } li { list-style-type: none; } a { text-decoration: none; font-size: 14px; } .nav { margin: 100px; } .nav>li { position: relative; float: left; width: 80px; height: 41px; text-align: center; } .nav li a { display: block; width: 100%; height: 100%; line-height: 41px; color: #333; } .nav>li>a:hover { background-color: #eee; } .nav ul { display: none; position: absolute; top: 41px; left: 0; width: 100%; border-left: 1px solid #FECC5B; border-right: 1px solid #FECC5B; } .nav ul li { border-bottom: 1px solid #FECC5B; } .nav ul li a:hover { background-color: #FFF5DA; } </style> </head> <body> <ul class="nav"> <li> <a href="#">微博</a> <ul> <li> <a href="">私信</a> </li> <li> <a href="">评论</a> </li> <li> <a href="">@我</a> </li> </ul> </li> <li> <a href="#">微博</a> <ul> <li> <a href="">私信</a> </li> <li> <a href="">评论</a> </li> <li> <a href="">@我</a> </li> </ul> </li> <li> <a href="#">微博</a> <ul> <li> <a href="">私信</a> </li> <li> <a href="">评论</a> </li> <li> <a href="">@我</a> </li> </ul> </li> <li> <a href="#">微博</a> <ul> <li> <a href="">私信</a> </li> <li> <a href="">评论</a> </li> <li> <a href="">@我</a> </li> </ul> </li> </ul> <script> // 1. 获取元素 var nav = document.querySelector('.nav'); var lis = nav.children; // 得到4个小li // 2.循环注册事件 for (var i = 0; i < lis.length; i++) { lis[i].onmouseover = function() { // console.log(i); // 移动到任意一个上面全是4,所以下面必须用this不能用lis[i],因为i已经循环完了; this.children[1].style.display = 'block'; //ul是li里的第二个孩子;this可以表示“当前这一个” } lis[i].onmouseout = function() { this.children[1].style.display = 'none'; } } </script> </body> </html>
兄弟节点
-
node.nextSibling
-
nextSibling
返回当前元素的下一个兄弟元素节点,找不到则返回null -
同样,也是包含所有的节点
-
-
node.previousSibling
-
previousSibling
返回当前元素上一个兄弟元素节点,找不到则返回null -
同样,也是包含所有的节点
-
-
node.nextElementSibling
(兼容性)-
nextElementSibling
返回当前元素下一个兄弟元素节点,找不到则返回null -
有兼容性问题,IE9才支持
-
-
node.previousElementSibling
(兼容性)-
previousElementSibling
返回当前元素上一个兄弟元素节点,找不到则返回null -
有兼容性问题,IE9才支持
-
<body> <div>我是div</div> <span>我是span</span> <script> var div = document.querySelector('div'); // 1.nextSibling 下一个兄弟节点 包含元素节点或者 文本节点等等 console.log(div.nextSibling); // #text console.log(div.previousSibling); // #text // 2. nextElementSibling 得到下一个兄弟元素节点 console.log(div.nextElementSibling); //<span>我是span</span> console.log(div.previousElementSibling);//null </script> </body>
-
解决兼容性问题
function getNextElementSibling(element) { var el = element; while (el = el.nextSibling) { if (el.nodeType === 1) { return el; } } return null; }
创建节点
document.createElement('tagName');
-
document.createElement()
方法创建由 tagName 指定的HTML 元素 -
因为这些元素原先不存在,是根据我们的需求动态生成的,所以我们也称为动态创建元素节点
添加节点
-
node.appendChild(child);
-
node.appendChild()
方法将一个节点添加到指定父节点的子节点列表末尾。类似于 CSS 里面的 after 伪元素和数组里的push()
-
-
node.insertBefore(child,指定元素)
-
node.insertBefore()
方法将一个节点添加到父节点的指定子节点前面。类似于 CSS 里面的 before 伪元素
-
<body> <ul> <li>123</li> </ul> <script> // 1. 创建节点元素节点 var li = document.createElement('li'); // 2. 添加节点 node.appendChild(child) node 父级 child 是子级 后面追加元素 // 先获取父亲ul var ul = document.querySelector('ul'); ul.appendChild(li); // 3. 添加节点 node.insertBefore(child, 指定元素); var lili = document.createElement('li'); ul.insertBefore(lili, ul.children[0]); // 在父元素最前面添加元素 // 4. 页面添加一个新的元素分两步: 1. 创建元素 2. 添加元素 </script> </body>
删除节点
node.removeChild(child)
-
node.removeChild()
方法从 DOM 中删除一个子节点,返回删除的节点
留言板:可新增可删除
阻止链接跳转需要在a的herf里写 javascript:void(0);
或者javascript:;
<!DOCTYPE html> <html lang="zh-CN"> <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; } body { padding: 100px; } textarea { width: 200px; height: 100px; border: 1px solid pink; outline: none; resize: none; } ul { margin-top: 50px; } li { width: 300px; padding: 5px; background-color: rgb(245, 209, 243); color: red; font-size: 14px; margin: 15px 0; } li a { float: right; } </style> </head> <body> <textarea name="" id=""></textarea> <button>发布</button> <ul> </ul> <script> // 1. 获取元素 var btn = document.querySelector('button'); var text = document.querySelector('textarea'); var ul = document.querySelector('ul'); // 2. 注册事件 btn.onclick = function() { if (text.value == '') { alert('您没有输入内容'); return false; } else { // console.log(text.value); // (1) 创建元素 var li = document.createElement('li'); // 先有li 才能赋值 li.innerHTML = text.value + "<a href='javascript:;'>删除</a>"; // (2) 添加元素 // ul.appendChild(li); ul.insertBefore(li, ul.children[0]); // 最新的留言放在最上面 // (3) 删除元素 删除的是当前链接的li的父亲;这一部分必须写在btn.onclick里面,否则未点击之前还没有a,设置a的功能也会无效;保证点击之后才会有a,对a进行设置 var as = document.querySelectorAll('a'); for (var i = 0; i < as.length; i++) { as[i].onclick = function() { // node.removeChild(child); 删除的是li当前a所在的li this.parentNode; ul.removeChild(this.parentNode); } } } text.value = ''; // 发布后清除文本框里的内容 } </script> </body> </html>
动态生成表格
① 因为里面的学生数据都是动态的,我们需要js 动态生成。 这里我们模拟数据,自己定义好 数据。 数据我们采取对象形式存储。 ② 所有的数据都是放到tbody里面的行里面。 ③ 因为行很多,我们需要循环创建多个行(对应多少人) ④ 每个行里面又有很多单元格(对应里面的数据),我们还继续使用循环创建多个单元格, 并且把数据存入里面(双重for循环) ⑤ 最后一列单元格是删除,需要单独创建单元格。 ⑥ 最后添加删除操作,单击删除,可以删除当前行。
步骤
-
根据对象的个数创建tr
-
根据每一个对象里属性的个数创建td
-
把每一个对象里的属性值给td
-
创建带有删除a链接的td
-
删除a链接的删除操作
<!DOCTYPE html> <html lang="zh-CN"> <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> table { width: 500px; margin: 100px auto; border-collapse: collapse; text-align: center; } td, th { border: 1px solid #333; } thead tr { height: 40px; background-color: #ccc; } </style> </head> <body> <table cellspacing="0"> <thead> <tr> <th>姓名</th> <th>科目</th> <th>成绩</th> <th>操作</th> </tr> </thead> <tbody> </tbody> </table> <script> // 1.先去准备好学生的数据(实际开发是数据是后台给的) //这里datas是个数组,这样就可以存很多对象 var datas = [{ name: '魏璎珞', subject: 'JavaScript', score: 100 }, { name: '弘历', subject: 'JavaScript', score: 98 }, { name: '傅恒', subject: 'JavaScript', score: 99 }, { name: '明玉', subject: 'JavaScript', score: 88 }, { name: '大猪蹄子', subject: 'JavaScript', score: 0 }]; // 2. 往tbody 里面创建行: 有几个人(通过数组的长度)我们就创建几行 var tbody = document.querySelector('tbody'); for (var i = 0; i < datas.length; i++) { // 外面的for循环管行 tr // 1. 创建 tr行 var tr = document.createElement('tr'); tbody.appendChild(tr); // 2. 行里面创建单元格(跟数据有关系的3个单元格) td 单元格的数量取决于每个对象里面的属性个数 for循环遍历对象 datas[i] for (var k in datas[i]) { // 里面的for循环管列 td;每一个对象里面有几个属性 // 创建单元格 var td = document.createElement('td'); // 把对象里面的属性值 datas[i][k] 给 td // console.log(datas[i][k]); td.innerHTML = datas[i][k]; tr.appendChild(td); } // 3. 创建删除的单元格 var td = document.createElement('td'); td.innerHTML = '<a href="javascript:;">删除</a>'; tr.appendChild(td); } // 4. 删除操作 开始 var as = document.querySelectorAll('a'); for (var i = 0; i < as.length; i++) { as[i].onclick = function() { // 点击a 删除 当前a 所在的行(链接的爸爸的爸爸) node.removeChild(child) tbody.removeChild(this.parentNode.parentNode) } } // for(var k in obj) { // k 得到的是属性名 // obj[k] 得到是属性值 // } </script> </body> </html>
复制节点(克隆节点)
node.cloneNode()
-
node.cloneNode()
方法返回调用该方法的节点的一个副本。 也称为克隆节点/拷贝节点 -
如果括号参数为空或者为 false ,则是浅拷贝,即只克隆复制节点本身,不克隆里面的子节点
-
如果括号参数为 true ,则是深度拷贝,会复制节点本身以及里面所有的子节点
<body> <ul> <li>1111</li> <li>2</li> <li>3</li> </ul> <script> var ul = document.querySelector('ul'); // 1. node.cloneNode(); 括号为空或者里面是false 浅拷贝 只复制标签不复制里面的内容 // 2. node.cloneNode(true); 括号为true 深拷贝 复制标签复制里面的内容 var lili = ul.children[0].cloneNode(true); ul.appendChild(lili); </script> </body>
三种动态创建元素区别
-
document.write()
是直接将内容写入页面的内容流,但是文档流执行完毕,则它会导致页面全部重绘(重新绘制了一个新的html页面,原来的全没了) -
innerHTML
是将内容写入某个 DOM 节点,不会导致页面全部重绘-
innerHTML
创建多个元素效率更高(不要拼接字符串,采取数组形式拼接),结构稍微复杂
-
-
createElement()
创建多个元素效率稍低一点点,但是结构更清晰
总结:不同浏览器下, innerHTML 效率要比 createElement 高(在不用字符串的情况下)
<body> <div class="inner"></div> <div class="create"></div> <script> var inner = document.querySelector('.inner'); // 2.1 innerHTML 用拼接字符串方法(慢) for (var i = 0; i <= 100; i++) { inner.innerHTML += '<a href="#">百度</a>'; } // 2.2 innerHTML 用数组形式拼接(快) var arr = []; for (var i = 0; i <= 100; i++) { arr.push('<a href="#">百度</a>'); } inner.innerHTML = arr.join(''); // 3.document.createElement() 创建元素(结构清晰) var create = document.querySelector('.create'); var a = document.createElement('a'); create.appendChild(a); </script> </body>
DOM核心
-
创建
-
document.write
-
innerHTML
-
createElement
-
-
增
-
appendChild
-
insertBefore
-
-
删
-
removeChild
-
-
改:主要修改dom的元素属性,dom元素的内容、属性、表单的值等
-
修改元素属性:src、href、title 等
-
修改普通元素内容:innerHTML、innerText
-
修改表单元素:value、type、disabled
-
修改元素样式:style、className
-
-
查:主要查询dom元素
-
DOM提供的API方法:getElementById、getElementsByTagName (古老用法,不推荐)
-
H5提供的新方法:querySelector、querySelectorAll (提倡)
-
利用节点操作获取元素:父(parentNode)、子(children)、兄(previousElementSibling、nextElementSibling)
-
-
属性操作:主要针对于自定义属性
-
setAttribute:设置dom的属性值
-
getAttribute:得到dom的属性值
-
removeAttribute:移除属性
-
-
事件操作:给元素注册事件, 采取 事件源.事件类型 = 事件处理程序
事件高级
注册事件(绑定事件)
给元素添加事件,称为注册事件或者绑定事件。
注册事件有两种方式:传统方式和方法监听注册方式
传统注册方式 | 方法监听注册方式 |
---|---|
利用 on 开头的事件 onclick | w3c 标准推荐方式 |
<button onclick = "alert("hi")"></button> | addEventListener() 它是一个方法 |
btn.onclick = function() {} | IE9 之前的 IE 不支持此方法,可使用 attachEvent() 代替 |
特点:注册事件的唯一性 | 特点:同一个元素同一个事件可以注册多个监听器 |
同一个元素同一个事件只能设置一个处理函数,最后注册的处理函数将会覆盖前面注册的处理函数 | 按注册顺序依次执行 |
addEventListener 事件监听方式
eventTarget.addEventListener(type,listener[,useCapture])
-
eventTarget.addEventListener()
方法将指定的监听器注册到 eventTarget(目标对象)上 -
当该对象触发指定的事件时,就会执行事件处理函数
-
该方法接收三个参数:
-
type
:事件类型字符串,比如click,mouseover,注意这里不要带on -
listener
:事件处理函数,事件发生时,会调用该监听函数 -
useCapture
:可选参数,是一个布尔值,默认是 false。
-
attachEvent 事件监听方式(兼容)
eventTarget.attachEvent(eventNameWithOn,callback)
-
eventTarget.attachEvent()
方法将指定的监听器注册到 eventTarget(目标对象) 上 -
当该对象触发指定的事件时,指定的回调函数就会被执行
-
该方法接收两个参数:
-
eventNameWithOn
:事件类型字符串,比如 onclick 、onmouseover ,这里要带 on -
callback
: 事件处理函数,当目标触发事件时回调函数被调用 -
IE8及以前的版本支持
-
<body> <button>传统注册事件</button> <button>方法监听注册事件</button> <button>ie9 attachEvent</button> <script> var btns = document.querySelectorAll('button'); // 1. 传统方式注册事件 btns[0].onclick = function() { alert('hi'); } btns[0].onclick = function() { alert('hao a u'); } // 2. 事件监听注册事件 addEventListener // (1) 里面的事件类型是字符串 所以加引号 而且不带on // (2) 同一个元素 同一个事件可以添加多个侦听器(事件处理程序) btns[1].addEventListener('click', function() { alert(22); }) btns[1].addEventListener('click', function() { alert(33); }) // 3. attachEvent ie9以前的版本支持 btns[2].attachEvent('onclick', function() { alert(11); }) </script> </body>
注册事件兼容性解决方案
兼容性处理的原则:首先照顾大多数浏览器,再处理特殊浏览器
function addEventListener(element, eventName, fn) { // 判断当前浏览器是否支持 addEventListener 方法 if (element.addEventListener) { element.addEventListener(eventName, fn); // 第三个参数 默认是false } else if (element.attachEvent) { element.attachEvent('on' + eventName, fn); } else { // 相当于 element.onclick = fn; element['on' + eventName] = fn; }
删除事件(解绑事件)
传统事件删除方式
eventTarget.onclick = null;
removeEventListener 删除事件方式
eventTarget.removeEventListener(type,listener[,useCapture]);
该方法接收三个参数:
-
type
:事件类型字符串,比如click,mouseover,注意这里不要带on -
listener
:事件处理函数,事件发生时,会调用该监听函数 -
useCapture
:可选参数,是一个布尔值,默认是 false。学完 DOM 事件流后,我们再进一步学习
detachEvent 删除事件方式(兼容)
eventTarget.detachEvent(eventNameWithOn,callback);
该方法接收两个参数:
-
eventNameWithOn
:事件类型字符串,比如 onclick 、onmouseover ,这里要带 on -
callback
: 事件处理函数,当目标触发事件时回调函数被调用 -
ie9以前的版本支持
<body> <div>1</div> <div>2</div> <div>3</div> <script> var divs = document.querySelectorAll('div'); divs[0].onclick = function() { alert(11); // 1. 传统方式删除事件 divs[0].onclick = null; } // 2.removeEventListener 删除事件 divs[1].addEventListener('click',fn); //里面的fn不需要调用加小括号 function fn(){ alert(22); divs[1].removeEventListener('click',fn); } // 3.IE9 中的删除事件方式 divs[2].attachEvent('onclick',fn1); function fn1() { alert(33); divs[2].detachEvent('onclick',fn1); } </script> </body>
删除事件兼容性解决方案
function removeEventListener(element, eventName, fn) { // 判断当前浏览器是否支持 removeEventListener 方法 if (element.removeEventListener) { element.removeEventListener(eventName, fn); // 第三个参数 默认是false } else if (element.detachEvent) { element.detachEvent('on' + eventName, fn); } else { element['on' + eventName] = null; }
DOM 事件流 - 捕获和冒泡
事件发生时会在元素节点之间按照特定的顺序传播,这个传播过程即 DOM 事件流。
-
事件流描述的是从页面中接收事件的顺序
-
事件发生时会在元素节点之间按照特定的顺序传播,这个传播过程即DOM事件流
-
事件冒泡: IE 最早提出,事件开始时由最具体的元素接收,然后逐级向上传播到到 DOM 最顶层节点的过程。
-
只点了son,但是father和document的事件也都发生了(son→father→body→html→document)
-
-
事件捕获: 网景最早提出,由 DOM 最顶层节点开始,然后逐级向下传播到到最具体的元素接收的过程。
-
注意
-
JS 代码中只能执行捕获或者冒泡其中的一个阶段。
-
onclick 和 attachEvent 只能得到冒泡阶段。
-
addEventListener(type, listener[, useCapture])第三个参数如果是 true,表示在事件捕 获阶段调用事件处理程序;如果是 false(不写默认就是false),表示在事件冒泡阶段调用事件处理 程序。
-
实际开发中我们很少使用事件捕获,我们更关注事件冒泡。
-
有些事件是没有冒泡的,比如 onblur、onfocus、onmouseenter、onmouseleave
-
事件冒泡有时候会带来麻烦,有时候又会帮助很巧妙的做某些事件
-
捕获阶段
捕获阶段由 DOM 最顶层节点开始,然后逐级向下传播到到最具体的元素接收
document→html→body→father→son
-
document先得到点击,有没有监听器?没有的话html得到点击,有没有监听器?没有的话body再得到点击,以此类推;如果过程中某个有监听器,就先执行它的再执行后面的
-
如两个盒子嵌套,一个父盒子一个子盒子,我们的需求是当点击父盒子时弹出 father ,当点击子盒子时弹出 son
-
但是因为DOM流的影响,我们点击子盒子,会先弹出 father,之后再弹出 son
-
<body> <div class="father"> <div class="son">son盒子</div> </div> <script> // dom 事件流 三个阶段 // 1. JS 代码中只能执行捕获或者冒泡其中的一个阶段。 // 2. onclick 和 attachEvent(ie) 只能得到冒泡阶段。 // 3. 捕获阶段 如果addEventListener 第三个参数是 true 那么则处于捕获阶段 document -> html -> body -> father -> son var son = document.querySelector('.son'); son.addEventListener('click', function() { alert('son'); }, true); var father = document.querySelector('.father'); father.addEventListener('click', function() { alert('father'); }, true); </script> </body>
冒泡阶段
冒泡阶段开始时由最具体的元素接收,然后逐级向上传播到到 DOM 最顶层节点
son -> father ->body -> html -> document
-
点击子盒子,会弹出 son、father、document
<body> <div class="father"> <div class="son">son盒子</div> </div> <script> // 4. 冒泡阶段 如果addEventListener 第三个参数是 false 或者 省略 那么则处于冒泡阶段 son -> father ->body -> html -> document var son = document.querySelector('.son'); son.addEventListener('click', function() { alert('son'); }, false); var father = document.querySelector('.father'); father.addEventListener('click', function() { alert('father'); }, false); document.addEventListener('click', function() { alert('document'); }) </script> </body>
事件对象
官方解释:event 对象代表事件的状态,比如键盘按键的状态、鼠标的位置、鼠标按钮的状态
-
简单理解:事件发生后,跟事件相关的一系列信息数据的集合都放到这个对象里面
-
这个对象就是事件对象 event,它有很多属性和方法,比如“
-
谁绑定了这个事件
-
鼠标触发事件的话,会得到鼠标的相关信息,如鼠标位置
-
键盘触发事件的话,会得到键盘的相关信息,如按了哪个键
-
-
这个 event 是个形参,系统帮我们设定为事件对象,不需要传递实参过去
-
当注册事件时, event 对象就会被系统自动创建,并依次传递给事件监听器(事件处理函数)
eventTarget.onclick = function(event) { // 这个 event 就是事件对象,我们还喜欢的写成 e 或者 evt } eventTarget.addEventListener('click', function(event) { // 这个 event 就是事件对象,我们还喜欢的写成 e 或者 evt })
事件对象的兼容性方案
事件对象本身的获取存在兼容问题:
-
标准浏览器中是浏览器给方法传递的参数,只需要定义形参 e 就可以获取到。
-
在 IE6~8 中,浏览器不会给方法传递参数,如果需要的话,需要到 window.event 中获取查找
e = e || window.event;
事件对象的常见属性和方法
事件对象属性方法 | 说明 |
---|---|
e.target | 返回触发事件的对象 标准 |
e.srcElement | 返回触发事件的对象 非标准 ie6-8使用 |
e.type | 返回事件的类型 比如click mouseover 不带on |
e.cancelBubble | 该属性阻止冒泡,非标准,ie6-8使用 |
e.returnValue | 该属性阻止默认行为 非标准,ie6-8使用 |
e.preventDefault() | 该方法阻止默认行为 标准 比如不让链接跳转或让提交按钮不提交 |
e.stopPropagation() | 阻止冒泡 标准 |
<body> <div>123</div> <a href="http://www.baidu.com">百度</a> <form action="http://www.baidu.com"> <input type="submit" value="提交" name="sub"> </form> <script> // 常见事件对象的属性和方法 // 1. 返回事件类型 var div = document.querySelector('div'); div.addEventListener('click', fn); div.addEventListener('mouseover', fn); div.addEventListener('mouseout', fn); function fn(e) { console.log(e.type); } // 2. 阻止默认行为(事件) 让链接不跳转 或者让提交按钮不提交 var a = document.querySelector('a'); a.addEventListener('click', function(e) { e.preventDefault(); // dom 标准写法 }) // 3. 传统的注册方式 a.onclick = function(e) { // 普通浏览器 e.preventDefault(); 方法 // e.preventDefault(); // 低版本浏览器 ie678 returnValue 属性 // e.returnValue; // 我们可以利用return false 也能阻止默认行为 没有兼容性问题 特点: return 后面的代码不执行了, 而且只限于传统的注册方式 return false; alert(11); } </script> </body>
-
兼容性处理:
div.onclik = function(e) { e = e || window.event; var target = e.target || e.srcElement; }
-
e.target 和 this 的区别:
-
this 是事件绑定的元素, 这个函数的调用者(绑定这个事件的元素)(哪个元素绑定了这个点击事件就返回谁)
-
e.target 是事件触发的元素(点击了那个元素就返回哪个元素)
-
如给ul绑定点击事件,但是点击的是li(冒泡),此时this是ul,e.target是li
-
e.currentTarget和this相似,ie678不认识阻止事件冒泡
-
<body> <div>123</div> <ul> <li>abc</li> <li>abc</li> <li>abc</li> </ul> <script> // 常见事件对象的属性和方法 // 1. e.target 返回的是触发事件的对象(元素) this 返回的是绑定事件的对象(元素) // 区别 : e.target 点击了那个元素,就返回那个元素 this 那个元素绑定了这个点击事件,那么就返回谁 var div = document.querySelector('div'); div.addEventListener('click', function(e) { console.log(e.target); // <div>123</div> console.log(this); // <div>123</div> }) var ul = document.querySelector('ul'); ul.addEventListener('click', function(e) { // 我们给ul 绑定了事件 那么this 就指向ul console.log(this); // <ul></ul> console.log(e.currentTarget); // <ul></ul> // e.target 指向我们点击的那个对象 谁触发了这个事件 我们点击的是li e.target 指向的就是li console.log(e.target); // <li></li> }) // 了解兼容性 // div.onclick = function(e) { // e = e || window.event; // var target = e.target || e.srcElement; // console.log(target); // } // 2. 了解 跟 this 有个非常相似的属性 currentTarget ie678不认识 </script> </body>
阻止事件冒泡
事件冒泡:开始时由最具体的元素接收,然后逐级向上传播到到 DOM 最顶层节点
-
标准写法
e.stopPropagation();
-
非标准写法: IE6-8 利用对象事件 cancelBubble属性
e.cancelBubble = true;
<body> <div class="father"> <div class="son">son儿子</div> </div> <script> // 常见事件对象的属性和方法 // 阻止冒泡 dom 推荐的标准 stopPropagation() var son = document.querySelector('.son'); son.addEventListener('click', function(e) { alert('son'); e.stopPropagation(); // stop 停止 Propagation 传播 e.cancelBubble = true; // 非标准 cancel 取消 bubble 泡泡 }, false); var father = document.querySelector('.father'); father.addEventListener('click', function() { alert('father'); // 被阻止冒泡了,没有反应 }, false); document.addEventListener('click', function() { alert('document'); // 被阻止冒泡了,没有反应 }) </script> </body>
-
兼容性解决方案
if(e && e.stopPropagation){ e.stopPropagation(); }else{ window.event.cancelBubble = true; }
事件委托
-
事件委托也称为事件代理,在 jQuery 里面称为事件委派
-
事件委托的原理
-
不是每个子节点单独设置事件监听器,而是事件监听器设置在其父节点上,然后利用冒泡原理影响设置每个子节点
-
如给 ul 注册点击事件,然后利用事件对象的 target 来找到当前点击的 li,因为点击 li,事件会冒泡到 ul 上,ul 有注册事件,就会触发事件监听器。
-
-
事件委托的作用:只操作了一次 DOM ,提高了程序的性能。
<body> <ul> <li>知否知否,点我应有弹框在手!</li> <li>知否知否,点我应有弹框在手!</li> <li>知否知否,点我应有弹框在手!</li> <li>知否知否,点我应有弹框在手!</li> <li>知否知否,点我应有弹框在手!</li> </ul> <script> // 事件委托的核心原理:给父节点添加侦听器, 利用事件冒泡影响每一个子节点 var ul = document.querySelector('ul'); ul.addEventListener('click', function(e) { // alert('知否知否,点我应有弹框在手!'); // e.target 这个可以得到我们点击的对象 e.target.style.backgroundColor = 'pink'; // 点了谁,就让谁的style里面的backgroundColor颜色变为pink }) </script> </body>
常见鼠标事件
鼠标事件 | 触发条件 |
---|---|
onclick | 鼠标点击左键触发 |
onmouseover | 鼠标经过触发 |
onmouseout | 鼠标离开触发 |
onfocus | 获得鼠标焦点触发 |
onblur | 失去鼠标焦点触发 |
onmousemove | 鼠标移动触发 |
onmouseup | 鼠标弹起触发 |
onmousedown | 鼠标按下触发 |
-
禁止鼠标右键菜单
-
contextmenu
主要控制应该何时显示上下文菜单,主要用于程序员取消默认的上下文菜单
-
-
禁止鼠标选中
-
selectstart
开始选中
-
<body> <h1>我是一段不愿意分享的文字</h1> <script> // 1. contextmenu 我们可以禁用右键菜单 document.addEventListener('contextmenu', function(e) { e.preventDefault(); // 阻止默认行为 }) // 2. 禁止选中文字 selectstart document.addEventListener('selectstart', function(e) { e.preventDefault(); }) </script> </body>
鼠标事件对象
-
event对象代表事件的状态,跟事件相关的一系列信息的集合
-
现阶段我们主要是用鼠标事件对象 MouseEvent 和键盘事件对象 KeyboardEvent。
鼠标事件对象 | 说明 |
---|---|
e.clientX | 返回鼠标相对于浏览器窗口可视区的X坐标(滚动条无影响) |
e.clientY | 返回鼠标相对于浏览器窗口可视区的Y坐标(滚动条无影响) |
e.pageX(重点) | 返回鼠标相对于文档页面的X坐标 IE9+ 支持 |
e.pageY(重点) | 返回鼠标相对于文档页面的Y坐标 IE9+ 支持 |
e.screenX | 返回鼠标相对于电脑屏幕的X坐标 |
e.screenY | 返回鼠标相对于电脑屏幕的Y坐标 |
<body> <script> // 鼠标事件对象 MouseEvent document.addEventListener('click', function(e) { // 1. client 鼠标在可视区的x和y坐标 console.log(e.clientX); console.log(e.clientY); console.log('---------------------'); // 2. page 鼠标在页面文档的x和y坐标 console.log(e.pageX); console.log(e.pageY); console.log('---------------------'); // 3. screen 鼠标在电脑屏幕的x和y坐标 console.log(e.screenX); console.log(e.screenY); }) </script> </body>
跟随鼠标的图片
<!DOCTYPE html> <html lang="zh-CN"> <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> img { position: absolute; top: 2px; } </style> </head> <body> <img src="./images.gif" alt=""> <script> var pic = document.querySelector('img'); document.addEventListener('mousemove', function(e) { //只要鼠标移动1px就会触发mousemove事件 var x = e.pageX; var y = e.pageY; // 核心原理:每次鼠标移动,都会重新获得最新的鼠标坐标,把这个x和y坐标作为图片绝对定位的top和left值,就可以移动图片 pic.style.left = x + -50 + 'px'; //50是图片的宽(为了把图片放在鼠标中央) pic.style.top = x - 40 + 'px'; }) </script> </body> </html>
常用键盘事件
键盘事件 | 触发条件 |
---|---|
onkeyup | 某个键盘按键被松开时触发 |
onkeydown | 某个键盘按键被按下时触发 |
onkeypress | 某个键盘按键被按下时触发,但是它不识别功能键,比如 ctrl shift 箭头等 |
-
如果使用addEventListener 不需要加 on
-
onkeypress
和前面2个的区别是,它不识别功能键,比如左右箭头,shift 等 -
三个事件的执行顺序是: keydown – keypress — keyup
<body> <script> //1. keyup 按键弹起的时候触发 // document.onkeyup = function() { // console.log('我弹起了'); // } document.addEventListener('keyup', function() { console.log('我弹起了'); }) //3. keypress 按键按下的时候触发 不能识别功能键 比如 ctrl shift 左右箭头 document.addEventListener('keypress', function() { console.log('我按下了press'); }) //2. keydown 按键按下的时候触发 能识别功能键 比如 ctrl shift 左右箭头 document.addEventListener('keydown', function() { console.log('我按下了down'); }) // 4. 三个事件的执行顺序 keydown -- keypress -- keyup </script> </body>
键盘事件对象
键盘事件对象 属性 | 说明 |
---|---|
keyCode | 返回该键值的ASCII值 |
-
onkeydown
和onkeyup
不区分字母大小写,onkeypress
区分字母大小写。 -
在我们实际开发中,我们更多的使用keydown和keyup, 它能识别所有的键(包括功能键)
-
Keypress
不识别功能键,但是keyCode
属性能区分大小写,返回不同的ASCII值
<body> <script> // 键盘事件对象中的keyCode属性可以得到相应键的ASCII码值 // 1. 我们的keyup 和keydown事件不区分字母大小写 a 和 A 得到的都是65 // 2. 我们的keypress 事件 区分字母大小写 a 97 和 A 得到的是65 document.addEventListener('keyup', function(e) { console.log('up:' + e.keyCode); // a:65 // 我们可以利用keycode返回的ASCII码值来判断用户按下了那个键 if (e.keyCode === 65) { alert('您按下的a键'); } else { alert('您没有按下a键') } }) document.addEventListener('keypress', function(e) { console.log('press:' + e.keyCode); // a:97 }) </script> </body>
按键输入内容
① 核心思路: 检测用户是否按下了s 键,如果按下s 键,就把光标定位到搜索框里面 ② 使用键盘事件对象里面的keyCode 判断用户按下的是否是s键 ③ 搜索框获得焦点: 使用 js 里面的 focus() 方法
var search = document.querySelector('input'); document.addEventListener('keyup', function(e) { // console.log(e.keyCode); if (e.keyCode === 83) { search.focus(); } })
快递单号查询
① 快递单号输入内容时, 上面的大号字体盒子(con)显示(这里面的文字 ② 同时把快递单号里面的值(value)获取过来赋值给 con盒子(innerText)做为内容 ③ 如果快递单号里面内容为空,则隐藏大号字体盒子(con)盒子 ④ 注意: keydown 和 keypress 在文本框里面的特点: 他们两个事件触发的时候,文字还 没有落入文本框中。 ⑤ keyup事件触发的时候, 文字已经落入文本框里面了 ⑥ 当我们失去焦点,就隐藏这个con盒子 ⑦ 当我们获得焦点,并且文本框内容不为空,就显示这个con盒子
<!DOCTYPE html> <html lang="zh-CN"> <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; } .search { position: relative; width: 178px; margin: 100px; } .con { display: none; position: absolute; top: -40px; width: 171px; border: 1px solid rgba(0, 0, 0, .2); box-shadow: 0 2px 4px rgba(0, 0, 0, .2); padding: 5px 0; font-size: 18px; line-height: 20px; color: #333; } .con::before { content: ''; width: 0; height: 0; position: absolute; top: 28px; left: 18px; border: 8px solid #000; border-style: solid dashed dashed; border-color: #fff transparent transparent; } </style> </head> <body> <div class="search"> <div class="con">123</div> <input type="text" placeholder="请输入您的快递单号" class="jd"> </div> <script> // 快递单号输入内容时, 上面的大号字体盒子(con)显示(这里面的字号更大) // 表单检测用户输入: 给表单添加键盘事件 // 同时把快递单号里面的值(value)获取过来赋值给 con盒子(innerText)做为内容 // 如果快递单号里面内容为空,则隐藏大号字体盒子(con)盒子 var con = document.querySelector('.con'); var jd_input = document.querySelector('.jd'); jd_input.addEventListener('keyup', function() { // console.log('输入内容啦'); if (this.value == '') { con.style.display = 'none'; } else { con.style.display = 'block'; con.innerText = this.value; } }) // 当我们失去焦点,就隐藏这个con盒子 jd_input.addEventListener('blur', function() { con.style.display = 'none'; }) // 当我们获得焦点,就显示这个con盒子 jd_input.addEventListener('focus', function() { if (this.value !== '') { con.style.display = 'block'; } }) </script> </body> </html>