---
title: "Workflow for thermal decay deconvolution with mixchar"
output:
html_document:
css: mixchar.css
fig_caption: yes
theme: lumen
toc: yes
toc_float:
collapsed: no
toc_depth: 4
vignette: >
%\VignetteIndexEntry{Workflow for thermal decay deconvolution with mixchar}
%\VignetteEngine{knitr::rmarkdown}
%\usepackage[utf8]{inputenc}
---
```{r setup, include=FALSE}
knitr::opts_chunk$set(echo = TRUE)
```
## 1. Load data
The first step is to load the data derived from the thermogravimetric analysis into R. This step will vary depending on the format of your exported data. Two of many options for reading data are `read.csv()` for .csv files and `read.table()` for .txt files. Both of these functions also have a `skip` argument, in which you can tell R to begin reading data from a certain line, useful if the exported data has several lines of metadata at the top. You may also then have to assign column names. For example:
``` {r, eval = FALSE}
my_data <- read.csv('your_file_path_here.csv', header = FALSE, skip = 15)
colnames(c('temperature_C', 'mass_mg'))
```
More information about reading data into R can be found [here](http://www.sthda.com/english/wiki/reading-data-from-txt-csv-files-r-base-functions).
This package was developed using data from a Netzsch TGA-FTIR thermogravimetric analyser. We have included two example datasets in this package, called `juncus` and `marsilea`, that we will use for this demonstration. The `juncus` dataset contains theromogravimetric data for a sample of the freshwater reed, *Juncus amabilis*, and `marsilea` contains data for the freshwater fern, *Marsilea drumondii*.
``` {r}
library(mixchar)
head(juncus)
```
## 2. `process` data
After we've loaded our data, can use the function `process()` to take the derivative of this data, resulting in rate of mass loss over temperature data. This produces the multi-peaked rate of decay curve that we will apply the mixture model to. The `process` function needs the dataset, the initial mass of sample, the name of the temperature data column, and the name of your mass column (mg). Mass might be recorded in your exported dataset in different ways:
* If you have *mass loss of sample*, such as in the `juncus` dataset, you can specify the column name using the `mass_loss` argument.
* If you have *mass remaining of sample*, you specify the column name using the `mass` argument.
* **You only need to specify one of these columns for the `process` function.** But make sure the argument matches the kind of data!
The function defaults to temperature data in Celsius, but you can also modify to indicate the data is provided in Kelvin, by specifying the argument `temp_units = 'K'`.
```{r}
deriv_juncus <- process(juncus, # dataframe name
init_mass = 18.96,
temp = 'temp_C', # column name for temperature
mass_loss = 'mass_loss') # column name for mass loss data
deriv_juncus
```
```{r}
deriv_marsilea <- process(marsilea,
init_mass = 15.29,
temp = 'temp_C',
mass_loss = 'mass_loss')
deriv_marsilea
```
## 3. Plot mass loss and rate of mass loss curves
At this point we can take a look at what we've done so far using the default plotting option for the processed data. If you `plot()` the output of the `process` function, you will get two curves: the mass of sample across time and the rate of mass loss curve. If you're only interested in one plot, you can specify `plot_type = 'mass'` or `plot_type = 'rate'`. You can also specify a size factor using `cex`.
The rate of mass loss curve helps us to visualise the three stages of mass loss:
1. A short period with of dehydration, up until approximately 120 $^{\circ}$C.
2. A wide mid-range of high mass loss, caused by devolatilisation of primary biomass carbon components, between approximately 120 -- 650 $^{\circ}$C.
3. A final period of little mass loss when carbonaceous material associated with the inorganic fraction decomposes, after approximately 650 $^{\circ}$C.
In the next step we will crop the data to include only phase 2, so visualising your own data is important to check that the default temperature bounds will be suitable, in case for example the dehydration phase extends past 120 $^{\circ}$C or ends earlier. Comparing the plots for the two species we can see similarities in the shape and location of the peaks of the overall rate of mass loss curve, but also subtle differences. It is these characteristics we will tease apart using the nonlinear mixture model in the next step.
```{r, fig.height = 3.7}
plot(deriv_juncus, cex = 0.9)
```
```{r, fig.height = 3.7}
plot(deriv_marsilea, cex = 0.9)
```
## 4. `deconvolve` data
The `decovolve` function takes care of modelling the rate of mass loss data with the nonlinear mixture model. To do so it first crops to the second phase, as mentioned above, to default temperature bounds of 120 $^{\circ}$C and 700 $^{\circ}$C. These can be modified with the `lower_temp` and `upper_temp` arguments. Although most biomass samples have only three main components (corresponding to hemicellulose, cellulose, and lignin), some have a second hemicellulose peak in the low temperature range. `deconvolve` will decide whether three or four peaks are best using an internal function that determines if there is a peak below 220 $^{\circ}$C. Upon inspection of your curve you can override this by modifying the `n_peaks` argument. The function also has built in starting values for the nonlinear optimisation. These values were tested on litter from 30 plant species, encompassing herbaceous, graminoid, as well as woody species. However, if they do not work for your sample, or you would like to play with the effect of changing them, you can do so with the `start_vec`, `lower_vec`, and `upper_vec` arguments.
```{r}
output_juncus <- deconvolve(deriv_juncus)
output_juncus
```
```{r}
# here's an example of specifying your starting vector.
# the order of values is height, skew, position, then width
my_starting_vec <- c(height_0 = 0.002, skew_0 = -0.15, position_0 = 210, width_0 = 50,
height_1 = 0.003, skew_1 = -0.15, position_1 = 250, width_1 = 50,
height_2 = 0.006, skew_2 = -0.15, position_2 = 320, width_2 = 30,
height_3 = 0.001, skew_3 = -0.15, position_3 = 390, width_3 = 200)
output_marsilea <- deconvolve(deriv_marsilea, n_peaks = 4, start_vec = my_starting_vec)
output_marsilea
```
## 5. Examine outputs
`deconvolve` results in a few different outputs that you can retrieve with accessor functions.
* `rate_data()` will show you the processed dataset that results from the `process` function, useful if you want to play around with other modelling approaches or plotting options:
```{r}
juncus_rate <- rate_data(output_juncus)
head(juncus_rate)
```
*`temp_bounds()` will print the temperature values at which the data were cropped for analysis:
```{r}
temp_bounds(output_juncus)
```
* `model_fit()` will show you the output of the mixture model and estimated parameter values for height, skew, position, and width for each peak. Peak 1 is hemicellulose, peak 2 is cellulose, and peak 3 is lignin. If present, the optional fourth peak located at the lowest temperature interval will be listed as peak 0:
```{r}
model_fit(output_juncus)
```
```{r}
model_fit(output_marsilea)
```
* `component_weights()` will display the mean, upper, and lower confidence intervals for each estimated component:
```{r}
component_weights(output_juncus)
```
```{r}
component_weights(output_marsilea)
```
## 6. Plot deconvolved curves
The default plotting function for the output of the `deconvolve` function shows you your raw mass data, the estimated full curve from the mixture model, and also plots the individual component peaks using their parameter estimates from the model. The default plot is in black and white, but you can plot a colour version that uses colour-blind friendly viridis colours by specifying `bw = FALSE`.
Take a look at your plot. Do the estimates seem reasonable?
```{r, out.width = '80%'}
plot(output_juncus, bw = FALSE)
```
```{r, out.width = '80%'}
plot(output_marsilea, bw = FALSE)
```
If the estimated peaks do not match your data well, it may be that your sample is a bit different than those used to develop the package. That's not a problem, but you may need to try the deconvolution again with different starting values. If you aren't managing to produce reasonable estimates, get in touch with us via the [issues page](http://www.github.com/smwindecker/mixchar/issues).
If you want to modify the aesthetics of this plot for your own work, then you can access the parameter estimates as follows:
```{r}
juncus_parameters <- model_parameters(output_juncus)
```
and use `fs_function()` to plot individual component peaks, and `fs_mixture` function to plot the overall mixture curve.
To read more about the methodology, check out the Detailed Methodology [article](http://smwindecker.github.io/mixchar/articles/Background.html).