--- title: "Getting started" output: rmarkdown::html_vignette: css: vignette.css bibliography: [references.bib] biblio-style: apalike link-citations: true vignette: > %\VignetteIndexEntry{Getting started} %\VignetteEngine{knitr::rmarkdown} %\VignetteEncoding{UTF-8} --- ```{r setup, include=FALSE} knitr::opts_chunk$set(echo = TRUE, fig.retina = 2) # options(width = 80) suppressPackageStartupMessages(library(dplyr)) suppressPackageStartupMessages(library(ggplot2)) ``` ## Overview The D-score is a one-number summary measure of early child development. The D-score has a fixed unit. In principle, we may use the D-score to [answer questions on the individual, group and population level](https://d-score.org/dbook1/sec-questions.html), but be aware that no instruments have yet been validated for individual application. For more background, see the introductory booklet [D-score: Turning milestones into measurement](https://d-score.org/dbook1/). This vignette shows how to estimate the D-score and the D-score age-adjusted Z-score (DAZ) from child data on developmental milestones. The vignette covers some typical actions needed when estimating the D-score and DAZ: 1. Identify whether the `dscore` package covers your measurement instrument; 2. Map your variable names to the GSED 9-position schema; 3. Calculate D-score and DAZ; 4. Summarise your results. ## Is your measurement instrument covered? The `dscore` package covers a subset of all possible assessment instruments. Moreover, it may have a restricted age range for a given instrument. Your first task is to evaluate whether the current `dscore` package can convert your measurements into D-scores. A 2017 Worldbank report [@fernald2017] identified 147 instruments for assessing the development of children aged 0-8 years. Well-known examples include the *Bayley Scales for Infant and Toddler Development* and the *Ages & Stages Questionnaires*. The D-score is defined by and calculated from, subsets of milestones from such instruments. Assessment instruments connect to the D-score through a *measurement model*, the Rasch model. We use the term **key** to refer to a particular set of parameters (difficulty estimates) of a fitted Rasch model. The key defines how 0/1 milestone scores are translated into a D-score, as thus can be seen as part of the scoring system. The `dscore` package contains a generic algorithm that takes: 1. the 0/1 scores on a set of milestones for a given child; 2. the age(s) of the child at which the test is administered; 3. a specification of the key; and then calculates the **D-score** and its **Standard Error of Measurement (SEM)** for the child at each age. The `dscore` package currently supports the following keys (in historic order): | Key name | Description | Reference | |:------------|:--------------------------------------------------------------------------------------------|:------------------------| | `dutch` | A model developed for the *Dutch development instrument* | @vanbuuren2014 | | `gcdg` | A model covering 13 instruments using *direct* measurements | @weber2019 | | `gsed1912` | Covers 20 instruments using a mix of *direct* and *caregiver-reported* measurements | @vanbuuren2025 (Study 1)| | `gsed2212` | Covers 23 instruments using a mix of *direct* and *caregiver-reported* measurements | @vanbuuren2025 (Study 2)| | `gsed2406` | Covers 23 instruments using a mix of *direct* and *caregiver-reported* measurements. Same as gsed2212, but with a different base population. | @vanbuuren2025 (Study 2)| | `gsed2510` | Currently covers 2 instruments - In development | In preparation (Study 3)| Different keys lead to (slightly) different D-scores. We may only compare D-scores that are calculated under the same key. Our advice to set the key is: - For new data, use the generic `key = "gsed"`. This choice will automatically fetch the latest GSED key, which is `key = "gsed2510"`; - Key `gsed2406` has a wider coverage, Use that if your instrument is not covered by `gsed2510`. - Use older keys `dutch`, `gcdg` or `gsed1912` to regenerate old results. These are unlikely to be useful for new data; - Superseded keys are: `gsed2212`, `gsed2206`, `gsed2208`, `lf2206`, `sf2006`, `294_0` and `293_0`. Use for research purposes only. The table given below displays the number of items per instrument for various keys. If the entry is blank, the key does not cover the instrument. | Code | Instrument | Items | dutch | gcdg | gsed1912 | gsed2406 | gsed2510 | |-------|-------------------------|------:|------:|------:|------:|------:|------:| | `aqi` | Ages & Stages Questionnaires-3 | 230 | | 29 | 17 | 17 | | | `bar` | Barrera Moncada | 22 | | 15 | 13 | 13 | | | `bat` | Battelle Development Inventory and Screener-2 | 137 | | | | | | | `by1` | Bayley Scales for Infant and Toddler Development-1 | 156 | | 85 | 76 | 76 | | | `by2` | Bayley Scales for Infant and Toddler Development-2 | 121 | | 16 | 16 | 16 | | | `by3` | Bayley Scales for Infant and Toddler Development-3 | 320 | | 105 | 67 | 172 | | | `cro` | Caregiver Reported Early Development Instrument (CREDI) | 149 | | | 62 | 64 | | | `ddi` | Dutch Development Instrument (Van Wiechen Schema) | 77 | 76 | 65 | 64 | 64 | | | `den` | Denver-2 | 111 | | 67 | 50 | 50 | | | `dmc` | Developmental Milestones Checklist | 66 | | | 43 | 43 | | | `gri` | Griffiths Mental Development Scales | 312 | | 104 | 93 | 18 | | | `gs1` | GSED SF (2023) | 139 | | | | 138 | 136 | | `gl1` | GSED LF (2023) | 155 | | | | 155 | 145 | | `gh1` | GSED HF (2023) | 55 | | | | 55 | | | `iyo` | Infant and Young Child Development (IYCD) | 90 | | | 55 | 57 | | | `kdi` | Kilifi Developmental Inventory | 69 | | | 48 | 48 | | | `mac` | MacArthur Communicative Development Inventory | 6 | | 3 | 3 | 3 | | | `mds` | WHO Motor Development Milestones | 6 | | | 1 | 1 | | | `mdt` | Malawi Developmental Assessment Tool (MDAT) | 136 | | | 126 | 126 | | | `peg` | Pegboard | 2 | | 1 | 1 | 1 | | | `pri` | Project on Child Development Indicators (PRIDI) | 63 | | | | | | | `sbi` | Stanford Binet Intelligence Scales-4/5 | 33 | | 6 | 5 | 5 | | | `sgr` | Griffiths for South Africa | 58 | | 19 | 19 | 19 | | | `tep` | Test de Desarrollo Psicomotor (TEPSI) | 61 | | 33 | 31 | 31 | | | `vin` | Vineland Social Maturity Scale | 50 | | 17 | 17 | 17 | | | | | | 76 | 565 | 807 | 818 | 281 | | | Extensions | | | | | | | | `ecd` | Early Child Development Indicators (ECDI) | 20 | | | | 18 | | | `mul` | Mullen Scales of Early Learning | 232 | | | 138 | | | You can’t compute a D-score if your instrument isn’t supported. If it does measure child development, we can often extend the key to include it. This requires a one-time effort—mapping your items to the milestone schema and estimating difficulty parameters. Once added, the key is reusable, and you’ll be able to compute valid D-scores for that instrument. The table shows two such extensions: one for Mullen and one for ECDI. ```{r graphkey, fig.width = 7, fig.height = 5, echo = FALSE} #| code-fold: true library(dscore) ib <- builtin_itembank |> filter(key == "gsed2406") |> mutate( a = get_age_equivalent( items = item, pct = 50, itembank = builtin_itembank )$a, a = a * 12 ) |> select(a, instrument, label) |> na.omit() ggplot(ib, aes(x = a, y = instrument, group = instrument)) + scale_y_discrete(limits = rev(unique(ib$instrument)), name = "") + scale_x_continuous( limits = c(0, 60), breaks = seq(0, 60, 12), name = "Age (months)" ) + geom_point(pch = 3, size = 1, colour = "blue") + theme_light() + theme(axis.text.y = element_text(hjust = 0, family = "mono")) ``` The designs of the original cohorts determine the age coverage for each instrument. The figure above indicates the age range currently supported by the `"gsed2406"` key. Some instruments contain many items for the first two years (e.g., `by1`, `dmc`), whereas others cover primarily upper ages (e.g., `tep`, `ecd`). ## GSED 9-position item names The `dscore()` function accepts item names that follow the GSED 9-position schema. A name with a length of nine characters identifies every milestone. The following table shows the construction of names. | Position | Description | Example | |---------:|:---------------------|:--------| | 1-3 | instrument | `by3` | | 4-5 | developmental domain | `cg` | | 6 | administration mode | `d` | | 7-9 | item number | `018` | Thus, item `by3cgd018` refers to the 18th item in the cognitive scale of the Bayley-III. The label of the item can be obtained by ```{r getlabels} library(dscore) get_labels("by3cgd018") ``` You may decompose item names into components as follows: ```{r decompose_itemnames} decompose_itemnames(c("by3cgd018", "denfmd014")) ``` This function returns a `data.frame` with four character vectors for further processing. The `dscore` package recognises `r nrow(dscore::builtin_itemtable)` item names. The expression `get_itemnames()` returns a (long) vector of all known item names. Let us study the table of instruments by domain: ```{r table} #| code-fold: true items <- get_itemnames() din <- decompose_itemnames(items) |> dplyr::filter(!instrument %in% c("gsd", "gpa", "gto", "rap")) knitr::kable(with(din, table(instrument, domain)), format = "html") |> kableExtra::column_spec(1, monospace = TRUE) ``` We obtain the first three item names and labels from the `mdt` domain `gm` as ```{r ddigm} items <- head(get_itemnames(instrument = "mdt", domain = "gm"), 3) get_labels(items) ``` In practice, you need to spend some time to figure out how item names in your data map to those in the `dscore` package. Once you've completed this mapping, rename the items into the GSED 9-position schema. For example, suppose that your first three gross motor MDAT items are called `mot1`, `mot2`, and `mot3`. ```{r smalldataset} data <- data.frame( id = c(1, 1, 2), age = c(1, 1.6, 0.9), mot1 = c(1, NA, NA), mot2 = c(0, 1, 1), mot3 = c(NA, 0, 1) ) data ``` Renaming is easy to do by changing the names attribute. ```{r rename} old_names <- names(data)[3:5] new_names <- get_itemnames(instrument = "mdt", domain = "gm")[1:3] names(data)[3:5] <- new_names data ``` There may be different versions and revision of the same instrument. Therefore, carefully check whether the item labels match up with the labels in version of the instrument that you administered. The `dscore` package assumes that response to milestones are dichotomous (1 = PASS, 0 = FAIL). If necessary, recode your data to match these response categories. ## Calculate the D-score and DAZ Once the data are in proper shape, calculation of the D-score and DAZ is easy. The `milestones` dataset in the `dscore` package contains responses of 27 preterm children measured at various age between birth and 2.5 years on the Dutch Development Instrument (`ddi`). The dataset looks like: ```{r milestones} head(milestones[, c(1, 3, 4, 9:14)]) ``` Each row corresponds to a visit. Most children have three or four visits. Columns starting with `ddi` hold the responses on DDI-items. A `1` means a PASS, a `0` means a FAIL, and `NA` means that the item was not administered. The `milestones` dataset has properly named columns that identify each item. Calculating the D-score and DAZ is then done by: ```{r} ds <- dscore(milestones, population = "dutch", key = "dutch") dim(ds) ``` Where `ds` is a `data.frame` with the same number of rows as the input data. The first six rows are ```{r} head(ds) ``` In addition, the package calculate the **Development-for-Age Z-score (DAZ)**, which gives the position of the child relative to age peers. The table below provides the interpretation of the output: | Name | Interpretation | |------------------|------------------------------------------------------| | `a` | Decimal age | | `n` | number of items used to calculate D-score | | `p` | Percentage of passed milestones | | `d` | D-score estimate, mean of posterior | | `sem` | Standard error of measurement, standard deviation of the posterior | | `daz` | D-score corrected for age | ## Summarise D-score and DAZ Combine the `milestones` data and the result by ```{r bind} md <- cbind(milestones, ds) ``` We may plot the 27 individual developmental curves by ```{r graphD, fig.width = 7, fig.height = 7} #| code-fold: true library(ggplot2) library(dplyr) # Prepare the reference ribbon data: sort by age and convert months → years r <- builtin_references %>% filter(population == "dutch" & key == "dutch") %>% transmute(age_years = age, SDM2, SD0, SDP2) %>% arrange(age_years) ggplot(md, aes(x = a, y = d, group = id, colour = sex)) + theme_light() + theme( legend.position = c(0.85, 0.15), legend.background = element_blank(), legend.key = element_blank() ) + geom_ribbon( data = r, inherit.aes = FALSE, aes(x = age_years, ymin = SDM2, ymax = SDP2), fill = "green", alpha = 0.1 ) + geom_line( data = r, inherit.aes = FALSE, aes(x = age_years, y = SDM2), linewidth = 0.3, alpha = 0.6, colour = "green" ) + geom_line( data = r, inherit.aes = FALSE, aes(x = age_years, y = SDP2), linewidth = 0.3, alpha = 0.6, colour = "green" ) + geom_line( data = r, inherit.aes = FALSE, aes(x = age_years, y = SD0), linewidth = 0.5, alpha = 0.8, colour = "green" ) + coord_cartesian(xlim = c(0, 2.5)) + ylab(expression(italic(D) * "-score")) + xlab("Age (years)") + scale_color_brewer(palette = "Set1") + geom_line(linewidth = 0.1) + geom_point(size = 1) ``` Note that similarity of these curves to growth curves for body height and weight. The DAZ is an age-standardized D-score with a standard normal distribution with mean 0 and variance 1. We plot the individual DAZ curves relative to the Dutch references by ```{r graphDAZ, fig.width = 7, fig.height = 5} #| code-fold: true ggplot(md, aes(x = a, y = daz, group = id, color = factor(sex))) + theme_light() + theme(legend.position = c(.85, .15)) + theme(legend.background = element_blank()) + theme(legend.key = element_blank()) + scale_color_brewer(palette = "Set1") + annotate( "rect", xmin = -Inf, xmax = Inf, ymin = -2, ymax = 2, alpha = 0.1, fill = "green" ) + coord_cartesian( xlim = c(0, 2.5), ylim = c(-4, 4) ) + geom_line(lwd = 0.1) + geom_point(size = 1) + xlab("Age (in years)") + ylab("DAZ") ``` Note that the D-scores and DAZ are a little lower than average. The explanation here is that these all children are born preterm. We can [account for prematurity](https://d-score.org/dbook1/sec-pops.html#age-adjustment) by correcting for gestational age. The distributions of DAZ for boys and girls show that a large overlap: ```{r density, fig.width = 7, fig.height = 4} #| code-fold: true ggplot(md) + theme_light() + scale_fill_brewer(palette = "Set1") + geom_density( aes(x = daz, group = sex, fill = sex), alpha = 0.4, adjust = 1.5, color = "transparent" ) + xlab("DAZ") ``` Under the assumption of independence, we may test whether sex differences are constant in age by a linear regression that includes the interaction between age and sex: ```{r independence} summary(lm(daz ~ age * sex, data = md)) ``` This group of children born very preterm starts around -2.5 SD, followed by a catch-up in child development of approximately 1.0 SD per year. The size of the catch-up is equal for boys and girls. We may account for the clustering effect by including random intercept and age effects, and rerun as ```{r multilevel} library(Matrix) library(lme4, quiet = TRUE) lmer(daz ~ 1 + age + sex + sex * age + (1 + age | id), data = md) ``` This analysis yields the same substantive conclusions as before. ## References