D3js / AngularJS - перетаскивание и привязка данных к координатам прямоугольника - PullRequest
0 голосов
/ 03 сентября 2018

Я хочу связать координаты x, y прямоугольника (записывается как «Координата»). И перетащите прямоугольник, надеясь, что запись (координата x, y) изменится синхронно.

Ниже приведена часть кода. (Полный код jsbin )

HTML

<tr ng-repeat="item in data">
    <td>({{ item.x }}, {{ item.y }})</td>
    <td>{{ item.width }}</td>
    <td>{{ item.height }}</td>
</tr>

JS

$scope.data = [{ 'x': 30, 'y': 50, 'width': 90, 'height': 70 }];

var drag = d3.drag()        
    .on('drag', function (d) {
        d3.select(this)
            .attr('x', d.x = d3.event.x)
            .attr('y', d.y = d3.event.y)
    })
    .on('end', function (d) {
        //update  coordinate x ,y to array
        arrayNum = this.id;   
        $scope.data.splice(arrayNum, 1, { 'x': d.x, 'y': d.y, 'width': d.width, 'height': d.height });
        console.log($scope.data);
    });

У меня также есть $ scope.data.splice для обновления массива. И это действительно обновляет $ scope.data . Но это не работает в представлении браузера. Как я могу изменить? Или на что я могу сослаться? Большое спасибо!

1 Ответ

0 голосов
/ 03 сентября 2018

Кажется, что angular не знает, что его область обновляется событием d3.

Я добавил $scope.$apply() в ваш обработчик on end, и отображение обновляется в представлении, как и предполагалось, после завершения каждого события перетаскивания.

var mainApp = angular.module("mainApp", []);

mainApp.controller('Controller', function($scope) {
  $scope.data = [{
    'x': 30,
    'y': 50,
    'width': 90,
    'height': 70
  }];

  var drag = d3.drag()
    .on('drag', function(d) {
      d3.select(this)
        .attr('x', d.x = d3.event.x)
        .attr('y', d.y = d3.event.y)

    })
    .on('end', function(d) {
      arrayNum = this.id;

      // Ensure angular knows about the update to its scope
      $scope.$apply(function() {
        //update  coordinate x ,y to array
        $scope.data.splice(arrayNum, 1, {
          'x': d.x,
          'y': d.y,
          'width': d.width,
          'height': d.height
        });
      });
      console.log($scope.data);
    });

  //create SVG
  var svg = d3.select('.Content')
    .append('svg')
    .attr('width', 300)
    .attr('height', 300)
    .style('border', '1px solid #000');

  var container = svg.append('g');

  container.append('svg:image')
    .attr('xlink:href', 'dog.jpg')
    .attr('x', 0)
    .attr('y', 0);

  container.data($scope.data)
    .append('rect')
    .attr('x', function(d) {
      return d.x;
    })
    .attr('y', function(d) {
      return d.y;
    })
    .attr('width', function(d) {
      return d.width;
    })
    .attr('height', function(d) {
      return d.height;
    })
    .attr('id', function(d, i) {
      return i;
    })
    .style('cursor', 'pointer')
    .call(drag);
});

Кроме того, прочитайте этот ответ , чтобы лучше понять, почему вам может потребоваться сделать это при работе с d3.

...