Подтвердить что ты не робот

Как проверить, указан ли аргумент метода директивы в AngularJS?

Я создал пользовательскую директиву, содержащую кнопку. Эта кнопка вызывает метод из родительской области, указанный атрибутом callback.

<!DOCTYPE html>
<html ng-app="app">
<head>
    <title>Simple directive</title>

    <script src="js/lib/angular/angular.js"></script>

    <script type="text/javascript">
        var app = angular.module('app', []);

        app.controller('TestController', function($scope) {

            $scope.doSomething = function(param) {
                alert('Something called with: ' + param);
            }
        })

        app.directive('myDirective', function() {
            var ret = {
                restrict: 'E',
                scope: {
                    user: '@',
                    callback: '&'       // bound a function from the scope
                },
                template: '<div>Hello {{user}}<button ng-show="hasCallback()" ng-click="callback({userData: user})">Callback</button>',
                controller: function($scope) {
                    $scope.hasCallback2 = function() {
                        var t = typeof $scope.callback;
                        return t == 'function';
                    }

                    $scope.hasCallback = function() {
                        return angular.isDefined($scope.callback);
                    }
                }
            };
            return ret;
        });

    </script>
</head>

<body ng-controller="TestController">

<my-directive user="cat" callback="doSomething(userData)"></my-directive>
<my-directive user="dog" callback="doSomething(userData)"></my-directive>
<my-directive user="pig"></my-directive>

</body>

</html>

Мой вопрос:

Как я могу контролировать видимость кнопки внутри шаблона? Я бы хотел скрыть его, если атрибут callback не указан в пользовательском теге (см. Третий тег my-директивы). Когда я проверяю тип обратного вызова, я всегда получаю "функцию" и angular. IsDefined (...) также возвращает true.

4b9b3361

Ответ 1

Использование '&?' возвращает undefined, если атрибут не был установлен.

'&' = функция обратного вызова всегда определена.

'&?' = функция обратного вызова определяется только тогда, когда атрибут определен в html-шаблоне.

bindToController: {
    callback: '&?'
},
controller: function() {
    if (this.callback === undefined) {
        // attribute "callback" was not defined
    }
}

Примечание: Работает в Angular 1.4.8. Я не уверен, работает ли он в более старых версиях.

Ответ 2

Посмотрев на исходный код angularjs, я вижу следующее:

case '&':
    parentGet = $parse(attrs[attrName]);
    isolateScope[scopeName] = function(locals) {
         return parentGet(scope, locals);
    };
    break;

parentGet - выражение связанной функции. К сожалению, это локальная переменная, доступная только функции, назначенной isolateScope[scopeName] через закрытие.

Вместо того, чтобы искать способ получить эту переменную, простое решение - просто проверить attrs. Попробуйте:

link: function(scope,elem,attrs) {

      scope.hasCallback = function() {
        return angular.isDefined(attrs.callback);
      }
    }

DEMO