Angularjs : '구문으로서의 컨트롤러'와 $ watch


153

controller as구문을 사용할 때 속성 변경을 구독하는 방법은 무엇입니까?

controller('TestCtrl', function ($scope) {
  this.name = 'Max';
  this.changeName = function () {
    this.name = new Date();
  }
  // not working       
  $scope.$watch("name",function(value){
    console.log(value)
  });
});
<div ng-controller="TestCtrl as test">
  <input type="text" ng-model="test.name" />
  <a ng-click="test.changeName()" href="#">Change Name</a>
</div>  

이건 어때? $ watch ()? 유효합니다 : this. $ watch ( 'name', ...)
Joao Polo

답변:


160

관련 컨텍스트를 바인딩하십시오.

$scope.$watch(angular.bind(this, function () {
  return this.name;
}), function (newVal) {
  console.log('Name changed to ' + newVal);
});

예 : http://jsbin.com/yinadoce/1/edit

최신 정보:

Bogdan Gersak의 대답은 실제로 동등한 것입니다. 두 대답 모두 this올바른 맥락에 구속력 이 있습니다. 그러나 나는 그의 대답이 더 깨끗하다는 것을 알았습니다.

그것이 무엇보다도 먼저, 당신은 그 배후의 기본 아이디어 를 이해해야 합니다 .

업데이트 2 :

ES6를 사용하는 사람들을 위해 arrow function올바른 컨텍스트 OOTB를 가진 함수를 얻을 수 있습니다.

$scope.$watch(() => this.name, function (newVal) {
  console.log('Name changed to ' + newVal);
});


9
이것과 $ scope의 혼합을 피하기 위해 $ scope없이 사용할 수 있습니까?
Miron

4
내가 아는 바는 없지만 완벽하게 괜찮습니다. $scope당신은 이런 종류의 방법을 제공하는 일종의 서비스입니다.
Roy Miloh

namein return this.name;이 컨트롤러 이름 인지 또는 name여기에 " " 속성 을 나타내는 지 알 수 있습니까 ?
Jannik Jochem

3
@Jannik, angular.bind경계 컨텍스트 (arg # 1)가있는 함수를 반환합니다. 이 경우 this컨트롤러 인스턴스 인을 함수 (arg # 2)에 바인딩 하므로 컨트롤러 인스턴스의 this.name속성 name을 의미합니다 .
Roy Miloh

나는 이것이 어떻게 작동하는지 이해했다고 생각합니다. 바운드 함수가 호출되면 단순히 관찰 된 값으로 평가됩니다.
Jannik Jochem

138

나는 보통 이것을한다 :

controller('TestCtrl', function ($scope) {
    var self = this;

    this.name = 'Max';
    this.changeName = function () {
        this.name = new Date();
   }

   $scope.$watch(function () {
       return self.name;
   },function(value){
        console.log(value)
   });
});

3
나는 이것이 최선의 대답이라는 데 동의하지만, 이것에 대한 혼란은 아마도 함수를 첫 번째 인수로 전달하고 $scope.$watch해당 함수를 사용하여 클로저에서 값을 반환하는 것입니다. 나는 이것의 다른 예를 아직 뛰어 넘지 못했지만 작동하고 최고입니다. 아래 답변 (예 $scope.$watch('test.name', function (value) {});:)을 선택하지 않은 이유 는 템플릿 또는 ui.router의 $ stateProvider에서 컨트롤러라는 이름을 하드 코딩해야하고 변경 사항이 있으면 실수로 감시자를 손상시킬 수 있기 때문입니다.
Morris Singer

또한이 답변과 현재 허용되는 답변 (을 사용하는 angular.bind)의 실질적인 차이점 은 클로저 내에 바인딩 this하거나 다른 참조를 추가 this하려는지 여부입니다. 이것들은 기능적으로 동등하며 제 경험상 이런 종류의 선택은 종종 주관적인 요구이며 매우 강한 의견의 문제입니다.
Morris Singer

1
ES6의 좋은 점 중 하나는 올바른 js 범위 를 얻기 위해 위에서 언급 한 두 가지 해결 방법을 수행하지 않아도된다는 것 입니다. $scope.$watch( ()=> { return this.name' }, function(){} ) 구조에 뚱뚱한 화살표
jusopi

1
당신은 또한 할 수 있습니다() => this.name
coblr

이 작업을 수행 $scope.$watchCollection하고 여전히 oldVal, newVal매개 변수를 얻을 수 있습니까?
Kraken

23

당신이 사용할 수있는:

   $scope.$watch("test.name",function(value){
        console.log(value)
   });

이것은 예제와 함께 JSFiddle 을 작동 시킵니다 .


25
이 접근법의 문제점은 JS가 이제 HTML에 의존하여 $ watch가 작동하기 위해 컨트롤러를 모든 곳에서 동일한 이름 (이 경우 "test")으로 바인딩해야한다는 것입니다. 미묘한 버그를 도입하기가 매우 쉽습니다.
jsdw

모든 것이 지시문 인 Angular 2와 같이 Angular 1을 작성하는 경우 훌륭하게 작동합니다. Object.observe는 지금 놀랍습니다.
Langdon

13

다른 답변에서 설명한대로 "TestCtrl as test"의 "test"를 사용하는 것과 유사하게 "self"를 범위에 할당 할 수 있습니다.

controller('TestCtrl', function($scope){
    var self = this;
    $scope.self = self;

    self.name = 'max';
    self.changeName = function(){
            self.name = new Date();
        }

    $scope.$watch("self.name",function(value){
            console.log(value)
        });
})

이런 식으로 DOM에 지정된 이름 ( "TestCtrl as test")에 연결되지 않으며 함수에 .bind (this) 할 필요가 없습니다.

... 지정된 원본 HTML과 함께 사용 :

<div ng-controller="TestCtrl as test">
    <input type="text" ng-model="test.name" />
    <a ng-click="test.changeName()" href="#">Change Name</a>
</div>

한 가지만 알고 싶습니다. 즉 $scope서비스입니다. 따라서을 추가 $scope.self = this하고 다른 컨트롤러에서 동일한 작업을 수행하면 어떻게됩니까?
Vivek Kumar

12

AngularJs 1.5는 ControllerAs 구조에 대한 기본 $ ctrl을 지원합니다.

$scope.$watch("$ctrl.name", (value) => {
    console.log(value)
});

$ watchGroup을 사용할 때 저에게 적합하지 않습니다. 알려진 한계입니까? 이 기능에 대한 링크를 공유 할 수 없습니다.
user1852503

@ user1852503 docs.angularjs.org/guide/component 비교 테이블 지시어 / 컴포넌트 정의를 참조하고 'controllerAs'레코드를 확인하십시오.
Niels Steenbeek

지금은 이해. 당신의 대답은 약간 오도합니다. 식별자 $ ctrl은 컨트롤러와 기능과 관련이 없으며 ($ index는 예를 들어 ng-repeat와 같이) 구성 요소 내부의 컨트롤러에 대한 기본 이름 일뿐입니다 (질문은 구성 요소).
user1852503

@ user1852503 1) $ ctrl은 컨트롤러와 컨트롤러를 상호 연관시킵니다. 2) "<div ng-controller ="TestCtrl as test ">"라는 언급이 있기 때문에 컴포넌트에 대한 질문입니다. 3)이 페이지의 모든 대답은 어떻게 든 내 대답과 동일합니다. 4) 설명서에 관해서 $ watchGroup은 $ watch를 기반으로하기 때문에 $ ctrl.name을 사용할 때 제대로 작동합니다.
Niels Steenbeek

2

실제로 $ watch ()의 첫 번째 인수로 함수를 전달할 수 있습니다.

 app.controller('TestCtrl', function ($scope) {
 this.name = 'Max';

// hmmm, a function
 $scope.$watch(function () {}, function (value){ console.log(value) });
 });

이것은 this.name 참조를 반환 할 수 있음을 의미합니다.

app.controller('TestCtrl', function ($scope) {
    this.name = 'Max';

    // boom
    $scope.$watch(angular.bind(this, function () {
    return this.name; // `this` IS the `this` above!!
    }), function (value) {
      console.log(value);
    });
});

controllerAs 주제 https://toddmotto.com/digging-into-angulars-controller-as-syntax/ 에 대한 흥미로운 게시물 읽기



0

ES6 구문으로 $ watch를 작성하는 것은 예상만큼 쉽지 않았습니다. 수행 할 수있는 작업은 다음과 같습니다.

// Assuming
// controllerAs: "ctrl"
// or
// ng-controller="MyCtrl as ctrl"
export class MyCtrl {
  constructor ($scope) {
    'ngInject';
    this.foo = 10;
    // Option 1
    $scope.$watch('ctrl.foo', this.watchChanges());
    // Option 2
    $scope.$watch(() => this.foo, this.watchChanges());
  }

  watchChanges() {
    return (newValue, oldValue) => {
      console.log('new', newValue);
    }
  }
}

-1

참고 : View와 Controller가 경로 나 지시문 정의 개체를 통해 연결된 경우에는 작동하지 않습니다. 아래에 표시된 것은 HTML에 "SomeController as SomeCtrl"이있는 경우에만 작동합니다. Mark V.가 아래 의견에서 지적한 것처럼 Bogdan 이하는 것처럼하는 것이 낫습니다.

나는 var vm = this;컨트롤러의 시작 부분에 "this"라는 단어를 내 방식에서 꺼내기 위해 사용합니다. 그리고 vm.name = 'Max';시계에서 나는 return vm.name. @Bogdan이 "self"를 사용하는 것처럼 "vm"을 사용합니다. 이 변수는 "vm"또는 "self"가 필요하기 때문에 "this"라는 단어는 함수 내부에서 다른 컨텍스트를 갖기 때문에 필요합니다. (그래서 this.name을 반환해도 작동하지 않습니다.) 예, $ watch에 도달하려면 아름다운 "컨트롤러"솔루션에 $ scope를 주입해야합니다. John Papa의 스타일 가이드를 참조하십시오 : https://github.com/johnpapa/angularjs-styleguide#controllers

function SomeController($scope, $log) {
    var vm = this;
    vm.name = 'Max';

    $scope.$watch('vm.name', function(current, original) {
        $log.info('vm.name was %s', original);
        $log.info('vm.name is now %s', current);
    });
}

11
HTML에 "SomeController as vm"이있는 한 작동합니다. 시계 식의 "vm.name"은 "var vm = this;"와 아무 관련이 없습니다. Bogdan이 위에서 설명한 것처럼 "controller as"와 함께 $ watch를 사용하는 유일한 안전한 방법은 함수를 첫 번째 인수로 전달하는 것입니다.
Mark Visser

-1

$ scope (및 $ watch!)없이이 작업을 수행하는 방법은 다음과 같습니다. 실수 5 가지-남용 시계

"controller as"구문을 사용하는 경우 $ scope를 사용하지 않는 것이 좋습니다.

JSFiddle의 코드는 다음과 같습니다 . (서비스를 사용하여 이름을 보유하고 있습니다. 그렇지 않으면 ES5 Object.defineProperty의 설정 및 get 메소드가 무한 호출을 유발합니다.

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

app.factory('testService', function() {
    var name = 'Max';

    var getName = function() {
        return name;
    }

    var setName = function(val) {
        name = val;
    }

    return {getName:getName, setName:setName};
});

app.controller('TestCtrl', function (testService) {
    var vm = this;

    vm.changeName = function () {
        vm.name = new Date();
    }

    Object.defineProperty(this, "name", {
        enumerable: true,
        configurable: false,
        get: function() {
            return testService.getName();
        },
        set: function (val) {
            testService.setName(val);
            console.log(vm.name);
        }
    }); 
});

바이올린이 작동하지 않아 객체 속성이 관찰되지 않습니다.
Rootical V.

@RooticalV. 바이올린이 작동합니다. (AngualrJS를 실행할 때로드 유형을 nowrap-head / nowrap-body로 지정해야합니다
Binu Jasim

죄송하지만 솔루션을 실행하는 데 시간이 많이 걸리기 때문에이 프로그램을 실행하지 못했습니다.
happyZZR1400

@happy 라이브러리를 Angular 1.4로 선택하십시오. (2.0이 작동하는지 확실하지 않음) 및 유형을 줄 바꿈 없음으로 설정하고 실행을 누릅니다. 작동해야합니다.
Binu Jasim 님
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.