(d3.js 强制定向图)无法创建 属性 'vx',但这仅在本地主机中发生

(d3.js force directed graph) Cannot create property 'vx', but this happens in localhost only

我正在整理文件夹结构(将 JS 移动到 .js 文件而不是内联,并将其放入其自己的文件夹中)突然发生这种情况。我的力导向图突然出现了这个奇怪的问题。所有节点和链接都移到了屏幕的左上角。它看起来像这样:

Javascript 控制台显示 Cannot create property 'vx' on string 'dki'

奇怪的是,当我尝试将整个代码粘贴到 jsfiddle 中时,它工作正常。您可以在此处找到 fiddle:https://jsfiddle.net/k6pf1hfw/1/

这里是JS代码:

var nodes = [
  {  id: "pusat",  group: 0,  label: "Pusat",  level: 0 },

  {  id: "dki",  group: 1,  label: "Prov. DKI",  level: 1 },
      {  id: "jaksel",  group: 1,  label: "Kota Jakarta Selatan",  level: 2 },
      {  id: "jakpus",  group: 1,  label: "Kota Jakarta Pusat",  level: 2},
  {  id: "jabar",  group: 2,  label: "Prov. Jawa Barat",  level: 1 },
      {  id: "sumedang",  group: 2,  label: "Kab. Sumedang",  level: 2 },
      {  id: "bekasi",  group: 2,  label: "Kota Bekasi",  level: 2 },
      {  id: "bandung",  group: 2,  label: "Kota Bandung",  level: 2 },
  {  id: "jatim",  group: 3,  label: "Prov. Jawa Timur",  level: 1 },
      {  id: "malang",  group: 3,  label: "Kota Malang",  level: 2 },
      {  id: "lamongan",  group: 3,  label: "Kota Lamongan",  level: 2 },
  {  id: "diy",  group: 4,  label: "Prov. DIY",  level: 1 },
      {  id: "sleman",  group: 4,  label: "Kab. Sleman",  level: 2 },
      {  id: "jogja",  group: 4,  label: "Kota Yogyakarta",  level: 2 },
  {  id: "bali",  group: 5,  label: "Prov. Bali",  level: 1 },
      {  id: "bali1",  group: 5,  label: "Kota Denpasar",  level: 2 },
      {  id: "bali2",  group: 5,  label: "Kab. Buleleng",  level: 2 },
  {  id: "ntt",  group: 6,  label: "Prov. NTT",  level: 1 },
      {  id: "ntt1",  group: 6,  label: "Kab. Alor",  level: 2 },
      {  id: "ntt2",  group: 6,  label: "Kab. Manggarai Timur",  level: 2},
  {  id: "ntb",  group: 7,  label: "Prov. NTB",  level: 1 },
      {  id: "kabima",  group: 7,  label: "Kab. Bima",  level: 2 },
      {  id: "kobima",  group: 7,  label: "Kota Bima",  level: 2 },
  {  id: "kaltara",  group: 8,  label: "Prov. Kaltara",  level: 1 },
      {  id: "kubar",  group: 8,  label: "Kab. Kutai Barat",  level: 2 },
      {  id: "kutim",  group: 8,  label: "Kab. Kutai Timur",  level: 2 },
  {  id: "kaltim",  group: 9,  label: "Prov. Kaltim",  level: 1 },
      {  id: "bpp",  group: 9,  label: "Kota Balikpapan",  level: 2 },
      {  id: "samarinda",  group: 9,  label: "Kota Samarinda",  level: 2 },
  {  id: "kalsel",  group: 10,  label: "Prov. Kalsel",  level: 1 },
      {  id: "banjar",  group: 10,  label: "Kota Banjarmasin",  level: 2 },
      {  id: "tapin",  group: 10,  label: "Kab. Tapin",  level: 2 },
  {  id: "kalbar",  group: 11,  label: "Prov. Kalbar",  level: 1 },
      {  id: "melawi",  group: 11,  label: "Kab. Melawi",  level: 2 },
      {  id: "sambas",  group: 11,  label: "Kab. Sambas",  level: 2}
]

var links = [
  // Pusat-Provinsi
  { source:"pusat", target:"dki", strength:.5, value:100000000000  },
  { source:"pusat", target:"jabar", strength:.5, value:30000000000},
  { source:"pusat", target:"jatim", strength:.5, value:100000000000},
  { source:"pusat", target:"diy", strength:.5, value:1000000000000},
  { source:"pusat", target:"bali", strength:.5, value:10000000000},
  { source:"pusat", target:"ntt", strength:.5, value:1000000000},
  { source:"pusat", target:"ntb", strength:.5, value:1000000000},
  { source:"pusat", target:"kaltim", strength:.5, value:5000000000000},
  { source:"pusat", target:"kaltara", strength:.5, value:5000000000000},
  { source:"pusat", target:"kalsel", strength:.5, value:10000000000000},
  { source:"pusat", target:"kalbar", strength:.5, value:1000000000},

  // Provinsi-Kab/Kota
  { source:"dki", target:"jaksel", strength:.7, value:2000000000},
  { source:"dki", target:"jakpus", strength:.7, value:4000000000000},
  { source:"jabar", target:"sumedang", strength:.7, value:400000000000},
  { source:"jabar", target:"bekasi", strength:.7, value:40000000000},
  { source:"jabar", target:"bandung", strength:.7, value:40000000000},
  { source:"jatim", target:"malang", strength:.7, value:300000000000},
  { source:"jatim", target:"lamongan", strength:.7, value:100000000000},
  { source:"diy", target:"sleman", strength:.7, value:4500000000000},
  { source:"diy", target:"jogja", strength:.7, value:6700000000000},
  { source:"bali", target:"bali1", strength:.7, value:100000000000000},
  { source:"bali", target:"bali2", strength:.7, value:2400000000000},
  { source:"ntt", target:"ntt1", strength:.7, value:60000000000000},
  { source:"ntt", target:"ntt2", strength:.7, value:100000000000},
  { source:"ntb", target:"kabima", strength:.7, value:126000000000},
  { source:"ntb", target:"kobima", strength:.7, value:1000000000000},
  { source:"kaltara", target:"kubar", strength:.7, value:12420000000000},
  { source:"kaltara", target:"kutim", strength:.7, value:14400000000000},
  { source:"kaltim", target:"bpp", strength:.7, value:1470000000000},
  { source:"kaltim", target:"samarinda", strength:.7, value:1000000000000000},
  { source:"kalsel", target:"banjar", strength:.7, value:137000000000},
  { source:"kalsel", target:"tapin", strength:.7, value:5050000000000},
  { source:"kalbar", target:"melawi", strength:.7, value:2400000000000},
  { source:"kalbar", target:"sambas", strength:.7, value:5500000000000}
];

function getNeighbors(node) {
  return links.reduce(function(neighbors, link) {
    if (link.target.id === node.id) {
      neighbors.push(link.source.id)
    } else if (link.source.id === node.id) {
      neighbors.push(link.target.id)
    }
    return neighbors
  }, [node.id])
}

function isNeighborLink(node, link) {
  return link.target.id === node.id || link.source.id === node.id
}


function getNodeColor(node, neighbors) {
  // If is neighbor
  if (Array.isArray(neighbors) && neighbors.indexOf(node.id) > -1) {
    return 'rgba(251, 130, 30, 1)'
    // return node.level === 1 ? '#9C4A9C' : 'rgba(251, 130, 30, 1)'
  } else {
    // Check the node level
      if (node.level === 0) {
        return '#E72148'
      } else if (node.level === 1) {
        return '#9C4A9C'
      } else {
        return '#D8ABD8'
      }
  }
  //return node.level === 0 ? '#91007B' : '#D8ABD8'
}

function getLinkColor(node, link) {
  return isNeighborLink(node, link) ? 'rgba(251, 130, 30, 1)' : 'rgba(251, 130, 30, 0.25)'
}

function getTextColor(node, neighbors) {
  return Array.isArray(neighbors) && neighbors.indexOf(node.id) > -1 ? '#333' : '#bbb'
}

var width = window.innerWidth
var height = window.innerHeight

var svg = d3.select('svg')
// svg.attr('width', width).attr('height', height)
    svg.attr("width", '100%')
    .attr("height", '500px')
    .attr('viewBox', '250 0 800 600')
    //.attr('viewBox','0 0 '+Math.min(width,height)+' '+Math.min(width,height))
    .attr('preserveAspectRatio','xMidYMid')
    .append("g")
    .attr("transform", "translate(" + Math.min(width,height) / 2 + "," + Math.min(width,height) / 2 + ")");

//add zoom capabilities
  var zoom_handler = d3.zoom()
      .scaleExtent([1 / 2, 8])
      .on("zoom", zoom_actions);

  zoom_handler(svg);

  function zoom_actions(){
      g.attr("transform", d3.event.transform)
  }

  function button_zoom_in(){
      zoom_handler.scaleBy(svg, 2);
  }
  function button_zoom_out(){
      zoom_handler.scaleBy(svg, 0.5);
  }

// simulation setup with all forces
var linkForce = d3
  .forceLink()
  .id(function (link) { return link.id })
  // Alternative: using the distance from the data "strength"
  //.distance(50).strength(function (link) { return link.strength })
  // If don't want to use this, use default here:
  .distance(50).strength(.7)

var simulation = d3
  .forceSimulation()
  .force('link', linkForce)
  .force('charge', d3.forceManyBody().strength(-1500))
  .force('radial', d3.forceRadial(function(d) {
    return d.level * 50
  }, width / 2, height / 2))
  .force('center', d3.forceCenter(width / 2, height / 2))

var dragDrop = d3.drag().on('start', function(node) {
  node.fx = node.x
  node.fy = node.y
}).on('drag', function(node) {
  simulation.alphaTarget(0.7).restart()
  node.fx = d3.event.x
  node.fy = d3.event.y
}).on('end', function(node) {
  if (!d3.event.active) {
    simulation.alphaTarget(0)
  }
  node.fx = null
  node.fy = null
})

function selectNode(selectedNode) {
  var neighbors = getNeighbors(selectedNode)

  // we modify the styles to highlight selected nodes
  nodeElements.attr('fill', function(node) {
    return getNodeColor(node, neighbors)
  })
  textElements.attr('fill', function(node) {
    return getTextColor(node, neighbors)
  })
  linkElements.attr('stroke', function(link) {
    return getLinkColor(selectedNode, link)
  })
}



// Enables zooming
var g = svg.append("g")
    .attr("class", "everything");
// Enables zooming end

// Create circling orbit
var circles = g.selectAll(null)   // use g.selectAll instead of svg.selectAll to enable zoom
  .data([200,350])  // sets the circle radius
  .enter()
  .append("circle")
  .attr("cx", width/2)
  .attr("cy", height/2)
  .attr("r", d=>d)
  .style("fill", "none")
  .style("stroke", "#ddd");

var linkElements = g.append("g")    // use g.append instead of svg.append to enable zoom
  .attr("class", "links")
  .selectAll("line")
  .data(links)
  .enter().append("line")
  .attr("id",function(d,i) { return "linkId_" + i; })
  .attr("stroke-width", function(link) {
    var linkWidthNormalize = link.value / 1000000000;       // in milyar
                                                            // under assumption that smallest 10 milyar, largest > 40 triliun
    if (linkWidthNormalize >= 40001) {
      return 12;
    } else if (linkWidthNormalize >= 20001 && linkWidthNormalize <= 40000) {
      return 10;
    } else if (linkWidthNormalize >= 9001 && linkWidthNormalize <= 20000) {
      return 8;
    } else if (linkWidthNormalize >= 4001 && linkWidthNormalize <= 9000) {
      return 6;
    } else if (linkWidthNormalize >= 10 && linkWidthNormalize <= 4000) {
      return 4;
    } else {
      return 2;
    }
    // return linkWidthNormalize;
  })
  .attr("stroke", "rgba(251, 130, 30, 0.5)")

var nodeElements = g.append("g")    // use g.append instead of svg.append to enable zoom
  .attr("class", "nodes")
  .selectAll("circle")
  .data(nodes)
  .enter().append("circle")
  .attr("r", 12)
  .attr("fill", getNodeColor)
  .attr("stroke", "#fff")
  .attr('stroke-width', 2)
  .call(dragDrop)
  //.on('click', selectNode)      // alternative
  .on('mouseover', selectNode)

var textElements = g.append("g")    // use g.append instead of svg.append to enable zoom
  .attr("class", "texts")
  .selectAll("text")
  .data(nodes)
  .enter().append("text")
  .text(function(node) {
    return node.label
  })
  .attr("font-size", 10)
  .attr("font-family", "sans-serif")
  .attr("text-anchor", "middle")
  .attr("fill", "#333")
  .attr("style", "font-weight:bold; -webkit-text-stroke: 1px #fff; text-shadow: 3px 3px 0 #fff, -1px -1px 0 #fff, 1px -1px 0 #fff, -1px 1px 0 #fff, 1px 1px 0 #fff")
  .attr("dx", 0)
  .attr("dy", 20)



simulation.nodes(nodes).on('tick', () => {
  nodeElements
    .attr('cx', function(node) {
      return node.x
    })
    .attr('cy', function(node) {
      return node.y
    })
  textElements
    .attr('x', function(node) {
      return node.x
    })
    .attr('y', function(node) {
      return node.y
    })
  linkElements
    .attr('x1', function(link) {
      return link.source.x
    })
    .attr('y1', function(link) {
      return link.source.y
    })
    .attr('x2', function(link) {
      return link.target.x
    })
    .attr('y2', function(link) {
      return link.target.y
    })
})

simulation.force("link").links(links)

我找不到代码有什么问题。我曾尝试恢复我的操作(将文件移回其初始位置)但没有任何变化。还是那样。

这有什么问题吗?这让我发疯。

终于找到罪魁祸首了

希望这对任何像我一样粗心的人发出警告:d3.js cannot read iso-8859-1 编码。 我的 HTML 文件使用 iso-8859-1 编码,同时 JSFiddle 使用 utf-8。这就是为什么它在 JSFiddle 中有效但在我这边不起作用的原因。

所以改变这个:

<meta http-equiv="Content-Type" content="text/html; charset=iso-8859-1">

对此:

<meta http-equiv="Content-Type" content="text/html; charset=utf-8">

解决了问题。