"use strict"; var __extends = (this && this.__extends) || (function () { var extendStatics = function (d, b) { extendStatics = Object.setPrototypeOf || ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) || function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; }; return extendStatics(d, b); }; return function (d, b) { extendStatics(d, b); function __() { this.constructor = d; } d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __()); }; })(); var __makeTemplateObject = (this && this.__makeTemplateObject) || function (cooked, raw) { if (Object.defineProperty) { Object.defineProperty(cooked, "raw", { value: raw }); } else { cooked.raw = raw; } return cooked; }; Object.defineProperty(exports, "__esModule", { value: true }); var compiler_1 = require("@angular/compiler"); var rules_1 = require("tslint/lib/rules"); var utils_1 = require("tslint/lib/utils"); var ngWalker_1 = require("./angular/ngWalker"); var recursiveAngularExpressionVisitor_1 = require("./angular/templates/recursiveAngularExpressionVisitor"); var unstrictEqualityOperator = '=='; var isAsyncBinding = function (ast) { return ast instanceof compiler_1.BindingPipe && ast.name === 'async'; }; var Rule = (function (_super) { __extends(Rule, _super); function Rule() { return _super !== null && _super.apply(this, arguments) || this; } Rule.prototype.apply = function (sourceFile) { var walkerConfig = { expressionVisitorCtrl: ExpressionVisitorCtrl }; var walker = new ngWalker_1.NgWalker(sourceFile, this.getOptions(), walkerConfig); return this.applyWithWalker(walker); }; Rule.metadata = { description: 'Ensures that strict equality is used when evaluating negations on async pipe output.', options: null, optionsDescription: 'Not configurable.', rationale: utils_1.dedent(templateObject_1 || (templateObject_1 = __makeTemplateObject(["\n Angular's async pipes emit null initially, prior to the observable emitting any values, or the promise resolving. This can cause negations, like\n *ngIf=\"!(myConditional | async)\" to thrash the layout and cause expensive side-effects like firing off XHR requests for a component which should not be shown.\n "], ["\n Angular's async pipes emit null initially, prior to the observable emitting any values, or the promise resolving. This can cause negations, like\n *ngIf=\"!(myConditional | async)\" to thrash the layout and cause expensive side-effects like firing off XHR requests for a component which should not be shown.\n "]))), ruleName: 'template-no-negated-async', type: 'functionality', typescriptOnly: true }; Rule.FAILURE_STRING_NEGATED_PIPE = 'Async pipes should not be negated. Use (observable | async) === (false | null | undefined) to check its value instead'; Rule.FAILURE_STRING_UNSTRICT_EQUALITY = 'Async pipes must use strict equality `===` when comparing with `false`'; return Rule; }(rules_1.AbstractRule)); exports.Rule = Rule; var ExpressionVisitorCtrl = (function (_super) { __extends(ExpressionVisitorCtrl, _super); function ExpressionVisitorCtrl() { return _super !== null && _super.apply(this, arguments) || this; } ExpressionVisitorCtrl.prototype.visitBinary = function (ast, context) { this.validateBinary(ast); _super.prototype.visitBinary.call(this, ast, context); }; ExpressionVisitorCtrl.prototype.visitPrefixNot = function (ast, context) { this.validatePrefixNot(ast); _super.prototype.visitPrefixNot.call(this, ast, context); }; ExpressionVisitorCtrl.prototype.generateFailure = function (ast, errorMessage) { var _a = ast.span, spanEnd = _a.end, spanStart = _a.start; this.addFailureFromStartToEnd(spanStart, spanEnd, errorMessage); }; ExpressionVisitorCtrl.prototype.validateBinary = function (ast) { var left = ast.left, operation = ast.operation, right = ast.right; if (!isAsyncBinding(left) || !(right instanceof compiler_1.LiteralPrimitive) || right.value !== false || operation !== unstrictEqualityOperator) { return; } this.generateFailure(ast, Rule.FAILURE_STRING_UNSTRICT_EQUALITY); }; ExpressionVisitorCtrl.prototype.validatePrefixNot = function (ast) { var expression = ast.expression; if (!isAsyncBinding(expression)) return; this.generateFailure(ast, Rule.FAILURE_STRING_NEGATED_PIPE); }; return ExpressionVisitorCtrl; }(recursiveAngularExpressionVisitor_1.RecursiveAngularExpressionVisitor)); var templateObject_1;