noaa <- readLines('http://www.ssd.noaa.gov/PS/TROP/2018/adt/archive.html')
library(stringr)
library(readr)
pattern = 'http[^\"]+text[^\"]+\\.txt'
urls <- str_extract(noaa,pattern)
urls <- urls[!is.na(urls)]
header1 <- c("date", "time", "intensity_ci", "intensity_mslp", "intensity_vmax","tno_fnl","tno_adj","tno_ini","constrnt","wkng","rpd","temp_ctr","temp_mean","scene","est_rmw","mw","lat","lon","mthd","sat","vza","comments")
widths1 <- c(9,7,5,7,6,5,4,4,10,5,5,8,7,8,7,6,8,8,6,9,5,NA)
header2 = header <- c("date", "time", "intensity_ci", "intensity_mslp", "intensity_mslpplat", "intensity_vmax","tno_fnl","tno_adj","tno_ini","constrnt","wkng","rpd","temp_ctr","temp_mean","scene","est_rmw","mw","lat","lon","mthd","sat","vza","comments")
widths2 <- c(9,7,5,7,8,6,5,4,4,10,5,5,8,7,8,7,6,8,8,6,9,5,NA)
first <- T
q <- 1
#REMOVE
for(url in urls) {
  print(q)
  q <- q + 1
  lines <- readLines(url)
  name <- strsplit(substr(url,52,10000), "-")[[1]][1]
  # check if "BiasAdj" in line 4 -- if so, need an extra column!
  if(length(grep("BiasAdj",lines))) {
    widths <- widths2
    h <- header2
  } else {
    widths <- widths1
    h <- header1
  }
  
  if(url=="http://www.ssd.noaa.gov/PS/TROP/DATA/2018/adt/text/17P-list.txt") {
    # Special rule for this file due to incorrect negative MSLP values offsetting columns
    for(i in 5:55) {
      lines[i] = paste0(substr(lines[i],1,22), substr(lines[i],24,nchar(lines[i])), sep='')
    }
  }
  lines <- lines[5:(length(lines)-3)]
  
  data <- read_fwf(paste0(lines,collapse="\n"), fwf_widths(widths, col_names=h), na="N/A")
  data$est_rmw <- as.character(data$est_rmw)
  data$typhoon_name <- name
  if(first) {
    first = F
    typhoon <- data
  } else {
    typhoon <- bind_rows(typhoon, data)
  }
}
write.csv(typhoon,"../../Data/shipping/typhoon.csv", row.names=F)
library(plotly)
library(dplyr)
# load data
df <- read.csv("../../Data/Shipping/shipping_dataset.csv", stringsAsFactors = F, na="NA")
df_ships <- read.csv("../../Data/Shipping/index.csv", stringsAsFactors = F, na="NA")
df_ports <- read.csv("../../Data/Shipping/port_dataset.csv", stringsAsFactors = F, na="NA")
typhoon <- read.csv("../../Data/Shipping/typhoon.csv", stringsAsFactors = F)
# rename lat and lon to avoid conflicts
df_ports <- df_ports %>% rename(port_lat=lat, port_lon=lon)
# Join in ships and ports
df <- left_join(df, df_ships)
Joining, by = "imo"
df <- df %>% filter(!imo == 9528029 | reported_destination == "AU ADL")
df <- left_join(df,df_ports, by=c("left_port"="port"))
df <- df %>% rename(left_lat=port_lat, left_lon=port_lon)
df <- left_join(df,df_ports, by=c("arrive_port"="port"))
df <- df %>% rename(arrive_lat=port_lat, arrive_lon=port_lon)
# fix dates
df$last_update = as.POSIXct(df$last_update, tz="UTC",origin="1970-01-01")
df$left_time = as.POSIXct(df$left_time, tz="UTC",origin="1970-01-01")
df$arrive_time = as.POSIXct(df$arrive_time, tz="UTC",origin="1970-01-01")
# Fix typhoon dates
typhoon$date <- as.POSIXct(paste(typhoon$date,typhoon$time), format="%Y%b%d %H%M%S", tz="UTC", origin="1970-01-01")
# Fix typhoon longitude -- they have lon * -1 in the data
typhoon$lon <- typhoon$lon * -1
typhoon_all <- typhoon
# filter to dates in sample
typhoon <- typhoon %>% filter(date > as.POSIXct("2018-08-30", tz="UTC"))
df <- df %>% filter(last_update > as.POSIXct("2018-08-30", tz="UTC"))
geo <- list(
  showland = TRUE,
  showlakes = TRUE,
  showcountries = TRUE,
  showocean = TRUE,
  countrywidth = 0.5,
  landcolor = toRGB("grey90"),
  lakecolor = toRGB("aliceblue"),
  oceancolor = toRGB("aliceblue"),
  projection = list(
    type = 'orthographic',  # detailed at https://plot.ly/r/reference/#layout-geo-projection
    rotation = list(
      lon = 100,
      lat = 1,
      roll = 0
    )
  ),
  lonaxis = list(
    showgrid = TRUE,
    gridcolor = toRGB("gray40"),
    gridwidth = 0.5
  ),
  lataxis = list(
    showgrid = TRUE,
    gridcolor = toRGB("gray40"),
    gridwidth = 0.5
  )
)
library(RColorBrewer)
# plot with boats, ports, and typhoons
palette = brewer.pal(8, "Dark2")[c(1,8,3,2)]
p <- plot_geo(colors=palette) %>%
  add_markers(data=df_ports, x = ~port_lon, y = ~port_lat, color = "Port") %>%
  add_markers(data=df[df$run == 1,], x = ~lon, y = ~lat, color = ~ship_type,
              text=~paste('Ship name',shipname)) %>%
  add_markers(data=typhoon[typhoon$date > as.POSIXct("2018-08-31 00:00:00", tz="UTC") & typhoon$date < as.POSIXct("2018-09-01 00:00:00", tz="UTC"),], x = ~lon, y = ~lat, color="TYPHOON", colors=toRGB("red"), text=~paste("Name", typhoon_name)) %>%
   layout(
    showlegend = TRUE, geo = geo,
    title = 'Singaporean owned container and tanker ships, August 31, 2018'
  )
p
# plot with boats, ports, and typhoons
palette = brewer.pal(8, "Dark2")[c(1,3,2)]
p <- plot_geo(colors=palette) %>%
  add_markers(data=df[df$run == 14,], x = ~lon, y = ~lat, color = ~ship_type,
              text=~paste('Ship name',shipname)) %>%
  add_markers(data=typhoon[typhoon$typhoon_name == "JEBI_Y",], x = ~lon, y = ~lat, color="Typhoon Jebi", text=~paste("Name", typhoon_name, "</br>Time: ", date)) %>%
   layout(
    showlegend = TRUE, geo = geo,
    title = 'Singaporean container/tanker ships, September 4, 2018, evening'
  )
p
world1 <- sf::st_as_sf(map('world', plot = FALSE, fill = TRUE))

p <- ggplot(data = world1) + 
  geom_sf() + 
  geom_point(data = df, aes(x = lon, y = lat, frame=run))
ggplotly(p) %>%
   animation_opts(
    1500, easing = "elastic", redraw = FALSE
  )
# distance between two geo coordinates
library(geosphere)

# Linear interpolation for matching hurricane locations
linear_interp <- function(from, to, mid, d_from, d_to) {
  d_from * (mid - from) / (to - from) + d_to * (to - mid) / (to - from)
}

# for line in df
# line_last_update, line$lat, line_lon
# typhoon %>% group_by(typhoon_name) %>%

df$date_only <- as.Date(df$last_update, "UTC")
typhoon$ty_date_only <- as.Date(typhoon$date, "UTC")
typhoon <- typhoon %>% rename(ty_lat=lat, ty_lon=lon, ty_date=date)

first <- TRUE
for(d in unique(df$date_only)) {
  df2 <- df[df$date_only == d,]
  ty2 <- typhoon[typhoon$ty_date_only == d,]
  dfm <- as.matrix(df2[,c("lon","lat")])
  tym <- as.matrix(ty2[,c("ty_lon","ty_lat")])
  mat <- distm(dfm, tym, fun = distVincentyEllipsoid)
  df2 <- cbind(df2, ty2[max.col(-mat),])
  if(first) {
    df3 <- df2
    first <- FALSE
  } else {
    df3 <- bind_rows(df3, df2)
  }
}
write.csv(df3,"../../Data/shipping/combined.csv", row.names=F)
summary(fit)

Call:
glm(formula = delayed ~ (typhoon_100 + typhoon_500 + typhoon_1000):typhoon_intensity, 
    family = binomial, data = df3)

Deviance Residuals: 
    Min       1Q   Median       3Q      Max  
-0.5770  -0.4648  -0.4648  -0.4648   2.2974  

Coefficients: (1 not defined because of singularities)
                                         Estimate Std. Error  z value Pr(>|z|)    
(Intercept)                               -2.1709     0.0144 -150.788   <2e-16 ***
typhoon_100:typhoon_intensity(0,34]       -0.2838     0.6195   -0.458   0.6469    
typhoon_100:typhoon_intensity(34,53]      -0.2609     0.4545   -0.574   0.5660    
typhoon_100:typhoon_intensity(53,92.4]    -7.7742    72.4631   -0.107   0.9146    
typhoon_100:typhoon_intensity(92.4,150]        NA         NA       NA       NA    
typhoon_500:typhoon_intensity(0,34]       -0.1955     0.1779   -1.099   0.2717    
typhoon_500:typhoon_intensity(34,53]       0.3250     0.1949    1.668   0.0953 .  
typhoon_500:typhoon_intensity(53,92.4]     0.2382     0.2517    0.946   0.3440    
typhoon_500:typhoon_intensity(92.4,150]   -0.5626     0.7576   -0.743   0.4577    
typhoon_1000:typhoon_intensity(0,34]       0.1935     0.1001    1.932   0.0533 .  
typhoon_1000:typhoon_intensity(34,53]      0.1373     0.1468    0.935   0.3496    
typhoon_1000:typhoon_intensity(53,92.4]    0.1409     0.1610    0.875   0.3814    
typhoon_1000:typhoon_intensity(92.4,150]   0.1686     0.1889    0.892   0.3722    
---
Signif. codes:  0 ‘***’ 0.001 ‘**’ 0.01 ‘*’ 0.05 ‘.’ 0.1 ‘ ’ 1

(Dispersion parameter for binomial family taken to be 1)

    Null deviance: 37178  on 55846  degrees of freedom
Residual deviance: 37157  on 55835  degrees of freedom
  (7204 observations deleted due to missingness)
AIC: 37181

Number of Fisher Scoring iterations: 8

OLD BELOW

library(readr)
library(dplyr)
library(ggplot2)

url_csv <- 'https://raw.githubusercontent.com/d4tagirl/R-Ladies-growth-maps/master/rladies.csv'
rladies <- read_csv(url(url_csv)) %>% 
  select(-1)
rladies2 <- rladies
rladies$frame <- 1
rladies2$frame <- 2
rladies2$lat <- rladies2$lat + 5
rladies3 <- rbind(rladies,rladies2)
rladies2$frame <- 3
rladies2$lat <- rladies2$lat + 5
rladies3 <- rbind(rladies3,rladies2)
rladies2$frame <- 4
rladies2$lat <- rladies2$lat + 5
rladies3 <- rbind(rladies3,rladies2)
rladies2$frame <- 5
rladies2$lat <- rladies2$lat + 5
rladies3 <- rbind(rladies3,rladies2)

library(plotly)

g <- list(resolution = 50,
          showland = TRUE,
          showlakes = TRUE,
          landcolor = toRGB("grey80"),
          countrycolor = toRGB("grey80"),
          lakecolor = toRGB("white"),
          projection = list(type = "equirectangular"),
          coastlinewidth = 2,
          lataxis = list(
            range = c(20, 60),
            showgrid = TRUE,
            tickmode = "linear",
            dtick = 10
          ),
          lonaxis = list(
            range = c(-100, 20),
            showgrid = TRUE,
            tickmode = "linear",
            dtick = 20
          ))

plot_geo() %>%
  add_markers(data = rladies3,
              x = ~lon,
              y = ~lat,
              frame = ~frame,
              color=I("purple"),
              hoverinfo = "text",
              text = ~paste('city: ', location,
                            '<br /> created: ', created_at,
                            '<br /> followers: ', followers)) %>%
  layout(geo = g) %>%
  animation_opts(
    1000, easing = "elastic", redraw = FALSE
  )

maps, maptools, sf, ggplot @v3, rgeos

world1 <- sf::st_as_sf(map('world', plot = FALSE, fill = TRUE))

p <- ggplot(data = world1) + 
  geom_sf() + 
  geom_point(data = rladies3, aes(x = lon, y = lat, frame=frame))
ggplotly(p) %>%
   animation_opts(
    1500, easing = "elastic", redraw = FALSE
  )
# ANIMATION: https://github.com/plotly/plotly.js/blob/master/src/plots/animation_attributes.js
---
title: "R Notebook"
output: html_notebook
---

```{r typhoons, eval=F}
noaa <- readLines('http://www.ssd.noaa.gov/PS/TROP/2018/adt/archive.html')
library(stringr)
library(readr)
pattern = 'http[^\"]+text[^\"]+\\.txt'
urls <- str_extract(noaa,pattern)
urls <- urls[!is.na(urls)]
header1 <- c("date", "time", "intensity_ci", "intensity_mslp", "intensity_vmax","tno_fnl","tno_adj","tno_ini","constrnt","wkng","rpd","temp_ctr","temp_mean","scene","est_rmw","mw","lat","lon","mthd","sat","vza","comments")
widths1 <- c(9,7,5,7,6,5,4,4,10,5,5,8,7,8,7,6,8,8,6,9,5,NA)
header2 = header <- c("date", "time", "intensity_ci", "intensity_mslp", "intensity_mslpplat", "intensity_vmax","tno_fnl","tno_adj","tno_ini","constrnt","wkng","rpd","temp_ctr","temp_mean","scene","est_rmw","mw","lat","lon","mthd","sat","vza","comments")
widths2 <- c(9,7,5,7,8,6,5,4,4,10,5,5,8,7,8,7,6,8,8,6,9,5,NA)
first <- T
q <- 1
#REMOVE
for(url in urls) {
  print(q)
  q <- q + 1
  lines <- readLines(url)
  name <- strsplit(substr(url,52,10000), "-")[[1]][1]
  # check if "BiasAdj" in line 4 -- if so, need an extra column!
  if(length(grep("BiasAdj",lines))) {
    widths <- widths2
    h <- header2
  } else {
    widths <- widths1
    h <- header1
  }
  
  if(url=="http://www.ssd.noaa.gov/PS/TROP/DATA/2018/adt/text/17P-list.txt") {
    # Special rule for this file due to incorrect negative MSLP values offsetting columns
    for(i in 5:55) {
      lines[i] = paste0(substr(lines[i],1,22), substr(lines[i],24,nchar(lines[i])), sep='')
    }
  }
  lines <- lines[5:(length(lines)-3)]
  
  data <- read_fwf(paste0(lines,collapse="\n"), fwf_widths(widths, col_names=h), na="N/A")
  data$est_rmw <- as.character(data$est_rmw)
  data$typhoon_name <- name
  if(first) {
    first = F
    typhoon <- data
  } else {
    typhoon <- bind_rows(typhoon, data)
  }
}
write.csv(typhoon,"../../Data/shipping/typhoon.csv", row.names=F)
```

```{r, fig.height=6, fig.width=8}
library(plotly)
library(dplyr)

# load data
df <- read.csv("../../Data/Shipping/shipping_dataset.csv", stringsAsFactors = F, na="NA")
df_ships <- read.csv("../../Data/Shipping/index.csv", stringsAsFactors = F, na="NA")
df_ports <- read.csv("../../Data/Shipping/port_dataset.csv", stringsAsFactors = F, na="NA")
typhoon <- read.csv("../../Data/Shipping/typhoon.csv", stringsAsFactors = F)

# rename lat and lon to avoid conflicts
df_ports <- df_ports %>% rename(port_lat=lat, port_lon=lon)

# Join in ships and ports
df <- left_join(df, df_ships)
df <- df %>% filter(!imo == 9528029 | reported_destination == "AU ADL")
df <- left_join(df,df_ports, by=c("left_port"="port"))
df <- df %>% rename(left_lat=port_lat, left_lon=port_lon)
df <- left_join(df,df_ports, by=c("arrive_port"="port"))
df <- df %>% rename(arrive_lat=port_lat, arrive_lon=port_lon)

# fix dates
df$last_update = as.POSIXct(df$last_update, tz="UTC",origin="1970-01-01")
df$left_time = as.POSIXct(df$left_time, tz="UTC",origin="1970-01-01")
df$arrive_time = as.POSIXct(df$arrive_time, tz="UTC",origin="1970-01-01")

# Fix typhoon dates
typhoon$date <- as.POSIXct(paste(typhoon$date,typhoon$time), format="%Y%b%d %H%M%S", tz="UTC", origin="1970-01-01")
# Fix typhoon longitude -- they have lon * -1 in the data
typhoon$lon <- typhoon$lon * -1

typhoon_all <- typhoon
# filter to dates in sample
typhoon <- typhoon %>% filter(date > as.POSIXct("2018-08-30", tz="UTC"))



df <- df %>% filter(last_update > as.POSIXct("2018-08-30", tz="UTC"))

geo <- list(
  showland = TRUE,
  showlakes = TRUE,
  showcountries = TRUE,
  showocean = TRUE,
  countrywidth = 0.5,
  landcolor = toRGB("grey90"),
  lakecolor = toRGB("aliceblue"),
  oceancolor = toRGB("aliceblue"),
  projection = list(
    type = 'orthographic',  # detailed at https://plot.ly/r/reference/#layout-geo-projection
    rotation = list(
      lon = 100,
      lat = 1,
      roll = 0
    )
  ),
  lonaxis = list(
    showgrid = TRUE,
    gridcolor = toRGB("gray40"),
    gridwidth = 0.5
  ),
  lataxis = list(
    showgrid = TRUE,
    gridcolor = toRGB("gray40"),
    gridwidth = 0.5
  )
)
library(RColorBrewer)

# plot with boats, ports, and typhoons
palette = brewer.pal(8, "Dark2")[c(1,8,3,2)]
p <- plot_geo(colors=palette) %>%
  add_markers(data=df_ports, x = ~port_lon, y = ~port_lat, color = "Port") %>%
  add_markers(data=df[df$run == 1,], x = ~lon, y = ~lat, color = ~ship_type,
              text=~paste('Ship name',shipname)) %>%
  add_markers(data=typhoon[typhoon$date > as.POSIXct("2018-08-31 00:00:00", tz="UTC") & typhoon$date < as.POSIXct("2018-09-01 00:00:00", tz="UTC"),], x = ~lon, y = ~lat, color="TYPHOON", colors=toRGB("red"), text=~paste("Name", typhoon_name)) %>%
   layout(
    showlegend = TRUE, geo = geo,
    title = 'Singaporean owned container and tanker ships, August 31, 2018'
  )
p
```

```{r, fig.height=6, fig.width=8}
# plot with boats and typhoons
palette = brewer.pal(8, "Dark2")[c(1,3,2)]
p <- plot_geo(colors=palette) %>%
  add_markers(data=df[df$run == 14,], x = ~lon, y = ~lat, color = ~ship_type,
              text=~paste('Ship name',shipname)) %>%
  add_markers(data=typhoon[typhoon$typhoon_name == "JEBI_Y",], x = ~lon, y = ~lat, color="Typhoon Jebi", text=~paste("Name", typhoon_name, "</br>Time: ", date)) %>%
   layout(
    showlegend = TRUE, geo = geo,
    title = 'Singaporean container/tanker ships, September 4, 2018, evening'
  )
p
```

```{r animated_ships}
world1 <- sf::st_as_sf(map('world', plot = FALSE, fill = TRUE))

p <- ggplot(data = world1) + 
  geom_sf() + 
  geom_point(data = df, aes(x = lon, y = lat, frame=run))
ggplotly(p) %>%
   animation_opts(
    1500, easing = "elastic", redraw = FALSE
  )
```

```{r, eval=F}
# distance between two geo coordinates
library(geosphere)

# Linear interpolation for matching hurricane locations
linear_interp <- function(from, to, mid, d_from, d_to) {
  d_from * (mid - from) / (to - from) + d_to * (to - mid) / (to - from)
}

# for line in df
# line_last_update, line$lat, line_lon
# typhoon %>% group_by(typhoon_name) %>%

df$date_only <- as.Date(df$last_update, "UTC")
typhoon$ty_date_only <- as.Date(typhoon$date, "UTC")
typhoon <- typhoon %>% rename(ty_lat=lat, ty_lon=lon, ty_date=date)

first <- TRUE
for(d in unique(df$date_only)) {
  df2 <- df[df$date_only == d,]
  ty2 <- typhoon[typhoon$ty_date_only == d,]
  dfm <- as.matrix(df2[,c("lon","lat")])
  tym <- as.matrix(ty2[,c("ty_lon","ty_lat")])
  mat <- distm(dfm, tym, fun = distVincentyEllipsoid)
  df2 <- cbind(df2, ty2[max.col(-mat),])
  if(first) {
    df3 <- df2
    first <- FALSE
  } else {
    df3 <- bind_rows(df3, df2)
  }
}
write.csv(df3,"../../Data/shipping/combined.csv", row.names=F)

```

```{r}
# distance between two geo coordinates
library(geosphere)

df3 <- read.csv("../../Data/Shipping/combined.csv", stringsAsFactors = F)

df3$dist_toport <- distVincentyEllipsoid(as.matrix(df3[,c("lon","lat")]), as.matrix(df3[,c("arrive_lon","arrive_lat")]))
df3$dist_fromport <- distVincentyEllipsoid(as.matrix(df3[,c("lon","lat")]), as.matrix(df3[,c("left_lon","left_lat")]))
df3$dist_typhoon <- distVincentyEllipsoid(as.matrix(df3[,c("lon","lat")]), as.matrix(df3[,c("ty_lon","ty_lat")]))

df3 <- df3 %>%
  arrange(imo, last_update) %>%
  group_by(imo) %>%
  mutate(delayed=ifelse(arrive_time>lag(arrive_time) & arrive_time > last_update,1,0)) %>%
  ungroup()



df3$typhoon_100 = ifelse(df3$dist_typhoon<100000,1,0)
df3$typhoon_500 = ifelse(df3$dist_typhoon<500000,1,0)
df3$typhoon_1000 = ifelse(df3$dist_typhoon<1000000,1,0)

df3$typhoon_intensity = cut(df3$intensity_vmax, c(92.4,150))

fit <- glm(delayed ~ typhoon_100 + typhoon_500 + typhoon_1000, data=df3, family=binomial)
summary(fit)
fit <- glm(delayed ~ (typhoon_100 + typhoon_500 + typhoon_1000) : typhoon_intensity, data=df3, family=binomial)
summary(fit)

#df$dist_toport <- distVincentyEllipsoid(as.matrix(df[,c("lon","lat")]), as.matrix(df[,c("arrive_lon","arrive_lat")]))
#df$dist_toport <- distVincentyEllipsoid(as.matrix(df[,c("lon","lat")]), as.matrix(df[,c("left_lon","left_lat")]))
# merging typhoons and ships...

#unique(as.Date(df$last_update, "UTC"))


```






OLD BELOW

```{r, fig.height=6, fig.width=8}
library(readr)
library(dplyr)
library(ggplot2)

url_csv <- 'https://raw.githubusercontent.com/d4tagirl/R-Ladies-growth-maps/master/rladies.csv'
rladies <- read_csv(url(url_csv)) %>% 
  select(-1)
rladies2 <- rladies
rladies$frame <- 1
rladies2$frame <- 2
rladies2$lat <- rladies2$lat + 5
rladies3 <- rbind(rladies,rladies2)
rladies2$frame <- 3
rladies2$lat <- rladies2$lat + 5
rladies3 <- rbind(rladies3,rladies2)
rladies2$frame <- 4
rladies2$lat <- rladies2$lat + 5
rladies3 <- rbind(rladies3,rladies2)
rladies2$frame <- 5
rladies2$lat <- rladies2$lat + 5
rladies3 <- rbind(rladies3,rladies2)

library(plotly)

g <- list(resolution = 50,
          showland = TRUE,
          showlakes = TRUE,
          landcolor = toRGB("grey80"),
          countrycolor = toRGB("grey80"),
          lakecolor = toRGB("white"),
          projection = list(type = "equirectangular"),
          coastlinewidth = 2,
          lataxis = list(
            range = c(20, 60),
            showgrid = TRUE,
            tickmode = "linear",
            dtick = 10
          ),
          lonaxis = list(
            range = c(-100, 20),
            showgrid = TRUE,
            tickmode = "linear",
            dtick = 20
          ))

plot_geo() %>%
  add_markers(data = rladies3,
              x = ~lon,
              y = ~lat,
              frame = ~frame,
              color=I("purple"),
              hoverinfo = "text",
              text = ~paste('city: ', location,
                            '<br /> created: ', created_at,
                            '<br /> followers: ', followers)) %>%
  layout(geo = g) %>%
  animation_opts(
    1000, easing = "elastic", redraw = FALSE
  )
```

maps, maptools, sf, ggplot @v3, rgeos

```{r,  fig.height=6, fig.width=8}
world1 <- sf::st_as_sf(map('world', plot = FALSE, fill = TRUE))

p <- ggplot(data = world1) + 
  geom_sf() + 
  geom_point(data = rladies3, aes(x = lon, y = lat, frame=frame))
ggplotly(p) %>%
   animation_opts(
    1500, easing = "elastic", redraw = FALSE
  )
# ANIMATION: https://github.com/plotly/plotly.js/blob/master/src/plots/animation_attributes.js
```
