Skip to contents

Common usage

A minimal call of mmrm(), consisting of only formula and data arguments will produce an object of class mmrm, mmrm_fit, and mmrm_tmb.

Here we fit a mmrm model with us (unstructured) covariance structure specified, as well as the defaults of reml = TRUE and control = mmrm_control().

library(mmrm)
fit <- mmrm(
  formula = FEV1 ~ RACE + SEX + ARMCD * AVISIT + us(AVISIT | USUBJID),
  data = fev_data
)

The code specifies an MMRM with the given covariates and an unstructured covariance matrix for the timepoints (also called visits in the clinical trial context, here given by AVISIT) within the subjects (here USUBJID). While by default this uses restricted maximum likelihood (REML), it is also possible to use ML, see ?mmrm.

Printing the object will show you output which should be familiar to anyone who has used any popular modeling functions such as stats::lm(), stats::glm(), glmmTMB::glmmTMB(), and lme4::nlmer(). From this print out we see the function call, the data used, the covariance structure with number of variance parameters, as well as the likelihood method, and model deviance achieved. Additionally the user is provided a printout of the estimated coefficients and the model convergence information:

fit
#> mmrm fit
#> 
#> Formula:     FEV1 ~ RACE + SEX + ARMCD * AVISIT + us(AVISIT | USUBJID)
#> Data:        fev_data (used 537 observations from 197 subjects with maximum 4 
#> timepoints)
#> Covariance:  unstructured (10 variance parameters)
#> Inference:   REML
#> Deviance:    3386.45
#> 
#> Coefficients: 
#>                   (Intercept) RACEBlack or African American 
#>                   30.77747548                    1.53049977 
#>                     RACEWhite                     SEXFemale 
#>                    5.64356535                    0.32606192 
#>                      ARMCDTRT                    AVISITVIS2 
#>                    3.77423004                    4.83958845 
#>                    AVISITVIS3                    AVISITVIS4 
#>                   10.34211288                   15.05389826 
#>           ARMCDTRT:AVISITVIS2           ARMCDTRT:AVISITVIS3 
#>                   -0.04192625                   -0.69368537 
#>           ARMCDTRT:AVISITVIS4 
#>                    0.62422703 
#> 
#> Model Inference Optimization:
#> Converged with code 0 and message: convergence: rel_reduction_of_f <= factr*epsmch

The summary() method then provides the coefficients table with Satterthwaite degrees of freedom as well as the covariance matrix estimate:

summary(fit)
#> mmrm fit
#> 
#> Formula:     FEV1 ~ RACE + SEX + ARMCD * AVISIT + us(AVISIT | USUBJID)
#> Data:        fev_data (used 537 observations from 197 subjects with maximum 4 
#> timepoints)
#> Covariance:  unstructured (10 variance parameters)
#> Method:      Satterthwaite
#> Vcov Method: Asymptotic
#> Inference:   REML
#> 
#> Model selection criteria:
#>      AIC      BIC   logLik deviance 
#>   3406.4   3439.3  -1693.2   3386.4 
#> 
#> Coefficients: 
#>                                Estimate Std. Error        df t value Pr(>|t|)
#> (Intercept)                    30.77748    0.88656 218.80000  34.715  < 2e-16
#> RACEBlack or African American   1.53050    0.62448 168.67000   2.451 0.015272
#> RACEWhite                       5.64357    0.66561 157.14000   8.479 1.56e-14
#> SEXFemale                       0.32606    0.53195 166.13000   0.613 0.540744
#> ARMCDTRT                        3.77423    1.07415 145.55000   3.514 0.000589
#> AVISITVIS2                      4.83959    0.80172 143.88000   6.037 1.27e-08
#> AVISITVIS3                     10.34211    0.82269 155.56000  12.571  < 2e-16
#> AVISITVIS4                     15.05390    1.31281 138.47000  11.467  < 2e-16
#> ARMCDTRT:AVISITVIS2            -0.04193    1.12932 138.56000  -0.037 0.970439
#> ARMCDTRT:AVISITVIS3            -0.69369    1.18765 158.17000  -0.584 0.559996
#> ARMCDTRT:AVISITVIS4             0.62423    1.85085 129.72000   0.337 0.736463
#>                                  
#> (Intercept)                   ***
#> RACEBlack or African American *  
#> RACEWhite                     ***
#> SEXFemale                        
#> ARMCDTRT                      ***
#> AVISITVIS2                    ***
#> AVISITVIS3                    ***
#> AVISITVIS4                    ***
#> ARMCDTRT:AVISITVIS2              
#> ARMCDTRT:AVISITVIS3              
#> ARMCDTRT:AVISITVIS4              
#> ---
#> Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
#> 
#> Covariance estimate:
#>         VIS1    VIS2    VIS3    VIS4
#> VIS1 40.5537 14.3960  4.9747 13.3867
#> VIS2 14.3960 26.5715  2.7855  7.4745
#> VIS3  4.9747  2.7855 14.8979  0.9082
#> VIS4 13.3867  7.4745  0.9082 95.5568

Common customizations

From the high-level mmrm() interface, common changes to the default function call can be specified.

Control Function

For fine control, mmrm_control() is provided. This function allows the user to choose the adjustment method for the degrees of freedom and the coefficients covariance matrix, specify optimization routines, number of cores to be used on Unix systems for trying several optimizers in parallel, provide a vector of starting parameter values, decide the action to be taken when the defined design matrix is singular, not drop unobserved visit levels. For example:

mmrm_control(
  method = "Kenward-Roger",
  optimizer = c("L-BFGS-B", "BFGS"),
  n_cores = 2,
  start = c(0, 1, 1, 0, 1, 0),
  accept_singular = FALSE,
  drop_visit_levels = FALSE
)

Note that this control list can either be passed via the control argument to mmrm, or selected controls can be directly specified in the mmrm call. We will see this below.

REML or ML

Users can specify if REML should be used (default) or if ML should be used in optimization.

fit_ml <- mmrm(
  formula = FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID),
  data = fev_data,
  reml = FALSE
)
fit_ml
#> mmrm fit
#> 
#> Formula:     FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID)
#> Data:        fev_data (used 537 observations from 197 subjects with maximum 4 
#> timepoints)
#> Covariance:  unstructured (10 variance parameters)
#> Inference:   ML
#> Deviance:    3397.934
#> 
#> Coefficients: 
#>                   (Intercept) RACEBlack or African American 
#>                    30.9663423                     1.5086851 
#>                     RACEWhite                      ARMCDTRT 
#>                     5.6133151                     3.7761037 
#>                    AVISITVIS2                    AVISITVIS3 
#>                     4.8270155                    10.3353319 
#>                    AVISITVIS4           ARMCDTRT:AVISITVIS2 
#>                    15.0487715                    -0.0156154 
#>           ARMCDTRT:AVISITVIS3           ARMCDTRT:AVISITVIS4 
#>                    -0.6663598                     0.6317222 
#> 
#> Model Inference Optimization:
#> Converged with code 0 and message: convergence: rel_reduction_of_f <= factr*epsmch

Optimizer

Users can specify which optimizer should be used, changing from the default of four optimizers, which starts with L-BFGS-B and proceeds through the other choices if optimization fails to converge. Other choices are BFGS, CG, nlminb and other user-defined custom optimizers.

L-BFGS-B, BFGS and CG are all implemented with stats::optim() and the Hessian is not used, while nlminb is using stats::nlminb() which in turn uses both the gradient and the Hessian (by default but can be switch off) for the optimization.

fit_opt <- mmrm(
  formula = FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID),
  data = fev_data,
  optimizer = "BFGS"
)
fit_opt
#> mmrm fit
#> 
#> Formula:     FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID)
#> Data:        fev_data (used 537 observations from 197 subjects with maximum 4 
#> timepoints)
#> Covariance:  unstructured (10 variance parameters)
#> Inference:   REML
#> Deviance:    3387.373
#> 
#> Coefficients: 
#>                   (Intercept) RACEBlack or African American 
#>                    30.9676902                     1.5046744 
#>                     RACEWhite                      ARMCDTRT 
#>                     5.6131048                     3.7755423 
#>                    AVISITVIS2                    AVISITVIS3 
#>                     4.8285855                    10.3331770 
#>                    AVISITVIS4           ARMCDTRT:AVISITVIS2 
#>                    15.0525706                    -0.0173504 
#>           ARMCDTRT:AVISITVIS3           ARMCDTRT:AVISITVIS4 
#>                    -0.6675190                     0.6309586 
#> 
#> Model Inference Optimization:
#> Converged with code 0 and message:

Covariance Structure

Covariance structures supported by the mmrm are being continuously developed. For a complete list and description please visit the covariance vignette. Below we see the function call for homogeneous compound symmetry (cs).

fit_cs <- mmrm(
  formula = FEV1 ~ RACE + ARMCD * AVISIT + cs(AVISIT | USUBJID),
  data = fev_data,
  reml = FALSE
)
fit_cs
#> mmrm fit
#> 
#> Formula:     FEV1 ~ RACE + ARMCD * AVISIT + cs(AVISIT | USUBJID)
#> Data:        fev_data (used 537 observations from 197 subjects with maximum 4 
#> timepoints)
#> Covariance:  compound symmetry (2 variance parameters)
#> Inference:   ML
#> Deviance:    3536.989
#> 
#> Coefficients: 
#>                   (Intercept) RACEBlack or African American 
#>                    31.4207077                     0.5357237 
#>                     RACEWhite                      ARMCDTRT 
#>                     5.4546329                     3.4305212 
#>                    AVISITVIS2                    AVISITVIS3 
#>                     4.8326353                    10.2395076 
#>                    AVISITVIS4           ARMCDTRT:AVISITVIS2 
#>                    15.0672680                     0.2801641 
#>           ARMCDTRT:AVISITVIS3           ARMCDTRT:AVISITVIS4 
#>                    -0.5894964                     0.7939750 
#> 
#> Model Inference Optimization:
#> Converged with code 0 and message: convergence: rel_reduction_of_f <= factr*epsmch

The time points have to be unique for each subject. That is, there cannot be time points with multiple observations for any subject. The rationale is that these observations would need to be correlated, but it is not possible within the currently implemented covariance structure framework to do that correctly. Moreover, for non-spatial covariance structures, the time variable must be coded as a factor.

Weighting

Users can perform weighted MMRM by specifying a numeric vector weights with positive values.

fit_wt <- mmrm(
  formula = FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID),
  data = fev_data,
  weights = fev_data$WEIGHT
)
fit_wt
#> mmrm fit
#> 
#> Formula:     FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID)
#> Data:        fev_data (used 537 observations from 197 subjects with maximum 4 
#> timepoints)
#> Weights:     fev_data$WEIGHT
#> Covariance:  unstructured (10 variance parameters)
#> Inference:   REML
#> Deviance:    3476.526
#> 
#> Coefficients: 
#>                   (Intercept) RACEBlack or African American 
#>                   31.20065229                    1.18452837 
#>                     RACEWhite                      ARMCDTRT 
#>                    5.36525917                    3.39695951 
#>                    AVISITVIS2                    AVISITVIS3 
#>                    4.85890820                   10.03942420 
#>                    AVISITVIS4           ARMCDTRT:AVISITVIS2 
#>                   14.79354054                    0.03418184 
#>           ARMCDTRT:AVISITVIS3           ARMCDTRT:AVISITVIS4 
#>                    0.01308088                    0.86701567 
#> 
#> Model Inference Optimization:
#> Converged with code 0 and message: convergence: rel_reduction_of_f <= factr*epsmch

Grouped Covariance Structure

Grouped covariance structures are supported by themmrm package. Covariance matrices for each group are identically structured (unstructured, compound symmetry, etc) but the estimates are allowed to vary across groups. We use the form cs(time | group / subject) to specify the group variable.

Here is an example of how we use ARMCD as group variable.

fit_cs <- mmrm(
  formula = FEV1 ~ RACE + ARMCD * AVISIT + cs(AVISIT | ARMCD / USUBJID),
  data = fev_data,
  reml = FALSE
)
VarCorr(fit_cs)
#> $PBO
#>           VIS1      VIS2      VIS3      VIS4
#> VIS1 37.823638  3.601296  3.601296  3.601296
#> VIS2  3.601296 37.823638  3.601296  3.601296
#> VIS3  3.601296  3.601296 37.823638  3.601296
#> VIS4  3.601296  3.601296  3.601296 37.823638
#> 
#> $TRT
#>          VIS1     VIS2     VIS3     VIS4
#> VIS1 49.58110 10.98112 10.98112 10.98112
#> VIS2 10.98112 49.58110 10.98112 10.98112
#> VIS3 10.98112 10.98112 49.58110 10.98112
#> VIS4 10.98112 10.98112 10.98112 49.58110

We can see that the estimated covariance matrices are different in different ARMCD groups.

Adjustment Method

In additional to the residual and Between-Within degrees of freedom, both Satterthwaite and Kenward-Roger adjustment methods are available. The default is Satterthwaite adjustment of the degrees of freedom. To use e.g. the Kenward-Roger adjustment of the degrees of freedom as well as the coefficients covariance matrix, use the method argument:

A list of all allowed method is

  1. “Kenward-Roger”
  2. “Satterthwaite”
  3. “Residual”
  4. “Between-Within”
fit_kr <- mmrm(
  formula = FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID),
  data = fev_data,
  method = "Kenward-Roger"
)

Note that this requires reml = TRUE, i.e. Kenward-Roger adjustment is not possible when using maximum likelihood inference. While this adjustment choice is not visible in the print() result of the fitted model (because the initial model fit is not affected by the choice of the adjustment method), looking at the summary we see the method and the correspondingly adjusted standard errors and degrees of freedom:

summary(fit_kr)
#> mmrm fit
#> 
#> Formula:     FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID)
#> Data:        fev_data (used 537 observations from 197 subjects with maximum 4 
#> timepoints)
#> Covariance:  unstructured (10 variance parameters)
#> Method:      Kenward-Roger
#> Vcov Method: Kenward-Roger
#> Inference:   REML
#> 
#> Model selection criteria:
#>      AIC      BIC   logLik deviance 
#>   3407.4   3440.2  -1693.7   3387.4 
#> 
#> Coefficients: 
#>                                Estimate Std. Error        df t value Pr(>|t|)
#> (Intercept)                    30.96770    0.83335 187.91000  37.160  < 2e-16
#> RACEBlack or African American   1.50465    0.62901 169.95000   2.392  0.01784
#> RACEWhite                       5.61310    0.67139 158.87000   8.360 2.98e-14
#> ARMCDTRT                        3.77556    1.07910 146.27000   3.499  0.00062
#> AVISITVIS2                      4.82859    0.80408 143.66000   6.005 1.49e-08
#> AVISITVIS3                     10.33317    0.82303 155.66000  12.555  < 2e-16
#> AVISITVIS4                     15.05256    1.30180 138.39000  11.563  < 2e-16
#> ARMCDTRT:AVISITVIS2            -0.01737    1.13154 138.39000  -0.015  0.98777
#> ARMCDTRT:AVISITVIS3            -0.66753    1.18714 158.21000  -0.562  0.57470
#> ARMCDTRT:AVISITVIS4             0.63094    1.83319 129.64000   0.344  0.73127
#>                                  
#> (Intercept)                   ***
#> RACEBlack or African American *  
#> RACEWhite                     ***
#> ARMCDTRT                      ***
#> AVISITVIS2                    ***
#> AVISITVIS3                    ***
#> AVISITVIS4                    ***
#> ARMCDTRT:AVISITVIS2              
#> ARMCDTRT:AVISITVIS3              
#> ARMCDTRT:AVISITVIS4              
#> ---
#> Signif. codes:  0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
#> 
#> Covariance estimate:
#>         VIS1    VIS2    VIS3    VIS4
#> VIS1 40.7335 14.2740  5.1411 13.5288
#> VIS2 14.2740 26.2243  2.6391  7.3219
#> VIS3  5.1411  2.6391 14.9497  1.0341
#> VIS4 13.5288  7.3219  1.0341 95.6006

For one-dimensional contrasts as in the coefficients table above, the degrees of freedom are the same for Kenward-Roger and Satterthwaite. However, Kenward-Roger uses adjusted standard errors, hence the p-values are different.

Note that if you would like to match SAS results for an unstructured covariance model, you can use the linear Kenward-Roger approximation:

fit_kr_lin <- mmrm(
  formula = FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID),
  data = fev_data,
  method = "Kenward-Roger",
  vcov = "Kenward-Roger-Linear"
)

This is due to the different parametrization of the unstructured covariance matrix, see the Kenward-Roger vignette for details.

Variance-covariance for Coefficients

There are multiple variance-covariance estimator available for the coefficients, including:

  1. “Asymptotic”
  2. “Empirical” (Cluster Robust Sandwich)
  3. “Empirical-Jackknife”
  4. “Empirical-Bias-Reduced”
  5. “Kenward-Roger”
  6. “Kenward-Roger-Linear”

Please note that, not all combinations of variance-covariance for coefficients and method of degrees of freedom are possible, e.g. “Kenward-Roger” and “Kenward-Roger-Linear” are available only when the degrees of freedom method is “Kenward-Roger”.

Details can be found in Coefficients Covariance Matrix Adjustment vignette and Weighted Least Square Empirical Covariance.

An example of using other variance-covariance is:

fit_emp <- mmrm(
  formula = FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID),
  data = fev_data,
  method = "Satterthwaite",
  vcov = "Empirical"
)

Keeping Unobserved Visits

Sometimes not all possible time points are observed in a given data set. When using a structured covariance matrix, e.g. with auto-regressive structure, then it can be relevant to keep the correct distance between the observed time points.

Consider the following example where we have deliberately removed the VIS3 observations from our initial example data set fev_data to obtain sparse_data. We first fit the model where we do not drop the visit level explicitly, using the drop_visit_levels = FALSE choice. Second we fit the model by default without this option.

sparse_data <- fev_data[fev_data$AVISIT != "VIS3", ]
sparse_result <- mmrm(
  FEV1 ~ RACE + ar1(AVISIT | USUBJID),
  data = sparse_data,
  drop_visit_levels = FALSE
)

dropped_result <- mmrm(
  FEV1 ~ RACE + ar1(AVISIT | USUBJID),
  data = sparse_data
)
#> In AVISIT there are dropped visits: VIS3

We see that we get a message about the dropped visit level by default. Now we can compare the estimated correlation matrices:

cov2cor(VarCorr(sparse_result))
#>            VIS1      VIS2      VIS3       VIS4
#> VIS1 1.00000000 0.4051834 0.1641736 0.06652042
#> VIS2 0.40518341 1.0000000 0.4051834 0.16417360
#> VIS3 0.16417360 0.4051834 1.0000000 0.40518341
#> VIS4 0.06652042 0.1641736 0.4051834 1.00000000
cov2cor(VarCorr(dropped_result))
#>            VIS1      VIS2       VIS4
#> VIS1 1.00000000 0.1468464 0.02156386
#> VIS2 0.14684640 1.0000000 0.14684640
#> VIS4 0.02156386 0.1468464 1.00000000

We see that when using the default, second result, we just drop the VIS3 from the covariance matrix. As a consequence, we model the correlation between VIS2 and VIS4 the same as the correlation between VIS1 and VIS2. Hence we get a smaller correlation estimate here compared to the first result, which includes VIS3 explicitly.

Extraction of model features

Similar to model objects created in other packages, components of mmrm and mmrm_tmb objects can be accessed with standard methods. Additionally, component() is provided to allow deeper and more precise access for those interested in digging through model output. Complete documentation of standard model output methods supported for mmrm_tmb objects can be found at the package website.

Summary

The summary method for mmrm objects provides easy access to frequently needed model components.

fit <- mmrm(
  formula = FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID),
  data = fev_data
)
fit_summary <- summary(fit)

From this summary object, you can easily retrieve the coefficients table.

fit_summary$coefficients
#>                                  Estimate Std. Error       df     t value
#> (Intercept)                   30.96769899  0.8293349 187.9132 37.34040185
#> RACEBlack or African American  1.50464863  0.6206596 169.9454  2.42427360
#> RACEWhite                      5.61309565  0.6630909 158.8700  8.46504747
#> ARMCDTRT                       3.77555734  1.0762774 146.2690  3.50797778
#> AVISITVIS2                     4.82858803  0.8017144 143.6593  6.02282805
#> AVISITVIS3                    10.33317002  0.8224414 155.6572 12.56401918
#> AVISITVIS4                    15.05255715  1.3128602 138.3916 11.46546844
#> ARMCDTRT:AVISITVIS2           -0.01737409  1.1291645 138.3926 -0.01538668
#> ARMCDTRT:AVISITVIS3           -0.66753189  1.1865359 158.2106 -0.56258887
#> ARMCDTRT:AVISITVIS4            0.63094392  1.8507884 129.6377  0.34090549
#>                                   Pr(>|t|)
#> (Intercept)                   7.122411e-89
#> RACEBlack or African American 1.638725e-02
#> RACEWhite                     1.605553e-14
#> ARMCDTRT                      6.001485e-04
#> AVISITVIS2                    1.366921e-08
#> AVISITVIS3                    1.927523e-25
#> AVISITVIS4                    8.242709e-22
#> ARMCDTRT:AVISITVIS2           9.877459e-01
#> ARMCDTRT:AVISITVIS3           5.745112e-01
#> ARMCDTRT:AVISITVIS4           7.337266e-01

Other model parameters and metadata available in the summary object is as follows:

str(fit_summary)
#> List of 15
#>  $ cov_type        : chr "us"
#>  $ reml            : logi TRUE
#>  $ n_groups        : int 1
#>  $ n_theta         : int 10
#>  $ n_subjects      : int 197
#>  $ n_timepoints    : int 4
#>  $ n_obs           : int 537
#>  $ beta_vcov       : num [1:10, 1:10] 0.688 -0.207 -0.163 -0.569 -0.422 ...
#>   ..- attr(*, "dimnames")=List of 2
#>   .. ..$ : chr [1:10] "(Intercept)" "RACEBlack or African American" "RACEWhite" "ARMCDTRT" ...
#>   .. ..$ : chr [1:10] "(Intercept)" "RACEBlack or African American" "RACEWhite" "ARMCDTRT" ...
#>  $ varcor          : num [1:4, 1:4] 40.73 14.27 5.14 13.53 14.27 ...
#>   ..- attr(*, "dimnames")=List of 2
#>   .. ..$ : chr [1:4] "VIS1" "VIS2" "VIS3" "VIS4"
#>   .. ..$ : chr [1:4] "VIS1" "VIS2" "VIS3" "VIS4"
#>  $ method          : chr "Satterthwaite"
#>  $ vcov            : chr "Asymptotic"
#>  $ coefficients    : num [1:10, 1:5] 30.97 1.5 5.61 3.78 4.83 ...
#>   ..- attr(*, "dimnames")=List of 2
#>   .. ..$ : chr [1:10] "(Intercept)" "RACEBlack or African American" "RACEWhite" "ARMCDTRT" ...
#>   .. ..$ : chr [1:5] "Estimate" "Std. Error" "df" "t value" ...
#>  $ n_singular_coefs: int 0
#>  $ aic_list        :List of 4
#>   ..$ AIC     : num 3407
#>   ..$ BIC     : num 3440
#>   ..$ logLik  : num -1694
#>   ..$ deviance: num 3387
#>  $ call            : language mmrm(formula = FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID), data = fev_data)
#>  - attr(*, "class")= chr "summary.mmrm"

Residuals

The residuals method for mmrm objects can be used to provide three different types of residuals:

  1. Response or raw residuals - the difference between the observed and fitted or predicted value. MMRMs can allow for heteroscedasticity, so these residuals should be interpreted with caution.
fit <- mmrm(
  formula = FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID),
  data = fev_data
)
residuals_resp <- residuals(fit, type = "response")
  1. Pearson residuals - the raw residuals scaled by the estimated standard deviation of the response. This residual type is better suited to identifying outlying observations and the appropriateness of the covariance structure, compared to the raw residuals.
residuals_pearson <- residuals(fit, type = "pearson")
  1. Normalized or scaled residuals - the raw residuals are ‘de-correlated’ based on the Cholesky decomposition of the variance-covariance matrix. These residuals should approximately follow the standard normal distribution, and therefore can be used to check for normality (@galecki2013linear).
residuals_norm <- residuals(fit, type = "normalized")

broom extensions

mmrm also contains S3 methods methods for tidy, glance and augment which were introduced by broom. Note that these methods will work also without loading the broom package. Please see ?mmrm_tidiers for the detailed documentation.

For example, we can apply the tidy method to return a summary table of coefficient estimates:

fit |>
  tidy()
#> # A tibble: 10 × 6
#>    term                          estimate std.error    df statistic  p.value
#>    <chr>                            <dbl>     <dbl> <dbl>     <dbl>    <dbl>
#>  1 (Intercept)                    31.0        0.829  188.   37.3    7.12e-89
#>  2 RACEBlack or African American   1.50       0.621  170.    2.42   1.64e- 2
#>  3 RACEWhite                       5.61       0.663  159.    8.47   1.61e-14
#>  4 ARMCDTRT                        3.78       1.08   146.    3.51   6.00e- 4
#>  5 AVISITVIS2                      4.83       0.802  144.    6.02   1.37e- 8
#>  6 AVISITVIS3                     10.3        0.822  156.   12.6    1.93e-25
#>  7 AVISITVIS4                     15.1        1.31   138.   11.5    8.24e-22
#>  8 ARMCDTRT:AVISITVIS2            -0.0174     1.13   138.   -0.0154 9.88e- 1
#>  9 ARMCDTRT:AVISITVIS3            -0.668      1.19   158.   -0.563  5.75e- 1
#> 10 ARMCDTRT:AVISITVIS4             0.631      1.85   130.    0.341  7.34e- 1

We can also specify some details to request confidence intervals of specific confidence level:

fit |>
  tidy(conf.int = TRUE, conf.level = 0.9)
#> # A tibble: 10 × 8
#>    term           estimate std.error    df statistic  p.value conf.low conf.high
#>    <chr>             <dbl>     <dbl> <dbl>     <dbl>    <dbl>    <dbl>     <dbl>
#>  1 (Intercept)     31.0        0.829  188.   37.3    7.12e-89   29.6       32.3 
#>  2 ARMCDTRT         3.78       1.08   146.    3.51   6.00e- 4    2.01       5.55
#>  3 ARMCDTRT:AVIS…  -0.0174     1.13   138.   -0.0154 9.88e- 1   -1.87       1.84
#>  4 ARMCDTRT:AVIS…  -0.668      1.19   158.   -0.563  5.75e- 1   -2.62       1.28
#>  5 ARMCDTRT:AVIS…   0.631      1.85   130.    0.341  7.34e- 1   -2.41       3.68
#>  6 AVISITVIS2       4.83       0.802  144.    6.02   1.37e- 8    3.51       6.15
#>  7 AVISITVIS3      10.3        0.822  156.   12.6    1.93e-25    8.98      11.7 
#>  8 AVISITVIS4      15.1        1.31   138.   11.5    8.24e-22   12.9       17.2 
#>  9 RACEBlack or …   1.50       0.621  170.    2.42   1.64e- 2    0.484      2.53
#> 10 RACEWhite        5.61       0.663  159.    8.47   1.61e-14    4.52       6.70

Or we can apply the glance method to return a summary table of goodness of fit statistics:

fit |>
  glance()
#> # A tibble: 1 × 4
#>     AIC   BIC logLik deviance
#>   <dbl> <dbl>  <dbl>    <dbl>
#> 1 3407. 3440. -1694.    3387.

Finally, we can use the augment method to return a merged tibble of the data, fitted values and residuals:

fit |>
  augment()
#> # A tibble: 537 × 8
#>    .rownames  FEV1 RACE                     ARMCD AVISIT USUBJID .fitted  .resid
#>        <dbl> <dbl> <fct>                    <fct> <fct>  <fct>     <dbl>   <dbl>
#>  1         2  40.0 Black or African Americ… TRT   VIS2   PT1        40.0  -1.09 
#>  2         4  20.5 Black or African Americ… TRT   VIS4   PT1        20.5 -31.4  
#>  3         6  31.5 Asian                    PBO   VIS2   PT2        31.5  -4.34 
#>  4         7  36.9 Asian                    PBO   VIS3   PT2        36.9  -4.42 
#>  5         8  48.8 Asian                    PBO   VIS4   PT2        48.8   2.79 
#>  6        10  36.0 Black or African Americ… PBO   VIS2   PT3        36.0  -1.31 
#>  7        12  37.2 Black or African Americ… PBO   VIS4   PT3        37.2 -10.4  
#>  8        13  33.9 Asian                    TRT   VIS1   PT4        33.9  -0.851
#>  9        14  33.7 Asian                    TRT   VIS2   PT4        33.7  -5.81 
#> 10        16  54.5 Asian                    TRT   VIS4   PT4        54.5   4.02 
#> # ℹ 527 more rows

Also here we can specify details for the prediction intervals and type of residuals via the arguments:

fit |>
  augment(interval = "confidence", type.residuals = "normalized")
#> # A tibble: 537 × 11
#>    .rownames  FEV1 RACE       ARMCD AVISIT USUBJID .fitted .lower .upper .se.fit
#>        <dbl> <dbl> <fct>      <fct> <fct>  <fct>     <dbl>  <dbl>  <dbl>   <dbl>
#>  1         2  40.0 Black or … TRT   VIS2   PT1        40.0   40.0   40.0       0
#>  2         4  20.5 Black or … TRT   VIS4   PT1        20.5   20.5   20.5       0
#>  3         6  31.5 Asian      PBO   VIS2   PT2        31.5   31.5   31.5       0
#>  4         7  36.9 Asian      PBO   VIS3   PT2        36.9   36.9   36.9       0
#>  5         8  48.8 Asian      PBO   VIS4   PT2        48.8   48.8   48.8       0
#>  6        10  36.0 Black or … PBO   VIS2   PT3        36.0   36.0   36.0       0
#>  7        12  37.2 Black or … PBO   VIS4   PT3        37.2   37.2   37.2       0
#>  8        13  33.9 Asian      TRT   VIS1   PT4        33.9   33.9   33.9       0
#>  9        14  33.7 Asian      TRT   VIS2   PT4        33.7   33.7   33.7       0
#> 10        16  54.5 Asian      TRT   VIS4   PT4        54.5   54.5   54.5       0
#> # ℹ 527 more rows
#> # ℹ 1 more variable: .resid <dbl>

Other components

Specific model quantities not supported by methods can be retrieved with the component() function. The default will output all supported components.

For example, a user may want information about convergence:

component(fit, name = c("convergence", "evaluations", "conv_message"))
#> $convergence
#> [1] 0
#> 
#> $evaluations
#> function gradient 
#>       17       17 
#> 
#> $conv_message
#> [1] "CONVERGENCE: REL_REDUCTION_OF_F <= FACTR*EPSMCH"

or the original low-level call:

component(fit, name = "call")
#> mmrm(formula = FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID), 
#>     data = fev_data)

the user could also ask for all provided components by not specifying the name argument.

Lower level functions

Low-level mmrm

The lower level function which is called by mmrm() is fit_mmrm(). This function is exported and can be used directly. It is similar to mmrm() but lacks some post-processing and support for Satterthwaite and Kenward-Roger calculations. It may be useful for other packages that want to fit the model without the adjustment calculations.

fit_mmrm(
  formula = FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID),
  data = fev_data,
  weights = rep(1, nrow(fev_data)),
  reml = TRUE,
  control = mmrm_control()
)
#> mmrm fit
#> 
#> Formula:     FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID)
#> Data:        fev_data (used 537 observations from 197 subjects with maximum 4 
#> timepoints)
#> Weights:     rep(1, nrow(fev_data))
#> Covariance:  unstructured (10 variance parameters)
#> Inference:   REML
#> Deviance:    3387.373
#> 
#> Coefficients: 
#>                   (Intercept) RACEBlack or African American 
#>                   30.96769899                    1.50464863 
#>                     RACEWhite                      ARMCDTRT 
#>                    5.61309565                    3.77555734 
#>                    AVISITVIS2                    AVISITVIS3 
#>                    4.82858803                   10.33317002 
#>                    AVISITVIS4           ARMCDTRT:AVISITVIS2 
#>                   15.05255715                   -0.01737409 
#>           ARMCDTRT:AVISITVIS3           ARMCDTRT:AVISITVIS4 
#>                   -0.66753189                    0.63094392 
#> 
#> Model Inference Optimization:
#> Converged with code 0 and message: convergence: rel_reduction_of_f <= factr*epsmch

Hypothesis testing

This package supports estimation of one- and multi-dimensional contrasts (t-test and F-test calculation) with the df_1d() and df_md() functions. Both functions utilize the chosen adjustment method from the initial mmrm call for the calculation of degrees of freedom and (for Kenward-Roger methods) the variance estimates for the test-statistics.

One-dimensional contrasts

Compute the test of a one-dimensional (vector) contrast for a mmrm object with Satterthwaite degrees of freedom.

fit <- mmrm(
  formula = FEV1 ~ RACE + SEX + ARMCD * AVISIT + us(AVISIT | USUBJID),
  data = fev_data
)

contrast <- numeric(length(component(fit, "beta_est")))
contrast[3] <- 1

df_1d(fit, contrast)
#> $est
#> [1] 5.643565
#> 
#> $se
#> [1] 0.6656093
#> 
#> $df
#> [1] 157.1382
#> 
#> $t_stat
#> [1] 8.478795
#> 
#> $p_val
#> [1] 1.564869e-14

This works similarly when choosing a Kenward-Roger adjustment:

fit_kr <- mmrm(
  formula = FEV1 ~ RACE + SEX + ARMCD * AVISIT + us(AVISIT | USUBJID),
  data = fev_data,
  method = "Kenward-Roger"
)

df_1d(fit_kr, contrast)
#> $est
#> [1] 5.643565
#> 
#> $se
#> [1] 0.6740941
#> 
#> $df
#> [1] 157.1382
#> 
#> $t_stat
#> [1] 8.372073
#> 
#> $p_val
#> [1] 2.931654e-14

We see that because this is a one-dimensional contrast, the degrees of freedoms are identical for Satterthwaite and Kenward-Roger. However, the standard errors are different and therefore the p-values are different.

Additional options for the degrees of freedom method are Residual and Between-Within.

Multi-dimensional contrasts

Compute the test of a multi-dimensional (matrix) contrast for the above defined mmrm object with Satterthwaite degrees of freedom:

contrast <- matrix(data = 0, nrow = 2, ncol = length(component(fit, "beta_est")))
contrast[1, 2] <- contrast[2, 3] <- 1

df_md(fit, contrast)
#> $num_df
#> [1] 2
#> 
#> $denom_df
#> [1] 165.5553
#> 
#> $f_stat
#> [1] 36.91143
#> 
#> $p_val
#> [1] 5.544575e-14

And for the Kenward-Roger adjustment:

df_md(fit_kr, contrast)
#> $num_df
#> [1] 2
#> 
#> $denom_df
#> [1] 165.5728
#> 
#> $f_stat
#> [1] 35.99422
#> 
#> $p_val
#> [1] 1.04762e-13

We see that for the multi-dimensional contrast we get slightly different denominator degrees of freedom for the two adjustment methods.

Also the simpler Residual and Between-Within method choices can be used of course together with multidimensional contrasts.

Support for emmeans

This package includes methods that allow mmrm objects to be used with the emmeans package. emmeans computes estimated marginal means (also called least-square means) for the coefficients of the MMRM.

fit <- mmrm(
  formula = FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID),
  data = fev_data
)

if (require(emmeans)) {
  emmeans(fit, ~ ARMCD | AVISIT)
}
#> Loading required package: emmeans
#> mmrm() registered as emmeans extension
#> AVISIT = VIS1:
#>  ARMCD emmean    SE  df lower.CL upper.CL
#>  PBO     33.3 0.757 149     31.8     34.8
#>  TRT     37.1 0.764 144     35.6     38.6
#> 
#> AVISIT = VIS2:
#>  ARMCD emmean    SE  df lower.CL upper.CL
#>  PBO     38.2 0.608 150     37.0     39.4
#>  TRT     41.9 0.598 146     40.7     43.1
#> 
#> AVISIT = VIS3:
#>  ARMCD emmean    SE  df lower.CL upper.CL
#>  PBO     43.7 0.462 131     42.8     44.6
#>  TRT     46.8 0.507 130     45.8     47.8
#> 
#> AVISIT = VIS4:
#>  ARMCD emmean    SE  df lower.CL upper.CL
#>  PBO     48.4 1.189 134     46.0     50.7
#>  TRT     52.8 1.188 133     50.4     55.1
#> 
#> Results are averaged over the levels of: RACE 
#> Confidence level used: 0.95

Note that the degrees of freedom choice is inherited here from the initial mmrm fit.

Tidymodels

Tidymodels

mmrm is compatible to work in a tidymodels workflow. The following is an example of how such a workflow would be constructed.

Direct fit

First we define the direct method to fit an mmrm model using the parsnip package functions linear_reg() and set_engine().

  • linear_reg() defines a model that can predict numeric values from predictors using a linear function
  • set_engine() is used to specify which package or system will be used to fit the model, along with any arguments specific to that software. We can set the method to adjust degrees of freedom directly in the call.
model <- linear_reg() |>
  set_engine("mmrm", method = "Satterthwaite") |>
  fit(FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID), fev_data)
model
#> parsnip model object
#> 
#> mmrm fit
#> 
#> Formula:     FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID)
#> Data:        data (used 537 observations from 197 subjects with maximum 4 
#> timepoints)
#> Weights:     weights
#> Covariance:  unstructured (10 variance parameters)
#> Inference:   REML
#> Deviance:    3387.373
#> 
#> Coefficients: 
#>                   (Intercept) RACEBlack or African American 
#>                   30.96769899                    1.50464863 
#>                     RACEWhite                      ARMCDTRT 
#>                    5.61309565                    3.77555734 
#>                    AVISITVIS2                    AVISITVIS3 
#>                    4.82858803                   10.33317002 
#>                    AVISITVIS4           ARMCDTRT:AVISITVIS2 
#>                   15.05255715                   -0.01737409 
#>           ARMCDTRT:AVISITVIS3           ARMCDTRT:AVISITVIS4 
#>                   -0.66753189                    0.63094392 
#> 
#> Model Inference Optimization:
#> Converged with code 0 and message: convergence: rel_reduction_of_f <= factr*epsmch

We can also pass in the full mmrm_control object into the set_engine() call:

model_with_control <- linear_reg() |>
  set_engine("mmrm", control = mmrm_control(method = "Satterthwaite")) |>
  fit(FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID), fev_data)

Predictions

Lastly, we can also obtain predictions with the predict() method:

predict(model, new_data = fev_data)
#> # A tibble: 800 × 1
#>    .pred
#>    <dbl>
#>  1  32.5
#>  2  40.0
#>  3  45.7
#>  4  20.5
#>  5  28.0
#>  6  31.5
#>  7  36.9
#>  8  48.8
#>  9  30.7
#> 10  36.0
#> # ℹ 790 more rows

Note that we need to explicitly pass new_data because the method definition does not allow to default it to the data set we used for the model fitting automatically.

By using the type = "numeric" default of predict() as above we cannot further customize the calculations. We obtain predicted values without confidence intervals or standard errors.

On the other hand, when using type = "raw" we can customize the calculations via the opts list:

predict(
  model,
  new_data = fev_data,
  type = "raw",
  opts = list(se.fit = TRUE, interval = "prediction", nsim = 10L)
)
#>          fit        se      lwr      upr
#> 1   32.47877  6.328341 20.07545 44.88209
#> 2   39.97105  0.000000 39.97105 39.97105
#> 3   45.70508  4.260127 37.35538 54.05477
#> 4   20.48379  0.000000 20.48379 20.48379
#> 5   28.01243  5.796614 16.65127 39.37358
#> 6   31.45522  0.000000 31.45522 31.45522
#> 7   36.87889  0.000000 36.87889 36.87889
#> 8   48.80809  0.000000 48.80809 48.80809
#> 9   30.73774  5.931329 19.11255 42.36293
#> 10  35.98699  0.000000 35.98699 35.98699
#> 11  42.64153  4.068537 34.66734 50.61571
#> 12  37.16444  0.000000 37.16444 37.16444
#> 13  33.89229  0.000000 33.89229 33.89229
#> 14  33.74637  0.000000 33.74637 33.74637
#> 15  44.04155  3.998944 36.20376 51.87934
#> 16  54.45055  0.000000 54.45055 54.45055
#> 17  32.31386  0.000000 32.31386 32.31386
#> 18  37.31982  4.680845 28.14553 46.49410
#> 19  46.79361  0.000000 46.79361 46.79361
#> 20  41.71154  0.000000 41.71154 41.71154
#> 21  31.17198  6.435192 18.55923 43.78472
#> 22  36.63341  5.139415 26.56034 46.70648
#> 23  39.02423  0.000000 39.02423 39.02423
#> 24  47.26333 10.183040 27.30494 67.22173
#> 25  31.93050  0.000000 31.93050 31.93050
#> 26  32.90947  0.000000 32.90947 32.90947
#> 27  41.27523  3.943471 33.54617 49.00429
#> 28  48.28031  0.000000 48.28031 48.28031
#> 29  32.23021  0.000000 32.23021 32.23021
#> 30  35.91080  0.000000 35.91080 35.91080
#> 31  45.54898  0.000000 45.54898 45.54898
#> 32  53.02877  0.000000 53.02877 53.02877
#> 33  47.16898  0.000000 47.16898 47.16898
#> 34  46.64287  0.000000 46.64287 46.64287
#> 35  50.84665  3.967415 43.07066 58.62264
#> 36  58.09713  0.000000 58.09713 58.09713
#> 37  33.21881  6.391593 20.69152 45.74610
#> 38  37.68412  5.101882 27.68462 47.68363
#> 39  44.97613  0.000000 44.97613 44.97613
#> 40  47.67506 10.168713 27.74474 67.60537
#> 41  44.32755  0.000000 44.32755 44.32755
#> 42  38.97813  0.000000 38.97813 38.97813
#> 43  43.72862  0.000000 43.72862 43.72862
#> 44  46.43393  0.000000 46.43393 46.43393
#> 45  40.34576  0.000000 40.34576 40.34576
#> 46  42.76568  0.000000 42.76568 42.76568
#> 47  40.11155  0.000000 40.11155 40.11155
#> 48  49.71974 10.017856 30.08511 69.35438
#> 49  41.46341  6.298989 29.11762 53.80920
#> 50  45.73510  5.093749 35.75154 55.71867
#> 51  53.31791  0.000000 53.31791 53.31791
#> 52  56.07641  0.000000 56.07641 56.07641
#> 53  32.16382  6.378763 19.66168 44.66597
#> 54  37.14256  5.089794 27.16675 47.11837
#> 55  41.90837  0.000000 41.90837 41.90837
#> 56  47.46284 10.152585 27.56414 67.36155
#> 57  27.78883  6.438650 15.16931 40.40835
#> 58  34.13887  5.213369 23.92086 44.35688
#> 59  34.65663  0.000000 34.65663 34.65663
#> 60  39.07791  0.000000 39.07791 39.07791
#> 61  31.18775  5.894336 19.63506 42.74044
#> 62  35.89612  0.000000 35.89612 35.89612
#> 63  41.31608  4.038753 33.40027 49.23189
#> 64  47.67264  0.000000 47.67264 47.67264
#> 65  22.65440  0.000000 22.65440 22.65440
#> 66  36.35488  4.824370 26.89929 45.81047
#> 67  45.20175  4.161558 37.04524 53.35825
#> 68  40.85376  0.000000 40.85376 40.85376
#> 69  32.60048  0.000000 32.60048 32.60048
#> 70  33.64329  0.000000 33.64329 33.64329
#> 71  44.00451  4.028095 36.10958 51.89943
#> 72  40.92278  0.000000 40.92278 40.92278
#> 73  32.14831  0.000000 32.14831 32.14831
#> 74  46.43604  0.000000 46.43604 46.43604
#> 75  41.34973  0.000000 41.34973 41.34973
#> 76  66.30382  0.000000 66.30382 66.30382
#> 77  42.79902  5.873658 31.28686 54.31118
#> 78  47.95358  0.000000 47.95358 47.95358
#> 79  53.97364  0.000000 53.97364 53.97364
#> 80  56.89204 10.092345 37.11141 76.67268
#> 81  46.35384  6.066128 34.46445 58.24323
#> 82  56.64544  0.000000 56.64544 56.64544
#> 83  49.70872  0.000000 49.70872 49.70872
#> 84  60.40497  0.000000 60.40497 60.40497
#> 85  45.98525  0.000000 45.98525 45.98525
#> 86  51.90911  0.000000 51.90911 51.90911
#> 87  41.50787  0.000000 41.50787 41.50787
#> 88  53.42727  0.000000 53.42727 53.42727
#> 89  23.86859  0.000000 23.86859 23.86859
#> 90  35.98563  0.000000 35.98563 35.98563
#> 91  43.60626  0.000000 43.60626 43.60626
#> 92  44.77520  9.977179 25.22029 64.33011
#> 93  29.59773  0.000000 29.59773 29.59773
#> 94  35.50688  0.000000 35.50688 35.50688
#> 95  55.42944  0.000000 55.42944 55.42944
#> 96  52.10530  0.000000 52.10530 52.10530
#> 97  31.69644  0.000000 31.69644 31.69644
#> 98  32.16159  0.000000 32.16159 32.16159
#> 99  51.04735  0.000000 51.04735 51.04735
#> 100 55.85987  0.000000 55.85987 55.85987
#> 101 49.11706  0.000000 49.11706 49.11706
#> 102 49.25544  0.000000 49.25544 49.25544
#> 103 51.72211  0.000000 51.72211 51.72211
#> 104 69.99128  0.000000 69.99128 69.99128
#> 105 22.07169  0.000000 22.07169 22.07169
#> 106 36.35845  4.838107 26.87593 45.84097
#> 107 46.08393  0.000000 46.08393 46.08393
#> 108 52.42288  0.000000 52.42288 52.42288
#> 109 37.69466  0.000000 37.69466 37.69466
#> 110 44.59400  0.000000 44.59400 44.59400
#> 111 52.08897  0.000000 52.08897 52.08897
#> 112 58.22961  0.000000 58.22961 58.22961
#> 113 37.22824  0.000000 37.22824 37.22824
#> 114 34.39863  0.000000 34.39863 34.39863
#> 115 45.88949  4.101886 37.84994 53.92904
#> 116 36.34012  0.000000 36.34012 36.34012
#> 117 45.44182  0.000000 45.44182 45.44182
#> 118 41.54847  0.000000 41.54847 41.54847
#> 119 43.92172  0.000000 43.92172 43.92172
#> 120 61.83243  0.000000 61.83243 61.83243
#> 121 27.25656  0.000000 27.25656 27.25656
#> 122 34.77803  4.776636 25.41600 44.14007
#> 123 45.65133  0.000000 45.65133 45.65133
#> 124 44.56078 10.005509 24.95034 64.17122
#> 125 33.19334  0.000000 33.19334 33.19334
#> 126 39.72671  4.712987 30.48942 48.96400
#> 127 45.59637  4.013838 37.72939 53.46335
#> 128 41.66826  0.000000 41.66826 41.66826
#> 129 27.12753  0.000000 27.12753 27.12753
#> 130 31.74858  0.000000 31.74858 31.74858
#> 131 44.57711  4.197659 36.34985 52.80437
#> 132 41.60000  0.000000 41.60000 41.60000
#> 133 39.45250  0.000000 39.45250 39.45250
#> 134 32.61823  0.000000 32.61823 32.61823
#> 135 34.62445  0.000000 34.62445 34.62445
#> 136 45.90515  0.000000 45.90515 45.90515
#> 137 36.17780  0.000000 36.17780 36.17780
#> 138 39.79796  0.000000 39.79796 39.79796
#> 139 45.87019  3.951097 38.12619 53.61420
#> 140 50.08272  0.000000 50.08272 50.08272
#> 141 36.27753  6.054156 24.41160 48.14346
#> 142 44.64316  0.000000 44.64316 44.64316
#> 143 44.88252  4.094265 36.85790 52.90713
#> 144 39.73529  0.000000 39.73529 39.73529
#> 145 34.06164  0.000000 34.06164 34.06164
#> 146 40.18592  0.000000 40.18592 40.18592
#> 147 41.17584  0.000000 41.17584 41.17584
#> 148 57.76669  0.000000 57.76669 57.76669
#> 149 38.18460  0.000000 38.18460 38.18460
#> 150 38.61735  4.708172 29.38950 47.84520
#> 151 47.19893  0.000000 47.19893 47.19893
#> 152 48.18237 10.147911 28.29283 68.07191
#> 153 37.32785  0.000000 37.32785 37.32785
#> 154 37.89476  4.665543 28.75046 47.03906
#> 155 43.16048  0.000000 43.16048 43.16048
#> 156 41.40349  0.000000 41.40349 41.40349
#> 157 30.15733  0.000000 30.15733 30.15733
#> 158 35.84353  0.000000 35.84353 35.84353
#> 159 40.95250  0.000000 40.95250 40.95250
#> 160 46.76086  9.886491 27.38369 66.13802
#> 161 36.09804  5.857808 24.61695 47.57913
#> 162 41.37928  0.000000 41.37928 41.37928
#> 163 50.17316  0.000000 50.17316 50.17316
#> 164 45.35226  0.000000 45.35226 45.35226
#> 165 39.06491  0.000000 39.06491 39.06491
#> 166 39.39597  4.722573 30.13990 48.65205
#> 167 43.69372  4.001901 35.85014 51.53731
#> 168 42.11960  0.000000 42.11960 42.11960
#> 169 29.81042  0.000000 29.81042 29.81042
#> 170 42.57055  0.000000 42.57055 42.57055
#> 171 47.81652  0.000000 47.81652 47.81652
#> 172 68.06024  0.000000 68.06024 68.06024
#> 173 35.62071  0.000000 35.62071 35.62071
#> 174 40.83933  4.711457 31.60505 50.07362
#> 175 45.83438  3.997712 37.99901 53.66975
#> 176 51.72310  9.985014 32.15283 71.29336
#> 177 33.89134  0.000000 33.89134 33.89134
#> 178 36.42808  0.000000 36.42808 36.42808
#> 179 37.57519  0.000000 37.57519 37.57519
#> 180 58.46873  0.000000 58.46873 58.46873
#> 181 19.54516  0.000000 19.54516 19.54516
#> 182 31.13541  0.000000 31.13541 31.13541
#> 183 40.89955  0.000000 40.89955 40.89955
#> 184 42.08852 10.033675 22.42288 61.75416
#> 185 22.18809  0.000000 22.18809 22.18809
#> 186 41.05857  0.000000 41.05857 41.05857
#> 187 37.32452  0.000000 37.32452 37.32452
#> 188 47.28633 10.506314 26.69433 67.87832
#> 189 35.28933  5.930491 23.66579 46.91288
#> 190 43.12432  0.000000 43.12432 43.12432
#> 191 41.99349  0.000000 41.99349 41.99349
#> 192 49.12250 10.133250 29.26169 68.98330
#> 193 44.03080  0.000000 44.03080 44.03080
#> 194 38.66417  0.000000 38.66417 38.66417
#> 195 53.45993  0.000000 53.45993 53.45993
#> 196 53.14026 10.340410 32.87343 73.40710
#> 197 29.81948  0.000000 29.81948 29.81948
#> 198 30.43859  0.000000 30.43859 30.43859
#> 199 40.18095  0.000000 40.18095 40.18095
#> 200 48.07272  9.967243 28.53728 67.60815
#> 201 26.78578  0.000000 26.78578 26.78578
#> 202 34.55115  0.000000 34.55115 34.55115
#> 203 44.66449  4.123926 36.58174 52.74724
#> 204 40.06421  0.000000 40.06421 40.06421
#> 205 36.66313  5.942700 25.01565 48.31061
#> 206 43.09329  0.000000 43.09329 43.09329
#> 207 46.09670  4.062984 38.13340 54.06000
#> 208 45.71567  0.000000 45.71567 45.71567
#> 209 40.74992  0.000000 40.74992 40.74992
#> 210 44.74635  0.000000 44.74635 44.74635
#> 211 47.51420  3.986462 39.70088 55.32753
#> 212 53.24620  9.955047 33.73466 72.75773
#> 213 40.35635  6.620353 27.38070 53.33201
#> 214 45.16757  5.186297 35.00261 55.33252
#> 215 50.02199  4.137823 41.91201 58.13197
#> 216 56.03985 10.197209 36.05369 76.02602
#> 217 40.14674  0.000000 40.14674 40.14674
#> 218 48.75859  0.000000 48.75859 48.75859
#> 219 46.43462  0.000000 46.43462 46.43462
#> 220 53.05319 10.007873 33.43812 72.66826
#> 221 29.33990  0.000000 29.33990 29.33990
#> 222 36.24424  4.689861 27.05228 45.43620
#> 223 42.39946  3.977660 34.60339 50.19553
#> 224 47.93165  0.000000 47.93165 47.93165
#> 225 36.59443  5.731392 25.36111 47.82775
#> 226 41.11632  0.000000 41.11632 41.11632
#> 227 47.05889  0.000000 47.05889 47.05889
#> 228 52.24599  0.000000 52.24599 52.24599
#> 229 45.12054  6.047408 33.26784 56.97325
#> 230 54.14236  0.000000 54.14236 54.14236
#> 231 50.44618  0.000000 50.44618 50.44618
#> 232 58.53593 10.219379 38.50632 78.56555
#> 233 37.53657  0.000000 37.53657 37.53657
#> 234 44.28282  4.653963 35.16122 53.40442
#> 235 49.45840  0.000000 49.45840 49.45840
#> 236 59.12866  0.000000 59.12866 59.12866
#> 237 40.31268  0.000000 40.31268 40.31268
#> 238 39.66049  0.000000 39.66049 39.66049
#> 239 50.89726  0.000000 50.89726 50.89726
#> 240 56.13116  0.000000 56.13116 56.13116
#> 241 32.82981  0.000000 32.82981 32.82981
#> 242 46.53837  0.000000 46.53837 46.53837
#> 243 44.46016  4.012140 36.59651 52.32382
#> 244 51.81265  0.000000 51.81265 51.81265
#> 245 27.76886  6.060476 15.89055 39.64718
#> 246 29.91939  0.000000 29.91939 29.91939
#> 247 40.70944  4.119124 32.63610 48.78277
#> 248 44.37942 10.161018 24.46419 64.29465
#> 249 43.47695  5.902016 31.90921 55.04469
#> 250 51.05656  0.000000 51.05656 51.05656
#> 251 50.50059  0.000000 50.50059 50.50059
#> 252 64.11388  0.000000 64.11388 64.11388
#> 253 32.21843  0.000000 32.21843 32.21843
#> 254 29.64732  0.000000 29.64732 29.64732
#> 255 42.48707  4.049783 34.54964 50.42449
#> 256 45.09919  0.000000 45.09919 45.09919
#> 257 39.75659  0.000000 39.75659 39.75659
#> 258 37.28894  0.000000 37.28894 37.28894
#> 259 44.80145  0.000000 44.80145 44.80145
#> 260 65.95920  0.000000 65.95920 65.95920
#> 261 33.43439  0.000000 33.43439 33.43439
#> 262 33.57042  0.000000 33.57042 33.57042
#> 263 39.91543  0.000000 39.91543 39.91543
#> 264 49.57098  0.000000 49.57098 49.57098
#> 265 38.91634  0.000000 38.91634 38.91634
#> 266 36.69011  0.000000 36.69011 36.69011
#> 267 45.66665  0.000000 45.66665 45.66665
#> 268 52.07431  0.000000 52.07431 52.07431
#> 269 42.21411  0.000000 42.21411 42.21411
#> 270 45.02901  0.000000 45.02901 45.02901
#> 271 50.22530  3.956570 42.47057 57.98004
#> 272 56.56084  9.940090 37.07862 76.04306
#> 273 30.98338  0.000000 30.98338 30.98338
#> 274 44.72932  0.000000 44.72932 44.72932
#> 275 40.68711  0.000000 40.68711 40.68711
#> 276 34.71530  0.000000 34.71530 34.71530
#> 277 27.30752  0.000000 27.30752 27.30752
#> 278 37.31585  0.000000 37.31585 37.31585
#> 279 42.23592  3.949193 34.49564 49.97619
#> 280 44.83000  0.000000 44.83000 44.83000
#> 281 32.93042  0.000000 32.93042 32.93042
#> 282 44.91911  0.000000 44.91911 44.91911
#> 283 45.68636  0.000000 45.68636 45.68636
#> 284 65.98800  0.000000 65.98800 65.98800
#> 285 46.60130  0.000000 46.60130 46.60130
#> 286 40.89786  0.000000 40.89786 40.89786
#> 287 46.66708  0.000000 46.66708 46.66708
#> 288 53.97575 10.170630 34.04169 73.90982
#> 289 34.51394  6.299838 22.16649 46.86140
#> 290 40.31956  5.091944 30.33954 50.29959
#> 291 43.83270  0.000000 43.83270 43.83270
#> 292 44.11604  0.000000 44.11604 44.11604
#> 293 38.29612  0.000000 38.29612 38.29612
#> 294 42.38074  4.705682 33.15777 51.60371
#> 295 51.38570  0.000000 51.38570 51.38570
#> 296 56.20979  0.000000 56.20979 56.20979
#> 297 35.93809  5.988841 24.20018 47.67600
#> 298 43.45819  0.000000 43.45819 43.45819
#> 299 38.38741  0.000000 38.38741 38.38741
#> 300 56.42818  0.000000 56.42818 56.42818
#> 301 39.05050  0.000000 39.05050 39.05050
#> 302 44.66707  4.690309 35.47423 53.85990
#> 303 49.86836  3.976885 42.07381 57.66292
#> 304 54.09200  0.000000 54.09200 54.09200
#> 305 31.40521  0.000000 31.40521 31.40521
#> 306 46.13330  0.000000 46.13330 46.13330
#> 307 45.29845  0.000000 45.29845 45.29845
#> 308 28.06936  0.000000 28.06936 28.06936
#> 309 39.05960  5.791717 27.70804 50.41116
#> 310 42.50283  0.000000 42.50283 42.50283
#> 311 46.45368  0.000000 46.45368 46.45368
#> 312 64.97366  0.000000 64.97366 64.97366
#> 313 30.67876  6.484535 17.96930 43.38821
#> 314 35.63991  5.137832 25.56994 45.70988
#> 315 41.27878  4.112017 33.21938 49.33819
#> 316 43.97847  0.000000 43.97847 43.97847
#> 317 35.33466  0.000000 35.33466 35.33466
#> 318 39.34378  0.000000 39.34378 39.34378
#> 319 41.27633  0.000000 41.27633 41.27633
#> 320 50.74572  9.900646 31.34081 70.15063
#> 321 34.18429  5.758417 22.89800 45.47058
#> 322 39.83058  0.000000 39.83058 39.83058
#> 323 43.49673  0.000000 43.49673 43.49673
#> 324 44.06114  0.000000 44.06114 44.06114
#> 325 41.43742  0.000000 41.43742 41.43742
#> 326 40.68384  4.671127 31.52860 49.83908
#> 327 46.16954  0.000000 46.16954 46.16954
#> 328 54.24024  0.000000 54.24024 54.24024
#> 329 36.61831  0.000000 36.61831 36.61831
#> 330 42.09272  0.000000 42.09272 42.09272
#> 331 50.69556  0.000000 50.69556 50.69556
#> 332 51.72563  0.000000 51.72563 51.72563
#> 333 32.08271  6.508659 19.32597 44.83945
#> 334 36.39974  5.148559 26.30875 46.49073
#> 335 41.38610  4.121854 33.30742 49.46479
#> 336 53.89947  0.000000 53.89947 53.89947
#> 337 31.94884  6.295848 19.60920 44.28848
#> 338 36.34138  5.063916 26.41629 46.26647
#> 339 39.94420  0.000000 39.94420 39.94420
#> 340 56.42482  0.000000 56.42482 56.42482
#> 341 41.86385  0.000000 41.86385 41.86385
#> 342 34.56420  0.000000 34.56420 34.56420
#> 343 38.68927  0.000000 38.68927 38.68927
#> 344 62.88743  0.000000 62.88743 62.88743
#> 345 28.85343  0.000000 28.85343 28.85343
#> 346 38.89808  4.802532 29.48529 48.31088
#> 347 49.29495  0.000000 49.29495 49.29495
#> 348 48.90884 10.028462 29.25341 68.56426
#> 349 28.74029  0.000000 28.74029 28.74029
#> 350 36.38836  4.701957 27.17269 45.60402
#> 351 43.59994  0.000000 43.59994 43.59994
#> 352 57.38616  0.000000 57.38616 57.38616
#> 353 35.36824  0.000000 35.36824 35.36824
#> 354 43.06110  0.000000 43.06110 43.06110
#> 355 31.27551  0.000000 31.27551 31.27551
#> 356 54.13245  0.000000 54.13245 54.13245
#> 357 25.97050  0.000000 25.97050 25.97050
#> 358 34.04514  4.728613 24.77723 43.31305
#> 359 40.67015  4.009238 32.81219 48.52812
#> 360 44.36054 10.015792 24.72995 63.99113
#> 361 39.69391  6.285098 27.37535 52.01248
#> 362 44.79720  5.063240 34.87343 54.72097
#> 363 51.17493  0.000000 51.17493 51.17493
#> 364 48.44043  0.000000 48.44043 48.44043
#> 365 43.33128  0.000000 43.33128 43.33128
#> 366 46.45918  4.710003 37.22775 55.69062
#> 367 55.93546  0.000000 55.93546 55.93546
#> 368 54.15312  0.000000 54.15312 54.15312
#> 369 33.79699  5.784463 22.45965 45.13433
#> 370 40.60252  0.000000 40.60252 40.60252
#> 371 44.44715  0.000000 44.44715 44.44715
#> 372 40.54161  0.000000 40.54161 40.54161
#> 373 33.95563  0.000000 33.95563 33.95563
#> 374 36.84083  4.649256 27.72845 45.95320
#> 375 43.67802  0.000000 43.67802 43.67802
#> 376 42.76023  0.000000 42.76023 42.76023
#> 377 34.18486  5.990614 22.44347 45.92625
#> 378 42.82678  0.000000 42.82678 42.82678
#> 379 39.59218  0.000000 39.59218 39.59218
#> 380 47.93427 10.186288 27.96951 67.89902
#> 381 33.49216  0.000000 33.49216 33.49216
#> 382 35.39266  0.000000 35.39266 35.39266
#> 383 42.88943  3.946494 35.15445 50.62442
#> 384 42.36266  0.000000 42.36266 42.36266
#> 385 48.54368  0.000000 48.54368 48.54368
#> 386 43.94366  0.000000 43.94366 43.94366
#> 387 50.98218  4.006133 43.13031 58.83406
#> 388 47.91204  0.000000 47.91204 47.91204
#> 389 20.72928  0.000000 20.72928 20.72928
#> 390 28.00599  0.000000 28.00599 28.00599
#> 391 40.19255  0.000000 40.19255 40.19255
#> 392 37.79360  0.000000 37.79360 37.79360
#> 393 32.17343  5.993973 20.42546 43.92141
#> 394 36.75177  0.000000 36.75177 36.75177
#> 395 42.75025  4.058659 34.79543 50.70508
#> 396 47.37158 10.101405 27.57319 67.16997
#> 397 34.59822  0.000000 34.59822 34.59822
#> 398 39.32034  0.000000 39.32034 39.32034
#> 399 40.65702  0.000000 40.65702 40.65702
#> 400 48.52002  9.901956 29.11254 67.92749
#> 401 40.41786  5.911790 28.83096 52.00475
#> 402 43.03255  0.000000 43.03255 43.03255
#> 403 54.65715  0.000000 54.65715 54.65715
#> 404 55.54195 10.155817 35.63691 75.44698
#> 405 35.55742  0.000000 35.55742 35.55742
#> 406 43.70215  0.000000 43.70215 43.70215
#> 407 42.52157  0.000000 42.52157 42.52157
#> 408 54.89337  0.000000 54.89337 54.89337
#> 409 32.03460  0.000000 32.03460 32.03460
#> 410 29.45107  0.000000 29.45107 29.45107
#> 411 45.35138  0.000000 45.35138 45.35138
#> 412 45.33026 10.169474 25.39845 65.26206
#> 413 38.73784  0.000000 38.73784 38.73784
#> 414 39.30063  4.703976 30.08100 48.52025
#> 415 41.42283  0.000000 41.42283 41.42283
#> 416 47.32385  0.000000 47.32385 47.32385
#> 417 40.41284  5.938378 28.77384 52.05185
#> 418 47.55310  0.000000 47.55310 47.55310
#> 419 49.06509  0.000000 49.06509 49.06509
#> 420 53.79511 10.121574 33.95719 73.63303
#> 421 29.22591  0.000000 29.22591 29.22591
#> 422 40.08175  0.000000 40.08175 40.08175
#> 423 45.68142  0.000000 45.68142 45.68142
#> 424 41.47403  0.000000 41.47403 41.47403
#> 425 37.55612  6.451121 24.91215 50.20009
#> 426 41.98123  5.136605 31.91366 52.04879
#> 427 42.51970  0.000000 42.51970 42.51970
#> 428 69.36099  0.000000 69.36099 69.36099
#> 429 42.39760  0.000000 42.39760 42.39760
#> 430 43.72376  0.000000 43.72376 43.72376
#> 431 49.47601  0.000000 49.47601 49.47601
#> 432 51.94188  0.000000 51.94188 51.94188
#> 433 31.77722  5.909596 20.19462 43.35981
#> 434 40.59100  0.000000 40.59100 40.59100
#> 435 39.97833  0.000000 39.97833 39.97833
#> 436 31.69049  0.000000 31.69049 31.69049
#> 437 33.99809  5.856154 22.52024 45.47594
#> 438 37.20517  0.000000 37.20517 37.20517
#> 439 46.28740  0.000000 46.28740 46.28740
#> 440 49.81365 10.088896 30.03978 69.58752
#> 441 35.15913  6.525518 22.36935 47.94891
#> 442 40.63997  5.164185 30.51835 50.76158
#> 443 46.80529  4.142376 38.68638 54.92420
#> 444 41.58720  0.000000 41.58720 41.58720
#> 445 32.17365  0.000000 32.17365 32.17365
#> 446 37.07479  4.677690 27.90669 46.24290
#> 447 40.69375  0.000000 40.69375 40.69375
#> 448 47.52336  9.947825 28.02598 67.02074
#> 449 32.28771  0.000000 32.28771 32.28771
#> 450 41.76205  0.000000 41.76205 41.76205
#> 451 40.06768  0.000000 40.06768 40.06768
#> 452 47.21582  9.975254 27.66469 66.76696
#> 453 29.14213  0.000000 29.14213 29.14213
#> 454 39.50989  0.000000 39.50989 39.50989
#> 455 43.32349  0.000000 43.32349 43.32349
#> 456 47.16756  0.000000 47.16756 47.16756
#> 457 40.93020  0.000000 40.93020 40.93020
#> 458 42.19406  0.000000 42.19406 42.19406
#> 459 41.21057  0.000000 41.21057 41.21057
#> 460 49.76205 10.011957 30.13898 69.38513
#> 461 38.54330  0.000000 38.54330 38.54330
#> 462 41.44104  4.704018 32.22133 50.66074
#> 463 43.96324  0.000000 43.96324 43.96324
#> 464 42.67652  0.000000 42.67652 42.67652
#> 465 22.79584  0.000000 22.79584 22.79584
#> 466 33.91004  4.770477 24.56008 43.26001
#> 467 41.58421  4.076348 33.59471 49.57370
#> 468 44.31106 10.110978 24.49391 64.12821
#> 469 31.43559  0.000000 31.43559 31.43559
#> 470 38.85064  0.000000 38.85064 38.85064
#> 471 48.24288  0.000000 48.24288 48.24288
#> 472 46.15940  9.976949 26.60494 65.71386
#> 473 44.71302  0.000000 44.71302 44.71302
#> 474 51.85370  0.000000 51.85370 51.85370
#> 475 50.77548  4.038611 42.85995 58.69101
#> 476 58.11432  9.999768 38.51513 77.71350
#> 477 30.56757  0.000000 30.56757 30.56757
#> 478 35.65607  4.710094 26.42446 44.88769
#> 479 41.25037  3.981832 33.44612 49.05462
#> 480 45.88736  9.985125 26.31688 65.45785
#> 481 37.37624  6.503985 24.62866 50.12381
#> 482 41.66978  5.150332 31.57531 51.76424
#> 483 45.99979  4.132235 37.90076 54.09882
#> 484 59.90473  0.000000 59.90473 59.90473
#> 485 33.87728  6.651903 20.83978 46.91477
#> 486 37.28988  5.348859 26.80631 47.77345
#> 487 49.76150  0.000000 49.76150 49.76150
#> 488 46.60552 10.364218 26.29203 66.91902
#> 489 47.21985  0.000000 47.21985 47.21985
#> 490 40.34525  0.000000 40.34525 40.34525
#> 491 48.29793  0.000000 48.29793 48.29793
#> 492 54.57153 10.174744 34.62940 74.51366
#> 493 36.13680  5.861094 24.64927 47.62434
#> 494 44.39634  0.000000 44.39634 44.39634
#> 495 41.71421  0.000000 41.71421 41.71421
#> 496 47.37535  0.000000 47.37535 47.37535
#> 497 42.03797  0.000000 42.03797 42.03797
#> 498 37.56100  0.000000 37.56100 37.56100
#> 499 45.11793  0.000000 45.11793 45.11793
#> 500 52.86788 10.007323 33.25389 72.48187
#> 501 34.62530  0.000000 34.62530 34.62530
#> 502 45.28206  0.000000 45.28206 45.28206
#> 503 44.51505  4.013990 36.64777 52.38232
#> 504 63.57761  0.000000 63.57761 63.57761
#> 505 35.80878  0.000000 35.80878 35.80878
#> 506 40.93038  4.685628 31.74672 50.11405
#> 507 45.85156  3.979761 38.05137 53.65175
#> 508 52.67314  0.000000 52.67314 52.67314
#> 509 35.88734  0.000000 35.88734 35.88734
#> 510 38.73222  0.000000 38.73222 38.73222
#> 511 46.70361  0.000000 46.70361 46.70361
#> 512 53.65398  0.000000 53.65398 53.65398
#> 513 36.71543  0.000000 36.71543 36.71543
#> 514 43.89170  4.726781 34.62738 53.15602
#> 515 49.56246  4.004442 41.71389 57.41102
#> 516 54.83060  9.998550 35.23380 74.42740
#> 517 37.85241  5.880195 26.32744 49.37738
#> 518 41.54317  0.000000 41.54317 41.54317
#> 519 51.67909  0.000000 51.67909 51.67909
#> 520 51.76691 10.135821 31.90106 71.63275
#> 521 27.40130  0.000000 27.40130 27.40130
#> 522 30.33517  0.000000 30.33517 30.33517
#> 523 37.73092  0.000000 37.73092 37.73092
#> 524 29.11668  0.000000 29.11668 29.11668
#> 525 30.03596  5.802185 18.66389 41.40804
#> 526 32.08830  0.000000 32.08830 32.08830
#> 527 41.66067  0.000000 41.66067 41.66067
#> 528 53.90815  0.000000 53.90815 53.90815
#> 529 34.02622  5.854819 22.55098 45.50145
#> 530 35.06937  0.000000 35.06937 35.06937
#> 531 47.17615  0.000000 47.17615 47.17615
#> 532 56.49347  0.000000 56.49347 56.49347
#> 533 34.02880  5.789778 22.68104 45.37655
#> 534 38.88006  0.000000 38.88006 38.88006
#> 535 47.54070  0.000000 47.54070 47.54070
#> 536 43.53705  0.000000 43.53705 43.53705
#> 537 31.82054  0.000000 31.82054 31.82054
#> 538 39.62816  0.000000 39.62816 39.62816
#> 539 44.95543  0.000000 44.95543 44.95543
#> 540 21.11543  0.000000 21.11543 21.11543
#> 541 34.74671  0.000000 34.74671 34.74671
#> 542 43.27308  4.715995 34.02990 52.51626
#> 543 49.29538  4.005188 41.44536 57.14541
#> 544 56.69249  0.000000 56.69249 56.69249
#> 545 22.73126  0.000000 22.73126 22.73126
#> 546 32.50075  0.000000 32.50075 32.50075
#> 547 42.37206  0.000000 42.37206 42.37206
#> 548 42.89847  0.000000 42.89847 42.89847
#> 549 55.62582  0.000000 55.62582 55.62582
#> 550 45.38998  0.000000 45.38998 45.38998
#> 551 52.66743  0.000000 52.66743 52.66743
#> 552 56.87348 10.773296 35.75821 77.98876
#> 553 30.66032  6.672129 17.58318 43.73745
#> 554 37.44228  5.411757 26.83543 48.04913
#> 555 34.18931  0.000000 34.18931 34.18931
#> 556 45.59740  0.000000 45.59740 45.59740
#> 557 28.89198  0.000000 28.89198 28.89198
#> 558 38.46147  0.000000 38.46147 38.46147
#> 559 42.42099  3.939954 34.69882 50.14315
#> 560 49.90357  0.000000 49.90357 49.90357
#> 561 39.74586  5.894422 28.19300 51.29872
#> 562 44.14167  0.000000 44.14167 44.14167
#> 563 49.91712  4.049872 41.97952 57.85473
#> 564 55.24278  0.000000 55.24278 55.24278
#> 565 36.24790  6.621579 23.26985 49.22596
#> 566 41.05912  5.185181 30.89635 51.22189
#> 567 45.91354  4.144486 37.79050 54.03659
#> 568 51.93141 10.196460 31.94671 71.91610
#> 569 27.38001  0.000000 27.38001 27.38001
#> 570 33.63251  0.000000 33.63251 33.63251
#> 571 44.70168  4.145116 36.57740 52.82596
#> 572 39.34410  0.000000 39.34410 39.34410
#> 573 26.98575  0.000000 26.98575 26.98575
#> 574 24.04175  0.000000 24.04175 24.04175
#> 575 42.16648  0.000000 42.16648 42.16648
#> 576 44.75380  0.000000 44.75380 44.75380
#> 577 31.55469  0.000000 31.55469 31.55469
#> 578 44.42696  0.000000 44.42696 44.42696
#> 579 44.10343  0.000000 44.10343 44.10343
#> 580 48.06505  9.995073 28.47507 67.65504
#> 581 34.87547  5.788135 23.53093 46.22001
#> 582 37.87445  0.000000 37.87445 37.87445
#> 583 48.31828  0.000000 48.31828 48.31828
#> 584 50.21520  0.000000 50.21520 50.21520
#> 585 41.94615  0.000000 41.94615 41.94615
#> 586 39.62690  0.000000 39.62690 39.62690
#> 587 46.69763  0.000000 46.69763 46.69763
#> 588 49.44653 10.156344 29.54046 69.35260
#> 589 38.01775  5.854587 26.54297 49.49253
#> 590 43.75255  0.000000 43.75255 43.75255
#> 591 47.38873  0.000000 47.38873 47.38873
#> 592 52.70780 10.069832 32.97129 72.44430
#> 593 32.43412  0.000000 32.43412 32.43412
#> 594 43.07163  0.000000 43.07163 43.07163
#> 595 42.99551  0.000000 42.99551 42.99551
#> 596 53.82759  0.000000 53.82759 53.82759
#> 597 39.45747  6.354089 27.00368 51.91125
#> 598 42.93167  5.158776 32.82065 53.04268
#> 599 50.64802  0.000000 50.64802 50.64802
#> 600 63.44051  0.000000 63.44051 63.44051
#> 601 34.48949  0.000000 34.48949 34.48949
#> 602 40.08056  0.000000 40.08056 40.08056
#> 603 41.86656  3.961055 34.10303 49.63008
#> 604 47.46553  0.000000 47.46553 47.46553
#> 605 32.03992  5.981829 20.31575 43.76409
#> 606 37.11697  0.000000 37.11697 37.11697
#> 607 44.12071  4.104464 36.07611 52.16531
#> 608 36.25120  0.000000 36.25120 36.25120
#> 609 29.20171  0.000000 29.20171 29.20171
#> 610 31.53773  0.000000 31.53773 31.53773
#> 611 42.35683  0.000000 42.35683 42.35683
#> 612 64.78352  0.000000 64.78352 64.78352
#> 613 32.72757  0.000000 32.72757 32.72757
#> 614 37.50022  0.000000 37.50022 37.50022
#> 615 42.76167  3.939749 35.03991 50.48344
#> 616 57.03861  0.000000 57.03861 57.03861
#> 617 36.32475  0.000000 36.32475 36.32475
#> 618 40.15241  4.671056 30.99730 49.30751
#> 619 41.46725  0.000000 41.46725 41.46725
#> 620 59.01411  0.000000 59.01411 59.01411
#> 621 30.14970  0.000000 30.14970 30.14970
#> 622 34.91740  0.000000 34.91740 34.91740
#> 623 52.13900  0.000000 52.13900 52.13900
#> 624 58.73839  0.000000 58.73839 58.73839
#> 625 35.83185  0.000000 35.83185 35.83185
#> 626 41.04423  4.689669 31.85265 50.23581
#> 627 45.82688  3.980329 38.02558 53.62818
#> 628 56.41409  0.000000 56.41409 56.41409
#> 629 37.80184  5.770783 26.49131 49.11236
#> 630 43.55593  0.000000 43.55593 43.55593
#> 631 44.26320  0.000000 44.26320 44.26320
#> 632 59.25579  0.000000 59.25579 59.25579
#> 633 28.47314  0.000000 28.47314 28.47314
#> 634 47.47581  0.000000 47.47581 47.47581
#> 635 44.01685  4.070225 36.03936 51.99435
#> 636 49.57489 10.292640 29.40168 69.74809
#> 637 39.38085  5.932315 27.75373 51.00797
#> 638 46.47483  0.000000 46.47483 46.47483
#> 639 51.22677  0.000000 51.22677 51.22677
#> 640 45.82777  0.000000 45.82777 45.82777
#> 641 33.43408  6.001982 21.67041 45.19775
#> 642 39.06783  0.000000 39.06783 39.06783
#> 643 42.98333  4.056639 35.03247 50.93420
#> 644 48.01822 10.105548 28.21171 67.82473
#> 645 29.99542  0.000000 29.99542 29.99542
#> 646 35.69583  4.702741 26.47863 44.91303
#> 647 41.11547  3.977183 33.32034 48.91061
#> 648 54.17796  0.000000 54.17796 54.17796
#> 649 39.32289  5.949171 27.66273 50.98305
#> 650 44.55743  0.000000 44.55743 44.55743
#> 651 47.26282  4.091564 39.24350 55.28214
#> 652 62.59579  0.000000 62.59579 62.59579
#> 653 31.80300  5.740763 20.55131 43.05469
#> 654 35.48396  0.000000 35.48396 35.48396
#> 655 44.07768  0.000000 44.07768 44.07768
#> 656 46.57837  0.000000 46.57837 46.57837
#> 657 47.67979  0.000000 47.67979 47.67979
#> 658 47.73388  4.735848 38.45179 57.01597
#> 659 50.94631  4.031499 43.04472 58.84790
#> 660 58.47218 10.080334 38.71508 78.22927
#> 661 22.15439  0.000000 22.15439 22.15439
#> 662 35.14301  4.826749 25.68276 44.60327
#> 663 42.82000  4.143302 34.69928 50.94073
#> 664 46.24563 10.193355 26.26703 66.22424
#> 665 34.27765  0.000000 34.27765 34.27765
#> 666 36.90059  0.000000 36.90059 36.90059
#> 667 43.05627  3.941943 35.33020 50.78234
#> 668 40.54285  0.000000 40.54285 40.54285
#> 669 29.09494  0.000000 29.09494 29.09494
#> 670 37.21768  0.000000 37.21768 37.21768
#> 671 43.08491  0.000000 43.08491 43.08491
#> 672 46.50100  9.892842 27.11139 65.89062
#> 673 27.12174  0.000000 27.12174 27.12174
#> 674 34.11916  0.000000 34.11916 34.11916
#> 675 45.56320  4.100041 37.52727 53.59914
#> 676 48.00823 10.027537 28.35462 67.66184
#> 677 35.93048  5.795433 24.57164 47.28932
#> 678 40.80230  0.000000 40.80230 40.80230
#> 679 45.89269  0.000000 45.89269 45.89269
#> 680 43.69153  0.000000 43.69153 43.69153
#> 681 28.56569  6.053402 16.70124 40.43014
#> 682 29.22869  0.000000 29.22869 29.22869
#> 683 40.67646  4.127091 32.58751 48.76541
#> 684 55.68362  0.000000 55.68362 55.68362
#> 685 31.90698  0.000000 31.90698 31.90698
#> 686 37.31061  0.000000 37.31061 37.31061
#> 687 40.75546  0.000000 40.75546 40.75546
#> 688 49.50911  9.907911 30.08996 68.92826
#> 689 42.19474  0.000000 42.19474 42.19474
#> 690 44.87228  0.000000 44.87228 44.87228
#> 691 47.55198  0.000000 47.55198 47.55198
#> 692 56.68097  9.896853 37.28349 76.07845
#> 693 50.62894  0.000000 50.62894 50.62894
#> 694 45.47551  0.000000 45.47551 45.47551
#> 695 48.62168  0.000000 48.62168 48.62168
#> 696 56.58212 10.214871 36.56134 76.60290
#> 697 29.66493  0.000000 29.66493 29.66493
#> 698 34.57406  0.000000 34.57406 34.57406
#> 699 42.45295  3.970420 34.67107 50.23483
#> 700 38.11676  0.000000 38.11676 38.11676
#> 701 33.77204  0.000000 33.77204 33.77204
#> 702 34.26148  0.000000 34.26148 34.26148
#> 703 45.29511  4.046987 37.36316 53.22706
#> 704 58.81037  0.000000 58.81037 58.81037
#> 705 31.46668  6.411558 18.90026 44.03311
#> 706 36.78469  5.118562 26.75250 46.81689
#> 707 39.88119  0.000000 39.88119 39.88119
#> 708 47.32261 10.169387 27.39098 67.25425
#> 709 31.62708  0.000000 31.62708 31.62708
#> 710 37.03239  4.683917 27.85208 46.21270
#> 711 42.69162  3.966035 34.91833 50.46490
#> 712 48.22049  0.000000 48.22049 48.22049
#> 713 42.58829  0.000000 42.58829 42.58829
#> 714 45.80410  4.649843 36.69057 54.91762
#> 715 49.33262  0.000000 49.33262 49.33262
#> 716 53.74331  0.000000 53.74331 53.74331
#> 717 29.71857  0.000000 29.71857 29.71857
#> 718 30.45651  0.000000 30.45651 30.45651
#> 719 38.29800  0.000000 38.29800 38.29800
#> 720 45.15328  9.913947 25.72230 64.58426
#> 721 36.81040  0.000000 36.81040 36.81040
#> 722 37.61606  4.675991 28.45129 46.78084
#> 723 42.35045  0.000000 42.35045 42.35045
#> 724 39.39860  0.000000 39.39860 39.39860
#> 725 36.09876  6.365383 23.62284 48.57468
#> 726 40.94066  5.098637 30.94751 50.93380
#> 727 49.73629  0.000000 49.73629 49.73629
#> 728 41.58082  0.000000 41.58082 41.58082
#> 729 43.58901  0.000000 43.58901 43.58901
#> 730 40.16762  0.000000 40.16762 40.16762
#> 731 46.70338  3.993923 38.87544 54.53133
#> 732 53.94830 10.072276 34.20701 73.68960
#> 733 39.60913  6.008652 27.83239 51.38587
#> 734 41.08206  0.000000 41.08206 41.08206
#> 735 49.65683  4.096129 41.62857 57.68510
#> 736 69.37409  0.000000 69.37409 69.37409
#> 737 34.12096  5.805419 22.74255 45.49937
#> 738 41.27625  0.000000 41.27625 41.27625
#> 739 44.76138  0.000000 44.76138 44.76138
#> 740 39.69815  0.000000 39.69815 39.69815
#> 741 38.44296  0.000000 38.44296 38.44296
#> 742 48.20586  0.000000 48.20586 48.20586
#> 743 47.54082  4.054538 39.59407 55.48757
#> 744 35.50735  0.000000 35.50735 35.50735
#> 745 32.08153  0.000000 32.08153 32.08153
#> 746 37.16398  4.708587 27.93532 46.39264
#> 747 42.75619  3.982077 34.95146 50.56092
#> 748 47.39510  9.984234 27.82636 66.96384
#> 749 44.69256  0.000000 44.69256 44.69256
#> 750 41.45664  4.821210 32.00725 50.90604
#> 751 42.18689  0.000000 42.18689 42.18689
#> 752 51.68534 10.155198 31.78152 71.58916
#> 753 37.01741  0.000000 37.01741 37.01741
#> 754 38.26920  0.000000 38.26920 38.26920
#> 755 49.28806  0.000000 49.28806 49.28806
#> 756 50.67485 10.012847 31.05003 70.29967
#> 757 40.45953  0.000000 40.45953 40.45953
#> 758 45.10337  0.000000 45.10337 45.10337
#> 759 45.58250  0.000000 45.58250 45.58250
#> 760 62.96989  0.000000 62.96989 62.96989
#> 761 30.78252  0.000000 30.78252 30.78252
#> 762 41.58139  4.771601 32.22922 50.93355
#> 763 48.87398  4.080179 40.87697 56.87098
#> 764 44.69667  0.000000 44.69667 44.69667
#> 765 32.72491  0.000000 32.72491 32.72491
#> 766 45.78702  0.000000 45.78702 45.78702
#> 767 48.74886  0.000000 48.74886 48.74886
#> 768 84.08449  0.000000 84.08449 84.08449
#> 769 28.60809  5.958125 16.93038 40.28580
#> 770 30.19495  0.000000 30.19495 30.19495
#> 771 36.78573  0.000000 36.78573 36.78573
#> 772 61.03588  0.000000 61.03588 61.03588
#> 773 20.36749  0.000000 20.36749 20.36749
#> 774 35.22480  0.000000 35.22480 35.22480
#> 775 37.42847  0.000000 37.42847 37.42847
#> 776 30.20501  0.000000 30.20501 30.20501
#> 777 41.72819  5.919206 30.12676 53.32962
#> 778 49.12862  0.000000 49.12862 49.12862
#> 779 47.31234  0.000000 47.31234 47.31234
#> 780 57.08286 10.144034 37.20091 76.96480
#> 781 19.28388  0.000000 19.28388 19.28388
#> 782 30.00682  0.000000 30.00682 30.00682
#> 783 39.69711  4.096855 31.66742 47.72680
#> 784 49.21768  0.000000 49.21768 49.21768
#> 785 31.42637  6.501262 18.68413 44.16861
#> 786 36.73485  5.143589 26.65360 46.81609
#> 787 42.72556  4.123392 34.64386 50.80726
#> 788 40.13353  0.000000 40.13353 40.13353
#> 789 42.34534  0.000000 42.34534 42.34534
#> 790 52.32575  0.000000 52.32575 52.32575
#> 791 46.92223  4.185411 38.71898 55.12549
#> 792 69.26254  0.000000 69.26254 69.26254
#> 793 40.35635  6.620353 27.38070 53.33201
#> 794 45.16757  5.186297 35.00261 55.33252
#> 795 50.02199  4.137823 41.91201 58.13197
#> 796 56.03985 10.197209 36.05369 76.02602
#> 797 35.70341  0.000000 35.70341 35.70341
#> 798 41.64454  0.000000 41.64454 41.64454
#> 799 43.29513  3.961002 35.53171 51.05855
#> 800 54.25081  0.000000 54.25081 54.25081

The result is now a matrix, because that is what the predict() method returns for mmrm objects. Note that this cannot be changed to return a tibble at the moment.

Similarly, we can also use the augment() method to add predicted values to a new data set:

augment(model, new_data = fev_data) |>
  select(USUBJID, AVISIT, .resid, .pred)
#> # A tibble: 800 × 4
#>    USUBJID AVISIT .resid .pred
#>    <fct>   <fct>   <dbl> <dbl>
#>  1 PT1     VIS1       NA  32.5
#>  2 PT1     VIS2        0  40.0
#>  3 PT1     VIS3       NA  45.7
#>  4 PT1     VIS4        0  20.5
#>  5 PT2     VIS1       NA  28.0
#>  6 PT2     VIS2        0  31.5
#>  7 PT2     VIS3        0  36.9
#>  8 PT2     VIS4        0  48.8
#>  9 PT3     VIS1       NA  30.7
#> 10 PT3     VIS2        0  36.0
#> # ℹ 790 more rows

Note that here we cannot customize the predict options as this is currently not supported by the augment() method in parsnip.

Using mmrm in workflows

We can leverage the workflows package in order to fit the same model.

  • First we define the specification for linear regression with the mmrm engine.
  • Second we define the workflow, by defining the outcome and predictors that will be used in the formula. We then add the model using the formula.
  • Lastly, we fit the model
mmrm_spec <- linear_reg() |>
  set_engine("mmrm", method = "Satterthwaite")

mmrm_wflow <- workflow() |>
  add_variables(outcomes = FEV1, predictors = c(RACE, ARMCD, AVISIT, USUBJID)) |>
  add_model(mmrm_spec, formula = FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID))

mmrm_wflow |>
  fit(data = fev_data)
#> ══ Workflow [trained] ══════════════════════════════════════════════════════════
#> Preprocessor: Variables
#> Model: linear_reg()
#> 
#> ── Preprocessor ────────────────────────────────────────────────────────────────
#> Outcomes: FEV1
#> Predictors: c(RACE, ARMCD, AVISIT, USUBJID)
#> 
#> ── Model ───────────────────────────────────────────────────────────────────────
#> mmrm fit
#> 
#> Formula:     FEV1 ~ RACE + ARMCD * AVISIT + us(AVISIT | USUBJID)
#> Data:        data (used 537 observations from 197 subjects with maximum 4 
#> timepoints)
#> Weights:     weights
#> Covariance:  unstructured (10 variance parameters)
#> Inference:   REML
#> Deviance:    3387.373
#> 
#> Coefficients: 
#>                   (Intercept) RACEBlack or African American 
#>                   30.96769899                    1.50464863 
#>                     RACEWhite                      ARMCDTRT 
#>                    5.61309565                    3.77555734 
#>                    AVISITVIS2                    AVISITVIS3 
#>                    4.82858803                   10.33317002 
#>                    AVISITVIS4           ARMCDTRT:AVISITVIS2 
#>                   15.05255715                   -0.01737409 
#>           ARMCDTRT:AVISITVIS3           ARMCDTRT:AVISITVIS4 
#>                   -0.66753189                    0.63094392 
#> 
#> Model Inference Optimization:
#> Converged with code 0 and message: convergence: rel_reduction_of_f <= factr*epsmch

We can separate out the data preparation step from the modeling step using the recipes package. Here we are converting the ARMCD variable into a dummy variable and creating an interaction term with the new dummy variable and each visit.

mmrm_recipe <- recipe(FEV1 ~ ., data = fev_data) |>
  step_dummy(ARMCD) |>
  step_interact(terms = ~ starts_with("ARMCD"):AVISIT)

Using prep() and juice() we can see what the transformed data that will be used in the model fit looks like.

mmrm_recipe |>
  prep() |>
  juice()
#> # A tibble: 800 × 13
#>    USUBJID AVISIT RACE       SEX   FEV1_BL WEIGHT VISITN VISITN2  FEV1 ARMCD_TRT
#>    <fct>   <fct>  <fct>      <fct>   <dbl>  <dbl>  <int>   <dbl> <dbl>     <dbl>
#>  1 PT1     VIS1   Black or … Fema…    25.3  0.677      1  -0.626  NA           1
#>  2 PT1     VIS2   Black or … Fema…    25.3  0.801      2   0.184  40.0         1
#>  3 PT1     VIS3   Black or … Fema…    25.3  0.709      3  -0.836  NA           1
#>  4 PT1     VIS4   Black or … Fema…    25.3  0.809      4   1.60   20.5         1
#>  5 PT2     VIS1   Asian      Male     45.0  0.465      1   0.330  NA           0
#>  6 PT2     VIS2   Asian      Male     45.0  0.233      2  -0.820  31.5         0
#>  7 PT2     VIS3   Asian      Male     45.0  0.360      3   0.487  36.9         0
#>  8 PT2     VIS4   Asian      Male     45.0  0.507      4   0.738  48.8         0
#>  9 PT3     VIS1   Black or … Fema…    43.5  0.682      1   0.576  NA           0
#> 10 PT3     VIS2   Black or … Fema…    43.5  0.892      2  -0.305  36.0         0
#> # ℹ 790 more rows
#> # ℹ 3 more variables: ARMCD_TRT_x_AVISITVIS2 <dbl>,
#> #   ARMCD_TRT_x_AVISITVIS3 <dbl>, ARMCD_TRT_x_AVISITVIS4 <dbl>

We can pass the covariance structure as well in the set_engine() definition. This allows for more flexibility on presetting different covariance structures in the pipeline while keeping the data preparation step independent.

mmrm_spec_with_cov <- linear_reg() |>
  set_engine(
    "mmrm",
    method = "Satterthwaite",
    covariance = as.cov_struct(~ us(AVISIT | USUBJID))
  )

We combine these steps into a workflow:

(mmrm_wflow_nocov <- workflow() |>
  add_model(mmrm_spec_with_cov, formula = FEV1 ~ SEX) |>
  add_recipe(mmrm_recipe))
#> ══ Workflow ════════════════════════════════════════════════════════════════════
#> Preprocessor: Recipe
#> Model: linear_reg()
#> 
#> ── Preprocessor ────────────────────────────────────────────────────────────────
#> 2 Recipe Steps
#> 
#> • step_dummy()
#> • step_interact()
#> 
#> ── Model ───────────────────────────────────────────────────────────────────────
#> Linear Regression Model Specification (regression)
#> 
#> Engine-Specific Arguments:
#>   method = Satterthwaite
#>   covariance = as.cov_struct(~us(AVISIT | USUBJID))
#> 
#> Computational engine: mmrm

Last step is to fit the data with the workflow object

(fit_tidy <- fit(mmrm_wflow_nocov, data = fev_data))
#> ══ Workflow [trained] ══════════════════════════════════════════════════════════
#> Preprocessor: Recipe
#> Model: linear_reg()
#> 
#> ── Preprocessor ────────────────────────────────────────────────────────────────
#> 2 Recipe Steps
#> 
#> • step_dummy()
#> • step_interact()
#> 
#> ── Model ───────────────────────────────────────────────────────────────────────
#> mmrm fit
#> 
#> Formula:     FEV1 ~ SEX
#> Data:        data (used 537 observations from 197 subjects with maximum 4 
#> timepoints)
#> Weights:     weights
#> Covariance:  unstructured (10 variance parameters)
#> Inference:   REML
#> Deviance:    3699.803
#> 
#> Coefficients: 
#> (Intercept)   SEXFemale 
#> 42.80540973  0.04513432 
#> 
#> Model Inference Optimization:
#> Converged with code 0 and message: convergence: rel_reduction_of_f <= factr*epsmch

To retrieve the fit object from within the workflow object run the following

fit_tidy |>
  hardhat::extract_fit_engine()
#> mmrm fit
#> 
#> Formula:     FEV1 ~ SEX
#> Data:        data (used 537 observations from 197 subjects with maximum 4 
#> timepoints)
#> Weights:     weights
#> Covariance:  unstructured (10 variance parameters)
#> Inference:   REML
#> Deviance:    3699.803
#> 
#> Coefficients: 
#> (Intercept)   SEXFemale 
#> 42.80540973  0.04513432 
#> 
#> Model Inference Optimization:
#> Converged with code 0 and message: convergence: rel_reduction_of_f <= factr*epsmch

Acknowledgments

The mmrm package is based on previous work internal in Roche, namely the tern and tern.mmrm packages which were based on lme4. The work done in the rbmi package has been important since it used glmmTMB for fitting MMRMs.

We would like to thank Ben Bolker from the glmmTMB team for multiple discussions when we tried to get the Satterthwaite degrees of freedom implemented with glmmTMB (see https://github.com/glmmTMB/glmmTMB/blob/satterthwaite_df/glmmTMB/vignettes/satterthwaite_unstructured_example2.Rmd). Also Ben helped us significantly with an example showing how to use TMB for a random effect vector (https://github.com/bbolker/tmb-case-studies/tree/master/vectorMixed).

References