Skip to content
This repository was archived by the owner on Jul 29, 2024. It is now read-only.

Commit a2cd6c8

Browse files
committed
feat(syntax): big syntax reboot, expose global $, $$, element, and by
In an effort to make tests more readable and clear, a few more global variables will now be exported. `browser` is an instance of protractor. This was previously accessed using `protractor.getInstance`. `by` is a collection of element locators. Previously, this was `protractor.By`. `$` is a shortcut for getting elements by css. `$('.foo')` === `element(by.css('.foo'))` All changes should be backwards compatible, as tested with the new 'backwardscompat' tests. Closes #156.
1 parent 3f0c249 commit a2cd6c8

17 files changed

+830
-36
lines changed

lib/cli.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ var argv = require('optimist').
2828
alias('build', 'capabilities.build').
2929
boolean('verbose').
3030
boolean('includeStackTrace').
31-
alias('verbose', 'jasmineNodeOpts.verbose').
31+
alias('verbose', 'jasmineNodeOpts.isVerbose').
3232
alias('includeStackTrace', 'jasmineNodeOpts.includeStackTrace').
3333
check(function(arg) {
3434
if (arg._.length > 1) {

lib/locators.js

+20
Original file line numberDiff line numberDiff line change
@@ -76,7 +76,9 @@ ProtractorBy.prototype.selectedOption = function(model) {
7676
}
7777
};
7878
};
79+
7980
/**
81+
* @DEPRECATED - use 'model' instead.
8082
* Usage:
8183
* <input ng-model="user" type="text"/>
8284
* ptor.findElement(protractor.By.input("user"));
@@ -94,6 +96,24 @@ ProtractorBy.prototype.input = function(model) {
9496
};
9597
};
9698

99+
/**
100+
* Usage:
101+
* <input ng-model="user" type="text"/>
102+
* ptor.findElement(protractor.By.model("user"));
103+
*/
104+
ProtractorBy.prototype.model = function(model) {
105+
return {
106+
findOverride: function(driver, using) {
107+
return driver.findElement(
108+
webdriver.By.js(clientSideScripts.findInput), using, model);
109+
},
110+
findArrayOverride: function(driver, using) {
111+
return driver.findElements(
112+
webdriver.By.js(clientSideScripts.findInputs), using, model);
113+
}
114+
};
115+
};
116+
97117
/**
98118
* Usage:
99119
* <textarea ng-model="user"></textarea>

lib/protractor.js

+116-31
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,11 @@ var ProtractorBy = require('./locators.js').ProtractorBy;
77

88
var DEFER_LABEL = 'NG_DEFER_BOOTSTRAP!';
99

10+
var WEB_ELEMENT_FUNCTIONS = [
11+
'click', 'sendKeys', 'getTagName', 'getCssValue', 'getAttribute', 'getText',
12+
'getSize', 'getLocation', 'isEnabled', 'isSelected', 'submit', 'clear',
13+
'isDisplayed', 'getOuterHtml', 'getInnerHtml'];
14+
1015
/**
1116
* Mix in other webdriver functionality to be accessible via protractor.
1217
*/
@@ -17,6 +22,8 @@ for (foo in webdriver) {
1722
/**
1823
* Mix a function from one object onto another. The function will still be
1924
* called in the context of the original object.
25+
*
26+
* @private
2027
* @param {Object} to
2128
* @param {Object} from
2229
* @param {string} fnName
@@ -28,7 +35,93 @@ var mixin = function(to, from, fnName, setupFn) {
2835
setupFn();
2936
}
3037
return from[fnName].apply(from, arguments);
38+
};
39+
};
40+
41+
/**
42+
* Build the helper 'element' function for a given instance of Protractor.
43+
*
44+
* @private
45+
* @param {Protractor} ptor
46+
* @return {function(webdriver.Locator): ElementFinder}
47+
*/
48+
var buildElementHelper = function(ptor) {
49+
var element = function(locator) {
50+
var elementFinder = {};
51+
var webElementFns = WEB_ELEMENT_FUNCTIONS.concat(
52+
['findElement', 'findElements', 'isElementPresent',
53+
'evaluate', '$', '$$']);
54+
webElementFns.forEach(function(fnName) {
55+
elementFinder[fnName] = function() {
56+
var elem = ptor.findElement(locator);
57+
return elem[fnName].apply(elem, arguments);
58+
};
59+
});
60+
61+
elementFinder.find = function() {
62+
return ptor.findElement(locator);
63+
};
64+
65+
elementFinder.isPresent = function() {
66+
return ptor.isElementPresent(locator);
67+
};
68+
69+
return elementFinder;
70+
};
71+
72+
/**
73+
* @type {function(webdriver.Locator): ElementArrayFinder}
74+
*/
75+
element.all = function(locator) {
76+
var elementArrayFinder = {};
77+
78+
elementArrayFinder.count = function() {
79+
return ptor.findElements(locator).then(function(arr) {
80+
return arr.length;
81+
});
82+
};
83+
84+
elementArrayFinder.get = function(index) {
85+
var id = ptor.findElements(locator).then(function(arr) {
86+
return arr[index];
87+
});
88+
return new webdriver.WebElement(ptor.driver, id);
89+
};
90+
91+
elementArrayFinder.then = function() {
92+
return ptor.findElements(locator);
93+
};
94+
95+
return elementArrayFinder;
3196
}
97+
98+
return element;
99+
};
100+
101+
/**
102+
* Build the helper '$' function for a given instance of Protractor.
103+
*
104+
* @private
105+
* @param {Protractor} ptor
106+
* @return {function(string): ElementFinder}
107+
*/
108+
var buildCssHelper = function(ptor) {
109+
return function(cssSelector) {
110+
return buildElementHelper(ptor)(webdriver.By.css(cssSelector));
111+
};
112+
};
113+
114+
/**
115+
* Build the helper '$$' function for a given instance of Protractor.
116+
*
117+
* @private
118+
* @param {Protractor} ptor
119+
* @return {function(string): ElementArrayFinder}
120+
*/
121+
var buildMultiCssHelper = function(ptor) {
122+
return function(cssSelector) {
123+
return buildElementHelper(ptor).all(webdriver.By.css(cssSelector));
124+
};
32125
};
33126

34127
/**
@@ -63,6 +156,27 @@ var Protractor = function(webdriver, opt_baseUrl, opt_rootElement) {
63156
*/
64157
this.driver = webdriver;
65158

159+
/**
160+
* Helper function for finding elements.
161+
*
162+
* @type {function(webdriver.Locator): ElementFinder}
163+
*/
164+
this.element = buildElementHelper(this);
165+
166+
/**
167+
* Helper function for finding elements by css.
168+
*
169+
* @type {function(string): ElementFinder}
170+
*/
171+
this.$ = buildCssHelper(this);
172+
173+
/**
174+
* Helper function for finding arrays of elements by css.
175+
*
176+
* @type {function(string): ElementArrayFinder}
177+
*/
178+
this.$$ = buildMultiCssHelper(this);
179+
66180
/**
67181
* All get methods will be resolved against this base URL. Relative URLs are =
68182
* resolved the way anchor tags resolve.
@@ -134,14 +248,10 @@ Protractor.prototype.waitForAngular = function() {
134248
*/
135249
Protractor.prototype.wrapWebElement = function(element) {
136250
var thisPtor = this;
137-
// Before any of these WebElement functions, Protractor will wait to make sure
251+
// Before any of the WebElement functions, Protractor will wait to make sure
138252
// Angular is synched up.
139-
var functionsToSync = [
140-
'click', 'sendKeys', 'getTagName', 'getCssValue', 'getAttribute', 'getText',
141-
'getSize', 'getLocation', 'isEnabled', 'isSelected', 'submit', 'clear',
142-
'isDisplayed', 'getOuterHtml', 'getInnerHtml'];
143253
var originalFns = {};
144-
functionsToSync.forEach(function(name) {
254+
WEB_ELEMENT_FUNCTIONS.forEach(function(name) {
145255
originalFns[name] = element[name];
146256
element[name] = function() {
147257
thisPtor.waitForAngular();
@@ -254,18 +364,6 @@ Protractor.prototype.wrapWebElement = function(element) {
254364
return element;
255365
};
256366

257-
/**
258-
* Shortcut for querying the document directly with css.
259-
*
260-
* @param {string} selector a css selector
261-
* @see webdriver.WebDriver.findElement
262-
* @return {!webdriver.WebElement}
263-
*/
264-
Protractor.prototype.$ = function(selector) {
265-
var locator = protractor.By.css(selector);
266-
return this.findElement(locator);
267-
};
268-
269367
/**
270368
* Waits for Angular to finish rendering before searching for elements.
271369
* @see webdriver.WebDriver.findElement
@@ -284,19 +382,6 @@ Protractor.prototype.findElement = function(locator, varArgs) {
284382
return this.wrapWebElement(found);
285383
};
286384

287-
/**
288-
* Shortcut for querying the document directly with css.
289-
*
290-
* @param {string} selector a css selector
291-
* @see webdriver.WebDriver.findElements
292-
* @return {!webdriver.promise.Promise} A promise that will be resolved to an
293-
* array of the located {@link webdriver.WebElement}s.
294-
*/
295-
Protractor.prototype.$$ = function(selector) {
296-
var locator = protractor.By.css(selector);
297-
return this.findElements(locator);
298-
};
299-
300385
/**
301386
* Waits for Angular to finish rendering before searching for elements.
302387
* @see webdriver.WebDriver.findElements

lib/runner.js

+8-3
Original file line numberDiff line numberDiff line change
@@ -191,16 +191,21 @@ var runJasmineTests = function() {
191191

192192
sessionId = session.getId();
193193

194-
var ptor = protractor.wrapDriver(
194+
var browser = protractor.wrapDriver(
195195
driver,
196196
config.baseUrl,
197197
config.rootElement)
198-
ptor.params = config.params;
198+
browser.params = config.params;
199199

200-
protractor.setInstance(ptor);
200+
protractor.setInstance(browser);
201201

202202
// Export protractor to the global namespace to be used in tests.
203203
global.protractor = protractor;
204+
global.browser = browser;
205+
global.$ = browser.$;
206+
global.$$ = browser.$$;
207+
global.element = browser.element;
208+
global.by = protractor.By;
204209

205210
// Let the configuration configure the protractor instance before running
206211
// the tests.
File renamed without changes.
File renamed without changes.
File renamed without changes.
File renamed without changes.
File renamed without changes.
File renamed without changes.

spec/backwardscompatConf.js

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
// The main suite of Protractor tests.
2+
exports.config = {
3+
seleniumServerJar: './selenium/selenium-server-standalone-2.35.0.jar',
4+
chromeDriver: './selenium/chromedriver',
5+
6+
seleniumAddress: 'http://localhost:4444/wd/hub',
7+
8+
// Spec patterns are relative to this directory.
9+
specs: [
10+
'backwardscompat/*_spec.js'
11+
],
12+
13+
capabilities: {
14+
'browserName': 'chrome'
15+
},
16+
17+
baseUrl: 'http://localhost:8000',
18+
19+
params: {
20+
login: {
21+
user: 'Jane',
22+
password: '1234'
23+
}
24+
}
25+
};

0 commit comments

Comments
 (0)