2.4JS基础笔记

这篇博客详细介绍了JavaScript中的DOM基本概念,包括DOM查询、元素操作、事件处理等。内容涵盖DOM简介、事件的绑定与传播、文档加载、DOM增删改节点、操作内联样式以及事件对象的相关知识点,并通过实例进行讲解。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


一、DOM简介

DOM,全称Document Object Model 文件对象模型
JS中通过DOM来对HTML文档进行操作

  • 文档
    表示的就是整个HTML网页文档
  • 对象
    表示将网页的每一个部分都转换为了一个对象
  • 模型
    使用模型来表示对象之间的关系,方便我们获取对象

节点:Node——构成HTML文档最基本的单元
常用节点分为四类

  • 文档节点:整个HTML文档
  • 元素节点:HTML文档中的HTML标签
  • 属性节点:元素的属性
  • 文本节点:HTML标签中的文本内容

节点的属性:

nodeNamenodeTypenodeValue
文档节点#document9null
元素节点标签名1null
属性节点属性名2属性值
文本节点#text3文本内容
<body>
    <button id="btn">我是一个按钮</button>
    <script> 
        /*
        * 浏览器已经为我们提供 文档节点 对象,这个对象是window属性
        * 可以在页面中直接使用,文档节点代表的是整个网页
        */
        console.log(document);
        // 获取到button对象
        var btn = document.getElementById("btn");
        btn.innerHTML = "I'm button.";
    </script>
</body>

在这里插入图片描述

二、事件的简介

事件,就是用户和浏览器之间的交互行为
比如:点击按钮、鼠标移动、关闭窗口……

我们可以在时间对应的属性中设置一些JS代码,这样当事件被触发时,这些代码将会执行

<button id="btn" onclick="alert('干什么!!')">我是一个按钮</button>

这种写法称为结构和行为耦合,不方便维护,不推荐使用


可以为按钮的对应事件绑定处理函数的形式来响应事件
这样当时间被触动时,其对应的函数将会被调用

<body>
    <button id="btn">我是一个按钮</button>
    <script> 
       var btn = document.getElementById("btn");
    //    绑定一个单击事件
    //    像这种为单击事件绑定的函数,我们称为单击响应函数
    btn.onclick= function(){
        alert("干嘛呀!!!");
    }
    </script>
</body>

三、文档的加载

浏览器在加载一个页面时,是按照自上向下的顺序加载的
读取到一行就运行一行,如果将JS标签写在页面的上边,在代码执行时,页面还没有加载,DOM对象也没有加载,会导致无法获取到DOM对象
所以将JS代码边界在页面下不就是为了,在页面加载完毕后再执行JS代码

如果想把JS代码写在head标签中
可以为window绑定一个onload事件
onload事件会在整个页面加载完成后才触发,这样可以确保我们的JS代码执行时所有的DOM对象已经加载完毕了

<!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>
    <script> 
        window.onload = function(){
            var btn = document.getElementById("btn");
            btn.onclick= function(){
                alert("干嘛呀!!!");
        };
    };
    </script>   
</head>
<body>
    <button id="btn">我是一个按钮</button>   
</body>
</html>

四、DOM查询

1.获取元素节点:通过document对象调用

  1. getElementById()
    通过id属性获取一个元素节点对象
  2. getElementsByTagName()
    通过标签名获取一组元素节点对象
  3. getElementsByName()
    通过name属性获取一组元素节点对象
    <script type="text/javascript">
        window.onload = function(){
            /* 
             * 查找#bj节点
             *      为id为btn01的按钮绑定一个单击响应函数
             */
            var  btn01 = document.getElementById("btn01");
            btn01.onclick = function(){
                var bj = document.getElementById("bj");
                alert(bj.innerHTML);
            }


            /* 
             * 查找所有li节点
             *      为id为btn02的按钮绑定一个单击响应函数
             */
            var btn02 = document.getElementById("btn02");
            btn02.onclick = function(){
                // getElementsTagName()可以根据标签名来获取一组元素节点对象
                // 这个方法会给我们返回一个类数组对象,所有查询到的元素都会封装到对象中
                var lis = document.getElementsByTagName("li");
                // alert(lis.length);
                for(var i=0;i<lis.length;i++){
                    alert(lis[i].innerHTML);
                }
            }


            /* 
             * 查找name=gender的所有节点
             *      为id为btn03的按钮绑定一个单击响应函数
             */
            var btn03 = document.getElementById("btn03");
            btn03.onclick = function(){
                var inputs = document.getElementsByName("gender");
                alert(inputs.length);
                /*
                *   innerHTML用于获取元素内部的HTML代码
                *   input属于自结束标签,内部没有代码
                *       如果想要读取元素节点属性,直接使用:元素.属性名
                *           例子:元素.id、元素.name、元素.value......
                *               读取class属性时需要使用 :元素.className
                */
               for(var i=0;i<inputs.length;i++){
                   alert(inputs[i].value);
               }
            }
        }
    </script>

练习——轮播图

<!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 type="text/css">
        *{
            margin: 0;
            padding: 0;
        }
        #outer{
            width: 200px;
            margin: 50px auto;
            padding: 10px;
            background-color: cornflowerblue;
            text-align: center;

        }
    </style>
    <script type="text/javascript">
        window.onload = function(){
            /*
            *   点击按钮切换图片
            */
        //    获取两个按钮
            var prev = document.getElementById("prev");

            // 获取img标签(这个方法获取到的是数组)
            var img = document.getElementsByTagName("img")[0];

            // 创建一个数组用来保存图片的路径
            var imgArr = ["./img/1.jpg","./img/2.jpg","./img/3.jpg","./img/4.jpg","./img/5.jpg"];

            // 创建一个变量,用来保存当前正在现实的图片的索引
            var index = 0;

            // 设置提示文字,获取id为info的p元素
            var info = document.getElementById("info");
            info.innerHTML = "一共"+imgArr.length+"张图片,当前第"+(index+1)+"张";
            prev.onclick = function(){
                // 要切换图片就是要修改img标签的src属性
                index--;
                if(index<0){
                    index = imgArr.length - 1;
                }
                img.src = imgArr[index];
                info.innerHTML = "一共"+imgArr.length+"张图片,当前第"+(index+1)+"张";
            };
            var next = document.getElementById("next");
            next.onclick = function(){
                index++;
                if(index>imgArr.length - 1){
                    index = 0;
                }
                img.src = imgArr[index];
                info.innerHTML = "一共"+imgArr.length+"张图片,当前第"+(index+1)+"张";
            };
        };
    </script>
</head>
<body>
    <div id="outer">
        <p id="info"></p>
        <img src="./img/1.jpg" alt="女侠">
        <button id="prev">上一张</button>
        <button id="next">下一张</button>
    </div>
</body>
</html>

2.获取元素节点的子节点:通过具体的元素节点调用

  1. getElementsByTagName()
    方法,返回当前节点的指定标签名后代节点
  2. childNodes
    属性,表示当前节点的所有子节点
  3. firstChild
    属性,表示当前节点的第一个子节点
  4. lastChild
    属性,表示当前节点的最后一个子节点
<script type="text/javascript">
window.onload = function(){
            /* 
             * 查找#city下的所有li节点
             *      为id为btn04的按钮绑定一个单击响应函数
             */
            var btn04 = document.getElementById("btn04");
            btn04.onclick = function(){
                var city = document.getElementById("city");
                var lis = city.getElementsByTagName("li");
                for(var i=0;i<lis.length;i++){
                    alert(lis[i].innerHTML);
                }
            };


            /* 
             * 返回#city下的所有子节点
             *      为id为btn05的按钮绑定一个单击响应函数
             */
            var btn05 = document.getElementById("btn05");
            btn05.onclick = function(){
                var city = document.getElementById("city");
                // childNodes 属性会获取包括文本节点在内的所有节点
                //      根据DOM标签标准标签间的空白也会当成文本节点
                //      注:在IE8及以下的浏览器中,不会将空白文本当作子节点
                var cns = city.childNodes;

                for(var i=0;i<cns.length;i++){
                    alert(cns[i].innerHTML);
                }
                
                // children 属性可以获取当前元素的所有子元素
                var cns2 = city.children;
                for(var i=0;i<cns2.length;i++){
                    alert(cns2[i].innerHTML);
                }
            };


            /* 
             * 返回#phone的第一个子节点
             *      为id为btn06的按钮绑定一个单击响应函数
             */
            var btn06 = document.getElementById("btn06");
            btn06.onclick = function(){
                var phone = document.getElementById("phone");
                // firstChild 可以获取到当前元素的第一个子节点(包括空白文本节点)
                var pho = phone.firstChild;
                alert(pho.innerHTML);
                // firstElementChild 可以获取当前元素的第一个子元素(但兼容性不好)
                var pho = phone.firstElementChild;
                alert(pho.innerHTML);
            };
}
</script>

3.获取父节点和兄弟节点:通过具体的节点调用

  • parentNode
    属性,表示当前节点的父节点
  • previousSibling
    属性,表示当前节点的前一个兄弟元素
  • nextSibling
    属性,表示当前节点的后一个兄弟元素
<script type="text/javascript">
window.onload = function(){
                        /* 
             * 返回#bj的父节点
             *      为id为btn07的按钮绑定一个单击响应函数
             */
            myClick("btn07",function(){
                var bj = document.getElementById("bj");
                var pn = bj.parentNode;
                alert(pn.innerHTML);
                // innerText 该属性可以获取到元素内部的文本内容
                //      它会innerHTML类似,不同的是它会自动将html去除
                var pn = bj.parentNode;
                alert(pn.innerText);
            });


            /* 
             * 返回#android的前一个兄弟节点
             *      为id为btn08的按钮绑定一个单击响应函数
             */
            myClick("btn08",function(){
                var and = document.getElementById("android");
                // previousSibling 也可能获取到空白的文本
                var ps = and.previousSibling;
                alert(ps.innerHTML);
                // previousElementSibling 获取前一个兄弟元素,IE8记一下不支持
                var ps = and.previousElementSibling;
                alert(ps.innerHTML);
            });
}
</script>
<script type="text/javascript">
			/* 
             * 定义一个函数,专门用来为指定元素绑定单击响应函数
             *      参数:
             *          idStr  要绑定单击响应函数的对象的id属性值
             *          fun    事件的回调函数,当单击元素时,该函数将会被触发
             */
             function myClick(idStr ,fun){
                var  btn = document.getElementById(idStr);
                btn.onclick = fun;
            }
        window.onload = function(){
            /* 
             * 返回#username的value属性值
             *      为id为btn09的按钮绑定一个单击响应函数
             */
            myClick("btn09",function(){
                var um = document.getElementById("username");
                // 文本框的value属性值,就是文本框中的内容
                alert(um.value);
            });


            /* 
             * 设置#username的value属性值
             *      为id为btn10的按钮绑定一个单击响应函数
             */
             myClick("btn10",function(){
                var um = document.getElementById("username");
                // 文本框的value属性值,就是文本框中的内容
                um.value = "快要结束啦";
            });


            /* 
             * 返回#bj的文本值
             *      为id为btn11的按钮绑定一个单击响应函数
             */
            myClick("btn11",function(){
                var bj = document.getElementById("bj");
                alert(bj.innerHTML);
                alert(bj.innerText);
                // 获取bj中的文本节点
                var fa = bj.firstChild;
                alert(fa.nodeValue);
                alert(bj.firstChild.nodeValue);

            });
}
</script>

全选练习

<!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>全选练习</title>
    <script type="text/javascript">
        window.onload = function(){
            var checkedAllBox = document.getElementById("checkedAllBox");
            var items = document.getElementsByName("items");

            // 1.#checkedAllBtn
            // 为id为checkedAllBtn的按钮绑定一个单击响应函数
            var checkedAllBtn = document.getElementById("checkedAllBtn");
            checkedAllBtn.onclick = function(){
                // 设置四个多选框全部被选中
                var items = document.getElementsByName("items");
                for(var i=0;i<items.length;i++){
                    // 通过checked 属性可以来获取或者设置多选框的选中状态
                    items[i].checked = true;
                } 
                checkedAllBox.checked = true;
            };

            // 2.#checkedNoBtn
            var checkedNoBtn = document.getElementById("checkedNoBtn");
            checkedNoBtn.onclick = function(){
                var items = document.getElementsByName("items");
                for(var i=0;i<items.length;i++){
                    items[i].checked = false;
                } 
                checkedAllBox.checked = false;
            };

            // 3.#checkedRevBtn
            var checkedRevBtn = document.getElementById("checkedRevBtn");
            checkedRevBtn.onclick = function(){
                var items = document.getElementsByName("items");
                checkedAllBox.checked =true;
                for(var i=0;i<items.length;i++){
                    // if(items[i].checked == false){
                    //     items[i].checked = true;
                    // }else{
                    //     items[i].checked = false;
                    // }
                    items[i].checked = !items[i].checked;
                    if(!items[i].checked){
                        checkedAllBox.checked = false;
                    }
                }
            };

            // 4.#sendBtn
            var sendBtn = document.getElementById("sendBtn");
            sendBtn.onclick = function(){
                var items = document.getElementsByName("items");
                for(var i=0;i<items.length;i++){
                    if(items[i].checked){
                        alert(items[i].value);

                    }
                }
            };
            // 5.#checkedAllBox
            // 在事件的响应函数中,响应函数是给谁绑定的this就是谁
            
            checkedAllBox.onclick = function(){
                var items = document.getElementsByName("items");
                for(var i=0;i<items.length;i++){
                    // items[i].checked = checkedAllBox.checked;
                    items[i].checked = this.checked;
                } 
            };
            // 6.items
            /*
            * 如果四个多选框全都选中,则checkedAllBox也应该选中
            * 如果四个多选框没有都选中,则checkedAllBox不应该选中
            */
            // 为四个多选框分别绑定单击响应
            for(var i=0;i<items.length;i++){
                items[i].onclick = function(){
                    checkedAllBox.checked =true;
                    for(var j=0;j<items.length;j++){
                        if(!items[j].checked){
                            checkedAllBox.checked = false;
                            break;
                        }
                    }
                };
            }
        };
    </script>
</head>
<body>
    <form method="post" action="">
        你爱好的运动是?<input type="checkbox" id="checkedAllBox" />全选/全不选

        <br/>
        <input type="checkbox" name="items" value="足球" />足球
        <input type="checkbox" name="items" value="篮球" />篮球
        <input type="checkbox" name="items" value="羽毛球" />羽毛球
        <input type="checkbox" name="items" value="乒乓球" />乒乓球

        <br/>
        <input type="button" id="checkedAllBtn" value="全 选"/>
        <input type="button" id="checkedNoBtn" value="全不选"/>
        <input type="button" id="checkedRevBtn" value="反 选"/>
        <input type="button" id="sendBtn" value="提 交"/>
    </form>
</body>
</html>

4.DOM查询的剩余方法

<script>
        window.onload = function(){
            /*
            *   在document中有一个属性body,他保存的是body的引用
            */
           var body = document.body;

            /*
            *   document.documentElement保存的是html跟标签
            */
           var html = document.documentElement;

           /*
            *   document.all代表页面中所有的元素
            *   这个两行代码一样
            */
           var all = document.all;
           all = document.getElementsByTagName("*");

            /*
            *   根据元素的class属性值,查询一组元素节点对象
            *   getElementsByClassName() 可以根据class属性值获取一组元素节点对象
            *   但该方法不支持IE8及一下浏览器,可以用querySelector() 代替
            */
            var box1 = document.getElementsByClassName("box1");

        };
    </script>

document.querySelector()
需要一个选择器的字符串作为参数,可以根据一个CSS选择器来查询一个元素节点对象

使用该方法总会返回唯一的元素,如果满足条件的元素有多个,只会返回第一个

            var div = document.querySelector(".box div");
            console.log(div.innerHTML);

document.querySelectorAll()
该方法和querySelector()用法类似,不同的是它会将符合条件的元素封装到一个数组中返回

五、DOM增删改

  1. document.createElement()
    可以用与创建一个元素节点对象,
    它需要一个标签名作为参数,将会根据该标签名创建元素节点对象,
    并将建好的对象作为返回值返回
  2. document.createTextNode()
    可以用来创建一个文本节点对象
    需要一个文本内容作为参数,将会根据该内容创建文本节点,并将新的节点返回
  3. appendChild()
    向一个父节点中添加一个新的子节点
    用法:父节点.appendChild(子节点);
  4. insertBefore()
    可以在指定的子节点前插入新的子节点
    语法:父节点.insertBefore(新节点,旧节点);
  5. replaceChild()
    可以使指定的子节点替换已有的子节点
    语法:父节点.replaceChild(新节点,旧节点);
  6. removeChild()
    可以删除子节点
    语法:父节点.removeChild(子节点); = = 子节点.parentNode.removeChild(子节点);
    city.removeChild(bj);== bj.parentNode.removeChild(bj);
<!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>
    <link rel="stylesheet" href="./css/style.css">
    <link rel="stylesheet" href="./css/reset.css">
    <script type="text/javascript">
        
        window.onload = function(){
            function myClick(idStr, fun){
                var btn = document.getElementById(idStr);
                btn.onclick = fun;
            }
            // 1.创建一个"广州"节点,添加到#city下
            myClick("btn01",function(){
                // 创建广州节点——创建li元素节点、创建广州文本节点
                var li = document.createElement("li");
                var gzText = document.createTextNode("广州");
                // 将gzText设置为li的子节点
                li.appendChild(gzText);

                // 将广州添加到city下
                var city = document.getElementById("city");
                city.appendChild(li);
            });


            // 2.将"广州"节点插入到#bj前面
            myClick("btn02",function(){
                var li = document.createElement("li");
                var gzText = document.createTextNode("广州");
                li.appendChild(gzText);
                var bj = document.getElementById("bj");
                var city = document.getElementById("city");
                city.insertBefore(li,bj);
            });


            // 3.使用"广州"节点代替#bj节点
            myClick("btn03",function(){
                var li = document.createElement("li");
                var gzText = document.createTextNode("广州");
                li.appendChild(gzText);
                var bj = document.getElementById("bj");
                var city = document.getElementById("city");
                city.replaceChild(li,bj);
            });


            // 4.删除#bj节点
            myClick("btn04",function(){
                var bj = document.getElementById("bj");
                // var city = document.getElementById("city");
                // city.removeChild(bj);
                bj.parentNode.removeChild(bj);
            });


            // 5.读取#city内的HTML代码
            myClick("btn05",function(){
                var city = document.getElementById("city");
                alert(city.innerHTML);
            });

            // 6.设置#bj内的HTML代码
            myClick("btn06",function(){
                var bj = document.getElementById("bj");
                bj.innerHTML = "昌平"

            });


            // 7.向#city中添加"广州"节点
            myClick("btn07",function(){
                var city = document.getElementById("city");
                /*
                *   使用innerHTML也可以完成DOM的增删改的相关操作
                *   一般我们会两种方式相结合使用
                */
                // city.innerHTML += "<li>天津</li>"
                var li = document.createElement("li");
                li.innerHTML = "天津";
                city.appendChild(li);
            });
        };
    </script>
</head>
<body>
    <div class="total">
        <div class="inner h">
            <p>你喜欢哪个城市?</p>
            <ul id="city">
                <li id="bj">北京</li>
                <li>上海</li>
                <li>东京</li>
                <li>首尔</li>
            </ul>
            
        </div>
    </div>
    <div class="btnList">
        <div><button id="btn01">创建一个"广州"节点,添加到#city下</button></div>
        <div><button id="btn02">将"广州"节点插入到#bj前面</button></div>
        <div><button id="btn03">使用"广州"节点代替#bj节点</button></div>
        <div><button id="btn04">删除#bj节点</button></div>
        <div><button id="btn05">读取#city内的HTML代码</button></div>
        <div><button id="btn06">设置#bj内的HTML代码</button></div>
        <div><button id="btn07">向#city中添加"天津"节点</button></div>
        
    </div>
</body>
</html>

增删记录练习

<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>添加删除记录练习</title>
<link rel="stylesheet" type="text/css" href="./css/css.css" />
<script type="text/javascript">
	window.onload = function(){
		/*
		*	for循环会在页面加载完成后立即执行
		*		而响应函数会在超链接被点击时才执行
		*	当响应函数执行时,for循环早已执行完毕
		*/
		function delA(){
			var tr = this.parentNode.parentNode;
				
				// 删除之前弹出的一个提示框
				/*
				*	confirm() 用于弹出一个带有确认和取消按钮的提示框
				*	需要一个字符串作为参数,该字符串会作为提示文字显示出来
				*		如果用户点击确认则返回true,如果点击取消则返回false
				*/
			var flag = confirm("你确认删除"+tr.children[0].innerHTML);
			if(flag){
				tr.parentNode.removeChild(tr);
			}

			return false;
		}

		var allA = document.getElementsByTagName("a");
		// 为每个超链接都绑定一个单击响应函数
		for(var i=0;i<allA.length;i++){
			allA[i].onclick = delA;
				// 点击超链接后需要删除超链接所在行
				// 这里我们点击哪个超链接this就是谁
				// 获取当前tr
		}

		//为提交按钮绑定一个单击响应函数
		var addEmpButton = document.getElementById("addEmpButton");
		addEmpButton.onclick = function(){

			// 获取用户填写的信息
			var name = document.getElementById("empName").value;
			var email = document.getElementById("email").value;
			var salary = document.getElementById("salary").value;

			var tr = document.createElement("tr");
			tr.innerHTML = "<td>"+name+"</td>"+
							"<td>"+email+"</td>"+
							"<td>"+salary+"</td>"+
							"<td><a href='javascript:;'>Delete</a></td>";
			
			var a = tr.getElementsByTagName("a")[0];
			a.onclick = delA;
			
			var employeeTable = document.getElementById("employeeTable");
			var tbody = employeeTable.getElementsByTagName("tbody")[0];
			tbody.appendChild(tr);
			
		};
			
		
	};
</script>
</head>
<body>

	<table id="employeeTable">
		<tr>
			<th>Name</th>
			<th>Email</th>
			<th>Salary</th>
			<th>&nbsp;</th>
		</tr>
		<tr>
			<td>Tom</td>
			<td>tom@tom.com</td>
			<td>5000</td>
			<td><a href="javascript:;">Delete</a></td>
		</tr>
		<tr>
			<td>Jerry</td>
			<td>jerry@sohu.com</td>
			<td>8000</td>
			<td><a href="javascript:;">Delete</a></td>
		</tr>
		<tr>
			<td>Bob</td>
			<td>bob@tom.com</td>
			<td>10000</td>
			<td><a href="javascript:;">Delete</a></td>
		</tr>
	</table>

	<div id="formDiv">
	
		<h4>添加新员工</h4>

		<table>
			<tr>
				<td class="word">name: </td>
				<td class="inp">
					<input type="text" name="empName" id="empName" />
				</td>
			</tr>
			<tr>
				<td class="word">email: </td>
				<td class="inp">
					<input type="text" name="email" id="email" />
				</td>
			</tr>
			<tr>
				<td class="word">salary: </td>
				<td class="inp">
					<input type="text" name="salary" id="salary" />
				</td>
			</tr>
			<tr>
				<td colspan="2" align="center">
					<button id="addEmpButton" value="abc">
						Submit
					</button>
				</td>
			</tr>
		</table>

	</div>

</body>
</html>

六、操作内联样式

<!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>
        #box1{
            width: 200px;
            height: 200px;
            background-color: red;
        }
    </style>
    <script>
        window.onload = function(){
            var box1 = document.getElementById("box1");
            var btn1 = document.getElementById("btn1");
            btn1.onclick = function(){
                /*
                *   通过JS修改元素的样式:
                *       语法:元素.style.样式名 = 样式值;
                * 
                * 注意:如果CSS的样式名中含有-,这种名称在JS中是不合法的
                *       需要将这种样式名修改为驼峰命名法
                *       去掉 - ,然后将-后的字母大写
                * 
                *  我们通过style属性设置的样式都是内联样式,
                *   而内联样式有较高的优先级,所以通过JS修改的样式往往会立即显示
                * 
                *   但是如果在样式中写了!important,则此样式会有最高的优先级
                *       即使JS也不能覆盖该样式,所以尽量不要为样式添加!important
                */
                box1.style.width = "300px";
                box1.style.height = "300px";
                box1.style.backgroundColor = "yellow";

            };



            // 点击第二个按钮后,读取元素样式
            var btn2 = document.getElementById("btn2");
            btn2.onclick = function(){
                // 读取box1样式
                /*
                *   语法:元素.style.样式名
                *
                *   通过style属性设置和读取的都是内联样式
                *   无法读取样式表中的样式
                */
               alert(box1.style.width);
            };
        };
    </script>
</head>
<body>
    <button id="btn1">第一个按钮</button>
    <button id="btn2">第二个按钮</button>
    <div id="box1"></dic>
</body>
</html>

获取元素的当前显示的样式
语法:元素.currentStyle.样式名
它可以用来读取当前元素正在显示的样式
如果当前元素没有设置样式,则获取它的默认值
currentStyle 只有IE浏览器支持,其他的浏览器都不支持


在其他浏览器中可以使用:
getComputedStyle()这个方法来获取元素当前的样式

这个方法是window的方法,可以直接使用
需要两个参数:

  • 第一个:要获取样式的元素
  • 第二个:可以传递一个伪元素,一般都传null

该方法会返回一个对象,对象中封装了当前元素对应的样式
如果获取的样式没有设置,则会获取到真实的值,而不是默认值
比如:没有设置width,它不会获取到auto,而是一个长度

         var obj = getComputedStyle(box1,null);
         alert(obj.width);

通过currentStyle和getComputedStyle()读取到的样式都是只读的,不能修改,如果需要修改必须通过styel属性

如果想要兼容所有浏览器:
定义一个函数,用来获取指定元素的当前的样式
参数:

  • obj 要获取样式的元素
  • name 要获取的样式名
function getStyle(obj,name){
	if(window.getComputedStyle){
		return getComputedStyle)(obj,null)[name];
	}else{
		return obj.currentStyle[name];
	}
}

七、其他样式相关的属性

<script>
        window.onload = function(){
            var box1 = document.getElementById("box1");
            var btn1 = document.getElementById("btn1");
            btn1.onclick = function(){
                /*
                *   clientWidth
                *   clientHeight
                *    - 这两个属性可以获取元素的可见高度和宽度
                *    - 这两个属性都是不带px的,返回的都是一个数子,可以直接进行计算
                *    - 会获取元素的宽度和高度,包括内容区和内边距
                *    - 这些属性都是只读的,不能修改
                */
               alert(box1.clientWidth);
               /*
                *   offsetWidth
                *   offsetHeight
                *    - 获取元素的整个的宽度和高度,包括内容区、内边距和边框
                */
                alert(box1.offsetWidth);
                /*
                *   offsetParent
                *    - 可以用来获取当前元素的定位父元素
                *    - 会获取到离当前元素最近的开启了定位的祖先元素
                *    - 如果所有的祖先元素都没有开启定位,则会返回body
                */
               var op = box1.offsetParent;
               alert(op);
               /*
                *   offsetLeft
                *    - 当前元素相对于其定位父元素的水平偏移量              
                *   offsetTop
                *    - 当前元素相对于其定位父元素的垂直偏移量
                */
               alert(box1.offsetLeft);
                /*
                *   scrollWidth
                *   scrollHeight
                *    - 可以获取元素整个滚动区域的宽度和高度
                */
               alert(box1.scrollHeight);
                /*
                *   scrollLeft
                *   scrollTop
                *    - k可以获取水平和垂直滚动条滚动的距离
                */
                alert(box1.scrollLeft);

                // 当满足 scrollHeight - scrollTop == clientHeight
                // 说明滚动到底了
                
            };



        };
    </script>

练习——注册

<!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>
        #info{
            width: 400px;
            height: 400px;
            background-color: #bfa;
            overflow: auto;
        }

    </style>
    <script>
        window.onload = function(){
            /*
            *   当垂直滚动条滚动到底时,使表单项可用
            *   onscroll
            *    - 该事件会在元素的滚动条滚动时触发
            */
            var info = document.getElementById("info");
            var inputs = document.getElementsByTagName("inprut");
            info.onscroll = function(){
                // 检查垂直滚动条是否滚动到底
                if(info.scrollHeight-info.scrollTop <= info.clientHeight){
                    // 是表单项可用
                    inputs[0].disabled=false;
                    inputs[1].disabled=false;
                }
            };
        };
    </script>
</head>
<body>
    <h3>欢迎注册</h3>
    <p id="info">
        请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!请仔细阅读,才能注册!!!!!!!!!!!!!
    </p>
    <!-- 如果为表单项添加disabled="disabled" 则表单项变成不可用状态 -->
    <input type="checkbox" disabled="disabled"/>我已仔细阅读
    <input type="submit" value="注册" disabled="disabled"/>
</body>
</html>

运行未成功😭😰😠🤕

八、事件对象

<!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 type="text/css">

        #areaDiv {
            border: 1px solid black;
            width: 300px;
            height: 50px;
            margin-bottom: 10px;
        }
        
        #showMsg {
            border: 1px solid black;
            width: 300px;
            height: 20px;
        }
    
    </style>
    <script>
        window.onload = function(){
            // 当鼠标在areaDiv中移动时,在showMsg中来显示鼠标的坐标
            var areaDiv = document.getElementById("areaDiv");
            var showMsg = document.getElementById("showMsg");
            
            /*
            *   onmousemove
            *    - 该事件将会在鼠标在元素中移动时被触发
            * 
            *   事件对象
            *     - 当事件的响应函数被触发时,浏览器每次都会将一个事件作为实参传递进响应函数
            *       在事件对象中封装了当前事件相关的一切信息,比如:鼠标的坐标、键盘哪个键被按下、鼠标滚轮滚动的方向……
            */
           areaDiv.onmousemove = function(event){

            /*
            *   在IE8中,响应函数被触发时,浏览器不会传递事件对象
            *       在IE8及以下的浏览器中,是将事件对象作为window对象的属性保存的
            *           所以改成 var x = window.event.clientX; 即可
            *   或者如下
            */
            if(!event){
                event = window.event;
            }
            // event = event || window.event;  同上

               /*
               *    clientX可以获取鼠标指针的水平坐标
               *    clientY可以获取鼠标指针的垂直坐标
               */
              var x = event.clientX;
              var y = event.clientY;
            //    在showMsg中显示坐标
            showMsg.innerHTML = "x ="+x+"y ="+y
           };


        };
    </script>
</head>
<body>
    <div id="areaDiv"></div>
    <div id="showMsg"></div>
</body>
</html>

1.练习——div跟随鼠标移动


<!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>
        html{
            height: 1000px;
        }
        #box1{
            width: 100px;
            height: 100px;
            background-color: red;
            /* 开启绝对定位 */
            position: absolute;
        }
    </style>
    <script>
        window.onload = function(){
            // 使div可以跟随鼠标移动
            var box1 = document.getElementById("box1");
            document.onmousemove = function(event){
                // 解决兼容问题
                event = event||window.event;

                // 获取鼠标坐标
                /*
                *   clientX和clientY
                *   用于获取鼠标在当前可见窗口的坐标
                *       div的偏移量是相对于整个页面的
                *   
                *   pageX和pageY可以获取鼠标相对于当前页面的坐标
                *   但这两个属性在IE8中不兼容
                */

                /*
                *   获取滚动条滚动的距离
                *   chrome认为浏览器的滚动条时body的,可以通过body.scrollTop来获取
                *   火狐等浏览器认为浏览器的滚动条是html的
                */
                var st = document.body.scrollTop || document.documentElement.scrollTop;
                var sl = document.body.scrollLeft || document.documentElement.scrollLeft;

                // var left = event.pageX;
                // var top = event.pageY;
                var left = event.clientX;
                var top = event.clientY;

                // 设置div的偏移量
                box1.style.left = left+ + sl + "px";
                box1.style.top = top+ st + "px";
            };
        };
    </script>
</head>
<body>
    <div id="box1"></div>
</body>
</html>

2.事件的冒泡

<!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>
        #box1{
            width: 200px;
            height: 200px;
            background-color: yellowgreen;
        }
        #s1{
            background-color: yellow;
        }
    </style>
    <script>
        window.onload = function(){

            /*
            *   事件的冒泡(Bubble)
            *       - 所谓的冒泡指的就是事件的向上传导,当后代元素上的事件被触发时,其祖先元素的相同事件也会被触发
            *       - 在开发中大部分冒泡都是有用的,如果不希望发生事件冒泡可以取消冒泡
            *       -       
            */

            var s1 = document.getElementById("s1");
            s1.onclick = function(event){
                event = event || window.event;
                alert("1");
                
                // 取消冒泡
                // 可以将事件对象的cancelBubble设置为true,即可取消冒泡
                event.cancelBubble = true;
            };
            var box1 = document.getElementById("box1");
            box1.onclick = function(){
                alert("2");
            };
            document.body.onclick = function(){
                alert("3");
            };
        };
    </script>
</head>
<body>
    <div id="box1">
        我是box1
        <span id="s1">
            我是span
        </span>
    </div>
</body>
</html>

3.事件的委派

例如:增删记录练习中的超链接,在一开始为每一个超链接绑定了单击响应函数,但在增加记录后,新的超链接要再次绑定单击响应函数

我们希望,只绑定一次单击响应函数,即可应用到多个元素上,即使元素是后添加的,我们可以尝试将其绑定给元素的共同的祖先元素

就叫做:事件的委派
指将事件统一绑定给元素共同的祖先元素,这样当后代元素上的事件触发时,会一直冒泡到祖先元素,从而通过祖先元素的响应函数来处理事件

事件委派是利用了冒泡,减少绑定次数,提高程序性能

u1.onclick = function(event){
	event = event || window.event;
	/*
	*	target
	*	 - event中的target表示的是触发事件的对象
	*/
	if(event.target.className == "link"){
		alert("我是ul里的单击响应函数");
	}
};

4.事件的绑定

使用 对象.事件 = 函数 的形式绑定响应函数
它只能同时为一个元素的一个事件绑定一个响应函数
不能同时绑定多个,如果绑定了多个,则后面的会覆盖掉前面的

addEventListener()

  • 通过这个方法也可以为元素绑定单击响应函数
  • 参数:
    1.事件的字符串,不要 on
    2.回调函数,但事件触发时该函数会被调用
    3.是否在捕获阶段触发事件,需要一个布尔值,一般都传false
  • 使用addEventListener()可以同时为一个元素的相同事件同时绑定多个响应函数
  • 这样当事件被触发时,响应函数将会按照函数的绑定顺序执行
  • 但这个方法不支持IE8及以下的浏览器
btn01.addEventListener("click",function{
	alert(1);
},false);
btn01.addEventListener("click",function{
	alert(2);
},false);
btn01.addEventListener("click",function{
	alert(3);
},false);

attachEvent()

  • 在IE8中可以使用attachEvent()来绑定事件
  • 参数:
    1.事件的字符串,要on
    2.回调函数
  • 这个方法也可以同时为一个事件绑定多个处理函数
  • 不同的是它是后绑定先执行,执行顺序和addEventListener()相反
btn01.attachEvent("onclick",function(){
	alert(1);
});
btn01.attachEvent("onclick",function(){
	alert(2);
});
btn01.attachEvent("onclick",function(){
	alert(3);
});

addEventListener()中的this,是绑定事件的对象
attachEvent()中的this,是window
this是谁是由调用方式决定的
callback.call(obj)在调用回调函数时,无法使用
所以添加以个匿名函数加入回调函数就可以使用了

所以为了解决兼容问题,定义一个函数,用来为指定元素绑定单击响应
参数:
obj 要绑定事件的对象
eventStr 事件的字符串(不要on)
callback 回调函数

		function bind(obj, eventStr, callback) {
               if (obj.addEventListener) {
                    //大部分浏览器兼容的方式
                    obj.addEventListener(eventStr, callback, false);
                } else {
                    //IE8及一下
                    obj.attachEvent("on" + eventStr, function () {
                        callback.call(obj);
                    });
                }
            }

            bind(btn01, "click", function () {
                alert("我是box1的响应函数")
            });

5.事件的传播

关于事件的传播,网景公司和微软公司有不同的理解

  • 微软公司:认为事件应该是由内向外传播,也就是当事件触发时,应该先触发当前元素上的事件,然后再向当前元素的祖先元素上传播,也就是说事件应该在冒泡阶段执行
  • 网景公司:认为事件应该是由外向内传播的,也就是当事件触发时,应该先触发当前元素的最外层的祖先元素的事件,然后再向内传播给后代元素
  • W3C:综合了两个公司的方案,将事件传播分成了三个阶段
    1.捕获阶段 —— 在捕获阶段时从最外层的祖先元素,向目标元素进行事件的捕获,但是默认此时不会触发事件
    2.目标阶段 —— 事件捕获的目标元素,捕获结束开始在目标元素上触发元素
    3.冒泡阶段 —— 事件从目标元素向他的祖先元素传递,依次触发祖先元素上的事件
  • 如果想要在捕获阶段就触发事件,可以将addEventListener()的第三个参数设置为true(一般用不到)
  • 在IE8及一下的浏览器中没有捕获阶段

<!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>
        #box1 {
            width: 300px;
            height: 300px;
            background-color: aqua;
        }

        #box2 {
            width: 200px;
            height: 200px;
            background-color: antiquewhite;
        }

        #box3 {
            width: 100px;
            height: 100px;
            background-color: crimson;
        }
    </style>
    <script>
        window.onload = function () {
            var box1 = document.getElementById("box1");
            var box2 = document.getElementById("box2");
            var box3 = document.getElementById("box3");
            function bind(obj, eventStr, callback) {
                if (obj.addEventListener) {
                    //大部分浏览器兼容的方式
                    obj.addEventListener(eventStr, callback, false);
                } else {
                    //IE8及一下
                    obj.attachEvent("on" + eventStr, function () {
                        callback.call(obj);
                    });
                }
            }

            bind(box1, "click", function () {
                alert("我是box1的响应函数")
            });

            bind(box2, "click", function () {
                alert("我是box2的响应函数")
            });

            bind(box3, "click", function () {
                alert("我是box3的响应函数")
            });


        }
    </script>
</head>

<body>
    <div id="box1">
        <div id="box2">
            <div id="box3"></div>
        </div>
    </div>
</body>

</html>

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值