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

fix(ngAnimate): safe-guard against missing document #14633

Merged
merged 2 commits into from
May 24, 2016

Conversation

Narretz
Copy link
Contributor

@Narretz Narretz commented May 18, 2016

A different fix for the mocked document problem (in 1.4.x) that also fixes another possibly breaking access to the document.

One caveat: since we are now failing gracefully, and a missing document means no animations fire, tests that test animations and mock out document will not run animations, and this will be opaque to developers. However, I assume that most tests that mock document are actually very basic, as lots of other things should break when $document is mocked

Please check if the PR fulfills these requirements

The commit message follows our guidelines: https://github.com/angular/angular.js/blob/master/CONTRIBUTING.md#commit-message-format

In tests, the $document service might be mocked without providing a real
document, which will lead to errors.

With this fix, we provide an empty object as a default document to
safe-guard against this. If the empty object is used, no animations
will run, as the animated node won't have a body as a parent.

This commit also changes the check for document.hidden slightly. It
should be accessed independently of the current animationsEnabled state.
Since animations are only enabled after two digests, it's possible that
some tests never reach the animationsEnabled = true state and therefore
aren't actually checking the document.hidden state, which means that
the previous fix only works if no more than two digests happen in the test.
@@ -331,7 +334,7 @@ var $$AnimateQueueProvider = ['$animateProvider', function($animateProvider) {

var isStructural = ['enter', 'move', 'leave'].indexOf(event) >= 0;

var documentHidden = animationsEnabled && $document[0].hidden;
var documentHidden = rawDocument.hidden;
Copy link

@ThomasBurleson ThomasBurleson May 18, 2016

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you also need to include animationsEnabled ?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@ThomasBurleson - that was a hack by @IgorMinar to get the tests passing in G3

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You mean "not need"? animationsEnabled was added by Igor to fix a test in the g3 sync, but as I wrote in the commit message, this doesn't actually prevent the access to the document in all cases. If animations are enabled, $document[0] is still checked, and it might still be undefined

@petebacondarwin
Copy link
Contributor

LGTM

@@ -583,7 +586,7 @@ var $$AnimateQueueProvider = ['$animateProvider', function($animateProvider) {
* d) the element is not a child of the $rootElement
*/
function areAnimationsAllowed(element, parentElement, event) {
var bodyElement = jqLite($document[0].body);
var bodyElement = jqLite(rawDocument.body);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

IIRC there is one more access to $document[0].body. Should we change that too ?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In this file?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@IgorMinar
Copy link
Contributor

does it really make sense to run animations when $document is mocked out? what's the use-case for that?

I think that all of the animation code assumes that either the animations enabled and document is not mocked out (or is mocked out by a special animations-compatible mock used for unit testing of the animations code) or the animations are disabled.

With all these changes we are adding a forth scenario - animations are enabled but $document is mocked out by third-party that is not aware of the $document requirements by the animation code.

@IgorMinar
Copy link
Contributor

I think the assumption we should make is that if someone mocks out $document then they don't want animations to run.

@Narretz
Copy link
Contributor Author

Narretz commented May 18, 2016

We can make this assumption but this might not necessarily be what someone else expects because it's not clear that animations need a document. But i like it since it makes the code simpler.
The bigger issue is that once we start to account for a missing or insufficiently mocked out document there's a lot more to change.

@jbedard
Copy link
Collaborator

jbedard commented May 18, 2016

Shouldn't tests just mock the document properly? Seems weird putting that mock in the actual code...

@Narretz
Copy link
Contributor Author

Narretz commented May 18, 2016

To clarify, even in the current form, without a document no animations are performed, as the areAnimationsAllowed check won't find a body element: https://github.com/angular/angular.js/pull/14633/files#diff-c15d87c15661cba4a721501ba4fda5b2R589

So I'd be okay with making this explicit if we find no document in the first place.

@petebacondarwin
Copy link
Contributor

I just realised that this actually checks the opposite... if the document is mocked out (and rawDocument is set to {}) then we will indeed try to run the animations.

Perhaps we should set rawDocument to { hidden: true } if $document[0] is undefined?

@Narretz
Copy link
Contributor Author

Narretz commented May 19, 2016

Yes that would short-circuit the animations earlier and be more explicit. Sounds good to me.

@Narretz
Copy link
Contributor Author

Narretz commented May 23, 2016

I've updated the PR to use an object {hidden: true} when no document is available. This will prevent the animator from attempting to run any animations

@matsko
Copy link
Contributor

matsko commented May 24, 2016

LGTM

@Narretz Narretz merged commit 0d764b5 into angular:v1.4.x May 24, 2016
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

8 participants