微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

D3JS + R - 无法显示数据

如何解决D3JS + R - 无法显示数据

目前,我正在开发一个 R 闪亮的应用程序,我想在其中使用 D3js 可视化数据。因为我是 D3 的新手,所以我完成了创建条形图的教程。不幸的是,虽然我能够为 x 和 y 绘制轴,但我无法使绘图工作以显示我的数据。这是我的代码(我只发送我的 .js 文件,因为我的数据甚至无法显示在我的 R Studio 的预览窗口中):

Histogram_Beispiel.js

// !preview r2d3 data=readr::read_tsv("Beispiel.tsv"),d3_version = "5",container = "div"
//r2d3: https://rstudio.github.io/r2d3,d3_version=3
// d3.tip
// Returns a tip
/* !preview r2d3 data=readr::read_tsv("Beispiel.tsv"),d3_version = "3",container = "div"*/
/* !preview r2d3 data=readr::read_tsv("Histogram_Data_Barplot.tsv"),container = "div" */
d3.tip = function() {
  var direction   = d3TipDirection,offset      = d3TipOffset,html        = d3TipHTML,rootElement = document.body,node        = initNode(),svg         = null,point       = null,target      = null;

  function tip(vis) {
    svg = getSVGNode(vis);
    if (!svg) return;
    point = svg.createSVGPoint();
    rootElement.appendChild(node);
  }

  // Public - show the tooltip on the screen
  //
  // Returns a tip
  tip.show = function() {
    var args = Array.prototype.slice.call(arguments);
    if (args[args.length - 1] instanceof SVGElement) target = args.pop();

    var content = html.apply(this,args),poffset = offset.apply(this,dir     = direction.apply(this,nodel   = getNodeEl(),i       = directions.length,coords,scrollTop  = document.documentElement.scrollTop ||
      rootElement.scrollTop,scrollLeft = document.documentElement.scrollLeft ||
      rootElement.scrollLeft;

    nodel.html(content)
      .style('opacity',1).style('pointer-events','all');

    while (i--) nodel.classed(directions[i],false);
    coords = directionCallbacks.get(dir).apply(this);
    nodel.classed(dir,true)
      .style('top',(coords.top + poffset[0]) + scrollTop + 'px')
      .style('left',(coords.left + poffset[1]) + scrollLeft + 'px');

    return tip;
  };

  // Public - hide the tooltip
  //
  // Returns a tip
  tip.hide = function() {
    var nodel = getNodeEl();
    nodel.style('opacity',0).style('pointer-events','none');
    return tip;
  };

  // Public: Proxy attr calls to the d3 tip container.
  // Sets or gets attribute value.
  //
  // n - name of the attribute
  // v - value of the attribute
  //
  // Returns tip or attribute value
  // eslint-disable-next-line no-unused-vars
  tip.attr = function(n,v) {
    if (arguments.length < 2 && typeof n === 'string') {
      return getNodeEl().attr(n);
    }

    var args =  Array.prototype.slice.call(arguments);
    d3.selection.prototype.attr.apply(getNodeEl(),args);
    return tip;
  };

  // Public: Proxy style calls to the d3 tip container.
  // Sets or gets a style value.
  //
  // n - name of the property
  // v - value of the property
  //
  // Returns tip or style property value
  // eslint-disable-next-line no-unused-vars
  tip.style = function(n,v) {
    if (arguments.length < 2 && typeof n === 'string') {
      return getNodeEl().style(n);
    }

    var args = Array.prototype.slice.call(arguments);
    selection.prototype.style.apply(getNodeEl(),args);
    return tip;
  };

  // Public: Set or get the direction of the tooltip
  //
  // v - One of n(north),s(south),e(east),or w(west),nw(northwest),//     sw(southwest),ne(northeast) or se(southeast)
  //
  // Returns tip or direction
  tip.direction = function(v) {
    if (!arguments.length) return direction;
    direction = v == null ? v : functor(v)

    return tip;
  };

  // Public: Sets or gets the offset of the tip
  //
  // v - Array of [x,y] offset
  //
  // Returns offset or
  tip.offset = function(v) {
    if (!arguments.length) return offset;
    offset = v == null ? v : functor(v)

    return tip;
  };

  // Public: sets or gets the html value of the tooltip
  //
  // v - String value of the tip
  //
  // Returns html value or tip
  tip.html = function(v) {
    if (!arguments.length) return html;
    html = v == null ? v : functor(v)

    return tip;
  };

  // Public: sets or gets the root element anchor of the tooltip
  //
  // v - root element of the tooltip
  //
  // Returns root node of tip
  tip.rootElement = function(v) {
    if (!arguments.length) return rootElement;
    rootElement = v == null ? v : functor(v)

    return tip;
  };

  // Public: destroys the tooltip and removes it from the DOM
  //
  // Returns a tip
  tip.destroy = function() {
    if (node) {
      getNodeEl().remove()
      node = null;
    }
    return tip;
  };

  function d3TipDirection() { return 'n' }
  function d3TipOffset() { return [0,0] }
  function d3TipHTML() { return ' ' }

  var directionCallbacks = d3.map({
        n:  directionnorth,s:  directionSouth,e:  directionEast,w:  directionWest,nw: directionnorthWest,ne: directionnorthEast,sw: directionSouthWest,se: directionSouthEast
      }),directions = directionCallbacks.keys();

  function directionnorth() {
    var bBox = getScreenBBox(this);
    return {
      top:  bBox.n.y - node.offsetHeight,left: bBox.n.x - node.offsetWidth / 2
    };
  }

  function directionSouth() {
    var bBox = getScreenBBox(this);
    return {
      top:  bBox.s.y,left: bBox.s.x - node.offsetWidth / 2
    };
  }

  function directionEast() {
    var bBox = getScreenBBox(this);
    return {
      top:  bBox.e.y - node.offsetHeight / 2,left: bBox.e.x
    };
  }

  function directionWest() {
    var bBox = getScreenBBox(this);
    return {
      top:  bBox.w.y - node.offsetHeight / 2,left: bBox.w.x - node.offsetWidth
    };
  }

  function directionnorthWest() {
    var bBox = getScreenBBox(this);
    return {
      top:  bBox.nw.y - node.offsetHeight,left: bBox.nw.x - node.offsetWidth
    }
  }

  function directionnorthEast() {
    var bBox = getScreenBBox(this)
    return {
      top:  bBox.ne.y - node.offsetHeight,left: bBox.ne.x
    }
  }

  function directionSouthWest() {
    var bBox = getScreenBBox(this)
    return {
      top:  bBox.sw.y,left: bBox.sw.x - node.offsetWidth
    }
  }

  function directionSouthEast() {
    var bBox = getScreenBBox(this)
    return {
      top:  bBox.se.y,left: bBox.se.x
    }
  }

  function initNode() {
    var div = d3.select(document.createElement('div'))
    div
      .style('position','absolute')
      .style('top',0)
      .style('opacity',0)
      .style('pointer-events','none')
      .style('Box-sizing','border-Box')

    return div.node()
  }

  function getSVGNode(element) {
    var svgNode = element.node()
    if (!svgNode) return null
    if (svgNode.tagName.toLowerCase() === 'svg') return svgNode
    return svgNode.ownerSVGElement
  }

  function getNodeEl() {
    if (node == null) {
      node = initNode()
      // re-add node to DOM
      rootElement.appendChild(node)
    }
    return d3.select(node)
  }

  // Private - gets the screen coordinates of a shape
  //
  // Given a shape on the screen,will return an SVGPoint for the directions
  // n(north),w(west),ne(northeast),se(southeast),// nw(northwest),sw(southwest).
  //
  //    +-+-+
  //    |   |
  //    +   +
  //    |   |
  //    +-+-+
  //
  // Returns an Object {n,s,e,w,nw,sw,ne,se}
  function getScreenBBox(targetShape) {
    var targetel   = target || targetShape

    while (targetel.getScreenCTM == null && targetel.parentNode != null) {
      targetel = targetel.parentNode
    }

    var bBox       = {},matrix     = targetel.getScreenCTM(),tbBox      = targetel.getBBox(),width      = tbBox.width,height     = tbBox.height,x          = tbBox.x,y          = tbBox.y

    point.x = x
    point.y = y
    bBox.nw = point.matrixTransform(matrix)
    point.x += width
    bBox.ne = point.matrixTransform(matrix)
    point.y += height
    bBox.se = point.matrixTransform(matrix)
    point.x -= width
    bBox.sw = point.matrixTransform(matrix)
    point.y -= height / 2
    bBox.w = point.matrixTransform(matrix)
    point.x += width
    bBox.e = point.matrixTransform(matrix)
    point.x -= width / 2
    point.y -= height / 2
    bBox.n = point.matrixTransform(matrix)
    point.y += height
    bBox.s = point.matrixTransform(matrix)

    return bBox
  }

  // Private - replace D3JS 3.X d3.functor() function
  function functor(v) {
    return typeof v === 'function' ? v : function() {
      return v
    }
  }

  return tip
}



//d3----------------------------------
    
var margin = {top: 40,right: 20,bottom: 30,left: 40},/// width and height will be generated by R automatically. Removed parts: Numbers after width = xxx and height = xxx.
    width = width - margin.left - margin.right,height = height - margin.top - margin.bottom;

var formatPercent = d3.format("");

var x = d3.scaleBand()
    .rangeRound([0,width])
    .padding(.1);

var y = d3.scaleLinear()
    .rangeRound([height,0]);

var xAxis = d3.axisBottom()
    .scale(x);
    
var yAxis = d3.axisLeft()
    .scale(y)
//    .tickFormat(formatPercent);

var tip = d3.tip()
  .attr('class','d3-tip')
  .offset([-10,0])
  .html(function(d) {
    return "<strong>SubCluster:</strong> <span style='color:red'>" + d.SubCluster + "</span>";
  })

//R already created a SVG element. So we do not need to create it again. Removing the following part:
//d3.select("body")
var svg = div.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 + ")");

svg.call(tip);

// To make the data input dynamically,we would do this: Create .tsv file (only in this example)
// Then change the header to read this specific .tsv file
// r2d3.onRender = To read the data into r2d3. 
r2d3.onRender(function(data,h,options) {
  x.domain(data.map(function(d) { return d.DimensionName; }));
  y.domain([0,d3.max(data,function(d) { return d.Amount;})]);
  //d3.count(data,d => d.amount);

  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("Occurence");

  svg.selectAll(".bar")
      .data(data)
    .enter().append("rect")
      .attr("class","bar")
      .attr("x",function(d) { return x(d.DimensionName); })
      .attr("width",x.range())
      .attr("y",function(d) { return y(d.Amount); })
      .attr("height",function(d) { return height - y(d.Amount); })
      .on('mouSEOver',tip.show)
      .on('mouSEOut',tip.hide)

});

function type(d) {
  d.Amount = +d.Amount;
  return d;
}

Beispiel.tsv

DimensionName   SubCluster  Amount
f1  1,7,22    3
f10 15  1
f11 3,8,44    3
f12 36,42    2
f13 4,10,11,19,33,37,38   7
f14 29  1
f15 18  1
f16 9   1
f17 2   1
f18 13,14,35,40,41,44    6
f19 31  1
f2  34  1
f20 21  1
f3  19,30    2
f4  10,13,16,17,20,22,23,25,26,27,28  13
f5  4,12 2
f6  5,6,28,38,43,45 13
f7  3,36,39,45   7
f8  32  1
f9  24  1

有人可以帮助我或给我建议吗?如何使条形图正常工作?

编辑:共享 R. 文件(我只包括调用 R2D3 对象的方式,因为文件很长):

        if (interactive()) {
          ui <- [Code about navigation bar etc.]
       body = argonDashBody(
          argonTabItems(
            
            argonTabItem(
              tabName = "subcluster_analysis",fluidRow(
              column(width = 8,argonTable(
                       cardWrap = FALSE,#title = "Barplots",headTitles = "Barplots",#                 width = 4,shinycssloaders::withSpinner(
                         d3Output(outputId = "out_barplotD3"))
                     )
                     ),[...rest of the body...]
  #### Server initiation ---------------------------------
  server <- function(input,output,session){
      [Server output]

            #Barplot in D3.Js
        output$out_barplotD3 <- renderD3({
          r2d3(data = data,script= "Histogram_Beispiel.js",container = "div")
          
        })

    
  }  
  shinyApp(ui,server)  
}

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。