Как визуализировать временную шкалу событий в D3?

Я хочу визуализировать временные метки событий в моих данных, используя визуализацию, как показано ниже:

введите здесь описание изображения

Я предполагаю, что это можно сделать несколькими различными способами, и, учитывая отсутствие у меня опыта работы с D3, я хотел бы услышать рекомендацию о хорошем подходе. Возможно, этого можно добиться, замысловато манипулируя какой-то обычной визуализацией?

РЕДАКТИРОВАТЬ: Основываясь на отзывах от ee2Dev, я сделаю 4 строки вместо 2, отдельно записывая входящие и исходящие звонки/тексты. Что касается детализации, было бы лучше, если бы визуализация могла фиксировать данные за одну неделю с 5-минутными интервалами (одно текстовое сообщение будет равняться 5 минутам разговора, а 48-минутный звонок будет округлен до 50-минутного звонка). Это составит 7*24*12 = 2016 возможных интервалов, что кажется несколько разумным. Возможно, 10-минутные интервалы были бы более подходящими, чем 5-минутные, но я думаю, что код можно легко настроить для этого. Кое-что, что я не показал, это то, как должна быть отмечена полночь, чтобы показать, какие дни активны, а какие нет.

Ниже приведен мой код и мои образцы данных:

// Data:
timestamp (yyyy-MM-dd HH:mm),type
1/1/2015 10:12,inc_call
1/2/2015 10:12,inc_call
1/2/2015 10:12,out_text
1/3/2015 10:12,out_call
1/4/2015 10:12,inc_text
1/5/2015 10:12,inc_text


// Code
<!DOCTYPE html>
<meta charset="utf-8">
<style>

</style>
<body>
<script type="text/javascript" src="d3/d3.js"></script>
<script type="text/javascript" src="papaparse.js"></script> 
<script type="text/javascript" src="jquery.js"></script> 
<script type="text/javascript" src="jquery.tipsy.js"></script>
<link href="tipsy.css" rel="stylesheet" type="text/css" />
<script>

var width = 1000;
var height = 500;

var events;

var results = Papa.parse("events.csv", {
    header: true,
    download: true, // is needed even for local files as this interprets the input value as a path instead of simply the data
    dynamicTyping: true,
    delimiter: ",",
    skipEmptyLines: true,
    complete: function(results) {
        events = results.data;  
        CreateVisualizationFromData();
      }
});


var svg = d3.select("body").append("svg")
            .attr("width", width)
            .attr("height", height);

var total_interactions;

function CreateVisualizationFromData()
{

    total_interactions = events.length;
    console.log(total_interactions);

    svg
      .append("marker")
      .attr("id", "arrowhead")
      .attr("refX", 6 + 7)
      .attr("refY", 2)
      .attr("markerWidth", 6)
      .attr("markerHeight", 4)
      .attr("orient", "auto")
      .append("path")
      .attr("d", "M 0,0 V 4 L6,2 Z");


 }

person pir    schedule 06.03.2015    source источник
comment
Прежде всего: выбор визуализации не кажется удачным: входящие и исходящие звонки или тексты рисуются в одной последовательности. Сначала предлагаю вам разделить их на четыре последовательности. Как алгоритм должен выбирать отрисовку, если в один день много входящих и исходящих вызовов. Заполните половину области светло-голубым, а вторую половину темно-синим? Чередовать светлый с темно-синим? Во-вторых, обязательно покажите зрителю, какое время изображено: я не вижу, где заканчивается 01.01.2015 или где начинается 05.01.2015? Есть ли между 2-м и 4-м или мы показываем только 2 дня?   -  person ee2Dev    schedule 06.03.2015
comment
Вы можете выбрать 4 отдельных графика, так как это действительно больше подходит. В качестве альтернативы также используются четыре визуализации calendarView bl.ocks.org/mbostock/4063318 и вместо заполнив весь день одним цветом, сделайте дневной квадрат 24х24 и проведите по 1 вертикали на каждый час, если часовая дискретизация достаточно зернистая   -  person ee2Dev    schedule 06.03.2015
comment
Спасибо за комментарии. Я полностью согласен. Конечным результатом должно стать 4 строки. Данные будут показывать последовательные дни, так что это будет с 1-го по 5-е. Я думал о том, как проиллюстрировать смену дней (например, с помощью черных линий, пересекающих полночь), но я все еще обдумываю, как сделать так, чтобы это вписывалось в общую визуализацию. Я обновлю исходный текст соответственно.   -  person pir    schedule 06.03.2015
comment
какую степень детализации времени вы хотите изобразить? звонить да/нет в час? за 30 минут?   -  person ee2Dev    schedule 06.03.2015
comment
См. обновленный текст. Считаете ли вы, что 5-минутные интервалы в неделю — это слишком амбициозно?   -  person pir    schedule 06.03.2015
comment
Ну, вы можете это сделать, но если вы используете только ширину 1 пикселя для каждого интервала, у вас будет общая ширина 2016 пикселей + допустим 30 пикселей для меток!   -  person ee2Dev    schedule 06.03.2015
comment
Хе-хе, правда. Наверное, я не думал. Просто было бы интересно как-то увидеть, был ли человек активен с 48-минутным звонком или просто звонком продолжительностью 2 минуты. Возможно, другим вариантом было бы делать интервалы в 1 час, а затем кодировать этот интервал цветом в зависимости от активности?   -  person pir    schedule 06.03.2015


Ответы (1)


Чтобы дать вам отправную точку, используйте этот код и идите оттуда.

<!DOCTYPE html>
<meta charset="utf-8">
<style>

.axis {
  font: 10px sans-serif;
}

.axis path,
.axis line {
  fill: none;
  stroke: #000;
  shape-rendering: crispEdges;
}

.x.axis path {
  display: none;
}

</style>
<body>
<script src="http://d3js.org/d3.v3.min.js"></script>
<script>

var margin = {top: 20, right: 20, bottom: 30, left: 60},
    width = 960 - margin.left - margin.right,
    height = 200 - margin.top - margin.bottom;

var colorOf = d3.scale.category10();

var y = d3.scale.ordinal()
    .rangeRoundBands([height, 0], 0.3);

var x = d3.time.scale()
    .range([5, width]);

var yAxis = d3.svg.axis()
    .scale(y)
    .orient("left");

var xAxis = d3.svg.axis()
    .scale(x)
    .orient("bottom")
    .tickFormat(d3.time.format("%m/%d/%Y %Hh"));;

var parseDate = d3.time.format("%m/%d/%Y %H:%M").parse;

var svg = d3.select("body").append("svg")
    .attr("width", width + margin.left + margin.right)
    .attr("height", height + margin.top + margin.bottom)
  .append("g")
    .attr("transform", "translate(" + margin.left + "," + margin.top + ")");

d3.csv("timeseries.csv", type, function(error, data) {
  y.domain(data.map(function(d) { return d.type; }));
  x.domain(d3.extent(data, function(d) { return d.time; }));
  colorOf.domain(data.map(function(d) { return d.type; }));

  svg.append("g")
      .attr("class", "x axis")
      .attr("transform", "translate(0," + height + ")")
      .call(xAxis);

  svg.append("g")
      .attr("class", "y axis")
      .call(yAxis);

  svg.selectAll(".bar")
      .data(data)
    .enter().append("rect")
      .attr("class", "bar")
      .attr("y", function(d) { return y(d.type); })
      .attr("height", y.rangeBand())
      .attr("x", function(d) { return x(d.time); })
      .attr("width", 1)
      .style("fill", function(d) { return colorOf(d.type); })
});

function type(d) {
  d.time = parseDate(d.time);
  return d;
}

</script>

а вот файл timeseries.csv

time,type
1/1/2015 10:12,inc_call
1/2/2015 10:12,inc_call
1/2/2015 10:12,out_text
1/3/2015 10:12,out_call
1/4/2015 10:12,inc_text
1/5/2015 10:12,inc_text

А вот JSFiddle: http://jsfiddle.net/ee2todev/6u0a0qnw/ Возможно, вы захотите удалите ось y, замените ее метками, отформатируйте ось x... отрегулируйте цвета и т.д..

person ee2Dev    schedule 06.03.2015
comment
Спасибо, это кажется хорошей отправной точкой :) Не могли бы вы вкратце рассказать подробнее о том, что делают эти 3 строки кода? d3.csv("timeseries.csv", type, function(error, data) { y.domain(data.map(function(d) { return d.type; })); x.domain(d3.extent(data, function(d) { return d.time; })); colorOf.domain(data.map(function(d) { return d.type; })); - person pir; 09.03.2015
comment
За несколько строк до этого я определяю три переменные: colorOf, x и y. Каждая из переменных указывает на функцию, в частности на функцию масштаба. Шкалы — это функции, которые отображают входной домен в выходной диапазон. Диапазон каждой шкалы был указан выше - поэтому в этих трех строках я задаю правильный домен. - person ee2Dev; 09.03.2015
comment
Таким образом, первые строки задают домен как массив различных типов из ваших данных (['inc_call', 'out_text', 'out_call', 'inc_text']. Эти значения вы хотите сопоставить с разными координатами y. Вторая строка назначает минимальное и максимальное время в качестве домена для x. Последний сопоставляет все типы с цветами. Обратите внимание, что y и colorOf имеют один и тот же домен (массив разных типов), но разные диапазоны (координаты y или цвета) - person ee2Dev; 09.03.2015
comment
вы можете посмотреть документацию шкал d3, тогда вы сможете легко настроить ее. - person ee2Dev; 09.03.2015