How to animate a bar plot in R that represents one variable increasing over time, synchronised with a second animation?

huangapple go评论107阅读模式
英文:

How to animate a bar plot in R that represents one variable increasing over time, synchronised with a second animation?

问题

我会为您翻译代码部分,如下所示:

  1. library(ggplot2)
  2. library(gganimate)
  3. library(dplyr)
  4. library(gifski)
  5. library(tidyverse)
  6. library(magick)
  7. plots <- bar_data %>%
  8. ggplot(aes(x = x_pos, y = cume_impulse)) + geom_col()
  9. # 将条形图的动画命名为bar_anim
  10. bar_anim <- plots + transition_time(time_steps)
  11. # 设置参数并显示条形图动画
  12. bar_gif <- animate(bar_anim, height = 500, width = 800, fps = 20, duration = 20)
  13. # 将条形图保存为gif文件
  14. anim_save("bar_impulse.gif", bar_gif)
  15. # 下面两行是在使用image_append函数处理之前所需的
  16. mgif_particle <- image_read(particle_gif)
  17. mgif_bar <- image_read(bar_gif)
  18. # 合并两个gif的第一帧
  19. paired_gif <- image_append(c(mgif_particle[1], mgif_bar[1]))
  20. for (i in 2:317) {
  21. combined <- image_append(c(mgif_particle[i], mgif_bar[i]))
  22. paired_gif <- c(paired_gif, combined)
  23. }
  24. anim_save("bar_plus_particle.gif", paired_gif)

这是您提供的R代码的翻译。如果您需要任何进一步的协助,请随时告诉我。

英文:

I would like to create an animated bar plot that increases at specified times, synchronized with another plot (links to both gifs at bottom). Whenever the ball hits the right side wall I'd like it to display as a synchronized increase in the bar graph.

The data (bar_data) is filtered from a data set of 17000 time steps. There is only 4 occasions where the ball hits the wall, so 4 occasions where the impulse should increase on the bar graph.

The data:

  1. impulse &lt;- c(8e-24, 8e-24, 8e-24, 8e-24)
  2. cume_impulse &lt;- c(8.0e-24, 1.6e-24, 2.4e-24, 3.2e-24) #cumulative total
  3. time_steps &lt;- c(1.132, 6.136, 11.140, 16.144) #time step at which the increase occurs
  4. bar_data &lt;- data.frame(time_steps, impulse, cume_impulse)

summary of my task:

  • create an animation which shows a growing bar plot, synchronized with right side wall contact. Two gifs side-by-side should form a single animation.

I've managed to get the animations paired into the same gif, but I can't get the timing right.

Here's what I've tried so far:

  1. library(ggplot2)
  2. library(gganimate)
  3. library(dplyr)
  4. library(gifski)
  5. library(tidyverse)
  6. library(magick)
  7. plots &lt;- bar_data %&gt;%
  8. ggplot(aes(x = x_pos, y = cume_impulse)) + geom_col()
  9. #label the animation for the bar plot as bar_anim
  10. bar_anim &lt;- plots + transition_time(time_steps)
  11. #set the parameters and display the bar plot animation
  12. bar_gif &lt;- animate(bar_anim, height = 500, width = 800, fps = 20, duration = 20)
  13. #save the bar plot to a gif file
  14. anim_save(&quot;bar_impulse.gif&quot;, bar_gif)
  15. #the following two lines are required before the image_append function will process
  16. mgif_particle &lt;- image_read(particle_gif)
  17. mgif_bar &lt;- image_read(bar_gif)
  18. #join the first fraomes of the gifs
  19. paired_gif &lt;- image_append(c(mgif_particle[1], mgif_bar[1]))
  20. for(i in 2:317){
  21. combined &lt;- image_append(c(mgif_particle[i], mgif_bar[i]))
  22. paired_gif &lt;- c(paired_gif, combined)
  23. }
  24. anim_save(&quot;bar_plus_particle.gif&quot;, paired_gif)

Any help would be greatly appreciated.

Gif links:
bar gif
ball motion gif

additional details. This code precedes the code above:

  1. m &lt;- 1e-26 #Mass of a particle in kg
  2. vx_i &lt;- 400 #initial x velocity in m/s
  3. vy_i &lt;- 200 #initial y velocity in m/s
  4. #creating variables for the initial coordinates of the particle
  5. x_i &lt;- round(runif(n = 1, min = 0,max = 1), 3)
  6. y_i &lt;- round(runif(n = 1, min = 0,max = 1), 3)
  7. #initialise vectors that will be added to a data frame
  8. x_pos &lt;- round(seq(x_i, (x_i + 17), 0.001), 3) #17 was an arbitrary choice to allow for a few bounces on the right wall
  9. y_pos &lt;- round(seq(y_i, (y_i + 17), 0.001), 3)
  10. time_steps &lt;- seq(0, 17.001, 0.001)
  11. #loop to detect x direction wall contact
  12. xv = 0.0004
  13. for (i in 1 + seq_along(x_pos)) {
  14. if (x_pos[i-1] &gt; 1 | x_pos[i-1] &lt; 0 ){xv = xv*(-1)}
  15. x_pos[[i]] &lt;- round(x_pos[[i-1]] + xv, 4)
  16. }
  17. #loop for y direction wall contact
  18. yv = 0.0002
  19. for (i in 1 + seq_along(y_pos)) {
  20. if (y_pos[i-1] &gt; 1 | y_pos[i-1] &lt; 0 ){yv = yv*(-1)}
  21. y_pos[[i]] &lt;- round(y_pos[[i-1]] + yv, 4)
  22. }
  23. #creating a data frame with the particle&#39;s kinematic information (this line must come after the loops)
  24. pos_data &lt;- data.frame(x_pos, y_pos, time_steps)
  25. graph1 = pos_data %&gt;% ggplot(aes(x_pos, y_pos)) +
  26. geom_point(colour = &quot;red&quot;, size = 2) +
  27. xlab(NULL) + ylab(NULL) + xlim(0, 1) + ylim(0, 1)
  28. #initialising vector for x-momentum. Same length as x_pos with arbitrary values.
  29. px &lt;- rep(xv,length(x_pos))
  30. #add a vector to pos_data called x_dif to pos_data that represents the change in x position between successive time steps
  31. pos_data$x_dif &lt;- c(0, diff(pos_data$x_pos))
  32. # add a vector to pos_data called x_turn which will help detect changes in direction
  33. pos_data$x_turn &lt;- c(0, diff(sign(pos_data$x_dif)))
  34. # add a cumulative total column for impulse on right wall called cume_impulse
  35. pos_data$cume_impulse &lt;- cumsum(pos_data$impulse)
  36. #add a vector to pos_data which will include the impulse magnitude on the right wall at each time step
  37. pos_data &lt;- pos_data %&gt;% mutate(impulse = case_when(abs(pos_data$x_turn) == 2 ~ 2*(vx_i*m),
  38. abs(pos_data$x_turn) != 2 ~ 0))
  39. graph1_animation = graph1 + transition_time(time_steps) +
  40. labs(subtitle = &quot;{frame_time}&quot;) + ease_aes(&#39;linear&#39;) +
  41. shadow_wake(wake_length = 0.05) #animation code
  42. particle_gif &lt;- animate(graph1_animation, height = 500, width = 800, nframes = 317)
  43. anim_save(&quot;particle_bounce.gif&quot;, particle_gif) #saves animation as a gif file
  44. #Filter the data to only include frames that include impulse, and x_pos = 1 (right wall)
  45. bar_data &lt;- filter(pos_data,impulse!=0 &amp; x_pos == 1)
  46. #new column in bar_data called &quot;cume_impulse&quot; - cumulative total of the impulse on the right wall
  47. bar_data[&quot;cume_impulse&quot;] &lt;- cumsum(bar_data$impulse)

Most recent adjustments:

  1. ## initial position for particle
  2. x_i &lt;- round(runif(n = 1, min = 0,max = 1), 3)
  3. y_i &lt;- round(runif(n = 1, min = 0,max = 1), 3)
  4. ## more initial conditions
  5. m &lt;- 1e-26 #Mass of a particle in kg
  6. vx_i &lt;- 400 #initial x velocity in m/s
  7. vy_i &lt;- 200 #initial y velocity in m/s
  8. #initialise vectors that will be added to a data frame
  9. x_pos &lt;- round(seq(x_i, (x_i + 17), 0.001), 3)
  10. y_pos &lt;- round(seq(y_i, (y_i + 17), 0.001), 3)
  11. time_steps &lt;- seq(0, 17.001, 0.001)
  12. #xv_column &lt;- seq(0, 17.001, 0.001) # don&#39;t think this is necessary
  13. #loop for x direction wall contact
  14. xv = 0.0004
  15. for (i in 1 + seq_along(x_pos)) {
  16. if (x_pos[i-1] &gt; 1 | x_pos[i-1] &lt; 0 ){xv = xv*(-1)}
  17. x_pos[[i]] &lt;- round(x_pos[[i-1]] + xv, 4)
  18. }
  19. #loop for y direction wall contact
  20. yv = 0.0002
  21. for (i in 1 + seq_along(y_pos)) {
  22. if (y_pos[i-1] &gt; 1 | y_pos[i-1] &lt; 0 ){yv = yv*(-1)}
  23. y_pos[[i]] &lt;- round(y_pos[[i-1]] + yv, 4)
  24. }
  25. #creating a data frame with the particle&#39;s kinematic information (this line must come after the loops)
  26. pos_data &lt;- data.frame(x_pos, y_pos, time_steps)
  27. #initialising vector for x-momentum. Same length as x_pos with arbitrary values.
  28. px &lt;- rep(xv,length(x_pos))
  29. #add a vector to pos_data called x_dif to pos_data that represents the change in x position between successive time steps
  30. pos_data$x_dif &lt;- c(0, diff(pos_data$x_pos))
  31. # add a vector to pos_data called x_turn which will help detect changes in direction
  32. pos_data$x_turn &lt;- c(0, diff(sign(pos_data$x_dif)))
  33. #add a vector to pos_data which will include the impulse magnitude on the right wall at each time step
  34. pos_data &lt;- pos_data %&gt;% mutate(impulse = case_when(abs(pos_data$x_turn) == 2 ~ 2*(vx_i*m),
  35. abs(pos_data$x_turn) != 2 ~ 0))
  36. # add a cumulative total column for impulse on right wall called cume_impulse
  37. pos_data$cume_impulse &lt;- cumsum(pos_data$impulse)
  38. ## ------------------------------------ tjebo&#39;s ------------------------------------------------------
  39. ## --- I may have misunderstood which data you used.Did you use the the length 4 vectors below for the bar_data frame?
  40. ## ---- I presumed not, so I made the bar_data frame similar length to pos_data.
  41. ## using your pos_data frame and impulse vector
  42. ## I have changed the order of cume_impulse, because it somehow made more sense
  43. #cume_impulse &lt;- c(1.6e-24, 2.4e-24, 3.2e-24, 8.0e-24)
  44. ## time steps need to be character, for the later merge
  45. #time_steps &lt;- as.character(c(1.132, 6.136, 11.140, 16.144))
  46. bar_data &lt;- data.frame(pos_data$time_steps, pos_data$impulse, pos_data$cume_impulse)
  47. df1 &lt;- pos_data %&gt;% mutate(time_steps = as.character(time_steps))
  48. # --- data frames wouldn&#39;t merge because of the automatic name change to the columns. This was how I fixed that. Unsure why this didn&#39;t effect you
  49. names(bar_data)[names(bar_data)==&quot;pos_data.time_steps&quot;] &lt;- &quot;time_steps&quot;
  50. names(bar_data)[names(bar_data)==&quot;pos_data.cume_impulse&quot;] &lt;- &quot;cume_impulse&quot;
  51. names(bar_data)[names(bar_data)==&quot;pos_data.impulse&quot;] &lt;- &quot;impulse&quot;
  52. # --- I fragmented the code to help me understand the individual commands, and also to locate where it was breaking down
  53. df2 &lt;- pos_data %&gt;% left_join(bar_data, by = &quot;time_steps&quot;)
  54. df3 &lt;- pos_data %&gt;% bind_rows(., .)
  55. df4 &lt;- df3 %&gt;% mutate(
  56. time_steps = as.numeric(time_steps),
  57. panel = rep(c(&quot;bar&quot;, &quot;dot&quot;), each = nrow(pos_data)),
  58. ## removing x and y for the bar panel
  59. across(c(x_pos, y_pos), ~ifelse(panel == &quot;bar&quot;, NA, .x))
  60. )
  61. ## --- I wasn&#39;t quite sure what this command was for
  62. df5 &lt;- df4 %&gt;% fill(cume_impulse)
  63. ## --- I presumed this converted the NA&#39;s into zeros in cume_impulse?
  64. df5$cume_impulse[is.na(df5$cume_impulse)] &lt;- 0
  65. ## --- Looks like this converts the cume_impulse values that correspond to dot as NA. Unsure why it&#39;s necessary though.
  66. df5$cume_impulse[df5$panel == &quot;dot&quot;] &lt;- NA
  67. set.seed(1)
  68. times &lt;- sort(sample(unique(df5$time_steps), 100))
  69. df_frac &lt;- df5[df5$time_steps %in% times, ]
  70. ## dummy data for facet range
  71. bar_range &lt;- data.frame(x= .5, cume_impulse = range(df5$cume_impulse, na.rm = T), panel = &quot;bar&quot;)
  72. ## separate out the aesthetics to the respective layers
  73. p &lt;-
  74. ggplot(df_frac) +
  75. geom_blank(data = bar_range, aes(x = .5, y = cume_impulse)) +
  76. ## need position = &quot;identity&quot;
  77. geom_col(aes(x = .5, y = cume_impulse), na.rm = T, position = &quot;identity&quot;) +
  78. geom_point(aes(x_pos, y_pos), na.rm = T) +
  79. facet_wrap(~panel, scales = &quot;free_y&quot;) +
  80. scale_x_continuous(expand = c(0,0))
  81. p_anim &lt;- p + transition_time(time_steps)
  82. # this is not random = there are 100 unique time steps, and now 100 frames
  83. animate(p_anim, fps = 20, duration = 5)

答案1

得分: 0

以下是翻译好的部分:

"You could make your life (and your code How to animate a bar plot in R that represents one variable increasing over time, synchronised with a second animation? a bit easier by creating those two plots as one single plot - using facets. This requires just a bit of data hacking, so that you create two separate panels. The biggest challenge is actually to generate two different 'ease_aes' for geom_bar and geom_point. As both geom layers are using the same aesthetic (x and y), we cannot simply define ease_aes for one of those aesthetics. I don't think one can define two different ease_aes for different geoms without a little hack. You have already quite a lot of intermediate time steps - I am merging this data with the bar code data and then removing the x/y according to the facet panel. The merge only properly works when your time values are character, probably because of floating point issues. Because my computer is not great and it is extremely computationally intense to compute 17000 frames, I will reduce the data drastically.

The only issue that I see which is still left, is that gganimate seems to switch the aesthetic "between" frames - thus the bar increases with a small delay when compared with the dot touching the wall. I guess this can be tolerable. Maybe there are ways to change this by using different transition objects. I don't know.

  1. suppressMessages({
  2. library(ggplot2)
  3. library(dplyr)
  4. library(tidyr)
  5. library(gganimate)
  6. })
  7. ## using your pos_data frame and impulse vector
  8. ## I have changed the order of cume_impulse, because it somehow made more sense
  9. cume_impulse &lt;- c(1.6e-24, 2.4e-24, 3.2e-24, 8.0e-24)
  10. ## time steps need to be character, for the later merge
  11. time_steps &lt;- as.character(c(1.132, 6.136, 11.140, 16.144))
  12. bar_data &lt;- data.frame(time_steps, impulse, cume_impulse)
  13. df &lt;-
  14. pos_data %&gt;%
  15. ## you need time steps as character for a correct merging - probably a floating point issue
  16. mutate(time_steps = as.character(time_steps)) %&gt;%
  17. left_join(bar_data, by = "time_steps") %&gt;%
  18. bind_rows(., .) %&gt;%
  19. ## adding a facetting variable
  20. mutate(
  21. time_steps = as.numeric(time_steps),
  22. panel = rep(c("bar", "dot"), each = nrow(pos_data)),
  23. ## removing x and y for the bar panel
  24. across(c(x_pos, y_pos), ~ifelse(panel == "bar", NA, .x))
  25. ) %&gt;%
  26. ## giving values to all time steps
  27. fill(cume_impulse)
  28. df$cume_impulse[is.na(df$cume_impulse)] &lt;- 0
  29. ## removing y for dot panel
  30. df$cume_impulse[df$panel == "dot"] &lt;- NA
  31. ## I am reducing the data to a mere fraction, because you get the same plot with way
  32. ## less computation (also, otherwise my computer is crashing).
  33. ## I do this after merging etc so not to accidentally remove the bar values
  34. ## I'm deliberately selecting 100 unique time steps for the animation (so that frame number is equivalent to time stamps)
  35. set.seed(1)
  36. times &lt;- sort(sample(unique(df$time_steps), 100))
  37. df_frac &lt;- df[df$time_steps %in% times, ]
  38. ## dummy data for facet range
  39. bar_range &lt;- data.frame(x= .5, cume_impulse = range(df$cume_impulse, na.rm = T), panel = "bar")
  40. ## separate out the aesthetics to the respective layers
  41. p &lt;-
  42. ggplot(df_frac) +
  43. geom_blank(data = bar_range, aes(x = .5, y = cume_impulse)) +
  44. ## need position = "identity"
  45. geom_col(aes(x = .5, y = cume_impulse), na.rm = T, position = "identity") +
  46. geom_point(aes(x_pos, y_pos), na.rm = T) +
  47. facet_wrap(~panel, scales = "free_y") +
  48. scale_x_continuous(expand = c(0,0))
  49. p_anim &lt;- p + transition_time(time_steps)
  50. ## this is not random = there are 100 unique time steps, and now 100 frames
  51. animate(p_anim, fps = 20, duration = 5)

How to animate a bar plot in R that represents one variable increasing over time, synchronised with a second animation?

英文:

You could make your life (and your code How to animate a bar plot in R that represents one variable increasing over time, synchronised with a second animation? a bit easier by creating those two plots as one single plot - using facets. This requires just a bit of data hacking, so that you create two separate panels.

The biggest challenge is actually to generate two different "ease_aes" for geom_bar and geom_point. As both geom layers are using the same aesthetic (x and y), we cannot simply define ease_aes for one of those aesthetics. I don't think one can define two different ease_aes for different geoms without a little hack. You have already quite a lot of intermediate time steps - I am merging this data with the bar code data and the removing the x/y according to the facet panel. The merge only properly works when your time values are character, probably because of floating point issues. Because my computer is not great and it is extremely computationally intense to compute 17000 frames, I will reduce the data drastically.

The only issue that I see which is still left, is that gganimate seems to switch the aesthetic "between" frames - thus the bar increases with a small delay when compared with the dot touching the wall. I guess this can be tolerable. Maybe there are ways to change this by using different transition objects. I don't know.

  1. suppressMessages({
  2. library(ggplot2)
  3. library(dplyr)
  4. library(tidyr)
  5. library(gganimate)
  6. })
  7. ## using your pos_data frame and impulse vector
  8. ## I have changed the order of cume_impulse, because it somehow made more sense
  9. cume_impulse &lt;- c(1.6e-24, 2.4e-24, 3.2e-24, 8.0e-24)
  10. ## time steps need to be character, for the later merge
  11. time_steps &lt;- as.character(c(1.132, 6.136, 11.140, 16.144))
  12. bar_data &lt;- data.frame(time_steps, impulse, cume_impulse)
  13. df &lt;-
  14. pos_data %&gt;%
  15. ##&#160;you need time steps as character for a correct merging - probably a floating point issue
  16. mutate(time_steps = as.character(time_steps)) %&gt;%
  17. left_join(bar_data, by = &quot;time_steps&quot;) %&gt;%
  18. bind_rows(., .) %&gt;%
  19. ## adding a facetting variable
  20. mutate(
  21. time_steps = as.numeric(time_steps),
  22. panel = rep(c(&quot;bar&quot;, &quot;dot&quot;), each = nrow(pos_data)),
  23. ## removing x and y for the bar panel
  24. across(c(x_pos, y_pos), ~ifelse(panel == &quot;bar&quot;, NA, .x))
  25. ) %&gt;%
  26. ## giving values to all time steps
  27. fill(cume_impulse)
  28. df$cume_impulse[is.na(df$cume_impulse)] &lt;- 0
  29. ## removing y for dot panel
  30. df$cume_impulse[df$panel == &quot;dot&quot;] &lt;- NA
  31. ## I am reducing the data to a mere fraction, because you get the same plot with way
  32. ##&#160;less computation (also, otherwise my computer is crashing).
  33. ## I do this after merging etc so not to accidentally remove the bar values
  34. ## I&#39;m deliberately selecting 100 unique time steps for the animation (so that frame number is equivalent to time stamps)
  35. set.seed(1)
  36. times &lt;- sort(sample(unique(df$time_steps), 100))
  37. df_frac &lt;- df[df$time_steps %in% times, ]
  38. ## dummy data for facet range
  39. bar_range &lt;- data.frame(x= .5, cume_impulse = range(df$cume_impulse, na.rm = T), panel = &quot;bar&quot;)
  40. ## separate out the aesthetics to the respective layers
  41. p &lt;-
  42. ggplot(df_frac) +
  43. geom_blank(data = bar_range, aes(x = .5, y = cume_impulse)) +
  44. ## need position = &quot;identity&quot;
  45. geom_col(aes(x = .5, y = cume_impulse), na.rm = T, position = &quot;identity&quot;) +
  46. geom_point(aes(x_pos, y_pos), na.rm = T) +
  47. facet_wrap(~panel, scales = &quot;free_y&quot;) +
  48. scale_x_continuous(expand = c(0,0))
  49. p_anim &lt;- p + transition_time(time_steps)
  50. ## this is not random = there are 100 unique time steps, and now 100 frames
  51. animate(p_anim, fps = 20, duration = 5)

How to animate a bar plot in R that represents one variable increasing over time, synchronised with a second animation?<!-- -->

huangapple
  • 本文由 发表于 2023年3月1日 14:18:16
  • 转载请务必保留本文链接:https://go.coder-hub.com/75600163.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定