In the Serenity-js book we have one example of a Task with just one parameter :
// spec/screenplay/tasks/add_a_todo_item.ts
import { PerformsTasks, Task } from 'serenity-js/protractor';
export class AddATodoItem implements Task {
static called(itemName: string) { // static method to improve the readability
return new AddATodoItem(itemName);
}
performAs(actor: PerformsTasks): PromiseLike<void> { // required by the Task interface
return actor.attemptsTo( // delegates the work to lower-level tasks
// todo: interact with the UI
);
}
constructor(private itemName: string) { // constructor assigning the name of the item
// to a private field
}
Imagine you can add a date the TodoItem should be done. We would receive a date parameter, say 'deadline'. I cannot figure out how to do it.
First thoughts:
constructor:
constructor(private itemName: string, private deadline: Date) {
}
performAs: just add the interaction to type the deadline
We would have a second static method. And possibly the called method return would be changed.
Thanks for your explanations.
There are several ways to do it, depending on which parameters are mandatory, and which are optional, and how many of them you'd like the task to have.
No parameters
If you have a task with no parameters, the easier way to define it is using the
Task.where
factory function:This is almost the same as using a class-style definition below, but with much less code:
One parameter
You can use the above approach with tasks that should receive one parameter:
Which, alternatively, you could also implement as follows:
I find this second version a bit more elegant and more consistent with the built-in interactions like
Click.on
,Enter.theValue
, etc. since you'd be callingLogin.as
rather thanLoginAs
in your actor flow.N parameters
If there are more than 1 parameters, but all of them are required and you're simply after an elegant DSL, you could extend the above pattern as follows:
You'd then invoke the above task:
This design is not particularly flexible, as it doesn't allow you to change the order of parameters (i.e. you can't say
Login.identifiedBy(password).as(username)
) or make some of the parameters optional, but gives you a good-looking DSL with relatively little implementation effort.More flexibility
If you require more flexibility, for example in a scenario where some parameters are optional, you might opt for the class-style definition and a quasi-builder pattern. (I say "quasi" because it doesn't mutate the object, but instead produces new objects).
For example, let's assume that while the system required the username to be provided, the password might be optional.
You can, of course, take it even further and decouple the act of instantiating the task from the task itself, which is useful if the different tasks are different enough to justify separate implementations:
Both the above implementations allow you to call the task as
Login.as(username)
andLogin.as(username).identifiedBy(password)
, but while the first implementation uses a default value of an empty string for the password, the second implementation doesn't even touch the password field.I hope this helps!
Jan