R/xxirt_newton_raphson.R

Defines functions xxirt_newton_raphson

## File Name: xxirt_newton_raphson.R
## File Version: 0.261


xxirt_newton_raphson <- function(em_out, em_args, maxit_nr, optimizer_nr,
            control_nr, verbose=TRUE, estimator="ML")
{
    partable <- xxirt_em_args_extract(em_args=em_args, em_out=em_out,
                            object='partable')
    customTheta <- xxirt_em_args_extract(em_args=em_args, em_out=em_out,
                            object='customTheta')
    G <- em_args$G
    Theta <- em_args$Theta
    items <- em_args$items
    item_list <- em_args$item_list
    ncat <- em_args$ncat
    partable_index <- em_args$partable_index
    dat <- em_args$dat
    dat_resp_bool <- em_args$dat_resp_bool
    resp_index <- em_args$resp_index
    group <- em_args$group
    weights <- em_args$weights

    par1 <- par_items <- xxirt_partable_extract_freeParameters( partable=partable )
    par2 <- par_Theta <- xxirt_parTheta_extract_freeParameters( customTheta=customTheta )
    par0 <- par <- c(par1, par2)

    NPI <- length(par_items)
    NPT <- length(par_Theta)
    NP <- NPI+NPT

    em_args$parindex_items <- 1L:NPI
    if (NPI==0){
        em_args$parindex_items <- NULL
    }

    em_args$parindex_Theta <- (NPI+1):(NPI+NPT)
    x <- par0
    # define parameter tables for estimation
    I <- ncol(dat)
    partable$item_group_comp <- 0
    for (ii in 1L:I){
        ind_ii <- which(partable$itemnr==ii)
        partable$item_group_comp[ ind_ii ] <- 1L:length(ind_ii)
    }
    partable$item_group_comp[ partable$parfree==0 ] <- 0
    t2 <- table(partable$parindex)
    i3 <- as.numeric( names(t2)[ t2 > 1] )
    partable$item_group_comp[ partable$parindex %in% i3 ] <- 0

    MIGC <- max(partable$item_group_comp)

    em_args$partable <- partable
    em_args$MIGC <- MIGC
    em_args$customTheta <- customTheta
    em_args$NPI <- NPI
    em_args$NPT <- NPT
    em_args$NP <- NP

    i1 <- stats::aggregate(partable$itemnr, list(partable$parindex), min )
    i2 <- stats::aggregate(partable$itemnr, list(partable$parindex), max )

    if (NPI>0){
        free_pars_design <- data.frame( pid=1L:NPI, type='item', parlabel=names(par1) )
        free_pars_design$one_item <- i1[,2]==i2[,2]
        free_pars_design$itemnr <- ifelse(free_pars_design$one_item, i1[,2], -9 )
        partable_free <- partable[ partable$parfree==1 & partable$est, ]
        free_pars_design$item_group_comp <- partable_free$item_group_comp
    } else {
        free_pars_design <- NULL
    }

    em_args$free_pars_design <- free_pars_design
    em_args$group0 <- em_args$group - 1

    test <- TRUE
    test <- FALSE

    ##** select functions
    if (estimator=='ML'){
        opt_fun <- xxirt_nr_optim_fun
        grad_fun <- xxirt_nr_grad_fun_Rcpp
    }
    if (estimator=='PML'){
        opt_fun <- xxirt_nr_pml_opt_fun
        res <- xxirt_nr_pml_more_arguments( em_args=em_args,
                                            grad_fun=xxirt_nr_pml_grad_fun)
        em_args <- res$em_args
        grad_fun <- res$grad_fun
        pml_args <- em_args$pml_args
    }



    if (test){
        ll1 <- opt_fun(x=x, em_args=em_args)
        grad1 <- xxirt_nr_grad_fun_numapprox(x=x, em_args=em_args, opt_fun=opt_fun)
        grad2 <- grad_fun(x=x, em_args=em_args)
        Revalpr_round('grad1-grad2',5)
    }

    #-- optimize
    control_nr$maxit <- maxit_nr
    partable_free <- partable[ partable$parfree==1, ]

    if (NPI>0){
        lower <- c( partable_free$lower, customTheta$lower )
        upper <- c( partable_free$upper, customTheta$upper )
    } else {
        lower <- c( customTheta$lower )
        upper <- c( customTheta$upper )
    }
    if (!is.null(lower)){ names(lower) <- names(x) }
    if (!is.null(upper)){ names(upper) <- names(x) }

    if (verbose){
        cat( paste0('****** Newton-Raphson Optimization ********\n'))
        utils::flush.console()
    }
    res_opt_nr <- sirt_optimizer(optimizer=optimizer_nr,
                            par=x, fn=opt_fun, grad=grad_fun, hessian=FALSE,
                            control=control_nr, em_args=em_args,
                            lower=lower, upper=upper )

    #-- collect output
    res_opt_nr$opt_fun <- opt_fun
    res_opt_nr$grad_fun <- grad_fun
    x <- res_opt_nr$par
    partable <- xxirt_partable_include_freeParameters( partable=em_args$partable,
                                        x=x[ em_args$parindex_items ] )
    customTheta <- xxirt_parTheta_include_freeParameters(
                                customTheta=em_args$customTheta,
                                x=x[ em_args$parindex_Theta ])

    #-- optimization function values
    opt_values <- opt_fun(x=x, em_args=em_args, output_all=TRUE)

    #-- item probabilities
    probs_items <- xxirt_compute_itemprobs( item_list=item_list,
                            items=items, Theta=Theta, ncat=ncat,
                            partable=partable, partable_index=partable_index )

    #--- output
    res <- list(res_opt_nr=res_opt_nr, partable=partable, customTheta=customTheta,
                    opt_values=opt_values, probs_items=probs_items,
                    em_args=em_args)
    return(res)
}
alexanderrobitzsch/sirt documentation built on April 23, 2024, 2:31 p.m.