datahuborg/datahub

View on GitHub
src/apps/dbwipes/static/js/summary/drawingview.js

Summary

Maintainability
F
1 wk
Test Coverage
define(function(require) {
  var Backbone = require('backbone'),
      Handlebars = require('handlebars'),
      $ = require('jquery'),
      d3 = require('d3'),
      _ = require('underscore'),
      Where = require('summary/where'),
      util = require('summary/util');


  var Drawing = Backbone.Model.extend({
    defaults: function() {
      return {
        isDrawing: false,
        path: []
      };
    },

    interpolate: function(xs) {
      var xys = this.get('path');
      xys.sort(function(a,b) { return a[0] - b[0]; });

      if (xys.length == 0) 
        return null;
      if (xys[0][0] > xs[0] || xys[xys.length-1][0] < xs[xs.length-1])
        return null;

      var idx = -1,
          xyidx = 0,
          xy = xys[0],
          xyn = xys[1],
          ys = [],
          x = null;
      while (++idx < xs.length ){
        x = xs[idx];
        while (x >= xyn[0] && ++xyidx < xys.length) {
          xy = xyn;
          xyn = xys[xyidx]
        }
        if (xyidx >= xys.length) break;
        var ratio = (x-xy[0]) / (xyn[0]-xy[0]);
        ys.push(ratio * (xyn[1]-xy[1]) + xy[1])
      }
      return ys;
    },

    inverty: function(y) {
      return this.get('state').yscales.invert(y);
    }
  });

  var DrawingView = Backbone.View.extend({
    tagName: "div",

    // expect state
    initialize: function(attrs) {
      this.state = attrs.state || {w: 500, h: 500};
      this.model = new Drawing({state: attrs.state})
    },


    renderPath: function() {
      this.d3path.attr('d', this.d3line(this.model.get('path')))
    },


    renderDrawing: function() {
      this.$el.empty();

      var d3svg = this.d3svg = d3.select(this.el).append("svg")
        .classed("drawingpane", true)
        .attr("width", this.state.w)
        .attr("height", this.state.h)

      var g = this.d3g = d3svg.append('g')
        .classed("drawing-container", true)
        .style("pointer-events", "all")
        .on('mousedown', this.onDown.bind(this))
        .on('mousemove', this.onMove.bind(this))
        .on('mouseup', this.onUp.bind(this))

      g.append("rect")
        .style("visibility", "hidden")
        .attr("width", this.state.w)
        .attr("height", this.state.h)

      var line = this.d3line = d3.svg.line()
        .x(function(d) { return d[0] })
        .y(function(d) { return d[1] })
        .interpolate('basis');

      var path = this.d3path = g.append('path')
        .style({
          'stroke': 'black', 
          'stroke-width': '3px', 
          'fill': 'none'
        });
    },

    onDown: function() {
      this.model.set({
        isDrawing: true,
        path: []
      });
      this.renderPath();
    },

    onMove: function() {
      if (!this.model.get("isDrawing")) 
        return;

      var xy = d3.mouse(this.d3g[0][0]),
          path = this.model.get("path");

      function find_and_insert(path, x, y) {
        var prev = null;
        var insertidx = null;
        var dir = (path.length == 1)? null : ((path[0][0] < path[path.length-1][0])? 'inc' : 'dec');
        for (var idx = 0; idx < path.length; idx++) {
          var cur = path[idx][0];
          if (prev != null) {
            if (dir == 'inc' && prev < x && x < cur) {
                insertidx = idx-1;
            } else if (dir == 'dec' && cur < x && x < prev)  {
                insertidx = idx;
            }
          }
          prev = cur;
        }

        if (insertidx != null) {
          var minidx = maxidx = insertidx;
          while(minidx > 0) {
            if (Math.abs(path[minidx-1][0]-path[insertidx][0]) > 10)
              break;
            minidx--;
          }
          while(maxidx < path.length-1) {
            if (Math.abs(path[maxidx+1][0]-path[insertidx][0]) > 10)
              break;
            maxidx++;
          }
          path.splice(minidx, (maxidx-minidx)+1, [path[insertidx][0], y]);
        } else {
          if (dir == 'inc') {
            if (x < path[0][0]) 
              path.unshift(xy);
            else if (x > path[path.length-1][0])
              path.push(xy);
          } else if (dir == 'dec') {
            if (x > path[0][0]) 
              path.unshift(xy);
            if (x < path[path.length-1][0])
              path.push(xy);
          } else {
            if (path.length <= 1)
              path.push(xy);
            else
              return null;
          }
          return 1;
        }
        return insertidx;
      }


      if (path.length >= 1) {
        var insertidx = find_and_insert(path, xy[0], xy[1]);
        if (insertidx != null) {
          this.renderPath();
        }
      } else if (path.length == 1) {
        if (xy[0] != path[path.length-1][0]) {
          path.push(xy);
          this.renderPath();
        }
      } else {
        path.push(xy);
        this.renderPath();
      }

    },

    onUp: function() {
      if (!this.model.get('isDrawing')) 
        return;
      if (this.model.get("path").length <= 1) 
        this.model.get("path", [])
      this.model.set("isDrawing", false);
      this.trigger("change:drawing");
    },


    status: function() { 
      return this.d3g.style('pointer-events'); 
    },

    disable: function() { 
      return this.d3g.style('pointer-events', 'none') 
    },

    enable: function() { 
      return this.d3g.style('pointer-events', 'all') 
    },

    render: function() {
      this.renderDrawing();
      return this;
    }
  });

  return DrawingView;

});