aboutsummaryrefslogtreecommitdiffstats
path: root/src/scenario/SpecRunner.js
diff options
context:
space:
mode:
authorElliott Sprehn2010-10-18 14:02:18 -0700
committerElliott Sprehn2010-10-19 00:45:38 -0700
commite7e894a2e36e042be6d62af56b0f3126f4e4fc77 (patch)
tree5b9c8b94bf3e3935a3acd6a3c0ecb142c82f4b17 /src/scenario/SpecRunner.js
parenta1fa23397f12e0b52838530a993f14491ad50869 (diff)
downloadangular.js-e7e894a2e36e042be6d62af56b0f3126f4e4fc77.tar.bz2
Significantly clean up the way the scenario DSL works and implement many more DSL statements.
- "this" always means the current chain scope inside a DSL - addFutureAction callbacks now take ($window, $document, done) - $document has a special method elements() that uses the currently selected nodes in the document as defined by using() statements. - $document.elements() allows placeholder insertion into selectors to make them more readable. ex. $document.elements('input[name="$1"]', myVar) will substitute the value of myVar for $1 in the selector. Subsequent arguments are $2 and so on. - $document.elements() results have a special method trigger(event) which should be used to events. This method implements some hacks to make sure browser UI controls update and the correct angular events fire. - futures now allow custom formatting. By default any chain that results in a future can use toJson() or fromJson() to convert the future value to and from json. A custom parser can be provided with parsedWith(fn) where fn is a callback(value) that must return the parsed result. Note: The entire widgets.html UI is now able to be controlled and asserted through DSL statements!!! Victory! :)
Diffstat (limited to 'src/scenario/SpecRunner.js')
-rw-r--r--src/scenario/SpecRunner.js56
1 files changed, 52 insertions, 4 deletions
diff --git a/src/scenario/SpecRunner.js b/src/scenario/SpecRunner.js
index 8b6d4ef1..d6cbdcdc 100644
--- a/src/scenario/SpecRunner.js
+++ b/src/scenario/SpecRunner.js
@@ -41,7 +41,7 @@ angular.scenario.SpecRunner.prototype.run = function(ui, spec, specDone) {
});
} catch (e) {
stepUI.error(e);
- rethrow(e);
+ throw e;
}
},
function(e) {
@@ -71,8 +71,56 @@ angular.scenario.SpecRunner.prototype.addFuture = function(name, behavior) {
*/
angular.scenario.SpecRunner.prototype.addFutureAction = function(name, behavior) {
return this.addFuture(name, function(done) {
- this.application.executeAction(function() {
- behavior.call(this, done);
- });
+ this.application.executeAction(angular.bind(this, function($window, $document) {
+
+ $document.elements = angular.bind(this, function(selector) {
+ var args = Array.prototype.slice.call(arguments, 1);
+ if (this.selector) {
+ selector = this.selector + ' ' + (selector || '');
+ }
+ angular.foreach(args, function(value, index) {
+ selector = selector.replace('$' + (index + 1), value);
+ });
+ var result = $document.find(selector);
+ if (!result.length) {
+ throw {
+ type: 'selector',
+ message: 'Selector ' + selector + ' did not match any elements.'
+ };
+ }
+
+ result.trigger = function(type) {
+ result.each(function(index, node) {
+ var element = $window.angular.element(node);
+ //TODO(esprehn): HACK!!! Something is broken in angular event dispatching
+ // and if the real jQuery is used we need to set the attribtue after too
+ if (angular.isDefined(element.selector)) {
+ if (type === 'click' && node.nodeName.toLowerCase() === 'input') {
+ element.attr('checked', !element.attr('checked'));
+ }
+ }
+ //TODO(esprehn): HACK!! See above comment.
+ element.trigger(type);
+ if (angular.isDefined(element.selector)) {
+ if (type === 'click' && node.nodeName.toLowerCase() === 'input') {
+ element.attr('checked', !element.attr('checked'));
+ }
+ }
+ });
+ };
+
+ return result;
+ });
+
+ try {
+ behavior.call(this, $window, $document, done);
+ } catch(e) {
+ if (e.type && e.type === 'selector') {
+ done(e.message);
+ } else {
+ throw e;
+ }
+ }
+ }));
});
};