12

我正在将 Selenium 与 Python 和 Chrome 一起使用。我试图按住各种键——特别是“w、a、s 和 d”。我找到了 Selenium 的 action_chains.key_press 动作以及 elem.send_keys 方法。第一种方法的问题在于,它只在完成一个动作链所需的时间内按住键。elem.send_keys 方法的问题在于它没有将密钥发送到的元素。

我正在尝试使用 WASD 控制基于 Web 浏览器的机器人,因此我需要按住按键不同的持续时间。

我尝试了以下方法:

action_chains = ActionChains(driver)
action_chains.key_down("w")
action_chains.key_up("w")

也:

action_chains.key_press(elem, "w")
for x in range (0, 100):
    action_chains.perform()
    time.sleep(.01)

两者都不理想。

4

4 回答 4

5

Chrome 的当前驱动程序(2.30 版)实现了以前的协议,其中仅支持修改键(Control、Shift、Alt、Command)按住键。

因此,此代码适用于 Firefox,但适用于 Chrome,因为keyUp每个事件都会发出keyDown

action_key_down_w = ActionChains(driver).key_down("w")
action_key_up_w = ActionChains(driver).key_up("w")

endtime = time.time() + 1.0

while True:
  action_key_down_w.perform()

  if time.time() > endtime:
    action_key_up_w.perform()
    break

但是,从 2.30 版本开始,Chrome 驱动程序支持send_command通过 devtools 协议直接发送原始命令。因此,作为一种解决方法,您可以调用Input.dispatchKeyEvent来发出低级事件。

这是一个使用 Selenium/Chromew在一秒钟内保持密钥的工作示例:

from selenium import webdriver
import json, time

def dispatchKeyEvent(driver, name, options = {}):
  options["type"] = name
  body = json.dumps({'cmd': 'Input.dispatchKeyEvent', 'params': options})
  resource = "/session/%s/chromium/send_command" % driver.session_id
  url = driver.command_executor._url + resource
  driver.command_executor._request('POST', url, body)

def holdKeyW(driver, duration):
  endtime = time.time() + duration
  options = { \
    "code": "KeyW",
    "key": "w",
    "text": "w",
    "unmodifiedText": "w",
    "nativeVirtualKeyCode": ord("W"),
    "windowsVirtualKeyCode": ord("W")
  }

  while True:
    dispatchKeyEvent(driver, "rawKeyDown", options)
    dispatchKeyEvent(driver, "char", options)

    if time.time() > endtime:
      dispatchKeyEvent(driver, "keyUp", options)
      break

    options["autoRepeat"] = True
    time.sleep(0.01)


driver = webdriver.Chrome()
driver.get("https://stackoverflow.com/questions")

# set the focus on the targeted element
driver.find_element_by_css_selector("input[name=q]").click()

# press the key W during a second
holdKeyW(driver, 1.0)
于 2017-06-14T11:22:29.857 回答
2

Selenium 动作链 只能与修饰键(Control、Alt 和 Shift)一起使用。因此,您只想按字符 wasd。所以,它没有工作。

您可以使用任何 gui 自动化工具,如 pyautogui 等。

请尝试下面的代码并告诉我。

import pyautogui

pyautogui.PAUSE = 10
pyautogui.keyDown('w')
pyautogui.keyUp('w')

pyautogui.PAUSE=10 命令在每次 PyAutoGUI 调用后暂停 10 秒

于 2017-06-12T13:06:24.167 回答
1

根据Selenium Documentation for key_down,它指出:

只能与修饰键(Control、Alt 和 Shift)一起使用。

我已经在文档中搜索了一个替代解决方案,但似乎在 Selenium 中“按住”非修饰键的行为是不可能的。

于 2017-06-12T18:26:48.850 回答
0

ActionChains(driver).key_down("w").pause(0.1).key_up("w").perform()

通过链接key_down并在两者之间key_up插入.pause(0.1),您可以在任何自定义持续时间内按住键。

尽管根据文档key_down只能与修饰键一起使用,但事实证明它可以与任何键一起使用。

于 2022-01-20T18:50:21.253 回答