动态地将Angularjs绑定到新创建的html元素

时间:2013-11-08 19:59:36

标签: angularjs model-binding

我有一个包含多个标签的标签页,一旦点击就会调用服务来返回一些数据。其中一些数据返回html表单并且非常随机。我想收集输入的值,并通过服务将数据发送回服务器。我遇到的问题是我无法从动态创建的html中的输入元素中获取数据。

我创建了Plunker以显示问题所在。请注意,html值可以随时更改,因此硬编码html将无法正常工作。这里是来自plunker的代码,但请查看plunker以了解最新情况。

app.js

var app = angular.module('plunker', []);

app.controller('MainCtrl', function($scope, $sce, $compile) {
    $scope.name = 'World';
    $scope.html = "";

    $scope.htmlElement = function(){
        var html = "<input type='text' ng-model='html'></input>";
        return $sce.trustAsHtml(html);
    }

});

的index.html

<!DOCTYPE html>
<html ng-app="plunker">

  <head>
    <meta charset="utf-8" />
    <title>AngularJS Plunker</title>
    <script>document.write('<base href="' + document.location + '" />');</script>
    <link rel="stylesheet" href="style.css" />
    <script src="https://ajax.googleapis.com/ajax/libs/angularjs/1.2.0-rc.3/angular.js"></script>
    <script src="app.js"></script>
  </head>

  <body ng-controller="MainCtrl">
    <p>Hello {{name}}!</p>

    <div ng-bind-html="htmlElement()"></div>

    {{html}}

  </body>

</html>

3 个答案:

答案 0 :(得分:26)

一种解决方案是将ngInclude与$ templateCache一起使用,如Plunker所示。

有几点需要注意。

首先,您可以使用服务获取模板并将其添加到$ templateCache,如上所述here(示例已复制):

myApp.service('myTemplateService', ['$http', '$templateCache', function ($http, $templateCache) {
  $http(/* ... */).then(function (result) {
    $templateCache.put('my-dynamic-template', result);
  });
}]);

然后您可以将其包含在模板中,如下所示:

<div ng-include="'my-dynamic-template'"></div>

ngInclude将允许对html字符串进行数据绑定,因此您不需要ngBindHtml。

第二个是,当ngInclude创建一个新范围时,访问新创建范围之外的html属性将无法正常工作,除非您通过父范围内的对象访问它(例如{{1而不是ng-model="data.html"。请注意,在这种情况下,父范围中的ng-model="html"是使得html可以在ngInclude范围之外访问的原因。

(有关您为何应始终在ngModel中使用点数的详细信息,请参阅this answer。)


修改

正如您所指出的,使用服务返回HTML时,ngInclude选项的用处不大。

以下是编辑后的plunker,其中包含使用$ compile的基于指令的解决方案,如上面David的评论所示。

相关补充:

$scope.data = {}

答案 1 :(得分:6)

根据Sarah的回答,我创建了一个结构来放置指令

.directive('dynamic', function(AmazonService, $compile) {
  return {
    restrict: 'E',
    link: function(scope, element, attrs) {
      AmazonService.getHTML()
     .then(function(result){
       element.replaceWith($compile(result.data)(scope));
     })
     .catch(function(error){
       console.log(error);
     });
   }
 };
});

在html中:

<dynamic></dynamic>

谢谢Sarah,帮了很多!!!

答案 2 :(得分:0)

我有一个带有一些ng-repeat的dinamyc表,然后当我尝试使用javascript回调函数填充一列时,它只给出了html文本,如

<td class="tableList_"+myValue> "span class=someclass> some_text /span>" </td>
<td class="tableList_"+myValue> "span class=someclass> some_text /span>" </td>
<td class="tableList_"+myValue> "span class=someclass> some_text /span>" </td>

所以我用jquery解决了我的问题:

$(".tableListFilas td").each(function(){
   var td_class = $(this).attr("class");
     if(td_class == 'tableList_'+titulo)
     {
       **var toExtraHtml = $(this).text();**
       **$(this).html(toExtraHtml);**
     }
});

然后最终输出结果很好:

<td class="tableList_COLORS"> <span class=someclass>some_text</span> </td>
<td class="tableList_COLORS"> <span class=someclass>some_text</span> </td>
<td class="tableList_COLORS"> <span class=someclass>some_text</span> </td>