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

fix(ng-if): make ng-if evaluate things using javascript logic #4005

Closed
wants to merge 1 commit into from

Conversation

lgalfaso
Copy link
Contributor

Make ng-if evaluate things using javascript logic
Do not reevaluate everything contained within an ng-if when there
is a change of the value of the ng-if, but the truthy of falsy value
does not change

Fixes #3969

BREAKING CHANGE: The expressions

  • <div ng-if="[]">X</div>
  • <div ng-if="'f'">X</div>
  • <div ng-if="'[]'">X</div>

used to be evaluated to false

@mary-poppins
Copy link

Thanks for the PR!

  • Contributor signed CLA now or in the past
    • If you just signed, leave a comment here with your real name
  • PR's commit messages follow the commit message format

If you need to make changes to your pull request, you can update the commit with git commit --amend.
Then, update the pull request with git push -f.

Thanks again for your help!

@@ -86,7 +86,7 @@ var ngIfDirective = ['$animate', function($animate) {
compile: function (element, attr, transclude) {
return function ($scope, $element, $attr) {
var childElement, childScope;
$scope.$watch($attr.ngIf, function ngIfWatchAction(value) {
$scope.$watch('!!(' + $attr.ngIf + ')', function ngIfWatchAction(value) {
Copy link
Contributor

Choose a reason for hiding this comment

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

It would seem to me cleaner to do something like:

$scope.$watch($attr.ngIf, function ngIfWatchAction(value) {
  ...
  if (!!value) {
    ...
  }
...
});

Copy link
Contributor

Choose a reason for hiding this comment

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

Oh, I guess that if the value changes but it is still true we don't want to delete the childElement.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Yep, that was the idea

@mgol
Copy link
Member

mgol commented Oct 17, 2013

@petebacondarwin It would be better to get this merged for 1.2.0 than 1.2.1 as in this case you'd introduce a breaking change in a patch version which is not the best idea...

@caitp
Copy link
Contributor

caitp commented Feb 5, 2014

@lgalfaso could you rebase this? It would be good to get this into 1.3 --- And similar to the ngShowHide changes, please included a short summary of how affected users could migrate in the BREAKING CHANGE block

@caitp caitp modified the milestones: 1.3.x, Backlog Feb 5, 2014
@lgalfaso
Copy link
Contributor Author

lgalfaso commented Feb 5, 2014

@caitp 4e4c9e9 is a rebase with the changes for ng-if, ng-show and ng-hide

@caitp
Copy link
Contributor

caitp commented Feb 5, 2014

Yeah thanks =) Were you going to close your other PR then, since this one contains both fixes? (I think it would be better to have kept the ngShowHide changes out of this one, but well...)

@lgalfaso
Copy link
Contributor Author

lgalfaso commented Feb 5, 2014

@caitp if you think it would be best to keep these two as different PRs, then that should not be an issue. I put these two together just to be consistent

@mgol
Copy link
Member

mgol commented May 12, 2014

Won't watching '!!(' + attr.ngShow + ')' instead of just attr.ngShow make it harder to optimize it via passing to Object.observe? I guess we don't add support for O.o in 1.3 anyway so that's not an issue, @caitp?

I'd like to merge it rather soonish since it does change semantics so it's better to prepare people for that as early as possible.

@mgol
Copy link
Member

mgol commented May 12, 2014

Actually, shouldn't we just fix toBoolean once and for all? I mean, this whole function:
https://github.com/angular/angular.js/blob/master/src/Angular.js#L1001-L1011
should be removed and its uses replaced with regular !! where needed.

@lgalfaso
Copy link
Contributor Author

@mzgol doing '!!(' + attr.ngShow + ')' has the added benefit that if the value of attr.ngShow changes from one truthy value to another truthy value, then the watch is not called. BTW, +1 on fixing toBoolean

@mgol
Copy link
Member

mgol commented May 13, 2014

@lgalfaso

doing '!!(' + attr.ngShow + ')' has the added benefit that if the value of attr.ngShow changes from one truthy value to another truthy value, then the watch is not called.

I know, I just mentioned it because if we had Object.observe then passing a regular reference would make it not checked on every $digest which is required when you modify it on the fly so this approach would be bad in Angular 2. But here it may be better because of what you said. Context matters. :)

Make ngIf, ngShow and ngHide follow javascript `truthy`/`falsy`
logic and not the custom toBoolean logic

Fixes angular#5414 angular#4277 angular#3969

BREAKING CHANGE: The expressions

* `<div ng-hide="[]">X</div>`
* `<div ng-hide="'f'">X</div>`
* `<div ng-hide="'[]'">X</div>`

used to be evaluated to `false` and the elements were hidden.

The same effect is present for `ng-show` and the elements are now
  visible; and with `ng-if` and the elements are now removed

If you were previously doing `ng-show="exp"` where
  `$scope.exp = 'no' // (or 'n' or 'f')`, then instead write
  `ng-show="exp && exp !== 'no'` (or 'n' or 'f').
@petebacondarwin petebacondarwin removed this from the 1.3.0 milestone Jun 16, 2014
@petebacondarwin
Copy link
Contributor

Regarding '!!(' + attr.ngShow + ')' I think it is better to check the newValue against the oldValue in the watch handler. Yes, it does mean that the watch handler gets called a few more times if the value goes from say, '' to 0. But equally we are not having to do the !! operation on every watch function.

@mgol
Copy link
Member

mgol commented Jun 20, 2014

@petebacondarwin I'll be getting rid of toBoolean in a separate PR soon so this will probably be obsolete.

@petebacondarwin
Copy link
Contributor

Here are the places where toBoolean is used and how we should deal with them. Then we can remove toBoolean altogether.

  • For ngIf, ngShow and ngHide we should, as you say, remove the call to toBoolean and use !! instead, causing a breaking change.
  • Where toBoolean is used in input for ngTrim we should put in place specific functionality and only check whether the value of the attribute is the string "false". This would also be a breaking change.
  • Where toBoolean is used in the orderBy filter, we are guaranteed to only have a boolean value anyway so this should simply be removed, with no breaking change.

@petebacondarwin
Copy link
Contributor

@mzgol - are you dealing with the items I specified above then?

@mgol
Copy link
Member

mgol commented Jun 20, 2014

@petebacondarwin Yes, I'll go through every usage and see what needs to be done to fully replace it. Thanks for your analysis, it'll help.

@petebacondarwin
Copy link
Contributor

Assigning to @mzgol then!

@IgorMinar
Copy link
Contributor

Thanks @petebacondarwin for looking into this one.

@mzgol will you be able to get this done by Monday or Tuesday? If not, @petebacondarwin or someone else will take care of it.

@mgol
Copy link
Member

mgol commented Jun 22, 2014

I'll look into it tomorrow.

Michał Gołębiowski

@IgorMinar
Copy link
Contributor

Great! Thanks
On Jun 22, 2014 3:11 AM, "Michał Gołębiowski" notifications@github.com
wrote:

I'll look into it tomorrow.

Michał Gołębiowski


Reply to this email directly or view it on GitHub
#4005 (comment).

@mgol
Copy link
Member

mgol commented Jun 23, 2014

Closing in favor of #7960.

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.

ngIf and ngShow mistreats "[]" (empty array)
6 participants