block by nitaku 9933ee9ec46aacb1fe2d0d54840c84c9

Vue.js shared state

Full Screen

Like in the previous example, two Vue.js components are instantiated, each one displaying a different matrix of numbers. In this example, the two views have a notion of shared selection (the orange square). You can click another square to change the selection in both views.

index.js

// Generated by CoffeeScript 1.10.0
(function() {
  var app;

  Vue.component('matrix', {
    props: {
      data: {
        type: Array,
        required: true
      },
      value: {
        type: Object,
        "default": function() {
          return {
            i: 0,
            j: 0
          };
        }
      },
      padding: {
        type: Number,
        "default": 1
      },
      cellsize: {
        type: Number,
        "default": 20
      }
    },
    template: '<svg :width="ncols*cellsize" :height="nrows*cellsize" class="matrix">\n  <g v-for="(row,i) in data">\n    <rect\n      v-for="(cell,j) in row"\n\n      :class="{cell: true, selected: value.i == i && value.j == j}"\n\n      :x="j*cellsize + cellsize/2 - scale(cell)/2"\n      :y="i*cellsize + cellsize/2 - scale(cell)/2"\n      :width="scale(cell)"\n      :height="scale(cell)"\n      \n      v-on:click="select({i:i,j:j})"\n    />\n  </g>\n</svg>',
    computed: {
      max: function() {
        return d3.max(this.data, function(r) {
          return d3.max(r);
        });
      },
      nrows: function() {
        return this.data.length;
      },
      ncols: function() {
        if (this.data.length > 0) {
          return this.data[0].length;
        } else {
          return 0;
        }
      },
      scale: function() {
        return d3.scaleSqrt().domain([0, this.max]).range([0, Math.max(0, this.cellsize - this.padding)]);
      }
    },
    methods: {
      select: function(d) {
        return this.$emit('input', d);
      }
    }
  });

  app = new Vue({
    el: '#app',
    data: {
      matrix1: [[1, 2, 3], [4, 5, 6], [9, 9, 9]],
      matrix2: [[12, 24, 35], [45, 15, 60], [60, 60, 60]],
      shared_sel: {
        i: 1,
        j: 1
      }
    }
  });

}).call(this);

index.html

<!DOCTYPE html>
<html>
<head>
  <meta charset="utf-8">
  <title>Vue.js Shared State</title>
  <link type="text/css" href="index.css" rel="stylesheet"/>
  <script src="https://d3js.org/d3.v4.min.js"></script>
  <script src="https://unpkg.com/vue/dist/vue.js"></script>
</head>
<body>
  
  <div id="app">
    <matrix
      :data="matrix1"
      v-model="shared_sel"
      :cellsize="60"
    />
    <matrix
      :data="matrix2"
      v-model="shared_sel"
      :cellsize="40"
      :padding="2"
    />
  </div>
  
  <script src="index.js"></script>
</body>
</html>

index.coffee

Vue.component 'matrix',
  props:
    data:
      type: Array
      required: true
    value: # selection
      type: Object
      default: () -> {i:0, j:0}
    padding:
      type: Number
      default: 1
    cellsize:
      type: Number
      default: 20
  template: '''
    <svg :width="ncols*cellsize" :height="nrows*cellsize" class="matrix">
      <g v-for="(row,i) in data">
        <rect
          v-for="(cell,j) in row"

          :class="{cell: true, selected: value.i == i && value.j == j}"

          :x="j*cellsize + cellsize/2 - scale(cell)/2"
          :y="i*cellsize + cellsize/2 - scale(cell)/2"
          :width="scale(cell)"
          :height="scale(cell)"
          
          v-on:click="select({i:i,j:j})"
        />
      </g>
    </svg>
  '''
  computed:
    max: () -> d3.max @data, (r) -> d3.max r
    nrows: () -> @data.length
    ncols: () -> if @data.length > 0 then @data[0].length else 0
    scale: () ->
      d3.scaleSqrt()
        .domain [0, @max]
        .range [0, Math.max(0,@cellsize-@padding)]
  methods:
    select: (d) ->
      @$emit 'input', d

app = new Vue
  el: '#app'
  data:
    matrix1: [
      [1,2,3],
      [4,5,6],
      [9,9,9]
    ],
    matrix2: [
      [12,24,35],
      [45,15,60],
      [60,60,60]
    ],
    shared_sel: {i:1,j:1}
  

index.css

body, html {
  padding: 0;
  margin: 0;
  height: 100%;
}
.matrix {
  margin: 20px;
}
.cell {
  fill: teal;
}
.cell.selected {
  fill: orange;
}