diff options
Diffstat (limited to 'src/ngScenario/Application.js')
| -rw-r--r-- | src/ngScenario/Application.js | 102 |
1 files changed, 102 insertions, 0 deletions
diff --git a/src/ngScenario/Application.js b/src/ngScenario/Application.js new file mode 100644 index 00000000..d3a70569 --- /dev/null +++ b/src/ngScenario/Application.js @@ -0,0 +1,102 @@ +'use strict'; + +/** + * Represents the application currently being tested and abstracts usage + * of iframes or separate windows. + * + * @param {Object} context jQuery wrapper around HTML context. + */ +angular.scenario.Application = function(context) { + this.context = context; + context.append( + '<h2>Current URL: <a href="about:blank">None</a></h2>' + + '<div id="test-frames"></div>' + ); +}; + +/** + * Gets the jQuery collection of frames. Don't use this directly because + * frames may go stale. + * + * @private + * @return {Object} jQuery collection + */ +angular.scenario.Application.prototype.getFrame_ = function() { + return this.context.find('#test-frames iframe:last'); +}; + +/** + * Gets the window of the test runner frame. Always favor executeAction() + * instead of this method since it prevents you from getting a stale window. + * + * @private + * @return {Object} the window of the frame + */ +angular.scenario.Application.prototype.getWindow_ = function() { + var contentWindow = this.getFrame_().prop('contentWindow'); + if (!contentWindow) + throw 'Frame window is not accessible.'; + return contentWindow; +}; + +/** + * Changes the location of the frame. + * + * @param {string} url The URL. If it begins with a # then only the + * hash of the page is changed. + * @param {function()} loadFn function($window, $document) Called when frame loads. + * @param {function()} errorFn function(error) Called if any error when loading. + */ +angular.scenario.Application.prototype.navigateTo = function(url, loadFn, errorFn) { + var self = this; + var frame = this.getFrame_(); + //TODO(esprehn): Refactor to use rethrow() + errorFn = errorFn || function(e) { throw e; }; + if (url === 'about:blank') { + errorFn('Sandbox Error: Navigating to about:blank is not allowed.'); + } else if (url.charAt(0) === '#') { + url = frame.attr('src').split('#')[0] + url; + frame.attr('src', url); + this.executeAction(loadFn); + } else { + frame.remove(); + this.context.find('#test-frames').append('<iframe>'); + frame = this.getFrame_(); + frame.load(function() { + frame.unbind(); + try { + self.executeAction(loadFn); + } catch (e) { + errorFn(e); + } + }).attr('src', url); + } + this.context.find('> h2 a').attr('href', url).text(url); +}; + +/** + * Executes a function in the context of the tested application. Will wait + * for all pending angular xhr requests before executing. + * + * @param {function()} action The callback to execute. function($window, $document) + * $document is a jQuery wrapped document. + */ +angular.scenario.Application.prototype.executeAction = function(action) { + var self = this; + var $window = this.getWindow_(); + if (!$window.document) { + throw 'Sandbox Error: Application document not accessible.'; + } + if (!$window.angular) { + return action.call(this, $window, _jQuery($window.document)); + } + angularInit($window.document, function(element) { + element = $window.angular.element(element); + var $injector = element.inheritedData('$injector'); + $injector.invoke(function($browser){ + $browser.notifyWhenNoOutstandingRequests(function() { + action.call(self, $window, _jQuery($window.document)); + }); + }); + }); +}; |
