使用用户定义的颜色修改networkD3桑基图

13
我有一个使用networkD3包创建的桑基图,我想修改节点和链接的颜色和透明度。
我的数据networkD3_data附在末尾。
问题1:如何使用用户定义的调色板修改节点颜色?
我不确定如何使用用户定义的调色板修改颜色。为了保持与其他图形一致,我需要使用每个节点源特定的相同调色板。
目前,我能够通过定义NodeGroup使每个源节点成为不同的颜色,并使所有目标节点成为相同的颜色。下面的图已经接近我想要的效果,但现在我想自己指定每个源节点的颜色。
library(networkD3)
sankeyNetwork(Links = networkD3_data$links, Nodes = networkD3_data$nodes, Source = "source", Target = "target", Value = "value", NodeID = "name", NodeGroup="group", fontSize=14)

在此输入图片描述

?sankeyNetwork 中提到了一个 colourScale 参数,它接受“指定节点分类颜色比例尺的字符字符串”。我理解为我可以在 networkD3_data$nodes 数据框中添加一个包含所需颜色的列,并像这样调用 colourScale

sankeyNetwork(Links = networkD3_data$links, Nodes = networkD3_data$nodes, Source = "source", Target = "target", Value = "value", NodeID = "name", NodeGroup="group", fontSize=14, colourScale="colors")

但是这并不起作用,没有生成任何图形。我在stackoverflow上搜索到了一个来自2014年的答案:这里,但我认为这不再起作用,因为我调用这些提出的解决方案之一时会出现错误Error: unexpected symbol(或者我可能只是不知道如何正确实现它)。

d3.scale.ordinal().range(["#7d3945","#e0677b", "#244457"])
d3.scaleOrdinal().range(["#7d3945","#e0677b", "#244457"])  
问题2:是否可以定义源和目标的显示顺序? 所以源从Source0到Source10列出,目标从Target11到Target47列出?这个设置在哪里?我知道这与允许算法最优位置放置节点相矛盾。 问题3:如何使用用户定义的调色板修改链接颜色并改变透明度? 我还想使用与源相同的颜色方案对从源引出的链接进行着色。我可以通过定义LinkGroup来实现这一点,请参见下面的图。我离我想要的东西很近了,但我只需要自己指定使用的颜色,我不确定在哪里修改这个设置。我还想调整不透明度,以便源颜色比链接颜色更加饱满。
sankeyNetwork(Links = networkD3_data$links, Nodes = networkD3_data$nodes, Source = "source", Target = "target", Value = "value", NodeID = "name", NodeGroup="group", LinkGroup="group", fontSize=14)

这里是我的数据 -- dput(networkD3_data)

输入图像描述

structure(list(nodes = structure(list(name = c("Source0", "Source1", 
"Source2", "Source3", "Source4", "Source5", "Source6", "Source7", 
"Source8", "Source9", "Source10", "Target11", "Target12", "Target13", 
"Target14", "Target15", "Target16", "Target17", "Target18", "Target19", 
"Target20", "Target21", "Target22", "Target23", "Target24", "Target25", 
"Target26", "Target27", "Target28", "Target29", "Target30", "Target31", 
"Target32", "Target33", "Target34", "Target35", "Target36", "Target37", 
"Target38", "Target39", "Target40", "Target41", "Target42", "Target43", 
"Target44", "Target45", "Target46", "Target47"), group = c("Source0", 
"Source1", "Source2", "Source3", "Source4", "Source5", "Source6", 
"Source7", "Source8", "Source9", "Source10", "Target", "Target", 
"Target", "Target", "Target", "Target", "Target", "Target", "Target", 
"Target", "Target", "Target", "Target", "Target", "Target", "Target", 
"Target", "Target", "Target", "Target", "Target", "Target", "Target", 
"Target", "Target", "Target", "Target", "Target", "Target", "Target", 
"Target", "Target", "Target", "Target", "Target", "Target", "Target"
), colors = c("#9E0142", "#D53E4F", "#F46D43", "#FDAE61", "#FEE08B", 
"#FFFFBF", "#E6F598", "#ABDDA4", "#66C2A5", "#3288BD", "#5E4FA2", 
"#969696", "#969696", "#969696", "#969696", "#969696", "#969696", 
"#969696", "#969696", "#969696", "#969696", "#969696", "#969696", 
"#969696", "#969696", "#969696", "#969696", "#969696", "#969696", 
"#969696", "#969696", "#969696", "#969696", "#969696", "#969696", 
"#969696", "#969696", "#969696", "#969696", "#969696", "#969696", 
"#969696", "#969696", "#969696", "#969696", "#969696", "#969696", 
"#969696")), .Names = c("name", "group", "colors"), row.names = c(NA, 
-48L), class = "data.frame"), links = structure(list(source = c(0, 
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 
1, 1, 1, 1, 1, 1, 1, 1, 1, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 
2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 
3, 3, 3, 4, 4, 4, 4, 4, 4, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 6, 
6, 6, 6, 7, 7, 7, 7, 7, 7, 7, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 
8, 9, 9, 9, 9, 9, 9, 9, 9, 10, 10, 10, 10, 10, 10, 10, 10, 10
), target = c(11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 
23, 13, 18, 15, 11, 12, 24, 21, 25, 26, 27, 19, 28, 16, 22, 29, 
30, 31, 32, 18, 16, 15, 13, 27, 29, 19, 33, 34, 31, 35, 21, 24, 
11, 30, 36, 28, 37, 38, 39, 40, 26, 41, 11, 12, 15, 18, 19, 14, 
13, 16, 27, 34, 20, 22, 25, 12, 27, 16, 18, 13, 11, 12, 11, 14, 
27, 21, 16, 18, 22, 13, 15, 19, 16, 11, 12, 39, 12, 14, 18, 11, 
42, 43, 44, 13, 11, 18, 15, 12, 19, 45, 31, 16, 20, 46, 40, 47, 
11, 12, 18, 16, 14, 19, 15, 11, 12, 16, 13, 18, 14, 34, 31, 15
), value = c(5.8, 3.2, 5, 2.4, 2.5, 2.7, 3.5, 2.5, 3.5, 1.4, 
2.9, 2.4, 1.3, 12.1, 7.4, 5, 11.2, 5.6, 6.4, 8.8, 2.6, 3.5, 7, 
10, 4.5, 6, 6.5, 5.8, 5.4, 6.2, 8.9, 5.5, 4.8, 3.4, 6.5, 5, 4, 
6.4, 7.3, 4.4, 4.2, 1.7, 5.1, 3.6, 6.4, 3.4, 2.5, 2.6, 2.3, 2.3, 
3.2, 1.6, 1.7, 3.7, 8, 4.4, 3.1, 4.1, 5.9, 2.8, 5, 3.2, 3.7, 
3.4, 1.8, 3.2, 1.2, 4.1, 5.2, 4.5, 4.8, 7.1, 7.3, 4.6, 8.4, 3.4, 
5.2, 6.1, 4.3, 4.5, 4.5, 6.5, 2.8, 6.3, 5.3, 8.2, 3.8, 4.3, 4.2, 
3.4, 5.4, 7.9, 1.2, 1.4, 1.4, 6.6, 6.8, 4.2, 2.9, 3.1, 5.3, 2.6, 
3.2, 2.9, 1.7, 1.9, 1.4, 8, 8, 4, 5, 4.3, 2.9, 6.9, 3, 8.7, 4.5, 
4.2, 6.6, 4.4, 2.7, 4.4, 4.3, 2.8), group = c("Source0", "Source0", 
"Source0", "Source0", "Source0", "Source0", "Source0", "Source0", 
"Source0", "Source0", "Source0", "Source0", "Source0", "Source1", 
"Source1", "Source1", "Source1", "Source1", "Source1", "Source1", 
"Source1", "Source1", "Source1", "Source1", "Source1", "Source1", 
"Source1", "Source1", "Source1", "Source1", "Source1", "Source2", 
"Source2", "Source2", "Source2", "Source2", "Source2", "Source2", 
"Source2", "Source2", "Source2", "Source2", "Source2", "Source2", 
"Source2", "Source2", "Source2", "Source2", "Source2", "Source2", 
"Source2", "Source2", "Source2", "Source2", "Source3", "Source3", 
"Source3", "Source3", "Source3", "Source3", "Source3", "Source3", 
"Source3", "Source3", "Source3", "Source3", "Source3", "Source4", 
"Source4", "Source4", "Source4", "Source4", "Source4", "Source5", 
"Source5", "Source5", "Source5", "Source5", "Source5", "Source5", 
"Source5", "Source5", "Source5", "Source5", "Source6", "Source6", 
"Source6", "Source6", "Source7", "Source7", "Source7", "Source7", 
"Source7", "Source7", "Source7", "Source8", "Source8", "Source8", 
"Source8", "Source8", "Source8", "Source8", "Source8", "Source8", 
"Source8", "Source8", "Source8", "Source9", "Source9", "Source9", 
"Source9", "Source9", "Source9", "Source9", "Source9", "Source10", 
"Source10", "Source10", "Source10", "Source10", "Source10", "Source10", 
"Source10", "Source10")), .Names = c("source", "target", "value", 
"group"), row.names = c(NA, -124L), class = "data.frame")), .Names = c("nodes", 
"links"))
1个回答

13

colourScale参数确定了颜色调色板。它应该是一个字符串,包含有效的D3代码来定义调色板。在当前版本的networkD3中,使用D3v4+,语法d3.scaleOrdinal().range(["#7d3945", "#e0677b", "#244457"])是有效的,但它也可以缩短为d3.scaleOrdinal(["#7d3945", "#e0677b", "#244457"])。还有其他可能性,请参考这里。您选择/定义调色板中的颜色数量必须等于或大于数据中定义的唯一组的数量,否则它将循环回到您的调色板的开头,以分配颜色给更多的组。您数据中定义的第一组将被分配到您的颜色调色板中的第一种颜色,依此类推。

NodeGroupLinkGroup参数分别定义NodesLinks数据框中列的名称,该列为每个节点/链接定义组值。根据使用的颜色调色板,节点和链接都将根据其组值和分配的颜色进行着色。

iteration参数设置为0将有效地防止放置算法运行,因此您的节点将按照原始数据中的顺序排序。正如您指出的,这基本上击败了sankeyNetwork的主要目的。

使用您发布的数据...

library(networkD3)

colors <- paste(networkD3_data$nodes$colors, collapse = '", "')
colorJS <- paste('d3.scaleOrdinal(["', colors, '"])')

sankeyNetwork(Links = networkD3_data$links, Nodes = networkD3_data$nodes, 
              Source = 'source', Target = 'target', Value = 'value', 
              NodeID = 'name', NodeGroup = "group", LinkGroup = "group",
              colourScale = colorJS,
              iterations = 0)

透明度/不透明度可以通过在D3颜色调色板中设置RGBA颜色来实现,但需要使用十进制RGB表示法(据我所知)。您将需要将类似#ff0043的内容转换为类似d3.rgb(255,0,67,0.5)的内容,其中最后一个数字是介于0和1之间的数字,用于定义不透明度水平。例如...

colors <- paste(sapply(networkD3_data$nodes$colors, function(x) { paste0("d3.rgb(", paste(c(col2rgb(x), 0.5), collapse = "," ), ")") }), collapse = ", ")
colorJS <- paste0('d3.scaleOrdinal([', colors, '])')
sankeyNetwork(Links = networkD3_data$links, Nodes = networkD3_data$nodes,
              Source = 'source', Target = 'target', Value = 'value',
              NodeID = 'name', NodeGroup = "group", LinkGroup = "group",
              colourScale = colorJS,
              iterations = 0)

1
第二个问题是,有没有一种方法可以保持源节点Source0-Source10的顺序,但让算法最优地定位目标节点? - Djork
1
透明度已经硬编码到JS中,所以不,不容易。您可以使用htmlwidget函数在加载后注入其他JS,这些JS可以选择所有链接并在最初生成后设置它们的透明度。 - CJ Yetman
不,如果不对基础JS进行重大更改,您无法有选择地选择节点应用算法。 - CJ Yetman
1
节点和链接都从相同的分配颜色调色板中提取其颜色,但这并不妨碍您为每个节点和链接设置特定的唯一颜色。只需为每个节点和链接设置唯一的组值,并设置/选择具有足够独特颜色的颜色调色板即可。 - CJ Yetman
1
我更新了我的答案,并提供了一种定义颜色调色板中每种颜色不透明度的解决方案。 - CJ Yetman
显示剩余2条评论

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接