Skip to content

Instantly share code, notes, and snippets.

@timelyportfolio
Last active August 2, 2024 16:33
Show Gist options
  • Save timelyportfolio/866f2507ee2a8c50809cfacba575416b to your computer and use it in GitHub Desktop.
Save timelyportfolio/866f2507ee2a8c50809cfacba575416b to your computer and use it in GitHub Desktop.
summary widget with flexdashboard valueBox
---
title: "Example of filtered values boxes"
output:
flexdashboard::flex_dashboard:
orientation: rows
vertical_layout: fill
mathjax: null
---
```{r setup, include=FALSE}
library(flexdashboard)
library(htmltools)
library(summarywidget)
library(dplyr)
library(crosstalk)
valueBoxSummaryWidget <- function (value, caption = NULL, icon = NULL, color = NULL, href = NULL)
{
if (!is.null(color) && color %in% c("primary", "info",
"success", "warning", "danger"))
color <- paste0("bg-", color)
valueOutput <- tags$span(class = "value-summarywidget-output", `data-caption` = caption,
`data-icon` = icon, `data-color` = color,
`data-href` = href, value)
hasPrefix <- function(x, prefix) {
if (!is.null(x))
grepl(paste0("^", prefix), x)
else FALSE
}
fontAwesome <- hasPrefix(icon, "fa")
ionicons <- hasPrefix(icon, "ion")
deps <- flexdashboard:::html_dependencies_fonts(fontAwesome, ionicons)
if (length(deps) > 0)
valueOutput <- attachDependencies(valueOutput, deps)
valueOutput
}
```
```{r}
mtcars_shared <-
SharedData$new(
mtcars
)
```
<script>
window.FlexDashboardComponents.push({
type: 'custom',
find: function(container) {
if (container.find('span.value-summarywidget-output').length)
return container;
else
return $();
},
flex: function(fillPage) {
return false;
},
layout: function(title, container, element, fillPage) {
// alias variables
var chartTitle = title;
var valueBox = element;
// add value-box class to container
container.addClass('value-box');
// value paragraph
var value = $('<p class="value"></p>');
// if we have shiny-text-output then just move it in
var valueOutputSpan = [];
var shinyOutput = valueBox.find('.shiny-valuebox-output').detach();
var summaryOutput = valueBox.find('.summarywidget').detach();
if (shinyOutput.length) {
valueBox.children().remove();
shinyOutput.html('&mdash;');
value.append(shinyOutput);
}
if (summaryOutput.length) {
value.append(summaryOutput);
valueOutputSpan = valueBox.find('span.value-summarywidget-output')
}
// caption
var caption = $('<p class="caption"></p>');
caption.append(chartTitle);
// build inner div for value box and add it
var inner = $('<div class="inner"></div>');
inner.append(value);
inner.append(caption);
valueBox.append(inner);
// add icon if specified
var icon = $('<div class="icon"><i></i></div>');
valueBox.append(icon);
function setIcon(chartIcon) {
var iconLib = '';
var iconSplit = chartIcon.split(' ');
if (iconSplit.length > 1) {
iconLib = iconSplit[0];
chartIcon = iconSplit.slice(1).join(' ');
} else {
var components = chartIcon.split('-');
if (components.length > 1)
iconLib = components[0];
}
icon.children('i').attr('class', iconLib + ' ' + chartIcon);
}
var chartIcon = valueBox.attr('data-icon');
if (chartIcon)
setIcon(chartIcon);
// set color based on data-background if necessary
var dataBackground = valueBox.attr('data-background');
if (dataBackground)
valueBox.css('background-color', bgColor);
else {
// default to bg-primary if no other background is specified
if (!valueBox.hasClass('bg-primary') &&
!valueBox.hasClass('bg-info') &&
!valueBox.hasClass('bg-warning') &&
!valueBox.hasClass('bg-success') &&
!valueBox.hasClass('bg-danger')) {
valueBox.addClass('bg-primary');
}
}
// handle data attributes in valueOutputSpan
function handleValueOutput(valueOutput) {
// caption
var dataCaption = valueOutput.attr('data-caption');
if (dataCaption)
caption.html(dataCaption);
// icon
var dataIcon = valueOutput.attr('data-icon');
if (dataIcon)
setIcon(dataIcon);
// color
var dataColor = valueOutput.attr('data-color');
if (dataColor) {
if (dataColor.indexOf('bg-') === 0) {
valueBox.css('background-color', '');
if (!valueBox.hasClass(dataColor)) {
valueBox.removeClass('bg-primary bg-info bg-warning bg-danger bg-success');
valueBox.addClass(dataColor);
}
} else {
valueBox.removeClass('bg-primary bg-info bg-warning bg-danger bg-success');
valueBox.css('background-color', dataColor);
}
}
// url
var dataHref = valueOutput.attr('data-href');
if (dataHref) {
valueBox.addClass('linked-value');
valueBox.off('click.value-box');
valueBox.on('click.value-box', function(e) {
window.FlexDashboardUtils.showLinkedValue(dataHref);
});
}
}
// check for a valueOutputSpan
if (valueOutputSpan.length > 0) {
handleValueOutput(valueOutputSpan);
}
// if we have a shinyOutput then bind a listener to handle
// new valueOutputSpan values
shinyOutput.on('shiny:value',
function(event) {
var element = $(event.target);
setTimeout(function() {
var valueOutputSpan = element.find('span.value-output');
if (valueOutputSpan.length > 0)
handleValueOutput(valueOutputSpan);
}, 10);
}
);
}
});
</script>
Row {data-width=650}
-----------------------------------------------------------------------
### Value Box A
```{r}
valueBox(count(mtcars),
color = "red")
```
### Value Box B
```{r}
valueBox(summarise(mtcars, mean = round(mean(mpg), 1)), color = "blue")
```
Row {data-width=350}
-----------------------------------------------------------------------
### Hacked value Box A
```{r}
valueBoxSummaryWidget(
summarywidget(
mtcars_shared,
statistic = 'count',
column = 'mpg',
digits = 0
),
color = "purple"
)
```
### Hacked value Box B
```{r}
valueBoxSummaryWidget(
summarywidget(
mtcars_shared,
statistic = 'mean',
column = 'mpg',
digits = 1
),
color = "hotpink"
)
```
Row {data-width=650}
-----------------------------------------------------------------------
### Filters
```{r fig.height = 5}
bscols(widths = c(4, 4),
filter_select("cyl", "Cylinders", mtcars_shared, ~cyl),
filter_slider("disp", "Disp", mtcars_shared, ~disp)
)
```
@zeroeuros
Copy link

Hi there!
Any chance to have the same summary widget operating on plot_ly gauge? No way to make it work in the value parameter !
That would be a great addition...
Thanks
Fabrizio

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment