An utility library for protractor providing several utility methods which extend the ElementFinder
, the ElementArrayFinder
and the protractor
objects.
- Description
- Installation
- API
- ElementFinder methods
- clickOnParent()
- getInputValue()
- getIdValue()
- getSelectCheckedOption()
- getCheckedValue()
- setInputValue(value)
- clearAndSetInputValue(value)
- setValueIfEnabledOrProceed(value)
- isEnabledIfDisplayedOrProceed(isEnabledIfNotDisplayed)
- waitAndThenExecute(maxWaitTime, fnToExecute)
- isDisplayedIfPresent()
- isEnabledAndPresent()
- isTagSelect()
- isTagInputType(inputType)
- isRadioInput()
- isCheckboxInput()
- ElementArrayFinder methods
- protractor methods
- onCatchGenericError(err)
- getLabelTextByForAttribute(forValue)
- getElementArrayFinderFromArrayOfElementFinder(arrayOfElementFinder)
- setRadioButtonValueByLabelFor(labelFor)
- setRadioButtonValueByLabelText(labelText[, elementContainer])
- setSelectValueByOptionText(optionText[, elementContainer])
- ifPresentAndEnabledDoAction(elementToCheck, actionToDo)
- checkErrorValidation(field, errorType[, errorMessage])
- getFirstAvailableSelectValue(selectElement)
- getFirstPresentElement(elements)
- ElementFinder methods
- Developer Usage
An utility library for protractor providing several utility methods which extend the ElementFinder
, the ElementArrayFinder
and the protractor
objects.
The source code provides also a really simple example AngularJS application where you can test your methods or test the usage of the existing methods.
Simply install it from the npm registry through:
npm install --save-dev qprotractor
Once installed it, require('qprotractor')
at the beginning of your onPrepare
hook of protractor, in order to make the methods available inside all the protractor life cycle and files:
exports.config = {
...
onPrepare: function() {
require('qprotractor');
}
};
Here the list of all the methods provided by the library. The methods extend ElementFinder
, ElementArrayFinder
and protractor
and here they are grouped by the kind of object which they extend.
You can see the usage of all the APIs inside the files contained in the folder test
. You will find an example for all the methods listed below.
Click on the parent of the current ElementFinder through it's property parentElementArrayFinder included in it
protractor.promise
A promise which will be resolved after the click on the element happens
element(by.id('my-id')).clickOnParent().then(
// click has been performed
);
Get the value of the current ElementFinder. For simple input types (like text, number, etc...) it will return directly the value. For complex types (like select, radio, etc...) it will return a composite value. See the explicit methods for these elements
protractor.promise
A promise which holds the value of the input
Get the id of the current ElementFinder
protractor.promise
A promise which holds the id of the input
element(by.model('my-model')).getIdValue().then(id => {
// id holds the value of the id the element with model my-model
});
Get the text of the checked option of an ElementFinder corresponding to a select
protractor.promise
A promise which holds the text of the checked option
element(by.id('my-select-id')).getSelectCheckedOption().then(value => {
// value holds the text of the selected option in a select
});
Get the value of the checked attribute of ElementFinder like an option, a radio input or a checkbox input
element(by.id('my-checkbox')).getCheckedValue().then(value => {
// value holds the value of the checked checkbox
});
protractor.promise
A promise which holds the value of the attribute checked
Set the provided value to an input ElementFinder. It doesn't clear the previous value so this will be appended at the end of the previous one
Name | Type | Description |
---|---|---|
value | string |
The value to set in the input ElementFinder |
protractor.promise
A promise resolved when the value will be set in the input ElementFinder
element(by.id('my-input')).setInputValue('test').then(resp => {
// in the value of the field my-input has been appended test, without clearing the previous value
});
Clear the actual value of an input ElementFinder and then it sets the provided value to it
Name | Type | Description |
---|---|---|
value | string |
The value to set in the input ElementFinder |
protractor.promise
A promise resolved when the input ElementFinder will be clear and the value will be set in it
element(by.id('my-input')).clearAndSetInputValue('test').then(resp => {
// the value of the field my-input has been set to test, clearing the previous value
});
Set a new value on an input ElementFinder if it is enabled, otherwise proceed without setting the value
Name | Type | Description |
---|---|---|
value | value |
The new value to set in the ElementFinder if it is enabled |
Boolean | clearAndSet |
Optional. Pass false if you want just to set without clearing the field before |
protractor.promise
A promise resolved when the input ElementFinder will be clear and the new value will be set in, if it is enabled. Otherwise it will be fulfilled if the ElementFinder is not enabled.
element(by.id('my-input')).setValueIfEnabledOrProceed('test').then(resp => {
// if the field my-input is enabled, it has been set to test, otherwise the step has been skipped
});
Check if an ElementFinder is enabled whether is displayed, otherwise checks if the element is enabled or disabled, depending on the provided value (true by default)
Name | Type | Description |
---|---|---|
isEnabledIfNotDisplayed | boolean |
The default value to be returned if the ElementFinder is not displayed. True by default |
protractor.promise
A promise which holds the isEnabled value if the ElementFinder is present, otherwise the provided isEnabledIfNotDisplayed or true by default
element(by.id('my-input')).isEnabledIfDisplayedOrProceed(false).then(resp => {
// If the element my-input is not present, proceed. Otherwise check if the element is not enabled because false has been passed through the method.
});
Wait for an ElementFinder to be present in a given delay, and if present then execute the provided function, otherwise reject with an error
Name | Type | Description |
---|---|---|
maxWaitTime | number |
The maximum time to wait for the ElementFinder presence |
fnToExecute | Function |
The function to be executed if the ElementFinder is present in the given delay |
protractor.promise
A promise rejected if the element is not present in the provided time, otherwise it holds the value returned by the provided function to call
// after waiting for a maximum of 5 seconds that the element my-id is displayed, get the text of the element and check if it is equal to Hello World
let el = element(by.id('my-id'));
el.waitAndThenExecute(5000, fnToExecute);
function fnToExecute() {
expect(el.getText()).toEqual('Hello World');
}
Check if an element is displayed wether is present. Returns a promise which holds the display value, if the element is present, otherwise false.
Name | Type | Description |
---|
protractor.promise
A promise which holds the display value, if the element is present, otherwise false
let el = element(by.id('my-id'));
el.isDisplayedIfPresent().then(onDisplayedIfPresent);
function onDisplayedIfPresent(isDisplayed) {
// false if the element is not present or present but not displayed
// true if the element is present and displayed
}
Check if an element is enabled whether is present. If not present, return a promise which holds false
protractor.promise
A promise which holds if the element is enabled, otherwise holds false if not present
let el = element(by.id('my-id'));
el.isEnabledAndPresent().then(onEnabledAndPresent);
function onEnabledAndPresent(isEnabledAndPresent) {
// isEnabledAndPresent if the element is not present or present but not enabled
// true if the element is present and enabled
}
Check if an element is a select tag
Boolean
True or false depending if the tag is a select
let el = element(by.id('my-id'));
var isSelect = el.isTagSelect(); // holds true if el is a checkbox input, or false otherwise
Check if an element is an input tag of the given type
Name | Type | Description |
---|---|---|
inputType | String |
The input type to check for the tag element |
Boolean
True or false depending if the tag is the given input type or not
let el = element(by.id('my-id'));
var isCheckbox = el.isTagInputType('checkbox'); // holds true if el is a checkbox input, or false otherwise
Check if an element is a radio input tag
Boolean
True or false depending if the tag is a radio input or not
let el = element(by.id('my-id'));
var isRadio = el.isRadioInput(); // holds true if el is a radio input, or false otherwise
Check if an element is a checkbox input tag
Boolean
True or false depending if the tag is a checkbox input or not
let el = element(by.id('my-id'));
let isRadio = el.isCheckboxInput(); // holds true if el is a checkbox input, or false otherwise
Get the value associated to the checked option of a radio button. Called on the ElementArrayFinder associated to all the elements of the radio input
protractor.promise
A promise which holds the value of the selected radio input, otherwise it rejects a promise with an error
element.all(by.model('my-radio-model')).getValueOfRadioSelectedItem().then(value => {
// value is the value of the selected radio item associated to radio input elements selected by the selector by the model my-radio-model
});
Get the text of the label of the checked option of a radio input. And specify if you want to return an empty string or an error in case no element is selected. Called on the ElementArrayFinder associated to all the elements of the radio input.
Name | Type | Description |
---|---|---|
ifEmptyThrowError | boolean |
Optional. Specify if you need to throw an error if the radio has not element selected or return an empty string |
protractor.promise
A promise which holds the text of the checked option of radio inputs, eventually empty value if specified through the parameter, otherwise it is rejected with an error
element.all(by.model('my-radio-model')).getLabelTextOfRadioSelectedItem(false).then(value => {
// value is the value of the selected radio item associated to radio input elements selected by the selector by the model my-radio-model
// passing false, it returns an empty string if the radio has not been selected
// passing true it returns an error
});
Get the text of all the rows of the table but associated only to the columns identified by the css classes provided as input array. Called on the ElementArrayFinder which holds all the tr of the table.
Name | Type | Description |
---|---|---|
columnClassesArray | array |
An array with all the css classes of the table columns you want to retrieve the text for |
protractor.promise
A promise which holds an array of arrays where the inner arrays represent the row texts of the table associated to the given columns
element.all(by.repeater('data in tableData')).getTableRowsFromCSSColumnsValues(['first-column', 'second-column']).then(data => {
// data is an array whose each item is an array with all the values of the rows, corresponding to the columns of the table which have the CSS classes given as input
});
Sort an ElementArrayFinder using a given compareFunction executed over the values returned by the application of the functionName with the inputParams over the ElementFinder items which compose the ElementArrayFinder.
Name | Type | Description |
---|---|---|
compareFunction | Function |
Function to be used for comparing elements inside the ElementArrayFinder |
functionName | string |
Name of the function called on the ElementFinder items. Should be a valid function of ElementFinder |
inputParams | array |
An array of input parameters to be passed in the functionName called on the single ElementFinder items |
protractor.promise
A promise which holds the sorted ElementArrayFinder. Then the sorted ElementArrayFinder will be available in result property of the object returned from the promise. Rejected if some error occurs.
let elementsToSort = element.all(by.className('elementsToSort'));
let compareFunction = (a, b) => a[0].localeCompare(b[0]);
elementsToSort.sort(compareFunction, 'getText', []).then(el => {
// el.result holds the sorted ElementArrayFinder by getText using the compareFunction, from the not sorted ElementArrayFinder elementsToSort
});
Get the text of the label with the for attribute equals to the provided forValue
Name | Type | Description |
---|---|---|
forValue | string |
The value of the for of the label |
protractor.promise
A promise resolved which holds the text value of the label
protractor.getLabelTextByForAttribute('my-for-attribute').then(value => {
// value holds the text of the label which has the given for attribute
});
Get an ElementArrayFinder from an array of ElementFinder items
Name | Type | Description |
---|---|---|
arrayOfElementFinder | array |
An array of ElementFinder items you want to convert into an ElementArrayFinder item |
protractor.ElementArrayFinder
The ElementArrayFinder associated to the given array of ElementFinder items
let arrayOfElementFinder = [
element(by.id('my-id-1')),
element(by.id('my-id-2'))
];
// get the ElementArrayFinder from the array of the given ElementFinder items
let convertedElementArrayFinder = protractor.getElementArrayFinderFromArrayOfElementFinder(arrayOfElementFinder);
Throw a given error
Name | Type | Description |
---|---|---|
err | Error |
Error to be thrown |
Void
// throw the given error
protractor.onCatchGenericError('My Error');
Set the value of a radio button clicking on the label associated to it through the HTML for attribute
Name | Type | Description |
---|---|---|
labelFor | string |
The value of the for attribute that will be used to look for the label element |
protractor.promise
A promise resolved when the label associated to the radio button will be clicked
protractor.setRadioButtonValueByLabelFor('my-label-for').then(res => {
// the radio button depending on the label with for attribute my-label-for has been selected
});
Set the radio button value from the given labelText associated to it.
Name | Type | Description |
---|---|---|
labelText | string |
The text of the label associated to the radio button you want to select |
elementContainer | ElementFinder |
Optional. The ElementFinder where to find the label. Otherwise it will find inside all the DOM. |
protractor.promise
A promise resolved when the label associated to the radio button will be clicked. Rejected if some error occurs.
protractor.setRadioButtonValueByLabelText('my-label-text').then(res => {
// the radio button depending on the label with the text my-label-text has been selected
});
Select an option from a select depending on the provided optionText
Name | Type | Description |
---|---|---|
optionText | string |
The text of the option want to select |
elementContainer | ElementFinder |
Optional. The ElementFinder where to find the option. Otherwise it will find inside all the DOM |
protractor.promise
A promise resolved when the option of the select has been selected. Rejected if some error occurs
let select = element(by.id('my-select'));
protractor.setSelectValueByOptionText('My Option Text', select).then(
// the option with text My Option Text contained inside the select element with id my-select has been selected
);
Perform the given action if and only if the given element is present and displayed, otherwise return a fulfilled promise without executing the action
Name | Type | Description |
---|---|---|
elementToCheck | ElementFinder |
The element that you want to check if is present and displayed before to perform the action |
actionToDo | Function |
A function which represents the action to perform if the element is present and displayed |
protractor.promise
A promise corresponding to the given action provided, or a fulfilled promise if the element is not present or not displayed
let element = element(by.id('my-element-id'));
protractor.ifPresentAndEnabledDoAction(element, element.click).then(
// action performed or promise fulfilled because the element is not present or displayed
);
Check if the given field presents the given error type, and eventually if the error message is the same as provided
Name | Type | Description |
---|---|---|
field | String |
A string representing the selector used in the ng-messages condition block, so the form, the field and the error type |
errorType | String |
A string representing the type of error you are going to check, so the one used in the ng-message block |
errorMessage | String |
A string representing the error message you are expecting to see associated to that field in case of that error type |
protractor.promise
A promise which holds two booleans (or one) that represent if the error message is displayed, and eventually if the error message is the same as the expected one
protractor.checkErrorValidation('formName.fieldName.$error', 'maxlength', 'The maximum length is 9').then(data => {
// data holds two booleans, the first one is if the maxlength error is displayed for the fieldName, and the second is if the displayed error is equal to "The Maximum length is 9"
});
Get the first present element within a list of ElementFinder items
Name | Type | Description |
---|---|---|
elements | Array |
An Array representing let list of ElementFinder items where to look for the first one which is present in the page |
protractor.promise
A promise resolved which holds the first present element of the array, undefined otherwise
protractor.getFirstPresentElement([element(by.id('a')), element(by.id('marital-status')), element(by.id('b'))]).then(() => {
// el is the first present element in the given list of ArrayFinder elements given as input
});
Given an ElementFinder corresponding to a select HTML element, returns the first valid option within the select with no empty text. Excludes the first element which usually corresponds to a placeholder of the select.
Name | Type | Description |
---|---|---|
selectElement | ElementFinder |
The ElementFinder corresponding to the select HTML element |
protractor.promise
A promise which holds an ElementFinder corresponding to the first valid option of the select with no empty text
let select = element(by.id('my-select-id'));
let firstOption = protractor.getFirstAvailableSelectValue(select);
// firstOption is the ElementFinder corresponding to the first valid not empty option of the given select ElementFinder
If you want to contribute to the current project, any help is more than welcome!
In order to contribute on it, first clone the repository and get the source code:
git clone https://github.com/quirimmo/Qprotractor.git
Then install all the dependencies:
npm install
Once you've done that, you may need to install the webdriver-manager through:
node_modules/protractor/bin/webdriver-manager update
After that, you can simply use the gulp tasks in order to run the application.
For example, if you want to run the AngularJS application used as example of usage of the library, you can trigger the following gulp task:
gulp serve
This will make the application available at the following address:
If you want to execute instead all the tests, you simply need to run the following task and everything will start automatically:
gulp protractor-test
The application uses husky
in order to trigger the e2e-tests before to push. If any of the tests will fail, you would not be able to push on the repository.
After pushed, travis
will be triggered in order to execute the job. If the job fails, the PR cannot be merged either.
So let's suppose you want to add a new method inside the library, these are the steps to be followed:
- Add your method inside
src/qprotractor.js
- Add the jsdoc of the method
- Add the related test and explanation of the method inside the relevant file inside the
test
folder - Execute the
gulp protractor-test
to ensure that all the tests are working - Add the documentation inside the README.md
- Push all the code in order to trigger the travis job
- Push again all the code in order to include the minified version of the files, but this time using
git push --no-verify
in order to bypass the local e2e tests