开学了软测的同学快来--详解selenium之JS元素定位--内附JS滚动条源码

Posted 清安无别事

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了开学了软测的同学快来--详解selenium之JS元素定位--内附JS滚动条源码相关的知识,希望对你有一定的参考价值。

        开学了系列前面讲了element元素定位以及elements元素组定位,有不少的小伙伴问我,这个元素定位不到,那个元素定位不到,该怎么办。今天讲到的可以帮你解决一部分问题,剩下的就需要靠个人检查一下,你自己真的学会元素定位了吗?

        你是否认为JS元素定位就跟前面所讲的element以及elements元素定位一样。在这儿就可以告诉各位,用法上确实有些类似,但是用途上就有很明显的差别。我们直接看正文吧!

目录

Windows对象:

获取窗口尺寸

获取内部宽高属性

获取外部宽高属性

操作窗口

Location对象

location属性

location方法

HTML DOM节点

元素定位

更改元素属性值

 控制滚动条



        JS全名:javascript,是WEB网页组成的三部分之一。JavaScript 代码放在位于<script>标签与<script>标签之间。先看一段JS代码认识一下JS:

 <!DOCTYPE html>
<html>
<body>
<h1>我的第一个 Web 页面</h1>
<p>我的第一个段落。</p>
<button onclick="myFunction()">点我</button>
<script>
function myFunction() {
    document.write(Date());
}
</script>
</body>
</html> 

        它通常会嵌在HTML中,以<script></script>的形式存在,有时候你看到的并非文中这样的,而是类似于链接的形式存在,那是因为吧JS代码写在的外面,通过链接形式引入,原则上是一样的,显得美观一些。如下:

        这里运行的话就很简单了,直接复制到txt文档,保存后重命名将后缀改为.html即可,双击直接打开就可以看到效果了。

        看完JS代码,我们先了解一下浏览器对象模型BOM,方便后面更好的学习,BOM全名Browser Object Mode,允许JS与浏览器交互。本章主要介绍BOM模型里面的Windows对象以及location对象。

Windows对象:

      1. 获取浏览器窗口的尺寸

      2. 操作其他窗口方法:如打开新窗口,关闭窗口等

      3. 操作当前窗口,如:获取当前页面的地址,重新加载一个新页面,刷新当前页面

获取窗口尺寸

获取内部宽高属性

        innerHeight,innerWidth 单位:px(像素)

        内部宽高:是指除去菜单栏、工具栏、边框等占位元素后,用于显示网页的净宽高

         • window.innerHeight - 浏览器窗口的内高度(以像素计)

        • window.innerWidth - 浏览器窗口的内宽度(以像素计)

获取外部宽高属性

        outerHeight,outerWidth 单位:px(像素)

        • window.outerHeight 浏览器窗口的外部高度(以像素计)

        • window.outerWidth 浏览器窗口的外部宽度(以像素计)

from selenium import webdriver

# 获取内宽,内高
fox = webdriver.Firefox()
fox.get('https://baidu.com')
js_ih = 'return window.innerHeight'
js_iw = 'return window.innerWidth'
i = fox.execute_script(js_ih)
o = fox.execute_script(js_iw)
#也可以写成
# i = fox.execute_script(js_ih,js_iw)
# print(i)
print(f"内高是:{i},内宽是:{o}")
# 外宽,外高
js_oh = 'return window.outerHeight'
js_ow = 'return window.outerWidth'
j = fox.execute_script(js_oh)
s = fox.execute_script(js_ow)
print(f"外高是:{j},外宽是:{s}")
fox.quit()

        看到这是不是诧异,JS代码写在python中还有有点复杂程度的或者说是有点麻烦的。如果你想获取宽高,你必须要加return不然是没有返回值的,并且没个JS语句你想执行的话都必须fox.execute_script()带上这么一个执行语句,不然写了跟没写一个样。引号内的代码都需要靠纯手写,所以考验键盘能力的时候到了。

操作窗口

        1. window.open(url) - 打开新窗口,并打开指定的url  

        2. window.close() - 关闭当前窗口

        3. selenium执行js语句 -fox.execute_script(js)

from selenium import webdriver

fox = webdriver.Firefox()
fox.implicitly_wait(5)
fox.get('https://baidu.com')
# 打开新窗口
js = "window.open('https://www.csdn.net/')"
fox.execute_script(js)
# 关闭新窗口
js_c = 'window.close()'
fox.execute_script(js_c)
fox.quit()

Location对象

location属性

        1. window.location.href 返回当前页面的 href (URL)

        2. window.location.hostname 返回 web 主机的域名

        3. window.location.pathname 返回当前页面的路径或文件名

        4. window.location.protocol 返回使用的 web 协议(http: 或 https:)

        这里的主机域名可能获取不到,还有就是协议。这里只做了解即可。

from selenium import webdriver

fox = webdriver.Firefox()
# 获取地址
fox.get('https://baidu.com')
js_href = "return location.href"
js_host = "return location.hostname"
js_path = "return location.pathname"
js_port = "return location.protocol"
i = fox.execute_script(js_href, js_host, js_path, js_port)
print(i)
fox.quit()

location方法

        1. location.assign(url) 加载新页面

        2. location.reload() 重新加载当前页面,刷新

        3. location.replace(url) 用输入的url替换当前的url

        值得注意的是:

        assign与replace的区别:

                1. location.assign(url) : 加载 URL 指定的新的 HTML 文档。就相当于一个链接,跳转到指定的url,当前页 面会转为新页面内容,可以点击后退返回上一个页面。

                2. location.replace(url) : 通过加载 URL 指定的文档来替换当前文档,这个方法是替换当前窗口页面,前 后两个页面共用一个窗口,所以是没有后退返回上一页的

        这些都是小练手,记得自己上手,提示:不需要得到值的就不需要添加return了,可以直接使用,举例:

from selenium import webdriver

fox = webdriver.Firefox()
fox.implicitly_wait(5)
# 获取地址
fox.get('https://baidu.com')
js = "return location.href"
i = fox.execute_script(js)
print(i)
# 打开一个新的窗口
fox.execute_script("location.assign('https://baidu.com')")

HTML DOM节点

        HTML DOM (Document Object Model) 中 , 每一个元素都是节点。

        Document 对象是 Window 对象的一部分,可通过 window.document 属性对其进行访问。

元素定位

        这里才是重点,JS元素定位其实说的就是Document元素定位。

        查找元素,最常用的查找是根据ID和Tag Name ,注意:除id,css选择器以外是元素组定位

                1. document.getElementById(id属性值) --》唯一的

                2. document.getElementsByClassName(class属性值) --》可以定位一组元素

                        • document.getElementsByClassName(class属性值)[索引值] 对一组元素中的单个元素的获取

                3. document.getElementsByName(name属性值)

                4. document.getElementsByTagName(标签名)

                5. document.querySelector(css选择器) --》对于没有id/name/classname属性,可以使用这个定位元 素

        这里与普通的元素定位没有什么太大的区别,元素组定位索引取值,元素定位。还有就是value与send_keys区别。

from selenium import webdriver

fox = webdriver.Firefox()
fox.implicitly_wait(5)
fox.get('https://baidu.com')
js_input = "document.getElementById('kw').value='python'"
fox.execute_script(js_input)
jx_click = "document.getElementById('su').click()"
fox.execute_script(jx_click)
fox.quit()
from selenium import webdriver

fox = webdriver.Firefox()
fox.implicitly_wait(5)
fox.get('https://baidu.com')
js_input = "document.getElementsByClassName('mnav')[0].click()"
fox.execute_script(js_input)
sleep(2)
fox.quit()

        这里看到区别了吧,JS中元素定位方式类似,写法还是有一定的区别的。我个人举得还是挺繁琐的。其他的元素定写法与上述类似,这里就不做全部的代码展示了。自行练习!

更改元素属性值

        看到这是不是很意外,还能更改元素属性值,JS的特点这里也有份,网页中有部分的属性是隐藏的,有些属性只读不可写让我们具体看看。

        1. document.getElementById('kw').autocomplete= 'off' --常用元素属性修改;百度

        2. document.getElementById('vip').style.visibility= 'visible' --设置元素的隐藏属性,是否显示;

        隐藏的元素:利用javascript更改元素属性值,让元素可见

        例如:visibility:hidden(隐藏),visible(显示)

                   display : none(隐藏),block(显示)

         看这个例子,我们可以通过更改属性值直接隐藏掉这个烦人的东西。

from time import sleep
from selenium import webdriver

fox = webdriver.Firefox()
fox.implicitly_wait(5)
fox.get('https://www.tmall.com/')
js = "document.querySelector('.j_doodleLink').style.display='none'"
fox.execute_script(js)
sleep(3)
fox.quit()

        还可以更改很多东西,比如text值,定位元素后.text=' '想要修改的值,具体的用法很多很多,对于测试而言,这部分其实也差不多,但是你想技多不压身就去研究研究。

 控制滚动条

        控制滚动条有什么好处,如果页面没有完全显示,element如果是在下拉之后才能显示出来,只能先滚动到该元素才能进 行click,否则是不能click。

        对于纵向滚动条,我们要获取滚动条的高度;

        对于横向滚动条则需要获取他的宽度

                1. document.body.scrollHeight 获取对象的滚动高度

                2. document.body.scrollWidth 获取对象的滚动宽度

        滑动滚动条:

                window.scrollTo(x,y) 方法可把内容滚动到指定的坐标。

        滑动到页面底部:

                1. 左下角:window.scrollTo(0,document.body.scrollHeight)

                2. 右下角:

                        window.scrollTo(document.body.scrollWidth,document.body.scrollHeight)

                3. 指定位置:window.scrollTo(0,数值)

                4. 滑动到指定元素:ele.srollIntoView() true:与元素顶部对其,false:与元素底部对其                         document.querySelector('').scrollIntoView()

                        document.querySelector('').scrollIntoView(false)        

        这里直接拿淘宝的例子来跑,先下拉到指定举例,再下拉到最底部:

from selenium import webdriver
from time import sleep

fox = webdriver.Firefox()
fox.get('https://www.taobao.com/')
fox.maximize_window()

# 获取高度
js_height = "return document.body.scrollHeight"
height = fox.execute_script(js_height)
print(height)
# 获取宽度
js_width = "return document.body.scrollWidth"
width = fox.execute_script(js_width)
print(width)
# 下滑1000px
js_left = "window.scrollTo(0,1000)"
fox.execute_script(js_left)
sleep(2)
# 下滑最底部
jx_left_down = "window.scrollTo(0,document.body.scrollHeight)"
fox.execute_script(jx_left_down)
# 下滑到指定元素
# jx_left_zd = "document.querySelector('ul.list > a:nth-child(3) > div:nth-child(1) > div:nth-child(2)').scrollIntoView(false)"
# fox.execute_script(jx_left_zd)

        上述是浏览器滚动条。接下来给到各位一个界面元素滚动条。自己写了一个JS,便于练习,源码给到各位:

<!DOCTYPE html>
<html>
	<head>
		<meta charset="utf-8">
		<title></title>
		<style type="text/css">
			*{
				margin: 0;
				padding: 0;
			}
			.news{
				width: 600px;
				height: 240px;
				border: 1px solid #333;
				line-height: 160%;
				margin: 50px auto;
				position: relative;
				padding-right: 30px;
				overflow: hidden;
			}
			.news .content{
				position: relative;
				padding: 20px;
			}
			.news .content p{
				margin-bottom: 20px;
			}
			.news .bar{
				width: 20px;
				height: 240px;
				background-color: #eee;
				position: absolute;
				right:10px;
				top: 0;
			}
			.news .bar b{
				position: absolute;
				width: 20px;
				height: 104px;
				top: 0;
				left: 0;
				border-radius: 6px;
				box-shadow: 1px 1px 1px black;

			}
			@-webkit-keyframes faguang{
				0%{
					box-shadow: 0px 0px 0px red;
				}
				 
				100%{
					box-shadow: 0px 0px 20px red;
				}
			}
			@-moz-keyframes faguang{
				0%{
					box-shadow: 0px 0px 0px red;
				}
				 
				100%{
					box-shadow: 0px 0px 20px red;
				}
			}
		</style>
	</head>
	<body>
		<div class="news" id="news">
			<div class="bar">
				<b id="barb"></b>
			</div>
			<div class="content" id="contentBox">
				<p>人们常说,犹豫一万次,不如实践一次。一个人越犹豫不决,就越容易患得患失.</p>
				<p>	西雨斜风作小寒,淡烟疏柳媚晴滩.</p>
				<p>入淮清洛渐漫漫,雪沫乳花浮午盏。</p>
				<p>蓼茸蒿笋试春盘,人间有味是清欢</p>
				<p>个人微信号qing_an_an</p>
			</div>
		</div>
		<script type="text/javascript">
			var barb = document.getElementById("barb");
			var contentBox = document.getElementById("contentBox");
			var news = document.getElementById("news");
			
			var newsJingTop = getAllY(news);
			var contentBoxHeight = parseInt(contentBox.clientHeight);
			var rate = 240 / contentBoxHeight;
			if(rate > 1) {
				box.style.display = "none";
			}
			var barbheight = 240 * rate;
			barb.style.height = barbheight + "px";
			
			//公共信号量,不管是滑块,都在操作它们两个数值
			var contentBoxTop = 0;
			var barbTop = 0;
			//滑块的拖曳
			//在滑块中按下的时候
			barb.onmousedown = function(event) {
				var dy = event.offsetY;
				//注册移动事件
				document.onmousemove = function(event) {
					var y = event.pageY - newsJingTop - dy;
					//验收
					if(y < 0) {
						y = 0;
					}else if(y>240 - barbheight) {
						y = 240 - barbheight;
					}
					//滑块的移动
					//改变信号量
					barbTop = y;
					contentBoxTop = -y / rate;
					
					barb.style.top = barbTop + "px";
					//内容的移动
					contentBox.style.top = contentBoxTop + "px";
					return false;
				}
				return false;
			}
			
			document.onmouseup = function() {
				document.onmousemove = null;
			}
			
			//滚轮事件
			news.onmousewheel = mousewheelhandler;
			news.addEventListener("DOMMouseScroll",mousewheelhandler,true);
			function mousewheelhandler(event) {
				event = event|| window.event;
				if(event.preventDefault) {
					event.preventDefault();
				}else {
					event.returnValue = true;
				}
	
			
				if(event.wheelDelta) {
					var direction = event.wheelDelta>0?1:-1;
				}else{
					var direction = event.detail>0?-1:1;
				}
				contentBoxTop += direction * 20;
				if(contentBoxTop > 0) {
					contentBoxTop = 0;
				}else if(contentBoxTop < 240 - contentBoxHeight) {
					contentBoxTop = 240 - contentBoxHeight;
				}
				barbTop = -contentBoxTop * rate;
				contentBox.style.top = contentBoxTop + "px";
				barb.style.top = barbTop + "px";
			}
			function getAllY(o) {
				var allY = o.offsetTop;
				while(o == o.offsetParent) {
					allY += o.offsetTop + Number(getComputedStyle(o)["border-top-width"]);
				}
				return allY;				
			}
		</script>
	</body>
</html>

        先定位到元素

                1. ele.scrollHeight # 获取滚动条高度

                2. ele.scrollWidth # 获取横向滚动条宽度

        这里提供思路,就是,定位到元素通过鼠标事件去拖拽即可,页面内的元素都可这样使用。

        大家对于获取高度宽度可以在控制台操作。也就是浏览器F12。

以上是关于开学了软测的同学快来--详解selenium之JS元素定位--内附JS滚动条源码的主要内容,如果未能解决你的问题,请参考以下文章

开学了软测的同学在哪里-selenium-必须要懂的鼠标键盘事件+实战

字节软测面试必问的Selenium自动化测试框架设计,你会了吗?

字节软测面试必问的Selenium自动化测试框架设计,你会了吗?

字节软测面试必问的Selenium自动化测试框架设计,你会了吗?

记浙江大学培训之开学篇

测试基础之06 测试基础理论