代替Watir中click_no_wait的方法。

我在刚学watir的时候被js弹出对话框折腾的死去活来,如何处理弹出框的方法网络上一搜一大堆,但是如何点出弹出框的文章却很少。因为如果用click或者click!方法点击会阻塞脚本,不能让脚本执行下去,而click_no_wait方法又不稳定(用ruby186-27_rc2.exe安装的ruby click_no_wait方法根本就不好用),当时差点让我对watir失去了信心。
还好在watir群里(群号25656482)的一位朋友给我提供了另一个替代click_no_wait的方法。
首先建一个ruby文件(我把它命名为ClickHelper.rb),把下面代码考进去。

#File Name: ClickHelper.rb
require 'watir'
require 'Win32API'
module Watir
class Element
def top_edge
assert_exists
assert_enabled
ole_object.getBoundingClientRect.top.to_i
end

def top_edge_absolute
top_edge + page_container.document.parentWindow.screenTop.to_i
end

def left_edge
assert_exists
assert_enabled
ole_object.getBoundingClientRect.left.to_i
end

def left_edge_absolute
left_edge + page_container.document.parentWindow.screenLeft.to_i
end

def right_click
x = left_edge_absolute
y = top_edge_absolute
#puts "x: #{x}, y: #{y}"
WindowsInput.move_mouse(x, y)
WindowsInput.right_click
end

def left_click
x = left_edge_absolute
y = top_edge_absolute
#puts "x: #{x}, y: #{y}"
# need some extra push to get the cursor in the right area
WindowsInput.move_mouse(x + 2, y + 2)
WindowsInput.left_click
end
end
end

module WindowsInput
# Windows API functions
SetCursorPos = Win32API.new('user32','SetCursorPos', 'II', 'I')
SendInput = Win32API.new('user32','SendInput', 'IPI', 'I')
# Windows API constants
INPUT_MOUSE = 0
MOUSEEVENTF_LEFTDOWN = 0x0002
MOUSEEVENTF_LEFTUP = 0x0004
MOUSEEVENTF_RIGHTDOWN = 0x0008
MOUSEEVENTF_RIGHTUP = 0x0010

module_function

def send_input(inputs)
n = inputs.size
ptr = inputs.collect {|i| i.to_s}.join # flatten arrays into single string
SendInput.call(n, ptr, inputs[0].size)
end

def create_mouse_input(mouse_flag)
mi = Array.new(7, 0)
mi[0] = INPUT_MOUSE
mi[4] = mouse_flag
mi.pack('LLLLLLL') # Pack array into a binary sequence usable to SendInput
end

def move_mouse(x, y)
SetCursorPos.call(x, y)
end

def right_click
rightdown = create_mouse_input(MOUSEEVENTF_RIGHTDOWN)
rightup = create_mouse_input(MOUSEEVENTF_RIGHTUP)
send_input( [rightdown, rightup] )
end

def left_click
leftdown = create_mouse_input(MOUSEEVENTF_LEFTDOWN)
leftup = create_mouse_input(MOUSEEVENTF_LEFTUP)
send_input( [leftdown, leftup] )
end
end


这段代码是给watir添加了几个方法(主角Watir模块的left_click方法),其原理就是通过DOM计算出控件的位置,然后再利用Windows的API在那个计算好的坐标模拟一下鼠标点击(left_click是左击,right_click是右击,我好像是在说废话)。

好了我们来测试一下,先建立一个测试用的网页,html代码如下。

<html>
<head>
<script>
function click_me()
{
alert("乖");
}
</script>
<title>test page</title>
</head>
<body>
<input id='btnClickMe' type = 'button' value = 'Click Me!' onclick='click_me()'/>
</body>
</html>

只有一个按钮,点击之后会弹出一个对话框。那我们现在写测试脚本。

require 'watir'
require 'ClickHelper'

ie = Watir::IE.attach(:title, "test page")
ie.button(:id, 'btnClickMe').click
#ie.button(:id, 'btnClickMe').click!
#ie.button(:id, 'btnClickMe').click_no_wait
#ie.button(:id, 'btnClickMe').left_click
puts "Successful"
# Deal with the pop up window.


分别使用四种方法点击button,结果:
click, click!方法都会弹出对话框,但是脚本会停止执行下面的步骤,也就是说不会输出"Successful",甚至脚本会永远卡在那里。
click_no_wait可以输出"Successful",但是不会弹出对话框(这个方法与ruby,watir的版本有关,相信很多人会遇到跟我一样的问题)。
left_click可以弹出对话框,也可以输出"Successful",正是我们想要的效果。

问题貌似要解决了,但是前几天我发现了left_click的一个缺陷,那就是当按钮不在屏幕中显示时按钮就按不到了。
我把一开始的html测试页代码改了一下,增加了点文字,然后把窗口缩放一下,弄成下图的样子。
[img]http://dl.iteye.com/upload/attachment/210696/50e03394-bac0-3d90-aa61-3e096f30a9c5.jpg[/img]
此时按钮还在页面中,但是再用left_click方法就点不到了 :cry: 。

但是遇到问题还是要解决的。这时候我想起当我们往文本框输入的时候,页面会自动将文本框调整到浏览器的屏幕中,那么如果按钮也能自动调整进来,那么就可以继续使用left_click了。
于是我查看text_field的set的源代码。在这里插一句,在cmd中输入gem server后,在浏览器地址栏里输127.0.0.1:8808就可以直接查看你已经安装的gem包的api了,非常方便(不要用IE,IE的显示有问题)。
当我看到text_field的set方法的代码时发现他要先执行一个focus的操作。
[img]http://dl.iteye.com/upload/attachment/210704/a8c5b15b-1bfd-39f2-ab98-2ae9e616eb09.jpg[/img]
于是在自己测试脚本上也加上这一句。

require 'watir'
require 'ClickHelper'

ie = Watir::IE.attach(:title, "test page")
ie.button(:id, 'btnClickMe').focus
ie.button(:id, 'btnClickMe').left_click
puts "Successful"
# Deal with the pop up window.

这下果真成功了!脚本会先把按钮调整到屏幕中,然后再点击 :D !

好了,问题已经解决了,剩下的就是优化一下,把focus封装到left_click里面,这样以后就不用自己写focus了。
观察了一下,在Element类的left_edge方法中添加一句话。
def left_edge
assert_exists
assert_enabled
ole_object.focus # 这句话是我加的
ole_object.getBoundingClientRect.left.to_i
end

为了保证脚本的健壮性,必须保证每次用left_click时被操作的控件都在最前。还好watir有一个方法专门干这件事,那就是bring_to_front。
还是在left_edge里加。
def left_edge
assert_exists
assert_enabled
page_container.bring_to_front # 这句话是我加的
ole_object.focus # 这句话也是我加的
ole_object.getBoundingClientRect.left.to_i
end

大功告成。附修改过后的脚本,嫌麻烦的直接用吧。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值