Skip to content

Commit

Permalink
Merge pull request #60 from bupaverse/dev
Browse files Browse the repository at this point in the history
cran 0.5.5
  • Loading branch information
gertjanssenswillen authored Aug 30, 2024
2 parents 5b3a53f + d9c5c6c commit 3da9b09
Show file tree
Hide file tree
Showing 56 changed files with 5,224 additions and 110 deletions.
10 changes: 6 additions & 4 deletions DESCRIPTION
Original file line number Diff line number Diff line change
@@ -1,14 +1,14 @@
Package: processmapR
Type: Package
Title: Construct Process Maps Using Event Data
Version: 0.5.3
Version: 0.5.5
Authors@R: c(person("Gert","Janssenswillen",email = "[email protected]", role = c("aut","cre")),
person("Gerard","van Hulzen", email = "[email protected]", role = c("ctb")),
person("Benoît","Depaire",email = "[email protected]", role = c("ctb")),
person("Felix","Mannhardt",email = "[email protected]", role = c("ctb")),
person("Thijs","Beuving", email = "[email protected]", role = c("ctb")),
person("urvikalia", role = c("ctb"),
person("Hasselt University", role = c("cph"))))
person("urvikalia", role = c("ctb")),
person("Hasselt University", role = c("cph")))
Description: Visualize event logs using directed graphs, i.e. process maps. Part of the 'bupaR' framework.
License: MIT + file LICENSE
LinkingTo: Rcpp, BH
Expand Down Expand Up @@ -36,7 +36,8 @@ Imports:
tidyr,
htmltools,
Rcpp,
lifecycle
lifecycle,
htmlwidgets
Encoding: UTF-8
RoxygenNote: 7.2.3
Suggests:
Expand Down Expand Up @@ -83,3 +84,4 @@ Collate:
'resource_matrix.R'
'trace_explorer.R'
'utils.R'
'utils_animateR.R'
1 change: 1 addition & 0 deletions NAMESPACE
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,7 @@ import(dplyr)
import(edeaR)
import(forcats)
import(ggplot2)
import(htmlwidgets)
import(miniUI)
import(shiny)
import(stringr)
Expand Down
4 changes: 2 additions & 2 deletions R/dotted_chart.R
Original file line number Diff line number Diff line change
Expand Up @@ -16,9 +16,9 @@
#' @param units [`character`] (default `"auto"`): Time units to use on the x-axis in case of relative time: `"auto"`
#' (default, see **Details**), `"secs"`, `"mins"`, `"hours"`, `"days"`, or `"weeks"`.
#' @param add_end_events [`logical`] (default `FALSE`): Whether to add dots for the complete lifecycle event with a different shape.
#' @param scale_color [`ggplot2`] scale function (default [`scale_color_discrete_bupaR`][`bupaR::scale_color_discrete_bupaR`]):
#' @param scale_color `ggplot2` scale function (default [`scale_color_discrete_bupaR`][`bupaR::scale_color_discrete_bupaR`]):
#' Set color scale. Defaults to [`scale_color_discrete_bupaR`][`bupaR::scale_color_discrete_bupaR`]. Replaced with [`scale_color_discrete`][`ggplot2::scale_color_discrete`] when more than 26 activities are present.
#' @param plotly [`logical`] (default `FALSE`): Return a [`plotly`] object, instead of a [`ggplot2`].
#' @param plotly [`logical`] (default `FALSE`): Return a `plotly` object, instead of a `ggplot2`.
#' @param eventlog `r lifecycle::badge("deprecated")`; please use `log` instead.
#'
#' @details
Expand Down
2 changes: 1 addition & 1 deletion R/layout.R
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
#'
#' @param fixed_positions When specified as a data.frame with three columns 'act', 'x', and 'y' the position of nodes is fixed. Note that using this option switches to the 'neato' layout engine.
#' @param edge_weight When `TRUE` then the frequency with which an edge appears in the process map has influence on the process map layout. Edges with higher frequency get higher priority in the layout algorithm, which increases the visibility of 'process highways'. Note that this has no effect when using the 'fixed_positions' parameters.
#' @param edge_cutoff Edges that appear in the process map below this frequency are not considered at all when calculating the layout. This may create very long and complicated edge routings when choosen too high. Note that this has no effect when using the 'fixed_positions' parameters.
#' @param edge_cutoff (\code{\link{numeric}}) Number between 0 and 1. Edges with a relative frequency below the cut off are not considered at all when calculating the layout. This may create very long and complicated edge routings when choosen too high. Note that this has no effect when using the 'fixed_positions' parameters.
#'
#' @export
#'
Expand Down
150 changes: 66 additions & 84 deletions R/process_map.R
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,6 @@
#' @param fixed_edge_width If TRUE, don't vary the width of edges.
#' @param layout List of parameters influencing the (automatic) layout of the process map. Use \code{\link{layout_pm}} to create a suitable parameter list.
#'
#' @param fixed_node_pos Deprecated, please use the 'layout' parameter instead.
#' @param ... Deprecated arguments
#'
#' @inheritParams dotted_chart
Expand Down Expand Up @@ -47,7 +46,6 @@ process_map <- function(log,
render = T,
fixed_edge_width = F,
layout = layout_pm(),
fixed_node_pos = NULL,
eventlog = deprecated(),
...) {
UseMethod("process_map")
Expand All @@ -68,7 +66,6 @@ process_map.eventlog <- function(log,
render = T,
fixed_edge_width = F,
layout = layout_pm(),
fixed_node_pos = NULL,
eventlog = deprecated(),
...) {

Expand Down Expand Up @@ -102,17 +99,10 @@ process_map.eventlog <- function(log,
weight <- NULL
constraint <- NULL

if(!is.null(fixed_node_pos)) {
warning("Argument fixed_node_pos deprecated, use layout argument instead.")
layout <- layout_pm(fixed_positions = fixed_node_pos)
}

if (any(is.na(eventlog %>% pull(!!timestamp_(eventlog))))) {
warning("Some of the timestamps in the supplied event log are missing (NA values). This may result in a invalid process map!")

}


#base_precedence <- create_base_precedence(eventlog, type_nodes, type_edges)

eventlog <- ungroup_eventlog(eventlog)
Expand Down Expand Up @@ -208,17 +198,17 @@ process_map.eventlog <- function(log,
data.table::setDT(base_log, key = c("start_time", "min_order"))
base_log[, ACTIVITY_CLASSIFIER_ := ordered(ACTIVITY_CLASSIFIER_,
levels = c("ARTIFICIAL_START", as.character(sort(activity_labels(eventlog))), "ARTIFICIAL_END"))
][, `:=`(next_act = data.table::shift(ACTIVITY_CLASSIFIER_, 1, type = "lead"),
next_start_time = data.table::shift(start_time, 1, type = "lead"),
next_end_time = data.table::shift(end_time, 1, type = "lead")),
by = CASE_CLASSIFIER_] %>%
merge(base_nodes, by.x = c("ACTIVITY_CLASSIFIER_"), by.y = c("ACTIVITY_CLASSIFIER_"), all = TRUE) %>%
merge(base_nodes, by.x = c("next_act"), by.y = c("ACTIVITY_CLASSIFIER_"), all = TRUE) %>%
][, `:=`(next_act = data.table::shift(ACTIVITY_CLASSIFIER_, 1, type = "lead"),
next_start_time = data.table::shift(start_time, 1, type = "lead"),
next_end_time = data.table::shift(end_time, 1, type = "lead")),
by = CASE_CLASSIFIER_] %>%
merge(base_nodes, by.x = c("ACTIVITY_CLASSIFIER_"), by.y = c("ACTIVITY_CLASSIFIER_"), all = TRUE) %>%
merge(base_nodes, by.x = c("next_act"), by.y = c("ACTIVITY_CLASSIFIER_"), all = TRUE) %>%
as.data.frame() %>%
select(everything(),
-n.x, -n.y,
from_id = node_id.x,
to_id = node_id.y) -> base_precedence
select(everything(),
-n.x, -n.y,
from_id = node_id.x,
to_id = node_id.y) -> base_precedence



Expand All @@ -232,6 +222,8 @@ process_map.eventlog <- function(log,
edges <- attr(type_edges, "create_edges")(base_precedence, type_edges, extra_data)




# secondary info
if(!is.null(sec_nodes)) {
nodes_secondary <- attr(sec_nodes, "create_nodes")(base_precedence, sec_nodes, extra_data) %>%
Expand Down Expand Up @@ -261,6 +253,10 @@ process_map.eventlog <- function(log,
edges %>% mutate(penwidth = 1) -> edges
}

edges %>%
mutate(labeltooltip = paste0(str_replace(ACTIVITY_CLASSIFIER_, "ARTIFICIAL_START",""), " > ",
str_replace(next_act, "ARTIFICIAL_END", ""))) -> edges

# This is to improve the DOT layout by using the frequency information
if (layout$edge_weight) {
edges %>% mutate(weight = as.integer(((n - min(n)) / max(n)) * 100)) -> edges
Expand Down Expand Up @@ -291,6 +287,7 @@ process_map.eventlog <- function(log,
create_node_df(n = nrow(nodes),
label = nodes$label,
shape = nodes$shape,
gradientangle = 0.1,
color_level = nodes$color_level,
style = "rounded,filled",
fontcolor = nodes$fontcolor,
Expand Down Expand Up @@ -333,6 +330,8 @@ process_map.eventlog <- function(log,

create_edge_df(from = edges$from_id,
to = edges$to_id,
labeltooltip = edges$labeltooltip,
edgetooltip = edges$labeltooltip,
label = edges$label,
penwidth = edges$penwidth,
# style = edges$style,
Expand All @@ -342,6 +341,10 @@ process_map.eventlog <- function(log,
weight = edges$weight,
constraint = edges$constraint) -> edges_df


edges_df %>%
mutate(len = weight,decorate = constraint) -> edges_df

create_graph(nodes_df, edges_df) %>%
add_global_graph_attrs(attr = "rankdir", value = rankdir,attr_type = "graph") %>%
add_global_graph_attrs(attr = "layout", value = if_else(is.data.frame(layout$fixed_positions), "neato", "dot"), attr_type = "graph") %>%
Expand All @@ -365,54 +368,35 @@ process_map.eventlog <- function(log,
rename(node = ACTIVITY_CLASSIFIER_) %>%
select(node, from_id, value) -> nodes

if(render == T) {

# Since DiagrammeR does not support the necessary GraphViz attributes,
# we use a workaround to add them tot the DOT code. See the issue logged here:
# https://github.com/rich-iannone/DiagrammeR/issues/360

# hack to add 'weight' attribute to the graph
graph$edges_df %>%
mutate(len = weight, decorate = constraint) -> graph$edges_df

graph %>% render_graph() -> graph
attr(graph, "base_precedence") <- base_precedence
attr(graph, "edges") <- edges
attr(graph, "nodes") <- nodes

# graph$x$diagram %>%
# stringr::str_replace_all("len", "weight") %>%
# stringr::str_replace_all("decorate", "constraint") -> graph$x$diagram

attr(graph, "base_precedence") <- base_precedence
attr(graph, "edges") <- edges
attr(graph, "nodes") <- nodes

graph %>% return()
} else {
attr(graph, "base_precedence") <- base_precedence
attr(graph, "edges") <- edges
attr(graph, "nodes") <- nodes
graph %>% return()
if(render == T) {
graph %>% render_map() -> graph
}

graph %>% return()

}

#' @describeIn process_map Process map for event log
#' @export


process_map.grouped_eventlog <- function(log,
type = frequency("absolute"),
sec = NULL,
type_nodes = type,
type_edges = type,
sec_nodes = sec,
sec_edges = sec,
rankdir = "LR",
render = T,
fixed_edge_width = F,
layout = layout_pm(),
fixed_node_pos = NULL,
eventlog = deprecated(),
...) {
type = frequency("absolute"),
sec = NULL,
type_nodes = type,
type_edges = type,
sec_nodes = sec,
sec_edges = sec,
rankdir = "LR",
render = T,
fixed_edge_width = F,
layout = layout_pm(),
eventlog = deprecated(),
...) {
log <- lifecycle_warning_eventlog(log, eventlog)

m <- mapping(log)
Expand All @@ -423,18 +407,17 @@ process_map.grouped_eventlog <- function(log,
distinct() %>%
unique(), collapse = ","),
group_map = process_map(re_map(., m),
type = type,
sec = sec,
type_nodes = type_nodes,
type_edges = type_edges,
sec_nodes = sec_nodes,
sec_edges = sec_edges,
rankdir = rankdir,
render = F,
fixed_edge_width = fixed_edge_width,
layout = layout,
fixed_node_pos = fixed_node_pos,
...)) -> grouped_map
type = type,
sec = sec,
type_nodes = type_nodes,
type_edges = type_edges,
sec_nodes = sec_nodes,
sec_edges = sec_edges,
rankdir = rankdir,
render = F,
fixed_edge_width = fixed_edge_width,
layout = layout,
...)) -> grouped_map

if (render) {
group_tags <-
Expand All @@ -458,23 +441,22 @@ process_map.grouped_eventlog <- function(log,
#' @export

process_map.activitylog <- function(log,
type = frequency("absolute"),
sec = NULL,
type_nodes = type,
type_edges = type,
sec_nodes = sec,
sec_edges = sec,
rankdir = "LR",
render = T,
fixed_edge_width = F,
layout = layout_pm(),
fixed_node_pos = NULL,
eventlog = deprecated(),
...) {
type = frequency("absolute"),
sec = NULL,
type_nodes = type,
type_edges = type,
sec_nodes = sec,
sec_edges = sec,
rankdir = "LR",
render = T,
fixed_edge_width = F,
layout = layout_pm(),
eventlog = deprecated(),
...) {
log <- lifecycle_warning_eventlog(log, eventlog)

process_map.eventlog(to_eventlog(log), type, sec, type_nodes, type_edges, sec_nodes, sec_edges, rankdir,
render, fixed_edge_width, layout, fixed_node_pos)
render, fixed_edge_width, layout)


}
1 change: 1 addition & 0 deletions R/processmapR.R
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@
#' @importFrom rlang arg_match is_integerish sym caller_env
#' @importFrom cli cli_abort cli_warn
#' @importFrom lifecycle deprecated
#' @import htmlwidgets

utils::globalVariables(c(".", ".order"))

Expand Down
Loading

0 comments on commit 3da9b09

Please sign in to comment.