2017-02-23 08:23:37 +00:00
|
|
|
package chromedp
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strconv"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2017-12-27 02:30:28 +00:00
|
|
|
"github.com/chromedp/cdproto/cdp"
|
|
|
|
"github.com/chromedp/cdproto/input"
|
2017-02-23 08:23:37 +00:00
|
|
|
)
|
|
|
|
|
2017-02-28 01:35:00 +00:00
|
|
|
const (
|
|
|
|
// inViewportJS is a javascript snippet that will get the specified node
|
|
|
|
// position relative to the viewport and returns true if the specified node
|
|
|
|
// is within the window's viewport.
|
|
|
|
inViewportJS = `(function(a) {
|
|
|
|
var r = a[0].getBoundingClientRect();
|
|
|
|
return r.top >= 0 && r.left >= 0 && r.bottom <= window.innerHeight && r.right <= window.innerWidth;
|
|
|
|
})($x('%s'))`
|
|
|
|
)
|
|
|
|
|
2017-02-23 08:23:37 +00:00
|
|
|
func TestMouseClickXY(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
2019-02-21 12:55:21 +00:00
|
|
|
var err error
|
|
|
|
|
2017-02-23 08:23:37 +00:00
|
|
|
c := testAllocate(t, "input.html")
|
|
|
|
defer c.Release()
|
|
|
|
|
2017-06-18 02:32:12 +00:00
|
|
|
err = c.Run(defaultContext, Sleep(100*time.Millisecond))
|
2017-02-23 08:23:37 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
x, y int64
|
|
|
|
}{
|
|
|
|
{100, 100},
|
|
|
|
{0, 0},
|
|
|
|
{9999, 100},
|
|
|
|
{100, 9999},
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, test := range tests {
|
|
|
|
err = c.Run(defaultContext, MouseClickXY(test.x, test.y))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test %d got error: %v", i, err)
|
|
|
|
}
|
|
|
|
|
2017-06-18 02:32:12 +00:00
|
|
|
time.Sleep(50 * time.Millisecond)
|
2017-02-25 07:34:53 +00:00
|
|
|
|
|
|
|
var xstr, ystr string
|
|
|
|
err = c.Run(defaultContext, Value("#input1", &xstr, ByID))
|
2017-02-23 08:23:37 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test %d got error: %v", i, err)
|
|
|
|
}
|
2017-02-25 07:34:53 +00:00
|
|
|
x, err := strconv.ParseInt(xstr, 10, 64)
|
2017-02-23 08:23:37 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test %d got error: %v", i, err)
|
|
|
|
}
|
|
|
|
if x != test.x {
|
|
|
|
t.Fatalf("test %d expected x to be: %d, got: %d", i, test.x, x)
|
|
|
|
}
|
|
|
|
|
2017-02-25 07:34:53 +00:00
|
|
|
err = c.Run(defaultContext, Value("#input2", &ystr, ByID))
|
2017-02-23 08:23:37 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test %d got error: %v", i, err)
|
|
|
|
}
|
2017-02-25 07:34:53 +00:00
|
|
|
y, err := strconv.ParseInt(ystr, 10, 64)
|
2017-02-23 08:23:37 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("test %d got error: %v", i, err)
|
|
|
|
}
|
|
|
|
if y != test.y {
|
|
|
|
t.Fatalf("test %d expected y to be: %d, got: %d", i, test.y, y)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMouseClickNode(t *testing.T) {
|
2019-02-21 12:55:21 +00:00
|
|
|
t.Parallel()
|
|
|
|
|
2017-02-23 08:23:37 +00:00
|
|
|
tests := []struct {
|
2017-02-25 07:34:53 +00:00
|
|
|
sel, exp string
|
|
|
|
opt MouseOption
|
|
|
|
by QueryOption
|
2017-02-23 08:23:37 +00:00
|
|
|
}{
|
2017-02-25 07:34:53 +00:00
|
|
|
{"button2", "foo", ButtonType(input.ButtonNone), ByID},
|
|
|
|
{"button2", "bar", ButtonType(input.ButtonLeft), ByID},
|
|
|
|
{"button2", "bar-middle", ButtonType(input.ButtonMiddle), ByID},
|
|
|
|
{"input3", "bar-right", ButtonType(input.ButtonRight), ByID},
|
|
|
|
{"input3", "bar-right", ButtonModifiers(input.ModifierNone), ByID},
|
|
|
|
{"input3", "bar-right", Button("right"), ByID},
|
2017-02-23 08:23:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for i, test := range tests {
|
|
|
|
t.Run(fmt.Sprintf("test %d", i), func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
c := testAllocate(t, "input.html")
|
|
|
|
defer c.Release()
|
|
|
|
|
2017-02-25 07:34:53 +00:00
|
|
|
var err error
|
2017-02-23 08:23:37 +00:00
|
|
|
var nodes []*cdp.Node
|
2017-02-25 07:34:53 +00:00
|
|
|
err = c.Run(defaultContext, Nodes(test.sel, &nodes, test.by))
|
2017-02-23 08:23:37 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("got error: %v", err)
|
|
|
|
}
|
|
|
|
if len(nodes) != 1 {
|
|
|
|
t.Fatalf("expected nodes to have exactly 1 element, got: %d", len(nodes))
|
|
|
|
}
|
|
|
|
|
|
|
|
err = c.Run(defaultContext, MouseClickNode(nodes[0], test.opt))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("got error: %v", err)
|
|
|
|
}
|
|
|
|
|
2017-06-18 02:32:12 +00:00
|
|
|
time.Sleep(50 * time.Millisecond)
|
2017-02-23 08:23:37 +00:00
|
|
|
|
|
|
|
var value string
|
|
|
|
err = c.Run(defaultContext, Value("#input3", &value, ByID))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("got error: %v", err)
|
|
|
|
}
|
|
|
|
if value != test.exp {
|
|
|
|
t.Fatalf("expected to have value %s, got: %s", test.exp, value)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMouseClickOffscreenNode(t *testing.T) {
|
2019-02-21 12:55:21 +00:00
|
|
|
t.Parallel()
|
|
|
|
|
2017-02-23 08:23:37 +00:00
|
|
|
tests := []struct {
|
|
|
|
sel string
|
|
|
|
exp int
|
|
|
|
by QueryOption
|
|
|
|
}{
|
|
|
|
{"#button3", 0, ByID},
|
|
|
|
{"#button3", 2, ByID},
|
|
|
|
{"#button3", 10, ByID},
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, test := range tests {
|
|
|
|
t.Run(fmt.Sprintf("test %d", i), func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
c := testAllocate(t, "input.html")
|
|
|
|
defer c.Release()
|
|
|
|
|
2017-02-25 07:34:53 +00:00
|
|
|
var err error
|
2017-02-23 08:23:37 +00:00
|
|
|
var nodes []*cdp.Node
|
|
|
|
err = c.Run(defaultContext, Nodes(test.sel, &nodes, test.by))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("got error: %v", err)
|
|
|
|
}
|
|
|
|
if len(nodes) != 1 {
|
|
|
|
t.Fatalf("expected nodes to have exactly 1 element, got: %d", len(nodes))
|
|
|
|
}
|
|
|
|
|
2017-02-25 07:34:53 +00:00
|
|
|
var ok bool
|
2017-02-28 01:35:00 +00:00
|
|
|
err = c.Run(defaultContext, EvaluateAsDevTools(fmt.Sprintf(inViewportJS, nodes[0].FullXPath()), &ok))
|
2017-02-25 07:34:53 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("got error: %v", err)
|
|
|
|
}
|
|
|
|
if ok {
|
|
|
|
t.Fatal("expected node to be offscreen")
|
|
|
|
}
|
|
|
|
|
2017-02-23 08:23:37 +00:00
|
|
|
for i := test.exp; i > 0; i-- {
|
|
|
|
err = c.Run(defaultContext, MouseClickNode(nodes[0]))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("got error: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-18 02:32:12 +00:00
|
|
|
time.Sleep(100 * time.Millisecond)
|
2017-02-23 08:23:37 +00:00
|
|
|
|
|
|
|
var value int
|
|
|
|
err = c.Run(defaultContext, Evaluate("window.document.test_i", &value))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("got error: %v", err)
|
|
|
|
}
|
|
|
|
if value != test.exp {
|
|
|
|
t.Fatalf("expected to have value %d, got: %d", test.exp, value)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestKeyAction(t *testing.T) {
|
2019-02-21 12:55:21 +00:00
|
|
|
t.Parallel()
|
|
|
|
|
2017-02-23 08:23:37 +00:00
|
|
|
tests := []struct {
|
|
|
|
sel, exp string
|
|
|
|
by QueryOption
|
|
|
|
}{
|
|
|
|
{"#input4", "foo", ByID},
|
|
|
|
{"#input4", "foo and bar", ByID},
|
|
|
|
{"#input4", "1234567890", ByID},
|
|
|
|
{"#input4", "~!@#$%^&*()_+=[];'", ByID},
|
|
|
|
{"#input4", "你", ByID},
|
|
|
|
{"#input4", "\n\nfoo\n\nbar\n\n", ByID},
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, test := range tests {
|
|
|
|
t.Run(fmt.Sprintf("test %d", i), func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
c := testAllocate(t, "input.html")
|
|
|
|
defer c.Release()
|
|
|
|
|
2017-02-25 07:34:53 +00:00
|
|
|
var err error
|
2017-02-23 08:23:37 +00:00
|
|
|
var nodes []*cdp.Node
|
|
|
|
err = c.Run(defaultContext, Nodes(test.sel, &nodes, test.by))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("got error: %v", err)
|
|
|
|
}
|
|
|
|
if len(nodes) != 1 {
|
|
|
|
t.Fatalf("expected nodes to have exactly 1 element, got: %d", len(nodes))
|
|
|
|
}
|
|
|
|
|
|
|
|
err = c.Run(defaultContext, Focus(test.sel, test.by))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("got error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = c.Run(defaultContext, KeyAction(test.exp))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("got error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var value string
|
|
|
|
err = c.Run(defaultContext, Value(test.sel, &value, test.by))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("got error: %v", err)
|
|
|
|
}
|
|
|
|
if value != test.exp {
|
|
|
|
t.Fatalf("expected to have value %s, got: %s", test.exp, value)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestKeyActionNode(t *testing.T) {
|
2019-02-21 12:55:21 +00:00
|
|
|
t.Parallel()
|
|
|
|
|
2017-02-23 08:23:37 +00:00
|
|
|
tests := []struct {
|
|
|
|
sel, exp string
|
|
|
|
by QueryOption
|
|
|
|
}{
|
|
|
|
{"#input4", "foo", ByID},
|
|
|
|
{"#input4", "foo and bar", ByID},
|
|
|
|
{"#input4", "1234567890", ByID},
|
|
|
|
{"#input4", "~!@#$%^&*()_+=[];'", ByID},
|
|
|
|
{"#input4", "你", ByID},
|
|
|
|
{"#input4", "\n\nfoo\n\nbar\n\n", ByID},
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, test := range tests {
|
|
|
|
t.Run(fmt.Sprintf("test %d", i), func(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
c := testAllocate(t, "input.html")
|
|
|
|
defer c.Release()
|
|
|
|
|
2017-02-25 07:34:53 +00:00
|
|
|
var err error
|
2017-02-23 08:23:37 +00:00
|
|
|
var nodes []*cdp.Node
|
|
|
|
err = c.Run(defaultContext, Nodes(test.sel, &nodes, test.by))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("got error: %v", err)
|
|
|
|
}
|
|
|
|
if len(nodes) != 1 {
|
|
|
|
t.Fatalf("expected nodes to have exactly 1 element, got: %d", len(nodes))
|
|
|
|
}
|
|
|
|
|
|
|
|
err = c.Run(defaultContext, KeyActionNode(nodes[0], test.exp))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("got error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var value string
|
|
|
|
err = c.Run(defaultContext, Value(test.sel, &value, test.by))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("got error: %v", err)
|
|
|
|
}
|
|
|
|
if value != test.exp {
|
|
|
|
t.Fatalf("expected to have value %s, got: %s", test.exp, value)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|