Custom function to loop over a list
Asked Answered
H

1

2

I have a working custom function but not sure how to allow it to loop with a list of inputs. Looks like I need to understand apply() and the such but I'm not quite there with my current setup. The function uses rollapply() to find the largest metric for a given time frame.

library(zoo)
library(dplyr)

# Data
set.seed(1)
df <- tibble(player = rep(LETTERS[1:2], each = 10),
             minute = rep(1:10, times = 2),
             tdc = sample(100:200,size = 20),
             sumad = sample(1:10, size = 20, replace = TRUE))

# Custom function
x_min_roll <- function(df, metric, n_minutes, fun){
  metric <- ensym(metric)
  newname <- glue::glue("{rlang::as_string(metric)}_x{as.character(n_minutes)}")
  df %>% 
    # dynamically create new column name based on input
    mutate("{newname}" := rollapply(!!metric, n_minutes, fun, align='left', fill=NA)) %>% 
    group_by(player) %>% 
    slice_max(.data[[newname]]) %>% 
    select(player, .data[[newname]])
}

# This works
df %>% 
  x_min_roll(metric = tdc, n_minutes = 2, fun = sum)

# A tibble: 2 x 2
# Groups:   player [2]
  player tdc_x2
  <chr>   <int>
1 A         339
2 B         380

I would like to be able to do this:

metric_list <- c('tdc', 'sumad')
minutes_list <- c(2,5)

df %>% 
  x_min_roll(metric = metric_list, n_minutes = minutes_list, fun = sum) %>% 
  # maybe a few more steps here.... to get this

# A tibble: 2 x 5
  player tdc_x2 tdc_x5 sumad_x2 sumad_x5
  <chr>   <dbl>  <dbl>    <dbl>    <dbl>
1 A         339    793       20       36
2 B         380    866       19       41
Hirsch answered 11/2, 2022 at 16:42 Comment(0)
P
2

We can use map2 to loop over the corresponding elements of both vectors

library(purrr)
library(dplyr)
map2(metric_list, minutes_list, 
  ~ df %>%
    x_min_roll(metric = !!.x, n_minutes = .y, fun = sum))

-output

[[1]]
# A tibble: 2 × 2
# Groups:   player [2]
  player tdc_x2
  <chr>   <int>
1 A         339
2 B         380

[[2]]
# A tibble: 3 × 2
# Groups:   player [2]
  player sumad_x5
  <chr>     <int>
1 A            36
2 B            41
3 B            41

EDIT: Based on @Onyambu's comments


If we want for each combination, then use crossing to create the combination

library(tidyr)
crossing(metric_list, minutes_list) %>% 
 pmap(~ df %>% 
      x_min_roll(metric = !!.x, n_minutes = .y, fun = sum))

Based on the comments from the OP, if we want to combine the datasets

crossing(metric_list, minutes_list) %>% 
 pmap(~ df %>% x_min_roll(metric = !!.x, n_minutes = .y, fun = sum)) %>%
    reduce(inner_join, by = 'player')
Priapic answered 11/2, 2022 at 16:44 Comment(10)
Since .y is numeric, I believe you could ignore the bang bang operator on it. in that case, the naming will be as required. Great solutionPlinth
@Plinth thanks, I overlooked that part.Priapic
If I add another value the minutes list like this minutes_list <- c(2,3,5). I receive and error message Error: Mapped vectors must have consistent lengths: *.x has length 2 * .y has length 3Hirsch
@Hirsch then what about metric_list it is only of length 2. I am not sure what you are trying to do with that function when one of the input is of length greater different than the otherPriapic
@Hirsch your function is constructed in such a way to take a single value for metric and n_minutes (if I understand correctly)Priapic
That's my problem!!! Sorry ~ let me edit the question to clarify.Hirsch
@Hirsch suppose you have minutes_list <- c(2, 3, 5) then the 5 should go with which value of metric_list?Priapic
@Hirsch if it is combination of both, then create the combination with crossing (updated my post)Priapic
@Priapic yes that's it!! Final request? How do I recombine back into one dataframe?Hirsch
Answering my own question because I realize the output is a list of tibbles.... this works. crossing(metric_list, minutes_list) %>% pmap(~ df %>% x_min_roll(metric = !!.x, n_minutes = .y, fun = sum)) %>% reduce(inner_join, by = 'player') @akron I will mark your answer as correct but if you want to add this bit to it it will answer the question in full - with you doing 99.9% of the answering ;)Hirsch

© 2022 - 2025 — McMap. All rights reserved.