selenium-core-runner 0.0.3

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