1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
|
var utils = {
/*
* Takes a dot-notation object string and call the function
* that it points to with the correct value for 'this'.
*/
invokeCommandString: function(str, argArray) {
var components = str.split('.');
var obj = window;
for (var i = 0; i < components.length - 1; i++)
obj = obj[components[i]];
var func = obj[components.pop()];
return func.apply(obj, argArray);
},
/*
* Takes an array of XPath selectors, adds the necessary namespaces (currently only XHTML), and applies them
* to the document root. The namespaceResolver in evaluateXPath should be kept in sync with the namespaces
* here.
*/
makeXPath: function(elementArray) {
var xpath = [];
for (var i in elementArray)
xpath.push("//" + elementArray[i], "//xhtml:" + elementArray[i]);
return xpath.join(" | ");
},
evaluateXPath: function(xpath, resultType) {
function namespaceResolver(namespace) {
return namespace == "xhtml" ? "http://www.w3.org/1999/xhtml" : null;
}
return document.evaluate(xpath, document.documentElement, namespaceResolver, resultType, null);
},
/**
* Returns the first visible clientRect of an element if it exists. Otherwise it returns null.
*/
getVisibleClientRect: function(element) {
// Note: this call will be expensive if we modify the DOM in between calls.
var clientRects = element.getClientRects();
var clientRectsLength = clientRects.length;
for (var i = 0; i < clientRectsLength; i++) {
if (clientRects[i].top < 0 || clientRects[i].top >= window.innerHeight - 4 ||
clientRects[i].left < 0 || clientRects[i].left >= window.innerWidth - 4)
continue;
if (clientRects[i].width < 3 || clientRects[i].height < 3)
continue;
// eliminate invisible elements (see test_harnesses/visibility_test.html)
var computedStyle = window.getComputedStyle(element, null);
if (computedStyle.getPropertyValue('visibility') != 'visible' ||
computedStyle.getPropertyValue('display') == 'none')
continue;
return clientRects[i];
}
for (var i = 0; i < clientRectsLength; i++) {
// If the link has zero dimensions, it may be wrapping visible
// but floated elements. Check for this.
if (clientRects[i].width == 0 || clientRects[i].height == 0) {
for (var j = 0, childrenCount = element.children.length; j < childrenCount; j++) {
var computedStyle = window.getComputedStyle(element.children[j], null);
// Ignore child elements which are not floated and not absolutely positioned for parent elements with zero width/height
if (computedStyle.getPropertyValue('float') == 'none' && computedStyle.getPropertyValue('position') != 'absolute')
continue;
var childClientRect = this.getVisibleClientRect(element.children[j]);
if (childClientRect === null)
continue;
return childClientRect;
}
}
};
return null;
},
/**
* Tries to convert :str into a valid URL.
* We don't bother with escaping characters, however, as Chrome will do that for us.
*/
ensureUrl: function(str) {
// trim str
str = str.replace(/^\s+|\s+$/g, '');
// definitely not a valid URL; treat as a search query
if (str.indexOf(" ") != -1 || (str.indexOf('.') == -1 && !/^((http|https|ftp):\/\/)?localhost/.test(str)))
return "http://www.google.com/search?q=" + str;
// possibly a valid URL, but not canonical
else if (!/^(http|https|ftp|chrome):\/\//.test(str))
return "http://" + str;
// cross our fingers and hope it is valid
else
return str;
},
};
|