inst/itemModelExplorer/server.R

library(rpf)
library(ggplot2)
library(reshape2)

#options(shiny.reactlog=TRUE)

verbose <- FALSE

moveSomeParameter <- function(input, state, whichPar, newValue) {
  pi <- rpf.paramInfo(isolate(state$spec), match(whichPar, names(isolate(state$par))))
  if (!is.na(pi$lower) && newValue < pi$lower) newValue <- pi$lower
  if (!is.na(pi$upper) && newValeu > pi$upper) newValue <- pi$upper
  if (verbose) cat("moveParameter", whichPar,"to",newValue, fill=T)
  oldValue <- isolate(state$par[whichPar])
  doit <- newValue != oldValue
  if (doit) {
    state$par[whichPar] <- newValue
  }
  return(doit)
}

moveParameter <- function(input, state, newValue) {
  whichPar <- isolate(input$editPar)
  moveSomeParameter(input, state, whichPar, newValue)
}

shinyServer(function(input, output, session) {
  state <- reactiveValues(spec = NULL, par = NULL)
  
  observe({
     newVal <- input$editParValue
     if (is.null(isolate(state$spec))) return()
     if (!is.numeric(newVal)) return()
     moveParameter(input, state, newVal)
   })
  
  observe({
    hit <- input$setParValue0
    if (is.null(isolate(state$spec))) return()
    if (verbose) cat("set0",hit,fill=T)
    if (moveParameter(input, state, 0)) {
      updateSliderInput(session, "editParValue", value = 0)
    }
  })
  
  observe({
    hit <- input$setParValue1
    if (is.null(isolate(state$spec))) return()
    if (verbose) cat("set1",hit,fill=T)
    if (moveParameter(input, state, 1)) {
      updateSliderInput(session, "editParValue", value = 1)
    }
  })
  
  observe({
    hit <- input$setAllValue0
    if (is.null(isolate(state$spec))) return()
    for (pname in names(isolate(state$par))) {
      doit <- moveSomeParameter(input, state, pname, 0)
      if (pname == isolate(input$editPar) && doit) {
        updateSliderInput(session, "editParValue", value = 0)
      }
    }
  })
  

  observe({
    if (verbose) cat("reset to new model/outcome", fill=T)
    Ta <- input$nominalTa
    Tc <- input$nominalTc
    outcomes <- input$outcomes
    if (Tc == "partial credit") {
      Tc <- lower.tri(diag(outcomes-1),TRUE) * -1
    }
    spec <- switch(input$model,
                   'dichotomous' = rpf.drm(),
                   'graded' = rpf.grm(outcomes),
                   'nominal' = rpf.nrm(outcomes, T.a=Ta, T.c=Tc))
    par <- rpf.rparam(spec)
    
    parNames <- names(rpf.rparam(spec))
    updateSelectInput(session, "editPar", choices = parNames, selected = parNames[1])
    
    state[['spec']] <- spec
    state[['par']] <- par
  })
  
  observe({
    hit <- input$drawNewParametersAction
    if (hit == 0) return()
    if (hit %% 2 == 1 && isolate(input$model) == "nominal") {
      spec <- isolate(state[['spec']])
      par <- rpf.rparam(spec)
      par[2] <- .4/par[1]
      par[3:spec$outcomes] <- 0  # zero alf2..alfn parameters (simpler)
      state[['par']] <- par
    } else {
      state[['par']] <- isolate(rpf.rparam(state[['spec']]))
    }
  })

  observe({
    whichPar <- input$editPar
    val <- isolate(state$par[whichPar])
    if (verbose) cat("switch editing parameter to", whichPar, val, fill = TRUE)
    updateSliderInput(session, "editParValue",
                      label = paste("Parameter", whichPar), value = as.numeric(val))
  })

  output$parView <- renderTable({
    data.frame(par=state$par)
  })
  
  output$plot1 <- renderPlot({
    width <- 4
    grid <- expand.grid(theta=seq(-width,width,.1))
    
    trace <- try(rpf.prob(state$spec, state$par, grid$theta), silent = TRUE)
    if (inherits(trace, "try-error") || any(is.na(trace))) {
      pl <- ggplot(grid, aes(theta, 0)) + geom_line() + ylim(0,1) +
        geom_text(label="Invalid parameters", y=.5, x=0, size=14, color="red")
      return(pl)
    }
    grid <- cbind(grid, t(trace))
    grid2 <- melt(grid, id.vars=c("theta"), variable.name="category", value.name="p")
    
    ggplot(grid2, aes(theta, p, color=category)) + geom_line() +
      ylim(0,1) + xlim(-width, width)  + theme(legend.position="none")
  })
  
  output$info <- renderPlot({
    width <- 4
    grid <- expand.grid(theta=seq(-width,width,.1))
    trace <- try(rpf.info(state$spec, state$par, t(grid$theta)), silent = TRUE)
    if (inherits(trace, "try-error") || any(is.na(trace))) {
      pl <- ggplot(grid, aes(theta, 0)) + geom_line() + ylim(0,1) +
        geom_text(label="Invalid parameters", y=.5, x=0, size=14, color="red")
      return(pl)
    }
    grid <- cbind(grid, information=trace)
    
    ggplot(grid, aes(theta, information)) + geom_line() +
      xlim(-width, width)
  })
})

# runApp('.',display.mode="showcase")

Try the ifaTools package in your browser

Any scripts or data that you put into this service are public.

ifaTools documentation built on Jan. 20, 2022, 1:08 a.m.