TABLE OF CONTENTS
Utilize the Table of Contents on the left to scroll through the
different sections and help navigate this report. You can click on the
title to go directly to the section.
If you’re viewing the PDF version, the Table of Contents stays at the
top of this report. To go directly to a section, copy and paste the
title into the search box. Use the following shortcut keys to help
navigate with ease:
Shortcut Keys:
- Hold
Ctrl
and press the Home
key to go to
the top of the page.
- Hold
Ctrl
and press the End
key to go to
the bottom of the page.
- Hold
Ctrl
and press the F
key to search
for specific text or strings of text.
- Press the
Esc
key to quickly exit the search box.
About This Report
This report is published on RPubs. KDF stands for Ken Does Fitness. Refer to
References at the bottom of this page for cited sources. The RMD file to
produce this report, and some of the figures from this report, may be
found in the Google Drive folder .
STRENGTH STANDARDS
Strength Standards provide valuable benchmarks for assessing an
individual’s performance in key strength exercises that are widely
practiced in fitness and strength training. These standards categorize
performance levels based on either amount of weight executed relative to
bodyweight, or repetitions executed, and are segmented by gender and
experience levels. By comparing an individual’s performance against
these established standards, trainers and athletes can identify
strengths and weaknesses in their training regimens, set realistic
goals, and measure progress in a structured manner.
Strength Standards Overview
Several strength standards are readily available online:
Finding accurate strength standards depends on the demographic,
exercise type, and required accuracy. Each has its pros and cons: some
cater to specific demographics, others focus on certain exercises, and
their credibility affects reliability. Usability and comprehensive data
for benchmarking are crucial, as are regular updates to reflect current
trends. Considering these factors helps trainers and athletes choose the
best standards for assessing and improving performance.
# Load necessary libraries
library(knitr)
library(kableExtra)
# Define the pros and cons as lists
pros <- c("Benchmarking: Strength standards provide clear benchmarks to measure progress and set goals.",
"Motivation: Seeing how your performance stacks up against established standards can be highly motivating.",
"Guidance: They offer guidance on what is considered strong or weak for different exercises, helping you tailor your training accordingly.",
"Comparison: They allow for meaningful comparisons between individuals, helping to identify strengths and weaknesses.",
"Structured Training: Strength standards can help in designing more structured and effective training programs.")
cons <- c("Generic: Strength standards can be too generic and may not account for individual differences in genetics, body composition, and training history.",
"Pressure: They can create undue pressure to meet certain benchmarks, potentially leading to injury or burnout.",
"Outdated Information: Some standards may be based on outdated information or not reflect recent advancements in sports science.",
"Not Comprehensive: They might not cover all exercises or account for different training goals, such as hypertrophy versus strength.",
"Misleading: For beginners, strength standards can be misleading if they don’t account for the starting point or rate of progress appropriate for different fitness levels.")
# Combine the pros and cons into a data frame
strength_standards_pnc <- data.frame(
Pros = pros,
Cons = cons,
stringsAsFactors = FALSE
)
# Print the table using kable and kableExtra
kable(strength_standards_pnc, format = "html", escape = FALSE, col.names = c("Pros", "Cons")) %>%
kable_styling(bootstrap_options = c("striped", "hover", "condensed", "responsive"), full_width = TRUE) %>%
column_spec(1, width = "50%") %>%
column_spec(2, width = "50%")
Pros
|
Cons
|
Benchmarking: Strength standards provide clear benchmarks to measure
progress and set goals.
|
Generic: Strength standards can be too generic and may not account for
individual differences in genetics, body composition, and training
history.
|
Motivation: Seeing how your performance stacks up against established
standards can be highly motivating.
|
Pressure: They can create undue pressure to meet certain benchmarks,
potentially leading to injury or burnout.
|
Guidance: They offer guidance on what is considered strong or weak for
different exercises, helping you tailor your training accordingly.
|
Outdated Information: Some standards may be based on outdated
information or not reflect recent advancements in sports science.
|
Comparison: They allow for meaningful comparisons between individuals,
helping to identify strengths and weaknesses.
|
Not Comprehensive: They might not cover all exercises or account for
different training goals, such as hypertrophy versus strength.
|
Structured Training: Strength standards can help in designing more
structured and effective training programs.
|
Misleading: For beginners, strength standards can be misleading if they
don’t account for the starting point or rate of progress appropriate for
different fitness levels.
|
I’ve compiled some of the standards into an Excel file. This allows for
comparison between personal performance measures and the specified
strength standards. I recommend consulting each source for confirmation
on the latest version, as the structure or numbers may have changed over
time.
# File Path Definition: The script defines a path to an Excel file containing "Strength Standards".
file_path_strength_standards <- "C://Users//kendr//OneDrive//Documents//Data Analytics//KenDoesFitness/Strength Standards Final.xlsm"
# Workbook Loading: Loads the Excel workbook into R using the openxlsx package, which provides more advanced control over Excel files than readxl.
strength_standards <- loadWorkbook(file_path_strength_standards)
Standards Categories
The standards compiled are separated into categories, as shown in the
following table, making it easy to determine which exercise to use as a
benchmark.
# Sheet Names Display: Retrieves and displays the names of sheets within the Excel workbook to help identify available data.
# Check to confirm which number of names to display
# names(strength_standards)
# Display specific elements from the strength_standards object with Indexing
names_subset_strength_standards <- names(strength_standards)[4:15]
names_subset_strength_standards_gsub = gsub("_"," ", names_subset_strength_standards)
# names_subset_strength_standards
# Calculate the number of names
num_names <- length(names_subset_strength_standards_gsub)
# Calculate the number of names per column
names_per_column <- ceiling(num_names / 2)
# Split the names into two columns
column_1 <- names_subset_strength_standards_gsub[1:min(names_per_column, num_names)]
column_2 <- names_subset_strength_standards_gsub[(min(names_per_column, num_names) + 1):num_names]
# Pad the shorter column with NA values if necessary
if(length(column_1) < length(column_2)) {
column_1 <- c(column_1, rep("", length(column_2) - length(column_1)))
} else if(length(column_2) < length(column_1)) {
column_2 <- c(column_2, rep("", length(column_1) - length(column_2)))
}
# Create a data frame with two columns
names_subset_strength_standards_gsub_df <- data.frame(Column_1 = column_1,
Column_2 = column_2)
# Calculate the width dynamically
col_width <- paste0(100 / ncol(names_subset_strength_standards_gsub_df), "%")
# Print the data frame using knitr::kable
knitr::kable(names_subset_strength_standards_gsub_df, format = "markdown", col.names = c("Categories", "Categories")) %>%
column_spec(1:2, width = col_width, extra_css = "text-align: left;")
Categories
|
Categories
|
Push Ups
|
Romanian Deadlift
|
Bench Press
|
T Bar Row
|
Incline Bench Press
|
Press
|
Pull Ups
|
Barbell Shoulder Press
|
Lat Pulldown
|
Dumbbell Shoulder Press
|
Deadlift
|
Military Press
|
Strength Standards List
Below is the list of Strength Standards compiled into the SS Excel
file.
This file references the source of each standard table. We’ll delve into
Mark Rippetoe’s Starting Strength Standards, as his and his colleagues’
methods and research are based on comprehensive and technical
information.
library(openxlsx)
library(readxl)
library(knitr)
library(kableExtra)
# Read Named Ranges: Reads a specific sheet named "Named_Ranges" that presumably contains metadata about other named ranges in the workbook. It keeps only the columns related to named ranges and their respective sheets.
named_ranges <- read_excel(file_path_strength_standards, sheet = "Named_Ranges")
sheet_names_filtered <- unique(named_ranges$`Category Sheet`)
named_ranges <- named_ranges[named_ranges$`Category Sheet` %in% sheet_names_filtered, c("Strength Standard", "Category Sheet")]
# Display the modified 'Strength Standard' column with spaces instead of underscores
strength_standards_list_display <- named_ranges %>%
mutate(`Strength Standard` = gsub("_", " ", `Strength Standard`)) %>%
mutate(`Category Name` = gsub("_", " ", `Category Sheet`)) %>%
select(-`Category Sheet`) # This excludes the 'Category Sheet' column
# View the data frame if needed (not during knitting). View function opens the raw_data dataframe in the RStudio data viewer for interactive exploration.
View(named_ranges)
View(strength_standards_list_display)
# Define the ratios and total width for column widths
total_width <- 100
ratio_strength_standards <- 2
ratio_category <- 1.25
total_ratio <- ratio_strength_standards + ratio_category
# Calculate the column widths
width_strength_standards <- (ratio_strength_standards / total_ratio) * total_width
width_category <- (ratio_category / total_ratio) * total_width
# Print the dataframe using kable for a nicely formatted markdown table.
kable(strength_standards_list_display, format = "html", col.names = c("Strength Standard", "Category")) %>%
kable_styling(full_width = TRUE) %>%
column_spec(1, width = paste0(width_strength_standards, "%")) %>%
column_spec(2, width = paste0(width_category, "%"))
Strength Standard
|
Category
|
Male Barbell Curl Std 1RM By Age Strengthlevel
|
Barbell Curl
|
Male Barbell Curl Std 1RM By Bodyweight Strengthlevel
|
Barbell Curl
|
Male Barbell Curl Std 1RM Strengthlevel
|
Barbell Curl
|
Male Barbell Shoulder Press Std 1RM By Age Strengthlevel
|
Barbell Shoulder Press
|
Male Barbell Shoulder Press Std 1RM By Bodyweight Strengthlevel
|
Barbell Shoulder Press
|
Male Barbell Shoulder Press Std 1RM Strengthlevel
|
Barbell Shoulder Press
|
Male Bench Press Std 1RM By Age Strengthlevel
|
Bench Press
|
Male Bench Press Std 1RM By Bodyweight Mark Rippetoe
|
Bench Press
|
Male Bench Press Std 1RM By Bodyweight Strengthlevel
|
Bench Press
|
Male Bench Press Std 1RM Strengthlevel
|
Bench Press
|
Male Deadlift Std 1RM By Age Strengthlevel
|
Deadlift
|
Male Deadlift Std 1RM By Bodyweight Mark Rippetoe
|
Deadlift
|
Male Deadlift Std 1RM By Bodyweight Strengthlevel
|
Deadlift
|
Male Deadlift Std 1RM Strengthlevel
|
Deadlift
|
Male Dumbbell Shoulder Press Std 1RM By Age Strengthlevel
|
Dumbbell Shoulder Press
|
Male Dumbbell Shoulder Press Std 1RM By Bodyweight Strengthlevel
|
Dumbbell Shoulder Press
|
Male Dumbbell Shoulder Press Std 1RM Strengthlevel
|
Dumbbell Shoulder Press
|
Male Incline Bench Press Std 1RM By Age Strengthlevel
|
Incline Bench Press
|
Male Incline Bench Press Std 1RM By Bodyweight Strengthlevel
|
Incline Bench Press
|
Male Incline Bench Press Std 1RM Strengthlevel
|
Incline Bench Press
|
Male Lat Pulldown Std 1RM By Age Strengthlevel
|
Lat Pulldown
|
Male Lat Pulldown Std 1RM By Bodyweight Strengthlevel
|
Lat Pulldown
|
Male Lat Pulldown Std 1RM Strengthlevel
|
Lat Pulldown
|
Male Military Press Std 1RM By Age Strengthlevel
|
Military Press
|
Male Military Press Std 1RM By Bodyweight Strengthlevel
|
Military Press
|
Male Military Press Std 1RM Strengthlevel
|
Military Press
|
Named Range List
|
Named Ranges
|
Male Press Std 1RM By Bodyweight Mark Rippetoe
|
Press
|
Male Pull Ups Std 1RM Strengthlevel
|
Pull Ups
|
Male Pull Ups Std 1RM Weight By Age Strengthlevel
|
Pull Ups
|
Male Pull Ups Std 1RM Weight By Bodyweight Strengthlevel
|
Pull Ups
|
Male Pull Ups Std Reps By Age Strengthlevel
|
Pull Ups
|
Male Pull Ups Std Reps By Bodyweight Strengthlevel
|
Pull Ups
|
Male Push Ups Std Reps By Age Strengthlevel
|
Push Ups
|
Male Push Ups Std Reps By Bodyweight Strengthlevel
|
Push Ups
|
Male Push Ups Std Reps Strengthlevel
|
Push Ups
|
Male Romanian Deadlift Std 1RM By Age Strengthlevel
|
Romanian Deadlift
|
Male Romanian Deadlift Std 1RM By Bodyweight Strengthlevel
|
Romanian Deadlift
|
Male Romanian Deadlift Std 1RM Strengthlevel
|
Romanian Deadlift
|
Male Squat Std 1RM By Age Strengthlevel
|
Squat
|
Male Squat Std 1RM By Bodyweight Mark Rippetoe
|
Squat
|
Male Squat Std 1RM By Bodyweight Strengthlevel
|
Squat
|
Male Squat Std 1RM Strengthlevel
|
Squat
|
Male T Bar Row Std 1RM By Age Strengthlevel
|
T Bar Row
|
Male T Bar Row Std 1RM By Bodyweight Strengthlevel
|
T Bar Row
|
Male T Bar Row Std 1RM Strengthlevel
|
T Bar Row
|
# Create lists to store data frames
named_ranges_list <- list()
named_ranges_sheets_list <- list()
# Loop through the named ranges
for (i in 1:nrow(named_ranges)) {
named_range_value <- named_ranges$`Strength Standard`[i]
named_range_sheet_value <- named_ranges$`Category Sheet`[i]
named_range_lower <- tolower(gsub(" ", "_", named_range_value))
named_range_sheet_lower <- tolower(gsub(" ", "_", named_range_sheet_value))
if (named_range_sheet_value == "Named_Ranges") {
next # Skip iteration if the sheet is "Named_Ranges" to avoid redundancy
}
# Read data from the named range
named_range_data <- tryCatch({
read.xlsx(file_path_strength_standards, namedRegion = named_range_value)
}, error = function(e) {
message(paste("Warning: Workbook has no such named region:", named_range_value))
NULL
})
if (!is.null(named_range_data) && is.data.frame(named_range_data)) {
# Update column names
colnames(named_range_data) <- gsub("\\.", " ", colnames(named_range_data))
# Store the data frame in the list
named_ranges_list[[named_range_lower]] <- named_range_data
# Assign the data frame to the global environment with the lower case name
assign(named_range_lower, named_range_data, envir = .GlobalEnv)
}
# Read data from the sheet
named_range_sheet_data <- tryCatch({
read.xlsx(file_path_strength_standards, sheet = named_range_sheet_value)
}, error = function(e) {
message(paste("Warning: Workbook has no such sheet:", named_range_sheet_value))
NULL
})
if (!is.null(named_range_sheet_data) && is.data.frame(named_range_sheet_data)) {
# Update column names
colnames(named_range_sheet_data) <- gsub("\\.", " ", colnames(named_range_sheet_data))
# Store the data frame in the list
named_ranges_sheets_list[[named_range_sheet_lower]] <- named_range_sheet_data
# Assign the data frame to the global environment with the lower case name
assign(named_range_sheet_lower, named_range_sheet_data, envir = .GlobalEnv)
}
}
# Display the structure of the lists
# str(named_ranges_list)
# str(named_ranges_sheets_list)
# Optional: View the contents of the lists in RStudio Viewer
# Instead of viewing the whole list, view individual data frames
# Uncomment the below lines to view specific named ranges or sheets
# View(named_ranges_list[["male_barbell_curl_std_1rm_by_age_strengthlevel"]])
# View(male_barbell_curl_std_1rm_by_age_strengthlevel)
MARK RIPPETOE’S STANDARDS
Mark Rippetoe’s Strength Standard: 1 Rep Max By Bodyweight.
Here we utilize Mark Rippetoe’s Strength Standards to benchmark
individual performance against established criteria for primary strength
exercises. These standards are instrumental in evaluating an
individual’s lifting capabilities in relation to their bodyweight and
training experience. By doing so, we can effectively gauge progress,
pinpoint areas that need improvement, and tailor training programs to
better meet the specific strength goals of each individual.
The exercises use the technique described in “Starting Strength:
Basic Barbell Training, 3rd ed.” The Bluebook may be purchased through
The Aasgaard Company online store and on Amazon in various mediums
(paperback, Kindle, audio). All lifts are performed with the barbell, as
indicated by the book title.
To properly utilize these standards, it is recommended to review
“Starting Strength: Basic Barbell Training” and “Practical
Programming for Strength Training.” The most recent and relevant 3rd editions
(2012) are available on their website, as well as in different mediums
on Amazon. You can download online book samples to determine if Starting
Strength suits your needs.
# Code extracts and refines a subset of data related to "Mark Rippetoe" standards from a broader dataset, making it more accessible and easier to work with by cleaning up the category names for further analysis or reporting
# subset() filters rows from the named_ranges dataframe.
# grepl() checks each entry in the "Strength Standard" column for the substring
mark_rippetoe_standards_list <- subset(named_ranges, grepl("Mark_Rippetoe", `Strength Standard`))
# Pipe Operator (%>%) is used to pass the result of one function to the next. It helps in writing cleaner and more readable code, especially when performing multiple operations on a dataset.
# gsub("_", " ", Category Sheet) replaces underscores (_) with spaces in the Category Sheet column.
# New column is created.
mark_rippetoe_standards_list <- mark_rippetoe_standards_list %>%
mutate(`Category Name` = gsub("_", " ", `Category Sheet`))
# Display the modified 'Strength Standard' column with spaces instead of underscores
display_mark_rippetoe_standards_list <- mark_rippetoe_standards_list %>%
mutate(`Strength Standard` = gsub("_", " ", `Strength Standard`)) %>%
select(-`Category Sheet`) # This excludes the 'Category Sheet' column
# The View function is used to open the raw_data dataframe in the RStudio data viewer for interactive exploration.
View(mark_rippetoe_standards_list) #Run this line of code to view table.
# Print for reference.
# print(mark_rippetoe_standards_list)
# print(display_mark_rippetoe_standards_list)
# Define the ratios and total width for column widths
total_width <- 100
ratio_strength_standards <- 2
ratio_category <- 1.25
total_ratio <- ratio_strength_standards + ratio_category
# Calculate the column widths
width_strength_standards <- (ratio_strength_standards / total_ratio) * total_width
width_category <- (ratio_category / total_ratio) * total_width
# Print the dataframe using kable for a nicely formatted markdown table.
kable(display_mark_rippetoe_standards_list, format = "html", col.names = c("Strength Standard", "Category")) %>%
kable_styling(full_width = TRUE) %>%
column_spec(1, width = paste0(width_strength_standards, "%")) %>%
column_spec(2, width = paste0(width_category, "%"))
Strength Standard
|
Category
|
Male Bench Press Std 1RM By Bodyweight Mark Rippetoe
|
Bench Press
|
Male Deadlift Std 1RM By Bodyweight Mark Rippetoe
|
Deadlift
|
Male Press Std 1RM By Bodyweight Mark Rippetoe
|
Press
|
Male Squat Std 1RM By Bodyweight Mark Rippetoe
|
Squat
|
There are five Starting Strength Standards, with the Power Clean
excluded in this report as my colleagues and I do not perform that
exercise. Each of the four remaining Standards is presented below,
followed by a plot to visualize performances for comparison and
motivation.
# Keep fig.width=10 for PDF display (via browser > Save As PDF). While fig.width=9 (instead of 10) fits width of output, saving as PDF with fig.width=9 will result in small figure; width=10 will fill the PDF page.
# Load the required packages if not already installed
library(ggplot2)
library(dplyr)
library(knitr)
library(kableExtra) # Ensure kableExtra is loaded for styling
# Function to generate ggplot for each data frame
generate_ggplot <- function(data, category_name, mark_rippetoe_strength_standard) {
data_number <- data %>%
mutate(Bodyweight = as.numeric(gsub("[^0-9]", "", Bodyweight)))
# Rename columns to remove double periods
colnames(data_number) <- gsub("\\.\\.", ". ", colnames(data_number))
# Print column names for debugging
# print(colnames(data_number))
ggplot(data_number, aes(x = Bodyweight)) +
geom_ribbon(aes(ymin = `Cat. I`, ymax = `Cat. II`), fill = "blue", alpha = 0.2) +
geom_ribbon(aes(ymin = `Cat. II`, ymax = `Cat. III`), fill = "green", alpha = 0.2) +
geom_ribbon(aes(ymin = `Cat. III`, ymax = `Cat. IV`), fill = "orange", alpha = 0.2) +
geom_ribbon(aes(ymin = `Cat. IV`, ymax = `Cat. V`), fill = "red", alpha = 0.2) +
geom_ribbon(aes(ymin = `Cat. V`, ymax = Inf), fill = "purple", alpha = 0.2) +
# Fill the remaining non-colored part
geom_ribbon(aes(ymin = -Inf, ymax = `Cat. I`), fill = "gray", alpha = 0.2) +
# Create lines
geom_line(aes(y = `Cat. I`), color = "blue", size = 1) +
geom_line(aes(y = `Cat. II`), color = "green", size = 1) +
geom_line(aes(y = `Cat. III`), color = "orange", size = 1) +
geom_line(aes(y = `Cat. IV`), color = "red", size = 1) +
geom_line(aes(y = `Cat. V`), color = "purple", size = 1) +
# Set labels
labs(
x = "Bodyweight (lb)",
y = "1RM Weight Load Execution",
title = paste("Male", category_name, "Standards"),
subtitle = "Comparison of Strength Standards Across Experience Levels",
caption = paste("Source:", mark_rippetoe_strength_standard) # Change Source info PRN
) +
# Add more markers on the y-axis at increments of 20
scale_y_continuous(breaks = seq(0, 999, by = 20)) +
# Add more markers on the x-axis at increments of 10
scale_x_continuous(breaks = seq(0, 999, by = 10)) +
# Add text labels for skill levels with matching colors
geom_text(aes(x = max(Bodyweight), y = min(`Cat. I`), label = ""), color = "dimgray", vjust = -1) +
geom_text(aes(x = max(Bodyweight), y = max(`Cat. I`), label = "Cat. I"), color = "darkblue", vjust = -1) +
geom_text(aes(x = max(Bodyweight), y = max(`Cat. II`), label = "Cat. II"), color = "darkgreen", vjust = -1) +
geom_text(aes(x = max(Bodyweight), y = max(`Cat. III`), label = "Cat. III"), color = "darkorange", vjust = -1) +
geom_text(aes(x = max(Bodyweight), y = max(`Cat. IV`), label = "Cat. IV"), color = "darkred", vjust = -1) +
geom_text(aes(x = max(Bodyweight), y = max(`Cat. V`), label = "Cat. V"), color = "purple", vjust = -1) +
# Customize theme
theme_minimal() +
theme(
plot.title = element_text(hjust = 0.5, size = 16),
plot.subtitle = element_text(hjust = 0.5, size = 12),
plot.caption = element_text(hjust = 0.5),
axis.title.x = element_text(size = 14),
axis.title.y = element_text(size = 14),
axis.text = element_text(size = 12),
legend.position = "right",
legend.title = element_blank(),
legend.text = element_text(size = 12),
legend.key.size = unit(1, "cm")
)
}
# Lists to store the plots and tables
plots_list_strength_standards_mark_rippetoe <- list()
tables_list_strength_standards_mark_rippetoe <- list()
# Check and create the 'plots' directory if it doesn't exist
if (!dir.exists("plots")) {
dir.create("plots")
}
# Check and create the 'tables' directory if it doesn't exist
if (!dir.exists("tables")) {
dir.create("tables")
}
# Loop through each named range from the filtered mark_rippetoe_standards_list dataframe
for (i in 1:nrow(mark_rippetoe_standards_list)) {
strength_standard_named_range <- mark_rippetoe_standards_list$`Strength Standard`[i]
category_name <- mark_rippetoe_standards_list$`Category Name`[i]
# Load the data from the named range
data_frame <- readWorkbook(strength_standards, namedRegion = strength_standard_named_range)
# Check if the 'Bodyweight' column exists in the data frame
if (!"Bodyweight" %in% colnames(data_frame)) {
cat(paste("Skipping", strength_standard_named_range, "- 'Bodyweight' column missing.\n"))
next
}
# Ensure column names are trimmed of whitespace
colnames(data_frame) <- trimws(colnames(data_frame))
# Rename columns to remove double periods for both table and plot
colnames(data_frame) <- gsub("\\.\\.", ". ", colnames(data_frame))
# Format the title of the plot and head of the table using the cleaned category name
formatted_header <- gsub("_", " ", strength_standard_named_range)
formatted_header <- (sub(" Std 1RM By Bodyweight Mark Rippetoe", " Standards", formatted_header))
# Generate the plot
plot <- generate_ggplot(data_frame, category_name, strength_standard_named_range)
# Print the name of the named range as a markdown header
cat(paste0("\n\n<h3> ", formatted_header, "</h3>\n"))
# Rename 'Bodyweight' column to 'Bodyweight (lb)'
names(data_frame)[names(data_frame) == "Bodyweight"] <- "Bodyweight (lb)"
# Print the plot
print(plot)
# Save the plot to the list
plots_list_strength_standards_mark_rippetoe[[formatted_header]] <- plot
# Output the data frame as a markdown table with even column widths
print(
table_html <- kable(data_frame, format = "html", align = 'l') %>%
kable_styling(full_width = TRUE) %>%
column_spec(1:ncol(data_frame), width = paste0(100 / ncol(data_frame), "%"))
)
# Save the table to the list
tables_list_strength_standards_mark_rippetoe[[formatted_header]] <- table_html
# Optionally save the plot and table
ggsave(filename = paste0("plots/", tolower(strength_standard_named_range), ".png"), plot = plot, width = 12, height = 8)
# Save the data frame as a CSV file
write.csv(data_frame, file = paste0("tables/", tolower(strength_standard_named_range), ".csv"), row.names = FALSE)
}
Male Bench Press Standards
Bodyweight (lb)
|
Cat. I
|
Cat. II
|
Cat. III
|
Cat. IV
|
Cat. V
|
114
|
84
|
107
|
130
|
179
|
222
|
123
|
91
|
116
|
142
|
194
|
242
|
132
|
98
|
125
|
153
|
208
|
260
|
148
|
109
|
140
|
172
|
234
|
291
|
165
|
119
|
152
|
187
|
255
|
319
|
181
|
128
|
164
|
201
|
275
|
343
|
198
|
135
|
173
|
213
|
289
|
362
|
220
|
142
|
183
|
225
|
306
|
381
|
242
|
149
|
190
|
232
|
316
|
395
|
275
|
153
|
196
|
239
|
325
|
407
|
319
|
156
|
199
|
244
|
333
|
416
|
320+
|
159
|
204
|
248
|
340
|
425
|

Male Deadlift Standards
Bodyweight (lb)
|
Cat. I
|
Cat. II
|
Cat. III
|
Cat. IV
|
Cat. V
|
114
|
97
|
179
|
204
|
299
|
387
|
123
|
105
|
194
|
222
|
320
|
414
|
132
|
113
|
209
|
239
|
342
|
438
|
148
|
126
|
234
|
269
|
380
|
482
|
165
|
137
|
254
|
293
|
411
|
518
|
181
|
148
|
274
|
315
|
438
|
548
|
198
|
156
|
289
|
333
|
457
|
567
|
220
|
164
|
305
|
351
|
479
|
586
|
242
|
172
|
318
|
363
|
490
|
596
|
275
|
176
|
326
|
373
|
499
|
602
|
319
|
180
|
333
|
381
|
506
|
608
|
320+
|
183
|
340
|
388
|
512
|
617
|

Male Press Standards
Bodyweight (lb)
|
Cat. I
|
Cat. II
|
Cat. III
|
Cat. IV
|
Cat. V
|
114
|
53
|
72
|
90
|
107
|
129
|
123
|
57
|
78
|
98
|
116
|
141
|
132
|
61
|
84
|
105
|
125
|
151
|
148
|
69
|
94
|
119
|
140
|
169
|
165
|
75
|
102
|
129
|
153
|
186
|
181
|
81
|
110
|
138
|
164
|
218
|
198
|
85
|
116
|
146
|
173
|
234
|
220
|
89
|
122
|
155
|
183
|
255
|
242
|
93
|
127
|
159
|
189
|
264
|
275
|
96
|
131
|
164
|
194
|
272
|
319
|
98
|
133
|
167
|
199
|
278
|
320+
|
100
|
136
|
171
|
203
|
284
|

Male Squat Standards
Bodyweight (lb)
|
Cat. I
|
Cat. II
|
Cat. III
|
Cat. IV
|
Cat. V
|
114
|
78
|
144
|
174
|
240
|
320
|
123
|
84
|
155
|
190
|
259
|
346
|
132
|
91
|
168
|
205
|
278
|
369
|
148
|
101
|
188
|
230
|
313
|
410
|
165
|
110
|
204
|
250
|
342
|
445
|
181
|
119
|
220
|
269
|
367
|
479
|
198
|
125
|
232
|
285
|
387
|
504
|
220
|
132
|
244
|
301
|
409
|
532
|
242
|
137
|
255
|
311
|
423
|
551
|
275
|
141
|
261
|
319
|
435
|
567
|
319
|
144
|
267
|
326
|
445
|
580
|
320+
|
147
|
272
|
332
|
454
|
593
|

# Example of how to recall and print a specific plot and table from the lists
# print(plots_list_strength_standards_mark_rippetoe[["Male Bench Press Std 1RM By Bodyweight Mark Rippetoe"]])
# print(tables_list_strength_standards_mark_rippetoe)
# print(tables_list_strength_standards_mark_rippetoe["Male Bench Press Std 1RM By Bodyweight Mark Rippetoe"])
# cat(as.character(tables_list_strength_standards_mark_rippetoe[["Male Bench Press Std 1RM By Bodyweight Mark Rippetoe"]]))
---
title: "Strength Standards"
author: "Kendrick Tham"
output:
  html_document:
    toc: yes
    toc_depth: 3
    toc_float:
      collapsed: no
      smooth_scroll: yes
      placement: top
      toc-subtree: yes
    code_folding: hide   # Hide the code by default
    code_download: true  # Add a button to download the R code

---

<!--
-->

Initial Print: 2024-07-31

Print Date: `r Sys.Date()`

<!-- RMD Setup -->

<style>

/* Reduce the font size of the TOC */
{
  font-size: 0.90em; /* Adjust the size as needed */
}

/* Add borders and spacing to all figures */
img {
  border: 1px solid #ddd; /* Specify border style */
  padding: 5px; /* Add padding around the figures */
  border-radius: 5px; /* Add rounded corners to the border */
  display: block; /* Ensure the image is treated as a block element to respect margin */
  margin-bottom: 20px; /* Add bottom margin to each image for spacing */
}

/* Add borders to all tables */
table {
  border-collapse: collapse; /* Collapse borders into a single border */
  border: 1px solid #ddd; /* Specify border style */
  margin-bottom: 20px; /* Add bottom margin to each table for spacing */
}

/* Add borders to table cells */
th, td {
  border: 1px solid #ddd; /* Specify border style for table cells */
  padding: 8px; /* Add padding inside the table cells */
  text-align: left; /* Align text to the left within table cells */
}

/* Apply styles to all headers */
th {
  position: sticky; /* Ensure headers stick when scrolling */
  top: 0; /* Position headers at the top */
  background-color: #f2f2f2; /* Background color for headers */
  z-index: 999; /* Ensure headers stay on top of other elements */
}

/* Apply styles to all tables */
table {
  border-collapse: collapse; /* Collapse borders into a single border */
  border: 1px solid #ddd; /* Specify border style */
}

/* Apply styles to table cells */
th, td {
  border: 1px solid #ddd; /* Specify border style for table cells */
  padding: 8px; /* Add padding inside the table cells */
  text-align: left; /* Align text to the left within table cells */
}

/* Style tables with alternating striped rows */
table {
  border-collapse: collapse;
  width: 100%;
}

th, td {
  border: 1px solid #dddddd;
  text-align: left;
  padding: 8px;
}

/* Add alternating background colors to table rows */
tr:nth-child(even) {
  background-color: #f2f2f2;
}

tr:nth-child(odd) {
  background-color: #ffffff; /* White background for odd rows */
}

.full-width-image img {
  display: block;
  width: 100%;
  height: auto;
}


</style>


<!-- Primary Pic -->

<div class="full-width-image">
  ![ ](C://Users//kendr//OneDrive//Documents//Data Analytics//KenDoesFitness//Strength Standards//Male_Bench_Press_Std_1RM_By_Bodyweight_Mark_Rippetoe.png)
</div>

<!-- Default Chunk Options -->
```{r setup, include=FALSE, }
# Set repository and default chunk options
options(repos = c(CRAN = "https://cloud.r-project.org"))
knitr::opts_chunk$set(echo=TRUE, eval=FALSE, results='hide', message=FALSE, warning=FALSE)
```

<!-- 
Options such as echo, message, warning, results, include, eval, cache, and more. 

- echo: Show or hide code in the output (echo = FALSE hides the code).
- message: Show or hide messages in the output (message = FALSE suppresses messages).
- warning: Show or hide warnings in the output (warning = FALSE suppresses warnings).
- results: Control how results are shown:
  - 'markup': Display results as they are.
  - 'asis': Include results as raw HTML or LaTeX.
  - 'hide': Hide results from output.
- include: Include or exclude both code and results from the output (include = FALSE excludes everything).
- eval: Evaluate the code chunk or not (eval = FALSE prevents code execution).
- cache: Cache the results of the chunk to speed up rendering (cache = TRUE enables caching).
- fig.cap: Add a caption to graphical outputs.
- out.width/out.height: Set the output size of plots.
- collapse: Concatenate textual output into a single block.
- tidy: Tidy up the display of code or results (tidy = TRUE tidies the output).
-->

<!-- ### Packages and Libraries -->
```{r packages_and_libraries, eval=TRUE}

# List of most of the required libraries used
required_libraries <- c("ggplot2", "stringr", "readxl", "dplyr", "lubridate", "hms", "openxlsx", "writexl", "rmarkdown", "knitr", "kableExtra")

# ggplot2     for 'declaratively' creating graphics, based on "The Grammar of Graphics". 
# stringr     for working with strings, i.e., character data.
# readxl      for reading Excel files by providing a way to read xls and xlsx files into R without dependencies on external software.
# dplyr       for data manipulation, providing a consistent set of verbs that help you solve the most common data manipulation challenges.
# lubridate   for working with dates and times in R.
# hms         for storing time-of-day values and interfacing these values with base R's date-time classes.
# openxlsx    for reading from and writing to Microsoft Excel .xlsx files. It provides more functionality than just handling data; it also allows you to modify various aspects of the Excel file itself.
# writexl     for a fast and portable way to write data frames to Excel .xlsx files, focusing solely on writing data without any dependencies.
# rmarkdown   for dynamic report generation from R. It integrates the core syntax of markdown (a simple markup language) with R code, allowing for the automatic update of reports to reflect data changes and analysis updates.
# knitr       for using functions such as kable() to format dataframes as markdown tables. 
# kableExtra  for enhanced functionality for styling tables such as displaying columns evenly

# Check which libraries are not installed
installed_libs <- rownames(installed.packages())
libraries_to_install <- required_libraries[!required_libraries %in% installed_libs]

# Install missing libraries
if(length(libraries_to_install) > 0) {
  install.packages(libraries_to_install)
}

# Load all required libraries
lapply(required_libraries, library, character.only = TRUE)
```

# TABLE OF CONTENTS
Utilize the Table of Contents on the left to scroll through the different sections and help navigate this report. You can click on the title to go directly to the section.

If you're viewing the PDF version, the Table of Contents stays at the top of this report. To go directly to a section, copy and paste the title into the search box. Use the following shortcut keys to help navigate with ease:

Shortcut Keys:

- Hold `Ctrl` and press the `Home` key to go to the top of the page.
- Hold `Ctrl` and press the `End` key to go to the bottom of the page.
- Hold `Ctrl` and press the `F` key to search for specific text or strings of text.
- Press the `Esc` key to quickly exit the search box.

## About This Report 

This report is published on RPubs[^1]. KDF stands for Ken Does Fitness. Refer to References at the bottom of this page for cited sources. The RMD file to produce this report, and some of the figures from this report, may be found in the Google Drive folder [^2]. 

# STRENGTH STANDARDS

Strength Standards provide valuable benchmarks for assessing an individual's performance in key strength exercises that are widely practiced in fitness and strength training. These standards categorize performance levels based on either amount of weight executed relative to bodyweight, or repetitions executed, and are segmented by gender and experience levels. By comparing an individual's performance against these established standards, trainers and athletes can identify strengths and weaknesses in their training regimens, set realistic goals, and measure progress in a structured manner.

## Strength Standards Overview 

Several strength standards are readily available online:

- ExRx Strength Standards: <https://exrx.net/Testing/WeightLifting/StrengthStandards>
- StrengthLevel.com Strength Standards: <https://strengthlevel.com/strength-standards>
- Symmetric Strength: <https://symmetricstrength.com/>
- Mark Rippetoe's Strength Standards: <https://startingstrength.com/files/standards.pdf>

Finding accurate strength standards depends on the demographic, exercise type, and required accuracy. Each has its pros and cons: some cater to specific demographics, others focus on certain exercises, and their credibility affects reliability. Usability and comprehensive data for benchmarking are crucial, as are regular updates to reflect current trends. Considering these factors helps trainers and athletes choose the best standards for assessing and improving performance.

```{r strength_standards_pnc, eval=TRUE, results='asis'}
# Load necessary libraries
library(knitr)
library(kableExtra)

# Define the pros and cons as lists
pros <- c("Benchmarking: Strength standards provide clear benchmarks to measure progress and set goals.",
          "Motivation: Seeing how your performance stacks up against established standards can be highly motivating.",
          "Guidance: They offer guidance on what is considered strong or weak for different exercises, helping you tailor your training accordingly.",
          "Comparison: They allow for meaningful comparisons between individuals, helping to identify strengths and weaknesses.",
          "Structured Training: Strength standards can help in designing more structured and effective training programs.")

cons <- c("Generic: Strength standards can be too generic and may not account for individual differences in genetics, body composition, and training history.",
          "Pressure: They can create undue pressure to meet certain benchmarks, potentially leading to injury or burnout.",
          "Outdated Information: Some standards may be based on outdated information or not reflect recent advancements in sports science.",
          "Not Comprehensive: They might not cover all exercises or account for different training goals, such as hypertrophy versus strength.",
          "Misleading: For beginners, strength standards can be misleading if they don’t account for the starting point or rate of progress appropriate for different fitness levels.")

# Combine the pros and cons into a data frame
strength_standards_pnc <- data.frame(
  Pros = pros,
  Cons = cons,
  stringsAsFactors = FALSE
)

# Print the table using kable and kableExtra
kable(strength_standards_pnc, format = "html", escape = FALSE, col.names = c("Pros", "Cons")) %>%
  kable_styling(bootstrap_options = c("striped", "hover", "condensed", "responsive"), full_width = TRUE) %>%
  column_spec(1, width = "50%") %>%
  column_spec(2, width = "50%")

```

I've compiled some of the standards into an Excel file[^3]. This allows for comparison between personal performance measures and the specified strength standards. I recommend consulting each source for confirmation on the latest version, as the structure or numbers may have changed over time. 

```{r read_strength_standards, eval=TRUE}

# File Path Definition: The script defines a path to an Excel file containing "Strength Standards".
file_path_strength_standards <- "C://Users//kendr//OneDrive//Documents//Data Analytics//KenDoesFitness/Strength Standards Final.xlsm"

# Workbook Loading: Loads the Excel workbook into R using the openxlsx package, which provides more advanced control over Excel files than readxl.
strength_standards <- loadWorkbook(file_path_strength_standards)
```

## Standards Categories  
The standards compiled are separated into categories, as shown in the following table, making it easy to determine which exercise to use as a benchmark.

```{r strength_standards_sheet_names, eval=TRUE, results='markup'}

# Sheet Names Display: Retrieves and displays the names of sheets within the Excel workbook to help identify available data.

# Check to confirm which number of names to display
# names(strength_standards)

# Display specific elements from the strength_standards object with Indexing
names_subset_strength_standards <- names(strength_standards)[4:15]
names_subset_strength_standards_gsub = gsub("_"," ", names_subset_strength_standards)
# names_subset_strength_standards

# Calculate the number of names
num_names <- length(names_subset_strength_standards_gsub)

# Calculate the number of names per column
names_per_column <- ceiling(num_names / 2)

# Split the names into two columns
column_1 <- names_subset_strength_standards_gsub[1:min(names_per_column, num_names)]
column_2 <- names_subset_strength_standards_gsub[(min(names_per_column, num_names) + 1):num_names]

# Pad the shorter column with NA values if necessary
if(length(column_1) < length(column_2)) {
  column_1 <- c(column_1, rep("", length(column_2) - length(column_1)))
} else if(length(column_2) < length(column_1)) {
  column_2 <- c(column_2, rep("", length(column_1) - length(column_2)))
}
  
# Create a data frame with two columns
names_subset_strength_standards_gsub_df <- data.frame(Column_1 = column_1,
                              Column_2 = column_2)

# Calculate the width dynamically
col_width <- paste0(100 / ncol(names_subset_strength_standards_gsub_df), "%")

# Print the data frame using knitr::kable
knitr::kable(names_subset_strength_standards_gsub_df, format = "markdown", col.names = c("Categories", "Categories")) %>%
  column_spec(1:2, width = col_width, extra_css = "text-align: left;")

```

## Strength Standards List 
Below is the list of Strength Standards compiled into the SS Excel file[^3]. This file references the source of each standard table. We'll delve into Mark Rippetoe's Starting Strength Standards, as his and his colleagues' methods and research are based on comprehensive and technical information.

```{r named_ranges, eval=TRUE, results='markup'}

library(openxlsx)
library(readxl)
library(knitr)
library(kableExtra)

# Read Named Ranges: Reads a specific sheet named "Named_Ranges" that presumably contains metadata about other named ranges in the workbook. It keeps only the columns related to named ranges and their respective sheets.
named_ranges <- read_excel(file_path_strength_standards, sheet = "Named_Ranges")
sheet_names_filtered <- unique(named_ranges$`Category Sheet`)
named_ranges <- named_ranges[named_ranges$`Category Sheet` %in% sheet_names_filtered, c("Strength Standard", "Category Sheet")]

# Display the modified 'Strength Standard' column with spaces instead of underscores
strength_standards_list_display <- named_ranges %>%
  mutate(`Strength Standard` = gsub("_", " ", `Strength Standard`)) %>%
  mutate(`Category Name` = gsub("_", " ", `Category Sheet`)) %>%
  select(-`Category Sheet`)  # This excludes the 'Category Sheet' column

# View the data frame if needed (not during knitting). View function opens the raw_data dataframe in the RStudio data viewer for interactive exploration.
View(named_ranges)
View(strength_standards_list_display)

# Define the ratios and total width for column widths
total_width <- 100
ratio_strength_standards <- 2
ratio_category <- 1.25
total_ratio <- ratio_strength_standards + ratio_category

# Calculate the column widths
width_strength_standards <- (ratio_strength_standards / total_ratio) * total_width
width_category <- (ratio_category / total_ratio) * total_width

# Print the dataframe using kable for a nicely formatted markdown table.
kable(strength_standards_list_display, format = "html", col.names = c("Strength Standard", "Category")) %>%
  kable_styling(full_width = TRUE) %>%
  column_spec(1, width = paste0(width_strength_standards, "%")) %>%
  column_spec(2, width = paste0(width_category, "%"))

# Create lists to store data frames
named_ranges_list <- list()
named_ranges_sheets_list <- list()

# Loop through the named ranges
for (i in 1:nrow(named_ranges)) {
  named_range_value <- named_ranges$`Strength Standard`[i]
  named_range_sheet_value <- named_ranges$`Category Sheet`[i]
  named_range_lower <- tolower(gsub(" ", "_", named_range_value))
  named_range_sheet_lower <- tolower(gsub(" ", "_", named_range_sheet_value))
  
  if (named_range_sheet_value == "Named_Ranges") {
    next  # Skip iteration if the sheet is "Named_Ranges" to avoid redundancy
  }
  
  # Read data from the named range
  named_range_data <- tryCatch({
    read.xlsx(file_path_strength_standards, namedRegion = named_range_value)
  }, error = function(e) {
    message(paste("Warning: Workbook has no such named region:", named_range_value))
    NULL
  })
  
  if (!is.null(named_range_data) && is.data.frame(named_range_data)) {
    # Update column names
    colnames(named_range_data) <- gsub("\\.", " ", colnames(named_range_data))
    # Store the data frame in the list
    named_ranges_list[[named_range_lower]] <- named_range_data
    # Assign the data frame to the global environment with the lower case name
    assign(named_range_lower, named_range_data, envir = .GlobalEnv)
  }
  
  # Read data from the sheet
  named_range_sheet_data <- tryCatch({
    read.xlsx(file_path_strength_standards, sheet = named_range_sheet_value)
  }, error = function(e) {
    message(paste("Warning: Workbook has no such sheet:", named_range_sheet_value))
    NULL
  })
  
  if (!is.null(named_range_sheet_data) && is.data.frame(named_range_sheet_data)) {
    # Update column names
    colnames(named_range_sheet_data) <- gsub("\\.", " ", colnames(named_range_sheet_data))
    # Store the data frame in the list
    named_ranges_sheets_list[[named_range_sheet_lower]] <- named_range_sheet_data
    # Assign the data frame to the global environment with the lower case name
    assign(named_range_sheet_lower, named_range_sheet_data, envir = .GlobalEnv)
  }
}

# Display the structure of the lists
# str(named_ranges_list)
# str(named_ranges_sheets_list)

# Optional: View the contents of the lists in RStudio Viewer
# Instead of viewing the whole list, view individual data frames
# Uncomment the below lines to view specific named ranges or sheets
# View(named_ranges_list[["male_barbell_curl_std_1rm_by_age_strengthlevel"]])
# View(male_barbell_curl_std_1rm_by_age_strengthlevel)

```

# MARK RIPPETOE'S STANDARDS

Mark Rippetoe's Strength Standard: 1 Rep Max By Bodyweight.  

Here we utilize Mark Rippetoe's Strength Standards[^4] to benchmark individual performance against established criteria for primary strength exercises. These standards are instrumental in evaluating an individual's lifting capabilities in relation to their bodyweight and training experience. By doing so, we can effectively gauge progress, pinpoint areas that need improvement, and tailor training programs to better meet the specific strength goals of each individual.

The exercises use the technique described in "Starting Strength: Basic Barbell Training, 3rd ed." The Bluebook may be purchased through The Aasgaard Company online store[^5] and on Amazon in various mediums (paperback, Kindle, audio). All lifts are performed with the barbell, as indicated by the book title.

To properly utilize these standards, it is recommended to review "Starting Strength: Basic Barbell Training"[^8] and "Practical Programming for Strength Training."[^9] The most recent and relevant 3rd editions (2012) are available on their website, as well as in different mediums on Amazon. You can download online book samples to determine if Starting Strength suits your needs.

```{r mark_rippetoe_standards_list, eval=TRUE, results='markup'}

# Code extracts and refines a subset of data related to "Mark Rippetoe" standards from a broader dataset, making it more accessible and easier to work with by cleaning up the category names for further analysis or reporting

# subset() filters rows from the named_ranges dataframe.
# grepl() checks each entry in the "Strength Standard" column for the substring 
mark_rippetoe_standards_list <- subset(named_ranges, grepl("Mark_Rippetoe", `Strength Standard`))

# Pipe Operator (%>%) is used to pass the result of one function to the next. It helps in writing cleaner and more readable code, especially when performing multiple operations on a dataset.
# gsub("_", " ", Category Sheet) replaces underscores (_) with spaces in the Category Sheet column.
# New column is created. 
mark_rippetoe_standards_list <- mark_rippetoe_standards_list %>%
  mutate(`Category Name` = gsub("_", " ", `Category Sheet`))

# Display the modified 'Strength Standard' column with spaces instead of underscores
display_mark_rippetoe_standards_list <- mark_rippetoe_standards_list %>%
  mutate(`Strength Standard` = gsub("_", " ", `Strength Standard`)) %>%
  select(-`Category Sheet`)  # This excludes the 'Category Sheet' column

# The View function is used to open the raw_data dataframe in the RStudio data viewer for interactive exploration.
View(mark_rippetoe_standards_list) #Run this line of code to view table. 

# Print for reference. 
# print(mark_rippetoe_standards_list)
# print(display_mark_rippetoe_standards_list)

# Define the ratios and total width for column widths
total_width <- 100
ratio_strength_standards <- 2
ratio_category <- 1.25
total_ratio <- ratio_strength_standards + ratio_category

# Calculate the column widths
width_strength_standards <- (ratio_strength_standards / total_ratio) * total_width
width_category <- (ratio_category / total_ratio) * total_width

# Print the dataframe using kable for a nicely formatted markdown table.
kable(display_mark_rippetoe_standards_list, format = "html", col.names = c("Strength Standard", "Category")) %>%
  kable_styling(full_width = TRUE) %>%
  column_spec(1, width = paste0(width_strength_standards, "%")) %>%
  column_spec(2, width = paste0(width_category, "%"))


```

There are five Starting Strength Standards, with the Power Clean excluded in this report as my colleagues and I do not perform that exercise. Each of the four remaining Standards is presented below, followed by a plot to visualize performances for comparison and motivation.

```{r mark_rippetoe_standards_tables_and_plots, fig.width=10, fig.height=7, eval=TRUE, results='asis'}

# Keep fig.width=10 for PDF display (via browser > Save As PDF). While fig.width=9 (instead of 10) fits width of output, saving as PDF with fig.width=9 will result in small figure; width=10 will fill the PDF page.

# Load the required packages if not already installed
library(ggplot2)
library(dplyr)
library(knitr)
library(kableExtra) # Ensure kableExtra is loaded for styling

# Function to generate ggplot for each data frame
generate_ggplot <- function(data, category_name, mark_rippetoe_strength_standard) {
  data_number <- data %>%
    mutate(Bodyweight = as.numeric(gsub("[^0-9]", "", Bodyweight)))

  # Rename columns to remove double periods
  colnames(data_number) <- gsub("\\.\\.", ". ", colnames(data_number))

  # Print column names for debugging
  # print(colnames(data_number))
    
  ggplot(data_number, aes(x = Bodyweight)) +
    geom_ribbon(aes(ymin = `Cat. I`, ymax = `Cat. II`), fill = "blue", alpha = 0.2) +
    geom_ribbon(aes(ymin = `Cat. II`, ymax = `Cat. III`), fill = "green", alpha = 0.2) +
    geom_ribbon(aes(ymin = `Cat. III`, ymax = `Cat. IV`), fill = "orange", alpha = 0.2) +
    geom_ribbon(aes(ymin = `Cat. IV`, ymax = `Cat. V`), fill = "red", alpha = 0.2) +
    geom_ribbon(aes(ymin = `Cat. V`, ymax = Inf), fill = "purple", alpha = 0.2) +
    
    # Fill the remaining non-colored part
    geom_ribbon(aes(ymin = -Inf, ymax = `Cat. I`), fill = "gray", alpha = 0.2) +
    
    # Create lines
    geom_line(aes(y = `Cat. I`), color = "blue", size = 1) +
    geom_line(aes(y = `Cat. II`), color = "green", size = 1) +
    geom_line(aes(y = `Cat. III`), color = "orange", size = 1) +
    geom_line(aes(y = `Cat. IV`), color = "red", size = 1) +
    geom_line(aes(y = `Cat. V`), color = "purple", size = 1) +
    
    # Set labels
    labs(
      x = "Bodyweight (lb)",
      y = "1RM Weight Load Execution",
      title = paste("Male", category_name, "Standards"),
      subtitle = "Comparison of Strength Standards Across Experience Levels",
      caption = paste("Source:", mark_rippetoe_strength_standard) # Change Source info PRN
    ) +
    
    # Add more markers on the y-axis at increments of 20
    scale_y_continuous(breaks = seq(0, 999, by = 20)) +
    # Add more markers on the x-axis at increments of 10
    scale_x_continuous(breaks = seq(0, 999, by = 10)) +
    
    # Add text labels for skill levels with matching colors
    geom_text(aes(x = max(Bodyweight), y = min(`Cat. I`), label = ""), color = "dimgray", vjust = -1) +
    geom_text(aes(x = max(Bodyweight), y = max(`Cat. I`), label = "Cat. I"), color = "darkblue", vjust = -1) +
    geom_text(aes(x = max(Bodyweight), y = max(`Cat. II`), label = "Cat. II"), color = "darkgreen", vjust = -1) +
    geom_text(aes(x = max(Bodyweight), y = max(`Cat. III`), label = "Cat. III"), color = "darkorange", vjust = -1) +
    geom_text(aes(x = max(Bodyweight), y = max(`Cat. IV`), label = "Cat. IV"), color = "darkred", vjust = -1) +
    geom_text(aes(x = max(Bodyweight), y = max(`Cat. V`), label = "Cat. V"), color = "purple", vjust = -1) +
    
    # Customize theme
    theme_minimal() +
    theme(
      plot.title = element_text(hjust = 0.5, size = 16),
      plot.subtitle = element_text(hjust = 0.5, size = 12),
      plot.caption = element_text(hjust = 0.5),
      axis.title.x = element_text(size = 14),
      axis.title.y = element_text(size = 14),
      axis.text = element_text(size = 12),
      legend.position = "right",
      legend.title = element_blank(),
      legend.text = element_text(size = 12),
      legend.key.size = unit(1, "cm")
    )
}

# Lists to store the plots and tables
plots_list_strength_standards_mark_rippetoe <- list()
tables_list_strength_standards_mark_rippetoe <- list()

# Check and create the 'plots' directory if it doesn't exist
if (!dir.exists("plots")) {
  dir.create("plots")
}

# Check and create the 'tables' directory if it doesn't exist
if (!dir.exists("tables")) {
  dir.create("tables")
}

# Loop through each named range from the filtered mark_rippetoe_standards_list dataframe
for (i in 1:nrow(mark_rippetoe_standards_list)) {
  strength_standard_named_range <- mark_rippetoe_standards_list$`Strength Standard`[i]
  category_name <- mark_rippetoe_standards_list$`Category Name`[i]
  
  # Load the data from the named range
  data_frame <- readWorkbook(strength_standards, namedRegion = strength_standard_named_range)
  
  # Check if the 'Bodyweight' column exists in the data frame
  if (!"Bodyweight" %in% colnames(data_frame)) {
    cat(paste("Skipping", strength_standard_named_range, "- 'Bodyweight' column missing.\n"))
    next
  }

  # Ensure column names are trimmed of whitespace
  colnames(data_frame) <- trimws(colnames(data_frame))

  # Rename columns to remove double periods for both table and plot
  colnames(data_frame) <- gsub("\\.\\.", ". ", colnames(data_frame))

  # Format the title of the plot and head of the table using the cleaned category name
  formatted_header <- gsub("_", " ", strength_standard_named_range)
  formatted_header <- (sub(" Std 1RM By Bodyweight Mark Rippetoe", " Standards", formatted_header))
  
  # Generate the plot
  plot <- generate_ggplot(data_frame, category_name, strength_standard_named_range)
  
  # Print the name of the named range as a markdown header
  cat(paste0("\n\n<h3> ", formatted_header, "</h3>\n"))

  # Rename 'Bodyweight' column to 'Bodyweight (lb)'
  names(data_frame)[names(data_frame) == "Bodyweight"] <- "Bodyweight (lb)"

  # Print the plot
  print(plot)

  # Save the plot to the list
  plots_list_strength_standards_mark_rippetoe[[formatted_header]] <- plot

  # Output the data frame as a markdown table with even column widths
  print(
    table_html <- kable(data_frame, format = "html", align = 'l') %>%
      kable_styling(full_width = TRUE) %>%
      column_spec(1:ncol(data_frame), width = paste0(100 / ncol(data_frame), "%"))
  )
  
  # Save the table to the list
  tables_list_strength_standards_mark_rippetoe[[formatted_header]] <- table_html

  # Optionally save the plot and table
  ggsave(filename = paste0("plots/", tolower(strength_standard_named_range), ".png"), plot = plot, width = 12, height = 8)

  # Save the data frame as a CSV file
  write.csv(data_frame, file = paste0("tables/", tolower(strength_standard_named_range), ".csv"), row.names = FALSE)
}

# Example of how to recall and print a specific plot and table from the lists
# print(plots_list_strength_standards_mark_rippetoe[["Male Bench Press Std 1RM By Bodyweight Mark Rippetoe"]])
# print(tables_list_strength_standards_mark_rippetoe)
# print(tables_list_strength_standards_mark_rippetoe["Male Bench Press Std 1RM By Bodyweight Mark Rippetoe"])
# cat(as.character(tables_list_strength_standards_mark_rippetoe[["Male Bench Press Std 1RM By Bodyweight Mark Rippetoe"]]))

```

# REFERENCES 


[^1]: Tham, K. (2024). *Strength Standards (`r Sys.Date()`).* RPubs by RStudio. Retrieved from <https://rpubs.com/KenDoesData/>.

[^2]: Tham, K. (2024). *KDF (KEN DOES FITNESS).* Google Drive. Available at <https://drive.google.com/drive/folders/1-41FJ_c41Kj7KaK0h1aJ5EBvEhPwwIsi?usp=sharing>

[^3]: Tham, K. (2024). *Strength Standards Final.xlsm.* Ken Does Data. Retrieved from <https://1drv.ms/x/s!AsYYZA9nL8lf8CK6k6rAP3LSkvry?e=ylARXi>.

[^4]: Rippetoe, M. (2012). *Strength Standards.* Starting Strength. Retrieved from <https://startingstrength.com/files/standards.pdf>.

[^5]: Rippetoe, M. (2017). *Starting Strength: Basic Barbell Training (3rd ed.).* The Aasgaard Company. Retrieved from <https://aasgaardco.com/store/books-posters-dvd/books/starting-strength-basic-barbell-training/>.

[^6]: Legion Athletics. (2019). *These Are the Best Strength Standards on the Internet.* Retrieved from <https://legionathletics.com/strength-standards/>.

[^7]: Rippetoe, M. (2012). *The Strength Standards Tables are back up.* Starting Strength Forum. Retrieved from <https://startingstrength.com/resources/forum/mark-rippetoe-q-and-a/33029-strength-standards-tables.html>.

[^8]: Rippetoe, M. (2017). *Starting Strength: Basic Barbell Training, 3rd Edition* by Mark Rippetoe. Available at <https://www.amazon.com/Starting-Strength-Mark-Rippetoe-ebook/dp/B006XJR5ZA/ref=sr_1_1?crid=21521YXBKX4AI&dib=eyJ2IjoiMSJ9.1qTeh_8DJhqDmxrrajmjTbLiQz-wPaKOifgi787M7WurNxwUwKVbhQadJEw8Wdu2ygDP8X0D23Y5xGqEZAJ2L5luEw0gvCIIyuTmugYZ_bRmG35hqLRPlbOyQx4e5pRf2wVoGnsQPzFQEfEOlrhiHKp8BSWCuQkf2RELFKSOLQO49Fq8Y57PU_MuPm9ge91UcC3ebmCZ8pCWHmqbUURN6tyocDT-CR3ZaQAKPJXnmYPsocCP4jz4Y_QTLE1hecsk8Vuugf3HIBjBWkOVMZFmSDlS9ujVxqF4vRqC7GsVJlw._9jNplpd9L6PPFzk7WMC-nSDfWXPhmckFiiaMSwaCyo>.

[^9]: Rippetoe, M. *Practical Programming for Strength Training, 3rd Edition* by Mark Rippetoe. Available at <https://www.amazon.com/Practical-Programming-Strength-Training-Rippetoe-ebook/dp/B00IU8YETW/ref=sr_1_2?crid=21521YXBKX4AI&dib=eyJ2IjoiMSJ9.1qTeh_8DJhqDmxrrajmjTbLiQz-wPaKOifgi787M7WurNxwUwKVbhQadJEw8Wdu2ygDP8X0D23Y5xGqEZAJ2L5luEw0gvCIIyuTmugYZ_bRmG35hqLRPlbOyQx4e5pRf2wVoGnsQPzFQEfEOlrhiHKp8BSWCuQkf2RELFKSOLQO49Fq8Y57PU_MuPm9ge91UcC3ebmCZ8pCWHmqbUURN6tyocDT-CR3ZaQAKPJXnmYPsocCP4jz4Y_QTLE1hecsk8Vuugf3HIBjBWkOVMZFmSDlS9ujVxqF4vRqC7GsVJlw._9jNplpd9L6PPFzk7WMC-nSDfWXPhmckFiiaMSwaCyo>.
