当前位置: 首页 > 面试题库 >

为什么ngModel。$ setViewValue(...)无法从

华瀚漠
2023-03-14
问题内容

我正在编写需要隔离范围的指令,但是我想通过ngModel将其绑定到父范围。

这里的问题是父级的范围值未更改。

标记

<form name="myForm" ng-app="customControl">
    <div ng-init="form.userContent"></div>
    <div contenteditable name="myWidget" ng-model="form.userContent" required>Change me!</div>
    <span ng-show="myForm.myWidget.$error.required">Required!</span>
    <hr />
    <textarea ng-model="form.userContent"></textarea>
</form>

JS

angular.module('customControl', []).directive('contenteditable', function() {
    return {
        restrict : 'A', // only activate on element attribute
        require : '?ngModel', // get a hold of NgModelController
        scope: {},
        link : function(scope, element, attrs, ngModel) {
            if (!ngModel)
                return; // do nothing if no ng-model

            // Specify how UI should be updated
            ngModel.$render = function() {
                element.html(ngModel.$viewValue || '');
            };

            // Listen for change events to enable binding
            element.bind('blur keyup change', function() {
                        scope.$apply(read);
                    });
            read(); // initialize

            // Write data to the model
            function read() {
                ngModel.$setViewValue(element.html());
            }
        }
    };
});

演示:小提琴。

如果我不对指令使用隔离范围,则此方法效果很好

演示:小提琴。


问题答案:

原因是因为您正在为contenteditable指令创建隔离范围,ng- model所以同一元素上的指令也将获得该隔离范围。这意味着您有两个彼此不连接的不同作用域,它们都具有form.userContent分别更改的属性。我想您可以通过以下代码举例说明:

<!doctype html>
<html ng-app="myApp">
<head>
    <script src="http://code.jquery.com/jquery-1.9.1.min.js"></script>
    <script src="http://code.angularjs.org/1.0.5/angular.min.js"></script>
    <script>
    angular.module('myApp', []).controller('Ctrl', function($scope) {

    })
    .directive('contenteditable', function() {
        return {
            restrict : 'A', // only activate on element attribute
            require : '?ngModel', // get a hold of NgModelController
            scope: {},
            link : function(scope, element, attrs, ngModel) {
                if (!ngModel)
                    return; // do nothing if no ng-model

                setInterval(function() {
                    if (angular.element('#contenteditable').scope().form)
                        console.log(angular.element('#contenteditable').scope().form.userContent);

                    if (angular.element('#textarea').scope().form)
                        console.log(angular.element('#textarea').scope().form.userContent);
                }, 1000);

                // Specify how UI should be updated
                ngModel.$render = function() {
                    element.html(ngModel.$viewValue || '');
                };

                // Listen for change events to enable binding
                element.bind('blur keyup change', function() {
                            scope.$apply(read);
                        });
                read(); // initialize

                // Write data to the model
                function read() {
                    ngModel.$setViewValue(element.html());
                }
            }
        };
    });
    </script>
</head>
<body ng-controller="Ctrl">
    <form name="myForm">
        <div ng-init="form.userContent"></div>
        <div contenteditable name="myWidget" ng-model="form.userContent" id="contenteditable" required>Change me!</div>
        <span ng-show="myForm.myWidget.$error.required">Required!</span>
        <hr />
        <textarea ng-model="form.userContent" id="textarea"></textarea>
    </form>
</body>
</html>

正如您将在控制台中看到的那样,有两个不同的作用域,form.userContent如果您更改了textarea中的文本或更改了contenteditable
div中的文本,则它们分别更改。

因此,我敢打赌您在考虑“有足够的解释,然后给我解决方案!”。嗯,(据我所知)没有一个很好的解决方案,但是有一个可行的解决方案。您要做的是将模型的引用带入隔离的范围,并确保模型在隔离范围中的名称与父范围中的名称相同。

这是您要做的,而不是像这样创建一个空作用域:

...
scope: {}
...

您可以像这样绑定模型:

...
scope: {
    model: '=ngModel'
}
....

现在,您model在隔离范围内具有一个属性,该属性是form.userContent对父范围的引用。但ng- model不是在寻找model属性,而是form.userProperty在我们隔离的范围内寻找尚不存在的属性。因此,要解决此问题,我们将其添加到链接函数中:

scope.$watch('model', function() {
    scope.$eval(attrs.ngModel + ' = model');
});

scope.$watch(attrs.ngModel, function(val) {
    scope.model = val;
});

第一个监视form.userContent将来自指令外部的更改同步到我们的Isolated
form.userContent,第二个监视确保我们将隔离form.userContent上的所有更改传播到父作用域。

我意识到这是一个冗长的答案,而且可能不太容易遵循。因此,我很乐意澄清您觉得模糊的所有内容。



 类似资料:
  • 问题内容: 我有以下ckEditor指令。在下面的示例中,我看到了两个有关如何在编辑器中设置数据的示例的两种变体: 有人可以告诉我有什么区别: 而我应该使用。我看了一下角度文档,上面写着: 我不知道作者在文档中写这个是什么意思:-( 问题答案: 您正在寻找正确的文档,但可能只是您有些困惑。该和有一个明显的区别。就是这个: 如前所述: 视图中的实际字符串 (或对象) 值。 模型中控件绑定到的值。 我

  • 问题内容: 我有一个子过程命令,输出一些字符,例如’\ xf1’。我正在尝试将其解码为utf8,但出现错误。 上面抛出: 当我使用’latin-1’时它可以工作,但是utf8也不能工作吗?我的理解是latin1是utf8的子集。 我在这里想念什么吗? 编辑: 问题答案: 您已经将Unicode与UTF-8混淆了。Latin-1是Unicode的子集,但不是UTF-8的子集。 避免像瘟疫一样思考各个

  • 我无法解决getWindow();由于某种原因的方法...

  • 问题内容: 考虑以下: 我一直在寻找背后的逻辑,但是没有运气。值得一提的是,如果将结构更改为类,则效果很好。 总是可以添加一种解决方法,并将fooArray的每个对象映射为强制类型,将它们转换为Any类型,但这不是这里的问题。我正在寻找一种解释,为什么会这样。 有人可以解释一下吗? 问题答案: Swift 3更新 从Swift 3(特别是Xcode 8 beta 6附带的内部版本)开始,集合类型现

  • 问题内容: 我刚刚下载了Hibernate 4.0.1最终版本的新版本。当我键入以下代码时,很奇怪: Eclipse IDE在STRING关键字上显示错误。我确定我的构建路径中有hibernatejar,并且还包含了以下代码: 我只是想知道为什么会发生这种情况? 问题答案: 因为该字段在4.0中不存在。 3.5中的现场文档说: 不推荐使用。 看到 http://opensource.atlassi

  • 类无法解析,为什么?这就引出了一个问题:在哪里可以找到的javadoc? null