web-dev-qa-db-ja.com

AngularJS:jQueryで変更するとng-modelバインディングが更新されない

これは私のHTMLです:

<input id="selectedDueDate" type="text" ng-model="selectedDate" />

ボックスに入力すると、モデルは2-way-bindingメカニズムを介して更新されます。甘い。

ただし JQuery経由でこれを行う場合...

$('#selectedDueDate').val(dateText);

モデルは更新されません。どうして?

98
Greg

Angularはその変更について知りません。このためには、$scope.$digest()を呼び出すか、$scope.$apply()内で変更を行う必要があります。

$scope.$apply(function() { 
   // every changes goes here
   $('#selectedDueDate').val(dateText); 
});

this を参照して理解を深めてくださいdirty-checking

UPDATEここ は例です

133

使用するだけです。

$('#selectedDueDate').val(dateText).trigger('input');
27
Jan Kuri

変数をスコープに対して直接配置しないと、より確実に更新されることがわかりました。

いくつかの「dateObj.selectedDate」を使用してみて、コントローラーでselectedDateをdateObjオブジェクトに追加します。

$scope.dateObj = {selectedDate: new Date()}

これは私のために働いた。

16
Ben Taliadoros

これを試して

var selectedDueDateField = document.getElementById("selectedDueDate");
var element = angular.element(selectedDueDateField);
element.val('new value here');
element.triggerHandler('input');
4
Ray

関数の最後で次の行を実行するだけです。

$ scope。$ apply()

3
Rohan M Nabar

角度の範囲外で何が起こっても、Angularは決してそれを知りません。

ダイジェストサイクルでは、モデル->コントローラから、次にコントローラ->モデルから変更が加えられます。

最新のモデルを見る必要がある場合は、ダイジェストサイクルをトリガーする必要があります

ただし、ダイジェストサイクルが進行中の可能性があるため、サイクルを確認して初期化する必要があります。

できれば、常に安全な申請を行ってください。

       $scope.safeApply = function(fn) {
            if (this.$root) {
                var phase = this.$root.$$phase;
                if (phase == '$apply' || phase == '$digest') {
                    if (fn && (typeof (fn) === 'function')) {
                        fn();
                    }
                } else {
                    this.$apply(fn);
                }
            }
        };


      $scope.safeApply(function(){
          // your function here.
      });

Ng-modelは入力イベントをリッスンし、スコープが更新されるため、入力要素のchangeイベントをトリガーする必要があります。ただし、通常のjQueryトリガーは機能しませんでした。しかし、これは魅力のように機能するものです

$("#myInput")[0].dispatchEvent(new Event("input", { bubbles: true })); //Works

以下は機能しませんでした

$("#myInput").trigger("change"); // Did't work for me

合成イベントの作成とディスパッチ について詳しく読むことができます。

1
Hari Das

AngularJSは、文字列、数値、およびブール値を値で渡しますが、配列およびオブジェクトを参照で渡します。したがって、空のオブジェクトを作成し、日付をそのオブジェクトのプロパティにすることができます。そのようにして、angularはモデルの変更を検出します。

コントローラー内

app.module('yourModule').controller('yourController',function($scope){
$scope.vm={selectedDate:''}
});

HTMLで

<div ng-controller="yourController">
<input id="selectedDueDate" type="text" ng-model="vm.selectedDate" />
</div>
1
Himanshu Mittal

ただ使用する:

$('#selectedDueDate').val(dateText).trigger('input');

の代わりに:

$('#selectedDueDate').val(dateText);
0
Wekerle Tibor

.val(value)へのすべての呼び出しを行い、angular要素を更新するjQuery用のこの小さなプラグインを作成しました:

(function($, ng) {
  'use strict';

  var $val = $.fn.val; // save original jQuery function

  // override jQuery function
  $.fn.val = function (value) {
    // if getter, just return original
    if (!arguments.length) {
      return $val.call(this);
    }

    // get result of original function
    var result = $val.call(this, value);

    // trigger angular input (this[0] is the DOM object)
    ng.element(this[0]).triggerHandler('input');

    // return the original result
    return result; 
  }
})(window.jQuery, window.angular);

JQueryとangular.jsおよびval(value)の更新がNiceになった後、このスクリプトをポップするだけです。


縮小版:

!function(n,t){"use strict";var r=n.fn.val;n.fn.val=function(n){if(!arguments.length)return r.call(this);var e=r.call(this,n);return t.element(this[0]).triggerHandler("input"),e}}(window.jQuery,window.angular);

例:

// the function
(function($, ng) {
  'use strict';
  
  var $val = $.fn.val;
  
  $.fn.val = function (value) {
    if (!arguments.length) {
      return $val.call(this);
    }
    
    var result = $val.call(this, value);
    
    ng.element(this[0]).triggerHandler('input');
    
    return result;
    
  }
})(window.jQuery, window.angular);

(function(ng){ 
  ng.module('example', [])
    .controller('ExampleController', function($scope) {
      $scope.output = "output";
      
      $scope.change = function() {
        $scope.output = "" + $scope.input;
      }
    });
})(window.angular);

(function($){  
  $(function() {
    var button = $('#button');
  
    if (button.length)
      console.log('hello, button');
    
    button.click(function() {
      var input = $('#input');
      
      var value = parseInt(input.val());
      value = isNaN(value) ? 0 : value;
      
      input.val(value + 1);
    });
  });
})(window.jQuery);
<script src="https://ajax.googleapis.com/ajax/libs/angularjs/1.2.23/angular.min.js"></script>
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<div ng-app="example" ng-controller="ExampleController">
  <input type="number" id="input" ng-model="input" ng-change="change()" />
  <span>{{output}}</span>
  <button id="button">+</button>
</div>

この回答は 別の同様の質問に対する私の回答 からそのままコピーされました。

0
dav_i