razvizThe purpose of this vignette is to demonstrate how to use the razviz package to produce a pdf report of longitudinal profile graphs for a set of HEC-RAS hydraulic model scenarios. A longitudinal Profile Graph displays water surface elevations on the y-axis by longitudinal river position (aka “river miles”) on the x-axis. For this example we will use a RAS model of the Upper Mississippi River. The razviz::longitudinal_profile_plot function is tailored to evaluating RAS model results on a large river system. A comprehensive longitudinal profile graph for a large river system must include a wide range of ancillary information to allow hydraulic engineers to evaluate model results during calibration and floodplain managers to evaluate the impact of model scenarios on built infrastructure. This longitudinal profile plot incorporates the following ancillary data:
Symbolization of these ancillary graph elements requires a dataset for each. The razviz user is required to develop these datasets first, following the structure of the package datasets provided.
HEC-RAS models are often used to model long river reaches. Effective longitudinal profile graphs must have sufficient x and y-axis resolution to see the differences between model scenarios and relationship to ancillary data (i.e., gage stage, high water marks, levees, bridges). This requires that long model reaches must be graphed using multiple graphs. However, each graph should not be too “zoomed-in” or medium and large scale trends are obscured. The purpose of the razviz:long_plot_pages function is to define the graph dimensions for multiple page longitudinal profile graph reports. Users will need to adjust the miles_per_plot parameter to find the right “scale” for your reach.
Begin by loading the needed packages.
library(tidyverse)
#> -- Attaching packages ----------------------------------------------------------------------------------------------------- tidyverse 1.3.0 --
#> v ggplot2 3.3.1     v purrr   0.3.4
#> v tibble  3.0.1     v dplyr   1.0.0
#> v tidyr   1.1.0     v stringr 1.4.0
#> v readr   1.3.1     v forcats 0.5.0
#> -- Conflicts -------------------------------------------------------------------------------------------------------- tidyverse_conflicts() --
#> x dplyr::filter() masks stats::filter()
#> x dplyr::lag()    masks stats::lag()
library(devtools)
#> Loading required package: usethisNext, we’ll install the razviz package.
Finally, we’ll load the razviz package.
In this step we’ll define the output folder.
In this step we will import the hydraulic parameter that needs to be plotted as a longitudinal profile. Commonly the parameter of interest will be water surface or stage profiles. However, as noted in these instructions other hydraulic parameters such as water velocity or lateral hydraulic connectivity can also be plotted longitudinally.
.csv format included in the package.Event field to identify each event as different series and plot separately in the longitudinal profile graph.#If importing multiple water surface profiles files, set the file path where all the profile data is being stored.
path <- system.file("extdata/longitudinal_profiles", 
                    package = "razviz")
#Identify the pattern in the file name for each profile that needs to be imported. 
pattern <- "Freq"
#Combines all the files that have been identified. 
#All columns must have the same column name to be combined 
hydro_model <- razviz::combine_files(path = path, pattern = pattern)
#Re-name `Event` field for labeling 
#in the example .csv files the  column with the return periods was labeled 'Freq'. 
hydro_model$Event <- hydro_model$Freq
#Filter the Events (optional)
#In the external data provided, there are 23 different water surface profiles related to return periods
#Filtering allows us to only plot a select few of them
model_events <- c("2 Year", "100 Year", "500 Year", "100000 Year")
hydro_model_1 <- dplyr::filter(hydro_model, Event %in% model_events)
#Set Event as an ordered factor
#Ordered factors are categorical variables that the data is grouped by. 
#Factors are used to specify how to plot the data later in the code
hydro_model_1$Event <- factor(hydro_model_1$Event,
                              levels = model_events,
                              labels = model_events)
#set the variable to plot
hydro_model_1$hydro_parameter <- hydro_model_1$WS_ElevIf the water surface profiles are still in HEC-RAS 5.0.7, the following steps will help you create the necessary .csv files
When the data is taken from HEC-RAS without reformatting in excel, the river mile column sometimes contains descriptions of the physical structures located at those river miles (e.g. bridge, lateral structure). In order for to plot river mile as a number on the x-axis, those text descriptions need to be removed and the river mile column needs to be a ‘numeric’ variable in r. Also, the total flow (Q_Total) will also have text for structures.
#Add a column with the Event year BEOFRE importing and combining in R. 
#If importing multiple water surface profiles files, set the file path where all the profile data is being stored.
path <- system.file("extdata/longitudinal_profiles", 
                    package = "razviz")
#Identify the pattern in the file name for each profile that needs to be imported. 
pattern <- "Profile_"
#Combines all the files that have been identified. 
#All columns must have the same column name to be combined
Profile_Summary <- razviz::combine_files(path = path, pattern = pattern)
#Rename the columns coming our of HEC-RAS to avoid special characters
data.table::setnames(Profile_Summary, old = c("River","Reach","River Sta",
                                            "Profile", "Q Total",
                                            "Min Ch El","W.S. Elev",
                                            "Crit W.S.","E.G. Elev",
                                            "E.G. Slope","Vel Chnl",
                                            "Flow Area","Top Width",
                                            "Froude # Chl","Event") , 
                                    new = c("River","Reach","River_Sta",
                                            "Profile", "Q_Total",
                                            "Min_Ch_El","WS_Elev",
                                            "Crit_WS","EG_Elev", 
                                            "EG_Slope","Vel_Chnl",
                                            "Flow_Area","Top_Width", 
                                            "Froude_Chl","Event"), skip_absent = TRUE)
#remove rows that do not have a number in the Q_Total column
#will create a "NAs introduced by coecion warning" because characters which are not numbers are converted in NA.
profile_data <- Profile_Summary[!is.na(as.numeric(as.character(Profile_Summary$Q_Total))),]
#Reformat river mile column to contain only numbers by splitting the column into numbers and text using a space (' ') as the delimiter
rm <- as.data.frame(str_split_fixed(profile_data$River_Sta, c(" "),2))
profile_data$River_Sta <- rm$V1
#save the river mile column as a number
profile_data$River_Sta <- as.numeric(as.character(profile_data$River_Sta))
#add ordered factor - this will be used during plotting
model_events<- c(2001, 2014, 2019)
#optional filtering not needed for this example
#profile_data_ordered <- dplyr::filter(profile_data, Event %in% model_events)
## Set Event as an ordered factor
profile_data$Event <- factor(profile_data$Event,
                              levels = model_events,
                              labels = model_events)
hydro_model_1 <- profile_data
#set the variable to plot
hydro_model_1$hydro_parameter <- hydro_model_1$WS_ElevIf you would like to important another parameter, at a minimum your ‘.csv’ file must have the following columns
CURRENTLY NO EXAMPLE DATA HAS BEEN UPLOADED INTO RAZVIZ AS AN EXAMPLE
#Add a column with the Event year BEOFRE importing and combining in R. 
#If importing multiple water surface profiles files, set the file path where all the profile data is being stored.
path <- system.file("extdata/longitudinal_profiles", 
                    package = "razviz")
#Identify the pattern in the file name for each profile that needs to be imported. 
pattern <- "Hydro_Parameter"
#Combines all the files that have been identified. 
#All columns must have the same column name to be combined
hydro_parameter_summary <- razviz::combine_files(path = path, pattern = pattern)
#save the river mile column as a number
hydro_parameter_summary$River_Sta <- as.numeric(as.character(profile_data$River_Sta))
#add ordered factor - this will be used during plotting
model_events<- c(2001, 2014, 2019)
#optional filtering not needed for this example
#profile_data_ordered <- dplyr::filter(profile_data, Event %in% model_events)
## Set Event as an ordered factor
hydro_parameter_summary$Event <- factor(hydro_parameter_summar$Event,
                              levels = model_events,
                              labels = model_events)
hydro_model_1 <- hydro_parameter_summary 
#set the variable to plot
hydro_model_1$hydro_parameter <- hydro_model_1$WS_ElevIn this section, we’ll setup how the longitudinal profile will look and then run a report
In this step we’ll define the number of longitudinal profile pages needed for the report. The razviz::long_plot_pages function determines the x and y-axis extents of each graph. This is a required data set.
In this step we’ll assign the colors to each of the series in the graph. This is a required data set.
razviz to particular data series depends on the use of R’s “named vector”.grDevices::colors()).In this step we’ll assign the legend labels. This is a required data set.
In this step we’ll assign the plot labels. This is a required data set.
In this step we’ll call the longitudinal profile graph report for the entire reach of the river. This report will plot only the hydraulic parameter of interest.
filename <- "Longitudinal_Profile_Report_No_Background.pdf"
razviz::longitudinal_profile_report(hydro_model = hydro_model_1,
                                    long_plot_pgs = long_plot_pgs,
                                    graph_colors = graph_cols,
                                    legend_labels = legend_labels,
                                    plot_labels = plot_labels,        
                                    output_dir = output_dir, 
                                    filename = filename)
#> \newpage
#> \newpage
#> \newpage
#> \newpage
#> \newpage
#> \newpage
#> png 
#>   2##Additional Data We will now import a series of optional dataframes that can be added to your longitundinal profile plot.
In this step we’ll prepare the high water marks dataset.
.csv format included in the package.high_water_csv <- system.file("extdata/longitudinal_profiles",
                              "high_water_marks.csv",
                              package = "razviz")
high_water <- readr::read_csv(high_water_csv)
#> Parsed with column specification:
#> cols(
#>   highwater_mark_name = col_character(),
#>   river_mile = col_double(),
#>   gage_zero = col_double(),
#>   gage_datum = col_character(),
#>   conversion_factor_2_NAVD88 = col_double(),
#>   stage = col_logical(),
#>   elevation_original = col_double(),
#>   elevation_NAVD88 = col_double(),
#>   peak_date = col_character(),
#>   hw = col_character()
#> )
## Define high water events in YEARS and add event year column to high water marks dataframe.
## NOTE: Your high water years do not have to match your model_events, but high_water_events must be a list of years. If the model_event and high_water_year event are different, there will be additional items in the legend. 
## The high_water_events function searches the date column for the year of the event and adds a column with the matching event year.
high_water_years <- c("2008", "2013", "2014")
high_water_events_df <- razviz::high_water_events(high_water, high_water_years)
##If you are working with a single profile or events that are not classified by years (e.g.25% duration flows or similar), and INSTEAD want to add a column which specifies the event, use the following code.
#high_water_events <- c("25% Duration")
#high_water_events_df <- add_column(high_water, event = high_water_events)In this step we’ll prepare the gage dataset. * Import the gage locations table in .csv format included in the package. * Min & Max stage are used to draw gage boxes and should be iteratively adjusted to match the profile * Remove the double backslashes introduced by R during the import.
##NOTE: If you'd like to have the gage identified without gage boxes, please add the gage as a river feature.)
gage_csv <- system.file("extdata/longitudinal_profiles", 
                        "gage_locations.csv", 
                        package = "razviz")
gages <- readr::read_csv(gage_csv)
#> Parsed with column specification:
#> cols(
#>   name = col_character(),
#>   river_mile = col_double(),
#>   elevation = col_double(),
#>   stage = col_double(),
#>   min_stage = col_double(),
#>   max_stage = col_double(),
#>   datum = col_character()
#> )
## Remove double backslashes introduced by R import
gages$name <- gsub("\\n", "\n", gages$name, fixed=TRUE)In this step we’ll define the stage text labels that appear next to the gage boxes. These labels represent the stage in units feet for each gage.
In this step we’ll define the dimensions of the boxes used to draw the gage stage levels for all gages.
In this step we’ll prepare the levee dataset.
.csv format included in the package.levees_csv <- system.file("extdata/longitudinal_profiles",
                          "levees_authorized_existing.csv",
                          package = "razviz")
levees <- readr::read_csv(levees_csv)
#> Parsed with column specification:
#> cols(
#>   levee = col_character(),
#>   river_mile = col_double(),
#>   elevation_NAVD88 = col_double(),
#>   descending_bank = col_character(),
#>   elevation_type = col_character()
#> )In this step we’ll prepare the river features dataset.
.csv format included in the package.In this step we’ll prepare the bridges dataset. This dataset must include elevations both lowest_elevation and highest_elevation for all bridges listed.
.csv format included in the package.bridges_csv <- system.file("extdata/longitudinal_profiles",
                           "bridge_elevations.csv",
                           package = "razviz")
bridges <- readr::read_csv(bridges_csv)
#> Parsed with column specification:
#> cols(
#>   name = col_character(),
#>   river_mile = col_double(),
#>   lowest_elevation = col_double(),
#>   highest_elevation = col_double()
#> )In this step we’ll define the number of longitudinal profile pages needed for the report. The razviz::long_plot_pages function determines the x and y-axis extents of each graph. This is a required data set.
In this step we’ll assign the colors to each of the series in the graph. This is a required data set.
razviz to particular data series depends on the use of R’s “named vector”.grDevices::colors()).In this step we’ll assign the legend labels. This is a required data set.
In this step we’ll assign the plot labels. This is a required data set.
In this step we’ll call the longitudinal profile graph report for the entire reach of the river that is in the data that was originally uploaded.
In this step we’ll call the longitudinal profile graph report, just like the previous section; however, prior to plotting we will specify the section of the river that we’d like plotted. This can be useful if there is a particular section of the river where there is considerable more elevation change than the remainder of the river.