D3.js Multi-Series Chart mit Y-Wert Verfolgung

Arbeits lösung: Im Moment arbeite ich am Styling und an der Lösung einiger Probleme in Bezug auf mein Problem beim Erstellen eines Diagramms, das aus mehreren Datenreihen mit Werteverfolgung besteht. Ich werde versuchen, Ihnen so schnell wie möglich ein Beispiel für den Arbeitscode zu geben, damit jemand, der auf dasselbe oder ein ähnliches Problem stößt wie ich, daran arbeiten kann. Im Moment sind die meisten Tipps, die ich verwendet habe, in den Kommentaren unten.

Dies ist meine erste Frage zu StackOverflow und ich freue mich darauf, welche Antworten Sie auf mein Problem haben könnten.

Kürzlich habe ich ein Projekt bekommen, in dem ich Javascript-Code zum Erzeugen von Diagrammen schreiben muss und in dem ich Y-Werte von jeder Zeile des Diagramms gleichzeitig lesen kann. Ich bin ein Neuling im D3-Framework und kann jetzt CSV-Daten lesen, Diagramme mit mehreren Reihen erstellen und den Y-Wert verfolgen und lesen, aber nur, wenn ich Diagramme aus einer einzelnen Datenreihe erstelle. Ich habe versucht, mehrere ähnliche Funktionen zu erstellen, mit denen Daten aus verschiedenen Datenreihen nachverfolgt werden können, aber dies funktioniert nicht. In der Konsole sehe ich, dass das Y nach meinem Verständnis null ist. Ich habe Beispiele von der D3-Website verwendet, um es zu lernen, und vorerst wird der Code diesen Beispielen sehr ähnlich sein.

Später würde ich einige andere Dinge damit machen müssen, aber ich denke, dass ich nach der Lösung dieses Problems in der Lage sein werde, weiterzumachen. Es wird geben wie:

Reduzieren Sie die Daten aus CSV durch Code, da ich die Header-Informationen löschen mussÄndern Sie den visuellen Stil des Diagramms und bearbeiten Sie die Achsenskalierung

Im Moment habe ich so etwas. Tut mir leid, wenn es ein bisschen chaotisch ist, aber ich lerne immer noch und probiere viele verschiedene Dinge aus. Ich habe auch einen Screenshot hinzugefügt, wie es für mich aussieht, und einige Konsoleninformationen, die ich bekommen könnte. Ich hoffe, es wird euch helfen zu sehen, was ich falsch mache und was ich lernen müsste. Auch dies ist nicht mein einziger Ansatz und es wäre zu lang, sie alle zu zeigen.

BEARBEITEN: Ich versuche einen etwas anderen Ansatz. Am Ende der Seite werde ich zeigen, was ich bis jetzt getan habe.

EDIT2: Tut mir leid, wenn ich nicht präzise genug über mein Ziel war. Damit versuche ich, alle Y-Achsenwerte der gezeichneten Linien (es werden 4 davon sein) gleichzeitig auf einem X-Achsenwert zu lesen. Ich habe einen Screenshot des zweiten Codes hinzugefügt, in dem ich nur einen Y-Achsenwert lesen kann und den über einen nicht lesen kann.

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

body {
  font: 10px sans-serif;
}

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

.x.axis path {
  display: none;
}

.line {
  fill: none;
  stroke: steelblue;
  stroke-width: 1.5px;
}

.overlay {
  fill: none;
  pointer-events: all;
}

.focus circle {
  fill: none;
  stroke: steelblue;
}

</style>
<body>
<script src="d3.min.js"></script>
<script>

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

var parseDate = d3.time.format("%Y-%M-%d %H:%M").parse,
    //dodane do sledzenia myszy i rysowania kuleczek
    bisectDate = d3.bisector(function(d) { return d.date; }).left,
    formatValue = d3.format(",.2f"),
    formatCurrency = function(d) { return "$" + formatValue(d); };

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

var y = d3.scale.linear()
    .range([height, 0]);

var color = d3.scale.category10();

var xAxis = d3.svg.axis()
    .scale(x)
    .orient("bottom");

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

var line = d3.svg.line()
    .interpolate("basis")
    .x(function(d) { return x(d.date); })
    .y(function(d) { return y(d.transfers); });

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("data2.csv", function(error, data) {
  color.domain(d3.keys(data[0]).filter(function(key) { return key !== "date"; }));

  data.forEach(function(d) {
    d.date = parseDate(d.date);
  });

  var bitrates = color.domain().map(function(name) {
    return {
      name: name,
      values: data.map(function(d) {
        return {date: d.date, transfers: +d[name]};
      })
    };
  });

  console.log(bitrates);

  //data.sort(function(a, b) {
    //return a.date - b.date;
  //});

  x.domain(d3.extent(data, function(d) { return d.date; }));
  y.domain([
    d3.min(bitrates, function(c) { return d3.min(c.values, function(v) { return v.transfers; }); }),
    d3.max(bitrates, function(c) { return d3.max(c.values, function(v) { return v.transfers; }); })
  ]);

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

  svg.append("g")
      .attr("class", "y axis")
      .call(yAxis)
    .append("text")
      .attr("transform", "rotate(-90)")
      .attr("y", 6)
      .attr("dy", ".71em")
      .style("text-anchor", "end")
      .text("Transfers");

  var chart = svg.selectAll(".chart")
      .data(bitrates)
    .enter().append("g")
      .attr("class", "chart");

  chart.append("path")
      .attr("class", "line")
      .attr("d", function(d) { return line(d.values); })
      //.attr("d", line);
      .style("stroke", function(d) { return color(d.name); });

  chart.append("text")
      .datum(function(d) { return {name: d.name, value: d.values[d.values.length - 1]}; })
      .attr("transform", function(d) { return "translate(" + x(d.value.date) + "," + y(d.value.transfers) + ")"; })
      .attr("x", 3)
      .attr("dy", ".35em");
      //.text(function(d) { return d.name; });

  //sledzenie myszy i rysowanie kuleczek
  var focus = svg.append("g")
      .attr("class", "focus")
      .style("display", "none");

  focus.append("circle")
      .attr("r", 4.5);

  focus.append("text")
      .attr("x", 9)
      .attr("dy", ".35em");

  svg.append("g").append("rect")
      .attr("class", "overlay")
      .attr("width", width)
      .attr("height", height)
      .on("mouseover", function() { focus.style("display", null); })
      .on("mouseout", function() { focus.style("display", "none"); })
      .on("mousemove", mousemove);

  function mousemove() {
    var x0 = x.invert(d3.mouse(this)[0]),
        i = bisectDate(data, x0, 1),
        d0 = data[i - 1],
        d1 = data[i],
        d = x0 - d0.date > d1.date - x0 ? d1 : d0;
    focus.attr("transform", "translate(" + x(d.date) + "," + y(d.value) + ")");
    focus.select("text").text(formatCurrency(d.value));
  }
});

</script>

Es sieht für mich so aus:Generiertes Diagramm Die CSV-Datendatei sieht folgendermaßen aus:

date,Średni wych.:,Średni wch.:,Maks. wych.:,Maks. wch.:
2014-02-14 15:40,86518717581,101989990772,88304882317,108036052338
2014-02-14 16:00,85739038102,98312113056,87060053514,107154908503

Einige über Informationen, die ich beim Versuch zu verstehen, was falsch ist, überprüft:

[Object, Object, Object, Object]
0: Object
name: "Średni wych.:"
values: Array[504]
__proto__: Object
1: Object
2: Object
name: "Maks. wych.:"
values: Array[504]
[0 … 99]
[100 … 199]
100: Object
date: Thu Jan 16 2014 01:00:00 GMT+0100 (Środkowoeuropejski czas stand.)
transfers: 49305177944
__proto__: Object
101: Object
date: Thu Jan 16 2014 01:20:00 GMT+0100 (Środkowoeuropejski czas stand.)
transfers: 42169641572
__proto__: Object
102: Object
date: Thu Jan 16 2014 01:40:00 GMT+0100 (Środkowoeuropejski czas stand.)
transfers: 39400112189
__proto__: Object
103: Object
104: Object
105: Object
106: Object
107: Object
108: Object
109: Object
110: Object

Ich würde mich über jede Hilfe von Ihnen sehr freuen. Ich kenne mich mit objektorientierter Programmierung, HTML und CSS aus, aber im Moment habe ich nicht wirklich mit einem Framework gearbeitet und es macht Spaß, es zu lernen, aber andererseits könnte es sehr frustrierend sein, herauszufinden, was zum Teufel ich tue falsch.

BEARBEITEN

Jetzt versuche ich zwei Linien getrennt zu zeichnen. Es funktioniert prima und es könnte mir später leichter fallen, den Linienstil zu ändern. Jetzt muss ich mousemove-Funktion für jede dieser Zeilen verwenden. Dann wäre es ziemlich einfach, gelesene Werte an einige Variablen zu übergeben und sie in einer Box oder so zu zeigen.

Dies ist der Code für meinen zweiten Versuch (Entschuldigung, dass der Beitrag lang wird):

Der Screenshot für den zweiten Code heißt Chart2.jpg

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

body {
  font: 10px sans-serif;
}

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

.x.axis path {
  display: none;
}

.line {
  fill: none;
  stroke: steelblue;
  stroke-width: 1.5px;
}

.overlay {
  fill: none;
  pointer-events: all;
}

.focus circle {
  fill: none;
  stroke: steelblue;
}

</style>
<body>
<script src="d3.js"></script>
<script>

var margin = {top: 20, right: 50, bottom: 30, left: 100},
    width = 960 - margin.left - margin.right,
    height = 500 - margin.top - margin.bottom;

var parseDate = d3.time.format("%d-%b-%y").parse,
    bisectDate = d3.bisector(function(d) { return d.date; }).left,
    formatValue = d3.format(",.2f"),
    formatCurrency = function(d) { return "$" + formatValue(d); };

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

var y = d3.scale.linear()
    .range([height, 0]);

var xAxis = d3.svg.axis()
    .scale(x)
    .orient("bottom");

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

var line = d3.svg.line()
    .x(function(d) { return x(d.date); })
    .y(function(d) { return y(d.close); });

var valueline2 = d3.svg.line()
    .x(function(d) { return x(d.date); })
    .y(function(d) { return y(d.open); });

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("data.csv", function(error, data) {
  data.forEach(function(d) {
    d.date = parseDate(d.date);
    d.close = +d.close;
    d.open = +d['open data'];
  });

  data.sort(function(a, b) {
    return a.date - b.date;
  });

  x.domain([data[0].date, data[data.length - 1].date]);
  y.domain([0, d3.max(data, function(d) { return Math.max(d.close, d.open); })]);

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

  svg.append("g")
      .attr("class", "y axis")
      .call(yAxis)
    .append("text")
      .attr("transform", "rotate(-90)")
      .attr("y", 6)
      .attr("dy", ".71em")
      .style("text-anchor", "end")
      .text("Price ($)");

  svg.append("path")
      .datum(data)
      .attr("class", "line")
      .attr("d", line);

  svg.append("path")
      .datum(data)
      .attr("class", "line")
      .style("stroke", "red")
      .attr("d", valueline2);

  var focus = svg.append("g")
      .attr("class", "focus")
      .style("display", "none");

  focus.append("circle")
      .attr("r", 4.5);

  focus.append("text")
      .attr("x", 9)
      .attr("dy", ".35em");

  svg.append("rect")
      .attr("class", "overlay")
      .attr("width", width)
      .attr("height", height)
      .on("mouseover", function() { focus.style("display", null); })
      .on("mouseout", function() { focus.style("display", "none"); })
      .on("mousemove", mousemove1)
      .on("mousemove", mousemove2);

  function mousemove1() {
    var x0 = x.invert(d3.mouse(this)[0]),
        i = bisectDate(data, x0, 1),
        d0 = data[i - 1],
        d1 = data[i],
        d = x0 - d0.date > d1.date - x0 ? d1 : d0;
    focus = focus.attr("transform", "translate(" + x(d.date) + "," + y(d.close) + ")");
    focus.select("text").text(formatCurrency(d.close));
    }

  function mousemove2() {
    var x0 = x.invert(d3.mouse(this)[0]),
        i = bisectDate(data, x0, 1),
        d0 = data[i - 1],
        d1 = data[i],
        d = x0 - d0.date > d1.date - x0 ? d1 : d0;
    focus = focus.attr("transform", "translate(" + x(d.date) + "," + y(d.open) + ")");
    focus.select("text").text(formatCurrency(d.open));
    }
});

</script>

Antworten auf die Frage(1)

Ihre Antwort auf die Frage