| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150 | /** * @fileoverview Rule to flag unnecessary bind calls * @author Bence Dányi <bence@danyi.me> */"use strict";//------------------------------------------------------------------------------// Requirements//------------------------------------------------------------------------------const astUtils = require("../ast-utils");//------------------------------------------------------------------------------// Rule Definition//------------------------------------------------------------------------------module.exports = {    meta: {        docs: {            description: "disallow unnecessary calls to `.bind()`",            category: "Best Practices",            recommended: false,            url: "https://eslint.org/docs/rules/no-extra-bind"        },        schema: [],        fixable: "code",        messages: {            unexpected: "The function binding is unnecessary."        }    },    create(context) {        let scopeInfo = null;        /**         * Reports a given function node.         *         * @param {ASTNode} node - A node to report. This is a FunctionExpression or         *      an ArrowFunctionExpression.         * @returns {void}         */        function report(node) {            context.report({                node: node.parent.parent,                messageId: "unexpected",                loc: node.parent.property.loc.start,                fix(fixer) {                    const firstTokenToRemove = context.getSourceCode()                        .getFirstTokenBetween(node.parent.object, node.parent.property, astUtils.isNotClosingParenToken);                    return fixer.removeRange([firstTokenToRemove.range[0], node.parent.parent.range[1]]);                }            });        }        /**         * Checks whether or not a given function node is the callee of `.bind()`         * method.         *         * e.g. `(function() {}.bind(foo))`         *         * @param {ASTNode} node - A node to report. This is a FunctionExpression or         *      an ArrowFunctionExpression.         * @returns {boolean} `true` if the node is the callee of `.bind()` method.         */        function isCalleeOfBindMethod(node) {            const parent = node.parent;            const grandparent = parent.parent;            return (                grandparent &&                grandparent.type === "CallExpression" &&                grandparent.callee === parent &&                grandparent.arguments.length === 1 &&                parent.type === "MemberExpression" &&                parent.object === node &&                astUtils.getStaticPropertyName(parent) === "bind"            );        }        /**         * Adds a scope information object to the stack.         *         * @param {ASTNode} node - A node to add. This node is a FunctionExpression         *      or a FunctionDeclaration node.         * @returns {void}         */        function enterFunction(node) {            scopeInfo = {                isBound: isCalleeOfBindMethod(node),                thisFound: false,                upper: scopeInfo            };        }        /**         * Removes the scope information object from the top of the stack.         * At the same time, this reports the function node if the function has         * `.bind()` and the `this` keywords found.         *         * @param {ASTNode} node - A node to remove. This node is a         *      FunctionExpression or a FunctionDeclaration node.         * @returns {void}         */        function exitFunction(node) {            if (scopeInfo.isBound && !scopeInfo.thisFound) {                report(node);            }            scopeInfo = scopeInfo.upper;        }        /**         * Reports a given arrow function if the function is callee of `.bind()`         * method.         *         * @param {ASTNode} node - A node to report. This node is an         *      ArrowFunctionExpression.         * @returns {void}         */        function exitArrowFunction(node) {            if (isCalleeOfBindMethod(node)) {                report(node);            }        }        /**         * Set the mark as the `this` keyword was found in this scope.         *         * @returns {void}         */        function markAsThisFound() {            if (scopeInfo) {                scopeInfo.thisFound = true;            }        }        return {            "ArrowFunctionExpression:exit": exitArrowFunction,            FunctionDeclaration: enterFunction,            "FunctionDeclaration:exit": exitFunction,            FunctionExpression: enterFunction,            "FunctionExpression:exit": exitFunction,            ThisExpression: markAsThisFound        };    }};
 |