2

我在使用 bPopup 插件的弹出对话框中使用了一个简单的验证码。

您可以将验证码参考这个 jsfiddle:https ://jsfiddle.net/Mazzu/hspxaeqa/

HTML:

<span id="popup-button-email">Click Here!</span>

<div id="popup-second"> <a class="b-close">x<a/>
        <simple-captcha valid="captchaValid"></simple-captcha>
</div>

JS:

app.directive('simpleCaptcha', function() {
    return {
        restrict: 'E',
        scope: { valid: '=' },
        template: '<input ng-model="a.value" ng-show="a.input" style="width:2em; text-align: center;"><span ng-hide="a.input">{{a.value}}</span>&nbsp;{{operation}}&nbsp;<input ng-model="b.value" ng-show="b.input" style="width:2em; text-align: center;"><span ng-hide="b.input">{{b.value}}</span>&nbsp;=&nbsp;{{result}}',
        controller: function($scope) {

            var show = Math.random() > 0.5;

            var value = function(max){
                return Math.floor(max * Math.random());
            };

            var int = function(str){
                return parseInt(str, 10);
            };

            $scope.a = {
                value: show? undefined : 1 + value(4),
                input: show
            };
            $scope.b = {
                value: !show? undefined : 1 + value(4),
                input: !show
            };
            $scope.operation = '+';

            $scope.result = 5 + value(5);

            var a = $scope.a;
            var b = $scope.b;
            var result = $scope.result;

            var checkValidity = function(){
                if (a.value && b.value) {
                    var calc = int(a.value) + int(b.value);
                    $scope.valid = calc == result;
                } else {
                    $scope.valid = false;
                }
                $scope.$apply(); // needed to solve 2 cycle delay problem;
            };


            $scope.$watch('a.value', function(){    
                checkValidity();
            });

            $scope.$watch('b.value', function(){    
                checkValidity();
            });
        }
    };
});

弹出:

$('#popup-button-email').bind('click', function(e) {
    // Prevents the default action to be triggered. 
    e.preventDefault();

    // Triggering bPopup when click event is fired
    $('#popup-second').bPopup();
});

一切看起来都很好,除了一件事。

在我关闭弹出窗口后,我打开另一个弹出窗口,它没有重新加载验证码,这很奇怪。

关闭弹出窗口后如何重新加载验证码的任何想法?

谢谢

4

1 回答 1

0

您只需要更新一两件事。在指令的检查有效性函数中, $scope.$apply()导致 $digest 循环在它已经在进行中时运行,因此中断它,以避免您可以在指令控制器的范围变量中拥有检查有效性函数并删除 $apply 的显式调用。此外,由于$('#popup-second').bPopup()调用实际上并不重新呈现指令,因此它始终显示指令的首次加载结果。因此,您可以做的是在视图中的该指令元素上使用 ng-if 并将其切换为onClosebPopup onOpen() 的事件。所以你的模板可以是:

<div id="popup-second" style="display: none">
    <div> From Controller : {{mymodel.captchaValid}}
    </div>
    <div ng-if="show">
      <simple-captcha valid="mymodel.captchaValid"></simple-captcha>
    </div>
</div>

将指令内的 checkvalidity 函数更新为:

$scope.checkValidity = function() {
    if (a.value && b.value) {
      var calc = int(a.value) + int(b.value);
      $scope.valid = calc == result;
    } else {
      $scope.valid = false;
    }
};

$scope.$watch('a.value', function(){    
    $scope.checkValidity();
});

$scope.$watch('b.value', function(){    
    $scope.checkValidity();
});

在视图控制器中调用 bPopup 为:

$('#popup-button-email').bind('click', function(e) {
    // Prevents the default action to be triggered.
    e.preventDefault();

    // Triggering bPopup when click event is fired
    $('#popup-second').bPopup({
      onOpen: function() {
        $scope.show = true;
        $scope.$digest();
      },
      onClose: function() {
        $scope.show = false;
        $scope.$digest();
      }
    });
});

因此,通过这种方式,每次触发时都会重新创建指令。因此,它会给出预期的结果,并且由于隔离范围,您可以在应用程序的同一视图或不同视图上多次重复使用它。

工作的笨蛋

于 2017-08-06T11:39:59.447 回答