diff --git a/BC.csv b/BC.csv new file mode 100644 index 0000000..17f64be --- /dev/null +++ b/BC.csv @@ -0,0 +1,39 @@ +cond,weight(g),femur(mm) +no_light,0.196,5.7 +no_light,0.145,5.5 +no_light,0.232,5.1 +no_light,0.185,5.05 +no_light,0.147,4.8 +no_light,0.108,4.55 +no_light,0.132,4.3 +no_light,0.132,4.7 +no_light,0.142,5.1 +no_light,0.142,5.5 +no_light,0.128,5.2 +no_light,0.131,5 +no_light,0.13,5.3 +no_light,0.115,4.5 +no_light,0.121,4.7 +light,0.209,5.15 +light,0.175,4.75 +light,0.196,5.05 +light,0.223,5.35 +light,0.244,5.6 +light,0.156,4.25 +light,0.176,4.45 +light,0.283,4.9 +light,0.18,4.85 +light,0.157,4.55 +light,0.211,5.25 +light,0.213,5.75 +light,0.181,4.7 +light,0.295,6 +light,0.15,4.6 +light,0.158,5.05 +light,0.147,4.1 +light,0.344,5.3 +light,0.178,5.3 +light,0.202,4.6 +light,0.188,5 +light,0.27,5.5 +light,0.189,5.55 diff --git a/README.txt b/README.txt new file mode 100644 index 0000000..7d2358d --- /dev/null +++ b/README.txt @@ -0,0 +1,7 @@ +This data is from August 13-23 of 2018 on the Saint-Symphorien suspension bridge over the Loire River in Tours, France (47.399158° N, 0.692519° E). This 350m pedestrian bridge consists of 324 individual metal side railing panels within which orb-weaving spiders (Family Araneidae) constructed webs. Bridge panels had an alternating lighting pattern; every other panel was lit from above by a blue fluorescent tube light, which was absent from the other panels. These two conditions acted as a natural block-on block-off design within I measured spider abundance, web-dimensions, prey capture success, and body condition. In artificially lit conditions, spiders caught more prey with smaller webs, and had higher body condition. However, there were fewer spiders with active webs in those lit areas. + +BC.csv contains data on individual Larinioides sclopetarius body size (femur length in mm) and weight (in g). Body condition can be calculated with the attached R script. +TOURS.csv contains data on spider abundance in bridge panels. This is the number of spiders which had active orb webs in each panel (either lit or unlit). +webs.csv contains data on spider web dimensions and prey capture success. These web dimensions can be used to calculate vertical web asymmetry and total web catch area. + +Tours_Spider.R contains the scripts to read all three csv files above, analyze the data, and plot the data. diff --git a/Spider_Light.R b/Spider_Light.R new file mode 100644 index 0000000..83f4588 --- /dev/null +++ b/Spider_Light.R @@ -0,0 +1,347 @@ + + + +# > sessionInfo() +# R version 3.5.1 (2018-07-02) +# Platform: x86_64-w64-mingw32/x64 (64-bit) +# Running under: Windows >= 8 x64 (build 9200) +# +# Matrix products: default +# +# locale: +# [1] LC_COLLATE=English_United States.1252 LC_CTYPE=English_United States.1252 +# [3] LC_MONETARY=English_United States.1252 LC_NUMERIC=C +# [5] LC_TIME=English_United States.1252 +# +# attached base packages: +# [1] grid stats graphics grDevices utils datasets methods base +# +# other attached packages: +# [1] cowplot_0.9.3 ggpubr_0.1.8 magrittr_1.5 ggplot2_3.0.0 DHARMa_0.2.4 glmmTMB_0.2.3 dplyr_0.8.1 + + +##### Load Libraries: #### +library(dplyr) +library(glmmTMB) +library(DHARMa) +library("ggpubr") +library(grid) +library(cowplot) + +#### LOAD DATA #### + +### Abundance +TOURS<-read.csv("TOURS.csv") +TOURS$Light[TOURS$Light=="0"]<-"No light" +TOURS$Light[TOURS$Light=="1"]<-"Light" + +### WEBS +WEB<-read.csv("webs.csv") +WEB$Area=pi*(WEB$Dv/2)*(WEB$Dh/2)-(pi*(WEB$Dfz/2)^2) # calculate web catch area +WEB$Asy<-(WEB$Rt-(WEB$Dv-WEB$Rt))/WEB$Dv # calculate web asymmetry +WEB$Prey[which(WEB$Prey=="NaN")]<-NA + +### Body Condition +SP<-read.csv("BC.csv") +SP$Wt<-SP$weight.g. ### Wt is measured in grams (g) # clean up names +SP$Fe<-SP$femur.mm. ### Fe is measured in milimeter (mm) # clean up names +SP$cond<-as.character(SP$cond) +SP$cond[which(SP$cond=="no_light")]<-("No Light") # clean up names +SP$cond[which(SP$cond=="light")]<-("Light") # clean up names +SP$cond<-as.factor(SP$cond) + +# +# +# + +### Abundance data #### + +## Summary Statistics for ABUNDANCE +Abun<-group_by(TOURS, Light) %>% + dplyr::summarise( + count = n(), + total = sum(Spiders), + mean = mean(Spiders, na.rm=TRUE), + SD = sd(Spiders, na.rm=TRUE), + median = median(Spiders, na.rm = TRUE), + IQR = IQR(Spiders, na.rm = TRUE), + min_range = min(Spiders, na.rm=TRUE), + max_range = max(Spiders, na.rm=TRUE) + ) + +Abun + +## Statistical testing of abundance data: + +# plot(density(TOURS$Spiders)) # look at distribution +# var(TOURS$Spiders)/mean(TOURS$Spiders) # overdispersed + +mod1<-glmmTMB(Spiders ~ Light, family=nbinom1(link="log"), data=TOURS) +# plot(simulateResiduals(mod1)) # check residuals = good fit + +# check model output: +summary(mod1) + +#95% confidence interval +# exp(0.32024+1.96*0.09572) # Upper 95% CI +# exp(0.32024-1.96*0.09572) # Lower 95% CI +# exp(0.32024) # Parameter estimate + + + +### WEB MEASUREMENTS #### + +### WEB AREA ### + +## summary stats +Web<-group_by(WEB, Condition) %>% + dplyr::summarise( + count = n(), + median = median(Area, na.rm = TRUE), + mean = mean(Area, na.rm=TRUE), + SD = sd(Area, na.rm=TRUE) + ) +Web + +# plot(density(WEB$Area)) # look at distribution +# var(WEB$Area)/mean(WEB$Area) # Overdispersed? + +## Statistical testing +mod2<-glm(Area~Condition,family=Gamma(link="log"),data=WEB) +# plot(simulateResiduals(mod2)) # check residual plots + +summary(mod2) + +# exp(0.43308) # parameter estimate +# exp(0.43308+ 1.96*0.12917) # Upper 95% CI +# exp(0.43308- 1.96*0.12917) # Lower 95% CI + + + +### VERTICAL WEB ASYMMETRY ### + +# plot(density(WEB$Asy)) # look at distribution + +WEB$Condition<-relevel(WEB$Condition,"No Light") +mod3<-lm(Asy ~ Condition + Area, data=WEB) + +# qqnorm(resid(mod3)) # check residual plots +# qqline(resid(mod3)) # check residual plots + +summary(mod3) +confint(mod3) + +## summary stats VWA +group_by(WEB, Condition) %>% + summarise( + count = n(), + median = median(Asy, na.rm = TRUE), + mean = mean(Asy, na.rm=TRUE), + SD = sd(Asy, na.rm=TRUE) + ) + + + + +### PREY CAPTURE #### +plot(density(PREY$Prey)) # plot data distribution +PREY<-WEB[!is.na(WEB$Prey),] # remove NA to look at dispersion +var(PREY$Prey)/mean(PREY$Prey) # overdispersed + +mod4<-glmmTMB(Prey ~ Condition+Area, data=PREY, family=nbinom2(link="log")) +# plot(simulateResiduals(mod4)) # check residual plots + +summary(mod4) + +# exp(1.9376593) # parameter estimate +# exp(1.9376593+1.96*0.1894091) # Upper 95% CI +# exp(1.9376593-1.96*0.1894091) # Lower 95% CI + +## Prey capture summary statistics +group_by(WEB,Condition) %>% + summarize( + count=length(which(!is.na(Prey))), + mean=mean(Prey,na.rm=TRUE), + SD = sd(Prey, na.rm=TRUE)) + + + + +## Body Condition #### + +## Summary stats for body condition +group_by(SP, cond) %>% + summarise( + count = n(), + meanWt=mean(Wt), + meanFe=mean(Fe) + ) + +## Generate residuals for Body Condition (following Jakob et al. 1996) +plot(Wt~Fe, data=SP) + +Bod.lm = lm(Wt~Fe, data=SP) # run lm for residuals +# plot(Bod.lm) # visualize + +SP$res<-resid(Bod.lm) # get residuals into original dataset + +## Summary stats for Residuals +Res<-group_by(SP, cond) %>% + dplyr::summarise( + count = n(), + mean=mean(res), + SD = sd(res), + median = median(res, na.rm = TRUE), + IQR = IQR(res, na.rm = TRUE), + min_range = min(res, na.rm=TRUE), + max_range = max(res, na.rm=TRUE) + ) + +# Res #View summary stats + +# plot(density(SP$res)) # look at distribution + +lm1<-lm(res ~ cond, data=SP) +# qqnorm(resid(lm1)) # check residual plots +# qqline(resid(lm1)) + +summary(lm1) +confint(lm1) + +lm2<-lm(Fe ~ cond, data=SP) +# qqnorm(resid(lm2)) # check residual plots +# qqline(resid(lm2)) + +summary(lm2) +confint(lm2) + +# Summary table of body size (Femur length) +group_by(SP, cond) %>% + dplyr::summarise( + count = n(), + mean=mean(Fe), + SD = sd(Fe), + median = median(Fe, na.rm = TRUE), + IQR = IQR(Fe, na.rm = TRUE), + min_range = min(Fe, na.rm=TRUE), + max_range = max(Fe, na.rm=TRUE) + ) + + + + + + +#### Graphical Visualization of Data #### + +## Figure 3 +### Visualization of abundance data + +grob <- grobTree(textGrob("*", x=0.475, y=0.9, hjust=0, + gp=gpar(col="black", fontsize=36, fontface="bold"))) + +A<-ggboxplot(data=TOURS, x = "Light", y = "Spiders", fill = "Light", palette = c("#00AFBB", "#FC4E07"), + ylab ="Spiders / \nbridge panel \n", xlab = "", legend="FALSE")+ + theme(axis.text=element_text(size=12), + axis.title=element_text(size=14))+ + scale_y_continuous(breaks=c(5,10,15,20,25))+ + annotation_custom(grob) + +### Web Area + +B<-ggboxplot(data=WEB, x = "Condition", y = "Area", fill = "Condition", order=c("No Light","Light"),palette = c("#00AFBB", "#FC4E07"), + ylab = expression(paste('Web area ',(cm^2))), xlab = "", legend="FALSE")+ + theme(axis.text=element_text(size=12), + axis.title=element_text(size=14))+ + scale_y_continuous(breaks=c(250,500,750,1000,1250))+ + annotation_custom(grob) + + +### Body Condition plot +C<-ggboxplot(data=SP, x = "cond", y = "res", fill = "cond", order=c("No Light","Light"), + palette = c("#00AFBB", "#FC4E07"), + ylab ="Body Condition\n(Residual Index) \n", xlab = "\nCondition", legend="FALSE")+ + theme(axis.text=element_text(size=12), + axis.title=element_text(size=14))+ + geom_hline(yintercept=0,linetype=2)+ + scale_y_continuous(breaks=c(-0.05,0.00,0.05), limits = c(-.05,.05))+ + annotation_custom(grob) + +grob <- grobTree(textGrob("N.S.", x=0.435, y=0.9, hjust=0, + gp=gpar(col="black", fontsize=28, fontface="bold"))) + +SP$sFe<-scale(SP$Fe) +D<-ggboxplot(data=SP, x = "cond", y = "sFe", fill = "cond", order=c("No Light","Light"), + palette = c("#00AFBB", "#FC4E07"), + ylab ="Body Size\n(Scaled Femur length) \n", xlab = "\nCondition", legend="FALSE")+ + theme(axis.text=element_text(size=12), + axis.title=element_text(size=14))+ + geom_hline(yintercept=0,linetype=2)+ + annotation_custom(grob) + + +## Making final plot of Figure 3 - combining ggplots in a grid of 3 panels + +plot_grid(A +rremove("x.text"),B+rremove("x.text"),C,D, ncol=2,align="hv",labels = "AUTO",label_x=.1,label_y=1,hjust=-3, label_size=22) +# +# +# +# +# +# +# +# + +### Figure 4 +Asy1<-ggplot(WEB,aes(Area,Asy, color=Condition, fill=Condition))+ + geom_point(size=2)+ + labs(color="Legend text")+ + geom_smooth(method='lm', se=F)+ + xlab(expression(paste('\nWeb catch area ',(cm^2))))+ + ylab("Vertical web asymmetry\n")+ + guides(fill=FALSE)+ + theme_classic()+ + geom_hline(yintercept=0,linetype=2)+ + theme(axis.text=element_text(size=12), + axis.title=element_text(size=14), + legend.position = c(0.7, 0.7), + legend.text = element_text(size=14), + legend.title = element_blank()) + + +# +# +# +# +# + +### Figure 5 +prey1<-ggplot(WEB,aes(Area,Prey, color=Condition, fill=Condition))+ + geom_point(size=2)+ + labs(color="Legend text")+ + geom_smooth(method="lm", alpha=.1, formula=y~x, size=1.1, + fullrange=F, se=F)+ + xlab(expression(paste('Web catch area ',(cm^2))))+ + ylab("Number of captured prey per web\n")+ + guides(fill=FALSE)+ + theme_classic()+ + scale_y_continuous(breaks=seq(0,100,by=10))+ + theme(axis.text=element_text(size=14), + axis.title=element_text(size=18), + legend.position = c(0.85, 0.8), + legend.text = element_text(size=20), + legend.title = element_blank()) + +grob <- grobTree(textGrob("*", x=0.475, y=0.9, hjust=0, + gp=gpar(col="black", fontsize=36, fontface="bold"))) + +prey2<-ggboxplot(data=WEB, x = "Condition", y = "Prey", fill = "Condition", order=c("No Light","Light"),palette = c("#00AFBB", "#FC4E07"), + ylab ="", xlab = "", legend="FALSE")+ + scale_y_continuous(breaks=seq(0,100,by=10))+ + theme(axis.text.x=element_text(size=12), + axis.title.x=element_text(size=14), + axis.text.y = element_blank(), + axis.title.y = element_blank())+ + annotation_custom(grob) + +plot_grid(prey1,prey2, align="hv") diff --git a/TOURS.csv b/TOURS.csv new file mode 100644 index 0000000..fba3646 --- /dev/null +++ b/TOURS.csv @@ -0,0 +1,89 @@ +Light,Spiders +0,23 +0,19 +1,15 +1,7 +0,10 +1,6 +0,7 +1,4 +0,7 +1,10 +0,21 +1,24 +0,15 +1,21 +0,9 +1,14 +0,20 +0,20 +1,5 +0,5 +1,6 +0,16 +1,11 +1,6 +0,15 +1,7 +0,7 +1,5 +0,3 +1,4 +0,15 +1,17 +1,4 +0,28 +1,10 +0,7 +1,5 +0,16 +1,10 +0,10 +1,9 +0,14 +1,16 +0,9 +1,12 +0,11 +1,11 +0,6 +1,11 +0,18 +0,16 +1,6 +0,18 +1,6 +0,5 +1,4 +0,19 +1,5 +0,9 +1,5 +0,15 +1,13 +0,27 +1,13 +0,14 +1,10 +0,16 +1,9 +0,11 +1,7 +0,10 +1,12 +0,20 +1,14 +0,18 +1,14 +0,13 +1,13 +0,18 +0,9 +1,4 +1,15 +0,13 +1,15 +0,12 +1,12 +0,22 +1,17 diff --git a/webs.csv b/webs.csv new file mode 100644 index 0000000..82fac2d --- /dev/null +++ b/webs.csv @@ -0,0 +1,87 @@ +Condition,Dv,Dh,Dfz,Rt,Prey +Light,16,19,4,8,NaN +No Light,19,27,4.5,7.5,NaN +Light,12,13,4.5,5.5,NaN +No Light,25,22,5,8,NaN +Light,17,11,2.5,8,NaN +No Light,27,26,6.5,9,NaN +Light,13,12.5,4,6,NaN +No Light,32,23,6,13,NaN +Light,12,13.5,4,5,NaN +No Light,32,25,6,11,NaN +Light,16,15.5,4.5,5,NaN +No Light,31,32,6,9,NaN +Light,14.5,10.5,2.5,5.5,NaN +No Light,18,18.5,4,6,NaN +Light,16,15,4.5,7,NaN +No Light,28,22,6,18,NaN +Light,36,30,8.5,10,NaN +No Light,37,39,5.5,11,NaN +Light,17,23,4,6,NaN +No Light,38,22,6,11,NaN +Light,29,22,4.5,12,NaN +No Light,36,27,6.5,18,NaN +Light,33,28,6,14,NaN +No Light,24,21,4.5,12,NaN +Light,16,15,4,8.5,NaN +No Light,39,35,6,13,NaN +Light,26,25,5.5,12,NaN +No Light,30,26,6.5,12,NaN +Light,45,37,8,19,NaN +No Light,43,33,6,13,NaN +Light,19,18,5,8,NaN +No Light,40,41,5,14,NaN +No Light,39,24,9,21,6 +No Light,22,24,5,11.5,5 +Light,24,18,6,9,28 +Light,38,27,6,15,41 +No Light,34,23,5,16,3 +No Light,22,22,4,11,2 +Light,36,30,8,15,67 +Light,35,34,6,16,80 +No Light,25,24,5,11,2 +No Light,25,20,4,7,1 +Light,14,14,4,8,32 +Light,47,35,5,17,54 +No Light,38,35,5,10,4 +No Light,26,22,6,9,3 +Light,23,17,4,7,16 +Light,23,17,4,7,52 +Light,30,22,5,10,25 +Light,24,24,4.5,10,13 +No Light,40,26,5,18,23 +No Light,28,27,5,11,17 +No Light,21,11,3.5,7,4 +No Light,22,23,5,11,12 +Light,21,16,4.5,11,48 +Light,22,13,4,11,24 +Light,19,20,5,9,62 +Light,25,16,4,10,81 +No Light,31,27,5,16,4 +No Light,38,26,6,17,12 +Light,13,10,3.5,6,19 +Light,22,33,5,16,22 +No Light,28,21,4,10,3 +No Light,15,18,3,6,2 +No Light,21,20,4,5,4 +Light,25,19,4,12,6 +Light,27,20,6.5,12,43 +No Light,36,26,5.5,13,9 +Light,22,19,5,9,57 +No Light,26,20,4,9,11 +Light,25,22,6,10,32 +No Light,34,28,4.5,14,3 +Light,23,21,5,10,14 +No Light,25,21,4.5,10,3 +No Light,51,43,8,21,31 +Light,33,27,6,13,NA +No Light,44,37,6,18,19 +Light,27,19,4,13,12 +No Light,33,30,5,12,7 +Light,27,25,4,11,27 +No Light,35,30,6,16,7 +Light,37,23,5,17,30 +No Light,39,32,6,13,10 +No Light,30,21,4,12,3 +Light,28,22,5,10,112 +Light,24,16,5,14,22