rainux-selenium-webdriver 0.0.17

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (192) hide show
  1. data/COPYING +204 -0
  2. data/chrome/prebuilt/Win32/Release/npchromedriver.dll +0 -0
  3. data/chrome/prebuilt/x64/Release/npchromedriver.dll +0 -0
  4. data/chrome/src/extension/background.html +9 -0
  5. data/chrome/src/extension/background.js +995 -0
  6. data/chrome/src/extension/content_script.js +1321 -0
  7. data/chrome/src/extension/icons/busy.png +0 -0
  8. data/chrome/src/extension/icons/free.png +0 -0
  9. data/chrome/src/extension/manifest-nonwin.json +19 -0
  10. data/chrome/src/extension/manifest-win.json +20 -0
  11. data/chrome/src/extension/utils.js +231 -0
  12. data/chrome/src/rb/lib/selenium/webdriver/chrome.rb +8 -0
  13. data/chrome/src/rb/lib/selenium/webdriver/chrome/bridge.rb +358 -0
  14. data/chrome/src/rb/lib/selenium/webdriver/chrome/command_executor.rb +124 -0
  15. data/chrome/src/rb/lib/selenium/webdriver/chrome/launcher.rb +135 -0
  16. data/common/src/js/abstractcommandprocessor.js +134 -0
  17. data/common/src/js/asserts.js +296 -0
  18. data/common/src/js/by.js +149 -0
  19. data/common/src/js/command.js +304 -0
  20. data/common/src/js/context.js +58 -0
  21. data/common/src/js/core/Blank.html +7 -0
  22. data/common/src/js/core/InjectedRemoteRunner.html +8 -0
  23. data/common/src/js/core/RemoteRunner.html +101 -0
  24. data/common/src/js/core/SeleniumLog.html +109 -0
  25. data/common/src/js/core/TestPrompt.html +145 -0
  26. data/common/src/js/core/TestRunner-splash.html +55 -0
  27. data/common/src/js/core/TestRunner.html +165 -0
  28. data/common/src/js/core/icons/all.png +0 -0
  29. data/common/src/js/core/icons/continue.png +0 -0
  30. data/common/src/js/core/icons/continue_disabled.png +0 -0
  31. data/common/src/js/core/icons/pause.png +0 -0
  32. data/common/src/js/core/icons/pause_disabled.png +0 -0
  33. data/common/src/js/core/icons/selected.png +0 -0
  34. data/common/src/js/core/icons/step.png +0 -0
  35. data/common/src/js/core/icons/step_disabled.png +0 -0
  36. data/common/src/js/core/lib/cssQuery/cssQuery-p.js +6 -0
  37. data/common/src/js/core/lib/cssQuery/src/cssQuery-level2.js +142 -0
  38. data/common/src/js/core/lib/cssQuery/src/cssQuery-level3.js +150 -0
  39. data/common/src/js/core/lib/cssQuery/src/cssQuery-standard.js +53 -0
  40. data/common/src/js/core/lib/cssQuery/src/cssQuery.js +356 -0
  41. data/common/src/js/core/lib/prototype.js +2006 -0
  42. data/common/src/js/core/lib/scriptaculous/builder.js +101 -0
  43. data/common/src/js/core/lib/scriptaculous/controls.js +815 -0
  44. data/common/src/js/core/lib/scriptaculous/dragdrop.js +915 -0
  45. data/common/src/js/core/lib/scriptaculous/effects.js +958 -0
  46. data/common/src/js/core/lib/scriptaculous/scriptaculous.js +47 -0
  47. data/common/src/js/core/lib/scriptaculous/slider.js +283 -0
  48. data/common/src/js/core/lib/scriptaculous/unittest.js +383 -0
  49. data/common/src/js/core/lib/snapsie.js +91 -0
  50. data/common/src/js/core/scripts/find_matching_child.js +69 -0
  51. data/common/src/js/core/scripts/htmlutils.js +8716 -0
  52. data/common/src/js/core/scripts/injection.html +72 -0
  53. data/common/src/js/core/scripts/selenium-api.js +3291 -0
  54. data/common/src/js/core/scripts/selenium-browserbot.js +2457 -0
  55. data/common/src/js/core/scripts/selenium-browserdetect.js +153 -0
  56. data/common/src/js/core/scripts/selenium-commandhandlers.js +379 -0
  57. data/common/src/js/core/scripts/selenium-executionloop.js +175 -0
  58. data/common/src/js/core/scripts/selenium-logging.js +148 -0
  59. data/common/src/js/core/scripts/selenium-remoterunner.js +695 -0
  60. data/common/src/js/core/scripts/selenium-testrunner.js +1362 -0
  61. data/common/src/js/core/scripts/selenium-version.js +5 -0
  62. data/common/src/js/core/scripts/ui-doc.html +808 -0
  63. data/common/src/js/core/scripts/ui-element.js +1644 -0
  64. data/common/src/js/core/scripts/ui-map-sample.js +979 -0
  65. data/common/src/js/core/scripts/user-extensions.js +3 -0
  66. data/common/src/js/core/scripts/user-extensions.js.sample +75 -0
  67. data/common/src/js/core/scripts/xmlextras.js +153 -0
  68. data/common/src/js/core/selenium-logo.png +0 -0
  69. data/common/src/js/core/selenium-test.css +43 -0
  70. data/common/src/js/core/selenium.css +316 -0
  71. data/common/src/js/core/xpath/dom.js +566 -0
  72. data/common/src/js/core/xpath/javascript-xpath-0.1.11.js +2816 -0
  73. data/common/src/js/core/xpath/util.js +549 -0
  74. data/common/src/js/core/xpath/xmltoken.js +149 -0
  75. data/common/src/js/core/xpath/xpath.js +2481 -0
  76. data/common/src/js/extension/README +2 -0
  77. data/common/src/js/extension/dommessenger.js +152 -0
  78. data/common/src/js/factory.js +55 -0
  79. data/common/src/js/future.js +141 -0
  80. data/common/src/js/jsunit.js +40 -0
  81. data/common/src/js/jsunit/app/css/jsUnitStyle.css +50 -0
  82. data/common/src/js/jsunit/app/css/readme +10 -0
  83. data/common/src/js/jsunit/app/emptyPage.html +11 -0
  84. data/common/src/js/jsunit/app/jsUnitCore.js +534 -0
  85. data/common/src/js/jsunit/app/jsUnitMockTimeout.js +81 -0
  86. data/common/src/js/jsunit/app/jsUnitTestManager.js +705 -0
  87. data/common/src/js/jsunit/app/jsUnitTestSuite.js +44 -0
  88. data/common/src/js/jsunit/app/jsUnitTracer.js +102 -0
  89. data/common/src/js/jsunit/app/jsUnitVersionCheck.js +59 -0
  90. data/common/src/js/jsunit/app/main-counts-errors.html +12 -0
  91. data/common/src/js/jsunit/app/main-counts-failures.html +13 -0
  92. data/common/src/js/jsunit/app/main-counts-runs.html +13 -0
  93. data/common/src/js/jsunit/app/main-counts.html +21 -0
  94. data/common/src/js/jsunit/app/main-data.html +178 -0
  95. data/common/src/js/jsunit/app/main-errors.html +23 -0
  96. data/common/src/js/jsunit/app/main-frame.html +19 -0
  97. data/common/src/js/jsunit/app/main-loader.html +45 -0
  98. data/common/src/js/jsunit/app/main-progress.html +25 -0
  99. data/common/src/js/jsunit/app/main-results.html +67 -0
  100. data/common/src/js/jsunit/app/main-status.html +13 -0
  101. data/common/src/js/jsunit/app/testContainer.html +16 -0
  102. data/common/src/js/jsunit/app/testContainerController.html +77 -0
  103. data/common/src/js/jsunit/app/xbDebug.js +306 -0
  104. data/common/src/js/jsunit/changelog.txt +60 -0
  105. data/common/src/js/jsunit/css/jsUnitStyle.css +83 -0
  106. data/common/src/js/jsunit/images/green.gif +0 -0
  107. data/common/src/js/jsunit/images/logo_jsunit.gif +0 -0
  108. data/common/src/js/jsunit/images/powerby-transparent.gif +0 -0
  109. data/common/src/js/jsunit/images/red.gif +0 -0
  110. data/common/src/js/jsunit/licenses/JDOM_license.txt +56 -0
  111. data/common/src/js/jsunit/licenses/Jetty_license.html +213 -0
  112. data/common/src/js/jsunit/licenses/MPL-1.1.txt +470 -0
  113. data/common/src/js/jsunit/licenses/gpl-2.txt +340 -0
  114. data/common/src/js/jsunit/licenses/index.html +141 -0
  115. data/common/src/js/jsunit/licenses/lgpl-2.1.txt +504 -0
  116. data/common/src/js/jsunit/licenses/mpl-tri-license-c.txt +35 -0
  117. data/common/src/js/jsunit/licenses/mpl-tri-license-html.txt +35 -0
  118. data/common/src/js/jsunit/readme.txt +19 -0
  119. data/common/src/js/jsunit/testRunner.html +167 -0
  120. data/common/src/js/jsunit/version.txt +1 -0
  121. data/common/src/js/key.js +117 -0
  122. data/common/src/js/localcommandprocessor.js +185 -0
  123. data/common/src/js/testcase.js +217 -0
  124. data/common/src/js/timing.js +89 -0
  125. data/common/src/js/webdriver.js +890 -0
  126. data/common/src/js/webelement.js +485 -0
  127. data/common/src/rb/README +30 -0
  128. data/common/src/rb/lib/selenium-webdriver.rb +1 -0
  129. data/common/src/rb/lib/selenium/webdriver.rb +67 -0
  130. data/common/src/rb/lib/selenium/webdriver/bridge_helper.rb +91 -0
  131. data/common/src/rb/lib/selenium/webdriver/child_process.rb +180 -0
  132. data/common/src/rb/lib/selenium/webdriver/core_ext/dir.rb +41 -0
  133. data/common/src/rb/lib/selenium/webdriver/driver.rb +252 -0
  134. data/common/src/rb/lib/selenium/webdriver/driver_extensions/takes_screenshot.rb +24 -0
  135. data/common/src/rb/lib/selenium/webdriver/element.rb +262 -0
  136. data/common/src/rb/lib/selenium/webdriver/error.rb +67 -0
  137. data/common/src/rb/lib/selenium/webdriver/find.rb +89 -0
  138. data/common/src/rb/lib/selenium/webdriver/keys.rb +84 -0
  139. data/common/src/rb/lib/selenium/webdriver/navigation.rb +27 -0
  140. data/common/src/rb/lib/selenium/webdriver/options.rb +50 -0
  141. data/common/src/rb/lib/selenium/webdriver/platform.rb +86 -0
  142. data/common/src/rb/lib/selenium/webdriver/target_locator.rb +70 -0
  143. data/firefox/prebuilt/Win32/Release/webdriver-firefox.dll +0 -0
  144. data/firefox/prebuilt/linux/Release/libwebdriver-firefox.so +0 -0
  145. data/firefox/prebuilt/linux/Release/x_ignore_nofocus.so +0 -0
  146. data/firefox/prebuilt/linux64/Release/libwebdriver-firefox.so +0 -0
  147. data/firefox/prebuilt/linux64/Release/x_ignore_nofocus.so +0 -0
  148. data/firefox/prebuilt/nsICommandProcessor.xpt +0 -0
  149. data/firefox/prebuilt/nsINativeEvents.xpt +0 -0
  150. data/firefox/prebuilt/nsIResponseHandler.xpt +0 -0
  151. data/firefox/src/extension/chrome.manifest +3 -0
  152. data/firefox/src/extension/components/badCertListener.js +294 -0
  153. data/firefox/src/extension/components/context.js +37 -0
  154. data/firefox/src/extension/components/driver-component.js +127 -0
  155. data/firefox/src/extension/components/firefoxDriver.js +810 -0
  156. data/firefox/src/extension/components/json2.js +273 -0
  157. data/firefox/src/extension/components/keytest.html +554 -0
  158. data/firefox/src/extension/components/nsCommandProcessor.js +643 -0
  159. data/firefox/src/extension/components/promptService.js +208 -0
  160. data/firefox/src/extension/components/screenshooter.js +81 -0
  161. data/firefox/src/extension/components/socketListener.js +185 -0
  162. data/firefox/src/extension/components/utils.js +1263 -0
  163. data/firefox/src/extension/components/webLoadingListener.js +57 -0
  164. data/firefox/src/extension/components/webdriverserver.js +106 -0
  165. data/firefox/src/extension/components/wrappedElement.js +683 -0
  166. data/firefox/src/extension/content/fxdriver.xul +30 -0
  167. data/firefox/src/extension/content/server.js +95 -0
  168. data/firefox/src/extension/idl/nsICommandProcessor.idl +38 -0
  169. data/firefox/src/extension/idl/nsIResponseHandler.idl +34 -0
  170. data/firefox/src/extension/install.rdf +29 -0
  171. data/firefox/src/rb/lib/selenium/webdriver/firefox.rb +31 -0
  172. data/firefox/src/rb/lib/selenium/webdriver/firefox/binary.rb +107 -0
  173. data/firefox/src/rb/lib/selenium/webdriver/firefox/bridge.rb +484 -0
  174. data/firefox/src/rb/lib/selenium/webdriver/firefox/extension_connection.rb +90 -0
  175. data/firefox/src/rb/lib/selenium/webdriver/firefox/launcher.rb +155 -0
  176. data/firefox/src/rb/lib/selenium/webdriver/firefox/profile.rb +233 -0
  177. data/firefox/src/rb/lib/selenium/webdriver/firefox/profiles_ini.rb +59 -0
  178. data/firefox/src/rb/lib/selenium/webdriver/firefox/util.rb +23 -0
  179. data/jobbie/prebuilt/Win32/Release/InternetExplorerDriver.dll +0 -0
  180. data/jobbie/prebuilt/x64/Release/InternetExplorerDriver.dll +0 -0
  181. data/jobbie/src/rb/lib/selenium/webdriver/ie.rb +14 -0
  182. data/jobbie/src/rb/lib/selenium/webdriver/ie/bridge.rb +565 -0
  183. data/jobbie/src/rb/lib/selenium/webdriver/ie/lib.rb +99 -0
  184. data/jobbie/src/rb/lib/selenium/webdriver/ie/util.rb +147 -0
  185. data/remote/client/src/rb/lib/selenium/webdriver/remote.rb +16 -0
  186. data/remote/client/src/rb/lib/selenium/webdriver/remote/bridge.rb +408 -0
  187. data/remote/client/src/rb/lib/selenium/webdriver/remote/capabilities.rb +105 -0
  188. data/remote/client/src/rb/lib/selenium/webdriver/remote/commands.rb +53 -0
  189. data/remote/client/src/rb/lib/selenium/webdriver/remote/default_http_client.rb +71 -0
  190. data/remote/client/src/rb/lib/selenium/webdriver/remote/response.rb +49 -0
  191. data/remote/client/src/rb/lib/selenium/webdriver/remote/server_error.rb +32 -0
  192. metadata +303 -0
@@ -0,0 +1,2457 @@
1
+ /*
2
+ * Copyright 2004 ThoughtWorks, Inc
3
+ *
4
+ * Licensed under the Apache License, Version 2.0 (the "License");
5
+ * you may not use this file except in compliance with the License.
6
+ * You may obtain a copy of the License at
7
+ *
8
+ * http://www.apache.org/licenses/LICENSE-2.0
9
+ *
10
+ * Unless required by applicable law or agreed to in writing, software
11
+ * distributed under the License is distributed on an "AS IS" BASIS,
12
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
+ * See the License for the specific language governing permissions and
14
+ * limitations under the License.
15
+ *
16
+ */
17
+
18
+ /*
19
+ * This script provides the Javascript API to drive the test application contained within
20
+ * a Browser Window.
21
+ * TODO:
22
+ * Add support for more events (keyboard and mouse)
23
+ * Allow to switch "user-entry" mode from mouse-based to keyboard-based, firing different
24
+ * events in different modes.
25
+ */
26
+
27
+ // The window to which the commands will be sent. For example, to click on a
28
+ // popup window, first select that window, and then do a normal click command.
29
+ var BrowserBot = function(topLevelApplicationWindow) {
30
+ this.topWindow = topLevelApplicationWindow;
31
+ this.topFrame = this.topWindow;
32
+ this.baseUrl=window.location.href;
33
+
34
+ // the buttonWindow is the Selenium window
35
+ // it contains the Run/Pause buttons... this should *not* be the AUT window
36
+ this.buttonWindow = window;
37
+ this.currentWindow = this.topWindow;
38
+ this.currentWindowName = null;
39
+ this.allowNativeXpath = true;
40
+ this.xpathEvaluator = new XPathEvaluator('ajaxslt'); // change to "javascript-xpath" for the newer, faster engine
41
+
42
+ // We need to know this in advance, in case the frame closes unexpectedly
43
+ this.isSubFrameSelected = false;
44
+
45
+ this.altKeyDown = false;
46
+ this.controlKeyDown = false;
47
+ this.shiftKeyDown = false;
48
+ this.metaKeyDown = false;
49
+
50
+ this.modalDialogTest = null;
51
+ this.recordedAlerts = new Array();
52
+ this.recordedConfirmations = new Array();
53
+ this.recordedPrompts = new Array();
54
+ this.openedWindows = {};
55
+ this.nextConfirmResult = true;
56
+ this.nextPromptResult = '';
57
+ this.newPageLoaded = false;
58
+ this.pageLoadError = null;
59
+
60
+ this.ignoreResponseCode = false;
61
+ this.xhr = null;
62
+ this.abortXhr = false;
63
+ this.isXhrSent = false;
64
+ this.isXhrDone = false;
65
+ this.xhrOpenLocation = null;
66
+ this.xhrResponseCode = null;
67
+ this.xhrStatusText = null;
68
+
69
+ this.shouldHighlightLocatedElement = false;
70
+
71
+ this.uniqueId = "seleniumMarker" + new Date().getTime();
72
+ this.pollingForLoad = new Object();
73
+ this.permDeniedCount = new Object();
74
+ this.windowPollers = new Array();
75
+ // DGF for backwards compatibility
76
+ this.browserbot = this;
77
+
78
+ var self = this;
79
+
80
+ objectExtend(this, PageBot.prototype);
81
+ this._registerAllLocatorFunctions();
82
+
83
+ this.recordPageLoad = function(elementOrWindow) {
84
+ LOG.debug("Page load detected");
85
+ try {
86
+ if (elementOrWindow.location && elementOrWindow.location.href) {
87
+ LOG.debug("Page load location=" + elementOrWindow.location.href);
88
+ } else if (elementOrWindow.contentWindow && elementOrWindow.contentWindow.location && elementOrWindow.contentWindow.location.href) {
89
+ LOG.debug("Page load location=" + elementOrWindow.contentWindow.location.href);
90
+ } else {
91
+ LOG.debug("Page load location unknown, current window location=" + this.getCurrentWindow(true).location);
92
+ }
93
+ } catch (e) {
94
+ LOG.error("Caught an exception attempting to log location; this should get noticed soon!");
95
+ LOG.exception(e);
96
+ self.pageLoadError = e;
97
+ return;
98
+ }
99
+ self.newPageLoaded = true;
100
+ };
101
+
102
+ this.isNewPageLoaded = function() {
103
+ var e;
104
+
105
+ if (this.pageLoadError) {
106
+ LOG.error("isNewPageLoaded found an old pageLoadError");
107
+ e = this.pageLoadError;
108
+ this.pageLoadError = null;
109
+ throw e;
110
+ }
111
+
112
+ if (self.ignoreResponseCode) {
113
+ return self.newPageLoaded;
114
+ } else {
115
+ if (self.isXhrSent && self.isXhrDone) {
116
+ if (!((self.xhrResponseCode >= 200 && self.xhrResponseCode <= 399) || self.xhrResponseCode == 0)) {
117
+ // TODO: for IE status like: 12002, 12007, ... provide corresponding statusText messages also.
118
+ LOG.error("XHR failed with message " + self.xhrStatusText);
119
+ e = "XHR ERROR: URL = " + self.xhrOpenLocation + " Response_Code = " + self.xhrResponseCode + " Error_Message = " + self.xhrStatusText;
120
+ self.abortXhr = false;
121
+ self.isXhrSent = false;
122
+ self.isXhrDone = false;
123
+ self.xhrResponseCode = null;
124
+ self.xhrStatusText = null;
125
+ throw new SeleniumError(e);
126
+ }
127
+ }
128
+ return self.newPageLoaded && (self.isXhrSent ? (self.abortXhr || self.isXhrDone) : true);
129
+ }
130
+ };
131
+
132
+ this.setAllowNativeXPath = function(allow) {
133
+ this.xpathEvaluator.setAllowNativeXPath(allow);
134
+ };
135
+
136
+ this.setIgnoreAttributesWithoutValue = function(ignore) {
137
+ this.xpathEvaluator.setIgnoreAttributesWithoutValue(ignore);
138
+ };
139
+
140
+ this.setXPathEngine = function(engineName) {
141
+ this.xpathEvaluator.setCurrentEngine(engineName);
142
+ };
143
+
144
+ this.getXPathEngine = function() {
145
+ return this.xpathEvaluator.getCurrentEngine();
146
+ };
147
+ };
148
+
149
+ // DGF PageBot exists for backwards compatibility with old user-extensions
150
+ var PageBot = function(){};
151
+
152
+ BrowserBot.createForWindow = function(window, proxyInjectionMode) {
153
+ var browserbot;
154
+ LOG.debug('createForWindow');
155
+ LOG.debug("browserName: " + browserVersion.name);
156
+ LOG.debug("userAgent: " + navigator.userAgent);
157
+ if (browserVersion.isIE) {
158
+ browserbot = new IEBrowserBot(window);
159
+ }
160
+ else if (browserVersion.isKonqueror) {
161
+ browserbot = new KonquerorBrowserBot(window);
162
+ }
163
+ else if (browserVersion.isOpera) {
164
+ browserbot = new OperaBrowserBot(window);
165
+ }
166
+ else if (browserVersion.isSafari) {
167
+ browserbot = new SafariBrowserBot(window);
168
+ }
169
+ else {
170
+ // Use mozilla by default
171
+ browserbot = new MozillaBrowserBot(window);
172
+ }
173
+ // getCurrentWindow has the side effect of modifying it to handle page loads etc
174
+ browserbot.proxyInjectionMode = proxyInjectionMode;
175
+ browserbot.getCurrentWindow(); // for modifyWindow side effect. This is not a transparent style
176
+ return browserbot;
177
+ };
178
+
179
+ // todo: rename? This doesn't actually "do" anything.
180
+ BrowserBot.prototype.doModalDialogTest = function(test) {
181
+ this.modalDialogTest = test;
182
+ };
183
+
184
+ BrowserBot.prototype.cancelNextConfirmation = function(result) {
185
+ this.nextConfirmResult = result;
186
+ };
187
+
188
+ BrowserBot.prototype.setNextPromptResult = function(result) {
189
+ this.nextPromptResult = result;
190
+ };
191
+
192
+ BrowserBot.prototype.hasAlerts = function() {
193
+ return (this.recordedAlerts.length > 0);
194
+ };
195
+
196
+ BrowserBot.prototype.relayBotToRC = function(s) {
197
+ // DGF need to do this funny trick to see if we're in PI mode, because
198
+ // "this" might be the window, rather than the browserbot (e.g. during window.alert)
199
+ var piMode = this.proxyInjectionMode;
200
+ if (!piMode) {
201
+ if (typeof(selenium) != "undefined") {
202
+ piMode = selenium.browserbot && selenium.browserbot.proxyInjectionMode;
203
+ }
204
+ }
205
+ if (piMode) {
206
+ this.relayToRC("selenium." + s);
207
+ }
208
+ };
209
+
210
+ BrowserBot.prototype.relayToRC = function(name) {
211
+ var object = eval(name);
212
+ var s = 'state:' + serializeObject(name, object) + "\n";
213
+ sendToRC(s,"state=true");
214
+ };
215
+
216
+ BrowserBot.prototype.resetPopups = function() {
217
+ this.recordedAlerts = [];
218
+ this.recordedConfirmations = [];
219
+ this.recordedPrompts = [];
220
+ };
221
+
222
+ BrowserBot.prototype.getNextAlert = function() {
223
+ var t = this.recordedAlerts.shift();
224
+ if (t) {
225
+ t = t.replace(/\n/g, " "); // because Selenese loses \n's when retrieving text from HTML table
226
+ }
227
+ this.relayBotToRC("browserbot.recordedAlerts");
228
+ return t;
229
+ };
230
+
231
+ BrowserBot.prototype.hasConfirmations = function() {
232
+ return (this.recordedConfirmations.length > 0);
233
+ };
234
+
235
+ BrowserBot.prototype.getNextConfirmation = function() {
236
+ var t = this.recordedConfirmations.shift();
237
+ this.relayBotToRC("browserbot.recordedConfirmations");
238
+ return t;
239
+ };
240
+
241
+ BrowserBot.prototype.hasPrompts = function() {
242
+ return (this.recordedPrompts.length > 0);
243
+ };
244
+
245
+ BrowserBot.prototype.getNextPrompt = function() {
246
+ var t = this.recordedPrompts.shift();
247
+ this.relayBotToRC("browserbot.recordedPrompts");
248
+ return t;
249
+ };
250
+
251
+ /* Fire a mouse event in a browser-compatible manner */
252
+
253
+ BrowserBot.prototype.triggerMouseEvent = function(element, eventType, canBubble, clientX, clientY, button) {
254
+ clientX = clientX ? clientX : 0;
255
+ clientY = clientY ? clientY : 0;
256
+
257
+ LOG.debug("triggerMouseEvent assumes setting screenX and screenY to 0 is ok");
258
+ var screenX = 0;
259
+ var screenY = 0;
260
+
261
+ canBubble = (typeof(canBubble) == undefined) ? true : canBubble;
262
+ var evt;
263
+ if (element.fireEvent && element.ownerDocument && element.ownerDocument.createEventObject) { //IE
264
+ evt = createEventObject(element, this.controlKeyDown, this.altKeyDown, this.shiftKeyDown, this.metaKeyDown);
265
+ evt.detail = 0;
266
+ evt.button = button ? button : 1; // default will be the left mouse click ( http://www.javascriptkit.com/jsref/event.shtml )
267
+ evt.relatedTarget = null;
268
+ if (!screenX && !screenY && !clientX && !clientY && !this.controlKeyDown && !this.altKeyDown && !this.shiftKeyDown && !this.metaKeyDown) {
269
+ element.fireEvent('on' + eventType);
270
+ }
271
+ else {
272
+ evt.screenX = screenX;
273
+ evt.screenY = screenY;
274
+ evt.clientX = clientX;
275
+ evt.clientY = clientY;
276
+
277
+ // when we go this route, window.event is never set to contain the event we have just created.
278
+ // ideally we could just slide it in as follows in the try-block below, but this normally
279
+ // doesn't work. This is why I try to avoid this code path, which is only required if we need to
280
+ // set attributes on the event (e.g., clientX).
281
+ try {
282
+ window.event = evt;
283
+ }
284
+ catch(e) {
285
+ // getting an "Object does not support this action or property" error. Save the event away
286
+ // for future reference.
287
+ // TODO: is there a way to update window.event?
288
+
289
+ // work around for http://jira.openqa.org/browse/SEL-280 -- make the event available somewhere:
290
+ selenium.browserbot.getCurrentWindow().selenium_event = evt;
291
+ }
292
+ element.fireEvent('on' + eventType, evt);
293
+ }
294
+ }
295
+ else {
296
+ evt = document.createEvent('MouseEvents');
297
+ if (evt.initMouseEvent)
298
+ {
299
+ // see http://developer.mozilla.org/en/docs/DOM:event.button and
300
+ // http://developer.mozilla.org/en/docs/DOM:event.initMouseEvent for button ternary logic logic
301
+ //Safari
302
+ evt.initMouseEvent(eventType, canBubble, true, document.defaultView, 1, screenX, screenY, clientX, clientY,
303
+ this.controlKeyDown, this.altKeyDown, this.shiftKeyDown, this.metaKeyDown, button ? button : 0, null);
304
+ }
305
+ else {
306
+ LOG.warn("element doesn't have initMouseEvent; firing an event which should -- but doesn't -- have other mouse-event related attributes here, as well as controlKeyDown, altKeyDown, shiftKeyDown, metaKeyDown");
307
+ evt.initEvent(eventType, canBubble, true);
308
+
309
+ evt.shiftKey = this.shiftKeyDown;
310
+ evt.metaKey = this.metaKeyDown;
311
+ evt.altKey = this.altKeyDown;
312
+ evt.ctrlKey = this.controlKeyDown;
313
+ if(button)
314
+ {
315
+ evt.button = button;
316
+ }
317
+ }
318
+ element.dispatchEvent(evt);
319
+ }
320
+ };
321
+
322
+ BrowserBot.prototype._windowClosed = function(win) {
323
+ var c = win.closed;
324
+ if (c == null) return true;
325
+ return c;
326
+ };
327
+
328
+ BrowserBot.prototype._modifyWindow = function(win) {
329
+ // In proxyInjectionMode, have to suppress LOG calls in _modifyWindow to avoid an infinite loop
330
+ if (this._windowClosed(win)) {
331
+ if (!this.proxyInjectionMode) {
332
+ LOG.error("modifyWindow: Window was closed!");
333
+ }
334
+ return null;
335
+ }
336
+ if (!this.proxyInjectionMode) {
337
+ LOG.debug('modifyWindow ' + this.uniqueId + ":" + win[this.uniqueId]);
338
+ }
339
+ if (!win[this.uniqueId]) {
340
+ win[this.uniqueId] = 1;
341
+ this.modifyWindowToRecordPopUpDialogs(win, this);
342
+ }
343
+ // In proxyInjection mode, we have our own mechanism for detecting page loads
344
+ if (!this.proxyInjectionMode) {
345
+ this.modifySeparateTestWindowToDetectPageLoads(win);
346
+ }
347
+ if (win.frames && win.frames.length && win.frames.length > 0) {
348
+ for (var i = 0; i < win.frames.length; i++) {
349
+ try {
350
+ this._modifyWindow(win.frames[i]);
351
+ } catch (e) {} // we're just trying to be opportunistic; don't worry if this doesn't work out
352
+ }
353
+ }
354
+ return win;
355
+ };
356
+
357
+ BrowserBot.prototype.selectWindow = function(target) {
358
+ if (!target || target == "null") {
359
+ this._selectTopWindow();
360
+ return;
361
+ }
362
+ var result = target.match(/^([a-zA-Z]+)=(.*)/);
363
+ if (!result) {
364
+ this._selectWindowByWindowId(target);
365
+ return;
366
+ }
367
+ locatorType = result[1];
368
+ locatorValue = result[2];
369
+ if (locatorType == "title") {
370
+ this._selectWindowByTitle(locatorValue);
371
+ }
372
+ // TODO separate name and var into separate functions
373
+ else if (locatorType == "name") {
374
+ this._selectWindowByName(locatorValue);
375
+ } else if (locatorType == "var") {
376
+ this._selectWindowByName(locatorValue);
377
+ } else {
378
+ throw new SeleniumError("Window locator not recognized: " + locatorType);
379
+ }
380
+ };
381
+
382
+ BrowserBot.prototype.selectPopUp = function(windowId) {
383
+ if (! windowId || windowId == 'null') {
384
+ this._selectFirstNonTopWindow();
385
+ }
386
+ else {
387
+ this._selectWindowByWindowId(windowId);
388
+ }
389
+ };
390
+
391
+ BrowserBot.prototype._selectTopWindow = function() {
392
+ this.currentWindowName = null;
393
+ this.currentWindow = this.topWindow;
394
+ this.topFrame = this.topWindow;
395
+ this.isSubFrameSelected = false;
396
+ };
397
+
398
+ BrowserBot.prototype._selectWindowByWindowId = function(windowId) {
399
+ try {
400
+ this._selectWindowByName(windowId);
401
+ }
402
+ catch (e) {
403
+ this._selectWindowByTitle(windowId);
404
+ }
405
+ };
406
+
407
+ BrowserBot.prototype._selectWindowByName = function(target) {
408
+ this.currentWindow = this.getWindowByName(target, false);
409
+ this.topFrame = this.currentWindow;
410
+ this.currentWindowName = target;
411
+ this.isSubFrameSelected = false;
412
+ };
413
+
414
+ BrowserBot.prototype._selectWindowByTitle = function(target) {
415
+ var windowName = this.getWindowNameByTitle(target);
416
+ if (!windowName) {
417
+ this._selectTopWindow();
418
+ } else {
419
+ this._selectWindowByName(windowName);
420
+ }
421
+ };
422
+
423
+ BrowserBot.prototype._selectFirstNonTopWindow = function() {
424
+ var names = this.getNonTopWindowNames();
425
+ if (names.length) {
426
+ this._selectWindowByName(names[0]);
427
+ }
428
+ };
429
+
430
+ BrowserBot.prototype.selectFrame = function(target) {
431
+ var frame;
432
+
433
+ if (target.indexOf("index=") == 0) {
434
+ target = target.substr(6);
435
+ frame = this.getCurrentWindow().frames[target];
436
+ if (frame == null) {
437
+ throw new SeleniumError("Not found: frames["+target+"]");
438
+ }
439
+ if (!frame.document) {
440
+ throw new SeleniumError("frames["+target+"] is not a frame");
441
+ }
442
+ this.currentWindow = frame;
443
+ this.isSubFrameSelected = true;
444
+ }
445
+ else if (target == "relative=up" || target == "relative=parent") {
446
+ this.currentWindow = this.getCurrentWindow().parent;
447
+ this.isSubFrameSelected = (this._getFrameElement(this.currentWindow) != null);
448
+ } else if (target == "relative=top") {
449
+ this.currentWindow = this.topFrame;
450
+ this.isSubFrameSelected = false;
451
+ } else {
452
+ frame = this.findElement(target);
453
+ if (frame == null) {
454
+ throw new SeleniumError("Not found: " + target);
455
+ }
456
+ // now, did they give us a frame or a frame ELEMENT?
457
+ var match = false;
458
+ if (frame.contentWindow) {
459
+ // this must be a frame element
460
+ if (browserVersion.isHTA) {
461
+ // stupid HTA bug; can't get in the front door
462
+ target = frame.contentWindow.name;
463
+ } else {
464
+ this.currentWindow = frame.contentWindow;
465
+ this.isSubFrameSelected = true;
466
+ match = true;
467
+ }
468
+ } else if (frame.document && frame.location) {
469
+ // must be an actual window frame
470
+ this.currentWindow = frame;
471
+ this.isSubFrameSelected = true;
472
+ match = true;
473
+ }
474
+
475
+ if (!match) {
476
+ // neither, let's loop through the frame names
477
+ var win = this.getCurrentWindow();
478
+
479
+ if (win && win.frames && win.frames.length) {
480
+ for (var i = 0; i < win.frames.length; i++) {
481
+ if (win.frames[i].name == target) {
482
+ this.currentWindow = win.frames[i];
483
+ this.isSubFrameSelected = true;
484
+ match = true;
485
+ break;
486
+ }
487
+ }
488
+ }
489
+ if (!match) {
490
+ throw new SeleniumError("Not a frame: " + target);
491
+ }
492
+ }
493
+ }
494
+ // modifies the window
495
+ this.getCurrentWindow();
496
+ };
497
+
498
+ BrowserBot.prototype.doesThisFrameMatchFrameExpression = function(currentFrameString, target) {
499
+ var isDom = false;
500
+ if (target.indexOf("dom=") == 0) {
501
+ target = target.substr(4);
502
+ isDom = true;
503
+ } else if (target.indexOf("index=") == 0) {
504
+ target = "frames[" + target.substr(6) + "]";
505
+ isDom = true;
506
+ }
507
+ var t;
508
+ try {
509
+ eval("t=" + currentFrameString + "." + target);
510
+ } catch (e) {
511
+ }
512
+ var autWindow = this.browserbot.getCurrentWindow();
513
+ if (t != null) {
514
+ try {
515
+ if (t.window == autWindow) {
516
+ return true;
517
+ }
518
+ if (t.window.uniqueId == autWindow.uniqueId) {
519
+ return true;
520
+ }
521
+ return false;
522
+ } catch (permDenied) {
523
+ // DGF if the windows are incomparable, they're probably not the same...
524
+ }
525
+ }
526
+ if (isDom) {
527
+ return false;
528
+ }
529
+ var currentFrame;
530
+ eval("currentFrame=" + currentFrameString);
531
+ if (target == "relative=up") {
532
+ if (currentFrame.window.parent == autWindow) {
533
+ return true;
534
+ }
535
+ return false;
536
+ }
537
+ if (target == "relative=top") {
538
+ if (currentFrame.window.top == autWindow) {
539
+ return true;
540
+ }
541
+ return false;
542
+ }
543
+ if (currentFrame.window == autWindow.parent) {
544
+ if (autWindow.name == target) {
545
+ return true;
546
+ }
547
+ try {
548
+ var element = this.findElement(target, currentFrame.window);
549
+ if (element.contentWindow == autWindow) {
550
+ return true;
551
+ }
552
+ } catch (e) {}
553
+ }
554
+ return false;
555
+ };
556
+
557
+ BrowserBot.prototype.abortXhrRequest = function() {
558
+ if (this.ignoreResponseCode) {
559
+ LOG.debug("XHR response code being ignored. Nothing to abort.");
560
+ } else {
561
+ if (this.abortXhr == false && this.isXhrSent && !this.isXhrDone) {
562
+ LOG.info("abortXhrRequest(): aborting request");
563
+ this.abortXhr = true;
564
+ this.xhr.abort();
565
+ }
566
+ }
567
+ };
568
+
569
+ BrowserBot.prototype.onXhrStateChange = function(method) {
570
+ LOG.info("onXhrStateChange(): xhr.readyState = " + this.xhr.readyState + " method = " + method + " time = " + new Date().getTime());
571
+ if (this.xhr.readyState == 4) {
572
+
573
+ // check if the request got aborted.
574
+ if (this.abortXhr == true) {
575
+ this.xhrResponseCode = 0;
576
+ this.xhrStatusText = "Request Aborted";
577
+ this.isXhrDone = true;
578
+ return;
579
+ }
580
+
581
+ try {
582
+ if (method == "HEAD" && (this.xhr.status == 501 || this.xhr.status == 405)) {
583
+ LOG.info("onXhrStateChange(): HEAD ajax returned 501 or 405, retrying with GET");
584
+ // handle 501 response code from servers that do not support 'HEAD' method.
585
+ // send GET ajax request with range 0-1.
586
+ this.xhr = XmlHttp.create();
587
+ this.xhr.onreadystatechange = this.onXhrStateChange.bind(this, "GET");
588
+ this.xhr.open("GET", this.xhrOpenLocation, true);
589
+ this.xhr.setRequestHeader("Range", "bytes:0-1");
590
+ this.xhr.send("");
591
+ this.isXhrSent = true;
592
+ return;
593
+ }
594
+ this.xhrResponseCode = this.xhr.status;
595
+ this.xhrStatusText = this.xhr.statusText;
596
+ } catch (ex) {
597
+ LOG.info("encountered exception while reading xhrResponseCode." + ex.message);
598
+ this.xhrResponseCode = -1;
599
+ this.xhrStatusText = "Request Error";
600
+ }
601
+
602
+ this.isXhrDone = true;
603
+ }
604
+ };
605
+
606
+ BrowserBot.prototype.checkedOpen = function(target) {
607
+ var url = absolutify(target, this.baseUrl);
608
+ LOG.debug("checkedOpen(): url = " + url);
609
+ this.isXhrDone = false;
610
+ this.abortXhr = false;
611
+ this.xhrResponseCode = null;
612
+ this.xhrOpenLocation = url;
613
+ try {
614
+ this.xhr = XmlHttp.create();
615
+ } catch (ex) {
616
+ LOG.error("Your browser doesnt support Xml Http Request");
617
+ return;
618
+ }
619
+ this.xhr.onreadystatechange = this.onXhrStateChange.bind(this, "HEAD");
620
+ this.xhr.open("HEAD", url, true);
621
+ this.xhr.send("");
622
+ this.isXhrSent = true;
623
+ };
624
+
625
+ BrowserBot.prototype.openLocation = function(target) {
626
+ // We're moving to a new page - clear the current one
627
+ var win = this.getCurrentWindow();
628
+ LOG.debug("openLocation newPageLoaded = false");
629
+ this.newPageLoaded = false;
630
+ if (!this.ignoreResponseCode) {
631
+ this.checkedOpen(target);
632
+ }
633
+ this.setOpenLocation(win, target);
634
+ };
635
+
636
+ BrowserBot.prototype.openWindow = function(url, windowID) {
637
+ if (url != "") {
638
+ url = absolutify(url, this.baseUrl);
639
+ }
640
+ if (browserVersion.isHTA) {
641
+ // in HTA mode, calling .open on the window interprets the url relative to that window
642
+ // we need to absolute-ize the URL to make it consistent
643
+ var child = this.getCurrentWindow().open(url, windowID);
644
+ selenium.browserbot.openedWindows[windowID] = child;
645
+ } else {
646
+ this.getCurrentWindow().open(url, windowID);
647
+ }
648
+ };
649
+
650
+ BrowserBot.prototype.setIFrameLocation = function(iframe, location) {
651
+ iframe.src = location;
652
+ };
653
+
654
+ BrowserBot.prototype.setOpenLocation = function(win, loc) {
655
+ loc = absolutify(loc, this.baseUrl);
656
+ if (browserVersion.isHTA) {
657
+ var oldHref = win.location.href;
658
+ win.location.href = loc;
659
+ var marker = null;
660
+ try {
661
+ marker = this.isPollingForLoad(win);
662
+ if (marker && win.location[marker]) {
663
+ win.location[marker] = false;
664
+ }
665
+ } catch (e) {} // DGF don't know why, but this often fails
666
+ } else {
667
+ win.location.href = loc;
668
+ }
669
+ };
670
+
671
+ BrowserBot.prototype.getCurrentPage = function() {
672
+ return this;
673
+ };
674
+
675
+ BrowserBot.prototype.modifyWindowToRecordPopUpDialogs = function(windowToModify, browserBot) {
676
+ var self = this;
677
+
678
+ windowToModify.seleniumAlert = windowToModify.alert;
679
+
680
+ windowToModify.alert = function(alert) {
681
+ browserBot.recordedAlerts.push(alert);
682
+ self.relayBotToRC.call(self, "browserbot.recordedAlerts");
683
+ };
684
+
685
+ windowToModify.confirm = function(message) {
686
+ browserBot.recordedConfirmations.push(message);
687
+ var result = browserBot.nextConfirmResult;
688
+ browserBot.nextConfirmResult = true;
689
+ self.relayBotToRC.call(self, "browserbot.recordedConfirmations");
690
+ return result;
691
+ };
692
+
693
+ windowToModify.prompt = function(message) {
694
+ browserBot.recordedPrompts.push(message);
695
+ var result = !browserBot.nextConfirmResult ? null : browserBot.nextPromptResult;
696
+ browserBot.nextConfirmResult = true;
697
+ browserBot.nextPromptResult = '';
698
+ self.relayBotToRC.call(self, "browserbot.recordedPrompts");
699
+ return result;
700
+ };
701
+
702
+ // Keep a reference to all popup windows by name
703
+ // note that in IE the "windowName" argument must be a valid javascript identifier, it seems.
704
+ var originalOpen = windowToModify.open;
705
+ var originalOpenReference;
706
+ if (browserVersion.isHTA) {
707
+ originalOpenReference = 'selenium_originalOpen' + new Date().getTime();
708
+ windowToModify[originalOpenReference] = windowToModify.open;
709
+ }
710
+
711
+ var isHTA = browserVersion.isHTA;
712
+
713
+ var newOpen = function(url, windowName, windowFeatures, replaceFlag) {
714
+ var myOriginalOpen = originalOpen;
715
+ if (isHTA) {
716
+ myOriginalOpen = this[originalOpenReference];
717
+ }
718
+ if (windowName == "" || windowName == "_blank") {
719
+ windowName = "selenium_blank" + Math.round(100000 * Math.random());
720
+ LOG.warn("Opening window '_blank', which is not a real window name. Randomizing target to be: " + windowName);
721
+ }
722
+ var openedWindow = myOriginalOpen(url, windowName, windowFeatures, replaceFlag);
723
+ LOG.debug("window.open call intercepted; window ID (which you can use with selectWindow()) is \"" + windowName + "\"");
724
+ if (windowName!=null) {
725
+ openedWindow["seleniumWindowName"] = windowName;
726
+ }
727
+ selenium.browserbot.openedWindows[windowName] = openedWindow;
728
+ return openedWindow;
729
+ };
730
+
731
+ if (browserVersion.isHTA) {
732
+ originalOpenReference = 'selenium_originalOpen' + new Date().getTime();
733
+ newOpenReference = 'selenium_newOpen' + new Date().getTime();
734
+ var setOriginalRef = "this['" + originalOpenReference + "'] = this.open;";
735
+
736
+ if (windowToModify.eval) {
737
+ windowToModify.eval(setOriginalRef);
738
+ windowToModify.open = newOpen;
739
+ } else {
740
+ // DGF why can't I eval here? Seems like I'm querying the window at a bad time, maybe?
741
+ setOriginalRef += "this.open = this['" + newOpenReference + "'];";
742
+ windowToModify[newOpenReference] = newOpen;
743
+ windowToModify.setTimeout(setOriginalRef, 0);
744
+ }
745
+ } else {
746
+ windowToModify.open = newOpen;
747
+ }
748
+ };
749
+
750
+ /**
751
+ * Call the supplied function when a the current page unloads and a new one loads.
752
+ * This is done by polling continuously until the document changes and is fully loaded.
753
+ */
754
+ BrowserBot.prototype.modifySeparateTestWindowToDetectPageLoads = function(windowObject) {
755
+ // Since the unload event doesn't fire in Safari 1.3, we start polling immediately
756
+ if (!windowObject) {
757
+ LOG.warn("modifySeparateTestWindowToDetectPageLoads: no windowObject!");
758
+ return;
759
+ }
760
+ if (this._windowClosed(windowObject)) {
761
+ LOG.info("modifySeparateTestWindowToDetectPageLoads: windowObject was closed");
762
+ return;
763
+ }
764
+ var oldMarker = this.isPollingForLoad(windowObject);
765
+ if (oldMarker) {
766
+ LOG.debug("modifySeparateTestWindowToDetectPageLoads: already polling this window: " + oldMarker);
767
+ return;
768
+ }
769
+
770
+ var marker = 'selenium' + new Date().getTime();
771
+ LOG.debug("Starting pollForLoad (" + marker + "): " + windowObject.location);
772
+ this.pollingForLoad[marker] = true;
773
+ // if this is a frame, add a load listener, otherwise, attach a poller
774
+ var frameElement = this._getFrameElement(windowObject);
775
+ // DGF HTA mode can't attach load listeners to subframes (yuk!)
776
+ var htaSubFrame = this._isHTASubFrame(windowObject);
777
+ if (frameElement && !htaSubFrame) {
778
+ LOG.debug("modifySeparateTestWindowToDetectPageLoads: this window is a frame; attaching a load listener");
779
+ addLoadListener(frameElement, this.recordPageLoad);
780
+ frameElement[marker] = true;
781
+ frameElement["frame"+this.uniqueId] = marker;
782
+ LOG.debug("dgf this.uniqueId="+this.uniqueId);
783
+ LOG.debug("dgf marker="+marker);
784
+ LOG.debug("dgf frameElement['frame'+this.uniqueId]="+frameElement['frame'+this.uniqueId]);
785
+ frameElement[this.uniqueId] = marker;
786
+ LOG.debug("dgf frameElement[this.uniqueId]="+frameElement[this.uniqueId]);
787
+ } else {
788
+ windowObject.location[marker] = true;
789
+ windowObject[this.uniqueId] = marker;
790
+ this.pollForLoad(this.recordPageLoad, windowObject, windowObject.document, windowObject.location, windowObject.location.href, marker);
791
+ }
792
+ };
793
+
794
+ BrowserBot.prototype._isHTASubFrame = function(win) {
795
+ if (!browserVersion.isHTA) return false;
796
+ // DGF this is wrong! what if "win" isn't the selected window?
797
+ return this.isSubFrameSelected;
798
+ };
799
+
800
+ BrowserBot.prototype._getFrameElement = function(win) {
801
+ var frameElement = null;
802
+ var caught;
803
+ try {
804
+ frameElement = win.frameElement;
805
+ } catch (e) {
806
+ caught = true;
807
+ }
808
+ if (caught) {
809
+ // on IE, checking frameElement in a pop-up results in a "No such interface supported" exception
810
+ // but it might have a frame element anyway!
811
+ var parentContainsIdenticallyNamedFrame = false;
812
+ try {
813
+ parentContainsIdenticallyNamedFrame = win.parent.frames[win.name];
814
+ } catch (e) {} // this may fail if access is denied to the parent; in that case, assume it's not a pop-up
815
+
816
+ if (parentContainsIdenticallyNamedFrame) {
817
+ // it can't be a coincidence that the parent has a frame with the same name as myself!
818
+ var result;
819
+ try {
820
+ result = parentContainsIdenticallyNamedFrame.frameElement;
821
+ if (result) {
822
+ return result;
823
+ }
824
+ } catch (e) {} // it was worth a try! _getFrameElementsByName is often slow
825
+ result = this._getFrameElementByName(win.name, win.parent.document, win);
826
+ return result;
827
+ }
828
+ }
829
+ LOG.debug("_getFrameElement: frameElement="+frameElement);
830
+ if (frameElement) {
831
+ LOG.debug("frameElement.name="+frameElement.name);
832
+ }
833
+ return frameElement;
834
+ };
835
+
836
+ BrowserBot.prototype._getFrameElementByName = function(name, doc, win) {
837
+ var frames;
838
+ var frame;
839
+ var i;
840
+ frames = doc.getElementsByTagName("iframe");
841
+ for (i = 0; i < frames.length; i++) {
842
+ frame = frames[i];
843
+ if (frame.name === name) {
844
+ return frame;
845
+ }
846
+ }
847
+ frames = doc.getElementsByTagName("frame");
848
+ for (i = 0; i < frames.length; i++) {
849
+ frame = frames[i];
850
+ if (frame.name === name) {
851
+ return frame;
852
+ }
853
+ }
854
+ // DGF weird; we only call this function when we know the doc contains the frame
855
+ LOG.warn("_getFrameElementByName couldn't find a frame or iframe; checking every element for the name " + name);
856
+ return BrowserBot.prototype.locateElementByName(win.name, win.parent.document);
857
+ };
858
+
859
+
860
+ /**
861
+ * Set up a polling timer that will keep checking the readyState of the document until it's complete.
862
+ * Since we might call this before the original page is unloaded, we first check to see that the current location
863
+ * or href is different from the original one.
864
+ */
865
+ BrowserBot.prototype.pollForLoad = function(loadFunction, windowObject, originalDocument, originalLocation, originalHref, marker) {
866
+ LOG.debug("pollForLoad original (" + marker + "): " + originalHref);
867
+ try {
868
+ if (this._windowClosed(windowObject)) {
869
+ LOG.debug("pollForLoad WINDOW CLOSED (" + marker + ")");
870
+ delete this.pollingForLoad[marker];
871
+ return;
872
+ }
873
+
874
+ var isSamePage = this._isSamePage(windowObject, originalDocument, originalLocation, originalHref, marker);
875
+ var rs = this.getReadyState(windowObject, windowObject.document);
876
+
877
+ if (!isSamePage && rs == 'complete') {
878
+ var currentHref = windowObject.location.href;
879
+ LOG.debug("pollForLoad FINISHED (" + marker + "): " + rs + " (" + currentHref + ")");
880
+ delete this.pollingForLoad[marker];
881
+ this._modifyWindow(windowObject);
882
+ var newMarker = this.isPollingForLoad(windowObject);
883
+ if (!newMarker) {
884
+ LOG.debug("modifyWindow didn't start new poller: " + newMarker);
885
+ this.modifySeparateTestWindowToDetectPageLoads(windowObject);
886
+ }
887
+ newMarker = this.isPollingForLoad(windowObject);
888
+ var currentlySelectedWindow;
889
+ var currentlySelectedWindowMarker;
890
+ currentlySelectedWindow =this.getCurrentWindow(true);
891
+ currentlySelectedWindowMarker = currentlySelectedWindow[this.uniqueId];
892
+
893
+ LOG.debug("pollForLoad (" + marker + ") restarting " + newMarker);
894
+ if (/(TestRunner-splash|Blank)\.html\?start=true$/.test(currentHref)) {
895
+ LOG.debug("pollForLoad Oh, it's just the starting page. Never mind!");
896
+ } else if (currentlySelectedWindowMarker == newMarker) {
897
+ loadFunction(currentlySelectedWindow);
898
+ } else {
899
+ LOG.debug("pollForLoad page load detected in non-current window; ignoring (currentlySelected="+currentlySelectedWindowMarker+", detection in "+newMarker+")");
900
+ }
901
+ return;
902
+ }
903
+ LOG.debug("pollForLoad continue (" + marker + "): " + currentHref);
904
+ this.reschedulePoller(loadFunction, windowObject, originalDocument, originalLocation, originalHref, marker);
905
+ } catch (e) {
906
+ LOG.debug("Exception during pollForLoad; this should get noticed soon (" + e.message + ")!");
907
+ //DGF this is supposed to get logged later; log it at debug just in case
908
+ //LOG.exception(e);
909
+ this.pageLoadError = e;
910
+ }
911
+ };
912
+
913
+ BrowserBot.prototype._isSamePage = function(windowObject, originalDocument, originalLocation, originalHref, marker) {
914
+ var currentDocument = windowObject.document;
915
+ var currentLocation = windowObject.location;
916
+ var currentHref = currentLocation.href;
917
+
918
+ var sameDoc = this._isSameDocument(originalDocument, currentDocument);
919
+
920
+ var sameLoc = (originalLocation === currentLocation);
921
+
922
+ // hash marks don't meant the page has loaded, so we need to strip them off if they exist...
923
+ var currentHash = currentHref.indexOf('#');
924
+ if (currentHash > 0) {
925
+ currentHref = currentHref.substring(0, currentHash);
926
+ }
927
+ var originalHash = originalHref.indexOf('#');
928
+ if (originalHash > 0) {
929
+ originalHref = originalHref.substring(0, originalHash);
930
+ }
931
+ LOG.debug("_isSamePage: currentHref: " + currentHref);
932
+ LOG.debug("_isSamePage: originalHref: " + originalHref);
933
+
934
+ var sameHref = (originalHref === currentHref);
935
+ var markedLoc = currentLocation[marker];
936
+
937
+ if (browserVersion.isKonqueror || browserVersion.isSafari) {
938
+ // the mark disappears too early on these browsers
939
+ markedLoc = true;
940
+ }
941
+
942
+ // since this is some _very_ important logic, especially for PI and multiWindow mode, we should log all these out
943
+ LOG.debug("_isSamePage: sameDoc: " + sameDoc);
944
+ LOG.debug("_isSamePage: sameLoc: " + sameLoc);
945
+ LOG.debug("_isSamePage: sameHref: " + sameHref);
946
+ LOG.debug("_isSamePage: markedLoc: " + markedLoc);
947
+
948
+ return sameDoc && sameLoc && sameHref && markedLoc;
949
+ };
950
+
951
+ BrowserBot.prototype._isSameDocument = function(originalDocument, currentDocument) {
952
+ return originalDocument === currentDocument;
953
+ };
954
+
955
+
956
+ BrowserBot.prototype.getReadyState = function(windowObject, currentDocument) {
957
+ var rs = currentDocument.readyState;
958
+ if (rs == null) {
959
+ if ((this.buttonWindow!=null && this.buttonWindow.document.readyState == null) // not proxy injection mode (and therefore buttonWindow isn't null)
960
+ || (top.document.readyState == null)) { // proxy injection mode (and therefore everything's in the top window, but buttonWindow doesn't exist)
961
+ // uh oh! we're probably on Firefox with no readyState extension installed!
962
+ // We'll have to just take a guess as to when the document is loaded; this guess
963
+ // will never be perfect. :-(
964
+ if (typeof currentDocument.getElementsByTagName != 'undefined'
965
+ && typeof currentDocument.getElementById != 'undefined'
966
+ && ( currentDocument.getElementsByTagName('body')[0] != null
967
+ || currentDocument.body != null )) {
968
+ if (windowObject.frameElement && windowObject.location.href == "about:blank" && windowObject.frameElement.src != "about:blank") {
969
+ LOG.info("getReadyState not loaded, frame location was about:blank, but frame src = " + windowObject.frameElement.src);
970
+ return null;
971
+ }
972
+ LOG.debug("getReadyState = windowObject.frames.length = " + windowObject.frames.length);
973
+ for (var i = 0; i < windowObject.frames.length; i++) {
974
+ LOG.debug("i = " + i);
975
+ if (this.getReadyState(windowObject.frames[i], windowObject.frames[i].document) != 'complete') {
976
+ LOG.debug("getReadyState aha! the nested frame " + windowObject.frames[i].name + " wasn't ready!");
977
+ return null;
978
+ }
979
+ }
980
+
981
+ rs = 'complete';
982
+ } else {
983
+ LOG.debug("pollForLoad readyState was null and DOM appeared to not be ready yet");
984
+ }
985
+ }
986
+ }
987
+ else if (rs == "loading" && browserVersion.isIE) {
988
+ LOG.debug("pageUnloading = true!!!!");
989
+ this.pageUnloading = true;
990
+ }
991
+ LOG.debug("getReadyState returning " + rs);
992
+ return rs;
993
+ };
994
+
995
+ /** This function isn't used normally, but was the way we used to schedule pollers:
996
+ asynchronously executed autonomous units. This is deprecated, but remains here
997
+ for future reference.
998
+ */
999
+ BrowserBot.prototype.XXXreschedulePoller = function(loadFunction, windowObject, originalDocument, originalLocation, originalHref, marker) {
1000
+ var self = this;
1001
+ window.setTimeout(function() {
1002
+ self.pollForLoad(loadFunction, windowObject, originalDocument, originalLocation, originalHref, marker);
1003
+ }, 500);
1004
+ };
1005
+
1006
+ /** This function isn't used normally, but is useful for debugging asynchronous pollers
1007
+ * To enable it, rename it to "reschedulePoller", so it will override the
1008
+ * existing reschedulePoller function
1009
+ */
1010
+ BrowserBot.prototype.XXXreschedulePoller = function(loadFunction, windowObject, originalDocument, originalLocation, originalHref, marker) {
1011
+ var doc = this.buttonWindow.document;
1012
+ var button = doc.createElement("button");
1013
+ var buttonName = doc.createTextNode(marker + " - " + windowObject.name);
1014
+ button.appendChild(buttonName);
1015
+ var tools = doc.getElementById("tools");
1016
+ var self = this;
1017
+ button.onclick = function() {
1018
+ tools.removeChild(button);
1019
+ self.pollForLoad(loadFunction, windowObject, originalDocument, originalLocation, originalHref, marker);
1020
+ };
1021
+ tools.appendChild(button);
1022
+ window.setTimeout(button.onclick, 500);
1023
+ };
1024
+
1025
+ BrowserBot.prototype.reschedulePoller = function(loadFunction, windowObject, originalDocument, originalLocation, originalHref, marker) {
1026
+ var self = this;
1027
+ var pollerFunction = function() {
1028
+ self.pollForLoad(loadFunction, windowObject, originalDocument, originalLocation, originalHref, marker);
1029
+ };
1030
+ this.windowPollers.push(pollerFunction);
1031
+ };
1032
+
1033
+ BrowserBot.prototype.runScheduledPollers = function() {
1034
+ LOG.debug("runScheduledPollers");
1035
+ var oldPollers = this.windowPollers;
1036
+ this.windowPollers = new Array();
1037
+ for (var i = 0; i < oldPollers.length; i++) {
1038
+ oldPollers[i].call();
1039
+ }
1040
+ LOG.debug("runScheduledPollers DONE");
1041
+ };
1042
+
1043
+ BrowserBot.prototype.isPollingForLoad = function(win) {
1044
+ var marker;
1045
+ var frameElement = this._getFrameElement(win);
1046
+ var htaSubFrame = this._isHTASubFrame(win);
1047
+ if (frameElement && !htaSubFrame) {
1048
+ marker = frameElement["frame"+this.uniqueId];
1049
+ } else {
1050
+ marker = win[this.uniqueId];
1051
+ }
1052
+ if (!marker) {
1053
+ LOG.debug("isPollingForLoad false, missing uniqueId " + this.uniqueId + ": " + marker);
1054
+ return false;
1055
+ }
1056
+ if (!this.pollingForLoad[marker]) {
1057
+ LOG.debug("isPollingForLoad false, this.pollingForLoad[" + marker + "]: " + this.pollingForLoad[marker]);
1058
+ return false;
1059
+ }
1060
+ return marker;
1061
+ };
1062
+
1063
+ BrowserBot.prototype.getWindowByName = function(windowName, doNotModify) {
1064
+ LOG.debug("getWindowByName(" + windowName + ")");
1065
+ // First look in the map of opened windows
1066
+ var targetWindow = this.openedWindows[windowName];
1067
+ if (!targetWindow) {
1068
+ targetWindow = this.topWindow[windowName];
1069
+ }
1070
+ if (!targetWindow && windowName == "_blank") {
1071
+ for (var winName in this.openedWindows) {
1072
+ // _blank can match selenium_blank*, if it looks like it's OK (valid href, not closed)
1073
+ if (/^selenium_blank/.test(winName)) {
1074
+ targetWindow = this.openedWindows[winName];
1075
+ var ok;
1076
+ try {
1077
+ if (!this._windowClosed(targetWindow)) {
1078
+ ok = targetWindow.location.href;
1079
+ }
1080
+ } catch (e) {}
1081
+ if (ok) break;
1082
+ }
1083
+ }
1084
+ }
1085
+ if (!targetWindow) {
1086
+ throw new SeleniumError("Window does not exist. If this looks like a Selenium bug, make sure to read http://selenium-core.openqa.org/reference.html#openWindow for potential workarounds.");
1087
+ }
1088
+ if (browserVersion.isHTA) {
1089
+ try {
1090
+ targetWindow.location.href;
1091
+ } catch (e) {
1092
+ targetWindow = window.open("", targetWindow.name);
1093
+ this.openedWindows[targetWindow.name] = targetWindow;
1094
+ }
1095
+ }
1096
+ if (!doNotModify) {
1097
+ this._modifyWindow(targetWindow);
1098
+ }
1099
+ return targetWindow;
1100
+ };
1101
+
1102
+ /**
1103
+ * Find a window name from the window title.
1104
+ */
1105
+ BrowserBot.prototype.getWindowNameByTitle = function(windowTitle) {
1106
+ LOG.debug("getWindowNameByTitle(" + windowTitle + ")");
1107
+
1108
+ // First look in the map of opened windows and iterate them
1109
+ for (var windowName in this.openedWindows) {
1110
+ var targetWindow = this.openedWindows[windowName];
1111
+
1112
+ // If the target window's title is our title
1113
+ try {
1114
+ // TODO implement Pattern Matching here
1115
+ if (!this._windowClosed(targetWindow) &&
1116
+ targetWindow.document.title == windowTitle) {
1117
+ return windowName;
1118
+ }
1119
+ } catch (e) {
1120
+ // You'll often get Permission Denied errors here in IE
1121
+ // eh, if we can't read this window's title,
1122
+ // it's probably not available to us right now anyway
1123
+ }
1124
+ }
1125
+
1126
+ try {
1127
+ if (this.topWindow.document.title == windowTitle) {
1128
+ return "";
1129
+ }
1130
+ } catch (e) {} // IE Perm denied
1131
+
1132
+ throw new SeleniumError("Could not find window with title " + windowTitle);
1133
+ };
1134
+
1135
+ BrowserBot.prototype.getNonTopWindowNames = function() {
1136
+ var nonTopWindowNames = [];
1137
+
1138
+ for (var windowName in this.openedWindows) {
1139
+ var win = this.openedWindows[windowName];
1140
+ if (! this._windowClosed(win) && win != this.topWindow) {
1141
+ nonTopWindowNames.push(windowName);
1142
+ }
1143
+ }
1144
+
1145
+ return nonTopWindowNames;
1146
+ };
1147
+
1148
+ BrowserBot.prototype.getCurrentWindow = function(doNotModify) {
1149
+ if (this.proxyInjectionMode) {
1150
+ return window;
1151
+ }
1152
+ var testWindow = this.currentWindow;
1153
+ if (!doNotModify) {
1154
+ this._modifyWindow(testWindow);
1155
+ LOG.debug("getCurrentWindow newPageLoaded = false");
1156
+ this.newPageLoaded = false;
1157
+ }
1158
+ testWindow = this._handleClosedSubFrame(testWindow, doNotModify);
1159
+ return testWindow;
1160
+ };
1161
+
1162
+ /**
1163
+ * Offer a method the end-user can reliably use to retrieve the current window.
1164
+ * This should work even for windows with an XPCNativeWrapper. Returns the
1165
+ * current window object.
1166
+ */
1167
+ BrowserBot.prototype.getUserWindow = function() {
1168
+ var userWindow = this.getCurrentWindow(true);
1169
+
1170
+ if (userWindow.wrappedJSObject) {
1171
+ userWindow = userWindow.wrappedJSObject;
1172
+ }
1173
+
1174
+ return userWindow;
1175
+ };
1176
+
1177
+ BrowserBot.prototype._handleClosedSubFrame = function(testWindow, doNotModify) {
1178
+ if (this.proxyInjectionMode) {
1179
+ return testWindow;
1180
+ }
1181
+
1182
+ if (this.isSubFrameSelected) {
1183
+ var missing = true;
1184
+ if (testWindow.parent && testWindow.parent.frames && testWindow.parent.frames.length) {
1185
+ for (var i = 0; i < testWindow.parent.frames.length; i++) {
1186
+ if (testWindow.parent.frames[i] == testWindow) {
1187
+ missing = false;
1188
+ break;
1189
+ }
1190
+ }
1191
+ }
1192
+ if (missing) {
1193
+ LOG.warn("Current subframe appears to have closed; selecting top frame");
1194
+ this.selectFrame("relative=top");
1195
+ return this.getCurrentWindow(doNotModify);
1196
+ }
1197
+ } else if (this._windowClosed(testWindow)) {
1198
+ var closedError = new SeleniumError("Current window or frame is closed!");
1199
+ closedError.windowClosed = true;
1200
+ throw closedError;
1201
+ }
1202
+ return testWindow;
1203
+ };
1204
+
1205
+ BrowserBot.prototype.highlight = function (element, force) {
1206
+ if (force || this.shouldHighlightLocatedElement) {
1207
+ try {
1208
+ highlight(element);
1209
+ } catch (e) {} // DGF element highlighting is low-priority and possibly dangerous
1210
+ }
1211
+ return element;
1212
+ };
1213
+
1214
+ BrowserBot.prototype.setShouldHighlightElement = function (shouldHighlight) {
1215
+ this.shouldHighlightLocatedElement = shouldHighlight;
1216
+ };
1217
+
1218
+ /*****************************************************************/
1219
+ /* BROWSER-SPECIFIC FUNCTIONS ONLY AFTER THIS LINE */
1220
+
1221
+
1222
+ BrowserBot.prototype._registerAllLocatorFunctions = function() {
1223
+ // TODO - don't do this in the constructor - only needed once ever
1224
+ this.locationStrategies = {};
1225
+ for (var functionName in this) {
1226
+ var result = /^locateElementBy([A-Z].+)$/.exec(functionName);
1227
+ if (result != null) {
1228
+ var locatorFunction = this[functionName];
1229
+ if (typeof(locatorFunction) != 'function') {
1230
+ continue;
1231
+ }
1232
+ // Use a specified prefix in preference to one generated from
1233
+ // the function name
1234
+ var locatorPrefix = locatorFunction.prefix || result[1].toLowerCase();
1235
+ this.locationStrategies[locatorPrefix] = locatorFunction;
1236
+ }
1237
+ }
1238
+
1239
+ /**
1240
+ * Find a locator based on a prefix.
1241
+ */
1242
+ this.findElementBy = function(locatorType, locator, inDocument, inWindow) {
1243
+ var locatorFunction = this.locationStrategies[locatorType];
1244
+ if (! locatorFunction) {
1245
+ throw new SeleniumError("Unrecognised locator type: '" + locatorType + "'");
1246
+ }
1247
+ return locatorFunction.call(this, locator, inDocument, inWindow);
1248
+ };
1249
+
1250
+ /**
1251
+ * The implicit locator, that is used when no prefix is supplied.
1252
+ */
1253
+ this.locationStrategies['implicit'] = function(locator, inDocument, inWindow) {
1254
+ if (locator.startsWith('//')) {
1255
+ return this.locateElementByXPath(locator, inDocument, inWindow);
1256
+ }
1257
+ if (locator.startsWith('document.')) {
1258
+ return this.locateElementByDomTraversal(locator, inDocument, inWindow);
1259
+ }
1260
+ return this.locateElementByIdentifier(locator, inDocument, inWindow);
1261
+ };
1262
+ };
1263
+
1264
+ BrowserBot.prototype.getDocument = function() {
1265
+ return this.getCurrentWindow().document;
1266
+ };
1267
+
1268
+ BrowserBot.prototype.getTitle = function() {
1269
+ var t = this.getDocument().title;
1270
+ if (typeof(t) == "string") {
1271
+ t = t.trim();
1272
+ }
1273
+ return t;
1274
+ };
1275
+
1276
+ BrowserBot.prototype.getCookieByName = function(cookieName, doc) {
1277
+ if (!doc) doc = this.getDocument();
1278
+ var ck = doc.cookie;
1279
+ if (!ck) return null;
1280
+ var ckPairs = ck.split(/;/);
1281
+ for (var i = 0; i < ckPairs.length; i++) {
1282
+ var ckPair = ckPairs[i].trim();
1283
+ var ckNameValue = ckPair.split(/=/);
1284
+ var ckName = decodeURIComponent(ckNameValue[0]);
1285
+ if (ckName === cookieName) {
1286
+ return decodeURIComponent(ckNameValue[1]);
1287
+ }
1288
+ }
1289
+ return null;
1290
+ };
1291
+
1292
+ BrowserBot.prototype.getAllCookieNames = function(doc) {
1293
+ if (!doc) doc = this.getDocument();
1294
+ var ck = doc.cookie;
1295
+ if (!ck) return [];
1296
+ var cookieNames = [];
1297
+ var ckPairs = ck.split(/;/);
1298
+ for (var i = 0; i < ckPairs.length; i++) {
1299
+ var ckPair = ckPairs[i].trim();
1300
+ var ckNameValue = ckPair.split(/=/);
1301
+ var ckName = decodeURIComponent(ckNameValue[0]);
1302
+ cookieNames.push(ckName);
1303
+ }
1304
+ return cookieNames;
1305
+ };
1306
+
1307
+ BrowserBot.prototype.deleteCookie = function(cookieName, domain, path, doc) {
1308
+ if (!doc) doc = this.getDocument();
1309
+ var expireDateInMilliseconds = (new Date()).getTime() + (-1 * 1000);
1310
+ var cookie = cookieName + "=deleted; ";
1311
+ if (path) {
1312
+ cookie += "path=" + path + "; ";
1313
+ }
1314
+ if (domain) {
1315
+ cookie += "domain=" + domain + "; ";
1316
+ }
1317
+ cookie += "expires=" + new Date(expireDateInMilliseconds).toGMTString();
1318
+ LOG.debug("Setting cookie to: " + cookie);
1319
+ doc.cookie = cookie;
1320
+ };
1321
+
1322
+ /** Try to delete cookie, return false if it didn't work */
1323
+ BrowserBot.prototype._maybeDeleteCookie = function(cookieName, domain, path, doc) {
1324
+ this.deleteCookie(cookieName, domain, path, doc);
1325
+ return (!this.getCookieByName(cookieName, doc));
1326
+ };
1327
+
1328
+
1329
+ BrowserBot.prototype._recursivelyDeleteCookieDomains = function(cookieName, domain, path, doc) {
1330
+ var deleted = this._maybeDeleteCookie(cookieName, domain, path, doc);
1331
+ if (deleted) return true;
1332
+ var dotIndex = domain.indexOf(".");
1333
+ if (dotIndex == 0) {
1334
+ return this._recursivelyDeleteCookieDomains(cookieName, domain.substring(1), path, doc);
1335
+ } else if (dotIndex != -1) {
1336
+ return this._recursivelyDeleteCookieDomains(cookieName, domain.substring(dotIndex), path, doc);
1337
+ } else {
1338
+ // No more dots; try just not passing in a domain at all
1339
+ return this._maybeDeleteCookie(cookieName, null, path, doc);
1340
+ }
1341
+ };
1342
+
1343
+ BrowserBot.prototype._recursivelyDeleteCookie = function(cookieName, domain, path, doc) {
1344
+ var slashIndex = path.lastIndexOf("/");
1345
+ var finalIndex = path.length-1;
1346
+ if (slashIndex == finalIndex) {
1347
+ slashIndex--;
1348
+ }
1349
+ if (slashIndex != -1) {
1350
+ deleted = this._recursivelyDeleteCookie(cookieName, domain, path.substring(0, slashIndex+1), doc);
1351
+ if (deleted) return true;
1352
+ }
1353
+ return this._recursivelyDeleteCookieDomains(cookieName, domain, path, doc);
1354
+ };
1355
+
1356
+ BrowserBot.prototype.recursivelyDeleteCookie = function(cookieName, domain, path, win) {
1357
+ if (!win) win = this.getCurrentWindow();
1358
+ var doc = win.document;
1359
+ if (!domain) {
1360
+ domain = doc.domain;
1361
+ }
1362
+ if (!path) {
1363
+ path = win.location.pathname;
1364
+ }
1365
+ var deleted = this._recursivelyDeleteCookie(cookieName, "." + domain, path, doc);
1366
+ if (deleted) return;
1367
+ // Finally try a null path (Try it last because it's uncommon)
1368
+ deleted = this._recursivelyDeleteCookieDomains(cookieName, "." + domain, null, doc);
1369
+ if (deleted) return;
1370
+ throw new SeleniumError("Couldn't delete cookie " + cookieName);
1371
+ };
1372
+
1373
+ /*
1374
+ * Finds an element recursively in frames and nested frames
1375
+ * in the specified document, using various lookup protocols
1376
+ */
1377
+ BrowserBot.prototype.findElementRecursive = function(locatorType, locatorString, inDocument, inWindow) {
1378
+
1379
+ var element = this.findElementBy(locatorType, locatorString, inDocument, inWindow);
1380
+ if (element != null) {
1381
+ return element;
1382
+ }
1383
+
1384
+ for (var i = 0; i < inWindow.frames.length; i++) {
1385
+ // On some browsers, the document object is undefined for third-party
1386
+ // frames. Make sure the document is valid before continuing.
1387
+ if (inWindow.frames[i].document) {
1388
+ element = this.findElementRecursive(locatorType, locatorString, inWindow.frames[i].document, inWindow.frames[i]);
1389
+
1390
+ if (element != null) {
1391
+ return element;
1392
+ }
1393
+ }
1394
+ }
1395
+ };
1396
+
1397
+ /*
1398
+ * Finds an element on the current page, using various lookup protocols
1399
+ */
1400
+ BrowserBot.prototype.findElementOrNull = function(locator, win) {
1401
+ locator = parse_locator(locator);
1402
+
1403
+ if (win == null) {
1404
+ win = this.getCurrentWindow();
1405
+ }
1406
+ var element = this.findElementRecursive(locator.type, locator.string, win.document, win);
1407
+
1408
+ if (element != null) {
1409
+ return this.browserbot.highlight(element);
1410
+ }
1411
+
1412
+ // Element was not found by any locator function.
1413
+ return null;
1414
+ };
1415
+
1416
+ BrowserBot.prototype.findElement = function(locator, win) {
1417
+ var element = this.findElementOrNull(locator, win);
1418
+ if (element == null) throw new SeleniumError("Element " + locator + " not found");
1419
+ return element;
1420
+ };
1421
+
1422
+ /**
1423
+ * In non-IE browsers, getElementById() does not search by name. Instead, we
1424
+ * we search separately by id and name.
1425
+ */
1426
+ BrowserBot.prototype.locateElementByIdentifier = function(identifier, inDocument, inWindow) {
1427
+ // HBC - use "this" instead of "BrowserBot.prototype"; otherwise we lose
1428
+ // the non-prototype fields of the object!
1429
+ return this.locateElementById(identifier, inDocument, inWindow)
1430
+ || BrowserBot.prototype.locateElementByName(identifier, inDocument, inWindow)
1431
+ || null;
1432
+ };
1433
+
1434
+ /**
1435
+ * Find the element with id - can't rely on getElementById, coz it returns by name as well in IE..
1436
+ */
1437
+ BrowserBot.prototype.locateElementById = function(identifier, inDocument, inWindow) {
1438
+ var element = inDocument.getElementById(identifier);
1439
+ if (element && element.getAttribute('id') === identifier) {
1440
+ return element;
1441
+ }
1442
+ else if (browserVersion.isIE || browserVersion.isOpera) {
1443
+ // SEL-484
1444
+ var elements = inDocument.getElementsByTagName('*');
1445
+
1446
+ for (var i = 0, n = elements.length; i < n; ++i) {
1447
+ element = elements[i];
1448
+
1449
+ if (element.tagName.toLowerCase() == 'form') {
1450
+ if (element.attributes['id'].nodeValue == identifier) {
1451
+ return element;
1452
+ }
1453
+ }
1454
+ else if (element.getAttribute('id') == identifier) {
1455
+ return element;
1456
+ }
1457
+ }
1458
+
1459
+ return null;
1460
+ }
1461
+ else {
1462
+ return null;
1463
+ }
1464
+ };
1465
+
1466
+ /**
1467
+ * Find an element by name, refined by (optional) element-filter
1468
+ * expressions.
1469
+ */
1470
+ BrowserBot.prototype.locateElementByName = function(locator, document, inWindow) {
1471
+ var elements = document.getElementsByTagName("*");
1472
+
1473
+ var filters = locator.split(' ');
1474
+ filters[0] = 'name=' + filters[0];
1475
+
1476
+ while (filters.length) {
1477
+ var filter = filters.shift();
1478
+ elements = this.selectElements(filter, elements, 'value');
1479
+ }
1480
+
1481
+ if (elements.length > 0) {
1482
+ return elements[0];
1483
+ }
1484
+ return null;
1485
+ };
1486
+
1487
+ /**
1488
+ * Finds an element using by evaluating the specfied string.
1489
+ */
1490
+ BrowserBot.prototype.locateElementByDomTraversal = function(domTraversal, document, window) {
1491
+
1492
+ var browserbot = this.browserbot;
1493
+ var element = null;
1494
+ try {
1495
+ element = eval(domTraversal);
1496
+ } catch (e) {
1497
+ return null;
1498
+ }
1499
+
1500
+ if (!element) {
1501
+ return null;
1502
+ }
1503
+
1504
+ return element;
1505
+ };
1506
+
1507
+ BrowserBot.prototype.locateElementByDomTraversal.prefix = "dom";
1508
+
1509
+ /**
1510
+ * Finds an element identified by the xpath expression. Expressions _must_
1511
+ * begin with "//".
1512
+ */
1513
+ BrowserBot.prototype.locateElementByXPath = function(xpath, inDocument, inWindow) {
1514
+ return this.xpathEvaluator.selectSingleNode(inDocument, xpath, null,
1515
+ this._namespaceResolver);
1516
+ };
1517
+
1518
+ BrowserBot.prototype._namespaceResolver = function(prefix) {
1519
+ if (prefix == 'html' || prefix == 'xhtml' || prefix == 'x') {
1520
+ return 'http://www.w3.org/1999/xhtml';
1521
+ } else if (prefix == 'mathml') {
1522
+ return 'http://www.w3.org/1998/Math/MathML';
1523
+ } else {
1524
+ throw new Error("Unknown namespace: " + prefix + ".");
1525
+ }
1526
+ };
1527
+
1528
+ /**
1529
+ * Returns the number of xpath results.
1530
+ */
1531
+ BrowserBot.prototype.evaluateXPathCount = function(xpath, inDocument) {
1532
+ return this.xpathEvaluator.countNodes(inDocument, xpath, null,
1533
+ this._namespaceResolver);
1534
+ };
1535
+
1536
+ /**
1537
+ * Finds a link element with text matching the expression supplied. Expressions must
1538
+ * begin with "link:".
1539
+ */
1540
+ BrowserBot.prototype.locateElementByLinkText = function(linkText, inDocument, inWindow) {
1541
+ var links = inDocument.getElementsByTagName('a');
1542
+ for (var i = 0; i < links.length; i++) {
1543
+ var element = links[i];
1544
+ if (PatternMatcher.matches(linkText, getText(element))) {
1545
+ return element;
1546
+ }
1547
+ }
1548
+ return null;
1549
+ };
1550
+
1551
+ BrowserBot.prototype.locateElementByLinkText.prefix = "link";
1552
+
1553
+ /**
1554
+ * Returns an attribute based on an attribute locator. This is made up of an element locator
1555
+ * suffixed with @attribute-name.
1556
+ */
1557
+ BrowserBot.prototype.findAttribute = function(locator) {
1558
+ // Split into locator + attributeName
1559
+ var attributePos = locator.lastIndexOf("@");
1560
+ var elementLocator = locator.slice(0, attributePos);
1561
+ var attributeName = locator.slice(attributePos + 1);
1562
+
1563
+ // Find the element.
1564
+ var element = this.findElement(elementLocator);
1565
+
1566
+ // Handle missing "class" attribute in IE.
1567
+ if (browserVersion.isIE && attributeName == "class") {
1568
+ attributeName = "className";
1569
+ }
1570
+
1571
+ // Get the attribute value.
1572
+ var attributeValue = element.getAttribute(attributeName);
1573
+
1574
+ // IE returns an object for the "style" attribute
1575
+ if (attributeName == 'style' && typeof(attributeValue) != 'string') {
1576
+ attributeValue = attributeValue.cssText;
1577
+ }
1578
+
1579
+ return attributeValue ? attributeValue.toString() : null;
1580
+ };
1581
+
1582
+ /*
1583
+ * Select the specified option and trigger the relevant events of the element.
1584
+ */
1585
+ BrowserBot.prototype.selectOption = function(element, optionToSelect) {
1586
+ triggerEvent(element, 'focus', false);
1587
+ var changed = false;
1588
+ for (var i = 0; i < element.options.length; i++) {
1589
+ var option = element.options[i];
1590
+ if (option.selected && option != optionToSelect) {
1591
+ option.selected = false;
1592
+ changed = true;
1593
+ }
1594
+ else if (!option.selected && option == optionToSelect) {
1595
+ option.selected = true;
1596
+ changed = true;
1597
+ }
1598
+ }
1599
+
1600
+ if (changed) {
1601
+ triggerEvent(element, 'change', true);
1602
+ }
1603
+ };
1604
+
1605
+ /*
1606
+ * Select the specified option and trigger the relevant events of the element.
1607
+ */
1608
+ BrowserBot.prototype.addSelection = function(element, option) {
1609
+ this.checkMultiselect(element);
1610
+ triggerEvent(element, 'focus', false);
1611
+ if (!option.selected) {
1612
+ option.selected = true;
1613
+ triggerEvent(element, 'change', true);
1614
+ }
1615
+ };
1616
+
1617
+ /*
1618
+ * Select the specified option and trigger the relevant events of the element.
1619
+ */
1620
+ BrowserBot.prototype.removeSelection = function(element, option) {
1621
+ this.checkMultiselect(element);
1622
+ triggerEvent(element, 'focus', false);
1623
+ if (option.selected) {
1624
+ option.selected = false;
1625
+ triggerEvent(element, 'change', true);
1626
+ }
1627
+ };
1628
+
1629
+ BrowserBot.prototype.checkMultiselect = function(element) {
1630
+ if (!element.multiple)
1631
+ {
1632
+ throw new SeleniumError("Not a multi-select");
1633
+ }
1634
+
1635
+ };
1636
+
1637
+ BrowserBot.prototype.replaceText = function(element, stringValue) {
1638
+ triggerEvent(element, 'focus', false);
1639
+ triggerEvent(element, 'select', true);
1640
+ var maxLengthAttr = element.getAttribute("maxLength");
1641
+ var actualValue = stringValue;
1642
+ if (maxLengthAttr != null) {
1643
+ var maxLength = parseInt(maxLengthAttr);
1644
+ if (stringValue.length > maxLength) {
1645
+ actualValue = stringValue.substr(0, maxLength);
1646
+ }
1647
+ }
1648
+
1649
+ if (getTagName(element) == "body") {
1650
+ if (element.ownerDocument && element.ownerDocument.designMode) {
1651
+ var designMode = new String(element.ownerDocument.designMode).toLowerCase();
1652
+ if (designMode == "on") {
1653
+ // this must be a rich text control!
1654
+ element.innerHTML = actualValue;
1655
+ }
1656
+ }
1657
+ } else {
1658
+ element.value = actualValue;
1659
+ }
1660
+ // DGF this used to be skipped in chrome URLs, but no longer. Is xpcnativewrappers to blame?
1661
+ try {
1662
+ triggerEvent(element, 'change', true);
1663
+ } catch (e) {}
1664
+ };
1665
+
1666
+ BrowserBot.prototype.submit = function(formElement) {
1667
+ var actuallySubmit = true;
1668
+ this._modifyElementTarget(formElement);
1669
+ if (formElement.onsubmit) {
1670
+ if (browserVersion.isHTA) {
1671
+ // run the code in the correct window so alerts are handled correctly even in HTA mode
1672
+ var win = this.browserbot.getCurrentWindow();
1673
+ var now = new Date().getTime();
1674
+ var marker = 'marker' + now;
1675
+ win[marker] = formElement;
1676
+ win.setTimeout("var actuallySubmit = "+marker+".onsubmit();" +
1677
+ "if (actuallySubmit) { " +
1678
+ marker+".submit(); " +
1679
+ "if ("+marker+".target && !/^_/.test("+marker+".target)) {"+
1680
+ "window.open('', "+marker+".target);"+
1681
+ "}"+
1682
+ "};"+
1683
+ marker+"=null", 0);
1684
+ // pause for up to 2s while this command runs
1685
+ var terminationCondition = function () {
1686
+ return !win[marker];
1687
+ };
1688
+ return Selenium.decorateFunctionWithTimeout(terminationCondition, 2000);
1689
+ } else {
1690
+ actuallySubmit = formElement.onsubmit();
1691
+ if (actuallySubmit) {
1692
+ formElement.submit();
1693
+ if (formElement.target && !/^_/.test(formElement.target)) {
1694
+ this.browserbot.openWindow('', formElement.target);
1695
+ }
1696
+ }
1697
+ }
1698
+ } else {
1699
+ formElement.submit();
1700
+ }
1701
+ };
1702
+
1703
+ BrowserBot.prototype.clickElement = function(element, clientX, clientY) {
1704
+ this._fireEventOnElement("click", element, clientX, clientY);
1705
+ };
1706
+
1707
+ BrowserBot.prototype.doubleClickElement = function(element, clientX, clientY) {
1708
+ this._fireEventOnElement("dblclick", element, clientX, clientY);
1709
+ };
1710
+
1711
+ // The contextmenu event is fired when the user right-clicks to open the context menu
1712
+ BrowserBot.prototype.contextMenuOnElement = function(element, clientX, clientY) {
1713
+ this._fireEventOnElement("contextmenu", element, clientX, clientY);
1714
+ };
1715
+
1716
+ BrowserBot.prototype._modifyElementTarget = function(element) {
1717
+ if (element.target) {
1718
+ if (element.target == "_blank" || /^selenium_blank/.test(element.target) ) {
1719
+ var tagName = getTagName(element);
1720
+ if (tagName == "a" || tagName == "form") {
1721
+ var newTarget = "selenium_blank" + Math.round(100000 * Math.random());
1722
+ LOG.warn("Link has target '_blank', which is not supported in Selenium! Randomizing target to be: " + newTarget);
1723
+ this.browserbot.openWindow('', newTarget);
1724
+ element.target = newTarget;
1725
+ }
1726
+ }
1727
+ }
1728
+ };
1729
+
1730
+
1731
+ BrowserBot.prototype._handleClickingImagesInsideLinks = function(targetWindow, element) {
1732
+ var itrElement = element;
1733
+ while (itrElement != null) {
1734
+ if (itrElement.href) {
1735
+ targetWindow.location.href = itrElement.href;
1736
+ break;
1737
+ }
1738
+ itrElement = itrElement.parentNode;
1739
+ }
1740
+ };
1741
+
1742
+ BrowserBot.prototype._getTargetWindow = function(element) {
1743
+ var targetWindow = element.ownerDocument.defaultView;
1744
+ if (element.target) {
1745
+ targetWindow = this._getFrameFromGlobal(element.target);
1746
+ }
1747
+ return targetWindow;
1748
+ };
1749
+
1750
+ BrowserBot.prototype._getFrameFromGlobal = function(target) {
1751
+
1752
+ if (target == "_self") {
1753
+ return this.getCurrentWindow();
1754
+ }
1755
+ if (target == "_top") {
1756
+ return this.topFrame;
1757
+ } else if (target == "_parent") {
1758
+ return this.getCurrentWindow().parent;
1759
+ } else if (target == "_blank") {
1760
+ // TODO should this set cleverer window defaults?
1761
+ return this.getCurrentWindow().open('', '_blank');
1762
+ }
1763
+ var frameElement = this.findElementBy("implicit", target, this.topFrame.document, this.topFrame);
1764
+ if (frameElement) {
1765
+ return frameElement.contentWindow;
1766
+ }
1767
+ var win = this.getWindowByName(target);
1768
+ if (win) return win;
1769
+ return this.getCurrentWindow().open('', target);
1770
+ };
1771
+
1772
+
1773
+ BrowserBot.prototype.bodyText = function() {
1774
+ if (!this.getDocument().body) {
1775
+ throw new SeleniumError("Couldn't access document.body. Is this HTML page fully loaded?");
1776
+ }
1777
+ return getText(this.getDocument().body);
1778
+ };
1779
+
1780
+ BrowserBot.prototype.getAllButtons = function() {
1781
+ var elements = this.getDocument().getElementsByTagName('input');
1782
+ var result = [];
1783
+
1784
+ for (var i = 0; i < elements.length; i++) {
1785
+ if (elements[i].type == 'button' || elements[i].type == 'submit' || elements[i].type == 'reset') {
1786
+ result.push(elements[i].id);
1787
+ }
1788
+ }
1789
+
1790
+ return result;
1791
+ };
1792
+
1793
+
1794
+ BrowserBot.prototype.getAllFields = function() {
1795
+ var elements = this.getDocument().getElementsByTagName('input');
1796
+ var result = [];
1797
+
1798
+ for (var i = 0; i < elements.length; i++) {
1799
+ if (elements[i].type == 'text') {
1800
+ result.push(elements[i].id);
1801
+ }
1802
+ }
1803
+
1804
+ return result;
1805
+ };
1806
+
1807
+ BrowserBot.prototype.getAllLinks = function() {
1808
+ var elements = this.getDocument().getElementsByTagName('a');
1809
+ var result = [];
1810
+
1811
+ for (var i = 0; i < elements.length; i++) {
1812
+ result.push(elements[i].id);
1813
+ }
1814
+
1815
+ return result;
1816
+ };
1817
+
1818
+ function isDefined(value) {
1819
+ return typeof(value) != undefined;
1820
+ };
1821
+
1822
+ BrowserBot.prototype.goBack = function() {
1823
+ this.getCurrentWindow().history.back();
1824
+ };
1825
+
1826
+ BrowserBot.prototype.goForward = function() {
1827
+ this.getCurrentWindow().history.forward();
1828
+ };
1829
+
1830
+ BrowserBot.prototype.close = function() {
1831
+ if (browserVersion.isIE) {
1832
+ // fix "do you want to close this window" warning in IE
1833
+ // You can only close windows that you have opened.
1834
+ // So, let's "open" it.
1835
+ try {
1836
+ this.topFrame.name=new Date().getTime();
1837
+ window.open("", this.topFrame.name, "");
1838
+ this.topFrame.close();
1839
+ return;
1840
+ } catch (e) {}
1841
+ }
1842
+ if (browserVersion.isChrome || browserVersion.isSafari || browserVersion.isOpera) {
1843
+ this.topFrame.close();
1844
+ } else {
1845
+ this.getCurrentWindow().eval("window.top.close();");
1846
+ }
1847
+ };
1848
+
1849
+ BrowserBot.prototype.refresh = function() {
1850
+ this.getCurrentWindow().location.reload(true);
1851
+ };
1852
+
1853
+ /**
1854
+ * Refine a list of elements using a filter.
1855
+ */
1856
+ BrowserBot.prototype.selectElementsBy = function(filterType, filter, elements) {
1857
+ var filterFunction = BrowserBot.filterFunctions[filterType];
1858
+ if (! filterFunction) {
1859
+ throw new SeleniumError("Unrecognised element-filter type: '" + filterType + "'");
1860
+ }
1861
+
1862
+ return filterFunction(filter, elements);
1863
+ };
1864
+
1865
+ BrowserBot.filterFunctions = {};
1866
+
1867
+ BrowserBot.filterFunctions.name = function(name, elements) {
1868
+ var selectedElements = [];
1869
+ for (var i = 0; i < elements.length; i++) {
1870
+ if (elements[i].name === name) {
1871
+ selectedElements.push(elements[i]);
1872
+ }
1873
+ }
1874
+ return selectedElements;
1875
+ };
1876
+
1877
+ BrowserBot.filterFunctions.value = function(value, elements) {
1878
+ var selectedElements = [];
1879
+ for (var i = 0; i < elements.length; i++) {
1880
+ if (elements[i].value === value) {
1881
+ selectedElements.push(elements[i]);
1882
+ }
1883
+ }
1884
+ return selectedElements;
1885
+ };
1886
+
1887
+ BrowserBot.filterFunctions.index = function(index, elements) {
1888
+ index = Number(index);
1889
+ if (isNaN(index) || index < 0) {
1890
+ throw new SeleniumError("Illegal Index: " + index);
1891
+ }
1892
+ if (elements.length <= index) {
1893
+ throw new SeleniumError("Index out of range: " + index);
1894
+ }
1895
+ return [elements[index]];
1896
+ };
1897
+
1898
+ BrowserBot.prototype.selectElements = function(filterExpr, elements, defaultFilterType) {
1899
+
1900
+ var filterType = (defaultFilterType || 'value');
1901
+
1902
+ // If there is a filter prefix, use the specified strategy
1903
+ var result = filterExpr.match(/^([A-Za-z]+)=(.+)/);
1904
+ if (result) {
1905
+ filterType = result[1].toLowerCase();
1906
+ filterExpr = result[2];
1907
+ }
1908
+
1909
+ return this.selectElementsBy(filterType, filterExpr, elements);
1910
+ };
1911
+
1912
+ /**
1913
+ * Find an element by class
1914
+ */
1915
+ BrowserBot.prototype.locateElementByClass = function(locator, document) {
1916
+ return elementFindFirstMatchingChild(document,
1917
+ function(element) {
1918
+ return element.className == locator;
1919
+ }
1920
+ );
1921
+ };
1922
+
1923
+ /**
1924
+ * Find an element by alt
1925
+ */
1926
+ BrowserBot.prototype.locateElementByAlt = function(locator, document) {
1927
+ return elementFindFirstMatchingChild(document,
1928
+ function(element) {
1929
+ return element.alt == locator;
1930
+ }
1931
+ );
1932
+ };
1933
+
1934
+ /**
1935
+ * Find an element by css selector
1936
+ */
1937
+ BrowserBot.prototype.locateElementByCss = function(locator, document) {
1938
+ var elements = eval_css(locator, document);
1939
+ if (elements.length != 0)
1940
+ return elements[0];
1941
+ return null;
1942
+ };
1943
+
1944
+ /**
1945
+ * This function is responsible for mapping a UI specifier string to an element
1946
+ * on the page, and returning it. If no element is found, null is returned.
1947
+ * Returning null on failure to locate the element is part of the undocumented
1948
+ * API for locator strategies.
1949
+ */
1950
+ BrowserBot.prototype.locateElementByUIElement = function(locator, inDocument) {
1951
+ // offset locators are delimited by "->", which is much simpler than the
1952
+ // previous scheme involving detecting the close-paren.
1953
+ var locators = locator.split(/->/, 2);
1954
+
1955
+ var locatedElement = null;
1956
+ var pageElements = UIMap.getInstance()
1957
+ .getPageElements(locators[0], inDocument);
1958
+
1959
+ if (locators.length > 1) {
1960
+ for (var i = 0; i < pageElements.length; ++i) {
1961
+ var locatedElements = eval_locator(locators[1], inDocument,
1962
+ pageElements[i]);
1963
+ if (locatedElements.length) {
1964
+ locatedElement = locatedElements[0];
1965
+ break;
1966
+ }
1967
+ }
1968
+ }
1969
+ else if (pageElements.length) {
1970
+ locatedElement = pageElements[0];
1971
+ }
1972
+
1973
+ return locatedElement;
1974
+ };
1975
+
1976
+ BrowserBot.prototype.locateElementByUIElement.prefix = 'ui';
1977
+
1978
+ // define a function used to compare the result of a close UI element
1979
+ // match with the actual interacted element. If they are close enough
1980
+ // according to the heuristic, consider them a match.
1981
+ /**
1982
+ * A heuristic function for comparing a node with a target node. Typically the
1983
+ * node is specified in a UI element definition, while the target node is
1984
+ * returned by the recorder as the leaf element which had some event enacted
1985
+ * upon it. This particular heuristic covers the case where the anchor element
1986
+ * contains other inline tags, such as "em" or "img".
1987
+ *
1988
+ * @param node the node being compared to the target node
1989
+ * @param target the target node
1990
+ * @return true if node equals target, or if node is a link
1991
+ * element and target is its descendant, or if node has
1992
+ * an onclick attribute and target is its descendant.
1993
+ * False otherwise.
1994
+ */
1995
+ BrowserBot.prototype.locateElementByUIElement.is_fuzzy_match = function(node, target) {
1996
+ try {
1997
+ var isMatch = (
1998
+ (node == target) ||
1999
+ ((node.nodeName == 'A' || node.onclick) && is_ancestor(node, target))
2000
+ );
2001
+ return isMatch;
2002
+ }
2003
+ catch (e) {
2004
+ return false;
2005
+ }
2006
+ };
2007
+
2008
+ /*****************************************************************/
2009
+ /* BROWSER-SPECIFIC FUNCTIONS ONLY AFTER THIS LINE */
2010
+
2011
+ function MozillaBrowserBot(frame) {
2012
+ BrowserBot.call(this, frame);
2013
+ };
2014
+ objectExtend(MozillaBrowserBot.prototype, BrowserBot.prototype);
2015
+
2016
+ function KonquerorBrowserBot(frame) {
2017
+ BrowserBot.call(this, frame);
2018
+ };
2019
+ objectExtend(KonquerorBrowserBot.prototype, BrowserBot.prototype);
2020
+
2021
+ KonquerorBrowserBot.prototype.setIFrameLocation = function(iframe, location) {
2022
+ // Window doesn't fire onload event when setting src to the current value,
2023
+ // so we set it to blank first.
2024
+ iframe.src = "about:blank";
2025
+ iframe.src = location;
2026
+ };
2027
+
2028
+ KonquerorBrowserBot.prototype.setOpenLocation = function(win, loc) {
2029
+ // Window doesn't fire onload event when setting src to the current value,
2030
+ // so we just refresh in that case instead.
2031
+ loc = absolutify(loc, this.baseUrl);
2032
+ loc = canonicalize(loc);
2033
+ var startUrl = win.location.href;
2034
+ if ("about:blank" != win.location.href) {
2035
+ var startLoc = parseUrl(win.location.href);
2036
+ startLoc.hash = null;
2037
+ startUrl = reassembleLocation(startLoc);
2038
+ }
2039
+ LOG.debug("startUrl="+startUrl);
2040
+ LOG.debug("win.location.href="+win.location.href);
2041
+ LOG.debug("loc="+loc);
2042
+ if (startUrl == loc) {
2043
+ LOG.debug("opening exact same location");
2044
+ this.refresh();
2045
+ } else {
2046
+ LOG.debug("locations differ");
2047
+ win.location.href = loc;
2048
+ }
2049
+ // force the current polling thread to detect a page load
2050
+ var marker = this.isPollingForLoad(win);
2051
+ if (marker) {
2052
+ delete win.location[marker];
2053
+ }
2054
+ };
2055
+
2056
+ KonquerorBrowserBot.prototype._isSameDocument = function(originalDocument, currentDocument) {
2057
+ // under Konqueror, there may be this case:
2058
+ // originalDocument and currentDocument are different objects
2059
+ // while their location are same.
2060
+ if (originalDocument) {
2061
+ return originalDocument.location == currentDocument.location;
2062
+ } else {
2063
+ return originalDocument === currentDocument;
2064
+ }
2065
+ };
2066
+
2067
+ function SafariBrowserBot(frame) {
2068
+ BrowserBot.call(this, frame);
2069
+ }
2070
+ objectExtend(SafariBrowserBot.prototype, BrowserBot.prototype);
2071
+
2072
+ SafariBrowserBot.prototype.setIFrameLocation = KonquerorBrowserBot.prototype.setIFrameLocation;
2073
+ SafariBrowserBot.prototype.setOpenLocation = KonquerorBrowserBot.prototype.setOpenLocation;
2074
+
2075
+
2076
+ function OperaBrowserBot(frame) {
2077
+ BrowserBot.call(this, frame);
2078
+ };
2079
+ objectExtend(OperaBrowserBot.prototype, BrowserBot.prototype);
2080
+ OperaBrowserBot.prototype.setIFrameLocation = function(iframe, location) {
2081
+ if (iframe.src == location) {
2082
+ iframe.src = location + '?reload';
2083
+ } else {
2084
+ iframe.src = location;
2085
+ }
2086
+ };
2087
+
2088
+ function IEBrowserBot(frame) {
2089
+ BrowserBot.call(this, frame);
2090
+ };
2091
+ objectExtend(IEBrowserBot.prototype, BrowserBot.prototype);
2092
+
2093
+ IEBrowserBot.prototype._handleClosedSubFrame = function(testWindow, doNotModify) {
2094
+ if (this.proxyInjectionMode) {
2095
+ return testWindow;
2096
+ }
2097
+
2098
+ try {
2099
+ testWindow.location.href;
2100
+ this.permDenied = 0;
2101
+ } catch (e) {
2102
+ this.permDenied++;
2103
+ }
2104
+ if (this._windowClosed(testWindow) || this.permDenied > 4) {
2105
+ if (this.isSubFrameSelected) {
2106
+ LOG.warn("Current subframe appears to have closed; selecting top frame");
2107
+ this.selectFrame("relative=top");
2108
+ return this.getCurrentWindow(doNotModify);
2109
+ } else {
2110
+ var closedError = new SeleniumError("Current window or frame is closed!");
2111
+ closedError.windowClosed = true;
2112
+ throw closedError;
2113
+ }
2114
+ }
2115
+ return testWindow;
2116
+ };
2117
+
2118
+ IEBrowserBot.prototype.modifyWindowToRecordPopUpDialogs = function(windowToModify, browserBot) {
2119
+ BrowserBot.prototype.modifyWindowToRecordPopUpDialogs(windowToModify, browserBot);
2120
+
2121
+ // we will call the previous version of this method from within our own interception
2122
+ oldShowModalDialog = windowToModify.showModalDialog;
2123
+
2124
+ windowToModify.showModalDialog = function(url, args, features) {
2125
+ // Get relative directory to where TestRunner.html lives
2126
+ // A risky assumption is that the user's TestRunner is named TestRunner.html
2127
+ var doc_location = document.location.toString();
2128
+ var end_of_base_ref = doc_location.indexOf('TestRunner.html');
2129
+ var base_ref = doc_location.substring(0, end_of_base_ref);
2130
+ var runInterval = '';
2131
+
2132
+ // Only set run interval if options is defined
2133
+ if (typeof(window.runOptions) != 'undefined') {
2134
+ runInterval = "&runInterval=" + runOptions.runInterval;
2135
+ }
2136
+
2137
+ var testRunnerURL = "TestRunner.html?auto=true&singletest="
2138
+ + escape(browserBot.modalDialogTest)
2139
+ + "&autoURL="
2140
+ + escape(url)
2141
+ + runInterval;
2142
+ var fullURL = base_ref + testRunnerURL;
2143
+ browserBot.modalDialogTest = null;
2144
+
2145
+ // If using proxy injection mode
2146
+ if (this.proxyInjectionMode) {
2147
+ var sessionId = runOptions.getSessionId();
2148
+ if (sessionId == undefined) {
2149
+ sessionId = injectedSessionId;
2150
+ }
2151
+ if (sessionId != undefined) {
2152
+ LOG.debug("Invoking showModalDialog and injecting URL " + fullURL);
2153
+ }
2154
+ fullURL = url;
2155
+ }
2156
+ var returnValue = oldShowModalDialog(fullURL, args, features);
2157
+ return returnValue;
2158
+ };
2159
+ };
2160
+
2161
+ IEBrowserBot.prototype.modifySeparateTestWindowToDetectPageLoads = function(windowObject) {
2162
+ this.pageUnloading = false;
2163
+ var self = this;
2164
+ var pageUnloadDetector = function() {
2165
+ self.pageUnloading = true;
2166
+ };
2167
+ windowObject.attachEvent("onbeforeunload", pageUnloadDetector);
2168
+ BrowserBot.prototype.modifySeparateTestWindowToDetectPageLoads.call(this, windowObject);
2169
+ };
2170
+
2171
+ IEBrowserBot.prototype.pollForLoad = function(loadFunction, windowObject, originalDocument, originalLocation, originalHref, marker) {
2172
+ LOG.debug("IEBrowserBot.pollForLoad: " + marker);
2173
+ if (!this.permDeniedCount[marker]) this.permDeniedCount[marker] = 0;
2174
+ BrowserBot.prototype.pollForLoad.call(this, loadFunction, windowObject, originalDocument, originalLocation, originalHref, marker);
2175
+ var self;
2176
+ if (this.pageLoadError) {
2177
+ if (this.pageUnloading) {
2178
+ self = this;
2179
+ LOG.debug("pollForLoad UNLOADING (" + marker + "): caught exception while firing events on unloading page: " + this.pageLoadError.message);
2180
+ this.reschedulePoller(loadFunction, windowObject, originalDocument, originalLocation, originalHref, marker);
2181
+ this.pageLoadError = null;
2182
+ return;
2183
+ } else if (((this.pageLoadError.message == "Permission denied") || (/^Access is denied/.test(this.pageLoadError.message)))
2184
+ && this.permDeniedCount[marker]++ < 8) {
2185
+ if (this.permDeniedCount[marker] > 4) {
2186
+ var canAccessThisWindow;
2187
+ var canAccessCurrentlySelectedWindow;
2188
+ try {
2189
+ windowObject.location.href;
2190
+ canAccessThisWindow = true;
2191
+ } catch (e) {}
2192
+ try {
2193
+ this.getCurrentWindow(true).location.href;
2194
+ canAccessCurrentlySelectedWindow = true;
2195
+ } catch (e) {}
2196
+ if (canAccessCurrentlySelectedWindow & !canAccessThisWindow) {
2197
+ LOG.debug("pollForLoad (" + marker + ") ABORTING: " + this.pageLoadError.message + " (" + this.permDeniedCount[marker] + "), but the currently selected window is fine");
2198
+ // returning without rescheduling
2199
+ this.pageLoadError = null;
2200
+ return;
2201
+ }
2202
+ }
2203
+
2204
+ self = this;
2205
+ LOG.debug("pollForLoad (" + marker + "): " + this.pageLoadError.message + " (" + this.permDeniedCount[marker] + "), waiting to see if it goes away");
2206
+ this.reschedulePoller(loadFunction, windowObject, originalDocument, originalLocation, originalHref, marker);
2207
+ this.pageLoadError = null;
2208
+ return;
2209
+ }
2210
+ //handy for debugging!
2211
+ //throw this.pageLoadError;
2212
+ }
2213
+ };
2214
+
2215
+ IEBrowserBot.prototype._windowClosed = function(win) {
2216
+ try {
2217
+ var c = win.closed;
2218
+ // frame windows claim to be non-closed when their parents are closed
2219
+ // but you can't access their document objects in that case
2220
+ if (!c) {
2221
+ try {
2222
+ win.document;
2223
+ } catch (de) {
2224
+ if (de.message == "Permission denied") {
2225
+ // the window is probably unloading, which means it's probably not closed yet
2226
+ return false;
2227
+ }
2228
+ else if (/^Access is denied/.test(de.message)) {
2229
+ // rare variation on "Permission denied"?
2230
+ LOG.debug("IEBrowserBot.windowClosed: got " + de.message + " (this.pageUnloading=" + this.pageUnloading + "); assuming window is unloading, probably not closed yet");
2231
+ return false;
2232
+ } else {
2233
+ // this is probably one of those frame window situations
2234
+ LOG.debug("IEBrowserBot.windowClosed: couldn't read win.document, assume closed: " + de.message + " (this.pageUnloading=" + this.pageUnloading + ")");
2235
+ return true;
2236
+ }
2237
+ }
2238
+ }
2239
+ if (c == null) {
2240
+ LOG.debug("IEBrowserBot.windowClosed: win.closed was null, assuming closed");
2241
+ return true;
2242
+ }
2243
+ return c;
2244
+ } catch (e) {
2245
+ LOG.debug("IEBrowserBot._windowClosed: Got an exception trying to read win.closed; we'll have to take a guess!");
2246
+
2247
+ if (browserVersion.isHTA) {
2248
+ if (e.message == "Permission denied") {
2249
+ // the window is probably unloading, which means it's not closed yet
2250
+ return false;
2251
+ } else {
2252
+ // there's a good chance that we've lost contact with the window object if it is closed
2253
+ return true;
2254
+ }
2255
+ } else {
2256
+ // the window is probably unloading, which means it's not closed yet
2257
+ return false;
2258
+ }
2259
+ }
2260
+ };
2261
+
2262
+ /**
2263
+ * In IE, getElementById() also searches by name - this is an optimisation for IE.
2264
+ */
2265
+ IEBrowserBot.prototype.locateElementByIdentifer = function(identifier, inDocument, inWindow) {
2266
+ return inDocument.getElementById(identifier);
2267
+ };
2268
+
2269
+ SafariBrowserBot.prototype.modifyWindowToRecordPopUpDialogs = function(windowToModify, browserBot) {
2270
+ BrowserBot.prototype.modifyWindowToRecordPopUpDialogs(windowToModify, browserBot);
2271
+
2272
+ var originalOpen = windowToModify.open;
2273
+ /*
2274
+ * Safari seems to be broken, so that when we manually trigger the onclick method
2275
+ * of a button/href, any window.open calls aren't resolved relative to the app location.
2276
+ * So here we replace the open() method with one that does resolve the url correctly.
2277
+ */
2278
+ windowToModify.open = function(url, windowName, windowFeatures, replaceFlag) {
2279
+
2280
+ if (url.startsWith("http://") || url.startsWith("https://") || url.startsWith("/")) {
2281
+ return originalOpen(url, windowName, windowFeatures, replaceFlag);
2282
+ }
2283
+
2284
+ // Reduce the current path to the directory
2285
+ var currentPath = windowToModify.location.pathname || "/";
2286
+ currentPath = currentPath.replace(/\/[^\/]*$/, "/");
2287
+
2288
+ // Remove any leading "./" from the new url.
2289
+ url = url.replace(/^\.\//, "");
2290
+
2291
+ newUrl = currentPath + url;
2292
+
2293
+ var openedWindow = originalOpen(newUrl, windowName, windowFeatures, replaceFlag);
2294
+ LOG.debug("window.open call intercepted; window ID (which you can use with selectWindow()) is \"" + windowName + "\"");
2295
+ if (windowName!=null) {
2296
+ openedWindow["seleniumWindowName"] = windowName;
2297
+ }
2298
+ return openedWindow;
2299
+ };
2300
+ };
2301
+
2302
+ MozillaBrowserBot.prototype._fireEventOnElement = function(eventType, element, clientX, clientY) {
2303
+ var win = this.getCurrentWindow();
2304
+ triggerEvent(element, 'focus', false);
2305
+
2306
+ // Add an event listener that detects if the default action has been prevented.
2307
+ // (This is caused by a javascript onclick handler returning false)
2308
+ // we capture the whole event, rather than the getPreventDefault() state at the time,
2309
+ // because we need to let the entire event bubbling and capturing to go through
2310
+ // before making a decision on whether we should force the href
2311
+ var savedEvent = null;
2312
+
2313
+ element.addEventListener(eventType, function(evt) {
2314
+ savedEvent = evt;
2315
+ }, false);
2316
+
2317
+ this._modifyElementTarget(element);
2318
+
2319
+ // Trigger the event.
2320
+ this.browserbot.triggerMouseEvent(element, eventType, true, clientX, clientY);
2321
+
2322
+ if (this._windowClosed(win)) {
2323
+ return;
2324
+ }
2325
+
2326
+ // Perform the link action if preventDefault was set.
2327
+ // In chrome URL, the link action is already executed by triggerMouseEvent.
2328
+ if (!browserVersion.isChrome && savedEvent != null && !savedEvent.getPreventDefault()) {
2329
+ var targetWindow = this.browserbot._getTargetWindow(element);
2330
+ if (element.href) {
2331
+ targetWindow.location.href = element.href;
2332
+ } else {
2333
+ this.browserbot._handleClickingImagesInsideLinks(targetWindow, element);
2334
+ }
2335
+ }
2336
+
2337
+ };
2338
+
2339
+
2340
+ OperaBrowserBot.prototype._fireEventOnElement = function(eventType, element, clientX, clientY) {
2341
+ var win = this.getCurrentWindow();
2342
+ triggerEvent(element, 'focus', false);
2343
+
2344
+ this._modifyElementTarget(element);
2345
+
2346
+ // Trigger the click event.
2347
+ this.browserbot.triggerMouseEvent(element, eventType, true, clientX, clientY);
2348
+
2349
+ if (this._windowClosed(win)) {
2350
+ return;
2351
+ }
2352
+
2353
+ };
2354
+
2355
+
2356
+ KonquerorBrowserBot.prototype._fireEventOnElement = function(eventType, element, clientX, clientY) {
2357
+ var win = this.getCurrentWindow();
2358
+ triggerEvent(element, 'focus', false);
2359
+
2360
+ this._modifyElementTarget(element);
2361
+
2362
+ if (element[eventType]) {
2363
+ element[eventType]();
2364
+ }
2365
+ else {
2366
+ this.browserbot.triggerMouseEvent(element, eventType, true, clientX, clientY);
2367
+ }
2368
+
2369
+ if (this._windowClosed(win)) {
2370
+ return;
2371
+ }
2372
+
2373
+ };
2374
+
2375
+ SafariBrowserBot.prototype._fireEventOnElement = function(eventType, element, clientX, clientY) {
2376
+ triggerEvent(element, 'focus', false);
2377
+ var wasChecked = element.checked;
2378
+
2379
+ this._modifyElementTarget(element);
2380
+
2381
+ // For form element it is simple.
2382
+ if (element[eventType]) {
2383
+ element[eventType]();
2384
+ }
2385
+ // For links and other elements, event emulation is required.
2386
+ else {
2387
+ var targetWindow = this.browserbot._getTargetWindow(element);
2388
+ // todo: deal with anchors?
2389
+ this.browserbot.triggerMouseEvent(element, eventType, true, clientX, clientY);
2390
+
2391
+ }
2392
+
2393
+ };
2394
+
2395
+ SafariBrowserBot.prototype.refresh = function() {
2396
+ var win = this.getCurrentWindow();
2397
+ if (win.location.hash) {
2398
+ // DGF Safari refuses to refresh when there's a hash symbol in the URL
2399
+ win.location.hash = "";
2400
+ var actuallyReload = function() {
2401
+ win.location.reload(true);
2402
+ };
2403
+ window.setTimeout(actuallyReload, 1);
2404
+ } else {
2405
+ win.location.reload(true);
2406
+ }
2407
+ };
2408
+
2409
+ IEBrowserBot.prototype._fireEventOnElement = function(eventType, element, clientX, clientY) {
2410
+ var win = this.getCurrentWindow();
2411
+ triggerEvent(element, 'focus', false);
2412
+
2413
+ var wasChecked = element.checked;
2414
+
2415
+ // Set a flag that records if the page will unload - this isn't always accurate, because
2416
+ // <a href="javascript:alert('foo'):"> triggers the onbeforeunload event, even thought the page won't unload
2417
+ var pageUnloading = false;
2418
+ var pageUnloadDetector = function() {
2419
+ pageUnloading = true;
2420
+ };
2421
+ win.attachEvent("onbeforeunload", pageUnloadDetector);
2422
+ this._modifyElementTarget(element);
2423
+ if (element[eventType]) {
2424
+ element[eventType]();
2425
+ }
2426
+ else {
2427
+ this.browserbot.triggerMouseEvent(element, eventType, true, clientX, clientY);
2428
+ }
2429
+
2430
+
2431
+ // If the page is going to unload - still attempt to fire any subsequent events.
2432
+ // However, we can't guarantee that the page won't unload half way through, so we need to handle exceptions.
2433
+ try {
2434
+ win.detachEvent("onbeforeunload", pageUnloadDetector);
2435
+
2436
+ if (this._windowClosed(win)) {
2437
+ return;
2438
+ }
2439
+
2440
+ // Onchange event is not triggered automatically in IE.
2441
+ if (isDefined(element.checked) && wasChecked != element.checked) {
2442
+ triggerEvent(element, 'change', true);
2443
+ }
2444
+
2445
+ }
2446
+ catch (e) {
2447
+ // If the page is unloading, we may get a "Permission denied" or "Unspecified error".
2448
+ // Just ignore it, because the document may have unloaded.
2449
+ if (pageUnloading) {
2450
+ LOG.logHook = function() {
2451
+ };
2452
+ LOG.warn("Caught exception when firing events on unloading page: " + e.message);
2453
+ return;
2454
+ }
2455
+ throw e;
2456
+ }
2457
+ };