Information Matrix Test

This vignette reproduces the example shows the use of the information matrix test when collapsing categories of a Partial Credit Model.

library(IMTest)
## Loading required package: ltm
## Loading required package: MASS
## Loading required package: msm
## Loading required package: polycor
library(lme4)
## Loading required package: Matrix
library(reshape2)

A commonly analyzed dataset in the illustration of IRT methods consists of 316 responses to 24 items referring to verbally aggressive reactions to situations that may be frustrating. Responses to the items are collected on a three point ordinal scale, with categories labelled as yes, perhaps, or no. The dataset is publicly available, for example through the lme4 package in R. This vignette analyzes item responses to the first 12 items, and investigates the potential model misspecification when collapsing categories of those items.

data(VerbAgg)

x = dcast(VerbAgg[,c( "id","item", "resp")], id~item)
## Using resp as value column: use value.var to override.
for(i in 2:25){
  x[,i] = factor(x[,i], levels = c("no", "perhaps", "yes"))
}

my_data = x[,2:13]

J = dim(my_data)[[2]]
n = dim(my_data)[[1]]

First, a set of index vectors is created to test both item-level parameters of each item separately.

# Create index vectors
ind = list()
for(i in 1:J){
  ind[[i]] = rep(0, J*2)
  ind[[i]][(2*i-1):(2*i)] = 1
}

A collapsing function is generated, showing that no collapsing has occurred, and the gpcm_IMT function is run. A matrix is initialized to store the p-values.

col = split(rep(c(1:3), J), rep(1:J, each = 3))
temp_data = collapse_data(my_data, col, "rasch")
mod = gpcm_IMT(temp_data$data, constraint = "rasch")

## Creates a matrix to store results.
pval = matrix(c(0), nrow = J, ncol = 3)

The following set of code runs the IMT for the uncollapsed data, and for two forms of collapsing: collapsing the middle category up, and collapsing the middle category down.

# No Collapsing
for(i in 1:J){
  test_fit = IMT(mod, "rasch", R = 5000, ind[[i]])
  pval[i,1] = pchisq(test_fit$Tstat, test_fit$df, lower.tail = F)
}

# Collapsing the middle categoy up.
for(i in 1:J){
  col = split(rep(c(1:3), J), rep(1:J, each = 3))
  col[[i]] = c(1,2,2)
  temp_data = collapse_data(my_data, col, "rasch")
  mod = gpcm_IMT(temp_data$data, constraint = "rasch")
  test_fit = IMT(mod, "rasch", R = 5000, temp_data$ind)
  pval[i,2] = pchisq(test_fit$Tstat, test_fit$df, lower.tail = F)
}

# Collapsing the middle category down.
for(i in 1:J){
  col = split(rep(c(1:3), J), rep(1:J, each = 3))
  col[[i]] = c(1,1,2)
  temp_data = collapse_data(my_data, col, "rasch")
  mod = gpcm_IMT(temp_data$data, constraint = "rasch")
  test_fit = IMT(mod, "rasch", R = 5000, temp_data$ind)
  pval[i,3] = pchisq(test_fit$Tstat, test_fit$df, lower.tail = F)
}

Originally, all but one item fails to reject goodness-of-fit. After collapsing in one direction, four items now reject goodness-of-fit. After collapsing in the other direction, no items reject goodness-of-fit.

# Original fit: all but question 8 fail to reject goodness-of-fit.
p.adjust(pval[,1], method = "fdr")
##  [1] 0.35758698 0.66462508 0.66462508 0.68084654 0.66462508 0.68084654
##  [7] 0.35758698 0.01586503 0.66462508 0.35758698 0.66462508 0.66462508
# After collapsing: question 8 now fails to reject goodness-of-fit, but 4 other questions reject.
p.adjust(pval[,2], method = "fdr")
##  [1] 0.01883634 0.01883634 0.32091366 0.02561956 0.01416300 0.35456531
##  [7] 0.23015659 0.14844349 0.65034862 0.66747176 0.08360686 0.79649680
# Had it been collapsed the other direction, all would fail to reject.
p.adjust(pval[,3], method = "fdr")
##  [1] 0.07232734 0.05944036 0.64898789 0.25289965 0.15327211 0.15327211
##  [7] 0.41280270 0.37245575 0.17008332 0.41280270 0.33396154 0.25360708