Skip to content

Commit

Permalink
Merge branch 'main' of github.com:uclahs-cds/package-CancerEvolutionV…
Browse files Browse the repository at this point in the history
…isualization into danknight-angle-calculation
  • Loading branch information
dan-knight committed Nov 9, 2023
2 parents 62f7e9e + e9fd02b commit 7c9d947
Show file tree
Hide file tree
Showing 16 changed files with 308 additions and 30 deletions.
2 changes: 1 addition & 1 deletion DESCRIPTION
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
Package: CancerEvolutionVisualization
Title: Publication Quality Phylogenetic Tree Plots
Version: 2.0.0
Date: 2023-09-29
Date: 2023-11-09
Authors@R: c(
person("Paul Boutros", role = "cre", email = "[email protected]"),
person("Adriana Salcedo", role = "aut"),
Expand Down
10 changes: 9 additions & 1 deletion NEWS
Original file line number Diff line number Diff line change
@@ -1,14 +1,22 @@
CancerEvolutionVisualization 2.0.0 2023-09-29 (Dan Knight, Helena Winata)
CancerEvolutionVisualization 2.0.0 2023-11-09

ADDED
* Support for specifying tree angles in either radians or degrees using
an optional "angle" column
* Generic functions to generate accompanying heatmaps
* Option to specify tree node colours with "node.col" column
* Option to specify tree node border colour, width, and line-type with
"border.col", "border.width", and "border.type" columns
* Option ot specify tree node label colour with "node.label.col" column

UPDATE
* Reimplemented tree angle calculations
* Fixed lopsided radial tree bug

REMOVED
* "node.col" parameter to SRCGrob. (Node colour only customizable through
tree input data.frame.)

--------------------------------------------------------------------------
CancerEvolutionVisualization 1.0.1 2022-10-03 (Dan Knight)

Expand Down
11 changes: 10 additions & 1 deletion R/SRCGrob.R
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,6 @@ SRCGrob <- function(
main.y = NULL,
main.cex = 1.7,
node.radius = 0.1,
node.col = 'grey29',
seg1.col = 'black',
seg2.col = 'green',
line.lwd = 3,
Expand All @@ -43,11 +42,17 @@ SRCGrob <- function(
yat <- prep.yat(yat);
yaxis.position <- get.y.axis.position(colnames(tree));

node.col <- 'grey40';

inputs <- prep.tree(
tree,
node.text,
colour.scheme = colour.scheme,
<<<<<<< HEAD
use.radians = use.radians
=======
default.node.colour = node.col
>>>>>>> e9fd02b08dad613689f57f3c74b7c0ee522d25be
);

fixed.angle <- pi / 6;
Expand Down Expand Up @@ -121,5 +126,9 @@ SRCGrob <- function(
cl = 'SRCGrob'
);

out.tree$input.data <- list(
tree = inputs$in.tree.df[-1, ], # Remove Normal node placeholder row
text = inputs$text.df
);
return(out.tree);
}
9 changes: 7 additions & 2 deletions R/add.nodes.R
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,12 @@ add.node.ellipse <- function(
y = unit(clone.out$v$y, 'native'),
size = node.radius * (1 + 0.2 * nchar(clone.out$v$plot.lab)),
ar = 1 - log2(nchar(clone.out$v$plot.lab)) / 10,
gp = gpar(fill = clone.out$v$colour, col = clone.out$v$colour),
gp = gpar(
fill = clone.out$v$node.colour,
col = clone.out$v$border.colour,
lty = clone.out$v$border.type,
lwd = clone.out$v$border.width
),
angle = pi / 2,
position.units = 'native',
size.units = 'inches',
Expand All @@ -57,7 +62,7 @@ add.node.ellipse <- function(
x = unit(clone.out$v$x, 'native'),
y = unit(clone.out$v$y, 'native'),
just = c('center', 'center'),
gp = gpar(col = '#FFFFFF', cex = label.cex - log2(nchar(clone.out$v$plot.lab)) / 10)
gp = gpar(col = clone.out$v$node.label.colour, cex = label.cex - log2(nchar(clone.out$v$plot.lab)) / 10)
);

clone.out$grobs <- c(clone.out$grobs, list(node.label.grob));
Expand Down
34 changes: 34 additions & 0 deletions R/get.colours.R
Original file line number Diff line number Diff line change
Expand Up @@ -16,3 +16,37 @@ get.colours <- function(
return(col.list[value.list]);
}
}

get.colour.luminance <- function(colour) {
# Formulas and values documented in:
# https://www.w3.org/WAI/GL/wiki/Relative_luminance
sRGB.values <- col2rgb(colour) / 255;
sRGB.values <- sapply(
sRGB.values,
FUN = function(sRGB.value) {
if (sRGB.value <= 0.03928) {
return(sRGB.value / 12.92);
} else {
return(((sRGB.value + 0.055 ) / 1.055) ** 2.4);
}
}
);

luminance.modifiers <- c(0.2126, 0.7152, 0.0722);
luminance <- sum(sRGB.values * luminance.modifiers);

return(luminance);
}

get.contrast.ratio <- function(luminance1, luminance2) {
# Based on WCAG accessibility standards:
# https://www.w3.org/TR/2008/REC-WCAG20-20081211/#visual-audio-contrast-contrast
luminance <- sort(
c(luminance1, luminance2),
decreasing = TRUE
);
luminance <- luminance + 0.05;

contrast.ratio <- luminance[1] / luminance[2];
return(contrast.ratio);
}
98 changes: 97 additions & 1 deletion R/prep.tree.R
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,8 @@ prep.tree <- function(
text.df,
bells = TRUE,
colour.scheme,
use.radians = FALSE
use.radians = FALSE,
default.node.colour = 'grey29'
) {

if (!('parent' %in% colnames(tree.df))) {
Expand Down Expand Up @@ -71,12 +72,65 @@ prep.tree <- function(
if (is.null(tree.df$label)) tree.df$child else tree.df$label
);

if (('node.col' %in% colnames(tree.df))) {
tree.df$node.col[is.na(tree.df$node.col)] <- default.node.colour;
} else {
tree.df$node.col <- default.node.colour;
}

tree.df$node.label.col <- prep.node.label.colours(tree.df);

tree.df$border.col <- apply(
tree.df,
MARGIN = 1,
FUN = function(row) {
if (is.na(row['border.col'])) row['node.col'] else row['border.col'];
}
);

if ('border.type' %in% colnames(tree.df)) {
valid.border.types <- c(
'blank',
'solid',
'dashed',
'dotted',
'dotdash',
'longdash',
'twodash'
);

border.type.is.valid <- tree.df$border.type %in% valid.border.types | is.na(tree.df$border.type);

if (!all(border.type.is.valid)) {
stop(paste(
'Invalid border type specified.',
'Must be one of', paste(c(valid.border.types, 'or NA.'), collapse = ', ')
));
}

tree.df$border.type[is.na(tree.df$border.type)] <- if (is.numeric(tree.df$border.type)) 1 else 'solid';
} else {
tree.df$border.type <- 'solid';
}

if ('border.width' %in% colnames(tree.df)) {
tree.df$border.width <- as.numeric(tree.df$border.width);
tree.df$border.width[is.na(tree.df$border.width)] <- 1;
} else {
tree.df$border.width <- 1;
}

out.df <- data.frame(
id = c(-1, tree.df$child),
label.text = c('', tree.df$label),
ccf = if (is.null(tree.df$CP)) NA else c(1, tree.df$CP),
color = colour.scheme[1:(nrow(tree.df) + 1)],
angle = c(NA, tree.df$angle),
node.colour = c(NA, tree.df$node.col),
node.label.colour = c(NA, tree.df$node.label.col),
border.colour = c(NA, tree.df$border.col),
border.type = c(NA, tree.df$border.type),
border.width = c(NA, tree.df$border.width),
parent = as.numeric(c(NA,tree.df$parent)),
excluded = c(TRUE, rep(FALSE, nrow(tree.df))),
bell = c(FALSE, rep(bells, nrow(tree.df))),
Expand Down Expand Up @@ -202,3 +256,45 @@ get.y.axis.position <- function(tree.colnames) {

return(y.axis.position);
}

prep.node.label.colours <- function(tree.df) {
node.col.error.message <- 'Cannot prepare node label colour without node colour values.';

if (!'node.col' %in% colnames(tree.df)) {
stop(paste(
node.col.error.message,
'"node.col" column not found in tree.df'
));
} else if (any(is.na(tree.df$node.col))) {
stop(paste(
node.col.error.message,
'NA values found in tree.df "node.col" column.'
));
}

label.colours <- if (!'node.label.col' %in% colnames(tree.df)) {
rep(NA, nrow(tree.df));
} else {
tree.df$node.label.col;
}

NA.indices <- is.na(label.colours);
label.colours[NA.indices] <- as.character(sapply(
tree.df$node.col[NA.indices],
FUN = get.default.node.label.colour
));

return(label.colours);
}

get.default.node.label.colour <- function(node.colour) {
white.luminance <- get.colour.luminance('black');
node.colour.luminance <- get.colour.luminance(node.colour);

contrast.ratio <- get.contrast.ratio(white.luminance, node.colour.luminance);

# WCAG minimum contrast for normal/small text
# https://www.w3.org/TR/2008/REC-WCAG20-20081211/#visual-audio-contrast-contrast
WCAG.contrast.threshold <- 7;
return(if (contrast.ratio < WCAG.contrast.threshold) 'white' else 'black');
}
2 changes: 0 additions & 2 deletions man/SRCGrob.Rd
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,6 @@ SRCGrob(
main.y = NULL,
main.cex = 1.7,
node.radius = 0.1,
node.col = "grey29",
seg1.col = "black",
seg2.col = "green",
line.lwd = 3,
Expand Down Expand Up @@ -73,7 +72,6 @@ SRCGrob(
\item{main.y}{Move the main plot title position up or down}
\item{main.cex}{Font size for the main plot title}
\item{node.radius}{Node size}
\item{node.col}{Node colour}
\item{seg1.col}{Colour of the first set of tree branch segments}
\item{seg2.col}{Colour of the second set of tree branch segments}
\item{line.lwd}{Branch segment thickness}
Expand Down
Binary file added tests/testthat/data/branching.data.Rda
Binary file not shown.
Binary file added tests/testthat/data/branching.plots.Rda
Binary file not shown.
Binary file modified tests/testthat/data/linear.data.Rda
Binary file not shown.
Binary file modified tests/testthat/data/linear.plots.Rda
Binary file not shown.
40 changes: 29 additions & 11 deletions tests/testthat/helper-compare.R
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ compare.trees <- function(example, test) {
Negate(is.null),
c(
list(getGrob(x, 'tree.segs.1')),
list(getGrob(x, 'tree.segs.2'))
list(getGrob(x, 'tree.segs.2'))=
)
);
axes <- sapply(
Expand Down Expand Up @@ -93,15 +93,21 @@ compare.trees <- function(example, test) {
example.grobs <- get.line.grobs(example);
test.grobs <- get.line.grobs(test);

all(sapply(
1:(length(example.grobs)),
FUN = function(i) {
compare.lines(
example.grobs[[i]],
test.grobs[[i]]
);
}
));
result <- if (length(example.grobs) > 0) {
all(sapply(
1:(length(example.grobs)),
FUN = function(i) {
compare.lines(
example.grobs[[i]],
test.grobs[[i]]
);
}
));
} else {
TRUE;
}

return(result);
}

test.text.grobs <- function(example, test) {
Expand Down Expand Up @@ -178,7 +184,6 @@ compare.trees <- function(example, test) {
));

gp.equal <- identical(x$gp, y$gp);

all(coords.equal, gp.equal);
}

Expand All @@ -196,6 +201,7 @@ compare.trees <- function(example, test) {
));
}

<<<<<<< HEAD
segments.match <- test.segment.grobs(example, test)
text.match <- test.text.grobs(example, test);
polygons.match <- test.polygon.grobs(example, test);
Expand All @@ -205,5 +211,17 @@ compare.trees <- function(example, test) {
text.match,
polygons.match,
lines.match
=======
segs.equal <- test.segment.grobs(example, test);
text.equal <- test.text.grobs(example, test);
polygons.equal <- test.polygon.grobs(example, test);
lines.equal <- test.line.grobs(example, test);
print(c(segs.equal, text.equal, polygons.equal, lines.equal))
all(
segs.equal,
text.equal,
polygons.equal,
lines.equal
>>>>>>> e9fd02b08dad613689f57f3c74b7c0ee522d25be
);
}
1 change: 0 additions & 1 deletion tests/testthat/helper-multitest.R
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,6 @@ create.test.tree <- function(tree, node.text, sample, ...) {
scale1 = 0.9,
seg1.col = 'navy',
seg2.col = 'gold',
node.col = 'grey40',
line.lwd = 4,
yaxis1.label = 'PGA',
yaxis2.label = 'SNV',
Expand Down
Loading

0 comments on commit 7c9d947

Please sign in to comment.