From da87df0362d9f2c26d9a0f3904673c9ac8df7b19 Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 8 May 2024 08:07:41 +0200 Subject: [PATCH 01/34] Revise mwu --- R/mann_whitney.R | 234 +++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 234 insertions(+) create mode 100644 R/mann_whitney.R diff --git a/R/mann_whitney.R b/R/mann_whitney.R new file mode 100644 index 0000000..6967bf7 --- /dev/null +++ b/R/mann_whitney.R @@ -0,0 +1,234 @@ +#' @title Mann-Whitney-U-Test +#' @name mwu +#' @description This function performs a Mann-Whitney-U-Test (or Wilcoxon rank +#' sum test for _unpaired_ samples, see [`wilcox.test()`] and [`coin::wilcox_test()`]) +#' comparing `x` by each group indicated by `grp`. If `grp` has more than two +#' categories, a comparison between each combination of two groups is performed. +#' +#' The function reports U, p and Z-values as well as effect size r and group-rank-means. +#' +#' @param x Bare (unquoted) variable name, or a character vector with the variable name. +#' @param distribution Indicates how the null distribution of the test statistic +#' should be computed. May be one of `"exact"`, `"approximate"` or `"asymptotic"` +#' (default). See [`coin::wilcox_test()`] for details. +#' +#' @inheritParams weighted_sd +#' @inheritParams means_by_group +#' +#' @return (Invisibly) returns a data frame with U, p and Z-values for each group-comparison +#' as well as effect-size r; additionally, group-labels and groups' n's are +#' also included. +#' +#' @note This function calls the \code{\link[coin]{wilcox_test}} with formula. If \code{grp} +#' has more than two groups, additionally a Kruskal-Wallis-Test (see \code{\link{kruskal.test}}) +#' is performed. \cr \cr +#' Interpretation of effect sizes, as a rule-of-thumb: +#' \itemize{ +#' \item small effect >= 0.1 +#' \item medium effect >= 0.3 +#' \item large effect >= 0.5 +#' } +#' +#' @examples +#' data(efc) +#' # Mann-Whitney-U-Tests for elder's age by elder's sex. +#' mwu(efc, e17age, e16sex) +#' +#' # using formula interface +#' mwu(e17age ~ e16sex, efc) +#' +#' # Mann-Whitney-Tests for elder's age by each level elder's dependency. +#' mwu(efc, e17age, e42dep) +#' +#' @importFrom stats na.omit wilcox.test kruskal.test +#' @importFrom sjmisc recode_to is_empty +#' @importFrom sjlabelled get_labels as_numeric +#' @importFrom rlang quo_name enquo +#' @export +mwu <- function(data, ...) { + UseMethod("mwu") +} + +#' @rdname mwu +#' @export +mwu.default <- function(data, + x, + grp, + distribution = "asymptotic", + out = c("txt", "viewer", "browser"), + encoding = "UTF-8", + file = NULL, + ...) { + + out <- match.arg(out) + + if (out != "txt" && !requireNamespace("sjPlot", quietly = TRUE)) { + message("Package `sjPlot` needs to be loaded to print HTML tables.") + out <- "txt" + } + + if (!requireNamespace("coin", quietly = TRUE)) { + stop("Package `coin` needs to be installed to compute the Mann-Whitney-U test.", call. = FALSE) + } + + + # create quosures + grp.name <- rlang::quo_name(rlang::enquo(grp)) + dv.name <- rlang::quo_name(rlang::enquo(x)) + + # create string with variable names + vars <- c(grp.name, dv.name) + + # get data + data <- suppressMessages(dplyr::select(data, !! vars)) + + grp <- data[[grp.name]] + dv <- data[[dv.name]] + + # coerce factor and character to numeric + if (is.factor(grp) || is.character(grp)) grp <- sjlabelled::as_numeric(grp) + + # group "counter" (index) should start with 1, not 0 + if (min(grp, na.rm = TRUE) < 1) grp <- sjmisc::recode_to(grp, lowest = 1, append = FALSE) + + # retrieve unique group values. need to iterate all values + grp_values <- sort(unique(stats::na.omit(grp))) + + # length of value range + cnt <- length(grp_values) + labels <- sjlabelled::get_labels( + grp, attr.only = FALSE, values = NULL, non.labelled = TRUE + ) + + df <- data.frame() + for (i in seq_len(cnt)) { + for (j in i:cnt) { + if (i != j) { + # retrieve cases (rows) of subgroups + xsub <- dv[which(grp == grp_values[i] | grp == grp_values[j])] + ysub <- grp[which(grp == grp_values[i] | grp == grp_values[j])] + + # this is for unpaired wilcox.test() + xsub_2 <- stats::na.omit(dv[which(grp == grp_values[i])]) + ysub_2 <- stats::na.omit(dv[which(grp == grp_values[j])]) + + # only use rows with non-missings + ysub <- ysub[which(!is.na(xsub))] + + # remove missings + xsub <- as.numeric(stats::na.omit(xsub)) + ysub.n <- stats::na.omit(ysub) + + # grouping variable is a factor + ysub <- as.factor(ysub.n) + + wcdat <- data.frame( + x = xsub, + y = ysub + ) + + # perfom wilcox test + wt <- coin::wilcox_test(x ~ y, data = wcdat, distribution = distribution) + + # compute statistics + u <- as.numeric(coin::statistic(wt, type = "linear")) + z <- as.numeric(coin::statistic(wt, type = "standardized")) + p <- coin::pvalue(wt) + r <- abs(z / sqrt(length(ysub))) + w <- stats::wilcox.test(xsub_2, ysub_2, paired = FALSE)$statistic + rkm.i <- mean(rank(xsub)[which(ysub.n == grp_values[i])], na.rm = TRUE) + rkm.j <- mean(rank(xsub)[which(ysub.n == grp_values[j])], na.rm = TRUE) + + # compute n for each group + n_grp1 <- length(xsub[which(ysub.n == grp_values[i])]) + n_grp2 <- length(xsub[which(ysub.n == grp_values[j])]) + + # generate result data frame + df <- + rbind( + df, + cbind( + grp1 = grp_values[i], + grp1.label = labels[i], + grp1.n = n_grp1, + grp2 = grp_values[j], + grp2.label = labels[j], + grp2.n = n_grp2, + u = u, + w = w, + p = p, + z = z, + r = r, + rank.mean.grp1 = rkm.i, + rank.mean.grp2 = rkm.j + ) + ) + } + } + } + + # convert variables + df[["grp1"]] <- as.numeric(as.character(df[["grp1"]])) + df[["grp2"]] <- as.numeric(as.character(df[["grp2"]])) + df[["grp1.n"]] <- as.numeric(as.character(df[["grp1.n"]])) + df[["grp2.n"]] <- as.numeric(as.character(df[["grp2.n"]])) + df[["grp1.label"]] <- as.character(df[["grp1.label"]]) + df[["grp2.label"]] <- as.character(df[["grp2.label"]]) + df[["u"]] <- as.numeric(as.character(df[["u"]])) + df[["w"]] <- as.numeric(as.character(df[["w"]])) + df[["p"]] <- as.numeric(as.character(df[["p"]])) + df[["z"]] <- as.numeric(as.character(df[["z"]])) + df[["r"]] <- as.numeric(as.character(df[["r"]])) + df[["rank.mean.grp1"]] <- as.numeric(as.character(df[["rank.mean.grp1"]])) + df[["rank.mean.grp2"]] <- as.numeric(as.character(df[["rank.mean.grp2"]])) + + # prepare a data frame that can be used for 'sjt.df'. + tab.df <- + data_frame( + Groups = sprintf("%s
%s", df$grp1.label, df$grp2.label), + N = sprintf("%s
%s", df$grp1.n, df$grp2.n), + 'Mean Rank' = sprintf("%.2f
%.2f", df$rank.mean.grp1, df$rank.mean.grp2), + 'Mann-Whitney-U' = as.character(df$u), + 'Wilcoxon-W' = as.character(df$w), + Z = sprintf("%.3f", df$z), + 'Effect Size' = sprintf("%.3f", df$r), + p = sprintf("%.3f", df$p) + ) + + # replace 0.001 with <0.001 + tab.df$p[which(tab.df$p == "0.001")] <- "<0.001" + + ret.df <- list(df = df, tab.df = tab.df, data = data.frame(dv, grp)) + + # save how to print output + attr(ret.df, "print") <- out + attr(ret.df, "encoding") <- encoding + attr(ret.df, "file") <- file + + if (out %in% c("viewer", "browser")) + class(ret.df) <- c("mwu", "sjt_mwu") + else + class(ret.df) <- c("mwu", "sj_mwu") + + ret.df +} + + +#' @importFrom dplyr select +#' @rdname mwu +#' @export +mwu.formula <- function(formula, + data, + distribution = "asymptotic", + out = c("txt", "viewer", "browser"), + encoding = "UTF-8", + file = NULL, + ...) { + vars <- all.vars(formula) + mwu(data, x = !! vars[1], grp = !! vars[2], distribution = distribution, out = out, encoding = encoding, file = file, ...) +} + + +#' @rdname mwu +#' @export +mannwhitney <- mwu From 50871902a89357f3e91c82fc3937591bd39b7966 Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 8 May 2024 08:44:06 +0200 Subject: [PATCH 02/34] mwu --- NAMESPACE | 1 + R/mann_whitney.R | 325 ++++++++++++++++++++++---------------------- man/mann_whitney.Rd | 68 +++++++++ 3 files changed, 232 insertions(+), 162 deletions(-) create mode 100644 man/mann_whitney.Rd diff --git a/NAMESPACE b/NAMESPACE index 2872f93..31a8e99 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -88,6 +88,7 @@ export(icc) export(inequ_trend) export(is_prime) export(link_inverse) +export(mann_whitney) export(mannwhitney) export(mean_n) export(means_by_group) diff --git a/R/mann_whitney.R b/R/mann_whitney.R index 6967bf7..54ac7d4 100644 --- a/R/mann_whitney.R +++ b/R/mann_whitney.R @@ -1,5 +1,5 @@ #' @title Mann-Whitney-U-Test -#' @name mwu +#' @name mann_whitney #' @description This function performs a Mann-Whitney-U-Test (or Wilcoxon rank #' sum test for _unpaired_ samples, see [`wilcox.test()`] and [`coin::wilcox_test()`]) #' comparing `x` by each group indicated by `grp`. If `grp` has more than two @@ -32,203 +32,204 @@ #' @examples #' data(efc) #' # Mann-Whitney-U-Tests for elder's age by elder's sex. -#' mwu(efc, e17age, e16sex) +#' mann_whitney(efc, e17age, e16sex) #' #' # using formula interface -#' mwu(e17age ~ e16sex, efc) +#' mann_whitney(e17age ~ e16sex, efc) #' #' # Mann-Whitney-Tests for elder's age by each level elder's dependency. -#' mwu(efc, e17age, e42dep) +#' mann_whitney(efc, e17age, e42dep) #' #' @importFrom stats na.omit wilcox.test kruskal.test #' @importFrom sjmisc recode_to is_empty #' @importFrom sjlabelled get_labels as_numeric #' @importFrom rlang quo_name enquo #' @export -mwu <- function(data, ...) { - UseMethod("mwu") -} - -#' @rdname mwu -#' @export -mwu.default <- function(data, - x, - grp, - distribution = "asymptotic", - out = c("txt", "viewer", "browser"), - encoding = "UTF-8", - file = NULL, - ...) { - - out <- match.arg(out) - - if (out != "txt" && !requireNamespace("sjPlot", quietly = TRUE)) { - message("Package `sjPlot` needs to be loaded to print HTML tables.") - out <- "txt" +mann_whitney <- function(data, + select = NULL, + by = NULL, + weights = NULL, + distribution = "asymptotic", + ...) { + insight::check_if_installed("datawizard") + + # check if "select" is in data + if (!select %in% colnames(data)) { + insight::format_error( + sprintf("Variable '%s' not found in data frame.", select), + .misspelled_string(select, colnames(data), "Maybe misspelled?") + ) } - - if (!requireNamespace("coin", quietly = TRUE)) { - stop("Package `coin` needs to be installed to compute the Mann-Whitney-U test.", call. = FALSE) + # check if "by" is in data + if (!by %in% colnames(data)) { + insight::format_error( + sprintf("Variable '%s' not found in data frame.", by), + .misspelled_string(by, colnames(data), "Maybe misspelled?") + ) + } + # check if "weights" is in data + if (!is.null(weights) && !weights %in% colnames(data)) { + insight::format_error( + sprintf("Weighting variable '%s' not found in data frame.", weights), + .misspelled_string(weights, colnames(data), "Maybe misspelled?") + ) } - - - # create quosures - grp.name <- rlang::quo_name(rlang::enquo(grp)) - dv.name <- rlang::quo_name(rlang::enquo(x)) - - # create string with variable names - vars <- c(grp.name, dv.name) # get data - data <- suppressMessages(dplyr::select(data, !! vars)) - - grp <- data[[grp.name]] - dv <- data[[dv.name]] - - # coerce factor and character to numeric - if (is.factor(grp) || is.character(grp)) grp <- sjlabelled::as_numeric(grp) - - # group "counter" (index) should start with 1, not 0 - if (min(grp, na.rm = TRUE) < 1) grp <- sjmisc::recode_to(grp, lowest = 1, append = FALSE) + dv <- data[[select]] + grp <- data[[by]] - # retrieve unique group values. need to iterate all values - grp_values <- sort(unique(stats::na.omit(grp))) + # coerce to factor and character to numeric + grp <- datawizard::to_factor(grp) # length of value range - cnt <- length(grp_values) labels <- sjlabelled::get_labels( - grp, attr.only = FALSE, values = NULL, non.labelled = TRUE + data[[by]], attr.only = FALSE, values = NULL, non.labelled = TRUE ) - df <- data.frame() - for (i in seq_len(cnt)) { - for (j in i:cnt) { - if (i != j) { - # retrieve cases (rows) of subgroups - xsub <- dv[which(grp == grp_values[i] | grp == grp_values[j])] - ysub <- grp[which(grp == grp_values[i] | grp == grp_values[j])] + .calculate_mwu(dv, grp, distribution, labels) +} - # this is for unpaired wilcox.test() - xsub_2 <- stats::na.omit(dv[which(grp == grp_values[i])]) - ysub_2 <- stats::na.omit(dv[which(grp == grp_values[j])]) - # only use rows with non-missings - ysub <- ysub[which(!is.na(xsub))] +.calculate_mwu <- function(dv, grp, distribution, labels) { + insight::check_if_installed("coin") + # prepare data + wcdat <- data.frame(dv, grp) + # perfom wilcox test + wt <- coin::wilcox_test(dv ~ grp, data = wcdat, distribution = distribution) + + # for rank mean + group_levels <- levels(grp) + + # compute statistics + u <- as.numeric(coin::statistic(wt, type = "linear")) + z <- as.numeric(coin::statistic(wt, type = "standardized")) + p <- coin::pvalue(wt) + r <- abs(z / sqrt(length(dv))) + w <- stats::wilcox.test(dv ~ grp, data = wcdat)$statistic + + # group means + rank_mean_1 <- mean(rank(dv)[which(grp == group_levels[1])], na.rm = TRUE) + rank_mean_2 <- mean(rank(dv)[which(grp == group_levels[2])], na.rm = TRUE) + + # compute n for each group + n_grp1 <- length(stats::na.omit(dv[which(grp == group_levels[1])])) + n_grp2 <- length(stats::na.omit(dv[which(grp == group_levels[2])])) + + out <- data.frame( + group1 = group_levels[1], + group_2 = group_levels[2], + u = u, + w = w, + p = p, + z = z, + r = r + ) + attr(out, "rank_means") <- stats::setNames( + c(rank_mean_1, rank_mean_2), + c("Mean Group 1", "Mean Group 2") + ) + attr(out, "n_groups") <- stats::setNames( + c(n_grp1, n_grp2), + c("N Group 1", "N Group 2") + ) + attr(out, "group_labels") <- labels + out +} - # remove missings - xsub <- as.numeric(stats::na.omit(xsub)) - ysub.n <- stats::na.omit(ysub) - # grouping variable is a factor - ysub <- as.factor(ysub.n) +.compute_weighted_mwu <- function(dv, grp, weights, labels) { + # check if pkg survey is available + insight::check_if_installed("survey") - wcdat <- data.frame( - x = xsub, - y = ysub - ) + dat <- data.frame(dv, grp, weights) + colnames(dat) <- c("x", "g", "w") - # perfom wilcox test - wt <- coin::wilcox_test(x ~ y, data = wcdat, distribution = distribution) - - # compute statistics - u <- as.numeric(coin::statistic(wt, type = "linear")) - z <- as.numeric(coin::statistic(wt, type = "standardized")) - p <- coin::pvalue(wt) - r <- abs(z / sqrt(length(ysub))) - w <- stats::wilcox.test(xsub_2, ysub_2, paired = FALSE)$statistic - rkm.i <- mean(rank(xsub)[which(ysub.n == grp_values[i])], na.rm = TRUE) - rkm.j <- mean(rank(xsub)[which(ysub.n == grp_values[j])], na.rm = TRUE) - - # compute n for each group - n_grp1 <- length(xsub[which(ysub.n == grp_values[i])]) - n_grp2 <- length(xsub[which(ysub.n == grp_values[j])]) - - # generate result data frame - df <- - rbind( - df, - cbind( - grp1 = grp_values[i], - grp1.label = labels[i], - grp1.n = n_grp1, - grp2 = grp_values[j], - grp2.label = labels[j], - grp2.n = n_grp2, - u = u, - w = w, - p = p, - z = z, - r = r, - rank.mean.grp1 = rkm.i, - rank.mean.grp2 = rkm.j - ) - ) - } - } + if (insight::n_unqiue(dat$g) > 2) { + m <- "Weighted Kruskal-Wallis test" + method <- "KruskalWallis" + } else { + m <- "Weighted Mann-Whitney-U test" + method <- "wilcoxon" } - # convert variables - df[["grp1"]] <- as.numeric(as.character(df[["grp1"]])) - df[["grp2"]] <- as.numeric(as.character(df[["grp2"]])) - df[["grp1.n"]] <- as.numeric(as.character(df[["grp1.n"]])) - df[["grp2.n"]] <- as.numeric(as.character(df[["grp2.n"]])) - df[["grp1.label"]] <- as.character(df[["grp1.label"]]) - df[["grp2.label"]] <- as.character(df[["grp2.label"]]) - df[["u"]] <- as.numeric(as.character(df[["u"]])) - df[["w"]] <- as.numeric(as.character(df[["w"]])) - df[["p"]] <- as.numeric(as.character(df[["p"]])) - df[["z"]] <- as.numeric(as.character(df[["z"]])) - df[["r"]] <- as.numeric(as.character(df[["r"]])) - df[["rank.mean.grp1"]] <- as.numeric(as.character(df[["rank.mean.grp1"]])) - df[["rank.mean.grp2"]] <- as.numeric(as.character(df[["rank.mean.grp2"]])) - - # prepare a data frame that can be used for 'sjt.df'. - tab.df <- - data_frame( - Groups = sprintf("%s
%s", df$grp1.label, df$grp2.label), - N = sprintf("%s
%s", df$grp1.n, df$grp2.n), - 'Mean Rank' = sprintf("%.2f
%.2f", df$rank.mean.grp1, df$rank.mean.grp2), - 'Mann-Whitney-U' = as.character(df$u), - 'Wilcoxon-W' = as.character(df$w), - Z = sprintf("%.3f", df$z), - 'Effect Size' = sprintf("%.3f", df$r), - p = sprintf("%.3f", df$p) - ) + design <- survey::svydesign(ids = ~0, data = dat, weights = ~w) + out <- survey::svyranktest(formula = x ~ g, design, test = method) - # replace 0.001 with <0.001 - tab.df$p[which(tab.df$p == "0.001")] <- "<0.001" + # for rank mean + group_levels <- levels(grp) - ret.df <- list(df = df, tab.df = tab.df, data = data.frame(dv, grp)) + design_mean1 <- survey::svydesign( + ids = ~0, + data = dat[dat$grp == group_levels[1]], + weights = ~w + ) + rank_mean_1 <- survey::svymean(~x, design_mean1) - # save how to print output - attr(ret.df, "print") <- out - attr(ret.df, "encoding") <- encoding - attr(ret.df, "file") <- file + design_mean2 <- survey::svydesign( + ids = ~0, + data = dat[dat$grp == group_levels[2]], + weights = ~w + ) + rank_mean_2 <- survey::svymean(~x, design_mean2) - if (out %in% c("viewer", "browser")) - class(ret.df) <- c("mwu", "sjt_mwu") - else - class(ret.df) <- c("mwu", "sj_mwu") + out$method <- m + attr(out, "rank_means") <- stats::setNames( + c(rank_mean_1, rank_mean_2), + c("Mean Group 1", "Mean Group 2") + ) - ret.df + out } -#' @importFrom dplyr select -#' @rdname mwu -#' @export -mwu.formula <- function(formula, - data, - distribution = "asymptotic", - out = c("txt", "viewer", "browser"), - encoding = "UTF-8", - file = NULL, - ...) { - vars <- all.vars(formula) - mwu(data, x = !! vars[1], grp = !! vars[2], distribution = distribution, out = out, encoding = encoding, file = file, ...) +.misspelled_string <- function(source, searchterm, default_message = NULL) { + if (is.null(searchterm) || length(searchterm) < 1) { + return(default_message) + } + # used for many matches + more_found <- "" + # init default + msg <- "" + # remove matching strings + same <- intersect(source, searchterm) + searchterm <- setdiff(searchterm, same) + source <- setdiff(source, same) + # guess the misspelled string + possible_strings <- unlist(lapply(searchterm, function(s) { + source[.fuzzy_grep(source, s)] # nolint + }), use.names = FALSE) + if (length(possible_strings)) { + msg <- "Did you mean " + if (length(possible_strings) > 1) { + # make sure we don't print dozens of alternatives for larger data frames + if (length(possible_strings) > 5) { + more_found <- sprintf( + " We even found %i more possible matches, not shown here.", + length(possible_strings) - 5 + ) + possible_strings <- possible_strings[1:5] + } + msg <- paste0(msg, "one of ", toString(paste0("\"", possible_strings, "\""))) + } else { + msg <- paste0(msg, "\"", possible_strings, "\"") + } + msg <- paste0(msg, "?", more_found) + } else { + msg <- default_message + } + # no double white space + insight::trim_ws(msg) } -#' @rdname mwu -#' @export -mannwhitney <- mwu +.fuzzy_grep <- function(x, pattern, precision = NULL) { + if (is.null(precision)) { + precision <- round(nchar(pattern) / 3) + } + if (precision > nchar(pattern)) { + return(NULL) + } + p <- sprintf("(%s){~%i}", pattern, precision) + grep(pattern = p, x = x, ignore.case = FALSE) +} diff --git a/man/mann_whitney.Rd b/man/mann_whitney.Rd new file mode 100644 index 0000000..7826dc7 --- /dev/null +++ b/man/mann_whitney.Rd @@ -0,0 +1,68 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/mann_whitney.R +\name{mann_whitney} +\alias{mann_whitney} +\title{Mann-Whitney-U-Test} +\usage{ +mann_whitney( + data, + select = NULL, + by = NULL, + weights = NULL, + distribution = "asymptotic", + ... +) +} +\arguments{ +\item{data}{A data frame.} + +\item{weights}{Bare (unquoted) variable name, or a character vector with +the variable name of the numeric vector of weights. If \code{weights = NULL}, +unweighted statistic is reported.} + +\item{distribution}{Indicates how the null distribution of the test statistic +should be computed. May be one of \code{"exact"}, \code{"approximate"} or \code{"asymptotic"} +(default). See \code{\link[coin:LocationTests]{coin::wilcox_test()}} for details.} + +\item{...}{For \code{weighted_ttest()} and \code{weighted_mannwhitney()}, currently not used. +For \code{weighted_chisqtest()}, further arguments passed down to +\code{\link[stats]{chisq.test}}.} + +\item{x}{Bare (unquoted) variable name, or a character vector with the variable name.} +} +\value{ +(Invisibly) returns a data frame with U, p and Z-values for each group-comparison +as well as effect-size r; additionally, group-labels and groups' n's are +also included. +} +\description{ +This function performs a Mann-Whitney-U-Test (or Wilcoxon rank +sum test for \emph{unpaired} samples, see \code{\link[=wilcox.test]{wilcox.test()}} and \code{\link[coin:LocationTests]{coin::wilcox_test()}}) +comparing \code{x} by each group indicated by \code{grp}. If \code{grp} has more than two +categories, a comparison between each combination of two groups is performed. + +The function reports U, p and Z-values as well as effect size r and group-rank-means. +} +\note{ +This function calls the \code{\link[coin]{wilcox_test}} with formula. If \code{grp} +has more than two groups, additionally a Kruskal-Wallis-Test (see \code{\link{kruskal.test}}) +is performed. \cr \cr +Interpretation of effect sizes, as a rule-of-thumb: +\itemize{ +\item small effect >= 0.1 +\item medium effect >= 0.3 +\item large effect >= 0.5 +} +} +\examples{ +data(efc) +# Mann-Whitney-U-Tests for elder's age by elder's sex. +mann_whitney(efc, e17age, e16sex) + +# using formula interface +mann_whitney(e17age ~ e16sex, efc) + +# Mann-Whitney-Tests for elder's age by each level elder's dependency. +mann_whitney(efc, e17age, e42dep) + +} From 21af06180bcb973c449e04773255925dda4ad05b Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 8 May 2024 08:48:34 +0200 Subject: [PATCH 03/34] update docs --- R/mann_whitney.R | 48 ++++++++++++++------------------------------- man/mann_whitney.Rd | 41 ++++++++++++-------------------------- 2 files changed, 27 insertions(+), 62 deletions(-) diff --git a/R/mann_whitney.R b/R/mann_whitney.R index 54ac7d4..6c9405a 100644 --- a/R/mann_whitney.R +++ b/R/mann_whitney.R @@ -1,56 +1,38 @@ #' @title Mann-Whitney-U-Test #' @name mann_whitney -#' @description This function performs a Mann-Whitney-U-Test (or Wilcoxon rank -#' sum test for _unpaired_ samples, see [`wilcox.test()`] and [`coin::wilcox_test()`]) -#' comparing `x` by each group indicated by `grp`. If `grp` has more than two -#' categories, a comparison between each combination of two groups is performed. +#' @description This function performs a Mann-Whitney-Test (or Wilcoxon rank +#' sum test for _unpaired_ samples, see [`wilcox.test()`] and [`coin::wilcox_test()`]). #' #' The function reports U, p and Z-values as well as effect size r and group-rank-means. #' -#' @param x Bare (unquoted) variable name, or a character vector with the variable name. +#' @param data A data frame. +#' @param select The dependent variable (numeric) to be used for the test. +#' @param by The grouping variable (factor) to be used for the test. If `by` is +#' not a factor, it will be coerced to a factor. +#' @param weights An optional weighting variable (numeric) to be used for the test. #' @param distribution Indicates how the null distribution of the test statistic #' should be computed. May be one of `"exact"`, `"approximate"` or `"asymptotic"` #' (default). See [`coin::wilcox_test()`] for details. #' -#' @inheritParams weighted_sd -#' @inheritParams means_by_group +#' @return A data frame. #' -#' @return (Invisibly) returns a data frame with U, p and Z-values for each group-comparison -#' as well as effect-size r; additionally, group-labels and groups' n's are -#' also included. +#' @note This function calls [`coin::wilcox_test()`] to extract effect sizes. +#' Interpretation of effect sizes, as a rule-of-thumb: #' -#' @note This function calls the \code{\link[coin]{wilcox_test}} with formula. If \code{grp} -#' has more than two groups, additionally a Kruskal-Wallis-Test (see \code{\link{kruskal.test}}) -#' is performed. \cr \cr -#' Interpretation of effect sizes, as a rule-of-thumb: -#' \itemize{ -#' \item small effect >= 0.1 -#' \item medium effect >= 0.3 -#' \item large effect >= 0.5 -#' } +#' - small effect >= 0.1 +#' - medium effect >= 0.3 +#' - large effect >= 0.5 #' #' @examples #' data(efc) #' # Mann-Whitney-U-Tests for elder's age by elder's sex. -#' mann_whitney(efc, e17age, e16sex) -#' -#' # using formula interface -#' mann_whitney(e17age ~ e16sex, efc) -#' -#' # Mann-Whitney-Tests for elder's age by each level elder's dependency. -#' mann_whitney(efc, e17age, e42dep) -#' -#' @importFrom stats na.omit wilcox.test kruskal.test -#' @importFrom sjmisc recode_to is_empty -#' @importFrom sjlabelled get_labels as_numeric -#' @importFrom rlang quo_name enquo +#' mann_whitney(efc, "e17age", "e16sex") #' @export mann_whitney <- function(data, select = NULL, by = NULL, weights = NULL, - distribution = "asymptotic", - ...) { + distribution = "asymptotic") { insight::check_if_installed("datawizard") # check if "select" is in data diff --git a/man/mann_whitney.Rd b/man/mann_whitney.Rd index 7826dc7..05c84c8 100644 --- a/man/mann_whitney.Rd +++ b/man/mann_whitney.Rd @@ -9,44 +9,34 @@ mann_whitney( select = NULL, by = NULL, weights = NULL, - distribution = "asymptotic", - ... + distribution = "asymptotic" ) } \arguments{ \item{data}{A data frame.} -\item{weights}{Bare (unquoted) variable name, or a character vector with -the variable name of the numeric vector of weights. If \code{weights = NULL}, -unweighted statistic is reported.} +\item{select}{The dependent variable (numeric) to be used for the test.} + +\item{by}{The grouping variable (factor) to be used for the test. If \code{by} is +not a factor, it will be coerced to a factor.} + +\item{weights}{An optional weighting variable (numeric) to be used for the test.} \item{distribution}{Indicates how the null distribution of the test statistic should be computed. May be one of \code{"exact"}, \code{"approximate"} or \code{"asymptotic"} (default). See \code{\link[coin:LocationTests]{coin::wilcox_test()}} for details.} - -\item{...}{For \code{weighted_ttest()} and \code{weighted_mannwhitney()}, currently not used. -For \code{weighted_chisqtest()}, further arguments passed down to -\code{\link[stats]{chisq.test}}.} - -\item{x}{Bare (unquoted) variable name, or a character vector with the variable name.} } \value{ -(Invisibly) returns a data frame with U, p and Z-values for each group-comparison -as well as effect-size r; additionally, group-labels and groups' n's are -also included. +A data frame. } \description{ -This function performs a Mann-Whitney-U-Test (or Wilcoxon rank -sum test for \emph{unpaired} samples, see \code{\link[=wilcox.test]{wilcox.test()}} and \code{\link[coin:LocationTests]{coin::wilcox_test()}}) -comparing \code{x} by each group indicated by \code{grp}. If \code{grp} has more than two -categories, a comparison between each combination of two groups is performed. +This function performs a Mann-Whitney-Test (or Wilcoxon rank +sum test for \emph{unpaired} samples, see \code{\link[=wilcox.test]{wilcox.test()}} and \code{\link[coin:LocationTests]{coin::wilcox_test()}}). The function reports U, p and Z-values as well as effect size r and group-rank-means. } \note{ -This function calls the \code{\link[coin]{wilcox_test}} with formula. If \code{grp} -has more than two groups, additionally a Kruskal-Wallis-Test (see \code{\link{kruskal.test}}) -is performed. \cr \cr +This function calls \code{\link[coin:LocationTests]{coin::wilcox_test()}} to extract effect sizes. Interpretation of effect sizes, as a rule-of-thumb: \itemize{ \item small effect >= 0.1 @@ -57,12 +47,5 @@ Interpretation of effect sizes, as a rule-of-thumb: \examples{ data(efc) # Mann-Whitney-U-Tests for elder's age by elder's sex. -mann_whitney(efc, e17age, e16sex) - -# using formula interface -mann_whitney(e17age ~ e16sex, efc) - -# Mann-Whitney-Tests for elder's age by each level elder's dependency. -mann_whitney(efc, e17age, e42dep) - +mann_whitney(efc, "e17age", "e16sex") } From 1d4fa1c44a0523c8ffe46b53b148978be04ca6fa Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 8 May 2024 08:51:27 +0200 Subject: [PATCH 04/34] update --- R/mann_whitney.R | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/R/mann_whitney.R b/R/mann_whitney.R index 6c9405a..4307670 100644 --- a/R/mann_whitney.R +++ b/R/mann_whitney.R @@ -61,13 +61,14 @@ mann_whitney <- function(data, dv <- data[[select]] grp <- data[[by]] - # coerce to factor and character to numeric + # coerce to factor grp <- datawizard::to_factor(grp) - # length of value range - labels <- sjlabelled::get_labels( - data[[by]], attr.only = FALSE, values = NULL, non.labelled = TRUE - ) + # value labels + labels <- names(attr(data[[by]], "labels", exact = TRUE)) + if (is.null(labels)) { + labels <- levels(grp) + } .calculate_mwu(dv, grp, distribution, labels) } From e988e47a38c5773e3e3c7ef1bdbc16b3718361b6 Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 8 May 2024 12:15:16 +0200 Subject: [PATCH 05/34] revise --- NAMESPACE | 2 +- R/{mann_whitney.R => test_mann_whitney.R} | 103 ++++++++++++------ man/{mann_whitney.Rd => mann_whitney_test.Rd} | 13 ++- 3 files changed, 80 insertions(+), 38 deletions(-) rename R/{mann_whitney.R => test_mann_whitney.R} (67%) rename man/{mann_whitney.Rd => mann_whitney_test.Rd} (89%) diff --git a/NAMESPACE b/NAMESPACE index 31a8e99..ee47378 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -88,7 +88,7 @@ export(icc) export(inequ_trend) export(is_prime) export(link_inverse) -export(mann_whitney) +export(mann_whitney_test) export(mannwhitney) export(mean_n) export(means_by_group) diff --git a/R/mann_whitney.R b/R/test_mann_whitney.R similarity index 67% rename from R/mann_whitney.R rename to R/test_mann_whitney.R index 4307670..cd50be1 100644 --- a/R/mann_whitney.R +++ b/R/test_mann_whitney.R @@ -1,9 +1,10 @@ -#' @title Mann-Whitney-U-Test -#' @name mann_whitney +#' @title Mann-Whitney-Test +#' @name mann_whitney_test #' @description This function performs a Mann-Whitney-Test (or Wilcoxon rank #' sum test for _unpaired_ samples, see [`wilcox.test()`] and [`coin::wilcox_test()`]). #' -#' The function reports U, p and Z-values as well as effect size r and group-rank-means. +#' The function reports U, p and Z-values as well as effect size r and +#' group-rank-means. #' #' @param data A data frame. #' @param select The dependent variable (numeric) to be used for the test. @@ -26,13 +27,13 @@ #' @examples #' data(efc) #' # Mann-Whitney-U-Tests for elder's age by elder's sex. -#' mann_whitney(efc, "e17age", "e16sex") +#' mann_whitney_test(efc, "e17age", "e16sex") #' @export -mann_whitney <- function(data, - select = NULL, - by = NULL, - weights = NULL, - distribution = "asymptotic") { +mann_whitney_test <- function(data, + select = NULL, + by = NULL, + weights = NULL, + distribution = "asymptotic") { insight::check_if_installed("datawizard") # check if "select" is in data @@ -65,16 +66,20 @@ mann_whitney <- function(data, grp <- datawizard::to_factor(grp) # value labels - labels <- names(attr(data[[by]], "labels", exact = TRUE)) - if (is.null(labels)) { - labels <- levels(grp) + group_labels <- names(attr(data[[by]], "labels", exact = TRUE)) + if (is.null(group_labels)) { + group_labels <- levels(droplevels(grp)) } - .calculate_mwu(dv, grp, distribution, labels) + if (is.null(weights)) { + .calculate_mwu(dv, grp, distribution, group_labels) + } else { + .calculate_weighted_mwu(dv, grp, weights, group_labels) + } } -.calculate_mwu <- function(dv, grp, distribution, labels) { +.calculate_mwu <- function(dv, grp, distribution, group_labels) { insight::check_if_installed("coin") # prepare data wcdat <- data.frame(dv, grp) @@ -92,16 +97,20 @@ mann_whitney <- function(data, w <- stats::wilcox.test(dv ~ grp, data = wcdat)$statistic # group means - rank_mean_1 <- mean(rank(dv)[which(grp == group_levels[1])], na.rm = TRUE) - rank_mean_2 <- mean(rank(dv)[which(grp == group_levels[2])], na.rm = TRUE) + dat_gr1 <- stats::na.omit(dv[grp == group_levels[1]]) + dat_gr2 <- stats::na.omit(dv[grp == group_levels[2]]) + + rank_mean_1 <- mean(rank(dat_gr1)) + rank_mean_2 <- mean(rank(dat_gr2)) # compute n for each group - n_grp1 <- length(stats::na.omit(dv[which(grp == group_levels[1])])) - n_grp2 <- length(stats::na.omit(dv[which(grp == group_levels[2])])) + n_grp1 <- length(dat_gr1) + n_grp2 <- length(dat_gr2) out <- data.frame( group1 = group_levels[1], group_2 = group_levels[2], + estimate = rank_mean_1 - rank_mean_2, u = u, w = w, p = p, @@ -116,51 +125,83 @@ mann_whitney <- function(data, c(n_grp1, n_grp2), c("N Group 1", "N Group 2") ) - attr(out, "group_labels") <- labels + attr(out, "group_labels") <- group_labels + attr(out, "method") <- "wilcoxon" + attr(out, "weighted") <- FALSE + class(out) <- c("sj_htest", "data.frame") + out } -.compute_weighted_mwu <- function(dv, grp, weights, labels) { +.calculate_weighted_mwu <- function(dv, grp, weights, group_labels) { # check if pkg survey is available insight::check_if_installed("survey") - dat <- data.frame(dv, grp, weights) + dat <- stats::na.omit(data.frame(dv, grp, weights)) colnames(dat) <- c("x", "g", "w") - if (insight::n_unqiue(dat$g) > 2) { - m <- "Weighted Kruskal-Wallis test" + if (insight::n_unique(dat$g) > 2) { method <- "KruskalWallis" } else { - m <- "Weighted Mann-Whitney-U test" method <- "wilcoxon" } design <- survey::svydesign(ids = ~0, data = dat, weights = ~w) - out <- survey::svyranktest(formula = x ~ g, design, test = method) + result <- survey::svyranktest(formula = x ~ g, design, test = method) # for rank mean - group_levels <- levels(grp) + group_levels <- levels(droplevels(grp)) + # subgroups + dat_gr1 <- dat[dat$g == group_levels[1]] + dat_gr2 <- dat[dat$g == group_levels[2]] + dat_gr1$rank_x <- rank(dat_gr1$x) + dat_gr2$rank_x <- rank(dat_gr2$x) + # rank means design_mean1 <- survey::svydesign( ids = ~0, - data = dat[dat$grp == group_levels[1]], + data = dat_gr1, weights = ~w ) - rank_mean_1 <- survey::svymean(~x, design_mean1) + rank_mean_1 <- survey::svymean(~rank_x, design_mean1) design_mean2 <- survey::svydesign( ids = ~0, - data = dat[dat$grp == group_levels[2]], + data = dat_gr2, weights = ~w ) - rank_mean_2 <- survey::svymean(~x, design_mean2) + rank_mean_2 <- survey::svymean(~rank_x, design_mean2) + + # group Ns + n_grp1 <- round(dat_gr1$x * dat_gr1$w) + n_grp2 <- round(dat_gr2$x * dat_gr2$w) + + # statistics and effect sizes + z <- result$statistic + r <- abs(z / sqrt(sum(n_grp1, n_grp2))) + + out <- data.frame( + group1 = group_levels[1], + group_2 = group_levels[2], + estimate = result$estimate, + z = z, + r = r, + p = result$p.value + ) - out$method <- m attr(out, "rank_means") <- stats::setNames( c(rank_mean_1, rank_mean_2), c("Mean Group 1", "Mean Group 2") ) + attr(out, "n_groups") <- stats::setNames( + c(n_grp1, n_grp2), + c("N Group 1", "N Group 2") + ) + attr(out, "group_labels") <- group_labels + attr(out, "method") <- method + attr(out, "weighted") <- TRUE + class(out) <- c("sj_htest", "data.frame") out } diff --git a/man/mann_whitney.Rd b/man/mann_whitney_test.Rd similarity index 89% rename from man/mann_whitney.Rd rename to man/mann_whitney_test.Rd index 05c84c8..261b1b6 100644 --- a/man/mann_whitney.Rd +++ b/man/mann_whitney_test.Rd @@ -1,10 +1,10 @@ % Generated by roxygen2: do not edit by hand % Please edit documentation in R/mann_whitney.R -\name{mann_whitney} -\alias{mann_whitney} -\title{Mann-Whitney-U-Test} +\name{mann_whitney_test} +\alias{mann_whitney_test} +\title{Mann-Whitney-Test} \usage{ -mann_whitney( +mann_whitney_test( data, select = NULL, by = NULL, @@ -33,7 +33,8 @@ A data frame. This function performs a Mann-Whitney-Test (or Wilcoxon rank sum test for \emph{unpaired} samples, see \code{\link[=wilcox.test]{wilcox.test()}} and \code{\link[coin:LocationTests]{coin::wilcox_test()}}). -The function reports U, p and Z-values as well as effect size r and group-rank-means. +The function reports U, p and Z-values as well as effect size r and +group-rank-means. } \note{ This function calls \code{\link[coin:LocationTests]{coin::wilcox_test()}} to extract effect sizes. @@ -47,5 +48,5 @@ Interpretation of effect sizes, as a rule-of-thumb: \examples{ data(efc) # Mann-Whitney-U-Tests for elder's age by elder's sex. -mann_whitney(efc, "e17age", "e16sex") +mann_whitney_test(efc, "e17age", "e16sex") } From a1e6624231971204486274d415e7fdafc7c7b874 Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 8 May 2024 12:42:18 +0200 Subject: [PATCH 06/34] final --- NAMESPACE | 1 + R/test_mann_whitney.R | 86 +++++++++++++++++++++++++++++++++------- man/mann_whitney_test.Rd | 14 ++++--- 3 files changed, 82 insertions(+), 19 deletions(-) diff --git a/NAMESPACE b/NAMESPACE index ee47378..d609f31 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -25,6 +25,7 @@ S3method(print,sj_check_assump) S3method(print,sj_chi2gof) S3method(print,sj_grpmean) S3method(print,sj_grpmeans) +S3method(print,sj_htest_mwu) S3method(print,sj_mwu) S3method(print,sj_outliers) S3method(print,sj_pval) diff --git a/R/test_mann_whitney.R b/R/test_mann_whitney.R index cd50be1..f7ce394 100644 --- a/R/test_mann_whitney.R +++ b/R/test_mann_whitney.R @@ -3,8 +3,7 @@ #' @description This function performs a Mann-Whitney-Test (or Wilcoxon rank #' sum test for _unpaired_ samples, see [`wilcox.test()`] and [`coin::wilcox_test()`]). #' -#' The function reports U, p and Z-values as well as effect size r and -#' group-rank-means. +#' The function reports p and Z-values as well as effect size r and group-rank-means. #' #' @param data A data frame. #' @param select The dependent variable (numeric) to be used for the test. @@ -17,13 +16,19 @@ #' #' @return A data frame. #' -#' @note This function calls [`coin::wilcox_test()`] to extract effect sizes. -#' Interpretation of effect sizes, as a rule-of-thumb: +#' @details This function calls [`coin::wilcox_test()`] to extract effect sizes. +#' Interpretation of the effect size **r**, as a rule-of-thumb: #' #' - small effect >= 0.1 #' - medium effect >= 0.3 #' - large effect >= 0.5 #' +#' **r** is calcuated as: +#' +#' ``` +#' r = |Z| / sqrt(n1 + n2) +#' ``` +#' #' @examples #' data(efc) #' # Mann-Whitney-U-Tests for elder's age by elder's sex. @@ -65,6 +70,11 @@ mann_whitney_test <- function(data, # coerce to factor grp <- datawizard::to_factor(grp) + # only two groups allowed + if (insight::n_unique(grp) > 2) { + insight::format_error("Only two groups are allowed for Mann-Whitney-Test. Please use `kruskal_wallis_test()` for more than two groups.") # nolint + } + # value labels group_labels <- names(attr(data[[by]], "labels", exact = TRUE)) if (is.null(group_labels)) { @@ -74,11 +84,13 @@ mann_whitney_test <- function(data, if (is.null(weights)) { .calculate_mwu(dv, grp, distribution, group_labels) } else { - .calculate_weighted_mwu(dv, grp, weights, group_labels) + .calculate_weighted_mwu(dv, grp, data[[weights]], group_labels) } } +# Mann-Whitney-Test for two groups -------------------------------------------- + .calculate_mwu <- function(dv, grp, distribution, group_labels) { insight::check_if_installed("coin") # prepare data @@ -109,13 +121,13 @@ mann_whitney_test <- function(data, out <- data.frame( group1 = group_levels[1], - group_2 = group_levels[2], + group2 = group_levels[2], estimate = rank_mean_1 - rank_mean_2, u = u, w = w, - p = p, z = z, - r = r + r = r, + p = as.numeric(p) ) attr(out, "rank_means") <- stats::setNames( c(rank_mean_1, rank_mean_2), @@ -128,12 +140,14 @@ mann_whitney_test <- function(data, attr(out, "group_labels") <- group_labels attr(out, "method") <- "wilcoxon" attr(out, "weighted") <- FALSE - class(out) <- c("sj_htest", "data.frame") + class(out) <- c("sj_htest_mwu", "data.frame") out } +# Weighted Mann-Whitney-Test for two groups ---------------------------------- + .calculate_weighted_mwu <- function(dv, grp, weights, group_labels) { # check if pkg survey is available insight::check_if_installed("survey") @@ -153,8 +167,8 @@ mann_whitney_test <- function(data, # for rank mean group_levels <- levels(droplevels(grp)) # subgroups - dat_gr1 <- dat[dat$g == group_levels[1]] - dat_gr2 <- dat[dat$g == group_levels[2]] + dat_gr1 <- dat[dat$g == group_levels[1], ] + dat_gr2 <- dat[dat$g == group_levels[2], ] dat_gr1$rank_x <- rank(dat_gr1$x) dat_gr2$rank_x <- rank(dat_gr2$x) @@ -183,11 +197,11 @@ mann_whitney_test <- function(data, out <- data.frame( group1 = group_levels[1], - group_2 = group_levels[2], + group2 = group_levels[2], estimate = result$estimate, z = z, r = r, - p = result$p.value + p = as.numeric(result$p.value) ) attr(out, "rank_means") <- stats::setNames( @@ -201,12 +215,14 @@ mann_whitney_test <- function(data, attr(out, "group_labels") <- group_labels attr(out, "method") <- method attr(out, "weighted") <- TRUE - class(out) <- c("sj_htest", "data.frame") + class(out) <- c("sj_htest_mwu", "data.frame") out } +# helper ---------------------------------------------------------------------- + .misspelled_string <- function(source, searchterm, default_message = NULL) { if (is.null(searchterm) || length(searchterm) < 1) { return(default_message) @@ -257,3 +273,45 @@ mann_whitney_test <- function(data, p <- sprintf("(%s){~%i}", pattern, precision) grep(pattern = p, x = x, ignore.case = FALSE) } + + +# methods --------------------------------------------------------------------- + +#' @export +print.sj_htest_mwu <- function(x, ...) { + # fetch attributes + group_labels <- attributes(x)$group_labels + rank_means <- attributes(x)$rank_means + n_groups <- attributes(x)$n_groups + weighted <- attributes(x)$weighted + + if (weighted) { + weight_string <- " (weighted)" + } else { + weight_string <- "" + } + + # same width + group_labels <- format(group_labels) + + # header + insight::print_color(sprintf("# Mann-Whitney-Test%s\n\n", weight_string), "blue") + + # group-1-info + insight::print_color( + sprintf( + " Group 1: %s (n = %i, rank mean = %s)\n", + group_labels[1], n_groups[1], insight::format_value(rank_means[1], protect_integers = TRUE) + ), "cyan" + ) + + # group-2-info + insight::print_color( + sprintf( + " Group 2: %s (n = %i, rank mean = %s)\n", + group_labels[2], n_groups[2], insight::format_value(rank_means[2], protect_integers = TRUE) + ), "cyan" + ) + + cat(sprintf("\n r = %.3f, Z = %.3f, %s\n\n", x$r, x$z, insight::format_p(x$p))) +} diff --git a/man/mann_whitney_test.Rd b/man/mann_whitney_test.Rd index 261b1b6..3610109 100644 --- a/man/mann_whitney_test.Rd +++ b/man/mann_whitney_test.Rd @@ -1,5 +1,5 @@ % Generated by roxygen2: do not edit by hand -% Please edit documentation in R/mann_whitney.R +% Please edit documentation in R/test_mann_whitney.R \name{mann_whitney_test} \alias{mann_whitney_test} \title{Mann-Whitney-Test} @@ -33,17 +33,21 @@ A data frame. This function performs a Mann-Whitney-Test (or Wilcoxon rank sum test for \emph{unpaired} samples, see \code{\link[=wilcox.test]{wilcox.test()}} and \code{\link[coin:LocationTests]{coin::wilcox_test()}}). -The function reports U, p and Z-values as well as effect size r and -group-rank-means. +The function reports p and Z-values as well as effect size r and group-rank-means. } -\note{ +\details{ This function calls \code{\link[coin:LocationTests]{coin::wilcox_test()}} to extract effect sizes. -Interpretation of effect sizes, as a rule-of-thumb: +Interpretation of the effect size \strong{r}, as a rule-of-thumb: \itemize{ \item small effect >= 0.1 \item medium effect >= 0.3 \item large effect >= 0.5 } + +\strong{r} is calcuated as: + +\if{html}{\out{
}}\preformatted{r = |Z| / sqrt(n1 + n2) +}\if{html}{\out{
}} } \examples{ data(efc) From be528c6116deaee5e142dcc67a34fa52057c8c42 Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 8 May 2024 14:58:36 +0200 Subject: [PATCH 07/34] add chi2 --- NAMESPACE | 2 + R/chi_squared_test.R | 188 ++++++++++++++++++ ...est_mann_whitney.R => mann_whitney_test.R} | 67 +++++-- R/xtab_statistics.R | 5 +- _pkgdown.yml | 2 + man/chi_squared_test.Rd | 55 +++++ man/mann_whitney_test.Rd | 2 +- 7 files changed, 297 insertions(+), 24 deletions(-) create mode 100644 R/chi_squared_test.R rename R/{test_mann_whitney.R => mann_whitney_test.R} (91%) create mode 100644 man/chi_squared_test.Rd diff --git a/NAMESPACE b/NAMESPACE index d609f31..02c7ebf 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -25,6 +25,7 @@ S3method(print,sj_check_assump) S3method(print,sj_chi2gof) S3method(print,sj_grpmean) S3method(print,sj_grpmeans) +S3method(print,sj_htest_chi) S3method(print,sj_htest_mwu) S3method(print,sj_mwu) S3method(print,sj_outliers) @@ -67,6 +68,7 @@ export(boot_est) export(boot_p) export(boot_se) export(bootstrap) +export(chi_squared_test) export(chisq_gof) export(ci) export(cohens_f) diff --git a/R/chi_squared_test.R b/R/chi_squared_test.R new file mode 100644 index 0000000..4520100 --- /dev/null +++ b/R/chi_squared_test.R @@ -0,0 +1,188 @@ +#' @title Chi-Squared Test +#' @name chi_squared_test +#' @description This function performs a Mann-Whitney-Test (or Wilcoxon rank +#' sum test for _unpaired_ samples, see [`wilcox.test()`] and [`coin::wilcox_test()`]). +#' +#' The function reports p and Z-values as well as effect size r and group-rank-means. +#' +#' @param data A data frame. +#' @param select The dependent variable (numeric) to be used for the test. +#' @param by The grouping variable (factor) to be used for the test. If `by` is +#' not a factor, it will be coerced to a factor. +#' @param weights An optional weighting variable (numeric) to be used for the test. +#' @param ... Additional arguments passed down to [`chisq.test()`]. +#' +#' @return A data frame. +#' +#' @details This function calls [`coin::wilcox_test()`] to extract effect sizes. +#' Interpretation of the effect size **r**, as a rule-of-thumb: +#' +#' - small effect >= 0.1 +#' - medium effect >= 0.3 +#' - large effect >= 0.5 +#' +#' **r** is calcuated as: +#' +#' ``` +#' r = |Z| / sqrt(n1 + n2) +#' ``` +#' +#' @examples +#' data(efc) +#' # Mann-Whitney-U-Tests for elder's age by elder's sex. +#' chi_squared_test(efc, "e17age", "e16sex") +#' @export +chi_squared_test <- function(data, + select = NULL, + by = NULL, + probabilities = NULL, + weights = NULL, + ...) { + if (is.null(probabilities)) { + .calculate_chisq(data, select, by, weights, ...) + } else { + .calculate_chisq_gof(data, select, probabilities, weights, ...) + } +} + + +# Mann-Whitney-Test for two groups -------------------------------------------- + +.calculate_chisq <- function(data, select, by, weights, ...) { + insight::check_if_installed("datawizard") + # sanity checks + .sanitize_htest_input(data, select, by, weights) + + # get data + grp1 <- data[[select]] + grp2 <- data[[by]] + + # create data frame for table + x <- data.frame( + grp1 = datawizard::to_factor(grp1), + grp2 = datawizard::to_factor(grp2) + ) + # add weights + if (!is.null(weights)) { + x$weights <- data[[weights]] + } + # remove missings + x <- stats::na.omit(x) + + # contingency table + if (is.null(weights)) { + tab <- table(x) + } else { + tab <- as.table(round(stats::xtabs(data[[3]] ~ data[[1]] + data[[2]]))) + class(tab) <- "table" + } + + # expected values, to identify whether Fisher's test is needed + expected_values <- as.table(round(as.array(margin.table(tab, 1)) %*% t(as.array(margin.table(tab, 2))) / margin.table(tab))) # nolint + + # chi-squared test + htest <- suppressWarnings(stats::chisq.test(tab, ...)) + test_statistic <- htest$statistic + + # need fisher? + if (min(expected_values$expected) < 5 || (min(expected_values$expected) < 10 && htest$parameter == 1)) { + htest <- stats::fisher.test(tab, simulate.p.value = TRUE, ...) + } + p_value <- htest$p.value + + # effect size + if (nrow(tab) > 2 || ncol(tab) > 2) { + effect_size <- stats::setNamed(cramer(tab), "Cramer's V") + } else { + effect_size <- stats::setNamed(phi(tab), "Phi") + } + + # return result + out <- data.frame( + statistic_name = "Chi-squared", + statistic = test_statistic, + effect_size_name = names(effect_size), + effect_size = as.numeric(effect_size), + p = p_value, + df = (nrow(tab) - 1) * (ncol(tab) - 1), + n_obs = sum(tab, na.rm = TRUE), + stringsAsFactors = FALSE + ) + class(out) <- c("sj_htest_chi", "data.frame") + attr(out, "caption") <- "Contingency Tables" + out +} + + +.calculate_chisq_gof <- function(data, select, probabilities, weights, ...) { + insight::check_if_installed("effectsize") + + # get data + x <- data.frame(grp = data[[select]]) + # add weights + if (!is.null(weights)) { + x$weights <- data[[weights]] + } + # remove missings + x <- stats::na.omit(x) + + # contingency table + if (is.null(weights)) { + tab <- table(x) + } else { + tab <- as.table(round(stats::xtabs(data[[2]] ~ data[[1]]))) + class(tab) <- "table" + } + + # sanity check + if (length(probabilities) != as.numeric(dim(tab))) { + insight::format_error("Length of probabilities must match number of cells in table (i.e. number of levels of input factor).") # nolint + } + + # chi-squared test + htest <- suppressWarnings(stats::chisq.test(tab, p = probabilities, rescale.p = TRUE, ...)) + test_statistic <- htest$statistic + p_value <- htest$p.value + + effect_size <- effectsize::chisq_to_fei( + test_statistic, + n = sum(tab), + nrow = as.numeric(dim(tab)), + ncol = 1, + p = probabilities, + alternative = "two.sided" + )$Fei + + # return result + out <- data.frame( + statistic_name = "Chi-squared", + statistic = test_statistic, + effect_size_name = "Fei", + effect_size = as.numeric(effect_size), + p = p_value, + df = (nrow(tab) - 1) * (ncol(tab) - 1), + n_obs = sum(tab, na.rm = TRUE), + stringsAsFactors = FALSE + ) + class(out) <- c("sj_htest_chi", "data.frame") + attr(out, "caption") <- "given Probabilities" + out +} + + +# methods --------------------------------------------------------------------- + +#' @export +print.sj_htest_chi <- function(x, ...) { + # get length of method name, to align output + l <- max(nchar(c(x$statistic_name, x$effect_size_name, "p-value", "Observations"))) + # headline + insight::print_color(sprintf("\n# Chi-squared Test for%s\n\n", attributes(x)$caption), "blue") + + # print test statistic + cat(sprintf(" %*s: %.4f\n", l, x$statistic_name, x$statistic)) + cat(sprintf(" %*s: %.4f\n", l, x$effect_size_name, x$effect_size)) + cat(sprintf(" %*s: %g\n", l, "df", x$df)) + cat(sprintf(" %*s: %s\n", l, "p-value", insight::format_p(x$p.value, stars = TRUE, name = NULL))) + cat(sprintf(" %*s: %g\n", l, "Observations", x$n_obs)) +} diff --git a/R/test_mann_whitney.R b/R/mann_whitney_test.R similarity index 91% rename from R/test_mann_whitney.R rename to R/mann_whitney_test.R index f7ce394..2708b15 100644 --- a/R/test_mann_whitney.R +++ b/R/mann_whitney_test.R @@ -41,27 +41,8 @@ mann_whitney_test <- function(data, distribution = "asymptotic") { insight::check_if_installed("datawizard") - # check if "select" is in data - if (!select %in% colnames(data)) { - insight::format_error( - sprintf("Variable '%s' not found in data frame.", select), - .misspelled_string(select, colnames(data), "Maybe misspelled?") - ) - } - # check if "by" is in data - if (!by %in% colnames(data)) { - insight::format_error( - sprintf("Variable '%s' not found in data frame.", by), - .misspelled_string(by, colnames(data), "Maybe misspelled?") - ) - } - # check if "weights" is in data - if (!is.null(weights) && !weights %in% colnames(data)) { - insight::format_error( - sprintf("Weighting variable '%s' not found in data frame.", weights), - .misspelled_string(weights, colnames(data), "Maybe misspelled?") - ) - } + # sanity checks + .sanitize_htest_input(data, select, by, weights) # get data dv <- data[[select]] @@ -275,6 +256,50 @@ mann_whitney_test <- function(data, } +.sanitize_htest_input <- function(data, select, by, weights) { + # check if arguments are NULL + if (is.null(select)) { + insight::format_error("Argument `select` is missing.") + } + if (is.null(by)) { + insight::format_error("Arguments `by` is missing.") + } + + # check if arguments have correct length (length of 1) + if (length(select) != 1 || !is.character(select)) { + insight::format_error("Argument `select` must be the name of a single variable.") + } + if (length(by) != 1 || !is.character(by)) { + insight::format_error("Argument `by` must be the name of a single variable.") + } + if (!is.null(weights) && length(weights) != 1) { + insight::format_error("Argument `weights` must be the name of a single variable.") + } + + # check if "select" is in data + if (!select %in% colnames(data)) { + insight::format_error( + sprintf("Variable '%s' not found in data frame.", select), + .misspelled_string(select, colnames(data), "Maybe misspelled?") + ) + } + # check if "by" is in data + if (!by %in% colnames(data)) { + insight::format_error( + sprintf("Variable '%s' not found in data frame.", by), + .misspelled_string(by, colnames(data), "Maybe misspelled?") + ) + } + # check if "weights" is in data + if (!is.null(weights) && !weights %in% colnames(data)) { + insight::format_error( + sprintf("Weighting variable '%s' not found in data frame.", weights), + .misspelled_string(weights, colnames(data), "Maybe misspelled?") + ) + } +} + + # methods --------------------------------------------------------------------- #' @export diff --git a/R/xtab_statistics.R b/R/xtab_statistics.R index 4fb592e..03a33bb 100644 --- a/R/xtab_statistics.R +++ b/R/xtab_statistics.R @@ -140,8 +140,9 @@ crosstable_statistics <- function(data, x1 = NULL, x2 = NULL, statistics = c("au if (!is.null(weights)) { tab <- as.table(round(stats::xtabs(data[[3]] ~ data[[1]] + data[[2]]))) class(tab) <- "table" - } else + } else { tab <- table(data) + } } else { # 'data' is a table - copy to table object tab <- data @@ -152,7 +153,7 @@ crosstable_statistics <- function(data, x1 = NULL, x2 = NULL, statistics = c("au "Need arguments `data`, `x1` and `x2` to compute %s-statistics.", statistics ), - call. = F + call. = FALSE ) } } diff --git a/_pkgdown.yml b/_pkgdown.yml index 4596c63..55cb5b2 100644 --- a/_pkgdown.yml +++ b/_pkgdown.yml @@ -31,6 +31,8 @@ reference: contents: - gmd - mannwhitney + - mann_whitney_test + - chi_squared_test - mean_n - means_by_group - var_pop diff --git a/man/chi_squared_test.Rd b/man/chi_squared_test.Rd new file mode 100644 index 0000000..610f522 --- /dev/null +++ b/man/chi_squared_test.Rd @@ -0,0 +1,55 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/chi_squared_test.R +\name{chi_squared_test} +\alias{chi_squared_test} +\title{Chi-Squared Test} +\usage{ +chi_squared_test( + data, + select = NULL, + by = NULL, + probabilities = NULL, + weights = NULL, + ... +) +} +\arguments{ +\item{data}{A data frame.} + +\item{select}{The dependent variable (numeric) to be used for the test.} + +\item{by}{The grouping variable (factor) to be used for the test. If \code{by} is +not a factor, it will be coerced to a factor.} + +\item{weights}{An optional weighting variable (numeric) to be used for the test.} + +\item{...}{Additional arguments passed down to \code{\link[=chisq.test]{chisq.test()}}.} +} +\value{ +A data frame. +} +\description{ +This function performs a Mann-Whitney-Test (or Wilcoxon rank +sum test for \emph{unpaired} samples, see \code{\link[=wilcox.test]{wilcox.test()}} and \code{\link[coin:LocationTests]{coin::wilcox_test()}}). + +The function reports p and Z-values as well as effect size r and group-rank-means. +} +\details{ +This function calls \code{\link[coin:LocationTests]{coin::wilcox_test()}} to extract effect sizes. +Interpretation of the effect size \strong{r}, as a rule-of-thumb: +\itemize{ +\item small effect >= 0.1 +\item medium effect >= 0.3 +\item large effect >= 0.5 +} + +\strong{r} is calcuated as: + +\if{html}{\out{
}}\preformatted{r = |Z| / sqrt(n1 + n2) +}\if{html}{\out{
}} +} +\examples{ +data(efc) +# Mann-Whitney-U-Tests for elder's age by elder's sex. +chi_squared_test(efc, "e17age", "e16sex") +} diff --git a/man/mann_whitney_test.Rd b/man/mann_whitney_test.Rd index 3610109..c72b69f 100644 --- a/man/mann_whitney_test.Rd +++ b/man/mann_whitney_test.Rd @@ -1,5 +1,5 @@ % Generated by roxygen2: do not edit by hand -% Please edit documentation in R/test_mann_whitney.R +% Please edit documentation in R/mann_whitney_test.R \name{mann_whitney_test} \alias{mann_whitney_test} \title{Mann-Whitney-Test} From 6c1109eacc6a65ea71f5589575faeeaa8a96076d Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 8 May 2024 16:04:37 +0200 Subject: [PATCH 08/34] chi2 test --- R/chi_squared_test.R | 34 +++++++++++++--------------------- R/mann_whitney_test.R | 6 +++--- man/chi_squared_test.Rd | 23 +++++++---------------- 3 files changed, 23 insertions(+), 40 deletions(-) diff --git a/R/chi_squared_test.R b/R/chi_squared_test.R index 4520100..d88f01f 100644 --- a/R/chi_squared_test.R +++ b/R/chi_squared_test.R @@ -14,23 +14,15 @@ #' #' @return A data frame. #' -#' @details This function calls [`coin::wilcox_test()`] to extract effect sizes. -#' Interpretation of the effect size **r**, as a rule-of-thumb: -#' -#' - small effect >= 0.1 -#' - medium effect >= 0.3 -#' - large effect >= 0.5 -#' -#' **r** is calcuated as: -#' -#' ``` -#' r = |Z| / sqrt(n1 + n2) -#' ``` -#' #' @examples #' data(efc) -#' # Mann-Whitney-U-Tests for elder's age by elder's sex. -#' chi_squared_test(efc, "e17age", "e16sex") +#' efc$weight <- abs(rnorm(nrow(efc), 1, 0.3)) +#' # Chi-squared-test +#' chi_squared_test(efc, "c161sex", "e16sex") +#' # weighted Chi-squared-test +#' chi_squared_test(efc, "c161sex", "e16sex", weights = "weight") +#' # Chi-squared-test for given probabilities +#' chi_squared_test(efc, "c161sex", probabilities = c(0.3, 0.7)) #' @export chi_squared_test <- function(data, select = NULL, @@ -73,7 +65,7 @@ chi_squared_test <- function(data, if (is.null(weights)) { tab <- table(x) } else { - tab <- as.table(round(stats::xtabs(data[[3]] ~ data[[1]] + data[[2]]))) + tab <- as.table(round(stats::xtabs(x[[3]] ~ x[[1]] + x[[2]]))) class(tab) <- "table" } @@ -85,16 +77,16 @@ chi_squared_test <- function(data, test_statistic <- htest$statistic # need fisher? - if (min(expected_values$expected) < 5 || (min(expected_values$expected) < 10 && htest$parameter == 1)) { + if (min(expected_values) < 5 || (min(expected_values) < 10 && htest$parameter == 1)) { htest <- stats::fisher.test(tab, simulate.p.value = TRUE, ...) } p_value <- htest$p.value # effect size if (nrow(tab) > 2 || ncol(tab) > 2) { - effect_size <- stats::setNamed(cramer(tab), "Cramer's V") + effect_size <- stats::setNames(cramer(tab), "Cramer's V") } else { - effect_size <- stats::setNamed(phi(tab), "Phi") + effect_size <- stats::setNames(phi(tab), "Phi") } # return result @@ -177,12 +169,12 @@ print.sj_htest_chi <- function(x, ...) { # get length of method name, to align output l <- max(nchar(c(x$statistic_name, x$effect_size_name, "p-value", "Observations"))) # headline - insight::print_color(sprintf("\n# Chi-squared Test for%s\n\n", attributes(x)$caption), "blue") + insight::print_color(sprintf("\n# Chi-Squared Test for %s\n\n", attributes(x)$caption), "blue") # print test statistic cat(sprintf(" %*s: %.4f\n", l, x$statistic_name, x$statistic)) cat(sprintf(" %*s: %.4f\n", l, x$effect_size_name, x$effect_size)) cat(sprintf(" %*s: %g\n", l, "df", x$df)) - cat(sprintf(" %*s: %s\n", l, "p-value", insight::format_p(x$p.value, stars = TRUE, name = NULL))) + cat(sprintf(" %*s: %s\n", l, "p-value", insight::format_p(x$p, stars = TRUE, name = NULL))) cat(sprintf(" %*s: %g\n", l, "Observations", x$n_obs)) } diff --git a/R/mann_whitney_test.R b/R/mann_whitney_test.R index 2708b15..f6302a8 100644 --- a/R/mann_whitney_test.R +++ b/R/mann_whitney_test.R @@ -280,21 +280,21 @@ mann_whitney_test <- function(data, if (!select %in% colnames(data)) { insight::format_error( sprintf("Variable '%s' not found in data frame.", select), - .misspelled_string(select, colnames(data), "Maybe misspelled?") + .misspelled_string(colnames(data), select, "Maybe misspelled?") ) } # check if "by" is in data if (!by %in% colnames(data)) { insight::format_error( sprintf("Variable '%s' not found in data frame.", by), - .misspelled_string(by, colnames(data), "Maybe misspelled?") + .misspelled_string(colnames(data), by, "Maybe misspelled?") ) } # check if "weights" is in data if (!is.null(weights) && !weights %in% colnames(data)) { insight::format_error( sprintf("Weighting variable '%s' not found in data frame.", weights), - .misspelled_string(weights, colnames(data), "Maybe misspelled?") + .misspelled_string(colnames(data), weights, "Maybe misspelled?") ) } } diff --git a/man/chi_squared_test.Rd b/man/chi_squared_test.Rd index 610f522..daf4d3b 100644 --- a/man/chi_squared_test.Rd +++ b/man/chi_squared_test.Rd @@ -34,22 +34,13 @@ sum test for \emph{unpaired} samples, see \code{\link[=wilcox.test]{wilcox.test( The function reports p and Z-values as well as effect size r and group-rank-means. } -\details{ -This function calls \code{\link[coin:LocationTests]{coin::wilcox_test()}} to extract effect sizes. -Interpretation of the effect size \strong{r}, as a rule-of-thumb: -\itemize{ -\item small effect >= 0.1 -\item medium effect >= 0.3 -\item large effect >= 0.5 -} - -\strong{r} is calcuated as: - -\if{html}{\out{
}}\preformatted{r = |Z| / sqrt(n1 + n2) -}\if{html}{\out{
}} -} \examples{ data(efc) -# Mann-Whitney-U-Tests for elder's age by elder's sex. -chi_squared_test(efc, "e17age", "e16sex") +efc$weight <- abs(rnorm(nrow(efc), 1, 0.3)) +# Chi-squared-test +chi_squared_test(efc, "c161sex", "e16sex") +# weighted Chi-squared-test +chi_squared_test(efc, "c161sex", "e16sex", weights = "weight") +# Chi-squared-test for given probabilities +chi_squared_test(efc, "c161sex", probabilities = c(0.3, 0.7)) } From f8e70cd5d9ac1087c8c094623d87d056d5352914 Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 8 May 2024 16:29:41 +0200 Subject: [PATCH 09/34] finalize chi2 --- NAMESPACE | 9 -- R/chi_squared_test.R | 16 ++- R/mwu.R | 234 ------------------------------------------- R/wtd_chisqtest.R | 91 ----------------- _pkgdown.yml | 1 - man/mwu.Rd | 104 ------------------- man/weighted_sd.Rd | 18 +--- 7 files changed, 15 insertions(+), 458 deletions(-) delete mode 100644 R/mwu.R delete mode 100644 R/wtd_chisqtest.R delete mode 100644 man/mwu.Rd diff --git a/NAMESPACE b/NAMESPACE index 02c7ebf..f586c75 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -13,8 +13,6 @@ S3method(formula,svyglm.zip) S3method(model.frame,svyglm.nb) S3method(model.frame,svyglm.zip) S3method(model.matrix,gls) -S3method(mwu,default) -S3method(mwu,formula) S3method(phi,formula) S3method(phi,ftable) S3method(phi,table) @@ -42,8 +40,6 @@ S3method(print,tidy_stan) S3method(residuals,svyglm.nb) S3method(summary,sj_pval) S3method(terms,svyglm.nb) -S3method(weighted_chisqtest,default) -S3method(weighted_chisqtest,formula) S3method(weighted_correlation,default) S3method(weighted_correlation,formula) S3method(weighted_mannwhitney,default) @@ -92,11 +88,9 @@ export(inequ_trend) export(is_prime) export(link_inverse) export(mann_whitney_test) -export(mannwhitney) export(mean_n) export(means_by_group) export(mse) -export(mwu) export(omega_sq) export(p_value) export(phi) @@ -119,7 +113,6 @@ export(typical_value) export(var_pop) export(weight) export(weight2) -export(weighted_chisqtest) export(weighted_correlation) export(weighted_mannwhitney) export(weighted_mean) @@ -179,7 +172,6 @@ importFrom(sjmisc,add_variables) importFrom(sjmisc,is_empty) importFrom(sjmisc,is_float) importFrom(sjmisc,is_num_fac) -importFrom(sjmisc,recode_to) importFrom(sjmisc,str_contains) importFrom(sjmisc,to_value) importFrom(sjmisc,trim) @@ -217,7 +209,6 @@ importFrom(stats,var) importFrom(stats,vcov) importFrom(stats,weighted.mean) importFrom(stats,weights) -importFrom(stats,wilcox.test) importFrom(stats,xtabs) importFrom(tidyr,gather) importFrom(tidyr,nest) diff --git a/R/chi_squared_test.R b/R/chi_squared_test.R index d88f01f..7d76ad8 100644 --- a/R/chi_squared_test.R +++ b/R/chi_squared_test.R @@ -40,7 +40,7 @@ chi_squared_test <- function(data, # Mann-Whitney-Test for two groups -------------------------------------------- -.calculate_chisq <- function(data, select, by, weights, ...) { +.calculate_chisq <- function(data, select, by, weights, verbose = TRUE, ...) { insight::check_if_installed("datawizard") # sanity checks .sanitize_htest_input(data, select, by, weights) @@ -122,14 +122,20 @@ chi_squared_test <- function(data, if (is.null(weights)) { tab <- table(x) } else { - tab <- as.table(round(stats::xtabs(data[[2]] ~ data[[1]]))) + tab <- as.table(round(stats::xtabs(x[[2]] ~ x[[1]]))) class(tab) <- "table" } + # table dimensions + n_rows <- nlevels(droplevels(as.factor(x$grp))) + # sanity check - if (length(probabilities) != as.numeric(dim(tab))) { + if (length(probabilities) != n_rows) { insight::format_error("Length of probabilities must match number of cells in table (i.e. number of levels of input factor).") # nolint } + if (!isTRUE(all.equal(sum(probabilities), 1))) { + insight::format_error("Probabilities must sum to 1.") + } # chi-squared test htest <- suppressWarnings(stats::chisq.test(tab, p = probabilities, rescale.p = TRUE, ...)) @@ -139,7 +145,7 @@ chi_squared_test <- function(data, effect_size <- effectsize::chisq_to_fei( test_statistic, n = sum(tab), - nrow = as.numeric(dim(tab)), + nrow = n_rows, ncol = 1, p = probabilities, alternative = "two.sided" @@ -152,7 +158,7 @@ chi_squared_test <- function(data, effect_size_name = "Fei", effect_size = as.numeric(effect_size), p = p_value, - df = (nrow(tab) - 1) * (ncol(tab) - 1), + df = n_rows - 1, n_obs = sum(tab, na.rm = TRUE), stringsAsFactors = FALSE ) diff --git a/R/mwu.R b/R/mwu.R deleted file mode 100644 index 6967bf7..0000000 --- a/R/mwu.R +++ /dev/null @@ -1,234 +0,0 @@ -#' @title Mann-Whitney-U-Test -#' @name mwu -#' @description This function performs a Mann-Whitney-U-Test (or Wilcoxon rank -#' sum test for _unpaired_ samples, see [`wilcox.test()`] and [`coin::wilcox_test()`]) -#' comparing `x` by each group indicated by `grp`. If `grp` has more than two -#' categories, a comparison between each combination of two groups is performed. -#' -#' The function reports U, p and Z-values as well as effect size r and group-rank-means. -#' -#' @param x Bare (unquoted) variable name, or a character vector with the variable name. -#' @param distribution Indicates how the null distribution of the test statistic -#' should be computed. May be one of `"exact"`, `"approximate"` or `"asymptotic"` -#' (default). See [`coin::wilcox_test()`] for details. -#' -#' @inheritParams weighted_sd -#' @inheritParams means_by_group -#' -#' @return (Invisibly) returns a data frame with U, p and Z-values for each group-comparison -#' as well as effect-size r; additionally, group-labels and groups' n's are -#' also included. -#' -#' @note This function calls the \code{\link[coin]{wilcox_test}} with formula. If \code{grp} -#' has more than two groups, additionally a Kruskal-Wallis-Test (see \code{\link{kruskal.test}}) -#' is performed. \cr \cr -#' Interpretation of effect sizes, as a rule-of-thumb: -#' \itemize{ -#' \item small effect >= 0.1 -#' \item medium effect >= 0.3 -#' \item large effect >= 0.5 -#' } -#' -#' @examples -#' data(efc) -#' # Mann-Whitney-U-Tests for elder's age by elder's sex. -#' mwu(efc, e17age, e16sex) -#' -#' # using formula interface -#' mwu(e17age ~ e16sex, efc) -#' -#' # Mann-Whitney-Tests for elder's age by each level elder's dependency. -#' mwu(efc, e17age, e42dep) -#' -#' @importFrom stats na.omit wilcox.test kruskal.test -#' @importFrom sjmisc recode_to is_empty -#' @importFrom sjlabelled get_labels as_numeric -#' @importFrom rlang quo_name enquo -#' @export -mwu <- function(data, ...) { - UseMethod("mwu") -} - -#' @rdname mwu -#' @export -mwu.default <- function(data, - x, - grp, - distribution = "asymptotic", - out = c("txt", "viewer", "browser"), - encoding = "UTF-8", - file = NULL, - ...) { - - out <- match.arg(out) - - if (out != "txt" && !requireNamespace("sjPlot", quietly = TRUE)) { - message("Package `sjPlot` needs to be loaded to print HTML tables.") - out <- "txt" - } - - if (!requireNamespace("coin", quietly = TRUE)) { - stop("Package `coin` needs to be installed to compute the Mann-Whitney-U test.", call. = FALSE) - } - - - # create quosures - grp.name <- rlang::quo_name(rlang::enquo(grp)) - dv.name <- rlang::quo_name(rlang::enquo(x)) - - # create string with variable names - vars <- c(grp.name, dv.name) - - # get data - data <- suppressMessages(dplyr::select(data, !! vars)) - - grp <- data[[grp.name]] - dv <- data[[dv.name]] - - # coerce factor and character to numeric - if (is.factor(grp) || is.character(grp)) grp <- sjlabelled::as_numeric(grp) - - # group "counter" (index) should start with 1, not 0 - if (min(grp, na.rm = TRUE) < 1) grp <- sjmisc::recode_to(grp, lowest = 1, append = FALSE) - - # retrieve unique group values. need to iterate all values - grp_values <- sort(unique(stats::na.omit(grp))) - - # length of value range - cnt <- length(grp_values) - labels <- sjlabelled::get_labels( - grp, attr.only = FALSE, values = NULL, non.labelled = TRUE - ) - - df <- data.frame() - for (i in seq_len(cnt)) { - for (j in i:cnt) { - if (i != j) { - # retrieve cases (rows) of subgroups - xsub <- dv[which(grp == grp_values[i] | grp == grp_values[j])] - ysub <- grp[which(grp == grp_values[i] | grp == grp_values[j])] - - # this is for unpaired wilcox.test() - xsub_2 <- stats::na.omit(dv[which(grp == grp_values[i])]) - ysub_2 <- stats::na.omit(dv[which(grp == grp_values[j])]) - - # only use rows with non-missings - ysub <- ysub[which(!is.na(xsub))] - - # remove missings - xsub <- as.numeric(stats::na.omit(xsub)) - ysub.n <- stats::na.omit(ysub) - - # grouping variable is a factor - ysub <- as.factor(ysub.n) - - wcdat <- data.frame( - x = xsub, - y = ysub - ) - - # perfom wilcox test - wt <- coin::wilcox_test(x ~ y, data = wcdat, distribution = distribution) - - # compute statistics - u <- as.numeric(coin::statistic(wt, type = "linear")) - z <- as.numeric(coin::statistic(wt, type = "standardized")) - p <- coin::pvalue(wt) - r <- abs(z / sqrt(length(ysub))) - w <- stats::wilcox.test(xsub_2, ysub_2, paired = FALSE)$statistic - rkm.i <- mean(rank(xsub)[which(ysub.n == grp_values[i])], na.rm = TRUE) - rkm.j <- mean(rank(xsub)[which(ysub.n == grp_values[j])], na.rm = TRUE) - - # compute n for each group - n_grp1 <- length(xsub[which(ysub.n == grp_values[i])]) - n_grp2 <- length(xsub[which(ysub.n == grp_values[j])]) - - # generate result data frame - df <- - rbind( - df, - cbind( - grp1 = grp_values[i], - grp1.label = labels[i], - grp1.n = n_grp1, - grp2 = grp_values[j], - grp2.label = labels[j], - grp2.n = n_grp2, - u = u, - w = w, - p = p, - z = z, - r = r, - rank.mean.grp1 = rkm.i, - rank.mean.grp2 = rkm.j - ) - ) - } - } - } - - # convert variables - df[["grp1"]] <- as.numeric(as.character(df[["grp1"]])) - df[["grp2"]] <- as.numeric(as.character(df[["grp2"]])) - df[["grp1.n"]] <- as.numeric(as.character(df[["grp1.n"]])) - df[["grp2.n"]] <- as.numeric(as.character(df[["grp2.n"]])) - df[["grp1.label"]] <- as.character(df[["grp1.label"]]) - df[["grp2.label"]] <- as.character(df[["grp2.label"]]) - df[["u"]] <- as.numeric(as.character(df[["u"]])) - df[["w"]] <- as.numeric(as.character(df[["w"]])) - df[["p"]] <- as.numeric(as.character(df[["p"]])) - df[["z"]] <- as.numeric(as.character(df[["z"]])) - df[["r"]] <- as.numeric(as.character(df[["r"]])) - df[["rank.mean.grp1"]] <- as.numeric(as.character(df[["rank.mean.grp1"]])) - df[["rank.mean.grp2"]] <- as.numeric(as.character(df[["rank.mean.grp2"]])) - - # prepare a data frame that can be used for 'sjt.df'. - tab.df <- - data_frame( - Groups = sprintf("%s
%s", df$grp1.label, df$grp2.label), - N = sprintf("%s
%s", df$grp1.n, df$grp2.n), - 'Mean Rank' = sprintf("%.2f
%.2f", df$rank.mean.grp1, df$rank.mean.grp2), - 'Mann-Whitney-U' = as.character(df$u), - 'Wilcoxon-W' = as.character(df$w), - Z = sprintf("%.3f", df$z), - 'Effect Size' = sprintf("%.3f", df$r), - p = sprintf("%.3f", df$p) - ) - - # replace 0.001 with <0.001 - tab.df$p[which(tab.df$p == "0.001")] <- "<0.001" - - ret.df <- list(df = df, tab.df = tab.df, data = data.frame(dv, grp)) - - # save how to print output - attr(ret.df, "print") <- out - attr(ret.df, "encoding") <- encoding - attr(ret.df, "file") <- file - - if (out %in% c("viewer", "browser")) - class(ret.df) <- c("mwu", "sjt_mwu") - else - class(ret.df) <- c("mwu", "sj_mwu") - - ret.df -} - - -#' @importFrom dplyr select -#' @rdname mwu -#' @export -mwu.formula <- function(formula, - data, - distribution = "asymptotic", - out = c("txt", "viewer", "browser"), - encoding = "UTF-8", - file = NULL, - ...) { - vars <- all.vars(formula) - mwu(data, x = !! vars[1], grp = !! vars[2], distribution = distribution, out = out, encoding = encoding, file = file, ...) -} - - -#' @rdname mwu -#' @export -mannwhitney <- mwu diff --git a/R/wtd_chisqtest.R b/R/wtd_chisqtest.R deleted file mode 100644 index 99f9a28..0000000 --- a/R/wtd_chisqtest.R +++ /dev/null @@ -1,91 +0,0 @@ -#' @rdname weighted_sd -#' @export -weighted_chisqtest <- function(data, ...) { - UseMethod("weighted_chisqtest") -} - -#' @importFrom dplyr select -#' @importFrom stats na.omit chisq.test as.formula -#' @rdname weighted_sd -#' @export -weighted_chisqtest.default <- function(data, x, y, weights, ...) { - x.name <- deparse(substitute(x)) - y.name <- deparse(substitute(y)) - w.name <- deparse(substitute(weights)) - - if (w.name == "NULL") { - w.name <- "weights" - data$weights <- 1 - } - - # create string with variable names - vars <- .compact_character(c(x.name, y.name, w.name)) - - # get data - dat <- suppressMessages(dplyr::select(data, !! vars)) - dat <- stats::na.omit(dat) - - colnames(dat)[ncol(dat)] <- ".weights" - - # check if we have chisq-test for given probabilities - dot_args <- list(...) - if ("p" %in% names(dot_args)) { - .weighted_chisq_for_prob(dat, x.name, prob = dot_args[["p"]]) - } else { - crosstable_statistics(data = dat, statistics = "auto", weights = ".weights", ...) - } -} - - -#' @importFrom stats xtabs -#' @rdname weighted_sd -#' @export -weighted_chisqtest.formula <- function(formula, data, ...) { - vars <- all.vars(formula) - dot_args <- list(...) - - if (length(vars) < 3 && !"p" %in% names(dot_args)) { - vars <- c(vars, ".weights") - data$.weights <- 1 - } - - if ("p" %in% names(dot_args)) { - dat <- data[vars] - colnames(dat)[ncol(dat)] <- ".weights" - .weighted_chisq_for_prob(dat, names(dat)[1], prob = dot_args[["p"]]) - } else { - tab <- as.table(round(stats::xtabs(data[[vars[3]]] ~ data[[vars[1]]] + data[[vars[2]]]))) - class(tab) <- "table" - crosstable_statistics(data = tab, statistics = "auto", weights = NULL, ...) - } -} - - - -.weighted_chisq_for_prob <- function(dat, x.name, prob) { - if (!requireNamespace("survey", quietly = TRUE)) { - stop("Package `survey` needed to for this function to work. Please install it.", call. = FALSE) - } - - if (abs(sum(prob) - 1) > sqrt(.Machine$double.eps)) { - prob <- prob / sum(prob) - } - - dat$sj_subject_id <- 1:nrow(dat) - dat$sj_weights <- dat$.weights - design <- survey::svydesign(id = ~sj_subject_id, weights = ~sj_weights, data = dat) - stable <- survey::svytable(stats::as.formula(paste0("~", x.name)), design) - out <- stats::chisq.test(stable, p = prob) - - structure(class = "sj_xtab_stat2", list( - estimate = out$statistic, - p.value = out$p.value, - stat.name = "Chi-squared", - stat.html = "χ2", - df = out$parameter, - n_obs = nrow(dat), - method = "Weighted chi-squared test for given probabilities", - method.html = "Weighted χ2 test for given probabilities", - method.short = "Chi-squared" - )) -} diff --git a/_pkgdown.yml b/_pkgdown.yml index 55cb5b2..cf57a80 100644 --- a/_pkgdown.yml +++ b/_pkgdown.yml @@ -30,7 +30,6 @@ reference: - title: "Other (Summary) Statistics" contents: - gmd - - mannwhitney - mann_whitney_test - chi_squared_test - mean_n diff --git a/man/mwu.Rd b/man/mwu.Rd deleted file mode 100644 index d8fe669..0000000 --- a/man/mwu.Rd +++ /dev/null @@ -1,104 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/mwu.R -\name{mwu} -\alias{mwu} -\alias{mwu.default} -\alias{mwu.formula} -\alias{mannwhitney} -\title{Mann-Whitney-U-Test} -\usage{ -mwu(data, ...) - -\method{mwu}{default}( - data, - x, - grp, - distribution = "asymptotic", - out = c("txt", "viewer", "browser"), - encoding = "UTF-8", - file = NULL, - ... -) - -\method{mwu}{formula}( - formula, - data, - distribution = "asymptotic", - out = c("txt", "viewer", "browser"), - encoding = "UTF-8", - file = NULL, - ... -) - -mannwhitney(data, ...) -} -\arguments{ -\item{data}{A data frame.} - -\item{...}{For \code{weighted_ttest()} and \code{weighted_mannwhitney()}, currently not used. -For \code{weighted_chisqtest()}, further arguments passed down to -\code{\link[stats]{chisq.test}}.} - -\item{x}{Bare (unquoted) variable name, or a character vector with the variable name.} - -\item{grp}{Bare (unquoted) name of the cross-classifying variable, where -\code{x} is grouped into the categories represented by \code{grp}, -or a character vector with the variable name.} - -\item{distribution}{Indicates how the null distribution of the test statistic -should be computed. May be one of \code{"exact"}, \code{"approximate"} or \code{"asymptotic"} -(default). See \code{\link[coin:LocationTests]{coin::wilcox_test()}} for details.} - -\item{out}{Character vector, indicating whether the results should be printed -to console (\code{out = "txt"}) or as HTML-table in the viewer-pane -(\code{out = "viewer"}) or browser (\code{out = "browser"}), of if the -results should be plotted (\code{out = "plot"}, only applies to certain -functions). May be abbreviated.} - -\item{encoding}{Character vector, indicating the charset encoding used -for variable and value labels. Default is \code{"UTF-8"}. Only used -when \code{out} is not \code{"txt"}.} - -\item{file}{Destination file, if the output should be saved as file. -Only used when \code{out} is not \code{"txt"}.} - -\item{formula}{A formula of the form \code{lhs ~ rhs1 + rhs2} where \code{lhs} is a -numeric variable giving the data values and \code{rhs1} a factor with two -levels giving the corresponding groups and \code{rhs2} a variable with weights.} -} -\value{ -(Invisibly) returns a data frame with U, p and Z-values for each group-comparison -as well as effect-size r; additionally, group-labels and groups' n's are -also included. -} -\description{ -This function performs a Mann-Whitney-U-Test (or Wilcoxon rank -sum test for \emph{unpaired} samples, see \code{\link[=wilcox.test]{wilcox.test()}} and \code{\link[coin:LocationTests]{coin::wilcox_test()}}) -comparing \code{x} by each group indicated by \code{grp}. If \code{grp} has more than two -categories, a comparison between each combination of two groups is performed. - -The function reports U, p and Z-values as well as effect size r and group-rank-means. -} -\note{ -This function calls the \code{\link[coin]{wilcox_test}} with formula. If \code{grp} -has more than two groups, additionally a Kruskal-Wallis-Test (see \code{\link{kruskal.test}}) -is performed. \cr \cr -Interpretation of effect sizes, as a rule-of-thumb: -\itemize{ -\item small effect >= 0.1 -\item medium effect >= 0.3 -\item large effect >= 0.5 -} -} -\examples{ -data(efc) -# Mann-Whitney-U-Tests for elder's age by elder's sex. -mwu(efc, e17age, e16sex) - -# using formula interface -mwu(e17age ~ e16sex, efc) - -# Mann-Whitney-Tests for elder's age by each level elder's dependency. -mwu(efc, e17age, e42dep) - -} diff --git a/man/weighted_sd.Rd b/man/weighted_sd.Rd index 3044601..576b8fe 100644 --- a/man/weighted_sd.Rd +++ b/man/weighted_sd.Rd @@ -1,12 +1,8 @@ % Generated by roxygen2: do not edit by hand -% Please edit documentation in R/svy_median.R, R/wtd_chisqtest.R, R/wtd_cor.R, -% R/wtd_mean.R, R/wtd_median.R, R/wtd_mwu.R, R/wtd_sd.R, R/wtd_se.R, -% R/wtd_ttest.R +% Please edit documentation in R/svy_median.R, R/wtd_cor.R, R/wtd_mean.R, +% R/wtd_median.R, R/wtd_mwu.R, R/wtd_sd.R, R/wtd_se.R, R/wtd_ttest.R \name{survey_median} \alias{survey_median} -\alias{weighted_chisqtest} -\alias{weighted_chisqtest.default} -\alias{weighted_chisqtest.formula} \alias{weighted_correlation} \alias{weighted_correlation.default} \alias{weighted_correlation.formula} @@ -26,12 +22,6 @@ \usage{ survey_median(x, design) -weighted_chisqtest(data, ...) - -\method{weighted_chisqtest}{default}(data, x, y, weights, ...) - -\method{weighted_chisqtest}{formula}(formula, data, ...) - weighted_correlation(data, ...) \method{weighted_correlation}{default}(data, x, y, weights, ci.lvl = 0.95, ...) @@ -101,12 +91,12 @@ the variable name.} the variable name of the numeric vector of weights. If \code{weights = NULL}, unweighted statistic is reported.} +\item{ci.lvl}{Confidence level of the interval.} + \item{formula}{A formula of the form \code{lhs ~ rhs1 + rhs2} where \code{lhs} is a numeric variable giving the data values and \code{rhs1} a factor with two levels giving the corresponding groups and \code{rhs2} a variable with weights.} -\item{ci.lvl}{Confidence level of the interval.} - \item{grp}{Bare (unquoted) name of the cross-classifying variable, where \code{x} is grouped into the categories represented by \code{grp}, or a character vector with the variable name.} From fb5e6857b83939317dfeaee0d4cba1f9872f36f7 Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 8 May 2024 16:37:34 +0200 Subject: [PATCH 10/34] cleanup --- R/wtd_sd.R | 10 ---------- man/weighted_sd.Rd | 10 ---------- tests/testthat/test-wtd.R | 15 --------------- 3 files changed, 35 deletions(-) diff --git a/R/wtd_sd.R b/R/wtd_sd.R index b7db04d..95af674 100644 --- a/R/wtd_sd.R +++ b/R/wtd_sd.R @@ -82,16 +82,6 @@ #' weighted_ttest(efc, e17age, weights = weight) #' weighted_ttest(efc, e17age, c160age, weights = weight) #' weighted_ttest(e17age ~ e16sex + weight, efc) -#' -#' # weighted Mann-Whitney-U-test ---- -#' weighted_mannwhitney(c12hour ~ c161sex + weight, efc) -#' -#' # weighted Chi-squared-test ---- -#' weighted_chisqtest(efc, c161sex, e16sex, weights = weight, correct = FALSE) -#' weighted_chisqtest(c172code ~ c161sex + weight, efc) -#' -#' # weighted Chi-squared-test for given probabilities ---- -#' weighted_chisqtest(c172code ~ weight, efc, p = c(.33, .33, .34)) #' @export weighted_sd <- function(x, weights = NULL) { UseMethod("weighted_sd") diff --git a/man/weighted_sd.Rd b/man/weighted_sd.Rd index 576b8fe..c553f99 100644 --- a/man/weighted_sd.Rd +++ b/man/weighted_sd.Rd @@ -167,14 +167,4 @@ efc$weight <- abs(rnorm(nrow(efc), 1, .3)) weighted_ttest(efc, e17age, weights = weight) weighted_ttest(efc, e17age, c160age, weights = weight) weighted_ttest(e17age ~ e16sex + weight, efc) - -# weighted Mann-Whitney-U-test ---- -weighted_mannwhitney(c12hour ~ c161sex + weight, efc) - -# weighted Chi-squared-test ---- -weighted_chisqtest(efc, c161sex, e16sex, weights = weight, correct = FALSE) -weighted_chisqtest(c172code ~ c161sex + weight, efc) - -# weighted Chi-squared-test for given probabilities ---- -weighted_chisqtest(c172code ~ weight, efc, p = c(.33, .33, .34)) } diff --git a/tests/testthat/test-wtd.R b/tests/testthat/test-wtd.R index 21077b3..7642253 100644 --- a/tests/testthat/test-wtd.R +++ b/tests/testthat/test-wtd.R @@ -17,21 +17,6 @@ if (require("testthat") && require("sjstats")) { expect_equal(weighted_median(efc$c12hour, weights = NULL), 20, tolerance = 1e-5) }) - test_that("weighted_chisqtest", { - w <- weighted_chisqtest(efc, c161sex, c172code, weights = weight) - expect_equal(w$estimate, 0.06668895, tolerance = 1e-5) - expect_equal(w$p.value, 0.1517221, tolerance = 1e-5) - - w <- weighted_chisqtest(c161sex ~ c172code + weight, efc) - expect_equal(w$estimate, 0.06668895, tolerance = 1e-5) - expect_equal(w$p.value, 0.1517221, tolerance = 1e-5) - }) - - test_that("weighted_mannwhitney", { - weighted_mannwhitney(efc, c12hour, c161sex, weights = weight) - weighted_mannwhitney(c12hour ~ c161sex + weight, efc) - }) - test_that("weighted_ttest", { weighted_ttest(efc, e17age, weights = weight) weighted_ttest(efc, e17age, c160age, weights = weight) From 005ff1e61971390546672ebd6c35ee0cc5e0bf1a Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 8 May 2024 16:40:05 +0200 Subject: [PATCH 11/34] fix --- R/chi_squared_test.R | 15 ++++++++++++++- 1 file changed, 14 insertions(+), 1 deletion(-) diff --git a/R/chi_squared_test.R b/R/chi_squared_test.R index 7d76ad8..e82a26b 100644 --- a/R/chi_squared_test.R +++ b/R/chi_squared_test.R @@ -101,6 +101,7 @@ chi_squared_test <- function(data, stringsAsFactors = FALSE ) class(out) <- c("sj_htest_chi", "data.frame") + attr(out, "weighted") <- !is.null(weights) attr(out, "caption") <- "Contingency Tables" out } @@ -164,6 +165,7 @@ chi_squared_test <- function(data, ) class(out) <- c("sj_htest_chi", "data.frame") attr(out, "caption") <- "given Probabilities" + attr(out, "weighted") <- !is.null(weights) out } @@ -172,10 +174,21 @@ chi_squared_test <- function(data, #' @export print.sj_htest_chi <- function(x, ...) { + weighted <- attributes(x)$weighted + if (weighted) { + weight_string <- " (weighted)" + } else { + weight_string <- "" + } + # get length of method name, to align output l <- max(nchar(c(x$statistic_name, x$effect_size_name, "p-value", "Observations"))) # headline - insight::print_color(sprintf("\n# Chi-Squared Test for %s\n\n", attributes(x)$caption), "blue") + insight::print_color(sprintf( + "\n# Chi-Squared Test for %s%s\n\n", + attributes(x)$caption, + weight_string + ), "blue") # print test statistic cat(sprintf(" %*s: %.4f\n", l, x$statistic_name, x$statistic)) From 60c1bf31edcd27f242b855eab5c0ed94f8446ba8 Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 8 May 2024 16:44:49 +0200 Subject: [PATCH 12/34] cleanup --- NAMESPACE | 11 -- R/Deprecated.R | 16 ++ R/grpmean.R | 335 ----------------------------------- R/mean_n.R | 78 -------- _pkgdown.yml | 2 - man/crosstable_statistics.Rd | 4 - man/mean_n.Rd | 69 -------- man/means_by_group.Rd | 94 ---------- man/r2.Rd | 6 + 9 files changed, 22 insertions(+), 593 deletions(-) delete mode 100644 R/grpmean.R delete mode 100644 R/mean_n.R delete mode 100644 man/mean_n.Rd delete mode 100644 man/means_by_group.Rd diff --git a/NAMESPACE b/NAMESPACE index f586c75..7947a77 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -82,7 +82,6 @@ export(find_beta2) export(find_cauchy) export(find_normal) export(gmd) -export(grpmean) export(icc) export(inequ_trend) export(is_prime) @@ -131,14 +130,10 @@ importFrom(dplyr,case_when) importFrom(dplyr,filter) importFrom(dplyr,group_vars) importFrom(dplyr,mutate) -importFrom(dplyr,n_distinct) -importFrom(dplyr,pull) importFrom(dplyr,quos) importFrom(dplyr,select) importFrom(dplyr,select_if) importFrom(dplyr,summarise) -importFrom(emmeans,contrast) -importFrom(emmeans,emmeans) importFrom(insight,export_table) importFrom(insight,find_formula) importFrom(insight,find_response) @@ -156,7 +151,6 @@ importFrom(performance,rmse) importFrom(purrr,flatten_df) importFrom(purrr,map) importFrom(purrr,map2) -importFrom(purrr,map_chr) importFrom(purrr,map_dbl) importFrom(purrr,map_df) importFrom(purrr,map_lgl) @@ -165,15 +159,10 @@ importFrom(rlang,.data) importFrom(rlang,enquo) importFrom(rlang,quo_name) importFrom(sjlabelled,as_numeric) -importFrom(sjlabelled,drop_labels) -importFrom(sjlabelled,get_label) -importFrom(sjlabelled,get_labels) -importFrom(sjmisc,add_variables) importFrom(sjmisc,is_empty) importFrom(sjmisc,is_float) importFrom(sjmisc,is_num_fac) importFrom(sjmisc,str_contains) -importFrom(sjmisc,to_value) importFrom(sjmisc,trim) importFrom(sjmisc,typical_value) importFrom(stats,approx) diff --git a/R/Deprecated.R b/R/Deprecated.R index 862b8da..94a27e2 100644 --- a/R/Deprecated.R +++ b/R/Deprecated.R @@ -83,3 +83,19 @@ se <- function(x, ...) { .Defunct("parameters::standard_error()") parameters::standard_error(x) } + + +#' @rdname r2 +#' @export +means_by_group <- function(x, ...) { + .Defunct("datawizard::means_by_group()") + datawizard::means_by_group(x, ...) +} + + +#' @rdname r2 +#' @export +mean_n <- function(x, ...) { + .Defunct("datawizard::row_means()") + datawizard::row_means(x, ...) +} diff --git a/R/grpmean.R b/R/grpmean.R deleted file mode 100644 index ab2eb2d..0000000 --- a/R/grpmean.R +++ /dev/null @@ -1,335 +0,0 @@ -#' @title Summary of mean values by group -#' @name means_by_group -#' -#' @description Computes mean, sd and se for each sub-group (indicated by \code{grp}) -#' of \code{dv}. -#' -#' @param x A (grouped) data frame. -#' @param dv Name of the dependent variable, for which the mean value, grouped -#' by \code{grp}, is computed. -#' @param grp Factor with the cross-classifying variable, where \code{dv} is -#' grouped into the categories represented by \code{grp}. Numeric vectors -#' are coerced to factors. -#' @param weights Name of variable in \code{x} that indicated the vector of -#' weights that will be applied to weight all observations. Default is -#' \code{NULL}, so no weights are used. -#' @param digits Numeric, amount of digits after decimal point when rounding -#' estimates and values. -#' @param file Destination file, if the output should be saved as file. -#' Only used when \code{out} is not \code{"txt"}. -#' @param encoding Character vector, indicating the charset encoding used -#' for variable and value labels. Default is \code{"UTF-8"}. Only used -#' when \code{out} is not \code{"txt"}. -#' @param out Character vector, indicating whether the results should be printed -#' to console (\code{out = "txt"}) or as HTML-table in the viewer-pane -#' (\code{out = "viewer"}) or browser (\code{out = "browser"}), of if the -#' results should be plotted (\code{out = "plot"}, only applies to certain -#' functions). May be abbreviated. -#' -#' @return For non-grouped data frames, \code{means_by_group()} returns a data frame with -#' following columns: \code{term}, \code{mean}, \code{N}, \code{std.dev}, -#' \code{std.error} and \code{p.value}. For grouped data frames, returns -#' a list of such data frames. -#' -#' @details This function performs a One-Way-Anova with \code{dv} as dependent -#' and \code{grp} as independent variable, by calling -#' \code{lm(count ~ as.factor(grp))}. Then \code{\link[emmeans]{contrast}} -#' is called to get p-values for each sub-group. P-values indicate whether -#' each group-mean is significantly different from the total mean. -#' -#' @examples -#' data(efc) -#' means_by_group(efc, c12hour, e42dep) -#' -#' data(iris) -#' means_by_group(iris, Sepal.Width, Species) -#' -#' # also works for grouped data frames -#' if (require("dplyr")) { -#' efc %>% -#' group_by(c172code) %>% -#' means_by_group(c12hour, e42dep) -#' } -#' -#' # weighting -#' efc$weight <- abs(rnorm(n = nrow(efc), mean = 1, sd = .5)) -#' means_by_group(efc, c12hour, e42dep, weights = weight) -#' @importFrom sjlabelled get_label drop_labels get_labels -#' @importFrom stats lm na.omit sd weighted.mean -#' @importFrom purrr map_chr map_df -#' @importFrom sjmisc to_value is_empty -#' @importFrom rlang enquo .data quo_name -#' @export -means_by_group <- function(x, - dv, - grp, - weights = NULL, - digits = 2, - out = c("txt", "viewer", "browser"), - encoding = "UTF-8", - file = NULL) { - - out <- match.arg(out) - - if (out != "txt" && !requireNamespace("sjPlot", quietly = TRUE)) { - message("Package `sjPlot` needs to be loaded to print HTML tables.") - out <- "txt" - } - - # create quosures - grp.name <- rlang::quo_name(rlang::enquo(grp)) - dv.name <- rlang::quo_name(rlang::enquo(dv)) - - # weights need extra checking, might be NULL - if (!missing(weights)) { - .weights <- try(rlang::quo_name(rlang::enquo(weights)), silent = TRUE) - if (inherits(.weights, "try-error")) .weights <- NULL - - w.string <- try(eval(weights), silent = TRUE) - if (!inherits(w.string, "try-error") && !is.null(w.string) && is.character(w.string)) .weights <- w.string - - if (sjmisc::is_empty(.weights) || .weights == "NULL") .weights <- NULL - } else - .weights <- NULL - - - # create string with variable names - vars <- c(grp.name, dv.name, .weights) - - # get data - x <- suppressMessages(dplyr::select(x, !! vars)) - - # set value and row labels - varGrpLabel <- sjlabelled::get_label(x[[grp.name]], def.value = grp.name) - varCountLabel <- sjlabelled::get_label(x[[dv.name]], def.value = dv.name) - - # first, drop unused labels - x[[grp.name]] <- sjlabelled::drop_labels(x[[grp.name]], drop.na = TRUE) - - # now get valid value labels - value.labels <- sjlabelled::get_labels( - x[[grp.name]], attr.only = F, values = "n", non.labelled = TRUE - ) - - # return values - dataframes <- list() - - # do we have a grouped data frame? - if (inherits(x, "grouped_df")) { - # get grouped data - grps <- get_grouped_data(x) - - # now plot everything - for (i in seq_len(nrow(grps))) { - # copy back labels to grouped data frame - tmp <- sjlabelled::copy_labels(grps$data[[i]], x) - - # get grouped means table - dummy <- means_by_group_helper( - x = tmp, - dv = dv.name, - grp = grp.name, - weight.by = .weights, - value.labels = value.labels, - varCountLabel = varCountLabel, - varGrpLabel = varGrpLabel - ) - - attr(dummy, "group") <- get_grouped_title(x, grps, i, sep = "\n") - - # save data frame for return value - dataframes[[length(dataframes) + 1]] <- dummy - } - - # add class-attr for print-method() - if (out == "txt") - class(dataframes) <- c("sj_grpmeans", "list") - else - class(dataframes) <- c("sjt_grpmeans", "list") - - } else { - dataframes <- means_by_group_helper( - x = x, - dv = dv.name, - grp = grp.name, - weight.by = .weights, - value.labels = value.labels, - varCountLabel = varCountLabel, - varGrpLabel = varGrpLabel - ) - - # add class-attr for print-method() - if (out == "txt") - class(dataframes) <- c("sj_grpmean", class(dataframes)) - else - class(dataframes) <- c("sjt_grpmean", class(dataframes)) - } - - # save how to print output - attr(dataframes, "print") <- out - attr(dataframes, "encoding") <- encoding - attr(dataframes, "file") <- file - attr(dataframes, "digits") <- digits - - dataframes -} - - -#' @importFrom stats pf lm weighted.mean na.omit sd -#' @importFrom sjmisc to_value add_variables -#' @importFrom emmeans emmeans contrast -#' @importFrom dplyr pull select n_distinct -#' @importFrom purrr map_chr -#' @importFrom rlang .data -means_by_group_helper <- function(x, dv, grp, weight.by, value.labels, varCountLabel, varGrpLabel) { - # copy vectors from data frame - dv <- x[[dv]] - grp <- x[[grp]] - - if (!is.null(weight.by)) - weight.by <- x[[weight.by]] - else - weight.by <- 1 - - # convert values to numeric - dv <- sjmisc::to_value(dv) - - # create data frame, for emmeans - mydf <- stats::na.omit(data.frame( - dv = dv, - grp = as.factor(grp), - weight.by = weight.by - )) - - # compute anova statistics for mean table - fit <- stats::lm(dv ~ grp, weights = weight.by, data = mydf) - - # p-values of contrast-means - means.p <- fit %>% - emmeans::emmeans(specs = "grp") %>% - emmeans::contrast(method = "eff") %>% - summary() %>% - dplyr::pull("p.value") - - ## TODO - # efc %>% - # group_by(c172code, c161sex) %>% - # means_by_group(c12hour, e42dep) - - - # check if value labels length matches group count - if (dplyr::n_distinct(mydf$grp) != length(value.labels)) { - # get unique factor levels and check if these are numeric. - # if so, we match the values from value labels and the remaining - # factor levels, so we get the correct value labels for printing - nl <- unique(mydf$grp) - if (sjmisc::is_num_fac(nl)) - value.labels <- value.labels[names(value.labels) %in% levels(nl)] - else - value.labels <- nl - } - - - # create summary - dat <- mydf %>% - dplyr::group_by(.data$grp) %>% - summarise( - mean = stats::weighted.mean(.data$dv, w = .data$weight.by, na.rm = TRUE), - N = round(sum(.data$weight.by)), - std.dev = weighted_sd(.data$dv, .data$weight.by), - std.error = weighted_se(.data$dv, .data$weight.by) - ) %>% - mutate(p.value = means.p) %>% - dplyr::select(-.data$grp) - - # finally, add total-row - dat <- dplyr::bind_rows( - dat, - data_frame( - mean = stats::weighted.mean(mydf$dv, w = mydf$weight.by, na.rm = TRUE), - N = nrow(mydf), - std.dev = weighted_sd(mydf$dv, mydf$weight.by), - std.error = weighted_se(mydf$dv, mydf$weight.by), - p.value = NA - ) - ) - - - # add row labels - dat <- sjmisc::add_variables( - dat, - term = c(unname(value.labels), "Total"), - .after = -1 - ) - - - # get anova statistics for mean table - sum.fit <- summary(fit) - - # r-squared values - r2 <- sum.fit$r.squared - r2.adj <- sum.fit$adj.r.squared - - # F-statistics - fstat <- sum.fit$fstatistic - pval <- stats::pf(fstat[1], fstat[2], fstat[3], lower.tail = F) - - - # copy as attributes - attr(dat, "r2") <- r2 - attr(dat, "adj.r2") <- r2.adj - attr(dat, "fstat") <- fstat[1] - attr(dat, "p.value") <- pval - attr(dat, "dv.label") <- varCountLabel - attr(dat, "grp.label") <- varGrpLabel - - dat -} - - -get_grouped_title <- function(x, grps, i, sep = "\n") { - # create title for first grouping level - tp <- get_title_part(x, grps, 1, i) - title <- sprintf("%s: %s", tp[1], tp[2]) - - # do we have another groupng variable? - if (length(dplyr::group_vars(x)) > 1) { - tp <- get_title_part(x, grps, 2, i) - title <- sprintf("%s%s%s: %s", title, sep, tp[1], tp[2]) - } - - # return title - title -} - - -get_title_part <- function(x, grps, level, i) { - # prepare title for group - var.name <- colnames(grps)[level] - - # get values from value labels - vals <- sjlabelled::get_values(x[[var.name]]) - # if we have no value labels, get values directly - if (is.null(vals)) { - vals <- unique(x[[var.name]]) - lab.pos <- i - } else { - # find position of value labels for current group - lab.pos <- which(vals == grps[[var.name]][i]) - } - - # get variable and value labels - t1 <- sjlabelled::get_label(x[[var.name]], def.value = var.name) - t2 <- sjlabelled::get_labels(x[[var.name]])[lab.pos] - - # if we have no value label, use value instead - if (is.null(t2)) t2 <- vals[lab.pos] - - # generate title - c(t1, t2) -} - - -#' @rdname means_by_group -#' @export -grpmean <- means_by_group diff --git a/R/mean_n.R b/R/mean_n.R deleted file mode 100644 index 5d6a2e0..0000000 --- a/R/mean_n.R +++ /dev/null @@ -1,78 +0,0 @@ -#' @title Row means with min amount of valid values -#' @name mean_n -#' @description This function is similar to the SPSS \code{MEAN.n} function and computes -#' row means from a \code{data.frame} or \code{matrix} if at least \code{n} -#' values of a row are valid (and not \code{NA}). -#' -#' @param dat A data frame with at least two columns, where row means are applied. -#' @param n May either be -#' \itemize{ -#' \item a numeric value that indicates the amount of valid values per row to calculate the row mean; -#' \item or a value between 0 and 1, indicating a proportion of valid values per row to calculate the row mean (see 'Details'). -#' } -#' If a row's sum of valid values is less than \code{n}, \code{NA} will be returned as row mean value. -#' @param digits Numeric value indicating the number of decimal places to be used for rounding mean -#' value. Negative values are allowed (see 'Details'). -#' -#' @return A vector with row mean values of \code{df} for those rows with at least \code{n} -#' valid values. Else, \code{NA} is returned. -#' -#' @details Rounding to a negative number of \code{digits} means rounding to a power of -#' ten, so for example mean_n(df, 3, digits = -2) rounds to the -#' nearest hundred. \cr \cr -#' For \code{n}, must be a numeric value from \code{0} to \code{ncol(dat)}. If -#' a \emph{row} in \code{dat} has at least \code{n} non-missing values, the -#' row mean is returned. If \code{n} is a non-integer value from 0 to 1, -#' \code{n} is considered to indicate the proportion of necessary non-missing -#' values per row. E.g., if \code{n = .75}, a row must have at least \code{ncol(dat) * n} -#' non-missing values for the row mean to be calculated. See 'Examples'. -#' -#' @references \href{https://r4stats.com/2014/09/03/adding-the-spss-mean-n-function-to-r/}{r4stats.com} -#' -#' @examples -#' dat <- data.frame(c1 = c(1,2,NA,4), -#' c2 = c(NA,2,NA,5), -#' c3 = c(NA,4,NA,NA), -#' c4 = c(2,3,7,8)) -#' -#' # needs at least 4 non-missing values per row -#' mean_n(dat, 4) # 1 valid return value -#' -#' # needs at least 3 non-missing values per row -#' mean_n(dat, 3) # 2 valid return values -#' -#' # needs at least 2 non-missing values per row -#' mean_n(dat, 2) -#' -#' # needs at least 1 non-missing value per row -#' mean_n(dat, 1) # all means are shown -#' -#' # needs at least 50% of non-missing values per row -#' mean_n(dat, .5) # 3 valid return values -#' -#' # needs at least 75% of non-missing values per row -#' mean_n(dat, .75) # 2 valid return values -#' -#' @export -mean_n <- function(dat, n, digits = 2) { - # is 'n' indicating a proportion? - digs <- n %% 1 - if (digs != 0) n <- round(ncol(dat) * digs) - - # coerce matrix to data frame - if (is.matrix(dat)) dat <- as.data.frame(dat) - - # check if we have a data framme with at least two columns - if (!is.data.frame(dat) || ncol(dat) < 2) { - warning("`dat` must be a data frame with at least two columns.", call. = TRUE) - return(NA) - } - - # n may not be larger as df's amount of columns - if (ncol(dat) < n) { - warning("`n` must be smaller or equal to number of columns in data frame.", call. = TRUE) - return(NA) - } - - round(apply(dat, 1, function(x) ifelse(sum(!is.na(x)) >= n, mean(x, na.rm = TRUE), NA)), digits) -} diff --git a/_pkgdown.yml b/_pkgdown.yml index cf57a80..bfe8987 100644 --- a/_pkgdown.yml +++ b/_pkgdown.yml @@ -32,8 +32,6 @@ reference: - gmd - mann_whitney_test - chi_squared_test - - mean_n - - means_by_group - var_pop - title: "Tools for Regression Models" diff --git a/man/crosstable_statistics.Rd b/man/crosstable_statistics.Rd index 358c283..20b2f03 100644 --- a/man/crosstable_statistics.Rd +++ b/man/crosstable_statistics.Rd @@ -81,10 +81,6 @@ will be irgnored.} \item{statistics}{Name of measure of association that should be computed. May be one of \code{"auto"}, \code{"cramer"}, \code{"phi"}, \code{"spearman"}, \code{"kendall"}, \code{"pearson"} or \code{"fisher"}. See 'Details'.} - -\item{weights}{Name of variable in \code{x} that indicated the vector of -weights that will be applied to weight all observations. Default is -\code{NULL}, so no weights are used.} } \value{ For \code{phi()}, the table's Phi value. For \code{cramer()}, the diff --git a/man/mean_n.Rd b/man/mean_n.Rd deleted file mode 100644 index 35f6915..0000000 --- a/man/mean_n.Rd +++ /dev/null @@ -1,69 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/mean_n.R -\name{mean_n} -\alias{mean_n} -\title{Row means with min amount of valid values} -\usage{ -mean_n(dat, n, digits = 2) -} -\arguments{ -\item{dat}{A data frame with at least two columns, where row means are applied.} - -\item{n}{May either be -\itemize{ -\item a numeric value that indicates the amount of valid values per row to calculate the row mean; -\item or a value between 0 and 1, indicating a proportion of valid values per row to calculate the row mean (see 'Details'). -} -If a row's sum of valid values is less than \code{n}, \code{NA} will be returned as row mean value.} - -\item{digits}{Numeric value indicating the number of decimal places to be used for rounding mean -value. Negative values are allowed (see 'Details').} -} -\value{ -A vector with row mean values of \code{df} for those rows with at least \code{n} -valid values. Else, \code{NA} is returned. -} -\description{ -This function is similar to the SPSS \code{MEAN.n} function and computes -row means from a \code{data.frame} or \code{matrix} if at least \code{n} -values of a row are valid (and not \code{NA}). -} -\details{ -Rounding to a negative number of \code{digits} means rounding to a power of -ten, so for example mean_n(df, 3, digits = -2) rounds to the -nearest hundred. \cr \cr -For \code{n}, must be a numeric value from \code{0} to \code{ncol(dat)}. If -a \emph{row} in \code{dat} has at least \code{n} non-missing values, the -row mean is returned. If \code{n} is a non-integer value from 0 to 1, -\code{n} is considered to indicate the proportion of necessary non-missing -values per row. E.g., if \code{n = .75}, a row must have at least \code{ncol(dat) * n} -non-missing values for the row mean to be calculated. See 'Examples'. -} -\examples{ -dat <- data.frame(c1 = c(1,2,NA,4), - c2 = c(NA,2,NA,5), - c3 = c(NA,4,NA,NA), - c4 = c(2,3,7,8)) - -# needs at least 4 non-missing values per row -mean_n(dat, 4) # 1 valid return value - -# needs at least 3 non-missing values per row -mean_n(dat, 3) # 2 valid return values - -# needs at least 2 non-missing values per row -mean_n(dat, 2) - -# needs at least 1 non-missing value per row -mean_n(dat, 1) # all means are shown - -# needs at least 50\% of non-missing values per row -mean_n(dat, .5) # 3 valid return values - -# needs at least 75\% of non-missing values per row -mean_n(dat, .75) # 2 valid return values - -} -\references{ -\href{https://r4stats.com/2014/09/03/adding-the-spss-mean-n-function-to-r/}{r4stats.com} -} diff --git a/man/means_by_group.Rd b/man/means_by_group.Rd deleted file mode 100644 index ad82ba3..0000000 --- a/man/means_by_group.Rd +++ /dev/null @@ -1,94 +0,0 @@ -% Generated by roxygen2: do not edit by hand -% Please edit documentation in R/grpmean.R -\name{means_by_group} -\alias{means_by_group} -\alias{grpmean} -\title{Summary of mean values by group} -\usage{ -means_by_group( - x, - dv, - grp, - weights = NULL, - digits = 2, - out = c("txt", "viewer", "browser"), - encoding = "UTF-8", - file = NULL -) - -grpmean( - x, - dv, - grp, - weights = NULL, - digits = 2, - out = c("txt", "viewer", "browser"), - encoding = "UTF-8", - file = NULL -) -} -\arguments{ -\item{x}{A (grouped) data frame.} - -\item{dv}{Name of the dependent variable, for which the mean value, grouped -by \code{grp}, is computed.} - -\item{grp}{Factor with the cross-classifying variable, where \code{dv} is -grouped into the categories represented by \code{grp}. Numeric vectors -are coerced to factors.} - -\item{weights}{Name of variable in \code{x} that indicated the vector of -weights that will be applied to weight all observations. Default is -\code{NULL}, so no weights are used.} - -\item{digits}{Numeric, amount of digits after decimal point when rounding -estimates and values.} - -\item{out}{Character vector, indicating whether the results should be printed -to console (\code{out = "txt"}) or as HTML-table in the viewer-pane -(\code{out = "viewer"}) or browser (\code{out = "browser"}), of if the -results should be plotted (\code{out = "plot"}, only applies to certain -functions). May be abbreviated.} - -\item{encoding}{Character vector, indicating the charset encoding used -for variable and value labels. Default is \code{"UTF-8"}. Only used -when \code{out} is not \code{"txt"}.} - -\item{file}{Destination file, if the output should be saved as file. -Only used when \code{out} is not \code{"txt"}.} -} -\value{ -For non-grouped data frames, \code{means_by_group()} returns a data frame with -following columns: \code{term}, \code{mean}, \code{N}, \code{std.dev}, -\code{std.error} and \code{p.value}. For grouped data frames, returns -a list of such data frames. -} -\description{ -Computes mean, sd and se for each sub-group (indicated by \code{grp}) -of \code{dv}. -} -\details{ -This function performs a One-Way-Anova with \code{dv} as dependent -and \code{grp} as independent variable, by calling -\code{lm(count ~ as.factor(grp))}. Then \code{\link[emmeans]{contrast}} -is called to get p-values for each sub-group. P-values indicate whether -each group-mean is significantly different from the total mean. -} -\examples{ -data(efc) -means_by_group(efc, c12hour, e42dep) - -data(iris) -means_by_group(iris, Sepal.Width, Species) - -# also works for grouped data frames -if (require("dplyr")) { - efc \%>\% - group_by(c172code) \%>\% - means_by_group(c12hour, e42dep) -} - -# weighting -efc$weight <- abs(rnorm(n = nrow(efc), mean = 1, sd = .5)) -means_by_group(efc, c12hour, e42dep, weights = weight) -} diff --git a/man/r2.Rd b/man/r2.Rd index 4b64705..d31c90e 100644 --- a/man/r2.Rd +++ b/man/r2.Rd @@ -11,6 +11,8 @@ \alias{icc} \alias{p_value} \alias{se} +\alias{means_by_group} +\alias{mean_n} \title{Deprecated functions} \usage{ r2(x) @@ -32,6 +34,10 @@ icc(x) p_value(x, ...) se(x, ...) + +means_by_group(x, ...) + +mean_n(x, ...) } \arguments{ \item{x}{An object.} From 0f253f881b151de51fb1d721d11e5bf00341ac42 Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 8 May 2024 16:46:55 +0200 Subject: [PATCH 13/34] fix --- _pkgdown.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/_pkgdown.yml b/_pkgdown.yml index bfe8987..059fc5f 100644 --- a/_pkgdown.yml +++ b/_pkgdown.yml @@ -22,12 +22,12 @@ reference: - crosstable_statistics - table_values -- title: "Weighted Statistics" +- title: "Weighted Estimates and Dispersion" contents: - weight - weighted_ttest -- title: "Other (Summary) Statistics" +- title: "Summary Statistics and Tests" contents: - gmd - mann_whitney_test From 20bf3f33333f5b9327d3a5509baa4d5cdb4fab54 Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 8 May 2024 16:50:33 +0200 Subject: [PATCH 14/34] reduce deps --- DESCRIPTION | 1 - NAMESPACE | 1 - R/anova_stats.R | 5 ++--- R/cv_error.R | 2 +- 4 files changed, 3 insertions(+), 6 deletions(-) diff --git a/DESCRIPTION b/DESCRIPTION index e1c10be..b30d975 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -19,7 +19,6 @@ Depends: utils Imports: bayestestR, - broom, datawizard, dplyr, effectsize, diff --git a/NAMESPACE b/NAMESPACE index 7947a77..450feeb 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -125,7 +125,6 @@ export(xtab_statistics) importFrom(MASS,glm.nb) importFrom(bayestestR,ci) importFrom(bayestestR,equivalence_test) -importFrom(broom,augment) importFrom(dplyr,case_when) importFrom(dplyr,filter) importFrom(dplyr,group_vars) diff --git a/R/anova_stats.R b/R/anova_stats.R index ef7c121..f791768 100644 --- a/R/anova_stats.R +++ b/R/anova_stats.R @@ -29,9 +29,7 @@ #' } #' @export anova_stats <- function(model, digits = 3) { - if (!requireNamespace("pwr", quietly = TRUE)) { - stop("Package `pwr` needed for this function to work. Please install it.", call. = FALSE) - } + insight::check_if_installed("pwr") # .Deprecated("effectsize::effectsize()", package = "effectsize") @@ -94,6 +92,7 @@ aov_stat <- function(model, type) { aov_stat_summary <- function(model) { + insight::check_if_installed("broom") # check if we have a mixed model mm <- is_merMod(model) ori.model <- model diff --git a/R/cv_error.R b/R/cv_error.R index 9ab6f87..c4880b1 100644 --- a/R/cv_error.R +++ b/R/cv_error.R @@ -34,13 +34,13 @@ #' @importFrom modelr crossv_kfold #' @importFrom dplyr mutate summarise #' @importFrom purrr map map2 map_dbl map_df -#' @importFrom broom augment #' @importFrom tidyr unnest #' @importFrom rlang .data #' @importFrom insight find_response #' @importFrom performance rmse #' @export cv_error <- function(data, formula, k = 5) { + insight::check_if_installed("broom") # compute cross validation data cv_data <- data %>% From 383b1a4ca9bffee7d8ff9f99dad56cd4436d1839 Mon Sep 17 00:00:00 2001 From: Daniel Date: Wed, 8 May 2024 18:49:45 +0200 Subject: [PATCH 15/34] fixes --- DESCRIPTION | 1 + R/chi_squared_test.R | 16 +++++++++------- R/mann_whitney_test.R | 12 +++++++----- R/xtab_statistics.R | 4 +++- man/chi_squared_test.Rd | 21 +++++++++++++++------ man/crosstable_statistics.Rd | 4 ++++ man/mann_whitney_test.Rd | 12 +++++++----- tests/testthat/test-grpmean.R | 30 ------------------------------ 8 files changed, 46 insertions(+), 54 deletions(-) delete mode 100644 tests/testthat/test-grpmean.R diff --git a/DESCRIPTION b/DESCRIPTION index b30d975..0ecf66c 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -38,6 +38,7 @@ Imports: tidyr Suggests: brms, + broom, car, coin, ggplot2, diff --git a/R/chi_squared_test.R b/R/chi_squared_test.R index e82a26b..13d5831 100644 --- a/R/chi_squared_test.R +++ b/R/chi_squared_test.R @@ -5,12 +5,14 @@ #' #' The function reports p and Z-values as well as effect size r and group-rank-means. #' -#' @param data A data frame. -#' @param select The dependent variable (numeric) to be used for the test. -#' @param by The grouping variable (factor) to be used for the test. If `by` is -#' not a factor, it will be coerced to a factor. -#' @param weights An optional weighting variable (numeric) to be used for the test. +#' @param probabilities A numeric vector of probabilities for each cell in the +#' contingency table. The length of the vector must match the number of cells +#' in the table, i.e. the number of unique levels of the variable specified +#' in `select`. If `probabilities` is provided, a chi-squared test for given +#' probabilities is conducted. Furthermore, if `probabilities` is given, `by` +#' must be `NULL`. The probabilities must sum to 1. #' @param ... Additional arguments passed down to [`chisq.test()`]. +#' @inheritParams mann_whitney_test #' #' @return A data frame. #' @@ -18,9 +20,9 @@ #' data(efc) #' efc$weight <- abs(rnorm(nrow(efc), 1, 0.3)) #' # Chi-squared-test -#' chi_squared_test(efc, "c161sex", "e16sex") +#' chi_squared_test(efc, "c161sex", by = "e16sex") #' # weighted Chi-squared-test -#' chi_squared_test(efc, "c161sex", "e16sex", weights = "weight") +#' chi_squared_test(efc, "c161sex", by = "e16sex", weights = "weight") #' # Chi-squared-test for given probabilities #' chi_squared_test(efc, "c161sex", probabilities = c(0.3, 0.7)) #' @export diff --git a/R/mann_whitney_test.R b/R/mann_whitney_test.R index f6302a8..0f71292 100644 --- a/R/mann_whitney_test.R +++ b/R/mann_whitney_test.R @@ -6,10 +6,12 @@ #' The function reports p and Z-values as well as effect size r and group-rank-means. #' #' @param data A data frame. -#' @param select The dependent variable (numeric) to be used for the test. -#' @param by The grouping variable (factor) to be used for the test. If `by` is -#' not a factor, it will be coerced to a factor. -#' @param weights An optional weighting variable (numeric) to be used for the test. +#' @param select Name of the dependent variable (as string) to be used for the +#' test. +#' @param by Name of the grouping variable to be used for the test. If `by` is +#' not a factor, it will be coerced to a factor. For `chi_squared_test()`, if +#' `probabilities` is provided, `by` must be `NULL`. +#' @param weights Name of an (optional) weighting variable to be used for the test. #' @param distribution Indicates how the null distribution of the test statistic #' should be computed. May be one of `"exact"`, `"approximate"` or `"asymptotic"` #' (default). See [`coin::wilcox_test()`] for details. @@ -32,7 +34,7 @@ #' @examples #' data(efc) #' # Mann-Whitney-U-Tests for elder's age by elder's sex. -#' mann_whitney_test(efc, "e17age", "e16sex") +#' mann_whitney_test(efc, "e17age", by = "e16sex") #' @export mann_whitney_test <- function(data, select = NULL, diff --git a/R/xtab_statistics.R b/R/xtab_statistics.R index 03a33bb..1f75875 100644 --- a/R/xtab_statistics.R +++ b/R/xtab_statistics.R @@ -29,11 +29,13 @@ #' \code{"kendall"}, \code{"pearson"} or \code{"fisher"}. See 'Details'. #' @param ci.lvl Scalar between 0 and 1. If not \code{NULL}, returns a data #' frame including lower and upper confidence intervals. +#' @param weights Name of variable in `x` that indicated the vector of weights +#' that will be applied to weight all observations. Default is `NULL`, so no +#' weights are used. #' @param ... Other arguments, passed down to the statistic functions #' \code{\link[stats]{chisq.test}}, \code{\link[stats]{fisher.test}} or #' \code{\link[stats]{cor.test}}. #' -#' @inheritParams means_by_group #' @inheritParams bootstrap #' @inheritParams boot_ci #' diff --git a/man/chi_squared_test.Rd b/man/chi_squared_test.Rd index daf4d3b..ace8a26 100644 --- a/man/chi_squared_test.Rd +++ b/man/chi_squared_test.Rd @@ -16,12 +16,21 @@ chi_squared_test( \arguments{ \item{data}{A data frame.} -\item{select}{The dependent variable (numeric) to be used for the test.} +\item{select}{Name of the dependent variable (as string) to be used for the +test.} -\item{by}{The grouping variable (factor) to be used for the test. If \code{by} is -not a factor, it will be coerced to a factor.} +\item{by}{Name of the grouping variable to be used for the test. If \code{by} is +not a factor, it will be coerced to a factor. For \code{chi_squared_test()}, if +\code{probabilities} is provided, \code{by} must be \code{NULL}.} -\item{weights}{An optional weighting variable (numeric) to be used for the test.} +\item{probabilities}{A numeric vector of probabilities for each cell in the +contingency table. The length of the vector must match the number of cells +in the table, i.e. the number of unique levels of the variable specified +in \code{select}. If \code{probabilities} is provided, a chi-squared test for given +probabilities is conducted. Furthermore, if \code{probabilities} is given, \code{by} +must be \code{NULL}. The probabilities must sum to 1.} + +\item{weights}{Name of an (optional) weighting variable to be used for the test.} \item{...}{Additional arguments passed down to \code{\link[=chisq.test]{chisq.test()}}.} } @@ -38,9 +47,9 @@ The function reports p and Z-values as well as effect size r and group-rank-mean data(efc) efc$weight <- abs(rnorm(nrow(efc), 1, 0.3)) # Chi-squared-test -chi_squared_test(efc, "c161sex", "e16sex") +chi_squared_test(efc, "c161sex", by = "e16sex") # weighted Chi-squared-test -chi_squared_test(efc, "c161sex", "e16sex", weights = "weight") +chi_squared_test(efc, "c161sex", by = "e16sex", weights = "weight") # Chi-squared-test for given probabilities chi_squared_test(efc, "c161sex", probabilities = c(0.3, 0.7)) } diff --git a/man/crosstable_statistics.Rd b/man/crosstable_statistics.Rd index 20b2f03..8dd677d 100644 --- a/man/crosstable_statistics.Rd +++ b/man/crosstable_statistics.Rd @@ -81,6 +81,10 @@ will be irgnored.} \item{statistics}{Name of measure of association that should be computed. May be one of \code{"auto"}, \code{"cramer"}, \code{"phi"}, \code{"spearman"}, \code{"kendall"}, \code{"pearson"} or \code{"fisher"}. See 'Details'.} + +\item{weights}{Name of variable in \code{x} that indicated the vector of weights +that will be applied to weight all observations. Default is \code{NULL}, so no +weights are used.} } \value{ For \code{phi()}, the table's Phi value. For \code{cramer()}, the diff --git a/man/mann_whitney_test.Rd b/man/mann_whitney_test.Rd index c72b69f..381512f 100644 --- a/man/mann_whitney_test.Rd +++ b/man/mann_whitney_test.Rd @@ -15,12 +15,14 @@ mann_whitney_test( \arguments{ \item{data}{A data frame.} -\item{select}{The dependent variable (numeric) to be used for the test.} +\item{select}{Name of the dependent variable (as string) to be used for the +test.} -\item{by}{The grouping variable (factor) to be used for the test. If \code{by} is -not a factor, it will be coerced to a factor.} +\item{by}{Name of the grouping variable to be used for the test. If \code{by} is +not a factor, it will be coerced to a factor. For \code{chi_squared_test()}, if +\code{probabilities} is provided, \code{by} must be \code{NULL}.} -\item{weights}{An optional weighting variable (numeric) to be used for the test.} +\item{weights}{Name of an (optional) weighting variable to be used for the test.} \item{distribution}{Indicates how the null distribution of the test statistic should be computed. May be one of \code{"exact"}, \code{"approximate"} or \code{"asymptotic"} @@ -52,5 +54,5 @@ Interpretation of the effect size \strong{r}, as a rule-of-thumb: \examples{ data(efc) # Mann-Whitney-U-Tests for elder's age by elder's sex. -mann_whitney_test(efc, "e17age", "e16sex") +mann_whitney_test(efc, "e17age", by = "e16sex") } diff --git a/tests/testthat/test-grpmean.R b/tests/testthat/test-grpmean.R deleted file mode 100644 index 354ef69..0000000 --- a/tests/testthat/test-grpmean.R +++ /dev/null @@ -1,30 +0,0 @@ -if (require("testthat") && require("sjstats") && require("dplyr")) { - data(efc) - set.seed(123) - efc$weight <- abs(rnorm(n = nrow(efc), mean = 1, sd = .5)) - efc_grouped <- group_by(efc, c172code) - - test_that("means_by_group", { - expect_equal(means_by_group(efc, c12hour, e42dep)$mean, c(9.90909, 17.53778, 34.51961, 75.90132, 42.4384), tolerance = 1e-3) - }) - - test_that("means_by_group, weighting", { - w <- "weight" - expect_equal(means_by_group(efc, c12hour, e42dep, weights = weight)$mean, c(9.43932, 17.28629, 35.16486, 79.23457, 43.0544), tolerance = 1e-3) - expect_equal(means_by_group(efc, c12hour, e42dep, weights = "weight")$mean, c(9.43932, 17.28629, 35.16486, 79.23457, 43.0544), tolerance = 1e-3) - expect_equal(means_by_group(efc, c12hour, e42dep, weights = w)$mean, c(9.43932, 17.28629, 35.16486, 79.23457, 43.0544), tolerance = 1e-3) - }) - - test_that("means_by_group, grouping", { - m <- means_by_group(efc_grouped, c12hour, e42dep) - expect_equal(length(m), 3) - }) - - test_that("means_by_group, grouped weighting", { - w <- "weight" - means_by_group(efc_grouped, c12hour, e42dep, weights = weight) - means_by_group(efc_grouped, c12hour, e42dep, weights = "weight") - means_by_group(efc_grouped, c12hour, e42dep, weights = w) - }) - -} From 7c7d32f1ce8874672fb2a47d36f0833aa1625587 Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 10:13:37 +0200 Subject: [PATCH 16/34] add kw --- NAMESPACE | 1 + R/chi_squared_test.R | 2 +- R/kruskal_wallis_test.R | 276 +++++++++++++++++++++++++++++++++++++ R/mann_whitney_test.R | 11 +- _pkgdown.yml | 3 +- man/chi_squared_test.Rd | 2 +- man/kruskal_wallis_test.Rd | 32 +++++ man/mann_whitney_test.Rd | 2 +- 8 files changed, 316 insertions(+), 13 deletions(-) create mode 100644 R/kruskal_wallis_test.R create mode 100644 man/kruskal_wallis_test.Rd diff --git a/NAMESPACE b/NAMESPACE index 450feeb..b61a035 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -85,6 +85,7 @@ export(gmd) export(icc) export(inequ_trend) export(is_prime) +export(kruskal_wallis_test) export(link_inverse) export(mann_whitney_test) export(mean_n) diff --git a/R/chi_squared_test.R b/R/chi_squared_test.R index 13d5831..6eeb53f 100644 --- a/R/chi_squared_test.R +++ b/R/chi_squared_test.R @@ -14,7 +14,7 @@ #' @param ... Additional arguments passed down to [`chisq.test()`]. #' @inheritParams mann_whitney_test #' -#' @return A data frame. +#' @return A data frame with test results. #' #' @examples #' data(efc) diff --git a/R/kruskal_wallis_test.R b/R/kruskal_wallis_test.R new file mode 100644 index 0000000..7741a2b --- /dev/null +++ b/R/kruskal_wallis_test.R @@ -0,0 +1,276 @@ +#' @title Kruskal-Wallis-Test +#' @name kruskal_wallis_test +#' @description This function performs a Kruskal-Wallis rank sum test, see +#' [`kruskal.test()`] and [`coin::kruskal_test()`]). +#' +#' @inheritParams mann_whitney_test +#' +#' @return A data frame with test results. +#' +#' @examples +#' data(efc) +#' # Mann-Whitney-U-Tests for elder's age by elder's sex. +#' kruskal_wallis_test(efc, "e17age", by = "e16sex") +#' @export +kruskal_wallis_test <- function(data, + select = NULL, + by = NULL, + weights = NULL) { + insight::check_if_installed("datawizard") + + # sanity checks + .sanitize_htest_input(data, select, by, weights) + + # get data + dv <- data[[select]] + grp <- data[[by]] + + # coerce to factor + grp <- datawizard::to_factor(grp) + + # only two groups allowed + if (insight::n_unique(grp) < 2) { + insight::format_error("At least two groups are required, i.e. data must have at least two unique levels in `by` for `kruskal_wallis_test()`.") # nolint + } + if (is.null(weights)) { + .calculate_kw(dv, grp) + } else { + .calculate_weighted_kw(dv, grp, data[[weights]]) + } +} + + +# Kruskal-Wallis-Test -------------------------------------------- + +.calculate_kw <- function(dv, grp) { + # prepare data + wcdat <- data.frame(dv, grp) + # perfom wilcox test + wt <- stats::kruskal.test(dv ~ grp, data = wcdat) + # number of groups + n_groups <- vapply(grp, function(g) sum(grp == g, na.rm = TRUE), numeric(1)) + + out <- data.frame( + data = wt$data.name, + Chi2 = wt$statistic, + df = wt$parameter, + p = as.numeric(wt$p.value), + stringsAsFactors = FALSE + ) + + attr(out, "n_groups") <- n_groups + attr(out, "method") <- "kruskal" + attr(out, "weighted") <- FALSE + class(out) <- c("sj_htest_kw", "data.frame") + + out +} + + +# Weighted Mann-Whitney-Test for two groups ---------------------------------- + +.calculate_weighted_kw <- function(dv, grp, weights) { + # check if pkg survey is available + insight::check_if_installed("survey") + + dat <- stats::na.omit(data.frame(dv, grp, weights)) + colnames(dat) <- c("x", "g", "w") + + design <- survey::svydesign(ids = ~0, data = dat, weights = ~w) + result <- survey::svyranktest(formula = x ~ g, design, test = "KruskalWallis") + + # for rank mean + group_levels <- levels(droplevels(grp)) + # subgroups + dat_gr1 <- dat[dat$g == group_levels[1], ] + dat_gr2 <- dat[dat$g == group_levels[2], ] + dat_gr1$rank_x <- rank(dat_gr1$x) + dat_gr2$rank_x <- rank(dat_gr2$x) + + # rank means + design_mean1 <- survey::svydesign( + ids = ~0, + data = dat_gr1, + weights = ~w + ) + rank_mean_1 <- survey::svymean(~rank_x, design_mean1) + + design_mean2 <- survey::svydesign( + ids = ~0, + data = dat_gr2, + weights = ~w + ) + rank_mean_2 <- survey::svymean(~rank_x, design_mean2) + + # group Ns + n_grp1 <- round(dat_gr1$x * dat_gr1$w) + n_grp2 <- round(dat_gr2$x * dat_gr2$w) + + # statistics and effect sizes + z <- result$statistic + r <- abs(z / sqrt(sum(n_grp1, n_grp2))) + + out <- data.frame( + group1 = group_levels[1], + group2 = group_levels[2], + estimate = result$estimate, + z = z, + r = r, + p = as.numeric(result$p.value) + ) + + attr(out, "rank_means") <- stats::setNames( + c(rank_mean_1, rank_mean_2), + c("Mean Group 1", "Mean Group 2") + ) + attr(out, "n_groups") <- stats::setNames( + c(n_grp1, n_grp2), + c("N Group 1", "N Group 2") + ) + attr(out, "group_labels") <- group_labels + attr(out, "method") <- method + attr(out, "weighted") <- TRUE + class(out) <- c("sj_htest_mwu", "data.frame") + + out +} + + +# helper ---------------------------------------------------------------------- + +.misspelled_string <- function(source, searchterm, default_message = NULL) { + if (is.null(searchterm) || length(searchterm) < 1) { + return(default_message) + } + # used for many matches + more_found <- "" + # init default + msg <- "" + # remove matching strings + same <- intersect(source, searchterm) + searchterm <- setdiff(searchterm, same) + source <- setdiff(source, same) + # guess the misspelled string + possible_strings <- unlist(lapply(searchterm, function(s) { + source[.fuzzy_grep(source, s)] # nolint + }), use.names = FALSE) + if (length(possible_strings)) { + msg <- "Did you mean " + if (length(possible_strings) > 1) { + # make sure we don't print dozens of alternatives for larger data frames + if (length(possible_strings) > 5) { + more_found <- sprintf( + " We even found %i more possible matches, not shown here.", + length(possible_strings) - 5 + ) + possible_strings <- possible_strings[1:5] + } + msg <- paste0(msg, "one of ", toString(paste0("\"", possible_strings, "\""))) + } else { + msg <- paste0(msg, "\"", possible_strings, "\"") + } + msg <- paste0(msg, "?", more_found) + } else { + msg <- default_message + } + # no double white space + insight::trim_ws(msg) +} + + +.fuzzy_grep <- function(x, pattern, precision = NULL) { + if (is.null(precision)) { + precision <- round(nchar(pattern) / 3) + } + if (precision > nchar(pattern)) { + return(NULL) + } + p <- sprintf("(%s){~%i}", pattern, precision) + grep(pattern = p, x = x, ignore.case = FALSE) +} + + +.sanitize_htest_input <- function(data, select, by, weights) { + # check if arguments are NULL + if (is.null(select)) { + insight::format_error("Argument `select` is missing.") + } + if (is.null(by)) { + insight::format_error("Arguments `by` is missing.") + } + + # check if arguments have correct length (length of 1) + if (length(select) != 1 || !is.character(select)) { + insight::format_error("Argument `select` must be the name of a single variable.") + } + if (length(by) != 1 || !is.character(by)) { + insight::format_error("Argument `by` must be the name of a single variable.") + } + if (!is.null(weights) && length(weights) != 1) { + insight::format_error("Argument `weights` must be the name of a single variable.") + } + + # check if "select" is in data + if (!select %in% colnames(data)) { + insight::format_error( + sprintf("Variable '%s' not found in data frame.", select), + .misspelled_string(colnames(data), select, "Maybe misspelled?") + ) + } + # check if "by" is in data + if (!by %in% colnames(data)) { + insight::format_error( + sprintf("Variable '%s' not found in data frame.", by), + .misspelled_string(colnames(data), by, "Maybe misspelled?") + ) + } + # check if "weights" is in data + if (!is.null(weights) && !weights %in% colnames(data)) { + insight::format_error( + sprintf("Weighting variable '%s' not found in data frame.", weights), + .misspelled_string(colnames(data), weights, "Maybe misspelled?") + ) + } +} + + +# methods --------------------------------------------------------------------- + +#' @export +print.sj_htest_mwu <- function(x, ...) { + # fetch attributes + group_labels <- attributes(x)$group_labels + rank_means <- attributes(x)$rank_means + n_groups <- attributes(x)$n_groups + weighted <- attributes(x)$weighted + + if (weighted) { + weight_string <- " (weighted)" + } else { + weight_string <- "" + } + + # same width + group_labels <- format(group_labels) + + # header + insight::print_color(sprintf("# Mann-Whitney-Test%s\n\n", weight_string), "blue") + + # group-1-info + insight::print_color( + sprintf( + " Group 1: %s (n = %i, rank mean = %s)\n", + group_labels[1], n_groups[1], insight::format_value(rank_means[1], protect_integers = TRUE) + ), "cyan" + ) + + # group-2-info + insight::print_color( + sprintf( + " Group 2: %s (n = %i, rank mean = %s)\n", + group_labels[2], n_groups[2], insight::format_value(rank_means[2], protect_integers = TRUE) + ), "cyan" + ) + + cat(sprintf("\n r = %.3f, Z = %.3f, %s\n\n", x$r, x$z, insight::format_p(x$p))) +} diff --git a/R/mann_whitney_test.R b/R/mann_whitney_test.R index 0f71292..2fa3c7e 100644 --- a/R/mann_whitney_test.R +++ b/R/mann_whitney_test.R @@ -16,7 +16,7 @@ #' should be computed. May be one of `"exact"`, `"approximate"` or `"asymptotic"` #' (default). See [`coin::wilcox_test()`] for details. #' -#' @return A data frame. +#' @return A data frame with test results. #' #' @details This function calls [`coin::wilcox_test()`] to extract effect sizes. #' Interpretation of the effect size **r**, as a rule-of-thumb: @@ -138,14 +138,8 @@ mann_whitney_test <- function(data, dat <- stats::na.omit(data.frame(dv, grp, weights)) colnames(dat) <- c("x", "g", "w") - if (insight::n_unique(dat$g) > 2) { - method <- "KruskalWallis" - } else { - method <- "wilcoxon" - } - design <- survey::svydesign(ids = ~0, data = dat, weights = ~w) - result <- survey::svyranktest(formula = x ~ g, design, test = method) + result <- survey::svyranktest(formula = x ~ g, design, test = "wilcoxon") # for rank mean group_levels <- levels(droplevels(grp)) @@ -196,7 +190,6 @@ mann_whitney_test <- function(data, c("N Group 1", "N Group 2") ) attr(out, "group_labels") <- group_labels - attr(out, "method") <- method attr(out, "weighted") <- TRUE class(out) <- c("sj_htest_mwu", "data.frame") diff --git a/_pkgdown.yml b/_pkgdown.yml index 059fc5f..2fb1d7b 100644 --- a/_pkgdown.yml +++ b/_pkgdown.yml @@ -30,8 +30,9 @@ reference: - title: "Summary Statistics and Tests" contents: - gmd - - mann_whitney_test - chi_squared_test + - kruskal_wallis_test + - mann_whitney_test - var_pop - title: "Tools for Regression Models" diff --git a/man/chi_squared_test.Rd b/man/chi_squared_test.Rd index ace8a26..abb066f 100644 --- a/man/chi_squared_test.Rd +++ b/man/chi_squared_test.Rd @@ -35,7 +35,7 @@ must be \code{NULL}. The probabilities must sum to 1.} \item{...}{Additional arguments passed down to \code{\link[=chisq.test]{chisq.test()}}.} } \value{ -A data frame. +A data frame with test results. } \description{ This function performs a Mann-Whitney-Test (or Wilcoxon rank diff --git a/man/kruskal_wallis_test.Rd b/man/kruskal_wallis_test.Rd new file mode 100644 index 0000000..75a760b --- /dev/null +++ b/man/kruskal_wallis_test.Rd @@ -0,0 +1,32 @@ +% Generated by roxygen2: do not edit by hand +% Please edit documentation in R/kruskal_wallis_test.R +\name{kruskal_wallis_test} +\alias{kruskal_wallis_test} +\title{Kruskal-Wallis-Test} +\usage{ +kruskal_wallis_test(data, select = NULL, by = NULL, weights = NULL) +} +\arguments{ +\item{data}{A data frame.} + +\item{select}{Name of the dependent variable (as string) to be used for the +test.} + +\item{by}{Name of the grouping variable to be used for the test. If \code{by} is +not a factor, it will be coerced to a factor. For \code{chi_squared_test()}, if +\code{probabilities} is provided, \code{by} must be \code{NULL}.} + +\item{weights}{Name of an (optional) weighting variable to be used for the test.} +} +\value{ +A data frame with test results. +} +\description{ +This function performs a Kruskal-Wallis rank sum test, see +\code{\link[=kruskal.test]{kruskal.test()}} and \code{\link[coin:LocationTests]{coin::kruskal_test()}}). +} +\examples{ +data(efc) +# Mann-Whitney-U-Tests for elder's age by elder's sex. +kruskal_wallis_test(efc, "e17age", by = "e16sex") +} diff --git a/man/mann_whitney_test.Rd b/man/mann_whitney_test.Rd index 381512f..2256df8 100644 --- a/man/mann_whitney_test.Rd +++ b/man/mann_whitney_test.Rd @@ -29,7 +29,7 @@ should be computed. May be one of \code{"exact"}, \code{"approximate"} or \code{ (default). See \code{\link[coin:LocationTests]{coin::wilcox_test()}} for details.} } \value{ -A data frame. +A data frame with test results. } \description{ This function performs a Mann-Whitney-Test (or Wilcoxon rank From f138c305b01141060152c864cd7849a20ed0aaac Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 10:16:35 +0200 Subject: [PATCH 17/34] fix --- R/kruskal_wallis_test.R | 61 +++++++++-------------------------------- 1 file changed, 13 insertions(+), 48 deletions(-) diff --git a/R/kruskal_wallis_test.R b/R/kruskal_wallis_test.R index 7741a2b..08afb73 100644 --- a/R/kruskal_wallis_test.R +++ b/R/kruskal_wallis_test.R @@ -48,7 +48,7 @@ kruskal_wallis_test <- function(data, # perfom wilcox test wt <- stats::kruskal.test(dv ~ grp, data = wcdat) # number of groups - n_groups <- vapply(grp, function(g) sum(grp == g, na.rm = TRUE), numeric(1)) + n_groups <- vapply(unique(grp), function(g) sum(grp == g, na.rm = TRUE), numeric(1)) out <- data.frame( data = wt$data.name, @@ -79,58 +79,23 @@ kruskal_wallis_test <- function(data, design <- survey::svydesign(ids = ~0, data = dat, weights = ~w) result <- survey::svyranktest(formula = x ~ g, design, test = "KruskalWallis") - # for rank mean - group_levels <- levels(droplevels(grp)) - # subgroups - dat_gr1 <- dat[dat$g == group_levels[1], ] - dat_gr2 <- dat[dat$g == group_levels[2], ] - dat_gr1$rank_x <- rank(dat_gr1$x) - dat_gr2$rank_x <- rank(dat_gr2$x) - - # rank means - design_mean1 <- survey::svydesign( - ids = ~0, - data = dat_gr1, - weights = ~w - ) - rank_mean_1 <- survey::svymean(~rank_x, design_mean1) - - design_mean2 <- survey::svydesign( - ids = ~0, - data = dat_gr2, - weights = ~w - ) - rank_mean_2 <- survey::svymean(~rank_x, design_mean2) - - # group Ns - n_grp1 <- round(dat_gr1$x * dat_gr1$w) - n_grp2 <- round(dat_gr2$x * dat_gr2$w) - - # statistics and effect sizes - z <- result$statistic - r <- abs(z / sqrt(sum(n_grp1, n_grp2))) + # number of groups + n_groups <- vapply(grp, function(g) { + + }, numeric(1)) out <- data.frame( - group1 = group_levels[1], - group2 = group_levels[2], - estimate = result$estimate, - z = z, - r = r, - p = as.numeric(result$p.value) + data = result$data.name, + Chi2 = result$statistic, + df = result$parameter, + p = as.numeric(result$p.value), + stringsAsFactors = FALSE ) - attr(out, "rank_means") <- stats::setNames( - c(rank_mean_1, rank_mean_2), - c("Mean Group 1", "Mean Group 2") - ) - attr(out, "n_groups") <- stats::setNames( - c(n_grp1, n_grp2), - c("N Group 1", "N Group 2") - ) - attr(out, "group_labels") <- group_labels - attr(out, "method") <- method + attr(out, "n_groups") <- n_groups + attr(out, "method") <- "kruskal" attr(out, "weighted") <- TRUE - class(out) <- c("sj_htest_mwu", "data.frame") + class(out) <- c("sj_htest_kw", "data.frame") out } From af7c729a699cc9acecbae21ffba8dfc142b5aa5d Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 10:39:09 +0200 Subject: [PATCH 18/34] fixes --- NAMESPACE | 1 + R/chi_squared_test.R | 4 +- R/helpfunctions.R | 37 ++++++++++- R/kruskal_wallis_test.R | 138 +++++----------------------------------- R/mann_whitney_test.R | 4 +- 5 files changed, 58 insertions(+), 126 deletions(-) diff --git a/NAMESPACE b/NAMESPACE index b61a035..3d00509 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -24,6 +24,7 @@ S3method(print,sj_chi2gof) S3method(print,sj_grpmean) S3method(print,sj_grpmeans) S3method(print,sj_htest_chi) +S3method(print,sj_htest_kw) S3method(print,sj_htest_mwu) S3method(print,sj_mwu) S3method(print,sj_outliers) diff --git a/R/chi_squared_test.R b/R/chi_squared_test.R index 6eeb53f..11c0aef 100644 --- a/R/chi_squared_test.R +++ b/R/chi_squared_test.R @@ -192,9 +192,11 @@ print.sj_htest_chi <- function(x, ...) { weight_string ), "blue") + stat_symbol <- .format_symbols(effect_size_name) + # print test statistic cat(sprintf(" %*s: %.4f\n", l, x$statistic_name, x$statistic)) - cat(sprintf(" %*s: %.4f\n", l, x$effect_size_name, x$effect_size)) + cat(sprintf(" %*s: %.4f\n", l, stat_symbol, x$effect_size)) cat(sprintf(" %*s: %g\n", l, "df", x$df)) cat(sprintf(" %*s: %s\n", l, "p-value", insight::format_p(x$p, stars = TRUE, name = NULL))) cat(sprintf(" %*s: %g\n", l, "Observations", x$n_obs)) diff --git a/R/helpfunctions.R b/R/helpfunctions.R index 0785b28..e1a8b12 100644 --- a/R/helpfunctions.R +++ b/R/helpfunctions.R @@ -64,8 +64,43 @@ get_grouped_data <- function(x) { } - .compact_character <- function(x) { x[!sapply(x, function(i) is.null(i) || nchar(i) == 0 || is.na(i) || any(i == "NULL", na.rm = TRUE))] } + +.format_symbols <- function(x) { + if (.unicode_symbols()) { + x <- gsub("Delta", "Δ", x, ignore.case = TRUE) + x <- gsub("Phi", "ϕ", x, ignore.case = TRUE) + x <- gsub("Eta", "η", x, ignore.case = TRUE) + x <- gsub("Epsilon", "ε", x, ignore.case = TRUE) + x <- gsub("Omega", "ε", x, ignore.case = TRUE) + x <- gsub("R2", "R²", x, ignore.case = TRUE) + x <- gsub("Chi2", "χ²", x, ignore.case = TRUE) + x <- gsub("Chi", "χ", x, ignore.case = TRUE) + x <- gsub("Sigma", "σ", x, ignore.case = TRUE) + x <- gsub("Rho", "ρ", x, ignore.case = TRUE) + x <- gsub("Mu", "μ", x, ignore.case = TRUE) + x <- gsub("Theta", "θ", x, ignore.case = TRUE) + } + x +} + + +.unicode_symbols <- function() { + win_os <- tryCatch( + { + si <- Sys.info() + if (is.null(si["sysname"])) { + FALSE + } else { + si["sysname"] == "Windows" || startsWith(R.version$os, "mingw") + } + }, + error = function(e) { + TRUE + } + ) + l10n_info()[["UTF-8"]] && ((win_os && getRversion() >= "4.2") || (!win_os && getRversion() >= "4.0")) +} diff --git a/R/kruskal_wallis_test.R b/R/kruskal_wallis_test.R index 08afb73..80b6b7b 100644 --- a/R/kruskal_wallis_test.R +++ b/R/kruskal_wallis_test.R @@ -9,8 +9,8 @@ #' #' @examples #' data(efc) -#' # Mann-Whitney-U-Tests for elder's age by elder's sex. -#' kruskal_wallis_test(efc, "e17age", by = "e16sex") +#' # Kruskal-Wallis-Test for elder's age by education +#' kruskal_wallis_test(efc, "e17age", by = "c172code") #' @export kruskal_wallis_test <- function(data, select = NULL, @@ -80,12 +80,12 @@ kruskal_wallis_test <- function(data, result <- survey::svyranktest(formula = x ~ g, design, test = "KruskalWallis") # number of groups - n_groups <- vapply(grp, function(g) { - + n_groups <- vapply(unique(grp), function(g) { + sum(dat$w[dat$grp == g], na.rm = TRUE) }, numeric(1)) out <- data.frame( - data = result$data.name, + data = paste(dv, "by", grp), Chi2 = result$statistic, df = result$parameter, p = as.numeric(result$p.value), @@ -101,111 +101,12 @@ kruskal_wallis_test <- function(data, } -# helper ---------------------------------------------------------------------- - -.misspelled_string <- function(source, searchterm, default_message = NULL) { - if (is.null(searchterm) || length(searchterm) < 1) { - return(default_message) - } - # used for many matches - more_found <- "" - # init default - msg <- "" - # remove matching strings - same <- intersect(source, searchterm) - searchterm <- setdiff(searchterm, same) - source <- setdiff(source, same) - # guess the misspelled string - possible_strings <- unlist(lapply(searchterm, function(s) { - source[.fuzzy_grep(source, s)] # nolint - }), use.names = FALSE) - if (length(possible_strings)) { - msg <- "Did you mean " - if (length(possible_strings) > 1) { - # make sure we don't print dozens of alternatives for larger data frames - if (length(possible_strings) > 5) { - more_found <- sprintf( - " We even found %i more possible matches, not shown here.", - length(possible_strings) - 5 - ) - possible_strings <- possible_strings[1:5] - } - msg <- paste0(msg, "one of ", toString(paste0("\"", possible_strings, "\""))) - } else { - msg <- paste0(msg, "\"", possible_strings, "\"") - } - msg <- paste0(msg, "?", more_found) - } else { - msg <- default_message - } - # no double white space - insight::trim_ws(msg) -} - - -.fuzzy_grep <- function(x, pattern, precision = NULL) { - if (is.null(precision)) { - precision <- round(nchar(pattern) / 3) - } - if (precision > nchar(pattern)) { - return(NULL) - } - p <- sprintf("(%s){~%i}", pattern, precision) - grep(pattern = p, x = x, ignore.case = FALSE) -} - - -.sanitize_htest_input <- function(data, select, by, weights) { - # check if arguments are NULL - if (is.null(select)) { - insight::format_error("Argument `select` is missing.") - } - if (is.null(by)) { - insight::format_error("Arguments `by` is missing.") - } - - # check if arguments have correct length (length of 1) - if (length(select) != 1 || !is.character(select)) { - insight::format_error("Argument `select` must be the name of a single variable.") - } - if (length(by) != 1 || !is.character(by)) { - insight::format_error("Argument `by` must be the name of a single variable.") - } - if (!is.null(weights) && length(weights) != 1) { - insight::format_error("Argument `weights` must be the name of a single variable.") - } - - # check if "select" is in data - if (!select %in% colnames(data)) { - insight::format_error( - sprintf("Variable '%s' not found in data frame.", select), - .misspelled_string(colnames(data), select, "Maybe misspelled?") - ) - } - # check if "by" is in data - if (!by %in% colnames(data)) { - insight::format_error( - sprintf("Variable '%s' not found in data frame.", by), - .misspelled_string(colnames(data), by, "Maybe misspelled?") - ) - } - # check if "weights" is in data - if (!is.null(weights) && !weights %in% colnames(data)) { - insight::format_error( - sprintf("Weighting variable '%s' not found in data frame.", weights), - .misspelled_string(colnames(data), weights, "Maybe misspelled?") - ) - } -} - - # methods --------------------------------------------------------------------- #' @export -print.sj_htest_mwu <- function(x, ...) { +print.sj_htest_kw <- function(x, ...) { + insight::check_if_installed("datawizard") # fetch attributes - group_labels <- attributes(x)$group_labels - rank_means <- attributes(x)$rank_means n_groups <- attributes(x)$n_groups weighted <- attributes(x)$weighted @@ -215,27 +116,20 @@ print.sj_htest_mwu <- function(x, ...) { weight_string <- "" } - # same width - group_labels <- format(group_labels) - # header - insight::print_color(sprintf("# Mann-Whitney-Test%s\n\n", weight_string), "blue") - - # group-1-info - insight::print_color( - sprintf( - " Group 1: %s (n = %i, rank mean = %s)\n", - group_labels[1], n_groups[1], insight::format_value(rank_means[1], protect_integers = TRUE) - ), "cyan" - ) + insight::print_color(sprintf("# Kruskal-Wallis-Test%s\n\n", weight_string), "blue") - # group-2-info + # data info insight::print_color( sprintf( - " Group 2: %s (n = %i, rank mean = %s)\n", - group_labels[2], n_groups[2], insight::format_value(rank_means[2], protect_integers = TRUE) + " Data: %s (%i groups, n = %s)\n", + x$data, length(n_groups), datawizard::text_concatenate(n_groups) ), "cyan" ) - cat(sprintf("\n r = %.3f, Z = %.3f, %s\n\n", x$r, x$z, insight::format_p(x$p))) + stat_symbol <- .format_symbols("Chi2") + cat(sprintf( + "\n %s = %.3f, df = %i, %s\n\n", + stat_symbol, x$Chi2, round(x$df), insight::format_p(x$p) + )) } diff --git a/R/mann_whitney_test.R b/R/mann_whitney_test.R index 2fa3c7e..6dbbc25 100644 --- a/R/mann_whitney_test.R +++ b/R/mann_whitney_test.R @@ -165,8 +165,8 @@ mann_whitney_test <- function(data, rank_mean_2 <- survey::svymean(~rank_x, design_mean2) # group Ns - n_grp1 <- round(dat_gr1$x * dat_gr1$w) - n_grp2 <- round(dat_gr2$x * dat_gr2$w) + n_grp1 <- round(sum(dat_gr1$w)) + n_grp2 <- round(sum(dat_gr2$w)) # statistics and effect sizes z <- result$statistic From 1fa72dc11e81cfdb8645379eba37bbc3995c8e80 Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 10:39:42 +0200 Subject: [PATCH 19/34] fix --- R/chi_squared_test.R | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/R/chi_squared_test.R b/R/chi_squared_test.R index 11c0aef..d7a63d2 100644 --- a/R/chi_squared_test.R +++ b/R/chi_squared_test.R @@ -192,7 +192,7 @@ print.sj_htest_chi <- function(x, ...) { weight_string ), "blue") - stat_symbol <- .format_symbols(effect_size_name) + stat_symbol <- .format_symbols(x$effect_size_name) # print test statistic cat(sprintf(" %*s: %.4f\n", l, x$statistic_name, x$statistic)) From c7d6a80bfaa638be58b2a06a479cd47cbed542ac Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 10:58:53 +0200 Subject: [PATCH 20/34] fixes --- R/chi_squared_test.R | 27 ++++++++++++++++++--------- R/helpfunctions.R | 1 + 2 files changed, 19 insertions(+), 9 deletions(-) diff --git a/R/chi_squared_test.R b/R/chi_squared_test.R index d7a63d2..c4a8851 100644 --- a/R/chi_squared_test.R +++ b/R/chi_squared_test.R @@ -93,6 +93,7 @@ chi_squared_test <- function(data, # return result out <- data.frame( + data = paste(select, "by", by), statistic_name = "Chi-squared", statistic = test_statistic, effect_size_name = names(effect_size), @@ -156,6 +157,11 @@ chi_squared_test <- function(data, # return result out <- data.frame( + data = paste( + select, + "against probabilities", + datawizard::text_concatenate(sprintf("%i%%", round(100 * probabilities))) + ), statistic_name = "Chi-squared", statistic = test_statistic, effect_size_name = "Fei", @@ -183,8 +189,6 @@ print.sj_htest_chi <- function(x, ...) { weight_string <- "" } - # get length of method name, to align output - l <- max(nchar(c(x$statistic_name, x$effect_size_name, "p-value", "Observations"))) # headline insight::print_color(sprintf( "\n# Chi-Squared Test for %s%s\n\n", @@ -192,12 +196,17 @@ print.sj_htest_chi <- function(x, ...) { weight_string ), "blue") - stat_symbol <- .format_symbols(x$effect_size_name) + # data info + if (!is.null(x$data)) { + insight::print_color(sprintf(" Data: %s (n = %i)\n", x$data, round(x$n_obs)), "cyan") + } + + # prepare and align strings + eff_symbol <- .format_symbols(x$effect_size_name) + stat_symbol <- .format_symbols(x$statistic_name) - # print test statistic - cat(sprintf(" %*s: %.4f\n", l, x$statistic_name, x$statistic)) - cat(sprintf(" %*s: %.4f\n", l, stat_symbol, x$effect_size)) - cat(sprintf(" %*s: %g\n", l, "df", x$df)) - cat(sprintf(" %*s: %s\n", l, "p-value", insight::format_p(x$p, stars = TRUE, name = NULL))) - cat(sprintf(" %*s: %g\n", l, "Observations", x$n_obs)) + cat(sprintf( + "\n %s = %.4f, %s = %.4f, df = %i, %s\n\n", + stat_symbol, x$statistic, eff_symbol, x$effect_size, round(x$df), insight::format_p(x$p) + )) } diff --git a/R/helpfunctions.R b/R/helpfunctions.R index e1a8b12..3802e39 100644 --- a/R/helpfunctions.R +++ b/R/helpfunctions.R @@ -78,6 +78,7 @@ get_grouped_data <- function(x) { x <- gsub("Omega", "ε", x, ignore.case = TRUE) x <- gsub("R2", "R²", x, ignore.case = TRUE) x <- gsub("Chi2", "χ²", x, ignore.case = TRUE) + x <- gsub("Chi-squared", "χ²", x, ignore.case = TRUE) x <- gsub("Chi", "χ", x, ignore.case = TRUE) x <- gsub("Sigma", "σ", x, ignore.case = TRUE) x <- gsub("Rho", "ρ", x, ignore.case = TRUE) From 8e726b3f439c1fcbd5d23cc14ee5270c28aede2d Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 10:59:31 +0200 Subject: [PATCH 21/34] fix --- R/chi_squared_test.R | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/R/chi_squared_test.R b/R/chi_squared_test.R index c4a8851..960ba31 100644 --- a/R/chi_squared_test.R +++ b/R/chi_squared_test.R @@ -197,9 +197,7 @@ print.sj_htest_chi <- function(x, ...) { ), "blue") # data info - if (!is.null(x$data)) { - insight::print_color(sprintf(" Data: %s (n = %i)\n", x$data, round(x$n_obs)), "cyan") - } + insight::print_color(sprintf(" Data: %s (n = %i)\n", x$data, round(x$n_obs)), "cyan") # prepare and align strings eff_symbol <- .format_symbols(x$effect_size_name) From 4790eb5fc5f541b8dfc1aab1ca0bfa5e1a0882bf Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 13:10:49 +0200 Subject: [PATCH 22/34] fix --- DESCRIPTION | 1 - R/helpfunctions.R | 26 +++++++++++++------------- 2 files changed, 13 insertions(+), 14 deletions(-) diff --git a/DESCRIPTION b/DESCRIPTION index 0ecf66c..e25170a 100644 --- a/DESCRIPTION +++ b/DESCRIPTION @@ -22,7 +22,6 @@ Imports: datawizard, dplyr, effectsize, - emmeans, insight, lme4, magrittr, diff --git a/R/helpfunctions.R b/R/helpfunctions.R index 3802e39..2db9790 100644 --- a/R/helpfunctions.R +++ b/R/helpfunctions.R @@ -71,19 +71,19 @@ get_grouped_data <- function(x) { .format_symbols <- function(x) { if (.unicode_symbols()) { - x <- gsub("Delta", "Δ", x, ignore.case = TRUE) - x <- gsub("Phi", "ϕ", x, ignore.case = TRUE) - x <- gsub("Eta", "η", x, ignore.case = TRUE) - x <- gsub("Epsilon", "ε", x, ignore.case = TRUE) - x <- gsub("Omega", "ε", x, ignore.case = TRUE) - x <- gsub("R2", "R²", x, ignore.case = TRUE) - x <- gsub("Chi2", "χ²", x, ignore.case = TRUE) - x <- gsub("Chi-squared", "χ²", x, ignore.case = TRUE) - x <- gsub("Chi", "χ", x, ignore.case = TRUE) - x <- gsub("Sigma", "σ", x, ignore.case = TRUE) - x <- gsub("Rho", "ρ", x, ignore.case = TRUE) - x <- gsub("Mu", "μ", x, ignore.case = TRUE) - x <- gsub("Theta", "θ", x, ignore.case = TRUE) + x <- gsub("Delta", "\u0394", x, ignore.case = TRUE) + x <- gsub("Phi", "\u03D5", x, ignore.case = TRUE) + x <- gsub("Eta", "\u03B7", x, ignore.case = TRUE) + x <- gsub("Epsilon", "\u03b5", x, ignore.case = TRUE) + x <- gsub("Omega", "\u03b5", x, ignore.case = TRUE) + x <- gsub("R2", "R\u00b2", x, ignore.case = TRUE) + x <- gsub("Chi2", "\u03C7\u00b2", x, ignore.case = TRUE) + x <- gsub("Chi-squared", "\u03C7\u00b2", x, ignore.case = TRUE) + x <- gsub("Chi", "\u03C7", x, ignore.case = TRUE) + x <- gsub("Sigma", "\u03C3", x, ignore.case = TRUE) + x <- gsub("Rho", "\u03C1", x, ignore.case = TRUE) + x <- gsub("Mu", "\u03BC", x, ignore.case = TRUE) + x <- gsub("Theta", "\u03B8", x, ignore.case = TRUE) } x } From c9730942953e7ae84442fbd317b5a18c451263dc Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 13:34:10 +0200 Subject: [PATCH 23/34] revise --- NAMESPACE | 7 +- R/cramer.R | 41 +- R/xtab_statistics.R | 103 ++- docs/404.html | 156 ---- docs/CODE_OF_CONDUCT.html | 164 ----- docs/CONTRIBUTING.html | 185 ----- docs/apple-touch-icon-120x120.png | Bin 13875 -> 0 bytes docs/apple-touch-icon-60x60.png | Bin 4831 -> 0 bytes docs/apple-touch-icon-76x76.png | Bin 6750 -> 0 bytes docs/apple-touch-icon.png | Bin 13875 -> 0 bytes docs/articles/anova-statistics.html | 237 ------- .../header-attrs-2.1/header-attrs.js | 12 - docs/articles/bayesian-statistics.html | 235 ------ .../figure-html/unnamed-chunk-6-1.png | Bin 26428 -> 0 bytes .../header-attrs-2.1/header-attrs.js | 12 - docs/articles/index.html | 175 ----- docs/articles/mixedmodels-statistics.html | 247 ------- docs/authors.html | 171 ----- docs/bootstrap-toc.css | 60 -- docs/bootstrap-toc.js | 159 ----- docs/docsearch.css | 148 ---- docs/docsearch.js | 85 --- docs/favicon-16x16.png | Bin 1158 -> 0 bytes docs/favicon-32x32.png | Bin 2191 -> 0 bytes docs/favicon.ico | Bin 15086 -> 0 bytes docs/index.html | 197 ----- docs/link.svg | 12 - docs/logo.png | Bin 18507 -> 0 bytes docs/news/index.html | 670 ------------------ docs/pkgdown.css | 367 ---------- docs/pkgdown.js | 108 --- docs/pkgdown.yml | 6 - docs/reference/auto_prior.html | 264 ------- docs/reference/boot_ci.html | 308 -------- docs/reference/bootstrap.html | 304 -------- docs/reference/check_assumptions.html | 431 ----------- docs/reference/chisq_gof.html | 236 ------ docs/reference/cod.html | 347 --------- docs/reference/converge_ok.html | 261 ------- docs/reference/crosstable_statistics.html | 353 --------- docs/reference/cv.html | 190 ----- docs/reference/cv_error.html | 219 ------ docs/reference/deff.html | 214 ------ docs/reference/design_effect.html | 202 ------ docs/reference/dot-badlink.html | 163 ----- docs/reference/dot-collapse_cond.html | 163 ----- docs/reference/dot-get_variance_beta.html | 163 ----- .../dot-get_variance_dispersion.html | 163 ----- docs/reference/dot-get_variance_fixed.html | 163 ----- docs/reference/dot-get_variance_random.html | 163 ----- docs/reference/dot-get_variance_residual.html | 163 ----- docs/reference/efc.html | 164 ----- docs/reference/eta_sq.html | 248 ------- docs/reference/figures/logo.png | Bin 18507 -> 0 bytes docs/reference/find_beta-1.png | Bin 14736 -> 0 bytes docs/reference/find_beta-2.png | Bin 14785 -> 0 bytes docs/reference/find_beta-3.png | Bin 13712 -> 0 bytes docs/reference/find_beta-4.png | Bin 12224 -> 0 bytes docs/reference/find_beta.html | 292 -------- docs/reference/fish.html | 164 ----- docs/reference/gmd.html | 198 ------ docs/reference/grpmean.html | 354 --------- docs/reference/hdi.html | 508 ------------- docs/reference/icc.html | 478 ------------- docs/reference/index.html | 467 ------------ docs/reference/inequ_trend-1.png | Bin 21467 -> 0 bytes docs/reference/inequ_trend-2.png | Bin 22634 -> 0 bytes docs/reference/inequ_trend.html | 275 ------- docs/reference/is_prime.html | 178 ----- docs/reference/mean_n.html | 223 ------ docs/reference/means_by_group.html | 340 --------- docs/reference/mediation.html | 228 ------ docs/reference/mwu.html | 299 -------- docs/reference/nhanes_sample.html | 168 ----- docs/reference/odds_to_rr.html | 256 ------- docs/reference/overdisp-1.png | Bin 8143 -> 0 bytes docs/reference/overdisp.html | 209 ------ docs/reference/p_value.html | 251 ------- docs/reference/pca.html | 242 ------- docs/reference/pred_accuracy.html | 254 ------- docs/reference/pred_vars.html | 417 ----------- docs/reference/prop.html | 294 -------- docs/reference/r2.html | 189 ----- docs/reference/re_var.html | 288 -------- docs/reference/reexports.html | 187 ----- docs/reference/reliab_test.html | 372 ---------- docs/reference/rmse-1.png | Bin 27572 -> 0 bytes docs/reference/rmse-2.png | Bin 25887 -> 0 bytes docs/reference/rmse.html | 416 ----------- docs/reference/robust.html | 351 --------- docs/reference/samplesize_mixed.html | 262 ------- docs/reference/scale_weights.html | 276 -------- docs/reference/se.html | 383 ---------- docs/reference/se_ybar.html | 188 ----- docs/reference/sjstats-package.html | 209 ------ docs/reference/smpsize_lmm.html | 260 ------- docs/reference/std_beta.html | 272 ------- docs/reference/svyglm.nb.html | 275 ------- docs/reference/svyglm.zip.html | 232 ------ docs/reference/table_values.html | 199 ------ docs/reference/tidy_stan.html | 265 ------- docs/reference/typical_value.html | 270 ------- docs/reference/var_pop.html | 190 ----- docs/reference/weight.html | 222 ------ docs/reference/weighted_sd.html | 402 ----------- docs/reference/wtd_sd.html | 421 ----------- docs/reference/xtab_statistics.html | 373 ---------- man/crosstable_statistics.Rd | 89 ++- man/kruskal_wallis_test.Rd | 4 +- pkgdown/extra.scss | 22 + 110 files changed, 141 insertions(+), 21010 deletions(-) delete mode 100644 docs/404.html delete mode 100644 docs/CODE_OF_CONDUCT.html delete mode 100644 docs/CONTRIBUTING.html delete mode 100644 docs/apple-touch-icon-120x120.png delete mode 100644 docs/apple-touch-icon-60x60.png delete mode 100644 docs/apple-touch-icon-76x76.png delete mode 100644 docs/apple-touch-icon.png delete mode 100644 docs/articles/anova-statistics.html delete mode 100644 docs/articles/anova-statistics_files/header-attrs-2.1/header-attrs.js delete mode 100644 docs/articles/bayesian-statistics.html delete mode 100644 docs/articles/bayesian-statistics_files/figure-html/unnamed-chunk-6-1.png delete mode 100644 docs/articles/bayesian-statistics_files/header-attrs-2.1/header-attrs.js delete mode 100644 docs/articles/index.html delete mode 100644 docs/articles/mixedmodels-statistics.html delete mode 100644 docs/authors.html delete mode 100644 docs/bootstrap-toc.css delete mode 100644 docs/bootstrap-toc.js delete mode 100644 docs/docsearch.css delete mode 100644 docs/docsearch.js delete mode 100644 docs/favicon-16x16.png delete mode 100644 docs/favicon-32x32.png delete mode 100644 docs/favicon.ico delete mode 100644 docs/index.html delete mode 100644 docs/link.svg delete mode 100644 docs/logo.png delete mode 100644 docs/news/index.html delete mode 100644 docs/pkgdown.css delete mode 100644 docs/pkgdown.js delete mode 100644 docs/pkgdown.yml delete mode 100644 docs/reference/auto_prior.html delete mode 100644 docs/reference/boot_ci.html delete mode 100644 docs/reference/bootstrap.html delete mode 100644 docs/reference/check_assumptions.html delete mode 100644 docs/reference/chisq_gof.html delete mode 100644 docs/reference/cod.html delete mode 100644 docs/reference/converge_ok.html delete mode 100644 docs/reference/crosstable_statistics.html delete mode 100644 docs/reference/cv.html delete mode 100644 docs/reference/cv_error.html delete mode 100644 docs/reference/deff.html delete mode 100644 docs/reference/design_effect.html delete mode 100644 docs/reference/dot-badlink.html delete mode 100644 docs/reference/dot-collapse_cond.html delete mode 100644 docs/reference/dot-get_variance_beta.html delete mode 100644 docs/reference/dot-get_variance_dispersion.html delete mode 100644 docs/reference/dot-get_variance_fixed.html delete mode 100644 docs/reference/dot-get_variance_random.html delete mode 100644 docs/reference/dot-get_variance_residual.html delete mode 100644 docs/reference/efc.html delete mode 100644 docs/reference/eta_sq.html delete mode 100644 docs/reference/figures/logo.png delete mode 100644 docs/reference/find_beta-1.png delete mode 100644 docs/reference/find_beta-2.png delete mode 100644 docs/reference/find_beta-3.png delete mode 100644 docs/reference/find_beta-4.png delete mode 100644 docs/reference/find_beta.html delete mode 100644 docs/reference/fish.html delete mode 100644 docs/reference/gmd.html delete mode 100644 docs/reference/grpmean.html delete mode 100644 docs/reference/hdi.html delete mode 100644 docs/reference/icc.html delete mode 100644 docs/reference/index.html delete mode 100644 docs/reference/inequ_trend-1.png delete mode 100644 docs/reference/inequ_trend-2.png delete mode 100644 docs/reference/inequ_trend.html delete mode 100644 docs/reference/is_prime.html delete mode 100644 docs/reference/mean_n.html delete mode 100644 docs/reference/means_by_group.html delete mode 100644 docs/reference/mediation.html delete mode 100644 docs/reference/mwu.html delete mode 100644 docs/reference/nhanes_sample.html delete mode 100644 docs/reference/odds_to_rr.html delete mode 100644 docs/reference/overdisp-1.png delete mode 100644 docs/reference/overdisp.html delete mode 100644 docs/reference/p_value.html delete mode 100644 docs/reference/pca.html delete mode 100644 docs/reference/pred_accuracy.html delete mode 100644 docs/reference/pred_vars.html delete mode 100644 docs/reference/prop.html delete mode 100644 docs/reference/r2.html delete mode 100644 docs/reference/re_var.html delete mode 100644 docs/reference/reexports.html delete mode 100644 docs/reference/reliab_test.html delete mode 100644 docs/reference/rmse-1.png delete mode 100644 docs/reference/rmse-2.png delete mode 100644 docs/reference/rmse.html delete mode 100644 docs/reference/robust.html delete mode 100644 docs/reference/samplesize_mixed.html delete mode 100644 docs/reference/scale_weights.html delete mode 100644 docs/reference/se.html delete mode 100644 docs/reference/se_ybar.html delete mode 100644 docs/reference/sjstats-package.html delete mode 100644 docs/reference/smpsize_lmm.html delete mode 100644 docs/reference/std_beta.html delete mode 100644 docs/reference/svyglm.nb.html delete mode 100644 docs/reference/svyglm.zip.html delete mode 100644 docs/reference/table_values.html delete mode 100644 docs/reference/tidy_stan.html delete mode 100644 docs/reference/typical_value.html delete mode 100644 docs/reference/var_pop.html delete mode 100644 docs/reference/weight.html delete mode 100644 docs/reference/weighted_sd.html delete mode 100644 docs/reference/wtd_sd.html delete mode 100644 docs/reference/xtab_statistics.html create mode 100644 pkgdown/extra.scss diff --git a/NAMESPACE b/NAMESPACE index 3d00509..e351d2c 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -3,9 +3,9 @@ S3method(AIC,svyglm.nb) S3method(as.data.frame,sj_resample) S3method(as.integer,sj_resample) -S3method(cramer,formula) -S3method(cramer,ftable) -S3method(cramer,table) +S3method(cramers_v,formula) +S3method(cramers_v,ftable) +S3method(cramers_v,table) S3method(deviance,svyglm.nb) S3method(family,svyglm.nb) S3method(formula,svyglm.nb) @@ -70,6 +70,7 @@ export(chisq_gof) export(ci) export(cohens_f) export(cramer) +export(cramers_v) export(crosstable_statistics) export(cv) export(cv_compare) diff --git a/R/cramer.R b/R/cramer.R index 66798e7..a623da4 100644 --- a/R/cramer.R +++ b/R/cramer.R @@ -1,53 +1,52 @@ #' @rdname crosstable_statistics #' @export -cramer <- function(tab, ...) { - UseMethod("cramer") +cramers_v <- function(tab, ...) { + UseMethod("cramers_v") } +#' @rdname crosstable_statistics +#' @export +cramer <- cramers_v #' @export -cramer.table <- function(tab, ...) { - .cramer(tab) +cramers_v.table <- function(tab, ...) { + .cramers_v(tab) } #' @export -cramer.ftable <- function(tab, ...) { - .cramer(tab) +cramers_v.ftable <- function(tab, ...) { + .cramers_v(tab) } #' @rdname crosstable_statistics #' @export -cramer.formula <- function(formula, data, ci.lvl = NULL, n = 1000, method = c("dist", "quantile"), ...) { +cramers_v.formula <- function(formula, data, ci.lvl = NULL, n = 1000, method = c("dist", "quantile"), ...) { terms <- all.vars(formula) tab <- table(data[[terms[1]]], data[[terms[2]]]) method <- match.arg(method) if (is.null(ci.lvl) || is.na(ci.lvl)) { - .cramer(tab) + .cramers_v(tab) } else { - ci <- data[, terms] %>% - sjstats::bootstrap(n) %>% - dplyr::mutate( - tables = lapply(.data$strap, function(x) { - dat <- as.data.frame(x) - table(dat[[1]], dat[[2]]) - }), - cramers = sapply(.data$tables, function(x) .cramer(x)) - ) %>% - dplyr::pull("cramers") %>% - boot_ci(ci.lvl = ci.lvl, method = method) + straps <- sjstats::bootstrap(data[terms], n) + tables <- lapply(straps$strap, function(x) { + dat <- as.data.frame(x) + table(dat[[1]], dat[[2]]) + }) + cramers <- sapply(tables, function(x) .cramers_v(x)) + ci <- boot_ci(cramers, ci.lvl = ci.lvl, method = method) data_frame( - cramer = .cramer(tab), + cramer = .cramers_v(tab), conf.low = ci$conf.low, conf.high = ci$conf.high ) } } -.cramer <- function(tab) { +.cramers_v <- function(tab) { # convert to flat table if (!inherits(tab, "ftable")) tab <- stats::ftable(tab) sqrt(phi(tab)^2 / min(dim(tab) - 1)) diff --git a/R/xtab_statistics.R b/R/xtab_statistics.R index 1f75875..2e9c619 100644 --- a/R/xtab_statistics.R +++ b/R/xtab_statistics.R @@ -6,70 +6,65 @@ #' Supported measures are Cramer's V, Phi, Spearman's rho, #' Kendall's tau and Pearson's r. #' -#' @param data A data frame or a table object. If a table object, \code{x1} and -#' \code{x2} will be ignored. For Kendall's \emph{tau}, Spearman's \emph{rho} -#' or Pearson's product moment correlation coefficient, \code{data} needs -#' to be a data frame. If \code{x1} and \code{x2} are not specified, -#' the first two columns of the data frames are used as variables -#' to compute the crosstab. -#' @param formula A formula of the form \code{lhs ~ rhs} where \code{lhs} is a -#' numeric variable giving the data values and \code{rhs} a factor giving the -#' corresponding groups. -#' @param tab A \code{\link{table}} or \code{\link[stats]{ftable}}. Tables of class -#' \code{\link[stats]{xtabs}} and other will be coerced to \code{ftable} -#' objects. +#' @param data A data frame or a table object. If a table object, `x1` and +#' `x2` will be ignored. For Kendall's _tau_, Spearman's _rho_ or Pearson's +#' product moment correlation coefficient, `data` needs to be a data frame. +#' If `x1` and `x2` are not specified, the first two columns of the data +#' frames are used as variables to compute the crosstab. +#' @param formula A formula of the form `lhs ~ rhs` where `lhs` is a +#' numeric variable giving the data values and `rhs` a factor giving the +#' corresponding groups. +#' @param tab A [`table()`] or [`ftable()`]. Tables of class [`xtabs()`] and +#' other will be coerced to `ftable` objects. #' @param x1 Name of first variable that should be used to compute the -#' contingency table. If \code{data} is a table object, this argument -#' will be irgnored. +#' contingency table. If `data` is a table object, this argument will be +#' irgnored. #' @param x2 Name of second variable that should be used to compute the -#' contingency table. If \code{data} is a table object, this argument -#' will be irgnored. +#' contingency table. If `data` is a table object, this argument will be +#' irgnored. #' @param statistics Name of measure of association that should be computed. May -#' be one of \code{"auto"}, \code{"cramer"}, \code{"phi"}, \code{"spearman"}, -#' \code{"kendall"}, \code{"pearson"} or \code{"fisher"}. See 'Details'. -#' @param ci.lvl Scalar between 0 and 1. If not \code{NULL}, returns a data -#' frame including lower and upper confidence intervals. +#' be one of `"auto"`, `"cramer"`, `"phi"`, `"spearman"`, `"kendall"`, +#' `"pearson"` or `"fisher"`. See 'Details'. +#' @param ci.lvl Scalar between 0 and 1. If not `NULL`, returns a data +#' frame including lower and upper confidence intervals. #' @param weights Name of variable in `x` that indicated the vector of weights #' that will be applied to weight all observations. Default is `NULL`, so no #' weights are used. #' @param ... Other arguments, passed down to the statistic functions -#' \code{\link[stats]{chisq.test}}, \code{\link[stats]{fisher.test}} or -#' \code{\link[stats]{cor.test}}. +#' [`chisq.test()`], [`fisher.test()`] or [`cor.test()`]. #' #' @inheritParams bootstrap #' @inheritParams boot_ci #' -#' @return For \code{phi()}, the table's Phi value. For \code{cramer()}, the -#' table's Cramer's V. -#' \cr \cr -#' For \code{crosstable_statistics()}, a list with following components: -#' \describe{ -#' \item{\code{estimate}}{the value of the estimated measure of association.} -#' \item{\code{p.value}}{the p-value for the test.} -#' \item{\code{statistic}}{the value of the test statistic.} -#' \item{\code{stat.name}}{the name of the test statistic.} -#' \item{\code{stat.html}}{if applicable, the name of the test statistic, in HTML-format.} -#' \item{\code{df}}{the degrees of freedom for the contingency table.} -#' \item{\code{method}}{character string indicating the name of the measure of association.} -#' \item{\code{method.html}}{if applicable, the name of the measure of association, in HTML-format.} -#' \item{\code{method.short}}{the short form of association measure, equals the \code{statistics}-argument.} -#' \item{\code{fisher}}{logical, if Fisher's exact test was used to calculate the p-value.} -#' } +#' @return For [`phi()`], the table's Phi value. For [`cramers_v()]`, the +#' table's Cramer's V. +#' +#' For `crosstable_statistics()`, a list with following components: +#' +#' - `estimate`: the value of the estimated measure of association. +#' - `p.value`: the p-value for the test. +#' - `statistic`: the value of the test statistic. +#' - `stat.name`: the name of the test statistic. +#' - `stat.html`: if applicable, the name of the test statistic, in HTML-format. +#' - `df`: the degrees of freedom for the contingency table. +#' - `method`: character string indicating the name of the measure of association. +#' - `method.html`: if applicable, the name of the measure of association, in HTML-format. +#' - `method.short`: the short form of association measure, equals the `statistics`-argument. +#' - `fisher`: logical, if Fisher's exact test was used to calculate the p-value. #' #' @details The p-value for Cramer's V and the Phi coefficient are based -#' on \code{chisq.test()}. If any expected value of a table cell is -#' smaller than 5, or smaller than 10 and the df is 1, then \code{fisher.test()} -#' is used to compute the p-value, unless \code{statistics = "fisher"}; in -#' this case, the use of \code{fisher.test()} is forced to compute the -#' p-value. The test statistic is calculated with \code{cramer()} resp. -#' \code{phi()}. -#' \cr \cr -#' Both test statistic and p-value for Spearman's rho, Kendall's tau -#' and Pearson's r are calculated with \code{cor.test()}. -#' \cr \cr -#' When \code{statistics = "auto"}, only Cramer's V or Phi are calculated, -#' based on the dimension of the table (i.e. if the table has more than -#' two rows or columns, Cramer's V is calculated, else Phi). +#' on `chisq.test()`. If any expected value of a table cell is smaller than 5, +#' or smaller than 10 and the df is 1, then `fisher.test()` is used to compute +#' the p-value, unless `statistics = "fisher"`; in this case, the use of +#' `fisher.test()` is forced to compute the p-value. The test statistic is +#' calculated with `cramers_v()` resp. `phi()`. +#' +#' Both test statistic and p-value for Spearman's rho, Kendall's tau and +#' Pearson's r are calculated with `cor.test()`. +#' +#' When `statistics = "auto"`, only Cramer's V or Phi are calculated, based on +#' the dimension of the table (i.e. if the table has more than two rows or +#' columns, Cramer's V is calculated, else Phi). #' #' @examples #' # Phi coefficient for 2x2 tables @@ -120,9 +115,9 @@ crosstable_statistics <- function(data, x1 = NULL, x2 = NULL, statistics = c("au weights <- deparse(substitute(weights)) # if names were quotes, remove quotes - x1 <- gsub("\"", "", x1, fixed = T) - x2 <- gsub("\"", "", x2, fixed = T) - weights <- gsub("\"", "", weights, fixed = T) + x1 <- gsub("\"", "", x1, fixed = TRUE) + x2 <- gsub("\"", "", x2, fixed = TRUE) + weights <- gsub("\"", "", weights, fixed = TRUE) if (sjmisc::is_empty(weights) || weights == "NULL") weights <- NULL diff --git a/docs/404.html b/docs/404.html deleted file mode 100644 index 2c858e8..0000000 --- a/docs/404.html +++ /dev/null @@ -1,156 +0,0 @@ - - - - - - - - -Page not found (404) • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
-
- - - - -
- -
-
- - -Content not found. Please use links in the navbar. - -
- - - -
- - - - -
- - - - - - - - diff --git a/docs/CODE_OF_CONDUCT.html b/docs/CODE_OF_CONDUCT.html deleted file mode 100644 index 26db543..0000000 --- a/docs/CODE_OF_CONDUCT.html +++ /dev/null @@ -1,164 +0,0 @@ - - - - - - - - -Contributor Code of Conduct • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
-
- - - - -
- -
-
- - -
- -

As contributors and maintainers of this project, we pledge to respect all people who contribute through reporting issues, posting feature requests, updating documentation, submitting pull requests or patches, and other activities.

-

We are committed to making participation in this project a harassment-free experience for everyone, regardless of level of experience, gender, gender identity and expression, sexual orientation, disability, personal appearance, body size, race, ethnicity, age, or religion.

-

Examples of unacceptable behavior by participants include the use of sexual language or imagery, derogatory comments or personal attacks, trolling, public or private harassment, insults, or other unprofessional conduct.

-

Project maintainers have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct. Project maintainers who do not follow the Code of Conduct may be removed from the project team.

-

Instances of abusive, harassing, or otherwise unacceptable behavior may be reported by opening an issue or contacting one or more of the project maintainers.

-

This Code of Conduct is adapted from the Contributor Covenant (http://contributor-covenant.org), version 1.0.0, available at http://contributor-covenant.org/version/1/0/0/

-
- -
- - - -
- - - - -
- - - - - - - - diff --git a/docs/CONTRIBUTING.html b/docs/CONTRIBUTING.html deleted file mode 100644 index 8c579ed..0000000 --- a/docs/CONTRIBUTING.html +++ /dev/null @@ -1,185 +0,0 @@ - - - - - - - - -Contributing to sjmisc • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
-
- - - - -
- -
-
- - -
- -

This outlines how to propose a change to sjmisc.

-
-

-Fixing typos

-

Small typos or grammatical errors in documentation may be edited directly using the GitHub web interface, so long as the changes are made in the source file. If you want to fix typos in the documentation, please edit the related .R file in the R/ folder. Do not edit an .Rd file in man/.

-
-
-

-Filing an issue

-

The easiest way to propose a change or new feature is to file an issue. If you’ve found a bug, you may also create an associated issue. If possible, try to illustrate your proposal or the bug with a minimal reproducible example.

-
-
-

-Pull request

-
    -
  • Please create a Git branch for each pull request (PR).
  • -
  • Your contributed code should roughly follow the tidyverse style guide. Exceptions from this guide: if separated, use underscores for function names, but dots for argument names. See as example set_na().
  • -
  • sjmisc uses roxygen2, with Markdown syntax, for documentation.
  • -
  • sjmisc uses testthat. Adding tests to the PR makes it easier for me to merge your PR into the code base.
  • -
  • If your PR is a user-visible change, you may add a bullet to the top of NEWS.md describing the changes made. You may optionally add your GitHub username, and links to relevant issue(s)/PR(s).
  • -
-
-
-

-Code of Conduct

-

Please note that this project is released with a Contributor Code of Conduct. By participating in this project you agree to abide by its terms.

-
-
- -
- - - -
- - - - -
- - - - - - - - diff --git a/docs/apple-touch-icon-120x120.png b/docs/apple-touch-icon-120x120.png deleted file mode 100644 index 61543c1cb9980ebbe1272522fe85b2a7c8e44ca7..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 13875 zcmZ{LWmFtZ(>6DRK+xbG+}$05I|NyrU;%==TOc@#dvLb}g1ZG+oW=oub~15=R*ev_bUVr?(x0o*FGGaI|m%xp$QzEP&ynOfm2qSy2yLQ zM>8ckX}GulU3s0QN$({{ZVD;>Y=pE5Bhwuj=EZ|h!P>a!|DE5D-hK@ zUNcssQge{Fb(~(kBvRpcziSKkIU)K*AND~U$MHh^bue!KSb3CJ=l1VKLZ3AAC7naa z6%e*&5#!&?Y1tSID5au`$~$z^Aq~%UW_vA2+k#ukoSNgmRx-GwBKsn%$=LF8J8CSF zc0&R(u7R6g#~+^R6P~QxYZ=*>ZJMn!3Xr&akT8+hHjE<&yI|!8q zxV!CRY}~zZ$y>aV3>jp(5)sH@0Ek!@##8edT^n?HQGSvNx*ryQ^(#BD~cL#_bFl?W1j>nUyb)AAj5D+miRn$55Y&bjm zt~4B_bic?Y0NhcSc4cWG%E?P2+Gb_*MfAJ%CG-%0sQ2==Rgn^eKSmZgm;aB!s9VT<7W z)q2JGkwknK9a<)lOm-Rf&CqcE0g8ca^f4nZhiWanDkZk5v)I6L7rc^?>yBR~6SzF; zlZFJf9bXQhG*^2L3DhRVyizl3W{-d~GxIp#9-zLm!cI;-*1m}2uaT&Y`m>4)2w_J-1mO61N2RBVot&~>G7N#d-0kLrZCU#s0 zX=L}>BvqDHfKF|byibUZhziGL*ZAk#!KfW?%r<=AYtjQ(`RbqA3$l|kGJrtkxT`dA zFIg*?M|lmF@B93@F#L@be}&5Fy1HwQh^9-OwAZQ|%wLTWn?H;5%m+;RypC=QC|i>< znv~e6EQyGWo6D6fYJyC0r@>BKv>&L-{w3)D?i(O(w3MD`PQ0nYa-R@{89ixO`aHpkUi=B=5<`A;CRE}aWs7PAQd#MW#7l=x># z2%tyoH);D_=FwN`Pu19%I;Z2s?@h7_A5cpPBvGB7wjH)I?P8yHHHv!O;>6aLOT z+Wji}v3Xs&ReusQ@IDdypg4ci{b>F__n=H^hc@jEp|irrKG8EpYL`;j(;1i9nbTq0 zoM1SU^Y-n~`*gRLR3!gHR{K!X7(4+k&%l!gDV8bD*BeeHSg1G#wb~Qi0L!9!BRI{J zvF79o{yq?)VmeqG7!iA2#!Fv#1d%fms;nPuLkqHdkLE@Dr7&;KcLN`})>TkdV9Y`? z7(W`?*RDR6E>zdfzb3#Q0vXI4qo54Yt`!wrnwaDBTSVSFL%6=tyal=nM8B~FnFqwk zOK$Q*REetj1iw|x*vTf|i7NP($OKy}_8Xd3`2&D#XSNt|rEN^qjl(|24x}awXyf

7;LN?SI$=Pfe85jvA?UcyC(OHjCsWJ< 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 rQH(Cm4)u@qllCKx{*upAWP>{-z;e5k4XTuJNv# diff --git a/docs/apple-touch-icon-60x60.png b/docs/apple-touch-icon-60x60.png deleted file mode 100644 index bbfc93368048e757dae4f8e96f2b269e614f8e47..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4831 zcmV<55+Ln~P)004R>004l5008;`004mK004C`008P>0026e000+ooVrmw00006 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 diff --git a/docs/apple-touch-icon.png b/docs/apple-touch-icon.png deleted file mode 100644 index 61543c1cb9980ebbe1272522fe85b2a7c8e44ca7..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 13875 zcmZ{LWmFtZ(>6DRK+xbG+}$05I|NyrU;%==TOc@#dvLb}g1ZG+oW=oub~15=R*ev_bUVr?(x0o*FGGaI|m%xp$QzEP&ynOfm2qSy2yLQ zM>8ckX}GulU3s0QN$({{ZVD;>Y=pE5Bhwuj=EZ|h!P>a!|DE5D-hK@ zUNcssQge{Fb(~(kBvRpcziSKkIU)K*AND~U$MHh^bue!KSb3CJ=l1VKLZ3AAC7naa z6%e*&5#!&?Y1tSID5au`$~$z^Aq~%UW_vA2+k#ukoSNgmRx-GwBKsn%$=LF8J8CSF zc0&R(u7R6g#~+^R6P~QxYZ=*>ZJMn!3Xr&akT8+hHjE<&yI|!8q zxV!CRY}~zZ$y>aV3>jp(5)sH@0Ek!@##8edT^n?HQGSvNx*ryQ^(#BD~cL#_bFl?W1j>nUyb)AAj5D+miRn$55Y&bjm zt~4B_bic?Y0NhcSc4cWG%E?P2+Gb_*MfAJ%CG-%0sQ2==Rgn^eKSmZgm;aB!s9VT<7W z)q2JGkwknK9a<)lOm-Rf&CqcE0g8ca^f4nZhiWanDkZk5v)I6L7rc^?>yBR~6SzF; zlZFJf9bXQhG*^2L3DhRVyizl3W{-d~GxIp#9-zLm!cI;-*1m}2uaT&Y`m>4)2w_J-1mO61N2RBVot&~>G7N#d-0kLrZCU#s0 zX=L}>BvqDHfKF|byibUZhziGL*ZAk#!KfW?%r<=AYtjQ(`RbqA3$l|kGJrtkxT`dA zFIg*?M|lmF@B93@F#L@be}&5Fy1HwQh^9-OwAZQ|%wLTWn?H;5%m+;RypC=QC|i>< znv~e6EQyGWo6D6fYJyC0r@>BKv>&L-{w3)D?i(O(w3MD`PQ0nYa-R@{89ixO`aHpkUi=B=5<`A;CRE}aWs7PAQd#MW#7l=x># z2%tyoH);D_=FwN`Pu19%I;Z2s?@h7_A5cpPBvGB7wjH)I?P8yHHHv!O;>6aLOT z+Wji}v3Xs&ReusQ@IDdypg4ci{b>F__n=H^hc@jEp|irrKG8EpYL`;j(;1i9nbTq0 zoM1SU^Y-n~`*gRLR3!gHR{K!X7(4+k&%l!gDV8bD*BeeHSg1G#wb~Qi0L!9!BRI{J zvF79o{yq?)VmeqG7!iA2#!Fv#1d%fms;nPuLkqHdkLE@Dr7&;KcLN`})>TkdV9Y`? z7(W`?*RDR6E>zdfzb3#Q0vXI4qo54Yt`!wrnwaDBTSVSFL%6=tyal=nM8B~FnFqwk zOK$Q*REetj1iw|x*vTf|i7NP($OKy}_8Xd3`2&D#XSNt|rEN^qjl(|24x}awXyf

7;LN?SI$=Pfe85jvA?UcyC(OHjCsWJ< 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 rQH(Cm4)u@qllCKx{*upAWP>{-z;e5k4XTuJNv# diff --git a/docs/articles/anova-statistics.html b/docs/articles/anova-statistics.html deleted file mode 100644 index 41980f9..0000000 --- a/docs/articles/anova-statistics.html +++ /dev/null @@ -1,237 +0,0 @@ - - - - - - - -Statistics for Anova Tables • sjstats - - - - - - - - - - - - - - - - - -

-
- - - - -
-
- - - - -
-

-Effect Size Statistics for Anova Tables

-

This vignettes demontrates those functions of the sjstats-package that deal with Anova tables. These functions report different effect size measures, which are useful beyond significance tests (p-values), because they estimate the magnitude of effects, independent from sample size. sjstats provides following functions:

- -

Befor we start, we fit a simple model:

-
library(sjstats)
-# load sample data
-data(efc)
-
-# fit linear model
-fit <- aov(
-  c12hour ~ as.factor(e42dep) + as.factor(c172code) + c160age,
-  data = efc
-)
-

All functions accept objects of class aov or anova, so you can also use model fits from the car package, which allows fitting Anova’s with different types of sum of squares. Other objects, like lm, will be coerced to anova internally.

-

The following functions return the effect size statistic as named numeric vector, using the model’s term names.

-
-

-Eta-Squared

-

The eta-squared is the proportion of the total variability in the dependent variable that is accounted for by the variation in the independent variable. It is the ratio of the sum of squares for each group level to the total sum of squares. It can be interpreted as percentage of variance accounted for by a variable.

-

For variables with 1 degree of freedom (in the numerator), the square root of eta-squared is equal to the correlation coefficient r. For variables with more than 1 degree of freedom, eta-squared equals R2. This makes eta-squared easily interpretable. Furthermore, these effect sizes can easily be converted into effect size measures that can be, for instance, further processed in meta-analyses.

-

Eta-squared can be computed simply with:

-
eta_sq(fit, ci.lvl = .95)
-#>                  term etasq conf.low conf.high
-#> 1   as.factor(e42dep) 0.266    0.217     0.312
-#> 2 as.factor(c172code) 0.005    0.000     0.018
-#> 3             c160age 0.048    0.024     0.080
-
-
-

-Partial Eta-Squared

-

The partial eta-squared value is the ratio of the sum of squares for each group level to the sum of squares for each group level plus the residual sum of squares. It is more difficult to interpret, because its value strongly depends on the variability of the residuals. Partial eta-squared values should be reported with caution, and Levine and Hullett (2002) recommend reporting eta- or omega-squared rather than partial eta-squared.

-

Use the partial-argument to compute partial eta-squared values:

-
eta_sq(fit, partial = TRUE, ci.lvl = .95)
-#>                  term partial.etasq conf.low conf.high
-#> 1   as.factor(e42dep)         0.281    0.232     0.327
-#> 2 as.factor(c172code)         0.008    0.000     0.023
-#> 3             c160age         0.066    0.038     0.101
-
-
-

-Omega-Squared

-

While eta-squared estimates tend to be biased in certain situations, e.g. when the sample size is small or the independent variables have many group levels, omega-squared estimates are corrected for this bias.

-

Omega-squared can be simply computed with:

-
omega_sq(fit, ci.lvl = .95)
-#>                  term omegasq conf.low conf.high
-#> 1   as.factor(e42dep)   0.263    0.214     0.310
-#> 2 as.factor(c172code)   0.004   -0.002     0.016
-#> 3             c160age   0.048    0.023     0.078
-
-
-

-Partial Omega-Squared

-

omega_sq() also has a partial-argument to compute partial omega-squared values. Computing the partial omega-squared statistics is based on bootstrapping. In this case, use n to define the number of samples (1000 by default.)

-
omega_sq(fit, partial = TRUE, ci.lvl = .95)
-#>                  term partial.omegasq conf.low conf.high
-#> 1   as.factor(e42dep)           0.278    0.229     0.325
-#> 2 as.factor(c172code)           0.005   -0.002     0.020
-#> 3             c160age           0.065    0.036     0.100
-
-
-
-

-Epsilon Squared

-

Espilon-squared is a less common measure of effect size. It is sometimes considered as an “adjusted r-squared” value. You can compute this effect size using epsilon_sq().

-
epsilon_sq(fit, ci.lvl = .95)
-#>                  term epsilonsq conf.low conf.high
-#> 1   as.factor(e42dep)     0.264    0.214     0.310
-#> 2 as.factor(c172code)     0.004   -0.002     0.016
-#> 3             c160age     0.048    0.023     0.079
-
-
-

-Complete Statistical Table Output

-

The anova_stats() function takes a model input and computes a comprehensive summary, including the above effect size measures, returned as tidy data frame:

-
anova_stats(fit)
-#>                  term  df      sumsq     meansq statistic p.value etasq partial.etasq omegasq partial.omegasq epsilonsq cohens.f power
-#> 1   as.factor(e42dep)   3  577756.33 192585.444   108.786   0.000 0.266         0.281   0.263           0.278     0.264    0.626  1.00
-#> 2 as.factor(c172code)   2   11722.05   5861.024     3.311   0.037 0.005         0.008   0.004           0.005     0.004    0.089  0.63
-#> 3             c160age   1  105169.60 105169.595    59.408   0.000 0.048         0.066   0.048           0.065     0.048    0.267  1.00
-#> 4           Residuals 834 1476436.34   1770.307        NA      NA    NA            NA      NA              NA        NA       NA    NA
-

Like the other functions, the input may also be an object of class anova, so you can also use model fits from the car package, which allows fitting Anova’s with different types of sum of squares:

-
anova_stats(car::Anova(fit, type = 3))
-#>                  term       sumsq     meansq  df statistic p.value etasq partial.etasq omegasq partial.omegasq epsilonsq cohens.f power
-#> 1   as.factor(e42dep)  426461.571 142153.857   3    80.299   0.000 0.212         0.224   0.209           0.221     0.209    0.537 1.000
-#> 2 as.factor(c172code)    7352.049   3676.025   2     2.076   0.126 0.004         0.005   0.002           0.003     0.002    0.071 0.429
-#> 3             c160age  105169.595 105169.595   1    59.408   0.000 0.052         0.066   0.051           0.065     0.051    0.267 1.000
-#> 4           Residuals 1476436.343   1770.307 834        NA      NA    NA            NA      NA              NA        NA       NA    NA
-
-
-

-References

-

Levine TR, Hullet CR. Eta Squared, Partial Eta Squared, and Misreporting of Effect Size in Communication Research. Human Communication Research 28(4); 2002: 612-625

-
-
- - - -
- - - - -
- - - - - - diff --git a/docs/articles/anova-statistics_files/header-attrs-2.1/header-attrs.js b/docs/articles/anova-statistics_files/header-attrs-2.1/header-attrs.js deleted file mode 100644 index dd57d92..0000000 --- a/docs/articles/anova-statistics_files/header-attrs-2.1/header-attrs.js +++ /dev/null @@ -1,12 +0,0 @@ -// Pandoc 2.9 adds attributes on both header and div. We remove the former (to -// be compatible with the behavior of Pandoc < 2.8). -document.addEventListener('DOMContentLoaded', function(e) { - var hs = document.querySelectorAll("div.section[class*='level'] > :first-child"); - var i, h, a; - for (i = 0; i < hs.length; i++) { - h = hs[i]; - if (!/^h[1-6]$/i.test(h.tagName)) continue; // it should be a header h1-h6 - a = h.attributes; - while (a.length > 0) h.removeAttribute(a[0].name); - } -}); diff --git a/docs/articles/bayesian-statistics.html b/docs/articles/bayesian-statistics.html deleted file mode 100644 index d0e15be..0000000 --- a/docs/articles/bayesian-statistics.html +++ /dev/null @@ -1,235 +0,0 @@ - - - - - - - -Summary of Mediation Analysis using Bayesian Regression Models • sjstats - - - - - - - - - - - - - - - - - -
-
- - - - -
-
- - - - - -

This vignettes demontrates the mediation()-function in sjstats. Before we start, we fit some models, including a mediation-object from the mediation-package, which we use for comparison with brms.

-
library(sjstats)
-library(mediation)
-library(brms)
-
-# load sample data
-data(jobs)
-set.seed(123)
-
-# linear models, for mediation analysis
-b1 <- lm(job_seek ~ treat + econ_hard + sex + age, data = jobs)
-b2 <- lm(depress2 ~ treat + job_seek + econ_hard + sex + age, data = jobs)
-
-# mediation analysis, for comparison with brms
-m1 <- mediate(b1, b2, sims = 1000, treat = "treat", mediator = "job_seek")
-
# Fit Bayesian mediation model
-f1 <- bf(job_seek ~ treat + econ_hard + sex + age)
-f2 <- bf(depress2 ~ treat + job_seek + econ_hard + sex + age)
-
-m2 <- brm(f1 + f2 + set_rescor(FALSE), data = jobs, cores = 4)
-

mediation() is a summary function, especially for mediation analysis, i.e. for multivariate response models with casual mediation effects.

-

In the model m2, treat is the treatment effect, job_seek is the mediator effect, f1 describes the mediator model and f2 describes the outcome model.

-

mediation() returns a data frame with information on the direct effect (median value of posterior samples from treatment of the outcome model), mediator effect (median value of posterior samples from mediator of the outcome model), indirect effect (median value of the multiplication of the posterior samples from mediator of the outcome model and the posterior samples from treatment of the mediation model) and the total effect (median value of sums of posterior samples used for the direct and indirect effect). The proportion mediated is the indirect effect divided by the total effect.

-

The simplest call just needs the model-object.

-
mediation(m2)
-#> 
-#> # Causal Mediation Analysis for Stan Model
-#> 
-#>   Treatment: treat
-#>    Mediator: job_seek
-#>    Response: depress2
-#> 
-#>                  Estimate    HDI (90%)
-#>   Direct effect:    -0.04 [-0.11 0.03]
-#> Indirect effect:    -0.02 [-0.04 0.00]
-#>    Total effect:    -0.05 [-0.13 0.02]
-#> 
-#> Proportion mediated: 28.14% [-79.57% 135.86%]
-

Typically, mediation() finds the treatment and mediator variables automatically. If this does not work, use the treatment and mediator arguments to specify the related variable names. For all values, the 90% HDIs are calculated by default. Use prob to calculate a different interval.

-

Here is a comparison with the mediation package. Note that the summary()-output of the mediation package shows the indirect effect first, followed by the direct effect.

-
summary(m1)
-#> 
-#> Causal Mediation Analysis 
-#> 
-#> Quasi-Bayesian Confidence Intervals
-#> 
-#>                Estimate 95% CI Lower 95% CI Upper p-value
-#> ACME            -0.0157      -0.0387         0.01    0.19
-#> ADE             -0.0438      -0.1315         0.04    0.35
-#> Total Effect    -0.0595      -0.1530         0.02    0.21
-#> Prop. Mediated   0.2137      -2.0277         2.70    0.32
-#> 
-#> Sample Size Used: 899 
-#> 
-#> 
-#> Simulations: 1000
-
-mediation(m2, prob = .95)
-#> 
-#> # Causal Mediation Analysis for Stan Model
-#> 
-#>   Treatment: treat
-#>    Mediator: job_seek
-#>    Response: depress2
-#> 
-#>                  Estimate    HDI (95%)
-#>   Direct effect:    -0.04 [-0.12 0.04]
-#> Indirect effect:    -0.02 [-0.04 0.01]
-#>    Total effect:    -0.05 [-0.15 0.03]
-#> 
-#> Proportion mediated: 28.14% [-178.65% 234.94%]
-

If you want to calculate mean instead of median values from the posterior samples, use the typical-argument. Furthermore, there is a print()-method, which allows to print more digits.

-
mediation(m2, typical = "mean", prob = .95) %>% print(digits = 4)
-#> 
-#> # Causal Mediation Analysis for Stan Model
-#> 
-#>   Treatment: treat
-#>    Mediator: job_seek
-#>    Response: depress2
-#> 
-#>                  Estimate        HDI (95%)
-#>   Direct effect:  -0.0395 [-0.1244 0.0450]
-#> Indirect effect:  -0.0158 [-0.0400 0.0086]
-#>    Total effect:  -0.0553 [-0.1482 0.0302]
-#> 
-#> Proportion mediated: 28.5975% [-178.1953% 235.3902%]
-

As you can see, the results are similar to what the mediation package produces for non-Bayesian models.

-
-

-References

-

Bürkner, P. C. (2017). brms: An R package for Bayesian multilevel models using Stan. Journal of Statistical Software, 80(1), 1-28

-
-
- - - -
- - - - -
- - - - - - diff --git a/docs/articles/bayesian-statistics_files/figure-html/unnamed-chunk-6-1.png b/docs/articles/bayesian-statistics_files/figure-html/unnamed-chunk-6-1.png deleted file mode 100644 index 478c0a0b74d837873148c9717bd22dea90f5c6f5..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 26428 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 a>OW :first-child"); - var i, h, a; - for (i = 0; i < hs.length; i++) { - h = hs[i]; - if (!/^h[1-6]$/i.test(h.tagName)) continue; // it should be a header h1-h6 - a = h.attributes; - while (a.length > 0) h.removeAttribute(a[0].name); - } -}); diff --git a/docs/articles/index.html b/docs/articles/index.html deleted file mode 100644 index 429c2ef..0000000 --- a/docs/articles/index.html +++ /dev/null @@ -1,175 +0,0 @@ - - - - - - - - -Articles • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
-
- - - - -
- -
- -
- - - -
- - - - - - - - diff --git a/docs/articles/mixedmodels-statistics.html b/docs/articles/mixedmodels-statistics.html deleted file mode 100644 index 5fa1720..0000000 --- a/docs/articles/mixedmodels-statistics.html +++ /dev/null @@ -1,247 +0,0 @@ - - - - - - - -Statistics for Mixed Effects Models • sjstats - - - - - - - - - - - - - - - - - -
-
- - - - -
-
- - - - -
-

-Statistics and Measures for Mixed Effects Models

-

This vignettes demontrates those functions of the sjstats-package that deal especially with mixed effects models. sjstats provides following functions:

- -

Befor we start, we fit a simple linear mixed model:

- -
-

-Sample Size Calculation for Mixed Models

-

The first two functions, design_effect() and samplesize_mixed(), can be used to approximately calculate the sample size in the context of power calculation. Calculating the sample size for simple linear models is pretty straightforward, however, for (linear) mixed models, statistical power is affected through the change of the variance of test statistics. This is what Hsieh et al. (2003) call a design effect (or variance inflation factor, VIF). Once this design effect is calculated, the sample size calculated for a standard design can be adjusted accordingly.

-
-

-Design Effect for Two-Level Mixed Models

-

design_effect() computes this design effect for linear mixed models with two-level design. It requires the approximated average number of observations per grouping cluster (i.e. level-2 unit) and the assumed intraclass correlation coefficient (ICC) for the multilevel-model. Typically, the minimum assumed value for the ICC is 0.05.

- -
-
-

-Calculating the Sample Size for Linear Mixed Models

-

samplesize_mixed() combines the functions for power calculation from the pwr-package and design effect design_effect(). It computes an approximated sample size for linear mixed models (two-level-designs), based on power-calculation for standard design and adjusted for design effect for 2-level-designs.

- -

There are more ways to perform power calculations for multilevel models, however, most of these require very detailed knowledge about the sample characteristics and performing simulation studys. samplesize_mixed() is a more pragmatic alternative to these approaches.

-
-
-
-

-Rescale model weights for complex samples

-

Most functions to fit multilevel and mixed effects models only allow to specify frequency weights, but not design (i.e. sampling or probability) weights, which should be used when analyzing complex samples and survey data.

-

scale_weights() implements an algorithm proposed by Aaparouhov (2006) and Carle (2009) to rescale design weights in survey data to account for the grouping structure of multilevel models, which then can be used for multilevel modelling.

-

To calculate a weight-vector that can be used in multilevel models, scale_weights() needs the data frame with survey data as x-argument. This data frame should contain 1) a cluster ID (argument cluster.id), which represents the strata of the survey data (the level-2-cluster variable) and 2) the probability weights (argument pweight), which represents the design or sampling weights of the survey data (level-1-weight).

-

scale_weights() then returns the original data frame, including two new variables: svywght_a, where the sample weights pweight are adjusted by a factor that represents the proportion of cluster size divided by the sum of sampling weights within each cluster. The adjustment factor for svywght_b is the sum of sample weights within each cluster devided by the sum of squared sample weights within each cluster (see Carle (2009), Appendix B, for details).

- -
-
-
-

-References

-

Aaparouhov T. 2006. General Multi-Level Modeling with Sampling Weights. Communications in Statistics—Theory and Methods (35): 439–460

-

Carle AC. 2009. Fitting multilevel models in complex survey data with design weights: Recommendations. BMC Medical Research Methodology 9(49): 1-13

-

Hsieh FY, Lavori PW, Cohen HJ, Feussner JR. 2003. An Overview of Variance Inflation Factors for Sample-Size Calculation. Evaluation & the Health Professions 26: 239–257. doi: 10.1177/0163278703255230

-
-
- - - -
- - - - -
- - - - - - diff --git a/docs/authors.html b/docs/authors.html deleted file mode 100644 index fb60f1a..0000000 --- a/docs/authors.html +++ /dev/null @@ -1,171 +0,0 @@ - - - - - - - - -Citation and Authors • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
-
- - - - -
- -
-
- - -

Lüdecke D (????). -sjstats: Statistical Functions for Regression Models (Version 0.18.0). -doi: 10.5281/zenodo.1284472, https://CRAN.R-project.org/package=sjstats. -

-
@Manual{,
-  title = {sjstats: Statistical Functions for Regression Models (Version 0.18.0)},
-  author = {Daniel Lüdecke},
-  url = {https://CRAN.R-project.org/package=sjstats},
-  doi = {10.5281/zenodo.1284472},
-}
- - - - - -
- -
- - - - -
- - - - - - - - diff --git a/docs/bootstrap-toc.css b/docs/bootstrap-toc.css deleted file mode 100644 index 5a85941..0000000 --- a/docs/bootstrap-toc.css +++ /dev/null @@ -1,60 +0,0 @@ -/*! - * Bootstrap Table of Contents v0.4.1 (http://afeld.github.io/bootstrap-toc/) - * Copyright 2015 Aidan Feldman - * Licensed under MIT (https://github.com/afeld/bootstrap-toc/blob/gh-pages/LICENSE.md) */ - -/* modified from https://github.com/twbs/bootstrap/blob/94b4076dd2efba9af71f0b18d4ee4b163aa9e0dd/docs/assets/css/src/docs.css#L548-L601 */ - -/* All levels of nav */ -nav[data-toggle='toc'] .nav > li > a { - display: block; - padding: 4px 20px; - font-size: 13px; - font-weight: 500; - color: #767676; -} -nav[data-toggle='toc'] .nav > li > a:hover, -nav[data-toggle='toc'] .nav > li > a:focus { - padding-left: 19px; - color: #563d7c; - text-decoration: none; - background-color: transparent; - border-left: 1px solid #563d7c; -} -nav[data-toggle='toc'] .nav > .active > a, -nav[data-toggle='toc'] .nav > .active:hover > a, -nav[data-toggle='toc'] .nav > .active:focus > a { - padding-left: 18px; - font-weight: bold; - color: #563d7c; - background-color: transparent; - border-left: 2px solid #563d7c; -} - -/* Nav: second level (shown on .active) */ -nav[data-toggle='toc'] .nav .nav { - display: none; /* Hide by default, but at >768px, show it */ - padding-bottom: 10px; -} -nav[data-toggle='toc'] .nav .nav > li > a { - padding-top: 1px; - padding-bottom: 1px; - padding-left: 30px; - font-size: 12px; - font-weight: normal; -} -nav[data-toggle='toc'] .nav .nav > li > a:hover, -nav[data-toggle='toc'] .nav .nav > li > a:focus { - padding-left: 29px; -} -nav[data-toggle='toc'] .nav .nav > .active > a, -nav[data-toggle='toc'] .nav .nav > .active:hover > a, -nav[data-toggle='toc'] .nav .nav > .active:focus > a { - padding-left: 28px; - font-weight: 500; -} - -/* from https://github.com/twbs/bootstrap/blob/e38f066d8c203c3e032da0ff23cd2d6098ee2dd6/docs/assets/css/src/docs.css#L631-L634 */ -nav[data-toggle='toc'] .nav > .active > ul { - display: block; -} diff --git a/docs/bootstrap-toc.js b/docs/bootstrap-toc.js deleted file mode 100644 index 1cdd573..0000000 --- a/docs/bootstrap-toc.js +++ /dev/null @@ -1,159 +0,0 @@ -/*! - * Bootstrap Table of Contents v0.4.1 (http://afeld.github.io/bootstrap-toc/) - * Copyright 2015 Aidan Feldman - * Licensed under MIT (https://github.com/afeld/bootstrap-toc/blob/gh-pages/LICENSE.md) */ -(function() { - 'use strict'; - - window.Toc = { - helpers: { - // return all matching elements in the set, or their descendants - findOrFilter: function($el, selector) { - // http://danielnouri.org/notes/2011/03/14/a-jquery-find-that-also-finds-the-root-element/ - // http://stackoverflow.com/a/12731439/358804 - var $descendants = $el.find(selector); - return $el.filter(selector).add($descendants).filter(':not([data-toc-skip])'); - }, - - generateUniqueIdBase: function(el) { - var text = $(el).text(); - var anchor = text.trim().toLowerCase().replace(/[^A-Za-z0-9]+/g, '-'); - return anchor || el.tagName.toLowerCase(); - }, - - generateUniqueId: function(el) { - var anchorBase = this.generateUniqueIdBase(el); - for (var i = 0; ; i++) { - var anchor = anchorBase; - if (i > 0) { - // add suffix - anchor += '-' + i; - } - // check if ID already exists - if (!document.getElementById(anchor)) { - return anchor; - } - } - }, - - generateAnchor: function(el) { - if (el.id) { - return el.id; - } else { - var anchor = this.generateUniqueId(el); - el.id = anchor; - return anchor; - } - }, - - createNavList: function() { - return $(''); - }, - - createChildNavList: function($parent) { - var $childList = this.createNavList(); - $parent.append($childList); - return $childList; - }, - - generateNavEl: function(anchor, text) { - var $a = $(''); - $a.attr('href', '#' + anchor); - $a.text(text); - var $li = $('
  • '); - $li.append($a); - return $li; - }, - - generateNavItem: function(headingEl) { - var anchor = this.generateAnchor(headingEl); - var $heading = $(headingEl); - var text = $heading.data('toc-text') || $heading.text(); - return this.generateNavEl(anchor, text); - }, - - // Find the first heading level (`

    `, then `

    `, etc.) that has more than one element. Defaults to 1 (for `

    `). - getTopLevel: function($scope) { - for (var i = 1; i <= 6; i++) { - var $headings = this.findOrFilter($scope, 'h' + i); - if ($headings.length > 1) { - return i; - } - } - - return 1; - }, - - // returns the elements for the top level, and the next below it - getHeadings: function($scope, topLevel) { - var topSelector = 'h' + topLevel; - - var secondaryLevel = topLevel + 1; - var secondarySelector = 'h' + secondaryLevel; - - return this.findOrFilter($scope, topSelector + ',' + secondarySelector); - }, - - getNavLevel: function(el) { - return parseInt(el.tagName.charAt(1), 10); - }, - - populateNav: function($topContext, topLevel, $headings) { - var $context = $topContext; - var $prevNav; - - var helpers = this; - $headings.each(function(i, el) { - var $newNav = helpers.generateNavItem(el); - var navLevel = helpers.getNavLevel(el); - - // determine the proper $context - if (navLevel === topLevel) { - // use top level - $context = $topContext; - } else if ($prevNav && $context === $topContext) { - // create a new level of the tree and switch to it - $context = helpers.createChildNavList($prevNav); - } // else use the current $context - - $context.append($newNav); - - $prevNav = $newNav; - }); - }, - - parseOps: function(arg) { - var opts; - if (arg.jquery) { - opts = { - $nav: arg - }; - } else { - opts = arg; - } - opts.$scope = opts.$scope || $(document.body); - return opts; - } - }, - - // accepts a jQuery object, or an options object - init: function(opts) { - opts = this.helpers.parseOps(opts); - - // ensure that the data attribute is in place for styling - opts.$nav.attr('data-toggle', 'toc'); - - var $topContext = this.helpers.createChildNavList(opts.$nav); - var topLevel = this.helpers.getTopLevel(opts.$scope); - var $headings = this.helpers.getHeadings(opts.$scope, topLevel); - this.helpers.populateNav($topContext, topLevel, $headings); - } - }; - - $(function() { - $('nav[data-toggle="toc"]').each(function(i, el) { - var $nav = $(el); - Toc.init($nav); - }); - }); -})(); diff --git a/docs/docsearch.css b/docs/docsearch.css deleted file mode 100644 index e5f1fe1..0000000 --- a/docs/docsearch.css +++ /dev/null @@ -1,148 +0,0 @@ -/* Docsearch -------------------------------------------------------------- */ -/* - Source: https://github.com/algolia/docsearch/ - License: MIT -*/ - -.algolia-autocomplete { - display: block; - -webkit-box-flex: 1; - -ms-flex: 1; - flex: 1 -} - -.algolia-autocomplete .ds-dropdown-menu { - width: 100%; - min-width: none; - max-width: none; - padding: .75rem 0; - background-color: #fff; - background-clip: padding-box; - border: 1px solid rgba(0, 0, 0, .1); - box-shadow: 0 .5rem 1rem rgba(0, 0, 0, .175); -} - -@media (min-width:768px) { - .algolia-autocomplete .ds-dropdown-menu { - width: 175% - } -} - -.algolia-autocomplete .ds-dropdown-menu::before { - display: none -} - -.algolia-autocomplete .ds-dropdown-menu [class^=ds-dataset-] { - padding: 0; - background-color: rgb(255,255,255); - border: 0; - max-height: 80vh; -} - -.algolia-autocomplete .ds-dropdown-menu .ds-suggestions { - margin-top: 0 -} - -.algolia-autocomplete .algolia-docsearch-suggestion { - padding: 0; - overflow: visible -} - -.algolia-autocomplete .algolia-docsearch-suggestion--category-header { - padding: .125rem 1rem; - margin-top: 0; - font-size: 1.3em; - font-weight: 500; - color: #00008B; - border-bottom: 0 -} - -.algolia-autocomplete .algolia-docsearch-suggestion--wrapper { - float: none; - padding-top: 0 -} - -.algolia-autocomplete .algolia-docsearch-suggestion--subcategory-column { - float: none; - width: auto; - padding: 0; - text-align: left -} - -.algolia-autocomplete .algolia-docsearch-suggestion--content { - float: none; - width: auto; - padding: 0 -} - -.algolia-autocomplete .algolia-docsearch-suggestion--content::before { - display: none -} - -.algolia-autocomplete .ds-suggestion:not(:first-child) .algolia-docsearch-suggestion--category-header { - padding-top: .75rem; - margin-top: .75rem; - border-top: 1px solid rgba(0, 0, 0, .1) -} - -.algolia-autocomplete .ds-suggestion .algolia-docsearch-suggestion--subcategory-column { - display: block; - padding: .1rem 1rem; - margin-bottom: 0.1; - font-size: 1.0em; - font-weight: 400 - /* display: none */ -} - -.algolia-autocomplete .algolia-docsearch-suggestion--title { - display: block; - padding: .25rem 1rem; - margin-bottom: 0; - font-size: 0.9em; - font-weight: 400 -} - -.algolia-autocomplete .algolia-docsearch-suggestion--text { - padding: 0 1rem .5rem; - margin-top: -.25rem; - font-size: 0.8em; - font-weight: 400; - line-height: 1.25 -} - -.algolia-autocomplete .algolia-docsearch-footer { - width: 110px; - height: 20px; - z-index: 3; - margin-top: 10.66667px; - float: right; - font-size: 0; - line-height: 0; -} - -.algolia-autocomplete .algolia-docsearch-footer--logo { - background-image: url("data:image/svg+xml;utf8,"); - background-repeat: no-repeat; - background-position: 50%; - background-size: 100%; - overflow: hidden; - text-indent: -9000px; - width: 100%; - height: 100%; - display: block; - transform: translate(-8px); -} - -.algolia-autocomplete .algolia-docsearch-suggestion--highlight { - color: #FF8C00; - background: rgba(232, 189, 54, 0.1) -} - - -.algolia-autocomplete .algolia-docsearch-suggestion--text .algolia-docsearch-suggestion--highlight { - box-shadow: inset 0 -2px 0 0 rgba(105, 105, 105, .5) -} - -.algolia-autocomplete .ds-suggestion.ds-cursor .algolia-docsearch-suggestion--content { - background-color: rgba(192, 192, 192, .15) -} diff --git a/docs/docsearch.js b/docs/docsearch.js deleted file mode 100644 index b35504c..0000000 --- a/docs/docsearch.js +++ /dev/null @@ -1,85 +0,0 @@ -$(function() { - - // register a handler to move the focus to the search bar - // upon pressing shift + "/" (i.e. "?") - $(document).on('keydown', function(e) { - if (e.shiftKey && e.keyCode == 191) { - e.preventDefault(); - $("#search-input").focus(); - } - }); - - $(document).ready(function() { - // do keyword highlighting - /* modified from https://jsfiddle.net/julmot/bL6bb5oo/ */ - var mark = function() { - - var referrer = document.URL ; - var paramKey = "q" ; - - if (referrer.indexOf("?") !== -1) { - var qs = referrer.substr(referrer.indexOf('?') + 1); - var qs_noanchor = qs.split('#')[0]; - var qsa = qs_noanchor.split('&'); - var keyword = ""; - - for (var i = 0; i < qsa.length; i++) { - var currentParam = qsa[i].split('='); - - if (currentParam.length !== 2) { - continue; - } - - if (currentParam[0] == paramKey) { - keyword = decodeURIComponent(currentParam[1].replace(/\+/g, "%20")); - } - } - - if (keyword !== "") { - $(".contents").unmark({ - done: function() { - $(".contents").mark(keyword); - } - }); - } - } - }; - - mark(); - }); -}); - -/* Search term highlighting ------------------------------*/ - -function matchedWords(hit) { - var words = []; - - var hierarchy = hit._highlightResult.hierarchy; - // loop to fetch from lvl0, lvl1, etc. - for (var idx in hierarchy) { - words = words.concat(hierarchy[idx].matchedWords); - } - - var content = hit._highlightResult.content; - if (content) { - words = words.concat(content.matchedWords); - } - - // return unique words - var words_uniq = [...new Set(words)]; - return words_uniq; -} - -function updateHitURL(hit) { - - var words = matchedWords(hit); - var url = ""; - - if (hit.anchor) { - url = hit.url_without_anchor + '?q=' + escape(words.join(" ")) + '#' + hit.anchor; - } else { - url = hit.url + '?q=' + escape(words.join(" ")); - } - - return url; -} diff --git a/docs/favicon-16x16.png b/docs/favicon-16x16.png deleted file mode 100644 index a081ac2e857268b521d93733104b80c3c71ce8a8..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1158 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 diff --git a/docs/favicon.ico b/docs/favicon.ico deleted file mode 100644 index e460caa7ecd09a53863515c4c7be74aaa16d7dd3..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 15086 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 owX?qE?6DO$sw!;e(eHB(nDf8LnHD39 - - - - - - -Collection of Convenient Functions for Common Statistical Computations • sjstats - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    -
    -
    - - -

    Collection of convenient functions for common statistical computations, which are not directly provided by R’s base or stats packages.

    -

    This package aims at providing, first, shortcuts for statistical measures, which otherwise could only be calculated with additional effort (like Cramer’s V, Phi, or effict size statistics like Eta or Omega squared), or for which currently no functions available.

    -

    Second, another focus lies on weighted variants of common statistical measures and tests like weighted standard error, mean, t-test, correlation, and more.

    -

    The comprised tools include:

    -
      -
    • Especially for mixed models: design effect, sample size calculation
    • -
    • Especially for Bayesian models: mediation analysis
    • -
    • For anova-tables: Eta-squared, Partial Eta-squared, Omega-squared, Partial Omega-squared and Epsilon-squared statistics
    • -
    • Weighted statistics and tests for: mean, median, standard error, standard deviation, correlation, Chi-squared test, t-test, Mann-Whitney-U-test
    • -
    -
    -

    -Documentation

    -

    Please visit https://strengejacke.github.io/sjstats/ for documentation and vignettes.

    -
    -
    -

    -Installation

    -
    -

    -Latest development build

    -

    To install the latest development snapshot (see latest changes below), type following commands into the R console:

    -
    library(devtools)
    -devtools::install_github("strengejacke/sjstats")
    -
    -
    -

    -Officiale, stable release

    -

    CRAN_Status_Badge    downloads    total

    -

    To install the latest stable release from CRAN, type following command into the R console:

    -
    install.packages("sjstats")
    -
    -
    -
    -

    -Citation

    -

    In case you want / have to cite my package, please use citation('sjstats') for citation information.

    -

    DOI

    -
    -
    -
    - - -
    - - - -
    - - - - - - diff --git a/docs/link.svg b/docs/link.svg deleted file mode 100644 index 88ad827..0000000 --- a/docs/link.svg +++ /dev/null @@ -1,12 +0,0 @@ - - - - - - diff --git a/docs/logo.png b/docs/logo.png deleted file mode 100644 index 7c24480b73f93451cf17c50396d77d087461396c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 18507 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 - - - - - - - -Changelog • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    -sjstats 0.18.0 2020-05-06 -

    -
    -

    -General

    -
      -
    • Effect size computation functions (like eta_sq()) now internally call the related functions from the effectsize package.
    • -
    • Remove packages from “Suggest” that have been removed from CRAN.
    • -
    -
    -
    -
    -

    -sjstats 0.17.9 2020-02-06 -

    -
    -

    -Bug fixes

    -
      -
    • Fixed documentation for chisq_gof().
    • -
    • Fixed issue in anova_stats() with incorrect effect sizes for certain Anova types (that included an intercept).
    • -
    -
    -
    -
    -

    -sjstats 0.17.8 2020-01-21 -

    -
    -

    -Deprecated and defunct

    -

    sjstats is being re-structured, and many functions are re-implemented in new packages that are part of a new project called easystats.

    -

    Therefore, following functions are now deprecated:

    - -
    -
    -

    -General

    - -
    -
    -
    -

    -sjstats 0.17.7 2019-11-14 -

    -
    -

    -Deprecated and defunct

    -

    sjstats is being re-structured, and many functions are re-implemented in new packages that are part of a new project called easystats. The aim of easystats is to provide a unifying and consistent framework to tame, discipline and harness the scary R statistics and their pesky models.

    -

    Therefore, following functions are now deprecated:

    - -
    -
    -

    -General

    -
      -
    • Revise some functions to cope with the forthcoming insight update.
    • -
    -
    -
    -
    -

    -sjstats 0.17.6 2019-09-08 -

    -
    -

    -General

    - -
    -
    -

    -New functions

    -
      -
    • -svyglm.zip() to fit zero-inflated Poisson models for survey-designs.
    • -
    -
    -
    -

    -Changes to functions

    -
      -
    • -phi() and cramer() can now compute confidence intervals.
    • -
    • -tidy_stan() removes prior parameters from output.
    • -
    • -tidy_stan() now also prints the probability of direction.
    • -
    -
    -
    -

    -Bug fixes

    - -
    -
    -
    -

    -sjstats 0.17.5 2019-06-04 -

    -
    -

    -New functions

    - -
    -
    -

    -Deprecated and defunct

    -

    sjstats is being re-structured, and many functions are re-implemented in new packages that are part of a new project called easystats. The aim of easystats is to provide a unifying and consistent framework to tame, discipline and harness the scary R statistics and their pesky models.

    -

    Therefore, following functions are now deprecated:

    - -
    -
    -

    -Changes to functions

    -
      -
    • Anova-stats functions (like eta_sq()) get a method-argument to define the method for computing confidence intervals from bootstrapping.
    • -
    -
    -
    -

    -Bug fixes

    -
      -
    • In some situations, smpsize_lmm() could result in negative sample-size recommendations. This was fixed, and a warning is now shown indicating that the parameters for the power-calculation should be modified.
    • -
    • Fixed issue with wrong calculated effect size r in mwu() if group-factor contained more than two groups.
    • -
    -
    -
    -
    -

    -sjstats 0.17.4 2019-03-15 -

    -
    -

    -General

    -
      -
    • Following models/objects are now supported by model-information functions like model_family(), link_inverse() or model_frame(): MixMod (package GLMMadaptive), MCMCglmm, mlogit and gmnl.
    • -
    • Reduce package dependencies.
    • -
    -
    -
    -

    -New functions

    -
      -
    • -cred_int(), to compute uncertainty intervals of Bayesian models. Mimics the behaviour and style of hdi() and is thus a convenient complement to functions like posterior_interval().
    • -
    -
    -
    -

    -Changes to functions

    -
      -
    • -equi_test() now finds better defaults for models with binomial outcome (like logistic regression models).
    • -
    • -r2() for mixed models now also should work properly for mixed models fitted with rstanarm.
    • -
    • -anova_stats() and alike (e.g. eta_sq()) now all preserve original term names.
    • -
    • -model_family() now returns $is_count = TRUE, when model is a count-model, and $is_beta = TRUE for models with beta-family.
    • -
    • -pred_vars() checks that return value has only unique values.
    • -
    • -pred_vars() gets a zi-argument to return the variables from a model’s zero-inflation-formula.
    • -
    -
    -
    -

    -Bug fixes

    -
      -
    • Fix minor issues in wtd_sd() and wtd_mean() when weight was NULL (which usually shoudln’t be the case anyway).
    • -
    • Fix potential issue with deparse(), cutting off very long formulas in various functions.
    • -
    • Fix encoding issues in help-files.
    • -
    -
    -
    -
    -

    -sjstats 0.17.3 2019-01-07 -

    -
    -

    -General

    -
      -
    • Export dplyr::n(), to meet forthcoming changes in dplyr 0.8.0.
    • -
    -
    -
    -

    -Changes to functions

    -
      -
    • -boot_ci() gets a ci.lvl-argument.
    • -
    • The rotation-argument in pca_rotate() now supports all rotations from psych::principal().
    • -
    • -pred_vars() gets a fe.only-argument to return only fixed effects terms from mixed models, and a disp-argument to return the variables from a model’s dispersion-formula.
    • -
    • -icc() for Bayesian models gets a adjusted-argument, to calculate adjusted and conditional ICC (however, only for Gaussian models).
    • -
    • For icc() for non-Gaussian Bayes-models, a message is printed that recommends setting argument ppd to TRUE.
    • -
    • -resp_val() and resp_var() now also work for brms-models with additional response information (like trial() in formula).
    • -
    • -resp_var() gets a combine-argument, to return either the name of the matrix-column or the original variable names for matrix-columns.
    • -
    • -model_frame() now also returns the original variables for matrix-column-variables.
    • -
    • -model_frame() now also returns the variable from the dispersion-formula of glmmTMB-models.
    • -
    • -model_family() and link_inverse() now supports glmmPQL, felm and lm_robust-models.
    • -
    • -anova_stats() and alike (omeqa_sq() etc.) now support gam-models from package gam.
    • -
    • -p_value() now supports objects of class svyolr.
    • -
    -
    -
    -

    -Bug fixes

    -
      -
    • Fix issue with se() and get_re_var() for objects returned by icc().
    • -
    • Fix issue with icc() for Stan-models.
    • -
    • -var_names() did not clear terms with log-log transformation, e.g. log(log(y)).
    • -
    • Fix issue in model_frame() for models with splines with only one column.
    • -
    -
    -
    -
    -

    -sjstats 0.17.2 2018-11-15 -

    -
    -

    -General

    -
      -
    • Revised help-files for r2() and icc(), also by adding more references.
    • -
    -
    -
    -

    -New functions

    -
      -
    • -re_grp_var() to find group factors of random effects in mixed models.
    • -
    -
    -
    -

    -Changes to functions

    -
      -
    • -omega_sq() and eta_sq() give more informative messages when using non-supported objects.
    • -
    • -r2() and icc() give more informative warnings and messages.
    • -
    • -tidy_stan() supports printing simplex parameters of monotonic effects of brms models.
    • -
    • -grpmean() and mwu() get a file and encoding argument, to save the HTML output as file.
    • -
    -
    -
    -

    -Bug fixes

    -
      -
    • -model_frame() now correctly names the offset-columns for terms provided as offset-argument (i.e. for models where the offset was not specified inside the formula).
    • -
    • Fixed issue with weights-argument in grpmean() when variable name was passed as character vector.
    • -
    • Fixed issue with r2() for glmmTMB models with ar1 random effects structure.
    • -
    -
    -
    -
    -

    -sjstats 0.17.1 2018-10-02 -

    -
    -

    -New functions

    -
      -
    • -wtd_chisqtest() to compute a weighted Chi-squared test.
    • -
    • -wtd_median() to compute the weighted median of variables.
    • -
    • -wtd_cor() to compute weighted correlation coefficients of variables.
    • -
    -
    -
    -

    -Changes to functions

    -
      -
    • -mediation() can now cope with models from different families, e.g. if the moderator or outcome is binary, while the treatment-effect is continuous.
    • -
    • -model_frame(), link_inverse(), pred_vars(), resp_var(), resp_val(), r2() and model_family() now support clm2-objects from package ordinal.
    • -
    • -anova_stats() gives a more informative message for non-supported models or ANOVA-options.
    • -
    -
    -
    -

    -Bug fixes

    -
      -
    • Fixed issue with model_family() and link_inverse() for models fitted with pscl::hurdle() or pscl::zeroinfl().
    • -
    • Fixed issue with wrong title in grpmean() for grouped data frames, when grouping variable was an unlabelled factor.
    • -
    • Fix issue with model_frame() for coxph-models with polynomial or spline-terms.
    • -
    • Fix issue with mediation() for logical variables.
    • -
    -
    -
    -
    -

    -sjstats 0.17.0 2018-08-20 -

    -
    -

    -General

    -
      -
    • Reduce package dependencies.
    • -
    -
    -
    -

    -New functions

    -
      -
    • -wtd_ttest() to compute a weighted t-test.
    • -
    • -wtd_mwu() to compute a weighted Mann-Whitney-U or Kruskal-Wallis test.
    • -
    -
    -
    -

    -Changes to functions

    -
      -
    • -robust() was revised, getting more arguments to specify different types of covariance-matrix estimation, and handling these more flexible.
    • -
    • Improved print()-method for tidy_stan() for brmsfit-objects with categorical-families.
    • -
    • -se() now also computes standard errors for relative frequencies (proportions) of a vector.
    • -
    • -r2() now also computes r-squared values for glmmTMB-models from genpois-families.
    • -
    • -r2() gives more precise warnings for non-supported model-families.
    • -
    • -xtab_statistics() gets a weights-argument, to compute measures of association for contingency tables for weighted data.
    • -
    • The statistics-argument in xtab_statistics() gets a "fisher"-option, to force Fisher’s Exact Test to be used.
    • -
    • Improved variance calculation in icc() for generalized linear mixed models with Poisson or negative binomial families.
    • -
    • -icc() gets an adjusted-argument, to calculate the adjusted and conditional ICC for mixed models.
    • -
    • To get consistent argument names across functions, argument weight.by is now deprecated and renamed into weights.
    • -
    -
    -
    -

    -Bug fixes

    -
      -
    • Fix issues with effect size computation for repeated-measure Anova when using bootstrapping to compute confidence intervals.
    • -
    • -grpmean() now also adjusts the n-columm for weighted data.
    • -
    • -icc(), re_var() and get_re_var() now correctly compute the random-effect-variances for models with multiple random slopes per random effect term (e.g., (1 + rs1 + rs2 | grp)).
    • -
    • Fix issues in tidy_stan(), mcse(), hdi() and n_eff() for stan_polr()-models.
    • -
    • Plotting equi_test() did not work for intercept-only models.
    • -
    -
    -
    -
    - - - -
    - - - -
    - - - - - - - - diff --git a/docs/pkgdown.css b/docs/pkgdown.css deleted file mode 100644 index c01e592..0000000 --- a/docs/pkgdown.css +++ /dev/null @@ -1,367 +0,0 @@ -/* Sticky footer */ - -/** - * Basic idea: https://philipwalton.github.io/solved-by-flexbox/demos/sticky-footer/ - * Details: https://github.com/philipwalton/solved-by-flexbox/blob/master/assets/css/components/site.css - * - * .Site -> body > .container - * .Site-content -> body > .container .row - * .footer -> footer - * - * Key idea seems to be to ensure that .container and __all its parents__ - * have height set to 100% - * - */ - -html, body { - height: 100%; -} - -body { - position: relative; -} - -body > .container { - display: flex; - height: 100%; - flex-direction: column; -} - -body > .container .row { - flex: 1 0 auto; -} - -footer { - margin-top: 45px; - padding: 35px 0 36px; - border-top: 1px solid #e5e5e5; - color: #666; - display: flex; - flex-shrink: 0; -} -footer p { - margin-bottom: 0; -} -footer div { - flex: 1; -} -footer .pkgdown { - text-align: right; -} -footer p { - margin-bottom: 0; -} - -img.icon { - float: right; -} - -img { - max-width: 100%; -} - -/* Fix bug in bootstrap (only seen in firefox) */ -summary { - display: list-item; -} - -/* Typographic tweaking ---------------------------------*/ - -.contents .page-header { - margin-top: calc(-60px + 1em); -} - -dd { - margin-left: 3em; -} - -/* Section anchors ---------------------------------*/ - -a.anchor { - margin-left: -30px; - display:inline-block; - width: 30px; - height: 30px; - visibility: hidden; - - background-image: url(./link.svg); - background-repeat: no-repeat; - background-size: 20px 20px; - background-position: center center; -} - -.hasAnchor:hover a.anchor { - visibility: visible; -} - -@media (max-width: 767px) { - .hasAnchor:hover a.anchor { - visibility: hidden; - } -} - - -/* Fixes for fixed navbar --------------------------*/ - -.contents h1, .contents h2, .contents h3, .contents h4 { - padding-top: 60px; - margin-top: -40px; -} - -/* Navbar submenu --------------------------*/ - -.dropdown-submenu { - position: relative; -} - -.dropdown-submenu>.dropdown-menu { - top: 0; - left: 100%; - margin-top: -6px; - margin-left: -1px; - border-radius: 0 6px 6px 6px; -} - -.dropdown-submenu:hover>.dropdown-menu { - display: block; -} - -.dropdown-submenu>a:after { - display: block; - content: " "; - float: right; - width: 0; - height: 0; - border-color: transparent; - border-style: solid; - border-width: 5px 0 5px 5px; - border-left-color: #cccccc; - margin-top: 5px; - margin-right: -10px; -} - -.dropdown-submenu:hover>a:after { - border-left-color: #ffffff; -} - -.dropdown-submenu.pull-left { - float: none; -} - -.dropdown-submenu.pull-left>.dropdown-menu { - left: -100%; - margin-left: 10px; - border-radius: 6px 0 6px 6px; -} - -/* Sidebar --------------------------*/ - -#pkgdown-sidebar { - margin-top: 30px; - position: -webkit-sticky; - position: sticky; - top: 70px; -} - -#pkgdown-sidebar h2 { - font-size: 1.5em; - margin-top: 1em; -} - -#pkgdown-sidebar h2:first-child { - margin-top: 0; -} - -#pkgdown-sidebar .list-unstyled li { - margin-bottom: 0.5em; -} - -/* bootstrap-toc tweaks ------------------------------------------------------*/ - -/* All levels of nav */ - -nav[data-toggle='toc'] .nav > li > a { - padding: 4px 20px 4px 6px; - font-size: 1.5rem; - font-weight: 400; - color: inherit; -} - -nav[data-toggle='toc'] .nav > li > a:hover, -nav[data-toggle='toc'] .nav > li > a:focus { - padding-left: 5px; - color: inherit; - border-left: 1px solid #878787; -} - -nav[data-toggle='toc'] .nav > .active > a, -nav[data-toggle='toc'] .nav > .active:hover > a, -nav[data-toggle='toc'] .nav > .active:focus > a { - padding-left: 5px; - font-size: 1.5rem; - font-weight: 400; - color: inherit; - border-left: 2px solid #878787; -} - -/* Nav: second level (shown on .active) */ - -nav[data-toggle='toc'] .nav .nav { - display: none; /* Hide by default, but at >768px, show it */ - padding-bottom: 10px; -} - -nav[data-toggle='toc'] .nav .nav > li > a { - padding-left: 16px; - font-size: 1.35rem; -} - -nav[data-toggle='toc'] .nav .nav > li > a:hover, -nav[data-toggle='toc'] .nav .nav > li > a:focus { - padding-left: 15px; -} - -nav[data-toggle='toc'] .nav .nav > .active > a, -nav[data-toggle='toc'] .nav .nav > .active:hover > a, -nav[data-toggle='toc'] .nav .nav > .active:focus > a { - padding-left: 15px; - font-weight: 500; - font-size: 1.35rem; -} - -/* orcid ------------------------------------------------------------------- */ - -.orcid { - font-size: 16px; - color: #A6CE39; - /* margins are required by official ORCID trademark and display guidelines */ - margin-left:4px; - margin-right:4px; - vertical-align: middle; -} - -/* Reference index & topics ----------------------------------------------- */ - -.ref-index th {font-weight: normal;} - -.ref-index td {vertical-align: top;} -.ref-index .icon {width: 40px;} -.ref-index .alias {width: 40%;} -.ref-index-icons .alias {width: calc(40% - 40px);} -.ref-index .title {width: 60%;} - -.ref-arguments th {text-align: right; padding-right: 10px;} -.ref-arguments th, .ref-arguments td {vertical-align: top;} -.ref-arguments .name {width: 20%;} -.ref-arguments .desc {width: 80%;} - -/* Nice scrolling for wide elements --------------------------------------- */ - -table { - display: block; - overflow: auto; -} - -/* Syntax highlighting ---------------------------------------------------- */ - -pre { - word-wrap: normal; - word-break: normal; - border: 1px solid #eee; -} - -pre, code { - background-color: #f8f8f8; - color: #333; -} - -pre code { - overflow: auto; - word-wrap: normal; - white-space: pre; -} - -pre .img { - margin: 5px 0; -} - -pre .img img { - background-color: #fff; - display: block; - height: auto; -} - -code a, pre a { - color: #375f84; -} - -a.sourceLine:hover { - text-decoration: none; -} - -.fl {color: #1514b5;} -.fu {color: #000000;} /* function */ -.ch,.st {color: #036a07;} /* string */ -.kw {color: #264D66;} /* keyword */ -.co {color: #888888;} /* comment */ - -.message { color: black; font-weight: bolder;} -.error { color: orange; font-weight: bolder;} -.warning { color: #6A0366; font-weight: bolder;} - -/* Clipboard --------------------------*/ - -.hasCopyButton { - position: relative; -} - -.btn-copy-ex { - position: absolute; - right: 0; - top: 0; - visibility: hidden; -} - -.hasCopyButton:hover button.btn-copy-ex { - visibility: visible; -} - -/* headroom.js ------------------------ */ - -.headroom { - will-change: transform; - transition: transform 200ms linear; -} -.headroom--pinned { - transform: translateY(0%); -} -.headroom--unpinned { - transform: translateY(-100%); -} - -/* mark.js ----------------------------*/ - -mark { - background-color: rgba(255, 255, 51, 0.5); - border-bottom: 2px solid rgba(255, 153, 51, 0.3); - padding: 1px; -} - -/* vertical spacing after htmlwidgets */ -.html-widget { - margin-bottom: 10px; -} - -/* fontawesome ------------------------ */ - -.fab { - font-family: "Font Awesome 5 Brands" !important; -} - -/* don't display links in code chunks when printing */ -/* source: https://stackoverflow.com/a/10781533 */ -@media print { - code a:link:after, code a:visited:after { - content: ""; - } -} diff --git a/docs/pkgdown.js b/docs/pkgdown.js deleted file mode 100644 index 7e7048f..0000000 --- a/docs/pkgdown.js +++ /dev/null @@ -1,108 +0,0 @@ -/* http://gregfranko.com/blog/jquery-best-practices/ */ -(function($) { - $(function() { - - $('.navbar-fixed-top').headroom(); - - $('body').css('padding-top', $('.navbar').height() + 10); - $(window).resize(function(){ - $('body').css('padding-top', $('.navbar').height() + 10); - }); - - $('[data-toggle="tooltip"]').tooltip(); - - var cur_path = paths(location.pathname); - var links = $("#navbar ul li a"); - var max_length = -1; - var pos = -1; - for (var i = 0; i < links.length; i++) { - if (links[i].getAttribute("href") === "#") - continue; - // Ignore external links - if (links[i].host !== location.host) - continue; - - var nav_path = paths(links[i].pathname); - - var length = prefix_length(nav_path, cur_path); - if (length > max_length) { - max_length = length; - pos = i; - } - } - - // Add class to parent
  • , and enclosing
  • if in dropdown - if (pos >= 0) { - var menu_anchor = $(links[pos]); - menu_anchor.parent().addClass("active"); - menu_anchor.closest("li.dropdown").addClass("active"); - } - }); - - function paths(pathname) { - var pieces = pathname.split("/"); - pieces.shift(); // always starts with / - - var end = pieces[pieces.length - 1]; - if (end === "index.html" || end === "") - pieces.pop(); - return(pieces); - } - - // Returns -1 if not found - function prefix_length(needle, haystack) { - if (needle.length > haystack.length) - return(-1); - - // Special case for length-0 haystack, since for loop won't run - if (haystack.length === 0) { - return(needle.length === 0 ? 0 : -1); - } - - for (var i = 0; i < haystack.length; i++) { - if (needle[i] != haystack[i]) - return(i); - } - - return(haystack.length); - } - - /* Clipboard --------------------------*/ - - function changeTooltipMessage(element, msg) { - var tooltipOriginalTitle=element.getAttribute('data-original-title'); - element.setAttribute('data-original-title', msg); - $(element).tooltip('show'); - element.setAttribute('data-original-title', tooltipOriginalTitle); - } - - if(ClipboardJS.isSupported()) { - $(document).ready(function() { - var copyButton = ""; - - $(".examples, div.sourceCode").addClass("hasCopyButton"); - - // Insert copy buttons: - $(copyButton).prependTo(".hasCopyButton"); - - // Initialize tooltips: - $('.btn-copy-ex').tooltip({container: 'body'}); - - // Initialize clipboard: - var clipboardBtnCopies = new ClipboardJS('[data-clipboard-copy]', { - text: function(trigger) { - return trigger.parentNode.textContent; - } - }); - - clipboardBtnCopies.on('success', function(e) { - changeTooltipMessage(e.trigger, 'Copied!'); - e.clearSelection(); - }); - - clipboardBtnCopies.on('error', function() { - changeTooltipMessage(e.trigger,'Press Ctrl+C or Command+C to copy'); - }); - }); - } -})(window.jQuery || window.$) diff --git a/docs/pkgdown.yml b/docs/pkgdown.yml deleted file mode 100644 index f852a13..0000000 --- a/docs/pkgdown.yml +++ /dev/null @@ -1,6 +0,0 @@ -pandoc: 2.9.2.1 -pkgdown: 1.5.1 -pkgdown_sha: ~ -articles: [] -last_built: 2020-05-28T09:16Z - diff --git a/docs/reference/auto_prior.html b/docs/reference/auto_prior.html deleted file mode 100644 index 362ac3a..0000000 --- a/docs/reference/auto_prior.html +++ /dev/null @@ -1,264 +0,0 @@ - - - - - - - - -Create default priors for brms-models — auto_prior • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    This function creates default priors for brms-regression - models, based on the same automatic prior-scale adjustment as in - rstanarm.

    -
    - -
    auto_prior(formula, data, gaussian, locations = NULL)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - -
    formula

    A formula describing the model, which just needs to contain -the model terms, but no notation of interaction, splines etc. Usually, -you want only those predictors in the formula, for which automatic -priors should be generated. Add informative priors afterwards to the -returned brmsprior-object.

    data

    The data that will be used to fit the model.

    gaussian

    Logical, if the outcome is gaussian or not.

    locations

    A numeric vector with location values for the priors. If -locations = NULL, 0 is used as location parameter.

    - -

    Value

    - -

    A brmsprior-object.

    -

    Details

    - -

    auto_prior() is a small, convenient function to create - some default priors for brms-models with automatically adjusted prior - scales, in a similar way like rstanarm does. The default scale for - the intercept is 10, for coefficients 2.5. If the outcome is gaussian, - both scales are multiplied with sd(y). Then, for categorical - variables, nothing more is changed. For numeric variables, the scales - are divided by the standard deviation of the related variable. -

    - All prior distributions are normal distributions. auto_prior() - is intended to quickly create default priors with feasible scales. If - more precise definitions of priors is necessary, this needs to be done - directly with brms-functions like set_prior().

    -

    Note

    - -

    As auto_prior() also sets priors on the intercept, the model - formula used in brms::brm() must be rewritten to something like - y ~ 0 + intercept ..., see set_prior.

    - -

    Examples

    -
    library(sjmisc) -data(efc) -efc$c172code <- as.factor(efc$c172code) -efc$c161sex <- to_label(efc$c161sex) - -mf <- formula(neg_c_7 ~ c161sex + c160age + c172code) - -if (requireNamespace("brms", quietly = TRUE)) - auto_prior(mf, efc, TRUE)
    #> prior class coef group resp dpar nlpar bound -#> 1 normal(0, 38.96) Intercept -#> 2 normal(0, 9.74) b c161sexFemale -#> 3 normal(0, 0.73) b c160age -#> 4 normal(0, 9.74) b c172code2 -#> 5 normal(0, 9.74) b c172code3
    -## compare to -# library(rstanarm) -# m <- stan_glm(mf, data = efc, chains = 2, iter = 200) -# ps <- prior_summary(m) -# ps$prior_intercept$adjusted_scale -# ps$prior$adjusted_scale - -## usage -# ap <- auto_prior(mf, efc, TRUE) -# brm(mf, data = efc, priors = ap) - -# add informative priors -mf <- formula(neg_c_7 ~ c161sex + c172code) - -if (requireNamespace("brms", quietly = TRUE)) { - auto_prior(mf, efc, TRUE) + - brms::prior(normal(.1554, 40), class = "b", coef = "c160age") -}
    #> prior class coef group resp dpar nlpar bound -#> 1 normal(0, 38.95) Intercept -#> 2 normal(0, 9.74) b c161sexFemale -#> 3 normal(0, 9.74) b c172code2 -#> 4 normal(0, 9.74) b c172code3 -#> 5 normal(0.1554, 40) b c160age
    -# example with binary response -efc$neg_c_7d <- ifelse(efc$neg_c_7 < median(efc$neg_c_7, na.rm = TRUE), 0, 1) -mf <- formula(neg_c_7d ~ c161sex + c160age + c172code + e17age) - -if (requireNamespace("brms", quietly = TRUE)) - auto_prior(mf, efc, FALSE)
    #> prior class coef group resp dpar nlpar bound -#> 1 normal(0, 10) Intercept -#> 2 normal(0, 2.5) b c161sexFemale -#> 3 normal(0, 0.19) b c160age -#> 4 normal(0, 2.5) b c172code2 -#> 5 normal(0, 2.5) b c172code3 -#> 6 normal(0, 0.31) b e17age
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/boot_ci.html b/docs/reference/boot_ci.html deleted file mode 100644 index 05414ea..0000000 --- a/docs/reference/boot_ci.html +++ /dev/null @@ -1,308 +0,0 @@ - - - - - - - - -Standard error and confidence intervals for bootstrapped estimates — boot_ci • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Compute nonparametric bootstrap estimate, standard error, - confidence intervals and p-value for a vector of bootstrap - replicate estimates.

    -
    - -
    boot_ci(data, ..., method = c("dist", "quantile"), ci.lvl = 0.95)
    -
    -boot_se(data, ...)
    -
    -boot_p(data, ...)
    -
    -boot_est(data, ...)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - -
    data

    A data frame that containts the vector with bootstrapped -estimates, or directly the vector (see 'Examples').

    ...

    Optional, unquoted names of variables with bootstrapped estimates. -Required, if either data is a data frame (and no vector), -and only selected variables from data should be processed. -You may also use functions like : or tidyselect's -select_helpers.

    method

    Character vector, indicating if confidence intervals should be -based on bootstrap standard error, multiplied by the value of the -quantile function of the t-distribution (default), or on sample -quantiles of the bootstrapped values. See 'Details' in boot_ci(). -May be abbreviated.

    ci.lvl

    Numeric, the level of the confidence intervals.

    - -

    Value

    - -

    A tibble with either bootstrap estimate, - standard error, the lower and upper confidence intervals or the - p-value for all bootstrapped estimates.

    -

    Details

    - -

    The methods require one or more vectors of bootstrap replicate estimates - as input.

      -
    • boot_est() returns the bootstrapped estimate, simply by - computing the mean value of all bootstrap estimates.

    • -
    • boot_se() computes the nonparametric bootstrap standard - error by calculating the standard deviation of the input vector.

    • -
    • The mean value of the input vector and its standard error is used - by boot_ci() to calculate the lower and upper confidence - interval, assuming a t-distribution of bootstrap estimate replicates - (for method = "dist", the default, which is - mean(x) +/- qt(.975, df = length(x) - 1) * sd(x)); for - method = "quantile", 95% sample quantiles are used to compute - the confidence intervals (quantile(x, probs = c(.025, .975))). - Use ci.lvl to change the level for the confidence interval.

    • -
    • P-values from boot_p() are also based on t-statistics, - assuming normal distribution.

    • -
    - -

    References

    - -

    Carpenter J, Bithell J. Bootstrap confdence intervals: when, which, what? A practical guide for medical statisticians. Statist. Med. 2000; 19:1141-1164

    -

    See also

    - -

    bootstrap to generate nonparametric bootstrap samples.

    - -

    Examples

    -
    library(dplyr) -library(purrr) -data(efc) -bs <- bootstrap(efc, 100) - -# now run models for each bootstrapped sample -bs$models <- map(bs$strap, ~lm(neg_c_7 ~ e42dep + c161sex, data = .x)) - -# extract coefficient "dependency" and "gender" from each model -bs$dependency <- map_dbl(bs$models, ~coef(.x)[2]) -bs$gender <- map_dbl(bs$models, ~coef(.x)[3]) - -# get bootstrapped confidence intervals -boot_ci(bs$dependency)
    #> term conf.low conf.high -#> 1 x 1.320903 1.77407
    -# compare with model fit -fit <- lm(neg_c_7 ~ e42dep + c161sex, data = efc) -confint(fit)[2, ]
    #> 2.5 % 97.5 % -#> 1.292945 1.796430
    -# alternative function calls. -boot_ci(bs$dependency)
    #> term conf.low conf.high -#> 1 x 1.320903 1.77407
    boot_ci(bs, dependency)
    #> term conf.low conf.high -#> 1 dependency 1.320903 1.77407
    boot_ci(bs, dependency, gender)
    #> term conf.low conf.high -#> 1 dependency 1.3209034 1.7740701 -#> 2 gender -0.1016646 0.9788897
    boot_ci(bs, dependency, gender, method = "q")
    #> term conf.low conf.high -#> 1 dependency 1.30501832 1.763890 -#> 2 gender -0.07012629 0.916922
    - -# compare coefficients -mean(bs$dependency)
    #> [1] 1.547487
    boot_est(bs$dependency)
    #> term estimate -#> 1 x 1.547487
    coef(fit)[2]
    #> e42dep -#> 1.544687
    - -# bootstrap() and boot_ci() work fine within pipe-chains -efc %>% - bootstrap(100) %>% - mutate( - models = map(strap, ~lm(neg_c_7 ~ e42dep + c161sex, data = .x)), - dependency = map_dbl(models, ~coef(.x)[2]) - ) %>% - boot_ci(dependency)
    #> term conf.low conf.high -#> 1 dependency 1.263093 1.779068
    -# check p-value -boot_p(bs$gender)
    #> term p.value -#> 1 x 0.1103975
    summary(fit)$coefficients[3, ]
    #> Estimate Std. Error t value Pr(>|t|) -#> 0.4339069 0.2818786 1.5393398 0.1240780
    -if (FALSE) { -# 'spread_coef()' from the 'sjmisc'-package makes it easy to generate -# bootstrapped statistics like confidence intervals or p-values -library(dplyr) -library(sjmisc) -efc %>% - # generate bootstrap replicates - bootstrap(100) %>% - # apply lm to all bootstrapped data sets - mutate( - models = map(strap, ~lm(neg_c_7 ~ e42dep + c161sex + c172code, data = .x)) - ) %>% - # spread model coefficient for all 100 models - spread_coef(models) %>% - # compute the CI for all bootstrapped model coefficients - boot_ci(e42dep, c161sex, c172code) - -# or... -efc %>% - # generate bootstrap replicates - bootstrap(100) %>% - # apply lm to all bootstrapped data sets - mutate( - models = map(strap, ~lm(neg_c_7 ~ e42dep + c161sex + c172code, data = .x)) - ) %>% - # spread model coefficient for all 100 models - spread_coef(models, append = FALSE) %>% - # compute the CI for all bootstrapped model coefficients - boot_ci()}
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/bootstrap.html b/docs/reference/bootstrap.html deleted file mode 100644 index f6e496b..0000000 --- a/docs/reference/bootstrap.html +++ /dev/null @@ -1,304 +0,0 @@ - - - - - - - - -Generate nonparametric bootstrap replications — bootstrap • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Generates n bootstrap samples of data and - returns the bootstrapped data frames as list-variable.

    -
    - -
    bootstrap(data, n, size)
    - -

    Arguments

    - - - - - - - - - - - - - - -
    data

    A data frame.

    n

    Number of bootstraps to be generated.

    size

    Optional, size of the bootstrap samples. May either be a number -between 1 and nrow(data) or a value between 0 and 1 to sample -a proportion of observations from data (see 'Examples').

    - -

    Value

    - -

    A data frame with one column: a list-variable - strap, which contains resample-objects of class sj_resample. - These resample-objects are lists with three elements:

      -
    1. the original data frame, data

    2. -
    3. the rownmumbers id, i.e. rownumbers of data, indicating the resampled rows with replacement

    4. -
    5. the resample.id, indicating the index of the resample (i.e. the position of the sj_resample-object in the list strap)

    6. -
    - -

    Details

    - -

    By default, each bootstrap sample has the same number of observations - as data. To generate bootstrap samples without resampling - same observations (i.e. sampling without replacement), use - size to get bootstrapped data with a specific number - of observations. However, specifying the size-argument is much - less memory-efficient than the bootstrap with replacement. Hence, - it is recommended to ignore the size-argument, if it is - not really needed.

    -

    Note

    - -

    This function applies nonparametric bootstrapping, i.e. the function - draws samples with replacement. -

    - There is an as.data.frame- and a print-method to get or - print the resampled data frames. See 'Examples'. The as.data.frame- - method automatically applies whenever coercion is done because a data - frame is required as input. See 'Examples' in boot_ci.

    -

    See also

    - -

    boot_ci to calculate confidence intervals from - bootstrap samples.

    - -

    Examples

    -
    data(efc) -bs <- bootstrap(efc, 5) - -# now run models for each bootstrapped sample -lapply(bs$strap, function(x) lm(neg_c_7 ~ e42dep + c161sex, data = x))
    #> [[1]] -#> -#> Call: -#> lm(formula = neg_c_7 ~ e42dep + c161sex, data = x) -#> -#> Coefficients: -#> (Intercept) e42dep c161sex -#> 6.9036 1.4385 0.4329 -#> -#> -#> [[2]] -#> -#> Call: -#> lm(formula = neg_c_7 ~ e42dep + c161sex, data = x) -#> -#> Coefficients: -#> (Intercept) e42dep c161sex -#> 7.8918 1.3693 -0.1371 -#> -#> -#> [[3]] -#> -#> Call: -#> lm(formula = neg_c_7 ~ e42dep + c161sex, data = x) -#> -#> Coefficients: -#> (Intercept) e42dep c161sex -#> 7.7762 1.3265 0.1469 -#> -#> -#> [[4]] -#> -#> Call: -#> lm(formula = neg_c_7 ~ e42dep + c161sex, data = x) -#> -#> Coefficients: -#> (Intercept) e42dep c161sex -#> 5.4334 1.7106 0.7536 -#> -#> -#> [[5]] -#> -#> Call: -#> lm(formula = neg_c_7 ~ e42dep + c161sex, data = x) -#> -#> Coefficients: -#> (Intercept) e42dep c161sex -#> 6.3228 1.5540 0.5277 -#> -#>
    -# generate bootstrap samples with 600 observations for each sample -bs <- bootstrap(efc, 5, 600) - -# generate bootstrap samples with 70% observations of the original sample size -bs <- bootstrap(efc, 5, .7) - -# compute standard error for a simple vector from bootstraps -# use the `as.data.frame()`-method to get the resampled -# data frame -bs <- bootstrap(efc, 100) -bs$c12hour <- unlist(lapply(bs$strap, function(x) { - mean(as.data.frame(x)$c12hour, na.rm = TRUE) -})) - -# or as tidyverse-approach -if (require("dplyr") && require("purrr")) { - bs <- efc %>% - bootstrap(100) %>% - mutate( - c12hour = map_dbl(strap, ~mean(as.data.frame(.x)$c12hour, na.rm = TRUE)) - ) - - # bootstrapped standard error - boot_se(bs, c12hour) -}
    #> Loading required package: dplyr
    #> -#> Attaching package: 'dplyr'
    #> The following objects are masked from 'package:stats': -#> -#> filter, lag
    #> The following objects are masked from 'package:base': -#> -#> intersect, setdiff, setequal, union
    #> Loading required package: purrr
    #> -#> Attaching package: 'purrr'
    #> The following object is masked from 'package:sjmisc': -#> -#> is_empty
    #> term std.err -#> 1 c12hour 1.6688
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/check_assumptions.html b/docs/reference/check_assumptions.html deleted file mode 100644 index b1a7dd5..0000000 --- a/docs/reference/check_assumptions.html +++ /dev/null @@ -1,431 +0,0 @@ - - - - - - - - -Check model assumptions — check_assumptions • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -
      -
    • outliers() detects outliers in (generalized) linear models.

    • -
    • heteroskedastic() checks a linear model for (non-)constant error variance.

    • -
    • autocorrelation() checks for independence of errors.

    • -
    • normality() checks linear models for (non-)normality of residuals.

    • -
    • multicollin() checks predictors of linear models for multicollinearity.

    • -
    • check_assumptions() checks all of the above assumptions.

    • -
    - -
    - -
    check_assumptions(x, model.column = NULL, as.logical = FALSE, ...)
    -
    -outliers(x, iterations = 5)
    -
    -heteroskedastic(x, model.column = NULL)
    -
    -autocorrelation(x, model.column = NULL, ...)
    -
    -normality(x, model.column = NULL)
    -
    -multicollin(x, model.column = NULL)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - -
    x

    Fitted lm (for outliers(), may also be a glm model), -or a (nested) data frame with a list-variable that contains fitted model -objects.

    model.column

    Name or index of the list-variable that contains the fitted -model objects. Only applies, if x is a nested data frame (e.g -with models fitted to bootstrap replicates).

    as.logical

    Logical, if TRUE, the values returned by -check_assumptions() are TRUE or FALSE, indicating -whether each violation of model assumotion holds true or not. If -FALSE (the default), the p-value of the respective test-statistics -is returned.

    ...

    Other arguments, passed down to durbinWatsonTest.

    iterations

    Numeric, indicates the number of iterations to remove -outliers.

    - -

    Value

    - -

    A data frame with the respective statistics.

    - -

    Details

    - -

    These functions are wrappers that compute various test statistics, - however, each of them returns a tibble instead of a list of values. - Furthermore, all functions can also be applied to multiples models - in stored in list-variables (see 'Examples'). -

    - outliers() wraps outlierTest and iteratively - removes outliers for iterations times, or if the r-squared value - (for glm: the AIC) did not improve after removing outliers. The function - returns a tibble with r-squared and AIC statistics for the original - and updated model, as well as the update model itself ($updated.model), - the number ($removed.count) and indices of the removed observations - ($removed.obs). -

    - heteroskedastic() wraps ncvTest and returns - the p-value of the test statistics as tibble. A p-value < 0.05 indicates - a non-constant variance (heteroskedasticity). -

    - autocorrelation() wraps durbinWatsonTest - and returns the p-value of the test statistics as tibble. A p-value - < 0.05 indicates autocorrelated residuals. In such cases, robust - standard errors (see robust return more accurate results - for the estimates, or maybe a mixed model with error term for the - cluster groups should be used. -

    - normality() calls shapiro.test - and checks the standardized residuals for normal distribution. - The p-value of the test statistics is returned as tibble. A p-value - < 0.05 indicates a significant deviation from normal distribution. - Note that this formal test almost always yields significant results - for the distribution of residuals and visual inspection (e.g. qqplots) - are preferable (see plot_model with - type = "diag"). -

    - multicollin() wraps vif and returns - the maximum vif-value from a model as tibble. If this value is - larger than about 4, multicollinearity exists, else not. - In case of multicollinearity, the names of independent - variables that vioalte contribute to multicollinearity are printed - to the console. -

    - check_assumptions() runs all of the above tests and returns - a tibble with all test statistics included. In case the p-values - are too confusing, use the as.logical argument, where all - p-values are replaced with either TRUE (in case of violation) - or FALSE (in case of model conforms to assumption of linar - regression).

    - -

    Note

    - -

    These formal tests are very strict and in most cases violation of model - assumptions are alerted, though the model is actually ok. It is - preferable to check model assumptions based on visual inspection - (see plot_model with type = "diag").

    - - -

    Examples

    -
    data(efc) - -fit <- lm(barthtot ~ c160age + c12hour + c161sex + c172code, data = efc) -outliers(fit)
    #> No outliers detected.
    heteroskedastic(fit)
    #> Heteroscedasticity (non-constant error variance) detected: p = 0.000
    #> heteroskedastic -#> 1 3.885808e-07
    autocorrelation(fit)
    #> Autocorrelated residuals detected: p = 0.000
    #> autocorrelation -#> 1 0
    normality(fit)
    #> Non-normality of residuals detected: p = 0.000
    #> non.normality -#> 1 1.535796e-13
    check_assumptions(fit)
    #> -#> # Checking Model-Assumptions -#> -#> Model: barthtot ~ c160age + c12hour + c161sex + c172code -#> -#> violated statistic -#> Heteroskedasticity yes p = 0.000 -#> Non-normal residuals yes p = 0.000 -#> Autocorrelated residuals yes p = 0.000 -#> Multicollinearity no vif = 1.153
    -fit <- lm(barthtot ~ c160age + c12hour + c161sex + c172code + neg_c_7, - data = efc) -outliers(fit)
    #> 2 outliers removed in updated model.
    #> models adjusted.r2 aic -#> 1 original 0.3458095 7487.639 -#> 2 updated 0.3530485 7468.980
    check_assumptions(fit, as.logical = TRUE)
    #> heteroskedasticity multicollinearity non.normal.resid autocorrelation -#> 1 TRUE FALSE TRUE TRUE
    -# apply function to multiple models in list-variable -library(purrr) -library(dplyr) -tmp <- efc %>% - bootstrap(50) %>% - mutate( - models = map(strap, ~lm(neg_c_7 ~ e42dep + c12hour + c161sex, data = .x)) - ) - -# for list-variables, argument 'model.column' is the -# quoted name of the list-variable with fitted models -tmp %>% normality("models")
    #> non.normality -#> 1 3.230058e-19 -#> 2 3.827347e-16 -#> 3 9.098247e-22 -#> 4 9.981165e-19 -#> 5 6.428575e-18 -#> 6 3.224082e-20 -#> 7 2.118794e-19 -#> 8 1.288646e-16 -#> 9 2.328003e-19 -#> 10 8.999720e-21 -#> 11 6.353047e-21 -#> 12 1.026013e-19 -#> 13 2.390006e-19 -#> 14 6.701265e-19 -#> 15 2.551566e-19 -#> 16 3.057700e-18 -#> 17 2.117815e-19 -#> 18 1.077834e-17 -#> 19 7.594577e-18 -#> 20 9.239676e-19 -#> 21 3.149844e-21 -#> 22 5.519879e-20 -#> 23 3.637194e-18 -#> 24 1.764174e-21 -#> 25 6.623029e-19 -#> 26 1.181069e-19 -#> 27 3.168530e-19 -#> 28 4.854560e-18 -#> 29 2.001162e-20 -#> 30 7.352369e-22 -#> 31 1.018042e-20 -#> 32 7.376915e-19 -#> 33 1.207461e-18 -#> 34 4.070908e-16 -#> 35 7.510733e-18 -#> 36 2.049412e-20 -#> 37 1.243810e-19 -#> 38 2.813157e-19 -#> 39 3.095559e-22 -#> 40 8.518521e-21 -#> 41 5.261510e-16 -#> 42 1.156834e-18 -#> 43 5.360721e-21 -#> 44 3.306733e-18 -#> 45 8.478782e-20 -#> 46 1.000268e-20 -#> 47 3.684449e-21 -#> 48 1.301398e-17 -#> 49 3.625559e-21 -#> 50 8.619752e-18
    tmp %>% heteroskedastic("models")
    #> heteroskedastic -#> 1 1.117145e-06 -#> 2 4.136378e-16 -#> 3 2.304193e-11 -#> 4 2.303091e-13 -#> 5 2.989163e-09 -#> 6 1.617715e-06 -#> 7 2.462472e-06 -#> 8 5.650847e-11 -#> 9 1.949355e-05 -#> 10 9.487583e-12 -#> 11 8.865950e-05 -#> 12 1.001106e-11 -#> 13 9.380742e-09 -#> 14 2.692178e-11 -#> 15 2.777129e-12 -#> 16 1.156712e-08 -#> 17 1.142231e-09 -#> 18 3.003215e-05 -#> 19 3.513974e-10 -#> 20 8.914693e-07 -#> 21 5.075324e-07 -#> 22 1.443102e-04 -#> 23 3.889332e-08 -#> 24 2.962249e-05 -#> 25 1.108236e-11 -#> 26 1.494601e-11 -#> 27 9.264544e-11 -#> 28 2.394459e-08 -#> 29 3.738415e-03 -#> 30 3.710447e-07 -#> 31 2.980297e-12 -#> 32 4.978182e-15 -#> 33 1.403946e-07 -#> 34 1.833919e-14 -#> 35 2.311357e-05 -#> 36 1.506452e-13 -#> 37 3.740443e-08 -#> 38 2.266819e-07 -#> 39 9.948937e-11 -#> 40 8.844100e-08 -#> 41 3.306678e-14 -#> 42 4.618659e-10 -#> 43 4.902720e-07 -#> 44 8.314707e-12 -#> 45 8.818074e-11 -#> 46 8.447289e-10 -#> 47 5.206397e-09 -#> 48 3.063647e-15 -#> 49 1.658216e-08 -#> 50 1.059808e-12
    -# Durbin-Watson-Test from package 'car' takes a little bit longer due -# to simulation of p-values... -
    # NOT RUN { -tmp %>% check_assumptions("models", as.logical = TRUE, reps = 100) -# }
    -
    -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/chisq_gof.html b/docs/reference/chisq_gof.html deleted file mode 100644 index 4a63755..0000000 --- a/docs/reference/chisq_gof.html +++ /dev/null @@ -1,236 +0,0 @@ - - - - - - - - -Compute model quality — chisq_gof • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    For logistic regression models, performs a Chi-squared - goodness-of-fit-test.

    -
    - -
    chisq_gof(x, prob = NULL, weights = NULL)
    - -

    Arguments

    - - - - - - - - - - - - - - -
    x

    A numeric vector or a glm-object.

    prob

    Vector of probabilities (indicating the population probabilities) -of the same length as x's amount of categories / factor levels. -Use nrow(table(x)) to determine the amount of necessary values -for prob. Only used, when x is a vector, and not a -glm-object.

    weights

    Vector with weights, used to weight x.

    - -

    Value

    - -

    For vectors, returns the object of the computed chisq.test. - For glm-objects, an object of class chisq_gof with - following values: p.value, the p-value for the goodness-of-fit test; - z.score, the standardized z-score for the goodness-of-fit test; - rss, the residual sums of squares term and chisq, the pearson - chi-squared statistic.

    -

    Details

    - -

    For vectors, this function is a convenient function for the - chisq.test(), performing goodness-of-fit test. For - glm-objects, this function performs a goodness-of-fit test. - A well-fitting model shows no significant difference between the - model and the observed data, i.e. the reported p-values should be - greater than 0.05.

    -

    References

    - -

    Hosmer, D. W., & Lemeshow, S. (2000). Applied Logistic Regression. Hoboken, NJ, USA: John Wiley & Sons, Inc. doi: 10.1002/0471722146

    - -

    Examples

    -
    data(efc) -efc$neg_c_7d <- ifelse(efc$neg_c_7 < median(efc$neg_c_7, na.rm = TRUE), 0, 1) -m <- glm( - neg_c_7d ~ c161sex + barthtot + c172code, - data = efc, - family = binomial(link = "logit") -) - -# goodness-of-fit test for logistic regression -chisq_gof(m)
    #> -#> # Chi-squared Goodness-of-Fit Test -#> -#> Chi-squared: 852.765 -#> z-score: 1.025 -#> p-value: 0.305 -#>
    #> Summary: model seems to fit well.
    -# goodness-of-fit test for vectors against probabilities -# differing from population -chisq_gof(efc$e42dep, c(0.3,0.2,0.22,0.28))
    #> -#> Chi-squared test for given probabilities -#> -#> data: dummy -#> X-squared = 234.76, df = 3, p-value < 2.2e-16 -#>
    -# equal to population -chisq_gof(efc$e42dep, prop.table(table(efc$e42dep)))
    #> -#> Chi-squared test for given probabilities -#> -#> data: dummy -#> X-squared = 0, df = 3, p-value = 1 -#>
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/cod.html b/docs/reference/cod.html deleted file mode 100644 index 7fc5bde..0000000 --- a/docs/reference/cod.html +++ /dev/null @@ -1,347 +0,0 @@ - - - - - - - - -Goodness-of-fit measures for regression models — cod • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    Compute Goodness-of-fit measures for various regression models, - including mixed and Bayesian regression models.

    - -
    - -
    cod(x)
    -
    -r2(x, ...)
    -
    -# S3 method for lme
    -r2(x, n = NULL, ...)
    -
    -# S3 method for stanreg
    -r2(x, loo = FALSE, ...)
    -
    -# S3 method for brmsfit
    -r2(x, loo = FALSE, ...)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - -
    x

    Fitted model of class lm, glm, merMod, -glmmTMB, lme, plm, stanreg or brmsfit. -For method cod(), only a glm with binrary response.

    ...

    Currently not used.

    n

    Optional, an lme object, representing the fitted null-model -(unconditional model) to x. If n is given, the pseudo-r-squared -for random intercept and random slope variances are computed -(Kwok et al. 2008) as well as the Omega squared value -(Xu 2003). See 'Examples' and 'Details'.

    loo

    Logical, if TRUE and x is a stanreg or -brmsfit object, a LOO-adjusted r-squared is calculated. Else, -a rather "unadjusted" r-squared will be returned by calling -rstantools::bayes_R2().

    - -

    Value

    - -

    For r2(), depending on the model, returns:

      -
    • For linear models, the r-squared and adjusted r-squared values.

    • -
    • For mixed models, the marginal and conditional r-squared values.

    • -
    • For glm objects, Cox & Snell's and Nagelkerke's pseudo r-squared values.

    • -
    • For brmsfit or stanreg objects, the Bayesian version of r-squared is computed, calling rstantools::bayes_R2().

    • -
    • If loo = TRUE, for brmsfit or stanreg objects a LOO-adjusted version of r-squared is returned.

    • -
    • Models that are not currently supported return NULL.

    • -

    For cod(), returns the D Coefficient of Discrimination, - also known as Tjur's R-squared value.

    - -

    Details

    - -

    For linear models, the r-squared and adjusted r-squared value is returned, - as provided by the summary-function. -

    - For mixed models (from lme4 or glmmTMB) marginal and - conditional r-squared values are calculated, based on - Nakagawa et al. 2017. The distributional variance - (or observation-level variance) is based on lognormal approximation, - log(1+var(x)/mu^2). -

    - For lme-models, an r-squared approximation by computing the - correlation between the fitted and observed values, as suggested by - Byrnes (2008), is returned as well as a simplified version of - the Omega-squared value (1 - (residual variance / response variance), - Xu (2003), Nakagawa, Schielzeth 2013), unless n - is specified. -

    - If n is given, for lme-models pseudo r-squared measures based - on the variances of random intercept (tau 00, between-group-variance) - and random slope (tau 11, random-slope-variance), as well as the - r-squared statistics as proposed by Snijders and Bosker 2012 and - the Omega-squared value (1 - (residual variance full model / residual - variance null model)) as suggested by Xu (2003) are returned. -

    - For generalized linear models, Cox & Snell's and Nagelkerke's - pseudo r-squared values are returned. -

    - The ("unadjusted") r-squared value and its standard error for - brmsfit or stanreg objects are robust measures, i.e. - the median is used to compute r-squared, and the median absolute - deviation as the measure of variability. If loo = TRUE, - a LOO-adjusted r-squared is calculated, which comes conceptionally - closer to an adjusted r-squared measure.

    - -

    Note

    - -
    -
    cod()

    This method calculates the Coefficient of Discrimination D - for generalized linear (mixed) models for binary data. It is - an alternative to other Pseudo-R-squared values like Nakelkerke's - R2 or Cox-Snell R2. The Coefficient of Discrimination D - can be read like any other (Pseudo-)R-squared value.

    -
    r2()

    For mixed models, the marginal r-squared considers only the variance - of the fixed effects, while the conditional r-squared takes both - the fixed and random effects into account. -

    - For lme-objects, if n is given, the Pseudo-R2 statistic - is the proportion of explained variance in the random effect after - adding co-variates or predictors to the model, or in short: the - proportion of the explained variance in the random effect of the - full (conditional) model x compared to the null (unconditional) - model n. -

    - The Omega-squared statistics, if n is given, is 1 - the proportion - of the residual variance of the full model compared to the null model's - residual variance, or in short: the the proportion of the residual - variation explained by the covariates. -

    - Alternative ways to assess the "goodness-of-fit" is to compare the ICC - of the null model with the ICC of the full model (see icc).

    -
    - -

    References

    - -
      -
    • DRAFT r-sig-mixed-models FAQ

    • -
    • Bolker B et al. (2017): GLMM FAQ

    • -
    • Byrnes, J. 2008. Re: Coefficient of determination (R^2) when using lme() (https://stat.ethz.ch/pipermail/r-sig-mixed-models/2008q2/000713.html)

    • -
    • Kwok OM, Underhill AT, Berry JW, Luo W, Elliott TR, Yoon M. 2008. Analyzing Longitudinal Data with Multilevel Models: An Example with Individuals Living with Lower Extremity Intra-Articular Fractures. Rehabilitation Psychology 53(3): 370-86. doi: 10.1037/a0012765

    • -
    • Nakagawa S, Schielzeth H. 2013. A general and simple method for obtaining R2 from generalized linear mixed-effects models. Methods in Ecology and Evolution, 4(2):133-142. doi: 10.1111/j.2041-210x.2012.00261.x

    • -
    • Nakagawa S, Johnson P, Schielzeth H (2017) The coefficient of determination R2 and intra-class correlation coefficient from generalized linear mixed-effects models revisted and expanded. J. R. Soc. Interface 14. doi: 10.1098/rsif.2017.0213

    • -
    • Rabe-Hesketh S, Skrondal A. 2012. Multilevel and longitudinal modeling using Stata. 3rd ed. College Station, Tex: Stata Press Publication

    • -
    • Raudenbush SW, Bryk AS. 2002. Hierarchical linear models: applications and data analysis methods. 2nd ed. Thousand Oaks: Sage Publications

    • -
    • Snijders TAB, Bosker RJ. 2012. Multilevel analysis: an introduction to basic and advanced multilevel modeling. 2nd ed. Los Angeles: Sage

    • -
    • Xu, R. 2003. Measuring explained variation in linear mixed effects models. Statist. Med. 22:3527-3541. doi: 10.1002/sim.1572

    • -
    • Tjur T. 2009. Coefficients of determination in logistic regression models - a new proposal: The coefficient of discrimination. The American Statistician, 63(4): 366-372

    • -
    - - -

    Examples

    -
    data(efc) - -# Tjur's R-squared value -efc$services <- ifelse(efc$tot_sc_e > 0, 1, 0) -fit <- glm(services ~ neg_c_7 + c161sex + e42dep, - data = efc, family = binomial(link = "logit")) -cod(fit)
    #> -#> R-Squared for (Generalized) Linear (Mixed) Model -#> -#> Tjur's D: 0.023 -#>
    -library(lme4)
    #> Loading required package: Matrix
    fit <- lmer(Reaction ~ Days + (Days | Subject), sleepstudy) -r2(fit)
    #> -#> R-Squared for (Generalized) Linear (Mixed) Model -#> -#> Family : gaussian (identity) -#> Formula: ~Days | Subject Reaction ~ Days NA -#> -#> Marginal R2: 0.279 -#> Conditional R2: 0.799 -#>
    -fit <- lm(barthtot ~ c160age + c12hour, data = efc) -r2(fit)
    #> -#> R-Squared for (Generalized) Linear (Mixed) Model -#> -#> R-squared: 0.256 -#> adjusted R-squared: 0.254 -#>
    -# Pseudo-R-squared values -fit <- glm(services ~ neg_c_7 + c161sex + e42dep, - data = efc, family = binomial(link = "logit")) -r2(fit)
    #> -#> R-Squared for (Generalized) Linear (Mixed) Model -#> -#> Cox & Snell's R-squared: 0.023 -#> Nagelkerke's R-squared: 0.030 -#>
    -
    -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/converge_ok.html b/docs/reference/converge_ok.html deleted file mode 100644 index b64e312..0000000 --- a/docs/reference/converge_ok.html +++ /dev/null @@ -1,261 +0,0 @@ - - - - - - - - -Convergence test for mixed effects models — converge_ok • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    converge_ok() provides an alternative convergence test for - merMod-objects; is_singular() checks - post-fitting convergence warnings. If the model fit is singular, - warning about negative eigenvalues of the Hessian can most likely - be ignored.

    - -
    - -
    converge_ok(x, tolerance = 0.001)
    -
    -is_singular(x, tolerance = 1e-05, ...)
    - -

    Arguments

    - - - - - - - - - - - - - - -
    x

    A merMod-object. For is_singluar(), may also be a -glmmTMB-object.

    tolerance

    Indicates up to which value the convergence result is -accepted. The smaller tolerance is, the stricter the test -will be.

    ...

    Currently not used.

    - -

    Value

    - -

    For converge_ok(), a logical vector, which is TRUE if - convergence is fine and FALSE if convergence is suspicious. - Additionally, the convergence value is returned as return value's name. - is_singluar() returns TRUE if the model fit is singular.

    - -

    Details

    - -

    converge_ok() provides an alternative convergence test for - merMod-objects, as discussed - here - and suggested by Ben Bolker in - this comment. -

    - If a model is "singular", this means that some dimensions of the variance-covariance - matrix have been estimated as exactly zero. is_singular() checks if - a model fit is singular, and can be used in case of post-fitting convergence - warnings, such as warnings about negative eigenvalues of the Hessian. If the fit - is singular (i.e. is_singular() returns TRUE), these warnings - can most likely be ignored. -

    - There is no gold-standard about how to deal with singularity and which - random-effects specification to choose. Beside using fully Bayesian methods - (with informative priors), proposals in a frequentist framework are:

      -
    • avoid fitting overly complex models, such that the variance-covariance matrices can be estimated precisely enough (Matuschek et al. 2017)

    • -
    • use some form of model selection to choose a model that balances predictive accuracy and overfitting/type I error (Bates et al. 2015, Matuschek et al. 2017)

    • -
    • “keep it maximal”, i.e. fit the most complex model consistent with the experimental design, removing only terms required to allow a non-singular fit (Barr et al. 2013)

    • -
    - -

    References

    - -
      -
    • Bates D, Kliegl R, Vasishth S, Baayen H. Parsimonious Mixed Models. arXiv:1506.04967, June 2015.

    • -
    • Barr DJ, Levy R, Scheepers C, Tily HJ. Random effects structure for confirmatory hypothesis testing: Keep it maximal. Journal of Memory and Language, 68(3):255-278, April 2013.

    • -
    • Matuschek H, Kliegl R, Vasishth S, Baayen H, Bates D. Balancing type I error and power in linear mixed models. Journal of Memory and Language, 94:305-315, 2017.

    • -
    - - -

    Examples

    -
    library(sjmisc) -library(lme4) -data(efc) -# create binary response -efc$hi_qol <- dicho(efc$quol_5) -# prepare group variable -efc$grp = as.factor(efc$e15relat) -# data frame for fitted model -mydf <- data.frame(hi_qol = as.factor(efc$hi_qol), - sex = as.factor(efc$c161sex), - c12hour = as.numeric(efc$c12hour), - neg_c_7 = as.numeric(efc$neg_c_7), - grp = efc$grp) -# fit glmer -fit <- glmer(hi_qol ~ sex + c12hour + neg_c_7 + (1|grp), - data = mydf, family = binomial("logit")) - -converge_ok(fit)
    #> 1.19757278379967e-05 -#> TRUE
    -
    -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/crosstable_statistics.html b/docs/reference/crosstable_statistics.html deleted file mode 100644 index 83c2cd9..0000000 --- a/docs/reference/crosstable_statistics.html +++ /dev/null @@ -1,353 +0,0 @@ - - - - - - - - -Measures of association for contingency tables — cramer • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    This function calculates various measure of association for - contingency tables and returns the statistic and p-value. - Supported measures are Cramer's V, Phi, Spearman's rho, - Kendall's tau and Pearson's r.

    -
    - -
    cramer(tab, ...)
    -
    -# S3 method for formula
    -cramer(
    -  formula,
    -  data,
    -  ci.lvl = NULL,
    -  n = 1000,
    -  method = c("dist", "quantile"),
    -  ...
    -)
    -
    -phi(tab, ...)
    -
    -crosstable_statistics(
    -  data,
    -  x1 = NULL,
    -  x2 = NULL,
    -  statistics = c("auto", "cramer", "phi", "spearman", "kendall", "pearson", "fisher"),
    -  weights = NULL,
    -  ...
    -)
    -
    -xtab_statistics(
    -  data,
    -  x1 = NULL,
    -  x2 = NULL,
    -  statistics = c("auto", "cramer", "phi", "spearman", "kendall", "pearson", "fisher"),
    -  weights = NULL,
    -  ...
    -)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    tab

    A table or ftable. Tables of class -xtabs and other will be coerced to ftable -objects.

    ...

    Other arguments, passed down to the statistic functions -chisq.test, fisher.test or -cor.test.

    formula

    A formula of the form lhs ~ rhs where lhs is a -numeric variable giving the data values and rhs a factor giving the -corresponding groups.

    data

    A data frame or a table object. If a table object, x1 and -x2 will be ignored. For Kendall's tau, Spearman's rho -or Pearson's product moment correlation coefficient, data needs -to be a data frame. If x1 and x2 are not specified, -the first two columns of the data frames are used as variables -to compute the crosstab.

    ci.lvl

    Scalar between 0 and 1. If not NULL, returns a data -frame including lower and upper confidence intervals.

    n

    Number of bootstraps to be generated.

    method

    Character vector, indicating if confidence intervals should be -based on bootstrap standard error, multiplied by the value of the -quantile function of the t-distribution (default), or on sample -quantiles of the bootstrapped values. See 'Details' in boot_ci(). -May be abbreviated.

    x1

    Name of first variable that should be used to compute the -contingency table. If data is a table object, this argument -will be irgnored.

    x2

    Name of second variable that should be used to compute the -contingency table. If data is a table object, this argument -will be irgnored.

    statistics

    Name of measure of association that should be computed. May -be one of "auto", "cramer", "phi", "spearman", -"kendall", "pearson" or "fisher". See 'Details'.

    weights

    Name of variable in x that indicated the vector of -weights that will be applied to weight all observations. Default is -NULL, so no weights are used.

    - -

    Value

    - -

    For phi(), the table's Phi value. For cramer(), the - table's Cramer's V. -

    - For crosstable_statistics(), a list with following components:

    -
    estimate

    the value of the estimated measure of association.

    -
    p.value

    the p-value for the test.

    -
    statistic

    the value of the test statistic.

    -
    stat.name

    the name of the test statistic.

    -
    stat.html

    if applicable, the name of the test statistic, in HTML-format.

    -
    df

    the degrees of freedom for the contingency table.

    -
    method

    character string indicating the name of the measure of association.

    -
    method.html

    if applicable, the name of the measure of association, in HTML-format.

    -
    method.short

    the short form of association measure, equals the statistics-argument.

    -
    fisher

    logical, if Fisher's exact test was used to calculate the p-value.

    - - - -

    Details

    - -

    The p-value for Cramer's V and the Phi coefficient are based - on chisq.test(). If any expected value of a table cell is - smaller than 5, or smaller than 10 and the df is 1, then fisher.test() - is used to compute the p-value, unless statistics = "fisher"; in - this case, the use of fisher.test() is forced to compute the - p-value. The test statistic is calculated with cramer() resp. - phi(). -

    - Both test statistic and p-value for Spearman's rho, Kendall's tau - and Pearson's r are calculated with cor.test(). -

    - When statistics = "auto", only Cramer's V or Phi are calculated, - based on the dimension of the table (i.e. if the table has more than - two rows or columns, Cramer's V is calculated, else Phi).

    - -

    Examples

    -
    # Phi coefficient for 2x2 tables -tab <- table(sample(1:2, 30, TRUE), sample(1:2, 30, TRUE)) -phi(tab)
    #> [1] 0.1336306
    -# Cramer's V for nominal variables with more than 2 categories -tab <- table(sample(1:2, 30, TRUE), sample(1:3, 30, TRUE)) -cramer(tab)
    #> [1] 0.180269
    -# formula notation -data(efc) -cramer(e16sex ~ c161sex, data = efc)
    #> [1] 0.05258249
    -# bootstrapped confidence intervals -cramer(e16sex ~ c161sex, data = efc, ci.lvl = .95, n = 100)
    #> cramer conf.low conf.high -#> 1 0.05258249 -0.00860759 0.111614
    -# 2x2 table, compute Phi automatically -crosstable_statistics(efc, e16sex, c161sex)
    #> -#> # Measure of Association for Contingency Tables -#> -#> Chi-squared: 2.2327 -#> Phi: 0.0526 -#> p-value: 0.1351
    -# more dimensions than 2x2, compute Cramer's V automatically -crosstable_statistics(efc, c172code, c161sex)
    #> -#> # Measure of Association for Contingency Tables -#> -#> Chi-squared: 4.1085 -#> Cramer's V: 0.0699 -#> p-value: 0.1282
    -# ordinal data, use Kendall's tau -crosstable_statistics(efc, e42dep, quol_5, statistics = "kendall")
    #> -#> # Measure of Association for Contingency Tables -#> -#> z: -9.5951 -#> Kendall's tau: -0.2496 -#> p-value: <0.001
    -# calcilate Spearman's rho, with continuity correction -crosstable_statistics(efc, - e42dep, - quol_5, - statistics = "spearman", - exact = FALSE, - continuity = TRUE -)
    #> -#> # Measure of Association for Contingency Tables -#> -#> S: 157974157.4198 -#> Spearman's rho: -0.3177 -#> p-value: <0.001
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/cv.html b/docs/reference/cv.html deleted file mode 100644 index 14f4738..0000000 --- a/docs/reference/cv.html +++ /dev/null @@ -1,190 +0,0 @@ - - - - - - - - -Compute model quality — cv • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Compute the coefficient of variation.

    -
    - -
    cv(x, ...)
    - -

    Arguments

    - - - - - - - - - - -
    x

    Fitted linear model of class lm, merMod (lme4) -or lme (nlme).

    ...

    More fitted model objects, to compute multiple coefficients of -variation at once.

    - -

    Value

    - -

    Numeric, the coefficient of variation.

    -

    Details

    - -

    The advantage of the cv is that it is unitless. This allows - coefficient of variation to be compared to each other in ways - that other measures, like standard deviations or root mean - squared residuals, cannot be.

    - -

    Examples

    -
    data(efc) -fit <- lm(barthtot ~ c160age + c12hour, data = efc) -cv(fit)
    #> [1] 0.3948098
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/cv_error.html b/docs/reference/cv_error.html deleted file mode 100644 index 3f01f6d..0000000 --- a/docs/reference/cv_error.html +++ /dev/null @@ -1,219 +0,0 @@ - - - - - - - - -Test and training error from model cross-validation — cv_error • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    cv_error() computes the root mean squared error from a model fitted - to kfold cross-validated test-training-data. cv_compare() - does the same, for multiple formulas at once (by calling cv_error() - for each formula).

    -
    - -
    cv_error(data, formula, k = 5)
    -
    -cv_compare(data, formulas, k = 5)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - -
    data

    A data frame.

    formula

    The formula to fit the linear model for the test and training data.

    k

    The number of folds for the kfold-crossvalidation.

    formulas

    A list of formulas, to fit linear models for the test and training data.

    - -

    Value

    - -

    A data frame with the root mean squared errors for the training and test data.

    -

    Details

    - -

    cv_error() first generates cross-validated test-training pairs, using - crossv_kfold and then fits a linear model, which - is described in formula, to the training data. Then, predictions - for the test data are computed, based on the trained models. - The training error is the mean value of the rmse for - all trained models; the test error is the rmse based on all - residuals from the test data.

    - -

    Examples

    -
    data(efc) -cv_error(efc, neg_c_7 ~ barthtot + c161sex)
    #> Warning: unnest() has a new interface. See ?unnest for details. -#> Try `df %>% unnest(c(predicted, residuals))`, with `mutate()` if needed
    #> model train.error test.error -#> 1 neg_c_7 ~ barthtot + c161sex 3.5065 3.519
    -cv_compare(efc, formulas = list( - neg_c_7 ~ barthtot + c161sex, - neg_c_7 ~ barthtot + c161sex + e42dep, - neg_c_7 ~ barthtot + c12hour -))
    #> Warning: unnest() has a new interface. See ?unnest for details. -#> Try `df %>% unnest(c(predicted, residuals))`, with `mutate()` if needed
    #> Warning: unnest() has a new interface. See ?unnest for details. -#> Try `df %>% unnest(c(predicted, residuals))`, with `mutate()` if needed
    #> Warning: unnest() has a new interface. See ?unnest for details. -#> Try `df %>% unnest(c(predicted, residuals))`, with `mutate()` if needed
    #> model train.error test.error -#> 1 neg_c_7 ~ barthtot + c161sex 3.5066 3.5223 -#> 2 neg_c_7 ~ barthtot + c161sex + e42dep 3.4865 3.5089 -#> 3 neg_c_7 ~ barthtot + c12hour 3.5028 3.5205
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/deff.html b/docs/reference/deff.html deleted file mode 100644 index 47609ce..0000000 --- a/docs/reference/deff.html +++ /dev/null @@ -1,214 +0,0 @@ - - - - - - - - -Design effects for two-level mixed models — deff • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    Compute the design effect (also called Variance Inflation Factor) - for mixed models with two-level design.

    - -
    - -
    deff(n, icc = 0.05)
    - -

    Arguments

    - - - - - - - - - - -
    n

    Average number of observations per grouping cluster (i.e. level-2 unit).

    icc

    Assumed intraclass correlation coefficient for multilevel-model.

    - -

    Value

    - -

    The design effect (Variance Inflation Factor) for the two-level model.

    - -

    Details

    - -

    The formula for the design effect is simply (1 + (n - 1) * icc).

    - -

    References

    - -

    Bland JM. 2000. Sample size in guidelines trials. Fam Pract. (17), 17-20. -

    - Hsieh FY, Lavori PW, Cohen HJ, Feussner JR. 2003. An Overview of Variance Inflation Factors for Sample-Size Calculation. Evaluation and the Health Professions 26: 239-257. doi: 10.1177/0163278703255230 -

    - Snijders TAB. 2005. Power and Sample Size in Multilevel Linear Models. In: Everitt BS, Howell DC (Hrsg.). Encyclopedia of Statistics in Behavioral Science. Chichester, UK: John Wiley and Sons, Ltd. doi: 10.1002/0470013192.bsa492 -

    - Thompson DM, Fernald DH, Mold JW. 2012. Intraclass Correlation Coefficients Typical of Cluster-Randomized Studies: Estimates From the Robert Wood Johnson Prescription for Health Projects. The Annals of Family Medicine;10(3):235-40. doi: 10.1370/afm.1347

    - - -

    Examples

    -
    # Design effect for two-level model with 30 observations per -# cluster group (level-2 unit) and an assumed intraclass -# correlation coefficient of 0.05. -deff(n = 30)
    #> [1] 2.45
    -# Design effect for two-level model with 24 observation per cluster -# group and an assumed intraclass correlation coefficient of 0.2. -deff(n = 24, icc = 0.2)
    #> [1] 5.6
    -
    -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/design_effect.html b/docs/reference/design_effect.html deleted file mode 100644 index a7cc0bd..0000000 --- a/docs/reference/design_effect.html +++ /dev/null @@ -1,202 +0,0 @@ - - - - - - - - -Design effects for two-level mixed models — design_effect • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Compute the design effect (also called Variance Inflation Factor) - for mixed models with two-level design.

    -
    - -
    design_effect(n, icc = 0.05)
    - -

    Arguments

    - - - - - - - - - - -
    n

    Average number of observations per grouping cluster (i.e. level-2 unit).

    icc

    Assumed intraclass correlation coefficient for multilevel-model.

    - -

    Value

    - -

    The design effect (Variance Inflation Factor) for the two-level model.

    -

    Details

    - -

    The formula for the design effect is simply (1 + (n - 1) * icc).

    -

    References

    - -

    Bland JM. 2000. Sample size in guidelines trials. Fam Pract. (17), 17-20. -

    - Hsieh FY, Lavori PW, Cohen HJ, Feussner JR. 2003. An Overview of Variance Inflation Factors for Sample-Size Calculation. Evaluation and the Health Professions 26: 239-257. doi: 10.1177/0163278703255230 - -

    - Snijders TAB. 2005. Power and Sample Size in Multilevel Linear Models. In: Everitt BS, Howell DC (Hrsg.). Encyclopedia of Statistics in Behavioral Science. Chichester, UK: John Wiley and Sons, Ltd. doi: 10.1002/0470013192.bsa492 - -

    - Thompson DM, Fernald DH, Mold JW. 2012. Intraclass Correlation Coefficients Typical of Cluster-Randomized Studies: Estimates From the Robert Wood Johnson Prescription for Health Projects. The Annals of Family Medicine;10(3):235-40. doi: 10.1370/afm.1347

    - -

    Examples

    -
    # Design effect for two-level model with 30 observations per -# cluster group (level-2 unit) and an assumed intraclass -# correlation coefficient of 0.05. -design_effect(n = 30)
    #> [1] 2.45
    -# Design effect for two-level model with 24 observation per cluster -# group and an assumed intraclass correlation coefficient of 0.2. -design_effect(n = 24, icc = 0.2)
    #> [1] 5.6
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/dot-badlink.html b/docs/reference/dot-badlink.html deleted file mode 100644 index de07849..0000000 --- a/docs/reference/dot-badlink.html +++ /dev/null @@ -1,163 +0,0 @@ - - - - - - - - -helper-function, telling user if model is supported or not — .badlink • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    helper-function, telling user if model is supported or not

    - -
    - -
    .badlink(link, family)
    - - -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/dot-collapse_cond.html b/docs/reference/dot-collapse_cond.html deleted file mode 100644 index 017fc37..0000000 --- a/docs/reference/dot-collapse_cond.html +++ /dev/null @@ -1,163 +0,0 @@ - - - - - - - - -glmmTMB returns a list of model information, one for conditional and one for zero-inflated part, so here we "unlist" it — .collapse_cond • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    glmmTMB returns a list of model information, one for conditional and one for zero-inflated part, so here we "unlist" it

    - -
    - -
    .collapse_cond(x)
    - - -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/dot-get_variance_beta.html b/docs/reference/dot-get_variance_beta.html deleted file mode 100644 index 270ca22..0000000 --- a/docs/reference/dot-get_variance_beta.html +++ /dev/null @@ -1,163 +0,0 @@ - - - - - - - - -Get distributional variance for beta-family — .get_variance_beta • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    Get distributional variance for beta-family

    - -
    - -
    .get_variance_beta(mu, phi)
    - - -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/dot-get_variance_dispersion.html b/docs/reference/dot-get_variance_dispersion.html deleted file mode 100644 index c89c6a5..0000000 --- a/docs/reference/dot-get_variance_dispersion.html +++ /dev/null @@ -1,163 +0,0 @@ - - - - - - - - -Get dispersion-specific variance — .get_variance_dispersion • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    Get dispersion-specific variance

    - -
    - -
    .get_variance_dispersion(x, vals, faminfo, obs.terms)
    - - -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/dot-get_variance_fixed.html b/docs/reference/dot-get_variance_fixed.html deleted file mode 100644 index 628a9ad..0000000 --- a/docs/reference/dot-get_variance_fixed.html +++ /dev/null @@ -1,163 +0,0 @@ - - - - - - - - -Get fixed effects variance — .get_variance_fixed • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    Get fixed effects variance

    - -
    - -
    .get_variance_fixed(vals)
    - - -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/dot-get_variance_random.html b/docs/reference/dot-get_variance_random.html deleted file mode 100644 index c01edb9..0000000 --- a/docs/reference/dot-get_variance_random.html +++ /dev/null @@ -1,163 +0,0 @@ - - - - - - - - -Compute variance associated with a random-effects term (Johnson 2014) — .get_variance_random • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    Compute variance associated with a random-effects term (Johnson 2014)

    - -
    - -
    .get_variance_random(terms, x, vals)
    - - -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/dot-get_variance_residual.html b/docs/reference/dot-get_variance_residual.html deleted file mode 100644 index 6340ec4..0000000 --- a/docs/reference/dot-get_variance_residual.html +++ /dev/null @@ -1,163 +0,0 @@ - - - - - - - - -Get residual (distribution specific) variance from random effects — .get_variance_residual • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    Get residual (distribution specific) variance from random effects

    - -
    - -
    .get_variance_residual(x, var.cor, faminfo, name)
    - - -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/efc.html b/docs/reference/efc.html deleted file mode 100644 index 5b83250..0000000 --- a/docs/reference/efc.html +++ /dev/null @@ -1,164 +0,0 @@ - - - - - - - - -Sample dataset from the EUROFAMCARE project — efc • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    German data set from the European study on family care of older people.

    -
    - - - -

    References

    - -

    Lamura G, Döhner H, Kofahl C, editors. Family carers of older people in Europe: a six-country comparative study. Münster: LIT, 2008.

    - -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/eta_sq.html b/docs/reference/eta_sq.html deleted file mode 100644 index 9a8d66d..0000000 --- a/docs/reference/eta_sq.html +++ /dev/null @@ -1,248 +0,0 @@ - - - - - - - - -Effect size statistics for anova — anova_stats • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Returns the (partial) eta-squared, (partial) omega-squared, - epsilon-squared statistic or Cohen's F for all terms in an anovas. - anova_stats() returns a tidy summary, including all these statistics - and power for each term.

    -
    - -
    anova_stats(model, digits = 3)
    -
    -epsilon_sq(model, partial = FALSE, ci.lvl = NULL)
    -
    -eta_sq(model, partial = FALSE, ci.lvl = NULL)
    -
    -omega_sq(model, partial = FALSE, ci.lvl = NULL)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - -
    model

    A fitted anova-model of class aov or anova. Other -models are coerced to anova.

    digits

    Amount of digits for returned values.

    partial

    Logical, if TRUE, the partial eta-squared is returned.

    ci.lvl

    Scalar between 0 and 1. If not NULL, returns a data -frame with effect sizes including lower and upper confidence intervals.

    - -

    Value

    - -

    A data frame with the term name(s) and effect size statistics; if - ci.lvl is not NULL, a data frame including lower and - upper confidence intervals is returned. For anova_stats(), a tidy - data frame with all statistics is returned (excluding confidence intervals).

    -

    Details

    - -

    See details in eta_squared.

    -

    References

    - -

    Levine TR, Hullett CR (2002): Eta Squared, Partial Eta Squared, and Misreporting of Effect Size in Communication Research (pdf) -

    - Tippey K, Longnecker MT (2016): An Ad Hoc Method for Computing Pseudo-Effect Size for Mixed Model. (pdf)

    - -

    Examples

    -
    # load sample data -data(efc) - -# fit linear model -fit <- aov( - c12hour ~ as.factor(e42dep) + as.factor(c172code) + c160age, - data = efc -) - -eta_sq(fit)
    #> term etasq -#> 1 as.factor(e42dep) 0.266 -#> 2 as.factor(c172code) 0.005 -#> 3 c160age 0.048
    omega_sq(fit)
    #> term omegasq -#> 1 as.factor(e42dep) 0.263 -#> 2 as.factor(c172code) 0.004 -#> 3 c160age 0.048
    eta_sq(fit, partial = TRUE)
    #> term partial.etasq -#> 1 as.factor(e42dep) 0.281 -#> 2 as.factor(c172code) 0.008 -#> 3 c160age 0.066
    eta_sq(fit, partial = TRUE, ci.lvl = .8)
    #> term partial.etasq conf.low conf.high -#> 1 as.factor(e42dep) 0.281 0.248 0.311 -#> 2 as.factor(c172code) 0.008 0.001 0.016 -#> 3 c160age 0.066 0.047 0.089
    -anova_stats(car::Anova(fit, type = 2))
    #> Registered S3 methods overwritten by 'car': -#> method from -#> influence.merMod lme4 -#> cooks.distance.influence.merMod lme4 -#> dfbeta.influence.merMod lme4 -#> dfbetas.influence.merMod lme4
    #> term sumsq meansq df statistic p.value etasq -#> 1 as.factor(e42dep) 426461.571 142153.857 3 80.299 0.000 0.212 -#> 2 as.factor(c172code) 7352.049 3676.025 2 2.076 0.126 0.004 -#> 3 c160age 105169.595 105169.595 1 59.408 0.000 0.052 -#> 4 Residuals 1476436.343 1770.307 834 NA NA NA -#> partial.etasq omegasq partial.omegasq epsilonsq cohens.f power -#> 1 0.224 0.209 0.221 0.209 0.537 1.000 -#> 2 0.005 0.002 0.003 0.002 0.071 0.429 -#> 3 0.066 0.051 0.065 0.051 0.267 1.000 -#> 4 NA NA NA NA NA NA
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/figures/logo.png b/docs/reference/figures/logo.png deleted file mode 100644 index 7c24480b73f93451cf17c50396d77d087461396c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 18507 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 diff --git a/docs/reference/find_beta-2.png b/docs/reference/find_beta-2.png deleted file mode 100644 index 8bbe844e6c943a81e71d7fe7fbf3f28dc4ec7ccc..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 14785 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 CD_NHS diff --git a/docs/reference/find_beta-3.png b/docs/reference/find_beta-3.png deleted file mode 100644 index 0a721d2e557c4e020ff44134c3cca6eb537a1369..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 13712 zcmeHucT`i~w`Tz9(xgN|np7L0)X=2}SSS{d7L+b6N=*n&2}KZ4Ku|i;R7#|V5HIUM3I-#A2YY za}NZfg@ZsewhS}?Wm(?@2mJgozVkqrazp_hK_CbSHCaI7$D5FTP=*+{gB`)8$&qNZoy@&AN}nS6;ePM5`<&^rty%#zjN2kk3I}MCs8j_s3&w%i3@)~ zxl$o`yYbn(u-y)l0^5y~su!_2?|!6kHI^b6PHO5B?#3n9BW$0fiT1knc$`$c&xK!J zS?R>$YHbR&fGT=$mDlN1yV;W1#peT->d4oh!~Mq_IT%B_xx#h`-g#w?<2yCo+U)P6 zCVKV+&h?#3!67GlI}a*1nf=WD%N&Y!=Z9T{`fmJc12lcG zUZqrxzv->ia4uBaK)Nr(w;9=(xg2PHP*AoxUVM4(=8na^#>_;p^X)DI1>G%}Gf4px z+--9FH;7p14Ix&%w=CnuN11WPJAtfdX4rymgu65PKp+Piz6yWPoL ztqafjqx>!r{Ot1<*`vL1rNFkZB*JDn8A8<8_|>SJhYitQ!XXpTYcl3}Km0$Gd;H0X zxpjRKRGo7f>lhPZKYu?6oP~3(qhHWb9XAl2rA>Rf=UvwAgf!2fPpZ4hGqV!N>vS9l zIr~1-lRnC!m^OZ$I@Vb^PVxAY$ByoH#!l9wi62Ff4z-b_iuZazhq8gaC5l-0S#Q{p zJm&UFeo~0Rt|jLTDFGi<%3h#mx<7`+4|KHME153$o+4PF2ONQ^#CXPQP*x|2v^#GuZdo>w4px`aLnM*M(F) z#EX2cu#q%u-J!v_j=*>7E#rR5#q!mo99*!dl#4arFG=yWp6<&Ju`vT8jp7Mdjhjndx{h)_nfS3_0QI`vk&b&X@^!Ql&X z?L0a-5&TJ)98Wm-y${6^Q`%xaVm0~WUFCmmcZWHmMy{bJbPz9iTjt;~r8RzR1fMRi z{R_WxG>PLHg@nD|Mwy4vWaU|8LO^h^rtpgm_|C1aH|fK*J1CX4hqwctYh$2C3Dus( zzs6*7#H8b^AF(0D!y_)$R<#hLZ7&?rkbzGJL3NwDFu}Us@=>9x8j%xfhI59V%>ixp zRSLlDO1Uq*S;@wRaNoroJr^p|JXH=!1b@w5bWCP#8Hxz;1os(-kU5~OTI0N(%x-by zCGO$xD0yBX=i&%OL~UIUvxdaG)&~)Ncsn|U^FohTBK$;f6L*{^&Wzif(yV4cn-bia zW@)Qh5LgS^>WhisQC7mkOYK*)vENLqL$p52U_^hIP-PRmucvEGzAX~^cxmnB* z7!U$OnPvB3UUtxB@e+n44r8*gnqp(*6wA}3K%X`10a&xNyi{QH1qLPsttNJ=I1~=) z^H}{B->l%5E%l**KYI_qo6)z5)E`0jXbWu_UPieaU(^2v{y>gh_d4rw5M$`{-rDPJ zMt9Ha1^zL$o2S)`OIhwHfVg#Vu2HG-vb`|>G*Os~8u<#gAio+-7m5+D_nnOqqQ9l^ zVx2nm6-x2)YB`Bbyrx%B;QL;%>-%s;WJvCNyK0rHz=NVjI+vIEj#(k9ugH$A@*A~t zfh3dAhrIbVaAj%C)`Sz~T3

    |4k`<&limH7nu()4cW*Bd(n6N7>@x|Uha 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 Qw0wZ{8%k;Q#;t diff --git a/docs/reference/find_beta-4.png b/docs/reference/find_beta-4.png deleted file mode 100644 index 03413a919b6c6d2451f898d36e8b479e196a585f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 12224 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 diff --git a/docs/reference/find_beta.html b/docs/reference/find_beta.html deleted file mode 100644 index f749ec1..0000000 --- a/docs/reference/find_beta.html +++ /dev/null @@ -1,292 +0,0 @@ - - - - - - - - -Determining distribution parameters — find_beta • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -

    -
    - - - - -
    - -
    -
    - - -
    -

    find_beta(), find_normal() and find_cauchy() find the - shape, mean and standard deviation resp. the location and scale parameters - to describe the beta, normal or cauchy distribution, based on two - percentiles. find_beta2() finds the shape parameters for a Beta - distribution, based on a probability value and its standard error - or confidence intervals.

    -
    - -
    find_beta(x1, p1, x2, p2)
    -
    -find_beta2(x, se, ci, n)
    -
    -find_cauchy(x1, p1, x2, p2)
    -
    -find_normal(x1, p1, x2, p2)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    x1

    Value for the first percentile.

    p1

    Probability of the first percentile.

    x2

    Value for the second percentile.

    p2

    Probability of the second percentile.

    x

    Numeric, a probability value between 0 and 1. Typically indicates -a prevalence rate of an outcome of interest; Or an integer value -with the number of observed events. In this case, specify n -to indicate the toral number of observations.

    se

    The standard error of x. Either se or ci must -be specified.

    ci

    The upper limit of the confidence interval of x. Either -se or ci must be specified.

    n

    Numeric, number of total observations. Needs to be specified, if -x is an integer (number of observed events), and no -probability. See 'Examples'.

    - -

    Value

    - -

    A list of length two, with the two distribution parameters than can - be used to define the distribution, which (best) describes - the shape for the given input parameters.

    -

    Details

    - -

    These functions can be used to find parameter for various distributions, - to define prior probabilities for Bayesian analyses. x1, - p1, x2 and p2 are parameters that describe two - quantiles. Given this knowledge, the distribution parameters are - returned.

    - Use find_beta2(), if the known parameters are, e.g. a prevalence - rate or similar probability, and its standard deviation or confidence - interval. In this case. x should be a probability, - for example a prevalence rate of a certain event. se then - needs to be the standard error for this probability. Alternatively, - ci can be specified, which should indicate the upper limit - of the confidence interval od the probability (prevalence rate) x. - If the number of events out of a total number of trials is known - (e.g. 12 heads out of 30 coin tosses), x can also be the number - of observed events, while n indicates the total amount of trials - (in the above example, the function call would be: find_beta2(x = 12, n = 30)).

    -

    References

    - -

    Cook JD. Determining distribution parameters from quantiles. 2010: Department of Biostatistics, Texas (PDF)

    - -

    Examples

    -
    # example from blogpost: -# https://www.johndcook.com/blog/2010/01/31/parameters-from-percentiles/ -# 10% of patients respond within 30 days of treatment -# and 80% respond within 90 days of treatment -find_normal(x1 = 30, p1 = .1, x2 = 90, p2 = .8)
    #> $mean -#> [1] 53.78387 -#> -#> $sd -#> [1] 30.48026 -#>
    find_cauchy(x1 = 30, p1 = .1, x2 = 90, p2 = .8)
    #> $location -#> [1] 48.54102 -#> -#> $scale -#> [1] 57.06339 -#>
    -parms <- find_normal(x1 = 30, p1 = .1, x2 = 90, p2 = .8) -curve( - dnorm(x, mean = parms$mean, sd = parms$sd), - from = 0, to = 200 -)
    -parms <- find_cauchy(x1 = 30, p1 = .1, x2 = 90, p2 = .8) -curve( - dcauchy(x, location = parms$location, scale = parms$scale), - from = 0, to = 200 -)
    - -find_beta2(x = .25, ci = .5)
    #> $shape1 -#> [1] 2.860267 -#> -#> $shape2 -#> [1] 7.93757 -#>
    -shapes <- find_beta2(x = .25, ci = .5) -curve(dbeta(x, shapes[[1]], shapes[[2]]))
    -# find Beta distribution for 3 events out of 20 observations -find_beta2(x = 3, n = 20)
    #> $shape1 -#> [1] 4.157811 -#> -#> $shape2 -#> [1] 22.03272 -#>
    -shapes <- find_beta2(x = 3, n = 20) -curve(dbeta(x, shapes[[1]], shapes[[2]]))
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/fish.html b/docs/reference/fish.html deleted file mode 100644 index 860c186..0000000 --- a/docs/reference/fish.html +++ /dev/null @@ -1,164 +0,0 @@ - - - - - - - - -Sample dataset — fish • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Sample data from the UCLA idre website.

    -
    - - - -

    References

    - -

    https://stats.idre.ucla.edu/r/dae/zip/

    - -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/gmd.html b/docs/reference/gmd.html deleted file mode 100644 index e1090b5..0000000 --- a/docs/reference/gmd.html +++ /dev/null @@ -1,198 +0,0 @@ - - - - - - - - -Gini's Mean Difference — gmd • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    gmd() computes Gini's mean difference for a numeric vector - or for all numeric vectors in a data frame.

    -
    - -
    gmd(x, ...)
    - -

    Arguments

    - - - - - - - - - - -
    x

    A vector or data frame.

    ...

    Optional, unquoted names of variables that should be selected for -further processing. Required, if x is a data frame (and no vector) -and only selected variables from x should be processed. You may also -use functions like : or tidyselect's select_helpers.

    - -

    Value

    - -

    For numeric vectors, Gini's mean difference. For non-numeric vectors - or vectors of length < 2, returns NA.

    -

    Note

    - -

    Gini's mean difference is defined as the mean absolute difference between - any two distinct elements of a vector. Missing values from x are - silently removed.

    -

    References

    - -

    David HA. Gini's mean difference rediscovered. Biometrika 1968(55): 573-575

    - -

    Examples

    -
    data(efc) -gmd(efc$e17age)
    #> [1] 9.297005
    gmd(efc, e17age, c160age, c12hour)
    #> # A tibble: 1 x 3 -#> e17age c160age c12hour -#> <dbl> <dbl> <dbl> -#> 1 9.30 15.2 47.9
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/grpmean.html b/docs/reference/grpmean.html deleted file mode 100644 index 64bc99f..0000000 --- a/docs/reference/grpmean.html +++ /dev/null @@ -1,354 +0,0 @@ - - - - - - - - -Summary of mean values by group — grpmean • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Computes mean, sd and se for each sub-group (indicated by grp) - of dv.

    -
    - -
    grpmean(
    -  x,
    -  dv,
    -  grp,
    -  weights = NULL,
    -  digits = 2,
    -  out = c("txt", "viewer", "browser"),
    -  encoding = "UTF-8",
    -  file = NULL
    -)
    -
    -means_by_group(
    -  x,
    -  dv,
    -  grp,
    -  weights = NULL,
    -  digits = 2,
    -  out = c("txt", "viewer", "browser"),
    -  encoding = "UTF-8",
    -  file = NULL
    -)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    x

    A (grouped) data frame.

    dv

    Name of the dependent variable, for which the mean value, grouped -by grp, is computed.

    grp

    Factor with the cross-classifying variable, where dv is -grouped into the categories represented by grp. Numeric vectors -are coerced to factors.

    weights

    Name of variable in x that indicated the vector of -weights that will be applied to weight all observations. Default is -NULL, so no weights are used.

    digits

    Numeric, amount of digits after decimal point when rounding -estimates and values.

    out

    Character vector, indicating whether the results should be printed -to console (out = "txt") or as HTML-table in the viewer-pane -(out = "viewer") or browser (out = "browser"), of if the -results should be plotted (out = "plot", only applies to certain -functions). May be abbreviated.

    encoding

    Character vector, indicating the charset encoding used -for variable and value labels. Default is "UTF-8". Only used -when out is not "txt".

    file

    Destination file, if the output should be saved as file. -Only used when out is not "txt".

    - -

    Value

    - -

    For non-grouped data frames, grpmean() returns a data frame with - following columns: term, mean, N, std.dev, - std.error and p.value. For grouped data frames, returns - a list of such data frames.

    -

    Details

    - -

    This function performs a One-Way-Anova with dv as dependent - and grp as independent variable, by calling - lm(count ~ as.factor(grp)). Then contrast - is called to get p-values for each sub-group. P-values indicate whether - each group-mean is significantly different from the total mean.

    - -

    Examples

    -
    data(efc) -grpmean(efc, c12hour, e42dep)
    #> -#> # Grouped Means for average number of hours of care per week by elder's dependency -#> -#> term mean N std.dev std.error p.value -#> 1 independent 9.91 66 8.01 0.99 <0.001 -#> 2 slightly dependent 17.54 225 17.74 1.18 <0.001 -#> 3 moderately dependent 34.52 306 41.54 2.37 0.98 -#> 4 severely dependent 75.90 304 61.72 3.54 <0.001 -#> 5 Total 42.44 901 50.82 1.69 -#> -#> Anova: R2=0.245; adj.R2=0.242; F=96.908; p=0.000
    -data(iris) -grpmean(iris, Sepal.Width, Species)
    #> -#> # Grouped Means for Sepal.Width by Species -#> -#> term mean N std.dev std.error p.value -#> 1 setosa 3.43 50 0.38 0.05 <0.001 -#> 2 versicolor 2.77 50 0.31 0.04 <0.001 -#> 3 virginica 2.97 50 0.32 0.05 0.04 -#> 4 Total 3.06 150 0.44 0.04 -#> -#> Anova: R2=0.401; adj.R2=0.393; F=49.160; p=0.000
    -# also works for grouped data frames -library(dplyr) -efc %>% - group_by(c172code) %>% - grpmean(c12hour, e42dep)
    #> -#> Grouped by: -#> carer's level of education: low level of education -#> -#> # Grouped Means for average number of hours of care per week by elder's dependency -#> -#> term mean N std.dev std.error p.value -#> 1 independent 16.33 12 10.74 3.10 0.02 -#> 2 slightly dependent 15.38 42 9.55 1.47 <0.001 -#> 3 moderately dependent 42.05 61 46.53 5.96 0.70 -#> 4 severely dependent 85.52 65 56.42 7.00 <0.001 -#> 5 Total 49.81 180 52.24 3.89 -#> -#> Anova: R2=0.307; adj.R2=0.295; F=25.955; p=0.000 -#> -#> -#> Grouped by: -#> carer's level of education: intermediate level of education -#> -#> # Grouped Means for average number of hours of care per week by elder's dependency -#> -#> term mean N std.dev std.error p.value -#> 1 independent 7.96 45 3.91 0.58 <0.001 -#> 2 slightly dependent 17.12 135 16.52 1.42 <0.001 -#> 3 moderately dependent 33.55 163 41.05 3.22 0.75 -#> 4 severely dependent 79.71 163 63.13 4.94 <0.001 -#> 5 Total 41.76 506 51.42 2.29 -#> -#> Anova: R2=0.284; adj.R2=0.280; F=66.374; p=0.000 -#> -#> -#> Grouped by: -#> carer's level of education: high level of education -#> -#> # Grouped Means for average number of hours of care per week by elder's dependency -#> -#> term mean N std.dev std.error p.value -#> 1 independent 15.20 5 18.43 8.24 0.36 -#> 2 slightly dependent 18.08 39 12.98 2.08 0.15 -#> 3 moderately dependent 28.42 62 35.64 4.53 0.67 -#> 4 severely dependent 63.38 50 62.69 8.87 <0.001 -#> 5 Total 36.62 156 46.38 3.71 -#> -#> Anova: R2=0.167; adj.R2=0.151; F=10.155; p=0.000 -#> -#>
    -# weighting -efc$weight <- abs(rnorm(n = nrow(efc), mean = 1, sd = .5)) -grpmean(efc, c12hour, e42dep, weights = weight)
    #> -#> # Grouped Means for average number of hours of care per week by elder's dependency -#> -#> term mean N std.dev std.error p.value -#> 1 independent 9.97 70 8.62 1.06 <0.001 -#> 2 slightly dependent 17.25 225 16.84 1.12 <0.001 -#> 3 moderately dependent 36.25 304 43.38 2.48 0.76 -#> 4 severely dependent 78.32 313 63.24 3.63 <0.001 -#> 5 Total 43.98 901 52.55 1.75 -#> -#> Anova: R2=0.250; adj.R2=0.248; F=99.732; p=0.000
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/hdi.html b/docs/reference/hdi.html deleted file mode 100644 index 2fc57f7..0000000 --- a/docs/reference/hdi.html +++ /dev/null @@ -1,508 +0,0 @@ - - - - - - - - -Compute statistics for MCMC samples and Stan models — hdi • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    hdi() computes the highest density interval for values from - MCMC samples, while cred_int() computes the credible interval (or - uncertainty interval). rope() calculates the proportion of a posterior - distribution that lies within a region of practical equivalence. - equi_test() combines these two functions and performs a - "HDI+ROPE decision rule" (Test for Practical Equivalence) (Kruschke 2018) - to check whether parameter values should be accepted or rejected against - the background of a formulated null hypothesis. n_eff() calculates - the the number of effective samples (effective sample size). mcse() - returns the Monte Carlo standard error. mediation() is a short - summary for multivariate-response mediation-models.

    - -
    - -
    hdi(x, ...)
    -
    -# S3 method for stanreg
    -hdi(x, prob = 0.9, trans = NULL, type = c("fixed",
    -  "random", "all"), ...)
    -
    -# S3 method for brmsfit
    -hdi(x, prob = 0.9, trans = NULL, type = c("fixed",
    -  "random", "all"), ...)
    -
    -cred_int(x, ...)
    -
    -# S3 method for stanreg
    -cred_int(x, prob = 0.9, trans = NULL,
    -  type = c("fixed", "random", "all"), ...)
    -
    -# S3 method for brmsfit
    -cred_int(x, prob = 0.9, trans = NULL,
    -  type = c("fixed", "random", "all"), ...)
    -
    -equi_test(x, ...)
    -
    -# S3 method for stanreg
    -equi_test(x, rope, eff_size, out = c("txt", "viewer",
    -  "browser", "plot"), ...)
    -
    -# S3 method for brmsfit
    -equi_test(x, rope, eff_size, out = c("txt", "viewer",
    -  "browser", "plot"), ...)
    -
    -mcse(x, ...)
    -
    -# S3 method for brmsfit
    -mcse(x, type = c("fixed", "random", "all"), ...)
    -
    -# S3 method for stanreg
    -mcse(x, type = c("fixed", "random", "all"), ...)
    -
    -mediation(x, ...)
    -
    -# S3 method for brmsfit
    -mediation(x, treatment, mediator, prob = 0.9,
    -  typical = "median", ...)
    -
    -n_eff(x, ...)
    -
    -# S3 method for stanreg
    -n_eff(x, type = c("fixed", "random", "all"), ...)
    -
    -# S3 method for brmsfit
    -n_eff(x, type = c("fixed", "random", "all"), ...)
    -
    -rope(x, rope, ...)
    -
    -# S3 method for stanreg
    -rope(x, rope, trans = NULL, type = c("fixed",
    -  "random", "all"), ...)
    -
    -# S3 method for brmsfit
    -rope(x, rope, trans = NULL, type = c("fixed",
    -  "random", "all"), ...)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    x

    A stanreg, stanfit, or brmsfit object. For -hdi() and rope(), may also be a data frame or a vector -of values from a probability distribution (e.g., posterior probabilities -from MCMC sampling).

    ...

    Further arguments passed down to equi_test() when -plot = TRUE:

      -
    • colors: - Color of the density regions for the 95% distribution of the posterior - samples.

    • -
    • rope.color and rope.alpha: - Fill color and alpha-value of the ROPE (region of practical equivalence).

    • -
    • x.title: - Title for the x-axis of the plot.

    • -
    • legend.title: - Title for the plot legend.

    • -
    • labels: - Character vector of same length as terms plotted on the y-axis, to - give axis labels user-defined labels.

    • -
    prob

    Vector of scalars between 0 and 1, indicating the mass within -the credible interval that is to be estimated. See hdi.

    trans

    Name of a function or character vector naming a function, used -to apply transformations on the returned HDI-values resp. -(for rope()) on the values of the posterior distribution, before -calculating the rope based on the boundaries given in rope. Note -that the values in rope are not transformed.

    type

    For mixed effects models, specify the type of effects that should -be returned. type = "fixed" returns fixed effects only, -type = "random" the random effects and type = "all" returns -both fixed and random effects.

    rope

    Vector of length two, indicating the lower and upper limit of a -range around zero, which indicates the region of practical equivalence. -Values of the posterior distribution within this range are considered as -being "practically equivalent to zero".

    eff_size

    A scalar indicating the effect size (the size of an negligible -effect) that is used to calculate the limits of the ROPE for the test of -practical equivalence. If not specified, an effect size of .1 is used for -linear models, as suggested by Kruschke 2018 (see 'Details'). -If rope is specified, this argument will be ignored.

    out

    Character vector, indicating whether the results should be printed -to console (out = "txt") or as HTML-table in the viewer-pane -(out = "viewer") or browser (out = "browser"), of if the -results should be plotted (out = "plot", only applies to certain -functions). May be abbreviated.

    treatment

    Character, name of the treatment variable (or direct effect) -in a (multivariate response) mediator-model. If missing, mediation() -tries to find the treatment variable automatically, however, this may fail.

    mediator

    Character, name of the mediator variable in a (multivariate -response) mediator-model. If missing, mediation() tries to find the -treatment variable automatically, however, this may fail.

    typical

    The typical value that will represent the Bayesian point estimate. -By default, the posterior median is returned. See typical_value -for possible values for this argument.

    - -

    Value

    - -

    For hdi(), if x is a vector, returns a vector of length - two with the lower and upper limit of the HDI; if x is a - stanreg, stanfit or brmsfit object, returns a - tibble with lower and upper HDI-limits for each predictor. To distinguish - multiple HDI values, column names for the HDI get a suffix when prob - has more than one element. -

    - For rope(), returns a tibble with two columns: the proportion of - values from x that are within and outside the boundaries of - rope. -

    - equi_test() returns a tibble with a column decision that - indicates whether or not a parameter value is accepted/rejected; - inside.rope, which indicates the proportion of the whole posterior - distribution that lies inside the ROPE (not just the proportion of - values from the 95% HDI); and the lower and upper interval from the 95%-HDI. -

    - mcse() and n_eff() return a tibble with two columns: one - with the term names and one with the related statistic resp. effective - sample size. -

    - mediation() returns a data frame with direct, indirect, mediator and - total effect of a multivariate-response mediation-model, as well as the - proportion mediated. The effect sizes are mean values of the posterior - samples.

    - -

    Details

    - -
    -
    HDI

    Computation for HDI is based on the code from Kruschke 2015, pp. 727f. - For default sampling in Stan (4000 samples), the 90% intervals for HDI are - more stable than, for instance, 95% intervals. An effective sample size - (see nsamples) of at least 10.000 is recommended if - 95% intervals should be computed (see Kruschke 2015, p. 183ff).

    -
    Credible Intervals

    Credible intervals (or uncertainty intervals) are simply the quantiles - for a given probability of the posterior draws. See - posterior_interval for more details.

    -
    MCSE

    The Monte Carlo Standard Error is another useful measure of accuracy of - the chains. It is defined as standard deviation of the chains divided by - their effective sample size (the formula for mcse() is from - Kruschke 2015, p. 187). The MCSE “provides a quantitative suggestion - of how big the estimation noise is”.

    -
    Number of Effective Samples

    The effective sample size divides the actual sample size by the amount - of autocorrelation. The effective sample size is a measure of “how - much independent information there is in autocorrelated chains”, or: - “What would be the sample size of a completely non-autocorrelated chain - that yielded the same information?” (Kruschke 2015, p182-3). - The ratio of effective number of samples and total number of samples - (provided in tidy_stan()) ranges from 0 to 1, and should be close - to 1. The closer this ratio comes to zero means that the chains may be - inefficient, but possibly still okay.

    -
    ROPE

    There are no fixed rules to set the limits for the region of practical - equivalence. However, there are some conventions described by - Kruschke (2018) how to specify the limits of the rope. One - convention for linear models is to set the limits about .1 SD of the - dependent variable around zero (i.e. 0 +/- .1 * sd(y)), where - .1 stands for half of a small effect size. Another, more conservative - convention to set the ROPE limits is a range of half a standard - deviation around zero (see Norman et al. 2003), which indicates - a clinical relevant effect (i.e. 0 +/- .25 * sd(y) or even - 0 +/- .5 * sd(y))

    -
    Test for Practical Equivalence

    equi_test() computes the 95%-HDI for x and checks if a - model predictor's HDI lies completely outside, completely inside or - partially inside the ROPE. If the HDI is completely outside the ROPE, - the "null hypothesis" for this parameter is "rejected". If the ROPE - completely covers the HDI, i.e. all most credible values of a parameter - are inside the region of practical equivalence, the null hypothesis - is accepted. Else, it's undecided whether to accept or reject the - null hypothesis. In short, desirable results are low proportions inside - the ROPE (the closer to zero the better) and the H0 should be rejected. -

    - If neither the rope nor eff_size argument are specified, - the effect size (the size of an negligible effect) will be set to 0.1 - and the ROPE is 0 +/- .1 * sd(y) for linear models. This is the - suggested way to specify the ROPE limits according to Kruschke (2018). - For models with binary outcome, there is no direct way to specify the - effect size that defines the ROPE limits. Two examples from Kruschke - suggest that a negligible change is about .05 on the logit-scale. - In these cases, it is recommended to specify the rope argument, - however, if not specified, the ROPE limits are caluclated as suggested - by Kruschke: The effect size is the probability of "success" for the - outcome, divided by pi. For all other models, - 0 +/- .1 * sd(intercept) is used to determine the ROPE limits. -

    - If eff_size is specified, but rope is not, then - the same formulas apply, except that .1 is replaced by the - value in eff_size. If rope is specified, eff_size - will be ignored. See also section ROPE in 'Details'. -

    - The advantage of Bayesian testing for practical equivalence over - classical frequentist null hypothesis significance testing is that - discrete decisions are avoided, “because such decisions encourage - people to ignore the magnitude of the parameter value and its uncertainty” - (Kruschke (2018)).

    -
    Mediation Analysis

    mediation() returns a data frame with information on the - direct effect (mean value of posterior samples from treatment - of the outcome model), mediator effect (mean value of posterior - samples from mediator of the outcome model), indirect effect - (mean value of the multiplication of the posterior samples from - mediator of the outcome model and the posterior samples from - treatment of the mediation model) and the total effect (mean - value of sums of posterior samples used for the direct and indirect - effect). The proportion mediated is the indirect effect divided - by the total effect. -

    - For all values, the 90% HDIs are calculated by default. Use prob - to calculate a different interval. -

    - The arguments treatment and mediator do not necessarily - need to be specified. If missing, mediation() tries to find the - treatment and mediator variable automatically. If this does not work, - specify these variables.

    -
    - -

    Note

    - -

    Since equi_test() computes 95% HDI, a number of 10.000 samples - produces more stable results (see Kruschke 2015, p183ff).

    - -

    References

    - -

    Kruschke JK. Doing Bayesian Data Analysis: A Tutorial with R, JAGS, and Stan. 2nd edition. Academic Press, 2015 -

    - Kruschke JK. Rejecting or Accepting Parameter Values in Bayesian Estimation. Advances in Methods and Practices in Psychological Science. 2018; doi: 10.1177/2515245918771304 -

    - Norman GR, Sloan JA, Wyrwich KW. Interpretation of Changes in Health-related Quality of Life: The Remarkable Universality of Half a Standard Deviation. Medical Care. 2003;41: 582-592. doi: 10.1097/01.MLR.0000062554.74615.4C

    - - -

    Examples

    -
    # NOT RUN {
    -if (require("rstanarm")) {
    -  fit <- stan_glm(mpg ~ wt + am, data = mtcars, chains = 1)
    -  hdi(fit)
    -
    -  # return multiple intervals
    -  hdi(fit, prob = c(.5, .7, .9))
    -
    -  # fit logistic regression model
    -  fit <- stan_glm(
    -    vs ~ wt + am,
    -    data = mtcars,
    -    family = binomial("logit"),
    -    chains = 1
    -  )
    -  # compute hdi, transform on "odds ratio scale"
    -  hdi(fit, trans = exp)
    -
    -  # compute rope, on scale of linear predictor. finds proportion
    -  # of posterior distribution values between -1 and 1.
    -  rope(fit, rope = c(-1, 1))
    -
    -  # compute rope, boundaries as "odds ratios". finds proportion of
    -  # posterior distribution values, which - after being exponentiated -
    -  # are between .8 and 1.25 (about -.22 and .22 on linear scale)
    -  rope(fit, rope = c(.8, 1.25), trans = exp)
    -
    -  # Test for Practical Equivalence
    -  equi_test(fit)
    -  equi_test(fit, out = "plot")
    -}
    -# }
    -
    -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/icc.html b/docs/reference/icc.html deleted file mode 100644 index 54b71a5..0000000 --- a/docs/reference/icc.html +++ /dev/null @@ -1,478 +0,0 @@ - - - - - - - - -Intraclass-Correlation Coefficient — icc • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    This function calculates the intraclass-correlation - (icc) - sometimes also called variance partition coefficient - (vpc) - for random intercepts of mixed effects models. Currently, - merMod, glmmTMB, - stanreg and brmsfit objects are supported.

    - -
    - -
    icc(x, ...)
    -
    -# S3 method for merMod
    -icc(x, adjusted = FALSE, ...)
    -
    -# S3 method for glmmTMB
    -icc(x, adjusted = FALSE, ...)
    -
    -# S3 method for stanreg
    -icc(x, re.form = NULL, typical = "mean",
    -  prob = 0.89, ppd = FALSE, adjusted = FALSE, ...)
    -
    -# S3 method for brmsfit
    -icc(x, re.form = NULL, typical = "mean",
    -  prob = 0.89, ppd = FALSE, ...)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    x

    Fitted mixed effects model (of class merMod, glmmTMB, -stanreg or brmsfit).

    ...

    Currently not used.

    adjusted

    Logical, if TRUE, the adjusted (and -conditional) ICC is calculated, which reflects the uncertainty of all -random effects (see 'Details'). For Bayesian models, if ppd = TRUE, -adjusted will be ignored.

    re.form

    Formula containing group-level effects to be considered in -the prediction. If NULL (default), include all group-level effects. -Else, for instance for nested models, name a specific group-level effect -to calculate the ICC for this group-level. Only applies if ppd = TRUE.

    typical

    Character vector, naming the function that will be used as -measure of central tendency for the ICC. The default is "mean". See -typical_value for options.

    prob

    Vector of scalars between 0 and 1, indicating the mass within -the credible interval that is to be estimated. See hdi.

    ppd

    Logical, if TRUE, variance decomposition is based on the -posterior predictive distribution, which is the correct way for Bayesian -non-Gaussian models. By default, ppd is set to TRUE for -non-Gaussian models.If adjusted = TRUE and ppd = FALSE, -variance decomposition is approximated following the suggestion by -Nakagawa et al. 2017 (see 'Details'), however, this is currently -only implemented for Gaussian models.

    - -

    Value

    - -

    A numeric vector with all random intercept intraclass-correlation-coefficients. - Furthermore, if adjusted = FALSE, between- and within-group variances - as well as random-slope variance are returned as attributes. -

    - For stanreg or brmsfit objects, the HDI for each statistic - is also included as attribute.

    - -

    Details

    - -

    The "simple" ICC (with both ppd and adjusted set to - FALSE) is calculated by dividing the between-group-variance (random - intercept variance) by the total variance (i.e. sum of between-group-variance - and within-group (residual) variance).

    - The calculation of the ICC for generalized linear mixed models with binary outcome is based on - Wu et al. (2012). For other distributions (negative binomial, poisson, ...), - calculation is based on Nakagawa et al. 2017, however, for - non-Gaussian models it is recommended to compute the adjusted ICC (with - adjusted = TRUE, see below). -

    - ICC for unconditional and conditional models -

    - Usually, the ICC is calculated for the null model ("unconditional model"). - However, according to Raudenbush and Bryk (2002) or - Rabe-Hesketh and Skrondal (2012) it is also feasible to compute the ICC - for full models with covariates ("conditional models") and compare how - much a level-2 variable explains the portion of variation in the grouping - structure (random intercept). -

    - ICC for random-slope models -

    - Caution: For models with random slopes and random intercepts, - the ICC would differ at each unit of the predictors. Hence, the ICC for these - kind of models cannot be understood simply as proportion of variance - (see Goldstein et al. 2010). For convenience reasons, as the - icc() function also extracts the different random effects - variances, the ICC for random-slope-intercept-models is reported - nonetheless, but it is usually no meaningful summary of the - proportion of variances. -

    - To get a meaningful ICC also for models with random slopes, use adjusted = TRUE. - The adjusted ICC uses the mean random effect variance, which is based - on the random effect variances for each value of the random slope - (see Johnson et al. 2014). -

    - ICC for models with multiple or nested random effects -

    - Caution: By default, for three-level-models, depending on the - nested structure of the model, or for models with multiple random effects, - icc() only reports the proportion of variance explained for each - grouping level. Use adjusted = TRUE to calculate the adjusted and - conditional ICC, which condition on all random effects. -

    - Adjusted and conditional ICC -

    - If adjusted = TRUE, an adjusted and conditional ICC are calculated, - which take all sources of uncertainty (of all random effects) - into account to report an "adjusted" ICC, as well as the conditional ICC. - The latter also takes the fixed effects variances into account (see - Nakagawa et al. 2017). If random effects are not nested and not - cross-classified, the adjusted (adjusted = TRUE) and unadjusted - (adjusted = FALSE) ICC are identical. adjust = TRUE returns - a meaningful ICC for models with random slopes. Furthermore, the adjusted - ICC is recommended for models with other distributions than Gaussian. -

    - ICC for specific group-levels -

    - To calculate the proportion of variance for specific levels related to each - other (e.g., similarity of level-1-units within - level-2-units or level-2-units within level-3-units) must be computed - manually. Use get_re_var to get the between-group-variances - and residual variance of the model, and calculate the ICC for the various level - correlations. -

    - For example, for the ICC between level 1 and 2:
    - sum(get_re_var(fit)) / (sum(get_re_var(fit)) + get_re_var(fit, "sigma_2")) -

    - or for the ICC between level 2 and 3:
    - get_re_var(fit)[2] / sum(get_re_var(fit)) -

    - ICC for Bayesian models -

    - If ppd = TRUE, icc() calculates a variance decomposition based on - the posterior predictive distribution. In this case, first, the draws from - the posterior predictive distribution not conditioned on group-level - terms (posterior_predict(..., re.form = NA)) are calculated as well - as draws from this distribution conditioned on all random effects - (by default, unless specified else in re.form) are taken. Then, second, - the variances for each of these draws are calculated. The "ICC" is then the - ratio between these two variances. This is the recommended way to - analyse random-effect-variances for non-Gaussian models. It is then possible - to compare variances accross models, also by specifying different group-level - terms via the re.form-argument. -

    - Sometimes, when the variance of the posterior predictive distribution is - very large, the variance ratio in the output makes no sense, e.g. because - it is negative. In such cases, it might help to use a more robust measure - to calculate the central tendency of the variances. For example, use - typical = "median".

    - -

    Note

    - -

    Some notes on why the ICC is useful, based on Grace-Martin:

      -
    • It can help you determine whether or not a linear mixed model is even necessary. If you find that the correlation is zero, that means the observations within clusters are no more similar than observations from different clusters. Go ahead and use a simpler analysis technique.

    • -
    • It can be theoretically meaningful to understand how much of the overall variation in the response is explained simply by clustering. For example, in a repeated measures psychological study you can tell to what extent mood is a trait (varies among people, but not within a person on different occasions) or state (varies little on average among people, but varies a lot across occasions).

    • -
    • It can also be meaningful to see how the ICC (as well as the between and within cluster variances) changes as variable are added to the model.

    • -

    In short, the ICC can be interpreted as “the proportion of the variance - explained by the grouping structure in the population” (Hox 2002: 15). -

    - The random effect variances indicate the between- and within-group - variances as well as random-slope variance and random-slope-intercept - correlation. The components are denoted as following:

      -
    • Within-group (residual) variance: sigma_2

    • -
    • Between-group-variance: tau.00 (variation between individual intercepts and average intercept)

    • -
    • Random-slope-variance: tau.11 (variation between individual slopes and average slope)

    • -
    • Random-Intercept-Slope-covariance: tau.01

    • -
    • Random-Intercept-Slope-correlation: rho.01

    • -
    - -

    References

    - -
      -
    • Aguinis H, Gottfredson RK, Culpepper SA. 2013. Best-Practice Recommendations for Estimating Cross-Level Interaction Effects Using Multilevel Modeling. Journal of Management 39(6): 1490-1528 (doi: 10.1177/0149206313478188 -)

    • -
    • Goldstein H, Browne W, Rasbash J. 2010. Partitioning Variation in Multilevel Models. Understanding Statistics, 1:4, 223-231 (doi: 10.1207/S15328031US0104_02 -)

    • -
    • Grace-Martion K. The Intraclass Correlation Coefficient in Mixed Models, web

    • -
    • Hox J. 2002. Multilevel analysis: techniques and applications. Mahwah, NJ: Erlbaum

    • -
    • Johnson PC, O'Hara RB. 2014. Extension of Nakagawa & Schielzeth's R2GLMM to random slopes models. Methods Ecol Evol, 5: 944-946. (doi: 10.1111/2041-210X.12225 -)

    • -
    • Nakagawa S, Johnson P, Schielzeth H (2017) The coefficient of determination R2 and intra-class correlation coefficient from generalized linear mixed-effects models revisted and expanded. J. R. Soc. Interface 14. doi: 10.1098/rsif.2017.0213

    • -
    • Rabe-Hesketh S, Skrondal A. 2012. Multilevel and longitudinal modeling using Stata. 3rd ed. College Station, Tex: Stata Press Publication

    • -
    • Raudenbush SW, Bryk AS. 2002. Hierarchical linear models: applications and data analysis methods. 2nd ed. Thousand Oaks: Sage Publications

    • -
    • Wu S, Crespi CM, Wong WK. 2012. Comparison of methods for estimating the intraclass correlation coefficient for binary responses in cancer prevention cluster randomized trials. Contempory Clinical Trials 33: 869-880 (doi: 10.1016/j.cct.2012.05.004 -)

    • -

    Further helpful online-ressources:

    - -

    See also

    - - - - -

    Examples

    -
    library(lme4) -fit0 <- lmer(Reaction ~ 1 + (1 | Subject), sleepstudy) -icc(fit0)
    #> -#> Intraclass Correlation Coefficient for Linear mixed model -#> -#> Family : gaussian (identity) -#> Formula: Reaction ~ 1 + (1 | Subject) -#> -#> ICC (Subject): 0.3949 -#>
    -# note: ICC for random-slope-intercept model usually not -# meaningful, unless you use "adjusted = TRUE" - see 'Note'. -fit1 <- lmer(Reaction ~ Days + (Days | Subject), sleepstudy) -icc(fit1)
    #> Caution! ICC for random-slope-intercept models usually not meaningful. Use `adjusted = TRUE` to use the mean random effect variance to calculate the ICC. See 'Note' in `?icc`.
    #> -#> Intraclass Correlation Coefficient for Linear mixed model -#> -#> Family : gaussian (identity) -#> Formula: Reaction ~ Days + (Days | Subject) -#> -#> ICC (Subject): 0.4830 -#>
    icc(fit1, adjusted = TRUE)
    #> -#> Intraclass Correlation Coefficient for Generalized Linear Mixed Model -#> -#> Family : gaussian (identity) -#> Formula: ~Days | Subject Reaction ~ Days NA -#> -#> Adjusted ICC: 0.7217 -#> Conditional ICC: 0.5206 -#>
    -sleepstudy$mygrp <- sample(1:45, size = 180, replace = TRUE) -fit2 <- lmer(Reaction ~ Days + (1 | mygrp) + (1 | Subject), sleepstudy) - -icc1 <- icc(fit1)
    #> Caution! ICC for random-slope-intercept models usually not meaningful. Use `adjusted = TRUE` to use the mean random effect variance to calculate the ICC. See 'Note' in `?icc`.
    icc2 <- icc(fit2) - -print(icc1, comp = "var")
    #> -#> Intraclass Correlation Coefficient for Linear mixed model -#> -#> Family : gaussian (identity) -#> Formula: Reaction ~ Days + (Days | Subject) -#> -#> Within-group-variance: 654.941 -#> Between-group-variance: 611.898 (Subject) -#> Random-slope-variance: 35.081 (Subject.Days) -#> Slope-Intercept-covariance: 9.614 (Subject.(Intercept)) -#> Slope-Intercept-correlation: 0.066 (Subject) -#>
    print(icc2, comp = "var")
    #> -#> Intraclass Correlation Coefficient for Linear mixed model -#> -#> Family : gaussian (identity) -#> Formula: Reaction ~ Days + (1 | mygrp) + (1 | Subject) -#> -#> Within-group-variance: 946.474 -#> Between-group-variance: 14.082 (mygrp) -#> Between-group-variance: 1381.597 (Subject) -#>
    -
    # NOT RUN { -# compute ICC for Bayesian mixed model, with an ICC for each -# sample of the posterior. The print()-method then shows -# the median ICC as well as 89% HDI for the ICC. -# Change interval with print-method: -# print(icc(m, posterior = TRUE), prob = .5) - -if (requireNamespace("brms", quietly = TRUE)) { - library(dplyr) - sleepstudy$mygrp <- sample(1:5, size = 180, replace = TRUE) - sleepstudy <- sleepstudy %>% - group_by(mygrp) %>% - mutate(mysubgrp = sample(1:30, size = n(), replace = TRUE)) - m <- brms::brm( - Reaction ~ Days + (1 | mygrp / mysubgrp) + (1 | Subject), - data = sleepstudy - ) - - # by default, 89% interval - icc(m) - - # show 50% interval - icc(m, prob = .5) - - # variances based on posterior predictive distribution - icc(m, ppd = TRUE) -} -# }
    -
    -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/index.html b/docs/reference/index.html deleted file mode 100644 index e56b618..0000000 --- a/docs/reference/index.html +++ /dev/null @@ -1,467 +0,0 @@ - - - - - - - - -Function reference • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -

    Regression Models

    -

    -
    -

    svyglm.nb()

    -

    Survey-weighted negative binomial generalised linear model

    -

    svyglm.zip()

    -

    Survey-weighted zero-inflated Poisson model

    -

    Bootstrapping

    -

    -
    -

    bootstrap()

    -

    Generate nonparametric bootstrap replications

    -

    boot_ci() boot_se() boot_p() boot_est()

    -

    Standard error and confidence intervals for bootstrapped estimates

    -

    Effect Size Statistics for Anova

    -

    -
    -

    anova_stats() epsilon_sq() eta_sq() omega_sq()

    -

    Effect size statistics for anova

    -

    Statistics for Crosstables

    -

    -
    -

    cramer() phi() crosstable_statistics() xtab_statistics()

    -

    Measures of association for contingency tables

    -

    table_values()

    -

    Expected and relative table values

    -

    Weighted Statistics

    -

    -
    -

    weight() weight2()

    -

    Weight a variable

    -

    survey_median() weighted_chisqtest() weighted_correlation() weighted_mean() weighted_median() weighted_mannwhitney() weighted_sd() wtd_sd() weighted_se() weighted_ttest()

    -

    Weighted statistics for tests and variables

    -

    Other (Summary) Statistics

    -

    -
    -

    gmd()

    -

    Gini's Mean Difference

    -

    mwu() mannwhitney()

    -

    Mann-Whitney-U-Test

    -

    mean_n()

    -

    Row means with min amount of valid values

    -

    means_by_group() grpmean()

    -

    Summary of mean values by group

    -

    var_pop() sd_pop()

    -

    Calculate population variance and standard deviation

    -

    Tools for Regression Models

    -

    -
    -

    chisq_gof()

    -

    Compute model quality

    -

    cv()

    -

    Compute model quality

    -

    cv_error() cv_compare()

    -

    Test and training error from model cross-validation

    -

    Tools for Mixed Models

    -

    -
    -

    design_effect()

    -

    Design effects for two-level mixed models

    -

    samplesize_mixed() smpsize_lmm()

    -

    Sample size for linear mixed models

    -

    se_ybar()

    -

    Standard error of sample mean for mixed models

    -

    Tools for Bayesian Models

    -

    -
    -

    auto_prior()

    -

    Create default priors for brms-models

    -

    mediation()

    -

    Summary of Bayesian multivariate-response mediation-models

    -

    Find Parameters of Distribution

    -

    -
    -

    find_beta() find_beta2() find_cauchy() find_normal()

    -

    Determining distribution parameters

    -

    Miscellaneous

    -

    -
    -

    inequ_trend()

    -

    Compute trends in status inequalities

    -

    is_prime()

    -

    Find prime numbers

    -

    odds_to_rr() or_to_rr()

    -

    Get relative risks estimates from logistic regressions or odds ratio values

    -

    prop() props()

    -

    Proportions of values in a vector

    -
    - - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/inequ_trend-1.png b/docs/reference/inequ_trend-1.png deleted file mode 100644 index cd9014d8cdebeb641a009b9b0e5262102c3fee5f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 21467 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 Wo;E}WNWP@=@B!<;v-X|4`hNjM@Jby3 diff --git a/docs/reference/inequ_trend-2.png b/docs/reference/inequ_trend-2.png deleted file mode 100644 index 2802b9ccc53e25514a22391e0c0865faf662c57d..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 22634 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 Wu%N->z#Cg3=T2Wbm2txI*8c(TK!+m$ diff --git a/docs/reference/inequ_trend.html b/docs/reference/inequ_trend.html deleted file mode 100644 index 6296ae5..0000000 --- a/docs/reference/inequ_trend.html +++ /dev/null @@ -1,275 +0,0 @@ - - - - - - - - -Compute trends in status inequalities — inequ_trend • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    This method computes the proportional change of absolute - (rate differences) and relative (rate ratios) inequalities - of prevalence rates for two different status groups, as proposed - by Mackenbach et al. (2015).

    -
    - -
    inequ_trend(data, prev.low, prev.hi)
    - -

    Arguments

    - - - - - - - - - - - - - - -
    data

    A data frame that contains the variables with prevalence rates for both low -and high status groups (see 'Examples').

    prev.low

    The name of the variable with the prevalence rates for -the low status groups.

    prev.hi

    The name of the variable with the prevalence rates for -the hi status groups.

    - -

    Value

    - -

    A data frame with the prevalence rates as well as the values for the - proportional change in absolute (rd) and relative (rr) - ineqqualities.

    -

    Details

    - -

    Given the time trend of prevalence rates of an outcome for two status - groups (e.g. the mortality rates for people with lower and higher - socioeconomic status over 40 years), this function computes the - proportional change of absolute and relative inequalities, expressed - in changes in rate differences and rate ratios. The function implements - the algorithm proposed by Mackenbach et al. 2015.

    -

    References

    - -

    Mackenbach JP, Martikainen P, Menvielle G, de Gelder R. 2015. The Arithmetic of Reducing Relative and Absolute Inequalities in Health: A Theoretical Analysis Illustrated with European Mortality Data. Journal of Epidemiology and Community Health 70(7): 730-36. doi: 10.1136/jech-2015-207018

    - -

    Examples

    -
    # This example reproduces Fig. 1 of Mackenbach et al. 2015, p.5 - -# 40 simulated time points, with an initial rate ratio of 2 and -# a rate difference of 100 (i.e. low status group starts with a -# prevalence rate of 200, the high status group with 100) - -# annual decline of prevalence is 1% for the low, and 3% for the -# high status group - -n <- 40 -time <- seq(1, n, by = 1) -lo <- rep(200, times = n) -for (i in 2:n) lo[i] <- lo[i - 1] * .99 - -hi <- rep(100, times = n) -for (i in 2:n) hi[i] <- hi[i - 1] * .97 - -prev.data <- data.frame(lo, hi) - -# print values -inequ_trend(prev.data, lo, hi)
    #> $data -#> lo hi rr rd -#> 1 200.0000 100.00000 2.000000 100.0000 -#> 2 198.0000 97.00000 2.041237 101.0000 -#> 3 196.0200 94.09000 2.083324 101.9300 -#> 4 194.0598 91.26730 2.126280 102.7925 -#> 5 192.1192 88.52928 2.170120 103.5899 -#> 6 190.1980 85.87340 2.214865 104.3246 -#> 7 188.2960 83.29720 2.260533 104.9988 -#> 8 186.4131 80.79828 2.307141 105.6148 -#> 9 184.5489 78.37434 2.354711 106.1746 -#> 10 182.7034 76.02311 2.403262 106.6803 -#> 11 180.8764 73.74241 2.452814 107.1340 -#> 12 179.0677 71.53014 2.503387 107.5375 -#> 13 177.2770 69.38424 2.555004 107.8927 -#> 14 175.5042 67.30271 2.607684 108.2015 -#> 15 173.7492 65.28363 2.661451 108.4655 -#> 16 172.0117 63.32512 2.716326 108.6866 -#> 17 170.2916 61.42537 2.772333 108.8662 -#> 18 168.5886 59.58260 2.829494 109.0060 -#> 19 166.9028 57.79513 2.887834 109.1076 -#> 20 165.2337 56.06127 2.947377 109.1725 -#> 21 163.5814 54.37943 3.008148 109.2020 -#> 22 161.9456 52.74805 3.070172 109.1975 -#> 23 160.3261 51.16561 3.133474 109.1605 -#> 24 158.7229 49.63064 3.198082 109.0922 -#> 25 157.1356 48.14172 3.264022 108.9939 -#> 26 155.5643 46.69747 3.331321 108.8668 -#> 27 154.0086 45.29655 3.400008 108.7121 -#> 28 152.4685 43.93765 3.470111 108.5309 -#> 29 150.9439 42.61952 3.541660 108.3243 -#> 30 149.4344 41.34093 3.614684 108.0935 -#> 31 147.9401 40.10071 3.689214 107.8394 -#> 32 146.4607 38.89769 3.765280 107.5630 -#> 33 144.9961 37.73076 3.842915 107.2653 -#> 34 143.5461 36.59883 3.922150 106.9473 -#> 35 142.1106 35.50087 4.003019 106.6098 -#> 36 140.6895 34.43584 4.085555 106.2537 -#> 37 139.2826 33.40277 4.169794 105.8799 -#> 38 137.8898 32.40068 4.255769 105.4891 -#> 39 136.5109 31.42866 4.343517 105.0823 -#> 40 135.1458 30.48580 4.433074 104.6600 -#> -#> attr(,"class") -#> [1] "sj_inequ_trend"
    -# plot trends - here we see that the relative inequalities -# are increasing over time, while the absolute inequalities -# are first increasing as well, but later are decreasing -# (while rel. inequ. are still increasing) -plot(inequ_trend(prev.data, lo, hi))
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/is_prime.html b/docs/reference/is_prime.html deleted file mode 100644 index 373dba7..0000000 --- a/docs/reference/is_prime.html +++ /dev/null @@ -1,178 +0,0 @@ - - - - - - - - -Find prime numbers — is_prime • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    This functions checks whether a number is, or numbers in a - vector are prime numbers.

    -
    - -
    is_prime(x)
    - -

    Arguments

    - - - - - - -
    x

    An integer, or a vector of integers.

    - -

    Value

    - -

    TRUE for each prime number in x, FALSE otherwise.

    - -

    Examples

    -
    is_prime(89)
    #> [1] TRUE
    is_prime(15)
    #> [1] FALSE
    is_prime(c(1, 2, 3, 4, 5, 6, 7, 8, 9, 10))
    #> [1] TRUE TRUE TRUE FALSE TRUE FALSE TRUE FALSE FALSE FALSE
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/mean_n.html b/docs/reference/mean_n.html deleted file mode 100644 index 9cb392d..0000000 --- a/docs/reference/mean_n.html +++ /dev/null @@ -1,223 +0,0 @@ - - - - - - - - -Row means with min amount of valid values — mean_n • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    This function is similar to the SPSS MEAN.n function and computes - row means from a data.frame or matrix if at least n - values of a row are valid (and not NA).

    -
    - -
    mean_n(dat, n, digits = 2)
    - -

    Arguments

    - - - - - - - - - - - - - - -
    dat

    A data frame with at least two columns, where row means are applied.

    n

    May either be

      -
    • a numeric value that indicates the amount of valid values per row to calculate the row mean;

    • -
    • or a value between 0 and 1, indicating a proportion of valid values per row to calculate the row mean (see 'Details').

    • -

    If a row's sum of valid values is less than n, NA will be returned as row mean value.

    digits

    Numeric value indicating the number of decimal places to be used for rounding mean -value. Negative values are allowed (see 'Details').

    - -

    Value

    - -

    A vector with row mean values of df for those rows with at least n - valid values. Else, NA is returned.

    -

    Details

    - -

    Rounding to a negative number of digits means rounding to a power of - ten, so for example mean_n(df, 3, digits = -2) rounds to the - nearest hundred.

    - For n, must be a numeric value from 0 to ncol(dat). If - a row in dat has at least n non-missing values, the - row mean is returned. If n is a non-integer value from 0 to 1, - n is considered to indicate the proportion of necessary non-missing - values per row. E.g., if n = .75, a row must have at least ncol(dat) * n - non-missing values for the row mean to be calculated. See 'Examples'.

    -

    References

    - -

    r4stats.com

    - -

    Examples

    -
    dat <- data.frame(c1 = c(1,2,NA,4), - c2 = c(NA,2,NA,5), - c3 = c(NA,4,NA,NA), - c4 = c(2,3,7,8)) - -# needs at least 4 non-missing values per row -mean_n(dat, 4) # 1 valid return value
    #> [1] NA 2.75 NA NA
    -# needs at least 3 non-missing values per row -mean_n(dat, 3) # 2 valid return values
    #> [1] NA 2.75 NA 5.67
    -# needs at least 2 non-missing values per row -mean_n(dat, 2)
    #> [1] 1.50 2.75 NA 5.67
    -# needs at least 1 non-missing value per row -mean_n(dat, 1) # all means are shown
    #> [1] 1.50 2.75 7.00 5.67
    -# needs at least 50% of non-missing values per row -mean_n(dat, .5) # 3 valid return values
    #> [1] 1.50 2.75 NA 5.67
    -# needs at least 75% of non-missing values per row -mean_n(dat, .75) # 2 valid return values
    #> [1] NA 2.75 NA 5.67
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/means_by_group.html b/docs/reference/means_by_group.html deleted file mode 100644 index f82f20d..0000000 --- a/docs/reference/means_by_group.html +++ /dev/null @@ -1,340 +0,0 @@ - - - - - - - - -Summary of mean values by group — means_by_group • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Computes mean, sd and se for each sub-group (indicated by grp) - of dv.

    -
    - -
    means_by_group(
    -  x,
    -  dv,
    -  grp,
    -  weights = NULL,
    -  digits = 2,
    -  out = c("txt", "viewer", "browser"),
    -  encoding = "UTF-8",
    -  file = NULL
    -)
    -
    -grpmean(
    -  x,
    -  dv,
    -  grp,
    -  weights = NULL,
    -  digits = 2,
    -  out = c("txt", "viewer", "browser"),
    -  encoding = "UTF-8",
    -  file = NULL
    -)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    x

    A (grouped) data frame.

    dv

    Name of the dependent variable, for which the mean value, grouped -by grp, is computed.

    grp

    Factor with the cross-classifying variable, where dv is -grouped into the categories represented by grp. Numeric vectors -are coerced to factors.

    weights

    Name of variable in x that indicated the vector of -weights that will be applied to weight all observations. Default is -NULL, so no weights are used.

    digits

    Numeric, amount of digits after decimal point when rounding -estimates and values.

    out

    Character vector, indicating whether the results should be printed -to console (out = "txt") or as HTML-table in the viewer-pane -(out = "viewer") or browser (out = "browser"), of if the -results should be plotted (out = "plot", only applies to certain -functions). May be abbreviated.

    encoding

    Character vector, indicating the charset encoding used -for variable and value labels. Default is "UTF-8". Only used -when out is not "txt".

    file

    Destination file, if the output should be saved as file. -Only used when out is not "txt".

    - -

    Value

    - -

    For non-grouped data frames, means_by_group() returns a data frame with - following columns: term, mean, N, std.dev, - std.error and p.value. For grouped data frames, returns - a list of such data frames.

    -

    Details

    - -

    This function performs a One-Way-Anova with dv as dependent - and grp as independent variable, by calling - lm(count ~ as.factor(grp)). Then contrast - is called to get p-values for each sub-group. P-values indicate whether - each group-mean is significantly different from the total mean.

    - -

    Examples

    -
    data(efc) -means_by_group(efc, c12hour, e42dep)
    #> -#> # Grouped Means for average number of hours of care per week by elder's dependency -#> -#> Category | Mean | N | SD | SE | p -#> ---------------------------------------------------------- -#> independent | 9.91 | 66 | 8.01 | 0.99 | <0.001 -#> slightly dependent | 17.54 | 225 | 17.74 | 1.18 | <0.001 -#> moderately dependent | 34.52 | 306 | 41.54 | 2.37 | 0.98 -#> severely dependent | 75.90 | 304 | 61.72 | 3.54 | <0.001 -#> Total | 42.44 | 901 | 50.82 | 1.69 | -#> -#> Anova: R2=0.245; adj.R2=0.242; F=96.908; p=0.000
    -data(iris) -means_by_group(iris, Sepal.Width, Species)
    #> -#> # Grouped Means for Sepal.Width by Species -#> -#> Category | Mean | N | SD | SE | p -#> ---------------------------------------------- -#> setosa | 3.43 | 50 | 0.38 | 0.05 | <0.001 -#> versicolor | 2.77 | 50 | 0.31 | 0.04 | <0.001 -#> virginica | 2.97 | 50 | 0.32 | 0.05 | 0.04 -#> Total | 3.06 | 150 | 0.44 | 0.04 | -#> -#> Anova: R2=0.401; adj.R2=0.393; F=49.160; p=0.000
    -# also works for grouped data frames -if (require("dplyr")) { - efc %>% - group_by(c172code) %>% - means_by_group(c12hour, e42dep) -}
    #> -#> Grouped by: -#> carer's level of education: low level of education -#> -#> # Grouped Means for average number of hours of care per week by elder's dependency -#> -#> Category | Mean | N | SD | SE | p -#> ---------------------------------------------------------- -#> independent | 16.33 | 12 | 10.74 | 3.10 | 0.02 -#> slightly dependent | 15.38 | 42 | 9.55 | 1.47 | <0.001 -#> moderately dependent | 42.05 | 61 | 46.53 | 5.96 | 0.70 -#> severely dependent | 85.52 | 65 | 56.42 | 7.00 | <0.001 -#> Total | 49.81 | 180 | 52.24 | 3.89 | -#> -#> Anova: R2=0.307; adj.R2=0.295; F=25.955; p=0.000 -#> -#> -#> Grouped by: -#> carer's level of education: intermediate level of education -#> -#> # Grouped Means for average number of hours of care per week by elder's dependency -#> -#> Category | Mean | N | SD | SE | p -#> ---------------------------------------------------------- -#> independent | 7.96 | 45 | 3.91 | 0.58 | <0.001 -#> slightly dependent | 17.12 | 135 | 16.52 | 1.42 | <0.001 -#> moderately dependent | 33.55 | 163 | 41.05 | 3.22 | 0.75 -#> severely dependent | 79.71 | 163 | 63.13 | 4.94 | <0.001 -#> Total | 41.76 | 506 | 51.42 | 2.29 | -#> -#> Anova: R2=0.284; adj.R2=0.280; F=66.374; p=0.000 -#> -#> -#> Grouped by: -#> carer's level of education: high level of education -#> -#> # Grouped Means for average number of hours of care per week by elder's dependency -#> -#> Category | Mean | N | SD | SE | p -#> ---------------------------------------------------------- -#> independent | 15.20 | 5 | 18.43 | 8.24 | 0.36 -#> slightly dependent | 18.08 | 39 | 12.98 | 2.08 | 0.15 -#> moderately dependent | 28.42 | 62 | 35.64 | 4.53 | 0.67 -#> severely dependent | 63.38 | 50 | 62.69 | 8.87 | <0.001 -#> Total | 36.62 | 156 | 46.38 | 3.71 | -#> -#> Anova: R2=0.167; adj.R2=0.151; F=10.155; p=0.000 -#> -#>
    -# weighting -efc$weight <- abs(rnorm(n = nrow(efc), mean = 1, sd = .5)) -means_by_group(efc, c12hour, e42dep, weights = weight)
    #> -#> # Grouped Means for average number of hours of care per week by elder's dependency -#> -#> Category | Mean | N | SD | SE | p -#> ---------------------------------------------------------- -#> independent | 8.86 | 72 | 6.09 | 0.75 | <0.001 -#> slightly dependent | 17.28 | 225 | 15.09 | 1.01 | <0.001 -#> moderately dependent | 35.26 | 296 | 41.66 | 2.38 | 0.75 -#> severely dependent | 76.53 | 297 | 61.73 | 3.54 | <0.001 -#> Total | 42.35 | 901 | 50.62 | 1.69 | -#> -#> Anova: R2=0.256; adj.R2=0.254; F=103.078; p=0.000
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/mediation.html b/docs/reference/mediation.html deleted file mode 100644 index 1e0d1cc..0000000 --- a/docs/reference/mediation.html +++ /dev/null @@ -1,228 +0,0 @@ - - - - - - - - -Summary of Bayesian multivariate-response mediation-models — mediation • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    mediation() is a short summary for multivariate-response - mediation-models.

    -
    - -
    mediation(x, ...)
    -
    -# S3 method for brmsfit
    -mediation(x, treatment, mediator, prob = 0.9, typical = "median", ...)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - -
    x

    A stanreg, stanfit, or brmsfit object.

    ...

    Not used.

    treatment

    Character, name of the treatment variable (or direct effect) -in a (multivariate response) mediator-model. If missing, mediation() -tries to find the treatment variable automatically, however, this may fail.

    mediator

    Character, name of the mediator variable in a (multivariate -response) mediator-model. If missing, mediation() tries to find the -treatment variable automatically, however, this may fail.

    prob

    Vector of scalars between 0 and 1, indicating the mass within -the credible interval that is to be estimated.

    typical

    The typical value that will represent the Bayesian point estimate. -By default, the posterior median is returned. See typical_value -for possible values for this argument.

    - -

    Value

    - -

    A data frame with direct, indirect, mediator and - total effect of a multivariate-response mediation-model, as well as the - proportion mediated. The effect sizes are mean values of the posterior - samples.

    -

    Details

    - -

    mediation() returns a data frame with information on the - direct effect (mean value of posterior samples from treatment - of the outcome model), mediator effect (mean value of posterior - samples from mediator of the outcome model), indirect effect - (mean value of the multiplication of the posterior samples from - mediator of the outcome model and the posterior samples from - treatment of the mediation model) and the total effect (mean - value of sums of posterior samples used for the direct and indirect - effect). The proportion mediated is the indirect effect divided - by the total effect. -

    - For all values, the 90% HDIs are calculated by default. Use prob - to calculate a different interval. -

    - The arguments treatment and mediator do not necessarily - need to be specified. If missing, mediation() tries to find the - treatment and mediator variable automatically. If this does not work, - specify these variables.

    - -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/mwu.html b/docs/reference/mwu.html deleted file mode 100644 index ae5377a..0000000 --- a/docs/reference/mwu.html +++ /dev/null @@ -1,299 +0,0 @@ - - - - - - - - -Mann-Whitney-U-Test — mwu • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    This function performs a Mann-Whitney-U-Test (or Wilcoxon rank sum test, - see wilcox.test and wilcox_test) - for x, for each group indicated by grp. If grp - has more than two categories, a comparison between each combination of - two groups is performed.

    - The function reports U, p and Z-values as well as effect size r - and group-rank-means.

    -
    - -
    mwu(
    -  data,
    -  x,
    -  grp,
    -  distribution = "asymptotic",
    -  out = c("txt", "viewer", "browser"),
    -  encoding = "UTF-8",
    -  file = NULL
    -)
    -
    -mannwhitney(
    -  data,
    -  x,
    -  grp,
    -  distribution = "asymptotic",
    -  out = c("txt", "viewer", "browser"),
    -  encoding = "UTF-8",
    -  file = NULL
    -)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    data

    A data frame.

    x

    Bare (unquoted) variable name, or a character vector with the variable name.

    grp

    Bare (unquoted) name of the cross-classifying variable, where -x is grouped into the categories represented by grp, -or a character vector with the variable name.

    distribution

    Indicates how the null distribution of the test statistic should be computed. -May be one of "exact", "approximate" or "asymptotic" -(default). See wilcox_test for details.

    out

    Character vector, indicating whether the results should be printed -to console (out = "txt") or as HTML-table in the viewer-pane -(out = "viewer") or browser (out = "browser"), of if the -results should be plotted (out = "plot", only applies to certain -functions). May be abbreviated.

    encoding

    Character vector, indicating the charset encoding used -for variable and value labels. Default is "UTF-8". Only used -when out is not "txt".

    file

    Destination file, if the output should be saved as file. -Only used when out is not "txt".

    - -

    Value

    - -

    (Invisibly) returns a data frame with U, p and Z-values for each group-comparison - as well as effect-size r; additionally, group-labels and groups' n's are - also included.

    -

    Note

    - -

    This function calls the wilcox_test with formula. If grp - has more than two groups, additionally a Kruskal-Wallis-Test (see kruskal.test) - is performed.

    - Interpretation of effect sizes, as a rule-of-thumb:

      -
    • small effect >= 0.1

    • -
    • medium effect >= 0.3

    • -
    • large effect >= 0.5

    • -
    - - -

    Examples

    -
    data(efc) -# Mann-Whitney-U-Tests for elder's age by elder's dependency. -mwu(efc, e17age, e42dep)
    #> -#> # Mann-Whitney-U-Test -#> -#> Groups 1 = independent (n = 65) | 2 = slightly dependent (n = 224): -#> U = 7635.000, W = 5490.000, p = 0.003, Z = -3.020 -#> effect-size r = 0.178 -#> rank-mean(1) = 117.46 -#> rank-mean(2) = 152.99 -#> -#> Groups 1 = independent (n = 65) | 3 = moderately dependent (n = 304): -#> U = 8692.000, W = 6547.000, p < 0.001, Z = -4.273 -#> effect-size r = 0.222 -#> rank-mean(1) = 133.72 -#> rank-mean(3) = 195.96 -#> -#> Groups 1 = independent (n = 65) | 4 = severely dependent (n = 297): -#> U = 7905.500, W = 5760.500, p < 0.001, Z = -5.096 -#> effect-size r = 0.268 -#> rank-mean(1) = 121.62 -#> rank-mean(4) = 194.60 -#> -#> Groups 2 = slightly dependent (n = 224) | 3 = moderately dependent (n = 304): -#> U = 54664.500, W = 29464.500, p = 0.008, Z = -2.647 -#> effect-size r = 0.115 -#> rank-mean(2) = 244.04 -#> rank-mean(3) = 279.58 -#> -#> Groups 2 = slightly dependent (n = 224) | 4 = severely dependent (n = 297): -#> U = 51007.500, W = 25807.500, p < 0.001, Z = -4.386 -#> effect-size r = 0.192 -#> rank-mean(2) = 227.71 -#> rank-mean(4) = 286.11 -#> -#> Groups 3 = moderately dependent (n = 304) | 4 = severely dependent (n = 297): -#> U = 87819.500, W = 41459.500, p = 0.083, Z = -1.732 -#> effect-size r = 0.071 -#> rank-mean(3) = 288.88 -#> rank-mean(4) = 313.41 -#> -#> # Kruskal-Wallis-Test -#> -#> chi-squared = 38.476 -#> df = 3 -#> p < 0.001
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/nhanes_sample.html b/docs/reference/nhanes_sample.html deleted file mode 100644 index f389f06..0000000 --- a/docs/reference/nhanes_sample.html +++ /dev/null @@ -1,168 +0,0 @@ - - - - - - - - -Sample dataset from the National Health and Nutrition Examination Survey — nhanes_sample • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Selected variables from the National Health and Nutrition Examination - Survey that are used in the example from Lumley (2010), Appendix E. - See svyglm.nb for examples.

    -
    - - - -

    References

    - -

    Lumley T (2010). Complex Surveys: a guide to analysis using R. Wiley

    - -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/odds_to_rr.html b/docs/reference/odds_to_rr.html deleted file mode 100644 index 01688bc..0000000 --- a/docs/reference/odds_to_rr.html +++ /dev/null @@ -1,256 +0,0 @@ - - - - - - - - -Get relative risks estimates from logistic regressions or odds ratio values — odds_to_rr • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    odds_to_rr() converts odds ratios from a logistic regression - model (including mixed models) into relative risks; or_to_rr() - converts a single odds ratio estimate into a relative risk estimate.

    -
    - -
    odds_to_rr(fit)
    -
    -or_to_rr(or, p0)
    - -

    Arguments

    - - - - - - - - - - - - - - -
    fit

    A fitted binomial generalized linear (mixed) model with logit-link function -(logistic (multilevel) regression model).

    or

    Numeric, an odds ratio estimate.

    p0

    Numeric, the risk of having a positive outcome in the control or -unexposed group (reference group), i.e. the number of outcome or "successes" -in the control divided by the total number of observations in the control -group.

    - -

    Value

    - -

    A data frame with relative risks and lower/upper confidence interval for - the relative risks estimates; for or_to_rr(), the risk ratio - estimate.

    -

    Details

    - -

    This function extracts the odds ratios (exponentiated model coefficients) - from logistic regressions (fitted with glm or glmer) - and their related confidence intervals, and transforms these values - into relative risks (and their related confidence intervals). -

    - The formula for transformation is based on Zhang and Yu (1998), - Wang (2013) and Grant (2014): - RR <- OR / (1 - P0 + (P0 * OR)), where OR is the odds - ratio and P0 indicates the proportion of the incidence in - the outcome variable for the control group (reference group).

    -

    References

    - -

    Grant RL. 2014. Converting an odds ratio to a range of plausible relative risks for better communication of research findings. BMJ 348:f7450. doi: 10.1136/bmj.f7450 - -

    - Wang Z. 2013. Converting Odds Ratio to Relative Risk in Cohort Studies with Partial Data Information. J Stat Soft 2013;55. doi: 10.18637/jss.v055.i05 - -

    - Zhang J, Yu KF. 1998. What's the Relative Risk? A Method of Correcting the Odds Ratio in Cohort Studies of Common Outcomes. JAMA; 280(19): 1690-1. doi: 10.1001/jama.280.19.1690

    - -

    Examples

    -
    library(sjmisc) -library(lme4)
    #> Loading required package: Matrix
    # create binary response -sleepstudy$Reaction.dicho <- dicho(sleepstudy$Reaction, dich.by = "median") -# fit model -fit <- glmer(Reaction.dicho ~ Days + (Days | Subject), - data = sleepstudy, family = binomial("logit")) -# convert to relative risks -odds_to_rr(fit)
    #> Parameter Odds Ratio Risk Ratio CI_low CI_high -#> 1 (Intercept) 0.02201714 0.04308565 0.004411356 0.3597023 -#> 2 Days 2.43719045 1.41812942 1.212151446 1.5885357
    - -data(efc) -# create binary response -y <- ifelse(efc$neg_c_7 < median(na.omit(efc$neg_c_7)), 0, 1) -# create data frame for fitted model -mydf <- data.frame( - y = as.factor(y), - sex = to_factor(efc$c161sex), - dep = to_factor(efc$e42dep), - barthel = efc$barthtot, - education = to_factor(efc$c172code) -) -# fit model -fit <- glm(y ~., data = mydf, family = binomial(link = "logit")) -# convert to relative risks -odds_to_rr(fit)
    #> Waiting for profiling to be done...
    #> Parameter Odds Ratio Risk Ratio CI_low CI_high -#> 1 (Intercept) 2.014220 1.2942650 0.8005943 1.6101037 -#> 2 sex2 1.913887 1.3686646 1.1632760 1.5625768 -#> 3 dep2 1.624596 1.4468228 0.8511852 2.3180826 -#> 4 dep3 3.080617 2.1859172 1.4029261 3.1216957 -#> 5 dep4 2.484804 1.9230853 1.0500095 3.0354848 -#> 6 barthel 0.970638 0.9865251 0.9822535 0.9906641 -#> 7 education2 1.254234 1.1027736 0.9233955 1.2698368 -#> 8 education3 1.327901 1.1280723 0.9057139 1.3294549
    -# replicate OR/RR for coefficient "sex" from above regression -# p0 ~ .44, or ~ 1.914 -prop.table(table(mydf$y, mydf$sex))
    #> -#> 1 2 -#> 0 0.1324355 0.3153760 -#> 1 0.1054994 0.4466891
    or_to_rr(1.914, 0.1055 / (.1324 + .1055))
    #> [1] 1.361962
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/overdisp-1.png b/docs/reference/overdisp-1.png deleted file mode 100644 index 24d2460b789b75833912b1c391064bdc34ac4a85..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 8143 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 F@$XfZdDj2{ diff --git a/docs/reference/overdisp.html b/docs/reference/overdisp.html deleted file mode 100644 index eabecf1..0000000 --- a/docs/reference/overdisp.html +++ /dev/null @@ -1,209 +0,0 @@ - - - - - - - - -Deprecated functions — overdisp • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    A list of deprecated functions.

    -
    - -
    overdisp(x, ...)
    -
    -zero_count(x, ...)
    -
    -pca(x, ...)
    -
    -pca_rotate(x, ...)
    -
    -r2(x)
    -
    -icc(x)
    -
    -p_value(x, ...)
    -
    -se(x, ...)
    - -

    Arguments

    - - - - - - - - - - -
    x

    An object.

    ...

    Currently not used.

    - -

    Value

    - -

    Nothing.

    - -
    - -
    - - -
    - - -
    -

    Site built with pkgdown 1.4.1.

    -
    - -
    -
    - - - - - - - - diff --git a/docs/reference/p_value.html b/docs/reference/p_value.html deleted file mode 100644 index 4eeb008..0000000 --- a/docs/reference/p_value.html +++ /dev/null @@ -1,251 +0,0 @@ - - - - - - - - -Get p-values from regression model objects — p_value • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    This function returns the p-values for fitted model objects.

    - -
    - -
    p_value(fit, ...)
    -
    -# S3 method for lmerMod
    -p_value(fit, p.kr = FALSE, ...)
    - -

    Arguments

    - - - - - - - - - - - - - - -
    fit

    A model object.

    ...

    Currently not used.

    p.kr

    Logical, if TRUE, the computation of p-values is based on -conditional F-tests with Kenward-Roger approximation for the df (see -'Details').

    - -

    Value

    - -

    A data.frame with the model coefficients' names (term), - p-values (p.value) and standard errors (std.error).

    - -

    Details

    - -

    For linear mixed models (lmerMod-objects), the computation of - p-values (if p.kr = TRUE) is based on conditional F-tests - with Kenward-Roger approximation for the df, using the - pbkrtest-package. If pbkrtest is not available or - p.kr = FALSE, or if x is a glmerMod-object, - computation of p-values is based on normal-distribution assumption, - treating the t-statistics as Wald z-statistics. -

    - If p-values already have been computed (e.g. for merModLmerTest-objects - from the lmerTest-package), these will be returned. -

    - The print()-method has a summary-argument, that - in - case p.kr = TRUE - also prints information on the approximated - degrees of freedom (see 'Examples'). A shortcut is the - summary()-method, which simply calls print(..., summary = TRUE).

    - - -

    Examples

    -
    data(efc) -# linear model fit -fit <- lm(neg_c_7 ~ e42dep + c172code, data = efc) -p_value(fit)
    #> term p.value std.error -#> 1 (Intercept) 0.000 0.566 -#> 2 e42dep 0.000 0.133 -#> 3 c172code 0.207 0.198
    -# Generalized Least Squares fit -library(nlme)
    #> -#> Attaching package: 'nlme'
    #> The following object is masked from 'package:lme4': -#> -#> lmList
    #> The following object is masked from 'package:dplyr': -#> -#> collapse
    fit <- gls(follicles ~ sin(2*pi*Time) + cos(2*pi*Time), Ovary, - correlation = corAR1(form = ~ 1 | Mare)) -p_value(fit)
    #> term p.value std.error -#> 1 (Intercept) 0.000 0.665 -#> 2 sin(2 * pi * Time) 0.000 0.645 -#> 3 cos(2 * pi * Time) 0.198 0.698
    -# lme4-fit -library(lme4) -sleepstudy$mygrp <- sample(1:45, size = 180, replace = TRUE) -fit <- lmer(Reaction ~ Days + (1 | mygrp) + (1 | Subject), sleepstudy)
    #> boundary (singular) fit: see ?isSingular
    pv <- p_value(fit, p.kr = TRUE)
    #> Computing p-values via Kenward-Roger approximation. Use `p.kr = FALSE` if computation takes too long.
    -# normal output -pv
    #> term p.value std.error -#> 1 (Intercept) 0 9.766 -#> 2 Days 0 0.815
    -# add information on df and t-statistic -print(pv, summary = TRUE)
    #> term p.value std.error df statistic -#> 1 (Intercept) 0 9.766 22.785 25.742 -#> 2 Days 0 0.815 160.682 12.838
    # or -summary(pv)
    #> term p.value std.error df statistic -#> 1 (Intercept) 0 9.766 22.785 25.742 -#> 2 Days 0 0.815 160.682 12.838
    -
    -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/pca.html b/docs/reference/pca.html deleted file mode 100644 index c6bb7f8..0000000 --- a/docs/reference/pca.html +++ /dev/null @@ -1,242 +0,0 @@ - - - - - - - - -Tidy summary of Principal Component Analysis — pca • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    ...

    - -
    - -
    pca(x)
    -
    -pca_rotate(x, nf = NULL, rotation = c("varimax", "quartimax", "promax",
    -  "oblimin", "simplimax", "cluster", "none"))
    - -

    Arguments

    - - - - - - - - - - - - - - -
    x

    A data frame or a prcomp object.

    nf

    Number of components to extract. If rotation = "varmiax" -and nf = NULL, number of components is based on the Kaiser-criteria.

    rotation

    Rotation of the factor loadings. May be one of -"varimax", "quartimax", "promax", "oblimin", "simplimax", "cluster" -or "none".

    - -

    Value

    - -

    A tidy data frame with either all loadings of principal components - (for pca()) or a rotated loadings matrix (for pca_rotate()).

    - -

    Details

    - -

    The print()-method for pca_rotate() has a - cutoff-argument, which is a scalar between 0 and 1, indicating - which (absolute) values from the loadings should be blank in the - output. By default, all loadings below .1 (or -.1) are not shown.

    - - -

    Examples

    -
    data(efc) -# recveive first item of COPE-index scale -start <- which(colnames(efc) == "c82cop1") -# recveive last item of COPE-index scale -end <- which(colnames(efc) == "c90cop9") - -# extract principal components -pca(efc[, start:end])
    #> PC1 PC2 PC3 PC4 PC5 PC6 PC7 PC8 -#> Standard deviation 1.7976 1.1311 0.9665 0.8609 0.8156 0.7951 0.7426 0.7251 -#> Eigenvalue 3.2314 1.2793 0.9342 0.7412 0.6651 0.6322 0.5515 0.5258 -#> Proportion variance 0.3590 0.1421 0.1038 0.0824 0.0739 0.0702 0.0613 0.0584 -#> Cumulative variance 0.3590 0.5012 0.6050 0.6873 0.7612 0.8315 0.8928 0.9512 -#> PC9 -#> Standard deviation 0.6627 -#> Eigenvalue 0.4392 -#> Proportion variance 0.0488 -#> Cumulative variance 1.0000
    -# extract principal components, varimax-rotation. -# number of components based on Kaiser-criteria -pca_rotate(efc[, start:end])
    #> variable PC1 PC2 -#> 1 c82cop1 0.2911 0.5964 -#> 2 c83cop2 -0.5976 -0.4235 -#> 3 c84cop3 -0.6885 -0.1564 -#> 4 c85cop4 -0.726 -0.119 -#> 5 c86cop5 -0.6426 -#> 6 c87cop6 -0.6934 0.1213 -#> 7 c88cop7 -0.6768 -0.3796 -#> 8 c89cop8 0.6364 -#> 9 c90cop9 0.7542 -#> -#> PC1 PC2 -#> Proportion variance 0.312 0.190 -#> Cumulative variance 0.312 0.501 -#> Proportion explained 0.622 0.378 -#> Cumulative explained 0.622 1.000
    -
    -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/pred_accuracy.html b/docs/reference/pred_accuracy.html deleted file mode 100644 index 84c1bbc..0000000 --- a/docs/reference/pred_accuracy.html +++ /dev/null @@ -1,254 +0,0 @@ - - - - - - - - -Accuracy of predictions from model fit — pred_accuracy • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    This function calculates the predictive accuracy of linear - or logistic regression models.

    - -
    - -
    pred_accuracy(data, fit, method = c("cv", "boot"), k = 5, n = 1000)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - -
    data

    A data frame.

    fit

    Fitted model object of class lm or glm, the latter -being a logistic regression model (binary response).

    method

    Character string, indicating whether crossvalidation -(method = "cv") or bootstrapping (method = "boot") -is used to compute the accuracy values.

    k

    The number of folds for the kfold-crossvalidation.

    n

    Number of bootstraps to be generated.

    - -

    Value

    - -

    A list with two values: The accuracy of the model predictions, i.e. - the proportion of accurately predicted values from the model and - its standard error, std.error.

    - -

    Details

    - -

    For linar models, the accuracy is the correlation coefficient - between the actual and the predicted value of the outcome. For - logistic regression models, the accuracy corresponds to the - AUC-value, calculated with the auc-function. -

    - The accuracy is the mean value of multiple correlation resp. - AUC-values, which are either computed with crossvalidation - or nonparametric bootstrapping (see argument method). - The standard error is the standard deviation of the computed - correlation resp. AUC-values.

    - -

    See also

    - - - - -

    Examples

    -
    data(efc) -fit <- lm(neg_c_7 ~ barthtot + c161sex, data = efc) - -# accuracy for linear model, with crossvalidation -pred_accuracy(efc, fit)
    #> -#> # Accuracy of Model Predictions -#> -#> Accuracy: 41.12% -#> SE: 6.56%-points -#> Method: Correlation between observed and predicted
    -# accuracy for linear model, with bootstrapping -pred_accuracy(efc, fit, method = "boot", n = 100)
    #> -#> # Accuracy of Model Predictions -#> -#> Accuracy: 41.30% -#> SE: 2.78%-points -#> Method: Correlation between observed and predicted
    -# accuracy for logistic regression, with crossvalidation -efc$services <- sjmisc::dicho(efc$tot_sc_e, dich.by = 0, as.num = TRUE) -fit <- glm(services ~ neg_c_7 + c161sex + e42dep, - data = efc, family = binomial(link = "logit")) -pred_accuracy(efc, fit)
    #> -#> # Accuracy of Model Predictions -#> -#> Accuracy: 58.38% -#> SE: 3.92%-points -#> Method: Area under Curve
    -
    -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/pred_vars.html b/docs/reference/pred_vars.html deleted file mode 100644 index 4adf9d6..0000000 --- a/docs/reference/pred_vars.html +++ /dev/null @@ -1,417 +0,0 @@ - - - - - - - - -Access information from model objects — link_inverse • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    Several functions to retrieve information from model objects, - like variable names, link-inverse function, model frame, - model family etc., in a tidy and consistent way.

    - -
    - -
    link_inverse(x, multi.resp = FALSE, mv = FALSE)
    -
    -model_family(x, multi.resp = FALSE, mv = FALSE)
    -
    -model_frame(x, fe.only = TRUE)
    -
    -pred_vars(x, ...)
    -
    -# S3 method for default
    -pred_vars(x, fe.only = FALSE, ...)
    -
    -# S3 method for glmmTMB
    -pred_vars(x, fe.only = FALSE, zi = FALSE,
    -  disp = FALSE, ...)
    -
    -# S3 method for MixMod
    -pred_vars(x, fe.only = FALSE, zi = FALSE, ...)
    -
    -re_grp_var(x)
    -
    -grp_var(x)
    -
    -resp_val(x)
    -
    -resp_var(x, combine = TRUE)
    -
    -var_names(x)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    x

    A fitted model; for var_names(), x may also be a -character vector.

    mv, multi.resp

    Logical, if TRUE and model is a multivariate response -model from a brmsfit object or of class stanmvreg, then a -list of values (one for each regression) is returned.

    fe.only

    Logical, if TRUE and x is a mixed effects -model, model_frame() returns the model frame for fixed effects only, -and pred_vars() returns only fixed effects terms. Note that the default -for model_frame() is fe.only = TRUE, while for pred_vars() -the default is fe.only = FALSE.

    ...

    Currently not used.

    zi

    Logical, if TRUE and model has a zero-inflation-formula, -the variable(s) used in this formula are also returned.

    disp

    Logical, if TRUE and model is of class glmmTMB and -has a dispersion-formula, the variable(s) used in the dispersion-formula -are also returned.

    combine

    Logical, if TRUE and the response is a matrix-column, -the name of the response matches the notation in formula, and would for -instance also contain patterns like "cbind(...)". Else, the original -variable names from the matrix-column are returned. See 'Examples'.

    - -

    Value

    - -

    For pred_vars() and resp_var(), the name(s) of the - response or predictor variables from x as character vector. - resp_val() returns the values from x's response vector. - re_grp_var() returns the group factor of random effects in - mixed models, or NULL if x has no such random effects term - (grp_var() is an alias for re_grp_var()). -

    - link_inverse() returns, if known, the inverse link function from - x; else NULL for those models where the inverse link function - can't be identified. -

    - model_frame() is similar to model.frame(), - but should also work for model objects that don't have a S3-generic for - model.frame(). -

    - var_names() returns the "cleaned" variable - names, i.e. things like s() for splines or log() are - removed. -

    - model_family() returns a list with information about the - model family (see 'Details').

    - -

    Details

    - -

    model_family() returns a list with information about the - model family for many different model objects. Following information - is returned, where all values starting with is_ are logicals.

      -
    • is_bin: family is binomial (but not negative binomial)

    • -
    • is_pois: family is either poisson or negative binomial

    • -
    • is_negbin: family is negative binomial

    • -
    • is_count: model is a count model (i.e. family is either poisson or negative binomial)

    • -
    • is_beta: family is beta

    • -
    • is_logit: model has logit link

    • -
    • is_linear: family is gaussian

    • -
    • is_ordinal: family is ordinal or cumulative link

    • -
    • is_categorical: family is categorical link

    • -
    • is_zeroinf: model has zero-inflation component

    • -
    • is_multivariate: model is a multivariate response model (currently only works for brmsfit objects)

    • -
    • is_trial: model response contains additional information about the trials

    • -
    • link.fun: the link-function

    • -
    • family: the family-object

    • -

    model_frame() slighty differs from model.frame(), especially - for spline terms and matrix-variables created with cbind() (for example - in binomial models, where the response is a combination of successes and - trials) . Where model.frame() returns a matrix for splines, - model_frame() returns the data of the original variable and uses - the same column name as in the data-argument from the model-function. - This makes it easier, for instance, to get data that should be used as new - data in predict(). For matrix-variables created with cbind(), - model_frame() returns the original variable as matrix and - additionally each column as own variable. See 'Examples'.

    - - -

    Examples

    -
    data(efc) -fit <- lm(neg_c_7 ~ e42dep + c161sex, data = efc) - -pred_vars(fit)
    #> [1] "e42dep" "c161sex"
    resp_var(fit)
    #> [1] "neg_c_7"
    resp_val(fit)
    #> [1] 12 20 11 10 12 19 15 11 15 10 28 18 13 18 16 13 11 11 13 17 11 9 8 14 11 -#> [26] 23 11 15 11 25 9 15 20 9 10 19 8 17 16 17 14 14 16 19 17 15 16 19 17 10 -#> [51] 14 14 9 12 25 17 22 13 15 7 19 11 15 13 11 14 17 7 15 11 19 10 10 20 10 -#> [76] 12 15 7 13 12 16 10 15 15 15 25 11 10 11 14 10 10 13 10 11 18 14 12 10 9 -#> [101] 13 14 10 10 13 13 12 12 18 7 13 14 11 16 15 15 9 17 17 22 16 14 9 13 9 -#> [126] 17 17 9 13 14 12 18 7 10 12 20 12 14 12 10 12 11 14 11 13 10 12 12 10 9 -#> [151] 15 12 11 14 16 18 11 11 14 14 12 10 9 12 8 10 11 10 11 13 7 10 11 12 15 -#> [176] 10 16 13 20 7 12 17 14 10 12 9 7 16 13 14 8 8 20 7 15 7 9 14 11 12 -#> [201] 11 12 18 8 13 16 8 13 14 11 8 12 24 11 11 13 9 13 20 12 16 15 20 10 12 -#> [226] 12 12 11 10 9 10 8 10 12 10 9 11 7 9 11 11 12 11 14 12 12 19 12 15 11 -#> [251] 17 8 13 11 10 8 10 19 10 18 8 11 9 10 13 11 9 9 8 9 8 8 11 9 10 -#> [276] 12 9 17 20 12 7 9 7 8 8 14 7 10 8 16 9 16 13 8 20 16 9 9 8 15 -#> [301] 16 19 8 12 17 12 14 11 9 11 9 8 10 8 10 15 13 8 10 10 12 14 12 7 8 -#> [326] 16 9 16 7 8 13 9 7 9 9 8 17 7 8 9 7 10 10 11 18 9 10 13 8 12 -#> [351] 9 7 10 8 7 7 12 12 9 8 10 18 16 11 15 10 9 9 12 18 12 13 17 9 8 -#> [376] 7 16 12 14 15 10 9 17 17 21 17 17 15 9 12 12 22 11 14 11 9 8 12 13 13 -#> [401] 9 10 12 9 11 13 11 17 10 18 10 16 10 10 14 11 11 10 11 8 15 12 10 13 13 -#> [426] 13 12 9 13 10 15 18 11 14 11 12 12 14 15 8 10 9 7 8 18 7 7 7 11 8 -#> [451] 11 11 16 13 14 14 7 9 7 17 7 10 9 9 7 12 14 7 10 20 7 8 9 11 10 -#> [476] 14 7 8 8 10 8 12 10 14 11 8 11 17 10 22 8 9 19 11 18 16 18 15 19 10 -#> [501] 13 15 7 8 22 8 20 19 10 7 25 9 11 7 11 9 8 12 9 20 7 12 9 9 8 -#> [526] 10 8 17 12 9 9 8 7 8 9 17 17 8 9 9 10 9 7 8 27 25 14 28 16 11 -#> [551] 15 7 9 7 7 8 13 19 15 14 20 20 14 10 11 15 7 14 11 13 16 13 10 17 10 -#> [576] 12 11 7 8 15 13 11 7 18 17 12 18 17 13 10 19 7 8 10 18 17 19 8 12 10 -#> [601] 14 10 13 9 8 8 9 15 11 7 8 11 21 8 11 10 10 11 10 11 9 13 17 9 8 -#> [626] 8 9 13 14 14 9 12 8 11 10 11 11 10 10 10 12 13 7 8 12 8 8 13 10 12 -#> [651] 16 8 10 13 10 9 10 12 11 9 10 9 13 10 9 10 8 7 8 7 7 9 8 11 9 -#> [676] 10 12 11 7 16 12 10 8 12 23 10 10 18 13 12 18 9 12 13 9 7 10 7 8 17 -#> [701] 11 14 11 23 14 8 7 15 8 12 9 15 17 13 13 10 20 10 11 25 10 12 10 12 10 -#> [726] 8 14 8 18 8 15 11 12 10 7 10 13 14 7 7 14 11 11 11 9 7 15 9 9 18 -#> [751] 8 15 7 8 13 8 8 9 7 7 9 8 8 13 10 11 13 11 8 12 8 9 16 11 19 -#> [776] 12 12 9 10 10 9 13 7 11 13 10 10 13 9 14 15 15 9 10 8 8 9 9 9 9 -#> [801] 9 13 9 12 14 12 8 10 7 22 18 16 13 15 24 11 14 12 11 10 7 10 10 12 10 -#> [826] 7 9 16 14 12 9 10 8 9 7 8 10 9 8 10 10 7 11 8 10 11 14 7 8 10 -#> [851] 10 11 11 8 8 9 11 7 7 8 9 9 7 13 15 11 24 8 9 7 10 15 18 22 18 -#> [876] 9 11 14 7 9 17 23 12 13 15 8 8 14 10 10 -#> attr(,"label") -#> [1] "Negative impact with 7 items"
    -link_inverse(fit)(2.3)
    #> [1] 2.3
    -# example from ?stats::glm -counts <- c(18, 17, 15, 20, 10, 20, 25, 13, 12) -outcome <- gl(3, 1, 9) -treatment <- gl(3, 3) -m <- glm(counts ~ outcome + treatment, family = poisson()) - -link_inverse(m)(.3)
    #> [1] 1.349859
    # same as -exp(.3)
    #> [1] 1.349859
    -outcome <- as.numeric(outcome) -m <- glm(counts ~ log(outcome) + as.factor(treatment), family = poisson()) -var_names(m)
    #> [1] "counts" "outcome" "treatment"
    -# model.frame and model_frame behave slightly different -library(splines) -m <- lm(neg_c_7 ~ e42dep + ns(c160age, knots = 2), data = efc) -head(model.frame(m))
    #> neg_c_7 e42dep ns(c160age, knots = 2).1 ns(c160age, knots = 2).2 -#> 1 12 3 0.49465270 0.08689310 -#> 2 20 3 0.49766116 0.04922034 -#> 3 11 3 0.45855117 0.53896628 -#> 4 10 4 0.47509770 0.33176607 -#> 5 12 4 0.50819077 -0.08263434 -#> 6 19 4 0.49465270 0.08689310
    head(model_frame(m))
    #> neg_c_7 e42dep c160age -#> 1 12 3 56 -#> 2 20 3 54 -#> 3 11 3 80 -#> 4 10 4 69 -#> 5 12 4 47 -#> 6 19 4 56
    -library(lme4) -data(cbpp) -cbpp$trials <- cbpp$size - cbpp$incidence -m <- glm(cbind(incidence, trials) ~ period, data = cbpp, family = binomial) -head(model.frame(m))
    #> cbind(incidence, trials).incidence cbind(incidence, trials).trials period -#> 1 2 12 1 -#> 2 3 9 2 -#> 3 4 5 3 -#> 4 0 5 4 -#> 5 3 19 1 -#> 6 1 17 2
    head(model_frame(m))
    #> cbind(incidence, trials).incidence cbind(incidence, trials).trials period -#> 1 2 12 1 -#> 2 3 9 2 -#> 3 4 5 3 -#> 4 0 5 4 -#> 5 3 19 1 -#> 6 1 17 2 -#> incidence trials -#> 1 2 12 -#> 2 3 9 -#> 3 4 5 -#> 4 0 5 -#> 5 3 19 -#> 6 1 17
    -resp_var(m, combine = TRUE)
    #> [1] "cbind(incidence, trials)"
    resp_var(m, combine = FALSE)
    #> [1] "incidence" "trials"
    -# get random effects grouping factor from mixed models -library(lme4) -data(sleepstudy) -m <- lmer(Reaction ~ Days + (1 + Days | Subject), data = sleepstudy) -re_grp_var(m)
    #> [1] "Subject"
    -# get model predictors, with and w/o dispersion formula -
    # NOT RUN { -library(glmmTMB) -data("Salamanders") -m <- glmmTMB( - count ~ spp + cover + mined + poly(DOP, 3) + (1 | site), - ziformula = ~spp + mined, - dispformula = ~DOY, - data = Salamanders, - family = nbinom2 -) - -pred_vars(m) -pred_vars(m, fe.only = TRUE) -pred_vars(m, disp = TRUE) -# }
    -
    -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/prop.html b/docs/reference/prop.html deleted file mode 100644 index de6fb55..0000000 --- a/docs/reference/prop.html +++ /dev/null @@ -1,294 +0,0 @@ - - - - - - - - -Proportions of values in a vector — prop • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    prop() calculates the proportion of a value or category - in a variable. props() does the same, but allows for - multiple logical conditions in one statement. It is similar - to mean() with logical predicates, however, both - prop() and props() work with grouped data frames.

    -
    - -
    prop(data, ..., weights = NULL, na.rm = TRUE, digits = 4)
    -
    -props(data, ..., na.rm = TRUE, digits = 4)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - -
    data

    A data frame. May also be a grouped data frame (see 'Examples').

    ...

    One or more value pairs of comparisons (logical predicates). Put -variable names the left-hand-side and values to match on the -right hand side. Expressions may be quoted or unquoted. See -'Examples'.

    weights

    Vector of weights that will be applied to weight all observations. -Must be a vector of same length as the input vector. Default is -NULL, so no weights are used.

    na.rm

    Logical, whether to remove NA values from the vector when the -proportion is calculated. na.rm = FALSE gives you the raw -percentage of a value in a vector, na.rm = TRUE the valid -percentage.

    digits

    Amount of digits for returned values.

    - -

    Value

    - -

    For one condition, a numeric value with the proportion of the values - inside a vector. For more than one condition, a tibble with one column - of conditions and one column with proportions. For grouped data frames, - returns a tibble with one column per group with grouping categories, - followed by one column with proportions per condition.

    -

    Details

    - -

    prop() only allows one logical statement per comparison, - while props() allows multiple logical statements per comparison. - However, prop() supports weighting of variables before calculating - proportions, and comparisons may also be quoted. Hence, prop() - also processes comparisons, which are passed as character vector - (see 'Examples').

    - -

    Examples

    -
    data(efc) - -# proportion of value 1 in e42dep -prop(efc, e42dep == 1)
    #> [1] 0.0733
    -# expression may also be completely quoted -prop(efc, "e42dep == 1")
    #> [1] 0.0733
    -# use "props()" for multiple logical statements -props(efc, e17age > 70 & e17age < 80)
    #> [1] 0.3199
    -# proportion of value 1 in e42dep, and all values greater -# than 2 in e42dep, including missing values. will return a tibble -prop(efc, e42dep == 1, e42dep > 2, na.rm = FALSE)
    #> condition prop -#> 1 e42dep==1 0.0727 -#> 2 e42dep>2 0.6718
    -# for factors or character vectors, use quoted or unquoted values -library(sjmisc) -# convert numeric to factor, using labels as factor levels -efc$e16sex <- to_label(efc$e16sex) -efc$n4pstu <- to_label(efc$n4pstu) - -# get proportion of female older persons -prop(efc, e16sex == female)
    #> [1] 0.6715
    -# get proportion of male older persons -prop(efc, e16sex == "male")
    #> [1] 0.3285
    -# "props()" needs quotes around non-numeric factor levels -props(efc, - e17age > 70 & e17age < 80, - n4pstu == 'Care Level 1' | n4pstu == 'Care Level 3' -)
    #> condition prop -#> 1 e17age>70&e17age<80 0.3199 -#> 2 n4pstu==CareLevel1|n4pstu==CareLevel3 0.3137
    -# also works with pipe-chains -library(dplyr) -efc %>% prop(e17age > 70)
    #> [1] 0.8092
    efc %>% prop(e17age > 70, e16sex == 1)
    #> condition prop -#> 1 e17age>70 0.8092 -#> 2 e16sex==1 0.0000
    -# and with group_by -efc %>% - group_by(e16sex) %>% - prop(e42dep > 2)
    #> Warning: Factor `e16sex` contains implicit NA, consider using `forcats::fct_explicit_na`
    #> Warning: Factor `e16sex` contains implicit NA, consider using `forcats::fct_explicit_na`
    #> elder's gender e42dep>2 -#> 1 male 0.6847 -#> 2 female 0.6744
    -efc %>% - select(e42dep, c161sex, c172code, e16sex) %>% - group_by(c161sex, c172code) %>% - prop(e42dep > 2, e16sex == 1)
    #> carer's gender carer's level of education e42dep>2 e16sex==1 -#> 1 Male low level of education 0.6829 0 -#> 5 Male intermediate level of education 0.6590 0 -#> 3 Male high level of education 0.7872 0 -#> 4 Female low level of education 0.7101 0 -#> 2 Female intermediate level of education 0.5929 0 -#> 6 Female high level of education 0.6881 0
    -# same for "props()" -efc %>% - select(e42dep, c161sex, c172code, c12hour, n4pstu) %>% - group_by(c161sex, c172code) %>% - props( - e42dep > 2, - c12hour > 20 & c12hour < 40, - n4pstu == 'Care Level 1' | n4pstu == 'Care Level 3' - )
    #> carer's gender carer's level of education e42dep>2 c12hour>20&c12hour<40 -#> 1 Male low level of education 0.6829 0.2439 -#> 5 Male intermediate level of education 0.6590 0.1756 -#> 3 Male high level of education 0.7872 0.1489 -#> 4 Female low level of education 0.7101 0.1957 -#> 2 Female intermediate level of education 0.5929 0.1504 -#> 6 Female high level of education 0.6881 0.2018 -#> n4pstu==CareLevel1|n4pstu==CareLevel3 -#> 1 0.2250 -#> 5 0.3111 -#> 3 0.3191 -#> 4 0.3433 -#> 2 0.3540 -#> 6 0.2752
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/r2.html b/docs/reference/r2.html deleted file mode 100644 index 93598f8..0000000 --- a/docs/reference/r2.html +++ /dev/null @@ -1,189 +0,0 @@ - - - - - - - - -Deprecated functions — r2 • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    A list of deprecated functions.

    -
    - -
    r2(x)
    -
    -icc(x)
    -
    -p_value(x, ...)
    -
    -se(x, ...)
    -
    -cohens_f(x, ...)
    -
    -std_beta(x, ...)
    -
    -robust(x, ...)
    - -

    Arguments

    - - - - - - - - - - -
    x

    An object.

    ...

    Currently not used.

    - -

    Value

    - -

    Nothing.

    - -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/re_var.html b/docs/reference/re_var.html deleted file mode 100644 index c295983..0000000 --- a/docs/reference/re_var.html +++ /dev/null @@ -1,288 +0,0 @@ - - - - - - - - -Random effect variances — re_var • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    These functions extracts random effect variances as well as - random-intercept-slope-correlation of mixed effects models. - Currently, merMod, glmmTMB, - stanreg and brmsfit - objects are supported.

    - -
    - -
    re_var(x, adjusted = FALSE)
    -
    -get_re_var(x, comp = c("tau.00", "tau.01", "tau.11", "rho.01",
    -  "sigma_2"))
    - -

    Arguments

    - - - - - - - - - - - - - - -
    x

    Fitted mixed effects model (of class merMod, glmmTMB, -stanreg or brmsfit). get_re_var() also accepts -an object returned by the icc function.

    adjusted

    Logical, if TRUE, returns the variance of the fixed -and random effects as well as of the additive dispersion and -distribution-specific variance, which are used to calculate the -adjusted and conditional r2 and icc.

    comp

    Name of the variance component to be returned. See 'Details'.

    - -

    Value

    - -

    get_re_var() returns the value of the requested variance component, - re_var() returns all random effects variances.

    - -

    Details

    - -

    The random effect variances indicate the between- and within-group - variances as well as random-slope variance and random-slope-intercept - correlation. Use following values for comp to get the particular - variance component:

    -
    "sigma_2"

    Within-group (residual) variance

    -
    "tau.00"

    Between-group-variance (variation between individual intercepts and average intercept)

    -
    "tau.11"

    Random-slope-variance (variation between individual slopes and average slope)

    -
    "tau.01"

    Random-Intercept-Slope-covariance

    -
    "rho.01"

    Random-Intercept-Slope-correlation

    -

    The within-group-variance is affected by factors at level one, i.e. - by the lower-level direct effects. Level two factors (i.e. cross-level - direct effects) affect the between-group-variance. Cross-level - interaction effects are group-level factors that explain the - variance in random slopes (Aguinis et al. 2013). -

    - If adjusted = TRUE, the variance of the fixed and random - effects as well as of the additive dispersion and - distribution-specific variance are returned (see Johnson et al. 2014 - and Nakagawa et al. 2017):

    -
    "fixed"

    variance attributable to the fixed effects

    -
    "random"

    (mean) variance of random effects

    -
    "dispersion"

    variance due to additive dispersion

    -
    "distribution"

    distribution-specific variance

    -
    "residual"

    sum of dispersion and distribution

    -
    - -

    References

    - -
      -
    • Aguinis H, Gottfredson RK, Culpepper SA. 2013. Best-Practice Recommendations for Estimating Cross-Level Interaction Effects Using Multilevel Modeling. Journal of Management 39(6): 1490-1528 (doi: 10.1177/0149206313478188 -)

    • -
    • Johnson PC, O'Hara RB. 2014. Extension of Nakagawa & Schielzeth's R2GLMM to random slopes models. Methods Ecol Evol, 5: 944-946. (doi: 10.1111/2041-210X.12225 -)

    • -
    • Nakagawa S, Johnson P, Schielzeth H (2017) The coefficient of determination R2 and intra-class correlation coefficient from generalized linear mixed-effects models revisted and expanded. J. R. Soc. Interface 14. doi: 10.1098/rsif.2017.0213

    • -
    - -

    See also

    - - - - -

    Examples

    -
    library(lme4) -fit1 <- lmer(Reaction ~ Days + (Days | Subject), sleepstudy) - -# all random effect variance components -re_var(fit1)
    #> Within-group-variance: 654.941 -#> Between-group-variance: 611.898 (Subject) -#> Random-slope-variance: 35.081 (Subject.Days) -#> Slope-Intercept-covariance: 9.614 (Subject.(Intercept)) -#> Slope-Intercept-correlation: 0.066 (Subject)
    re_var(fit1, adjusted = TRUE)
    #> -#> Variance Components of Mixed Models -#> -#> Family : gaussian (identity) -#> Formula: list(conditional = Reaction ~ Days, random = ~Days | Subject) -#> -#> fixed: 908.953 -#> random: 1698.233 -#> residual: 654.941 -#> dispersion: 0.000 -#> distribution: 654.941 -#> -#>
    -# just the rand. slope-intercept covariance -get_re_var(fit1, "tau.01")
    #> Subject.(Intercept) -#> 9.613886
    -sleepstudy$mygrp <- sample(1:45, size = 180, replace = TRUE) -fit2 <- lmer(Reaction ~ Days + (1 | mygrp) + (Days | Subject), sleepstudy) -re_var(fit2)
    #> Within-group-variance: 605.912 -#> Between-group-variance: 44.917 (mygrp) -#> Between-group-variance: 615.511 (Subject) -#> Random-slope-variance: 38.301 (Subject.Days) -#> Slope-Intercept-covariance: 1.108 (Subject.(Intercept)) -#> Slope-Intercept-correlation: 0.007 (Subject)
    -
    -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/reexports.html b/docs/reference/reexports.html deleted file mode 100644 index 778e83c..0000000 --- a/docs/reference/reexports.html +++ /dev/null @@ -1,187 +0,0 @@ - - - - - - - - -Objects exported from other packages — reexports • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    These objects are imported from other packages. Follow the links -below to see their documentation.

    - -
    bayestestR

    ci, equivalence_test

    - -
    insight

    link_inverse

    - -
    magrittr

    %>%

    - -
    performance

    mse, rmse

    - -
    sjmisc

    typical_value

    - - -
    - - - - -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/reliab_test.html b/docs/reference/reliab_test.html deleted file mode 100644 index 6bf2ef4..0000000 --- a/docs/reference/reliab_test.html +++ /dev/null @@ -1,372 +0,0 @@ - - - - - - - - -Check internal consistency of a test or questionnaire — reliab_test • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    These function compute various measures of internal consistencies - for tests or item-scales of questionnaires.

    - -
    - -
    reliab_test(x, scale.items = FALSE, digits = 3, out = c("txt",
    -  "viewer", "browser"))
    -
    -split_half(x, digits = 3)
    -
    -cronb(x)
    -
    -difficulty(x)
    -
    -mic(x, cor.method = c("pearson", "spearman", "kendall"))
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - -
    x

    Depending on the function, x may be a matrix as -returned by the cor-function, or a data frame -with items (e.g. from a test or questionnaire).

    scale.items

    Logical, if TRUE, the data frame's vectors will be scaled. Recommended, -when the variables have different measures / scales.

    digits

    Amount of digits for returned values.

    out

    Character vector, indicating whether the results should be printed -to console (out = "txt") or as HTML-table in the viewer-pane -(out = "viewer") or browser (out = "browser"), of if the -results should be plotted (out = "plot", only applies to certain -functions). May be abbreviated.

    cor.method

    Correlation computation method. May be one of -"spearman" (default), "pearson" or "kendall". -You may use initial letter only.

    - -

    Value

    - - -
    -
    reliab_test()

    A data frame with the corrected item-total correlations (item - discrimination, column item.discr) and Cronbach's alpha - (if item deleted, column alpha.if.deleted) for each item - of the scale, or NULL if data frame had too less columns.

    -
    split_half()

    A list with two values: the split-half reliability splithalf and - the Spearman-Brown corrected split-half reliability spearmanbrown.

    -
    cronb()

    The Cronbach's Alpha value for x.

    -
    mic()

    The mean inter-item-correlation value for x.

    -
    difficulty()

    The item difficulty value for x.

    -
    - - -

    Details

    - -
    -
    reliab_test()

    This function calculates the item discriminations (corrected item-total - correlations for each item of x with the remaining items) and - the Cronbach's alpha for each item, if it was deleted from the scale. - The absolute value of the item discrimination indices should be - above 0.1. An index between 0.1 and 0.3 is considered as "fair", - while an index above 0.3 (or below -0.3) is "good". Items with - low discrimination indices are often ambiguously worded and - should be examined. Items with negative indices should be - examined to determine why a negative value was obtained (e.g. - reversed answer categories regarding positive and negative poles).

    -
    split_half()

    This function calculates the split-half reliability for items in - the data frame x, including the Spearman-Brown adjustment. - Splitting is done by selecting odd versus even columns in x. - A value closer to 1 indicates greater internal consistency.

    -
    cronb()

    The Cronbach's Alpha value for x. A value closer to 1 - indicates greater internal consistency, where usually following - rule of thumb is applied to interprete the results: - α < 0.5 is unacceptable, - 0.5 < α < 0.6 is poor, - 0.6 < α < 0.7 is questionable, - 0.7 < α < 0.8 is acceptable, - and everything > 0.8 is good or excellent.

    -
    mic()

    This function calculates a mean inter-item-correlation, i.e. - a correlation matrix of x will be computed (unless - x is already a matrix as returned by the - cor-function) and the mean - of the sum of all item's correlation values is returned. - Requires either a data frame or a computed cor-object. -

    - “Ideally, the average inter-item correlation for a set of - items should be between .20 and .40, suggesting that while the - items are reasonably homogenous, they do contain sufficiently - unique variance so as to not be isomorphic with each other. - When values are lower than .20, then the items may not be - representative of the same content domain. If values are higher than - .40, the items may be only capturing a small bandwidth of the construct.” - (Piedmont 2014)

    -
    difficulty()

    This function calculates the item difficutly, which should - range between 0.2 and 0.8. Lower values are a signal for - more difficult items, while higher values close to one - are a sign for easier items. The ideal value for item difficulty - is p + (1 - p) / 2, where p = 1 / max(x). In most - cases, the ideal item difficulty lies between 0.5 and 0.8.

    -
    - -

    References

    - -

    Spearman C. 1910. Correlation calculated from faulty data. British Journal of Psychology (3): 271-295. doi: 10.1111/j.2044-8295.1910.tb00206.x -

    - Brown W. 1910. Some experimental results in the correlation of mental abilities. British Journal of Psychology (3): 296-322. doi: 10.1111/j.2044-8295.1910.tb00207.x -

    - Piedmont RL. 2014. Inter-item Correlations. In: Michalos AC (eds) Encyclopedia of Quality of Life and Well-Being Research. Dordrecht: Springer, 3303-3304. doi: 10.1007/978-94-007-0753-5_1493

    - - -

    Examples

    -
    library(sjlabelled) -# Data from the EUROFAMCARE sample dataset -data(efc) - -# retrieve variable and value labels -varlabs <- get_label(efc) - -# recveive first item of COPE-index scale -start <- which(colnames(efc) == "c82cop1") -# recveive last item of COPE-index scale -end <- which(colnames(efc) == "c90cop9") - -# create data frame with COPE-index scale -x <- efc[, c(start:end)] -colnames(x) <- varlabs[c(start:end)] - -# reliability tests -reliab_test(x)
    #> term -#> 1 do you feel you cope well as caregiver? -#> 2 do you find caregiving too demanding? -#> 3 does caregiving cause difficulties in your relationship with your friends? -#> 4 does caregiving have negative effect on your physical health? -#> 5 does caregiving cause difficulties in your relationship with your family? -#> 6 does caregiving cause financial difficulties? -#> 7 do you feel trapped in your role as caregiver? -#> 8 do you feel supported by friends/neighbours? -#> 9 do you feel caregiving worthwhile? -#> alpha.if.deleted item.discr -#> 1 0.539 -0.241 -#> 2 0.384 0.329 -#> 3 0.339 0.408 -#> 4 0.324 0.441 -#> 5 0.380 0.357 -#> 6 0.366 0.416 -#> 7 0.353 0.368 -#> 8 0.534 -0.029 -#> 9 0.556 -0.112
    -# split-half-reliability -split_half(x)
    #> -#> # Internal Consistency -#> -#> Split-Half Reliability: 0.410 -#> Spearman-Brown Adjustment: 0.581
    -# cronbach's alpha -cronb(x)
    #> [1] 0.459369
    -# mean inter-item-correlation -mic(x)
    #> [1] 0.09176831
    -# item difficulty -difficulty(x)
    #> -#> # Item Difficulty -#> -#> difficulty ideal -#> do you feel you cope well as caregiver? 0.78 0.62 -#> do you find caregiving too demanding? 0.51 0.62 -#> does caregiving cause difficulties in your relationship with your friends? 0.41 0.62 -#> does caregiving have negative effect on your physical health? 0.44 0.62 -#> does caregiving cause difficulties in your relationship with your family? 0.35 0.62 -#> does caregiving cause financial difficulties? 0.32 0.62 -#> do you feel trapped in your role as caregiver? 0.48 0.62 -#> do you feel supported by friends/neighbours? 0.54 0.62 -#> do you feel caregiving worthwhile? 0.73 0.62
    -
    # NOT RUN { -library(sjPlot) -sjt.df(reliab_test(x), describe = FALSE, show.cmmn.row = TRUE, - string.cmmn = sprintf("Cronbach's &alpha;=%.2f", cronb(x))) - -# Compute PCA on Cope-Index, and perform a -# reliability check on each extracted factor. -factors <- sjt.pca(x)$factor.index -findex <- sort(unique(factors)) -library(sjPlot) -for (i in seq_len(length(findex))) { - rel.df <- subset(x, select = which(factors == findex[i])) - if (ncol(rel.df) >= 3) { - sjt.df(reliab_test(rel.df), describe = FALSE, show.cmmn.row = TRUE, - use.viewer = FALSE, title = "Item-Total-Statistic", - string.cmmn = sprintf("Scale's overall Cronbach's &alpha;=%.2f", - cronb(rel.df))) - } - } -# }
    -
    -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/rmse-1.png b/docs/reference/rmse-1.png deleted file mode 100644 index 9b9a407b49f47c76ef51d989813bc71d2b4603b1..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 27572 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 diff --git a/docs/reference/rmse-2.png b/docs/reference/rmse-2.png deleted file mode 100644 index f1f2bcf626e25a8356d75fb11bd5ad3638643dd0..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 25887 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 diff --git a/docs/reference/rmse.html b/docs/reference/rmse.html deleted file mode 100644 index ba06ba3..0000000 --- a/docs/reference/rmse.html +++ /dev/null @@ -1,416 +0,0 @@ - - - - - - - - -Compute model quality — cv • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    Compute various measures or tests to assess the model quality, - like root mean squared error, residual standard error or mean square error - of fitted linear (mixed effects) models. For logistic regression models, - or mixed models with binary outcome, the error rate, binned residuals, - Chi-square goodness-of-fit-test or the Hosmer-Lemeshow Goodness-of-fit-test - can be performed.

    - -
    - -
    cv(x, ...)
    -
    -chisq_gof(x, prob = NULL, weights = NULL)
    -
    -hoslem_gof(x, n.bins = 10)
    -
    -rmse(x, normalized = FALSE)
    -
    -rse(x)
    -
    -mse(x)
    -
    -error_rate(x)
    -
    -binned_resid(x, term = NULL, n.bins = NULL)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    x

    Fitted linear model of class lm, merMod (lme4) -or lme (nlme). For error_rate(), hoslem_gof() -and binned_resid(), a glm-object with binomial-family. For -chisq_gof(), a numeric vector or a glm-object.

    ...

    More fitted model objects, to compute multiple coefficients of -variation at once.

    prob

    Vector of probabilities (indicating the population probabilities) -of the same length as x's amount of categories / factor levels. -Use nrow(table(x)) to determine the amount of necessary values -for prob. Only used, when x is a vector, and not a -glm-object.

    weights

    Vector with weights, used to weight x.

    n.bins

    Numeric, the number of bins to divide the data. For -hoslem_gof(), the default is 10. For binned_resid(), if -n.bins = NULL, the square root of the number of observations is -taken.

    normalized

    Logical, use TRUE if normalized rmse should be returned.

    term

    Name of independent variable from x. If not NULL, -average residuals for the categories of term are plotted; else, -average residuals for the estimated probabilities of the response are -plotted.

    - -

    Value

    - - -
    -
    rmse(), rse(), mse(), cv()

    These functions return a number, the requested statistic.

    -
    error_rate()

    A list with four values: the error rate of the full and the null model, - as well as the chi-squared and p-value from the Likelihood-Ratio-Test - between the full and null model.

    -
    binned_resid()

    A data frame representing the data that is mapped to the plot, which is - automatically plotted. In case all residuals are inside the error bounds, - points are black. If some of the residuals are outside the error bounds - (indicates by the grey-shaded area), blue points indicate residuals that - are OK, while red points indicate model under- or overfitting for the - related range of estimated probabilities.

    -
    chisq_gof()

    For vectors, returns the object of the computed chisq.test. - For glm-objects, an object of class chisq_gof with - following values: p.value, the p-value for the goodness-of-fit test; - z.score, the standardized z-score for the goodness-of-fit test; - rss, the residual sums of squares term and chisq, the pearson - chi-squared statistic.

    -
    hoslem_gof()

    An object of class hoslem_test with following values: chisq, - the Hosmer-Lemeshow chi-squared statistic; df, degrees of freedom - and p.value the p-value for the goodness-of-fit test.

    -
    - - -

    Details

    - -
    -
    Root Mean Square Error

    The RMSE is the square root of the variance of the residuals and indicates - the absolute fit of the model to the data (difference between observed data - to model's predicted values). “RMSE can be interpreted as the standard - deviation of the unexplained variance, and has the useful property - of being in the same units as the response variable. Lower values - of RMSE indicate better fit. RMSE is a good measure of how accurately - the model predicts the response, and is the most important criterion - for fit if the main purpose of the model is prediction.” - (Grace-Martin K: Assessing the Fit of Regression Models) -

    - The normalized RMSE is the proportion of the RMSE related to the - range of the response variable. Hence, lower values indicate - less residual variance.

    -
    Residual Standard Error

    The residual standard error is the square root of the residual - sum of squares divided by the residual degrees of freedom.

    -
    Mean Square Error

    The mean square error is the mean of the sum of squared residuals, - i.e. it measures the average of the squares of the errors. Lower - values (closer to zero) indicate better fit.

    -
    Coefficient of Variation

    The advantage of the cv is that it is unitless. This allows - coefficient of variation to be compared to each other in ways - that other measures, like standard deviations or root mean - squared residuals, cannot be. -

    - “It is interesting to note the differences between a model's CV - and R-squared values. Both are unitless measures that are indicative - of model fit, but they define model fit in two different ways: CV - evaluates the relative closeness of the predictions to the actual - values while R-squared evaluates how much of the variability in the - actual values is explained by the model.” - (source: UCLA-FAQ)

    -
    Error Rate

    The error rate is a crude measure for model fit for logistic regression - models. It is defined as the proportion of cases for which the - deterministic prediction is wrong, i.e. the proportion where the the - predicted probability is above 0.5, although y = 0 (and vice versa). - In general, the error rate should be below 0.5 (i.e. 50%), the - closer to zero, the better. Furthermore, the error rate of the full - model should be considerably below the null model's error rate - (cf. Gelman and Hill 2007, pp. 99). The print()-method also - prints the results from the Likelihood-Ratio-Test, comparing the full - to the null model.

    -
    Binned Residuals

    Binned residual plots are achieved by “dividing the data into - categories (bins) based on their fitted values, and then plotting - the average residual versus the average fitted value for each bin.” - (Gelman, Hill 2007: 97). If the model were true, one would - expect about 95% of the residuals to fall inside the error bounds. -

    - If term is not NULL, one can compare the residuals in - relation to a specific model predictor. This may be helpful to check - if a term would fit better when transformed, e.g. a rising and falling - pattern of residuals along the x-axis (the pattern is indicated by - a green line) is a signal to consider taking the logarithm of the - predictor (cf. Gelman and Hill 2007, pp. 97ff).

    -
    Chi-squared Goodness-of-Fit Test

    For vectors, this function is a convenient function for the - chisq.test(), performing goodness-of-fit test. For - glm-objects, this function performs a goodness-of-fit test. - A well-fitting model shows no significant difference between the - model and the observed data, i.e. the reported p-values should be - greater than 0.05.

    -
    Hosmer-Lemeshow Goodness-of-Fit Test

    A well-fitting model shows no significant difference between - the model and the observed data, i.e. the reported p-value should be - greater than 0.05.

    -
    - -

    References

    - -

    Gelman A, Hill J (2007) Data Analysis Using Regression and Multilevel/Hierarchical Models. Cambridge, New York: Cambridge University Press -

    - Everitt, Brian (1998). The Cambridge Dictionary of Statistics. Cambridge, UK New York: Cambridge University Press -

    - Hosmer, D. W., & Lemeshow, S. (2000). Applied Logistic Regression. Hoboken, NJ, USA: John Wiley & Sons, Inc. doi: 10.1002/0471722146 -

    - Grace-Martin K: Assessing the Fit of Regression Models

    - -

    See also

    - -

    r2 for R-squared or pseudo-R-squared values.

    - - -

    Examples

    -
    data(efc) -fit <- lm(barthtot ~ c160age + c12hour, data = efc) -rmse(fit)
    #> [1] 25.46792
    rse(fit)
    #> [1] 25.51134
    cv(fit)
    #> [1] 0.3948098
    -library(lme4) -fit <- lmer(Reaction ~ Days + (Days | Subject), sleepstudy) -rmse(fit)
    #> [1] 23.43815
    mse(fit)
    #> [1] 549.3468
    cv(fit)
    #> [1] 0.07851768
    -# normalized RMSE -library(nlme) -fit <- lme(distance ~ age, data = Orthodont) -rmse(fit, normalized = TRUE)
    #> [1] 0.07242178
    -#coefficient of variation for variable -cv(efc$e17age)
    #> [1] 0.1023027
    -# Error Rate -efc$neg_c_7d <- ifelse(efc$neg_c_7 < median(efc$neg_c_7, na.rm = TRUE), 0, 1) -m <- glm( - neg_c_7d ~ c161sex + barthtot + c172code, - data = efc, - family = binomial(link = "logit") -) -error_rate(m)
    #> -#> # Error Rate of Logistic Regression Model -#> -#> Full model: 29.45% -#> Null model: 45.15% -#> -#> # Likelihood-Ratio-Test -#> -#> Chi-squared: 166.050 -#> p-value: 0.000 -#>
    -# Binned residuals -binned_resid(m)
    binned_resid(m, "barthtot")
    -# goodness-of-fit test for logistic regression -chisq_gof(m)
    #> -#> # Chi-squared Goodness-of-Fit Test -#> -#> Chi-squared: 852.765 -#> z-score: 1.025 -#> p-value: 0.305 -#>
    #> Summary: model seems to fit well.
    -# goodness-of-fit test for logistic regression -hoslem_gof(m)
    #> -#> # Hosmer-Lemeshow Goodness-of-Fit Test -#> -#> Chi-squared: 18.707 -#> df: 8 -#> p-value: 0.017 -#>
    #> Summary: model does not fit well.
    -# goodness-of-fit test for vectors against probabilities -# differing from population -chisq_gof(efc$e42dep, c(0.3,0.2,0.22,0.28))
    #> -#> Chi-squared test for given probabilities -#> -#> data: dummy -#> X-squared = 234.76, df = 3, p-value < 2.2e-16 -#>
    # equal to population -chisq_gof(efc$e42dep, prop.table(table(efc$e42dep)))
    #> -#> Chi-squared test for given probabilities -#> -#> data: dummy -#> X-squared = 0, df = 3, p-value = 1 -#>
    - -
    -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/robust.html b/docs/reference/robust.html deleted file mode 100644 index 251e3a3..0000000 --- a/docs/reference/robust.html +++ /dev/null @@ -1,351 +0,0 @@ - - - - - - - - -Robust standard errors for regression models — robust • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    robust() computes robust standard error for regression models. - This method calls one of the vcov*()-functions from the - sandwich-package for robust covariance matrix estimators. Results are - returned as tidy data frame. -

    - svy() is intended to compute standard errors for survey - designs (complex samples) fitted with regular lm or - glm functions, as alternative to the survey-package. - It simulates sampling weights by adjusting the residual degrees - of freedom based on the precision weights used to fit x, - and then calls robust() with the adjusted model.

    -
    - -
    robust(
    -  x,
    -  vcov.fun = "vcovHC",
    -  vcov.type = c("HC3", "const", "HC", "HC0", "HC1", "HC2", "HC4", "HC4m", "HC5"),
    -  vcov.args = NULL,
    -  conf.int = FALSE,
    -  exponentiate = FALSE
    -)
    -
    -svy(
    -  x,
    -  vcov.fun = "vcovHC",
    -  vcov.type = c("HC1", "const", "HC", "HC0", "HC3", "HC2", "HC4", "HC4m", "HC5"),
    -  vcov.args = NULL,
    -  conf.int = FALSE,
    -  exponentiate = FALSE
    -)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - -
    x

    A fitted model of any class that is supported by the vcov*()-functions -from the sandwich package. For svy(), x must be -lm object, fitted with weights.

    vcov.fun

    String, indicating the name of the vcov*()-function -from the sandwich-package, e.g. vcov.fun = "vcovCL".

    vcov.type

    Character vector, specifying the estimation type for the -robust covariance matrix estimation (see vcovHC -for details).

    vcov.args

    List of named vectors, used as additional arguments that -are passed down to vcov.fun.

    conf.int

    Logical, TRUE if confidence intervals based on robust -standard errors should be included.

    exponentiate

    Logical, whether to exponentiate the coefficient estimates -and confidence intervals (typical for logistic regression).

    - -

    Value

    - -

    A summary of the model, including estimates, robust standard error, - p-value and - optionally - the confidence intervals.

    -

    Note

    - -

    svy() simply calls robust(), but first adjusts the - residual degrees of freedom based on the model weights. - Hence, for svy(), x should be fitted with weights. - This simulates sampling weights like in survey designs, though - lm and glm implement precision weights. - The results from svy() are usually more accurate than simple - weighted standard errors for complex samples. However, results from - the survey package are still more exactly, especially - regarding the estimates. -

    - vcov.type for svy() defaults to "HC1", because - standard errors with this estimation type come closest to the standard - errors from the survey-package. -

    - Currently, svy() only works for objects of class lm.

    - -

    Examples

    -
    data(efc) -fit <- lm(barthtot ~ c160age + c12hour + c161sex + c172code, data = efc) -summary(fit)
    #> -#> Call: -#> lm(formula = barthtot ~ c160age + c12hour + c161sex + c172code, -#> data = efc) -#> -#> Residuals: -#> Min 1Q Median 3Q Max -#> -74.639 -15.246 4.251 19.009 73.327 -#> -#> Coefficients: -#> Estimate Std. Error t value Pr(>|t|) -#> (Intercept) 90.06448 6.17237 14.592 <2e-16 *** -#> c160age -0.22156 0.07111 -3.116 0.0019 ** -#> c12hour -0.27810 0.01865 -14.915 <2e-16 *** -#> c161sex -0.26178 2.08649 -0.125 0.9002 -#> c172code -0.76215 1.41971 -0.537 0.5915 -#> --- -#> Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1 -#> -#> Residual standard error: 25.35 on 816 degrees of freedom -#> (87 observations deleted due to missingness) -#> Multiple R-squared: 0.2696, Adjusted R-squared: 0.266 -#> F-statistic: 75.28 on 4 and 816 DF, p-value: < 2.2e-16 -#>
    robust(fit)
    #> term estimate std.error statistic p.value -#> 1 (Intercept) 90.0644792 5.58174936 16.1355291 4.832660e-51 -#> 2 c160age -0.2215581 0.07106417 -3.1177189 1.886499e-03 -#> 3 c12hour -0.2781004 0.02119537 -13.1208077 8.012747e-36 -#> 4 c161sex -0.2617783 1.91864853 -0.1364389 8.915080e-01 -#> 5 c172code -0.7621525 1.39456606 -0.5465159 5.848608e-01
    -confint(fit)
    #> 2.5 % 97.5 % -#> (Intercept) 77.9488902 102.18006829 -#> c160age -0.3611297 -0.08198647 -#> c12hour -0.3146997 -0.24150107 -#> c161sex -4.3573007 3.83374416 -#> c172code -3.5488594 2.02455439
    robust(fit, conf.int = TRUE)
    #> term estimate std.error conf.low conf.high statistic -#> 1 (Intercept) 90.0644792 5.58174936 79.1082006 101.02075786 16.1355291 -#> 2 c160age -0.2215581 0.07106417 -0.3610482 -0.08206799 -3.1177189 -#> 3 c12hour -0.2781004 0.02119537 -0.3197042 -0.23649650 -13.1208077 -#> 4 c161sex -0.2617783 1.91864853 -4.0278463 3.50428977 -0.1364389 -#> 5 c172code -0.7621525 1.39456606 -3.4995120 1.97520692 -0.5465159 -#> p.value -#> 1 4.832660e-51 -#> 2 1.886499e-03 -#> 3 8.012747e-36 -#> 4 8.915080e-01 -#> 5 5.848608e-01
    robust(fit, vcov.type = "HC1", conf.int = TRUE) # "HC1" should be Stata default
    #> term estimate std.error conf.low conf.high statistic -#> 1 (Intercept) 90.0644792 5.55391758 79.1628309 100.96612755 16.2163874 -#> 2 c160age -0.2215581 0.07066514 -0.3602650 -0.08285124 -3.1353241 -#> 3 c12hour -0.2781004 0.02103734 -0.3193941 -0.23680669 -13.2193675 -#> 4 c161sex -0.2617783 1.90939743 -4.0096876 3.48613101 -0.1370999 -#> 5 c172code -0.7621525 1.38747118 -3.4855856 1.96128056 -0.5493105 -#> p.value -#> 1 1.788307e-51 -#> 2 1.778136e-03 -#> 3 2.730414e-36 -#> 4 8.909856e-01 -#> 5 5.829427e-01
    -library(sjmisc) -# dichtomozize service usage by "service usage yes/no" -efc$services <- sjmisc::dicho(efc$tot_sc_e, dich.by = 0) -fit <- glm(services ~ neg_c_7 + c161sex + e42dep, - data = efc, family = binomial(link = "logit")) - -robust(fit)
    #> term estimate std.error statistic p.value -#> 1 (Intercept) -0.5198021 0.38048112 -1.366171 0.172232067 -#> 2 neg_c_7 0.0419026 0.02035790 2.058296 0.039853669 -#> 3 c161sex -0.2189336 0.16263979 -1.346126 0.178606278 -#> 4 e42dep 0.2134784 0.07970607 2.678320 0.007536269
    robust(fit, conf.int = TRUE, exponentiate = TRUE)
    #> term estimate std.error conf.low conf.high statistic p.value -#> 1 (Intercept) 0.5946382 0.38048112 0.2818017 1.254764 -1.366171 0.172232067 -#> 2 neg_c_7 1.0427929 0.02035790 1.0019492 1.085302 2.058296 0.039853669 -#> 3 c161sex 0.8033751 0.16263979 0.5838345 1.105470 -1.346126 0.178606278 -#> 4 e42dep 1.2379767 0.07970607 1.0587020 1.447609 2.678320 0.007536269
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/samplesize_mixed.html b/docs/reference/samplesize_mixed.html deleted file mode 100644 index 964bcab..0000000 --- a/docs/reference/samplesize_mixed.html +++ /dev/null @@ -1,262 +0,0 @@ - - - - - - - - -Sample size for linear mixed models — samplesize_mixed • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Compute an approximated sample size for linear mixed models - (two-level-designs), based on power-calculation for standard - design and adjusted for design effect for 2-level-designs.

    -
    - -
    samplesize_mixed(
    -  eff.size,
    -  df.n = NULL,
    -  power = 0.8,
    -  sig.level = 0.05,
    -  k,
    -  n,
    -  icc = 0.05
    -)
    -
    -smpsize_lmm(
    -  eff.size,
    -  df.n = NULL,
    -  power = 0.8,
    -  sig.level = 0.05,
    -  k,
    -  n,
    -  icc = 0.05
    -)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    eff.size

    Effect size.

    df.n

    Optional argument for the degrees of freedom for numerator. See 'Details'.

    power

    Power of test (1 minus Type II error probability).

    sig.level

    Significance level (Type I error probability).

    k

    Number of cluster groups (level-2-unit) in multilevel-design.

    n

    Optional, number of observations per cluster groups -(level-2-unit) in multilevel-design.

    icc

    Expected intraclass correlation coefficient for multilevel-model.

    - -

    Value

    - -

    A list with two values: The number of subjects per cluster, and the - total sample size for the linear mixed model.

    -

    Details

    - -

    The sample size calculation is based on a power-calculation for the - standard design. If df.n is not specified, a power-calculation - for an unpaired two-sample t-test will be computed (using - pwr.t.test of the pwr-package). - If df.n is given, a power-calculation for general linear models - will be computed (using pwr.f2.test of the - pwr-package). The sample size of the standard design - is then adjusted for the design effect of two-level-designs (see - design_effect). Thus, the sample size calculation is appropriate - in particular for two-level-designs (see Snijders 2005). Models that - additionally include repeated measures (three-level-designs) may work - as well, however, the computed sample size may be less accurate.

    -

    References

    - -

    Cohen J. 1988. Statistical power analysis for the behavioral sciences (2nd ed.). Hillsdale,NJ: Lawrence Erlbaum. -

    - Hsieh FY, Lavori PW, Cohen HJ, Feussner JR. 2003. An Overview of Variance Inflation Factors for Sample-Size Calculation. Evaluation and the Health Professions 26: 239-257. doi: 10.1177/0163278703255230 - -

    - Snijders TAB. 2005. Power and Sample Size in Multilevel Linear Models. In: Everitt BS, Howell DC (Hrsg.). Encyclopedia of Statistics in Behavioral Science. Chichester, UK: John Wiley and Sons, Ltd. doi: 10.1002/0470013192.bsa492

    - -

    Examples

    -
    # Sample size for multilevel model with 30 cluster groups and a small to -# medium effect size (Cohen's d) of 0.3. 27 subjects per cluster and -# hence a total sample size of about 802 observations is needed. -samplesize_mixed(eff.size = .3, k = 30)
    #> $`Subjects per Cluster` -#> [1] 27 -#> -#> $`Total Sample Size` -#> [1] 802 -#>
    -# Sample size for multilevel model with 20 cluster groups and a medium -# to large effect size for linear models of 0.2. Five subjects per cluster and -# hence a total sample size of about 107 observations is needed. -samplesize_mixed(eff.size = .2, df.n = 5, k = 20, power = .9)
    #> $`Subjects per Cluster` -#> [1] 5 -#> -#> $`Total Sample Size` -#> [1] 107 -#>
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/scale_weights.html b/docs/reference/scale_weights.html deleted file mode 100644 index bbb9f62..0000000 --- a/docs/reference/scale_weights.html +++ /dev/null @@ -1,276 +0,0 @@ - - - - - - - - -Rescale design weights for multilevel analysis — scale_weights • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Most functions to fit multilevel and mixed effects models only - allow to specify frequency weights, but not design (i.e. sampling or probability) - weights, which should be used when analyzing complex samples and survey data. - scale_weights() implements an algorithm proposed by Aaparouhov (2006) - and Carle (2009) to rescale design weights in survey data to account for - the grouping structure of multilevel models, which then can be used for - multilevel modelling.

    -
    - -
    scale_weights(x, cluster.id, pweight)
    - -

    Arguments

    - - - - - - - - - - - - - - -
    x

    A data frame.

    cluster.id

    Variable indicating the grouping structure (strata) of -the survey data (level-2-cluster variable).

    pweight

    Variable indicating the probability (design or sampling) -weights of the survey data (level-1-weight).

    - -

    Value

    - -

    x, with two new variables: svywght_a and svywght_b, - which represent the rescaled design weights to use in multilevel models - (use these variables for the weights argument).

    -

    Details

    - -

    Rescaling is based on two methods: For svywght_a, the sample - weights pweight are adjusted by a factor that represents the proportion - of cluster size divided by the sum of sampling weights within each cluster. - The adjustment factor for svywght_b is the sum of sample weights - within each cluster devided by the sum of squared sample weights within - each cluster (see Carle (2009), Appendix B). -

    - Regarding the choice between scaling methods A and B, Carle suggests - that "analysts who wish to discuss point estimates should report results - based on weighting method A. For analysts more interested in residual - between-cluster variance, method B may generally provide the least biased - estimates". In general, it is recommended to fit a non-weighted model - and weighted models with both scaling methods and when comparing the - models, see whether the "inferential decisions converge", to gain - confidence in the results. -

    - Though the bias of scaled weights decreases with increasing cluster size, - method A is preferred when insufficient or low cluster size is a concern. -

    - The cluster ID and probably PSU may be used as random effects (e.g. - nested design, or cluster and PSU as varying intercepts), depending - on the survey design that should be mimicked.

    -

    References

    - -

    Carle AC. Fitting multilevel models in complex survey data with design weights: Recommendations BMC Medical Research Methodology 2009, 9(49): 1-13 -

    - Asparouhov T. General Multi-Level Modeling with Sampling Weights Communications in Statistics - Theory and Methods 2006, 35: 439-460

    - -

    Examples

    -
    data(nhanes_sample) -scale_weights(nhanes_sample, SDMVSTRA, WTINT2YR)
    #> This function will be removed in a future update. There's an enhanced implementation in 'parameters::rescale_weights()' that should be used instead.
    #> # A tibble: 2,992 x 9 -#> total age RIAGENDR RIDRETH1 SDMVPSU SDMVSTRA WTINT2YR svywght_a svywght_b -#> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> <dbl> -#> 1 1 2.2 1 3 2 31 97594. 1.57 1.20 -#> 2 7 2.08 2 3 1 29 39599. 0.623 0.525 -#> 3 3 1.48 2 1 2 42 26620. 0.898 0.544 -#> 4 4 1.32 2 4 2 33 34999. 0.708 0.550 -#> 5 1 2 2 1 1 41 14746. 0.422 0.312 -#> 6 6 2.2 2 4 1 38 28232. 0.688 0.516 -#> 7 350 1.6 1 3 2 33 93162. 1.89 1.46 -#> 8 NA 1.48 2 3 1 29 82276. 1.29 1.09 -#> 9 3 2.28 2 4 1 41 24726. 0.707 0.523 -#> 10 30 0.84 1 3 2 35 39895. 0.760 0.594 -#> # ... with 2,982 more rows
    -if (require("lme4")) { - nhanes_sample <- scale_weights(nhanes_sample, SDMVSTRA, WTINT2YR) - glmer( - total ~ factor(RIAGENDR) * (log(age) + factor(RIDRETH1)) + (1 | SDMVPSU), - family = poisson(), - data = nhanes_sample, - weights = svywght_a - ) -}
    #> This function will be removed in a future update. There's an enhanced implementation in 'parameters::rescale_weights()' that should be used instead.
    #> Generalized linear mixed model fit by maximum likelihood (Laplace -#> Approximation) [glmerMod] -#> Family: poisson ( log ) -#> Formula: total ~ factor(RIAGENDR) * (log(age) + factor(RIDRETH1)) + (1 | -#> SDMVPSU) -#> Data: nhanes_sample -#> Weights: svywght_a -#> AIC BIC logLik deviance df.resid -#> 78844.27 78920.47 -39409.14 78818.27 2582 -#> Random effects: -#> Groups Name Std.Dev. -#> SDMVPSU (Intercept) 0.1018 -#> Number of obs: 2595, groups: SDMVPSU, 2 -#> Fixed Effects: -#> (Intercept) factor(RIAGENDR)2 -#> 2.491801 -1.021308 -#> log(age) factor(RIDRETH1)2 -#> 0.838726 -0.088627 -#> factor(RIDRETH1)3 factor(RIDRETH1)4 -#> -0.013333 0.722511 -#> factor(RIDRETH1)5 factor(RIAGENDR)2:log(age) -#> -0.106521 -1.012695 -#> factor(RIAGENDR)2:factor(RIDRETH1)2 factor(RIAGENDR)2:factor(RIDRETH1)3 -#> -0.009086 0.732985 -#> factor(RIAGENDR)2:factor(RIDRETH1)4 factor(RIAGENDR)2:factor(RIDRETH1)5 -#> 0.275967 0.542074
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/se.html b/docs/reference/se.html deleted file mode 100644 index 09d8418..0000000 --- a/docs/reference/se.html +++ /dev/null @@ -1,383 +0,0 @@ - - - - - - - - -Standard Error for variables or coefficients — se • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    Compute standard error for a variable, for all variables - of a data frame, for joint random and fixed effects - coefficients of (non-/linear) mixed models, the adjusted - standard errors for generalized linear (mixed) models, or - for intraclass correlation coefficients (ICC).

    - -
    - -
    se(x, ...)
    - -

    Arguments

    - - - - - - - - - - -
    x

    (Numeric) vector, a data frame, an lm, glm, -merMod (lme4), or stanreg model object, an ICC object -(as obtained by the icc-function), a table or -xtabs object, or a list with estimate and p-value. For the latter -case, the list must contain elements named estimate and -p.value (see 'Examples' and 'Details').

    ...

    Currently not used.

    - -

    Value

    - -

    The standard error of x.

    - -

    Details

    - -

    Standard error for variables -

    - For variables and data frames, the standard error is the square root of the - variance divided by the number of observations (length of vector). -

    - Standard error for mixed models -

    - For linear mixed models, and generalized linear mixed models, this - function computes the standard errors for joint (sums of) random and fixed - effects coefficients (unlike se.coef, which returns the - standard error for fixed and random effects separately). Hence, se() - returns the appropriate standard errors for coef.merMod. -

    - Standard error for generalized linear models -

    - For generalized linear models, approximated standard errors, using the delta - method for transformed regression parameters are returned (Oehlert 1992). -

    - Standard error for proportions and mean value -

    - To compute the standard error for relative frequencies (i.e. proportions, or - mean value if x has only two categories), this vector must be supplied - as table, e.g. se(table(iris$Species)). se() than computes the - relative frequencies (proportions) for each value and the related standard - error for each value. This might be useful to add standard errors or confidence - intervals to descriptive statistics. If standard errors for weighted variables - are required, use xtabs(), e.g. se(xtabs(weights ~ variable)). -

    - Standard error for regression coefficient and p-value -

    - se() also returns the standard error of an estimate (regression - coefficient) and p-value, assuming a normal distribution to compute - the z-score from the p-value (formula in short: b / qnorm(p / 2)). - See 'Examples'.

    - -

    Note

    - -

    Computation of standard errors for coefficients of mixed models - is based on this code. - Standard errors for generalized linear (mixed) models, if - type = "re", are approximations based on the delta - method (Oehlert 1992). -

    - A remark on standard errors: - “Standard error represents variation in the point estimate, but - confidence interval has usual Bayesian interpretation only with flat prior.” - (Gelman 2017)

    - -

    References

    - -

    Oehlert GW. 1992. A note on the delta method. American Statistician 46(1). -

    - Gelman A 2017. How to interpret confidence intervals? http://andrewgelman.com/2017/03/04/interpret-confidence-intervals/

    - - -

    Examples

    -
    library(lme4) -library(sjmisc) - -# compute standard error for vector -se(rnorm(n = 100, mean = 3))
    #> [1] 0.09589874
    -# compute standard error for each variable in a data frame -data(efc) -se(efc[, 1:3])
    #> c12hour e15relat e16sex -#> 1.69162290 0.06942207 0.01565588
    -# compute standard error for merMod-coefficients -library(lme4) -fit <- lmer(Reaction ~ Days + (Days | Subject), sleepstudy) -se(fit)
    #> $Subject -#> (Intercept) Days -#> 1 13.86572 2.775269 -#> 2 13.86572 2.775269 -#> 3 13.86572 2.775269 -#> 4 13.86572 2.775269 -#> 5 13.86572 2.775269 -#> 6 13.86572 2.775269 -#> 7 13.86572 2.775269 -#> 8 13.86572 2.775269 -#> 9 13.86572 2.775269 -#> 10 13.86572 2.775269 -#> 11 13.86572 2.775269 -#> 12 13.86572 2.775269 -#> 13 13.86572 2.775269 -#> 14 13.86572 2.775269 -#> 15 13.86572 2.775269 -#> 16 13.86572 2.775269 -#> 17 13.86572 2.775269 -#> 18 13.86572 2.775269 -#>
    -# compute odds-ratio adjusted standard errors, based on delta method -# with first-order Taylor approximation. -data(efc) -efc$services <- sjmisc::dicho(efc$tot_sc_e, dich.by = 0) -fit <- glm( - services ~ neg_c_7 + c161sex + e42dep, - data = efc, - family = binomial(link = "logit") -) -se(fit)
    #> # A tibble: 4 x 3 -#> term estimate std.error -#> <chr> <dbl> <dbl> -#> 1 (Intercept) 0.595 0.224 -#> 2 neg_c_7 1.04 0.0204 -#> 3 c161sex 0.803 0.130 -#> 4 e42dep 1.24 0.0972
    -# compute odds-ratio adjusted standard errors for generalized -# linear mixed model, also based on delta method - -# create binary response -sleepstudy$Reaction.dicho <- dicho(sleepstudy$Reaction, dich.by = "median") -fit <- glmer( - Reaction.dicho ~ Days + (Days | Subject), - data = sleepstudy, - family = binomial("logit") -) -se(fit)
    #> $Subject -#> (Intercept) Days -#> 1 1.859251 0.4700877 -#> 2 2.622321 0.4115004 -#> 3 2.622321 0.4115004 -#> 4 1.633616 0.3289753 -#> 5 1.745522 0.4826391 -#> 6 1.714059 0.3560052 -#> 7 1.715195 0.4646524 -#> 8 2.219966 0.4133501 -#> 9 2.622321 0.4115004 -#> 10 1.806673 0.5077360 -#> 11 2.314690 0.3959941 -#> 12 2.111495 0.4335945 -#> 13 1.830037 0.3388438 -#> 14 1.686571 0.4933028 -#> 15 1.988285 0.4534415 -#> 16 2.167228 0.4013762 -#> 17 2.314690 0.3959941 -#> 18 1.778644 0.4321839 -#>
    -# compute standard error for proportions -efc$e42dep <- to_label(efc$e42dep) -se(table(efc$e42dep))
    #> value proportion std.error -#> 1 independent 0.07325194 0.008680166 -#> 2 slightly dependent 0.24972253 0.014420404 -#> 3 moderately dependent 0.33962264 0.015777276 -#> 4 severely dependent 0.33740289 0.015752039
    -# including weights -efc$weights <- rnorm(nrow(efc), 1, .25) -se(xtabs(efc$weights ~ efc$e42dep))
    #> value proportion std.error -#> 1 independent 0.07061821 0.008552969 -#> 2 slightly dependent 0.25359865 0.014525167 -#> 3 moderately dependent 0.33761265 0.015787989 -#> 4 severely dependent 0.33817049 0.015794372
    -# compute standard error from regression coefficient and p-value -se(list(estimate = .3, p.value = .002))
    #> [1] 0.09708008
    -
    # NOT RUN { -# compute standard error of ICC for the linear mixed model -icc(fit) -se(icc(fit)) - -# the standard error for the ICC can be computed manually in this way, -# taking the fitted model example from above -library(dplyr) -library(purrr) -dummy <- sleepstudy %>% - # generate 100 bootstrap replicates of dataset - bootstrap(100) %>% - # run mixed effects regression on each bootstrap replicate - # and compute ICC for each "bootstrapped" regression - mutate( - models = map(strap, ~lmer(Reaction ~ Days + (Days | Subject), data = .x)), - icc = map_dbl(models, ~icc(.x)) - ) - -# now compute SE and p-values for the bootstrapped ICC, values -# may differ from above example due to random seed -boot_se(dummy, icc) -boot_p(dummy, icc) -# }
    - -
    -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/se_ybar.html b/docs/reference/se_ybar.html deleted file mode 100644 index 969a20d..0000000 --- a/docs/reference/se_ybar.html +++ /dev/null @@ -1,188 +0,0 @@ - - - - - - - - -Standard error of sample mean for mixed models — se_ybar • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Compute the standard error for the sample mean for mixed models, - regarding the extent to which clustering affects the standard errors. - May be used as part of the multilevel power calculation for cluster sampling - (see Gelman and Hill 2007, 447ff).

    -
    - -
    se_ybar(fit)
    - -

    Arguments

    - - - - - - -
    fit

    Fitted mixed effects model (merMod-class).

    - -

    Value

    - -

    The standard error of the sample mean of fit.

    -

    References

    - -

    Gelman A, Hill J. 2007. Data analysis using regression and multilevel/hierarchical models. Cambridge, New York: Cambridge University Press

    - -

    Examples

    -
    if (require("lme4")) { - fit <- lmer(Reaction ~ 1 + (1 | Subject), sleepstudy) - se_ybar(fit) -}
    #> Subject -#> 9.049936
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/sjstats-package.html b/docs/reference/sjstats-package.html deleted file mode 100644 index 925573e..0000000 --- a/docs/reference/sjstats-package.html +++ /dev/null @@ -1,209 +0,0 @@ - - - - - - - - -Collection of Convenient Functions for Common Statistical Computations — sjstats-package • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Collection of convenient functions for common statistical computations, which are not directly provided by R's base or stats packages.

    -

    This package aims at providing, first, shortcuts for statistical measures, which otherwise could only be calculated with additional effort (like standard errors or root mean squared errors).

    -

    Second, these shortcut functions are generic (if appropriate), and can be applied not only to vectors, but also to other objects as well (e.g., the Coefficient of Variation can be computed for vectors, linear models, or linear mixed models; the r2()-function returns the r-squared value for lm, glm, merMod, glmmTMB, or lme and other objects).

    -

    Most functions of this package are designed as summary functions, i.e. they do not transform the input vector; rather, they return a summary, which is sometimes a vector and sometimes a tidy data frame. he focus of most functions lies on summary statistics or fit measures for regression models, including generalized linear models, mixed effects models or Bayesian models. However, some of the functions deal with other statistical measures, like Cronbach's Alpha, Cramer's V, Phi etc.

    -

    The comprised tools include:

      -
    • For regression and mixed models: Coefficient of Variation, Root Mean Squared Error, Residual Standard Error, Coefficient of Discrimination, R-squared and pseudo-R-squared values, standardized beta values

    • -
    • Especially for mixed models: Design effect, ICC, sample size calculation and convergence tests

    • -
    • Especially for Bayesian models: Highest Density Interval, region of practical equivalence (rope), Monte Carlo Standard Errors, ratio of number of effective samples, mediation analysis, Test for Practical Equivalence

    • -
    • Fit and accuracy measures for regression models: Overdispersion tests, accuracy of predictions, test/training-error comparisons, error rate and binned residual plots for logistic regression models

    • -
    • For anova-tables: Eta-squared, Partial Eta-squared, Omega-squared and Partial Omega-squared statistics

    • -
    - -

    Furthermore, sjstats has functions to access information from model objects, which either support more model objects than their stats counterparts, or provide easy access to model attributes, like:

    -
      -
    • model_frame() to get the model frame

    • -
    • model_family() to get information about the model family, link functions etc.

    • -
    • link_inverse() to get the link-inverse function

    • -
    • pred_vars() and resp_var() to get the names of either the dependent or independent variables, or

    • -
    • var_names() to get the "cleaned" variables names from a model object (cleaned means, things like s() or log() are removed from the returned character vector with variable names.)

    • -
    - -

    Other statistics:

      -
    • Cramer's V, Cronbach's Alpha, Mean Inter-Item-Correlation, Mann-Whitney-U-Test, Item-scale reliability tests

    • -
    - -
    - - - - -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/smpsize_lmm.html b/docs/reference/smpsize_lmm.html deleted file mode 100644 index 2da7400..0000000 --- a/docs/reference/smpsize_lmm.html +++ /dev/null @@ -1,260 +0,0 @@ - - - - - - - - -Sample size for linear mixed models — smpsize_lmm • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    Compute an approximated sample size for linear mixed models - (two-level-designs), based on power-calculation for standard - design and adjusted for design effect for 2-level-designs.

    - -
    - -
    smpsize_lmm(eff.size, df.n = NULL, power = 0.8, sig.level = 0.05, k,
    -  n, icc = 0.05)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    eff.size

    Effect size.

    df.n

    Optional argument for the degrees of freedom for numerator. See 'Details'.

    power

    Power of test (1 minus Type II error probability).

    sig.level

    Significance level (Type I error probability).

    k

    Number of cluster groups (level-2-unit) in multilevel-design.

    n

    Optional, number of observations per cluster groups -(level-2-unit) in multilevel-design.

    icc

    Expected intraclass correlation coefficient for multilevel-model.

    - -

    Value

    - -

    A list with two values: The number of subjects per cluster, and the - total sample size for the linear mixed model.

    - -

    Details

    - -

    The sample size calculation is based on a power-calculation for the - standard design. If df.n is not specified, a power-calculation - for an unpaired two-sample t-test will be computed (using - pwr.t.test of the pwr-package). - If df.n is given, a power-calculation for general linear models - will be computed (using pwr.f2.test of the - pwr-package). The sample size of the standard design - is then adjusted for the design effect of two-level-designs (see - deff). Thus, the sample size calculation is appropriate - in particular for two-level-designs (see Snijders 2005). Models that - additionally include repeated measures (three-level-designs) may work - as well, however, the computed sample size may be less accurate.

    - -

    References

    - -

    Cohen J. 1988. Statistical power analysis for the behavioral sciences (2nd ed.). Hillsdale,NJ: Lawrence Erlbaum. -

    - Hsieh FY, Lavori PW, Cohen HJ, Feussner JR. 2003. An Overview of Variance Inflation Factors for Sample-Size Calculation. Evaluation and the Health Professions 26: 239-257. doi: 10.1177/0163278703255230 -

    - Snijders TAB. 2005. Power and Sample Size in Multilevel Linear Models. In: Everitt BS, Howell DC (Hrsg.). Encyclopedia of Statistics in Behavioral Science. Chichester, UK: John Wiley and Sons, Ltd. doi: 10.1002/0470013192.bsa492

    - - -

    Examples

    -
    # Sample size for multilevel model with 30 cluster groups and a small to -# medium effect size (Cohen's d) of 0.3. 27 subjects per cluster and -# hence a total sample size of about 802 observations is needed. -smpsize_lmm(eff.size = .3, k = 30)
    #> $`Subjects per Cluster` -#> [1] 27 -#> -#> $`Total Sample Size` -#> [1] 802 -#>
    -# Sample size for multilevel model with 20 cluster groups and a medium -# to large effect size for linear models of 0.2. Five subjects per cluster and -# hence a total sample size of about 107 observations is needed. -smpsize_lmm(eff.size = .2, df.n = 5, k = 20, power = .9)
    #> $`Subjects per Cluster` -#> [1] 5 -#> -#> $`Total Sample Size` -#> [1] 107 -#>
    - -
    -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/std_beta.html b/docs/reference/std_beta.html deleted file mode 100644 index 5702cf9..0000000 --- a/docs/reference/std_beta.html +++ /dev/null @@ -1,272 +0,0 @@ - - - - - - - - -Standardized beta coefficients and CI of linear and mixed models — std_beta • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Returns the standardized beta coefficients, std. error and confidence intervals - of a fitted linear (mixed) models.

    -
    - -
    std_beta(fit, ...)
    -
    -# S3 method for merMod
    -std_beta(fit, ci.lvl = 0.95, ...)
    -
    -# S3 method for lm
    -std_beta(fit, type = "std", ci.lvl = 0.95, ...)
    -
    -# S3 method for gls
    -std_beta(fit, type = "std", ci.lvl = 0.95, ...)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - -
    fit

    Fitted linear (mixed) model of class lm, merMod -(lme4 package), gls or stanreg.

    ...

    Currently not used.

    ci.lvl

    Numeric, the level of the confidence intervals.

    type

    If fit is of class lm, normal standardized coefficients -are computed by default. Use type = "std2" to follow -Gelman's (2008) -suggestion, rescaling the estimates by deviding them by two standard -deviations, so resulting coefficients are directly comparable for -untransformed binary predictors.

    - -

    Value

    - -

    A tibble with term names, standardized beta coefficients, - standard error and confidence intervals of fit.

    -

    Details

    - -

    “Standardized coefficients refer to how many standard deviations a dependent variable will change, - per standard deviation increase in the predictor variable. Standardization of the coefficient is - usually done to answer the question of which of the independent variables have a greater effect - on the dependent variable in a multiple regression analysis, when the variables are measured - in different units of measurement (for example, income measured in dollars and family size - measured in number of individuals)” (Source: Wikipedia)

    -

    Note

    - -

    For gls-objects, standardized beta coefficients may be wrong - for categorical variables (factors), because the model.matrix for - gls objects returns the original data of the categorical vector, - and not the 'dummy' coded vectors as for other classes. See, as example:

    - head(model.matrix(lm(neg_c_7 ~ as.factor(e42dep), data = efc, na.action = na.omit))) -

    and

    - head(model.matrix(nlme::gls(neg_c_7 ~ as.factor(e42dep), data = efc, na.action = na.omit))). -

    - In such cases, use to_dummy to create dummies from - factors.

    -

    References

    - -

    Wikipedia: Standardized coefficient -

    - Gelman A. 2008. Scaling regression inputs by dividing by two standard deviations. Statistics in Medicine 27: 2865-2873 http://www.stat.columbia.edu/~gelman/research/published/standardizing7.pdf

    - -

    Examples

    -
    # fit linear model -fit <- lm(Ozone ~ Wind + Temp + Solar.R, data = airquality) -# print std. beta coefficients -std_beta(fit)
    #> term std.estimate std.error conf.low conf.high -#> 1 Wind -0.3564122 0.06996619 -0.4935434 -0.2192810 -#> 2 Temp 0.4731461 0.07260889 0.3308353 0.6154569 -#> 3 Solar.R 0.1638655 0.06351430 0.0393798 0.2883513
    -# print std. beta coefficients and ci, using -# 2 sd and center binary predictors -std_beta(fit, type = "std2")
    #> term std.estimate std.error conf.low conf.high -#> 1 Wind -23.71992 4.656386 -32.846272 -14.59358 -#> 2 Temp 31.48879 4.832262 22.017729 40.95985 -#> 3 Solar.R 10.90557 4.226999 2.620802 19.19034
    -# std. beta for mixed models -library(lme4) -fit1 <- lmer(Reaction ~ Days + (Days | Subject), sleepstudy) -std_beta(fit)
    #> term std.estimate std.error conf.low conf.high -#> 1 Wind -0.3564122 0.06996619 -0.4935434 -0.2192810 -#> 2 Temp 0.4731461 0.07260889 0.3308353 0.6154569 -#> 3 Solar.R 0.1638655 0.06351430 0.0393798 0.2883513
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/svyglm.nb.html b/docs/reference/svyglm.nb.html deleted file mode 100644 index 190254d..0000000 --- a/docs/reference/svyglm.nb.html +++ /dev/null @@ -1,275 +0,0 @@ - - - - - - - - -Survey-weighted negative binomial generalised linear model — svyglm.nb • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    svyglm.nb() is an extension to the survey-package - to fit survey-weighted negative binomial models. It uses - svymle to fit sampling-weighted - maximum likelihood estimates, based on starting values provided - by glm.nb, as proposed by Lumley - (2010, pp249).

    -
    - -
    svyglm.nb(formula, design, ...)
    - -

    Arguments

    - - - - - - - - - - - - - - -
    formula

    An object of class formula, i.e. a symbolic description -of the model to be fitted. See 'Details' in glm.

    design

    An object of class svydesign, providing -a specification of the survey design.

    ...

    Other arguments passed down to glm.nb.

    - -

    Value

    - -

    An object of class svymle and svyglm.nb, - with some additional information about the model.

    -

    Details

    - -

    For details on the computation method, see Lumley (2010), Appendix E - (especially 254ff.) -

    - sjstats implements following S3-methods for svyglm.nb-objects: - family(), model.frame(), formula(), print(), - predict() and residuals(). However, these functions have some - limitations:

      -
    • family() simply returns the family-object from the - underlying glm.nb-model.

    • -
    • The predict()-method just re-fits the svyglm.nb-model - with glm.nb, overwrites the $coefficients - from this model-object with the coefficients from the returned - svymle-object and finally calls - predict.glm to compute the predicted values.

    • -
    • residuals() re-fits the svyglm.nb-model with - glm.nb and then computes the Pearson-residuals - from the glm.nb-object.

    • -
    - -

    References

    - -

    Lumley T (2010). Complex Surveys: a guide to analysis using R. Wiley

    - -

    Examples

    -
    # ------------------------------------------ -# This example reproduces the results from -# Lumley 2010, figure E.7 (Appendix E, p256) -# ------------------------------------------ -if (require("survey")) { - data(nhanes_sample) - - # create survey design - des <- svydesign( - id = ~SDMVPSU, - strat = ~SDMVSTRA, - weights = ~WTINT2YR, - nest = TRUE, - data = nhanes_sample - ) - - # fit negative binomial regression - fit <- svyglm.nb(total ~ factor(RIAGENDR) * (log(age) + factor(RIDRETH1)), des) - - # print coefficients and standard errors - fit -}
    #> Loading required package: survey
    #> Loading required package: grid
    #> Loading required package: survival
    #> -#> Attaching package: 'survey'
    #> The following object is masked from 'package:sjstats': -#> -#> cv
    #> The following object is masked from 'package:graphics': -#> -#> dotchart
    #> term irr std.error conf.low conf.high -#> 2 (Intercept) 9.8463 0.1556 7.2578 13.3580 -#> 3 factor(RIAGENDR)2 0.4511 0.1805 0.3167 0.6426 -#> 4 log(age) 2.9163 0.2331 1.8467 4.6056 -#> 5 factor(RIDRETH1)2 1.0859 0.1477 0.8130 1.4504 -#> 6 factor(RIDRETH1)3 1.0977 0.1779 0.7746 1.5556 -#> 7 factor(RIDRETH1)4 2.2686 0.2974 1.2665 4.0634 -#> 8 factor(RIDRETH1)5 1.0589 0.3789 0.5039 2.2250 -#> 9 factor(RIAGENDR)2:log(age) 0.2947 0.2651 0.1753 0.4955 -#> 10 factor(RIAGENDR)2:factor(RIDRETH1)2 0.8314 0.2611 0.4984 1.3870 -#> 11 factor(RIAGENDR)2:factor(RIDRETH1)3 1.8285 0.1931 1.2523 2.6698 -#> 12 factor(RIAGENDR)2:factor(RIDRETH1)4 1.0668 0.3747 0.5119 2.2232 -#> 13 factor(RIAGENDR)2:factor(RIDRETH1)5 1.4564 0.4427 0.6116 3.4680 -#> p.value -#> 2 <0.001 *** -#> 3 <0.001 *** -#> 4 <0.001 *** -#> 5 0.5769 -#> 6 0.6003 -#> 7 0.0059 ** -#> 8 0.8800 -#> 9 <0.001 *** -#> 10 0.4795 -#> 11 0.0018 ** -#> 12 0.8630 -#> 13 0.3957 -#> -#> Dispersion parameter Theta: 0.8062 -#> Standard Error of Theta: 0.0216
    #> -#> Showing robust standard errors on link-scale (untransformed).
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/svyglm.zip.html b/docs/reference/svyglm.zip.html deleted file mode 100644 index bb9fd14..0000000 --- a/docs/reference/svyglm.zip.html +++ /dev/null @@ -1,232 +0,0 @@ - - - - - - - - -Survey-weighted zero-inflated Poisson model — svyglm.zip • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    svyglm.zip() is an extension to the survey-package - to fit survey-weighted zero-inflated Poisson models. It uses - svymle to fit sampling-weighted - maximum likelihood estimates, based on starting values provided - by zeroinfl.

    -
    - -
    svyglm.zip(formula, design, ...)
    - -

    Arguments

    - - - - - - - - - - - - - - -
    formula

    An object of class formula, i.e. a symbolic description -of the model to be fitted. See 'Details' in zeroinfl.

    design

    An object of class svydesign, providing -a specification of the survey design.

    ...

    Other arguments passed down to zeroinfl.

    - -

    Value

    - -

    An object of class svymle and svyglm.zip, - with some additional information about the model.

    -

    Details

    - -

    Code modified from https://notstatschat.rbind.io/2015/05/26/zero-inflated-poisson-from-complex-samples/.

    - -

    Examples

    -
    if (require("survey")) { - data(nhanes_sample) - set.seed(123) - nhanes_sample$malepartners <- rpois(nrow(nhanes_sample), 2) - nhanes_sample$malepartners[sample(1:2992, 400)] <- 0 - - # create survey design - des <- svydesign( - id = ~SDMVPSU, - strat = ~SDMVSTRA, - weights = ~WTINT2YR, - nest = TRUE, - data = nhanes_sample - ) - - # fit negative binomial regression - fit <- svyglm.zip( - malepartners ~ age + factor(RIDRETH1) | age + factor(RIDRETH1), - des - ) - - # print coefficients and standard errors - fit -}
    #> Warning: non-integer #successes in a binomial glm!
    #> term estimate std.error conf.low conf.high p.value -#> 2 age 0.0149 0.0354 0.9469 1.0879 0.6745 -#> 3 factor(RIDRETH1)2 0.0185 0.0754 0.8787 1.1810 0.8062 -#> 4 factor(RIDRETH1)3 -0.0449 0.0284 0.9043 1.0107 0.1133 -#> 5 factor(RIDRETH1)4 -0.0240 0.0276 0.9250 1.0305 0.3843 -#> 6 factor(RIDRETH1)5 0.0371 0.0617 0.9197 1.1712 0.5470 -#> 7 tp.(Intercept) -1.6694 0.4717 0.0747 0.4748 <0.001 *** -#> 8 tp.age -0.0333 0.2831 0.5553 1.6848 0.9064 -#> 9 tp.factor(RIDRETH1)2 0.1548 0.2571 0.7053 1.9323 0.5472 -#> 10 tp.factor(RIDRETH1)3 -0.3969 0.2111 0.4446 1.0169 0.0601 . -#> 11 tp.factor(RIDRETH1)4 -0.2330 0.3050 0.4357 1.4402 0.4450 -#> 12 tp.factor(RIDRETH1)5 -0.3303 0.4744 0.2836 1.8214 0.4863
    #> -#> Showing robust standard errors on link-scale (untransformed).
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/table_values.html b/docs/reference/table_values.html deleted file mode 100644 index f040d6a..0000000 --- a/docs/reference/table_values.html +++ /dev/null @@ -1,199 +0,0 @@ - - - - - - - - -Expected and relative table values — table_values • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    This function calculates a table's cell, row and column percentages as - well as expected values and returns all results as lists of tables.

    -
    - -
    table_values(tab, digits = 2)
    - -

    Arguments

    - - - - - - - - - - -
    tab

    Simple table or ftable of which -cell, row and column percentages as well as expected values are calculated. -Tables of class xtabs and other will be coerced to -ftable objects.

    digits

    Amount of digits for the table percentage values.

    - -

    Value

    - -

    (Invisibly) returns a list with four tables:

      -
    1. cell a table with cell percentages of tab

    2. -
    3. row a table with row percentages of tab

    4. -
    5. col a table with column percentages of tab

    6. -
    7. expected a table with expected values of tab

    8. -
    - - -

    Examples

    -
    tab <- table(sample(1:2, 30, TRUE), sample(1:3, 30, TRUE)) -# show expected values -table_values(tab)$expected
    #> A B C -#> A 4 5 4 -#> B 5 7 5
    # show cell percentages -table_values(tab)$cell
    #> 1 2 3 -#> -#> 1 16.67 10.00 13.33 -#> 2 13.33 30.00 16.67
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/tidy_stan.html b/docs/reference/tidy_stan.html deleted file mode 100644 index bbceb10..0000000 --- a/docs/reference/tidy_stan.html +++ /dev/null @@ -1,265 +0,0 @@ - - - - - - - - -Tidy summary output for stan models — tidy_stan • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Returns a tidy summary output for stan models.

    -
    - -
    tidy_stan(
    -  x,
    -  prob = 0.89,
    -  typical = "median",
    -  trans = NULL,
    -  effects = c("all", "fixed", "random"),
    -  component = c("all", "conditional", "zero_inflated", "zi"),
    -  digits = 2
    -)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    x

    A stanreg, stanfit or brmsfit object.

    prob

    Vector of scalars between 0 and 1, indicating the mass within -the credible interval that is to be estimated.

    typical

    The typical value that will represent the Bayesian point estimate. -By default, the posterior median is returned. See typical_value -for possible values for this argument.

    trans

    Name of a function or character vector naming a function, used -to apply transformations on the estimates and uncertainty intervals. The -values for standard errors are not transformed! If trans -is not NULL, credible intervals instead of HDI -are computed, due to the possible asymmetry of the HDI.

    effects

    Should results for fixed effects, random effects or both be returned? -Only applies to mixed models. May be abbreviated.

    component

    Should results for all parameters, parameters for the conditional model -or the zero-inflated part of the model be returned? May be abbreviated. Only -applies to brms-models.

    digits

    Amount of digits to round numerical values in the output.

    - -

    Value

    - -

    A data frame, summarizing x, with consistent column names. - To distinguish multiple HDI values, column names for the HDI get a suffix - when prob has more than one element.

    -

    Details

    - -

    The returned data frame has an additonal class-attribute, - tidy_stan, to pass the result to its own print()-method. - The print()-method creates a cleaner output, especially for multilevel, - zero-inflated or multivariate response models, where - for instance - - the conditional part of a model is printed separately from the zero-inflated - part, or random and fixed effects are printed separately. -

    - The returned data frame gives information on:

      -
    • The Bayesian point estimate (column estimate, which is by - default the posterior median; other statistics are also possible, - see argument typical).

    • -
    • The standard error (which is actually the median absolute deviation).

    • -
    • The HDI. Computation for HDI is based on the - code from Kruschke 2015, pp. 727f.

    • -
    • The Probability of Direction (pd), which is an index for "effect significance" - (see Makowski et al. 2019). A value of 95% or higher indicates a - "significant" (i.e. statistically clear) effect.

    • -
    • The effective numbers of samples, ESS.

    • -
    • The Rhat statistics. When Rhat is above 1, it usually indicates that - the chain has not yet converged, indicating that the drawn samples - might not be trustworthy. Drawing more iteration may solve this issue.

    • -
    • The Monte Carlo standard error (see mcse). It is defined - as standard deviation of the chains divided by their effective sample - size and “provides a quantitative suggestion of how big the - estimation noise is” (Kruschke 2015, p.187).

    • -
    - -

    References

    - -

    Kruschke JK. Doing Bayesian Data Analysis: A Tutorial with R, JAGS, and Stan 2nd edition. Academic Press, 2015 -

    -Gelman A, Carlin JB, Stern HS, Dunson DB, Vehtari A, Rubin DB. Bayesian data analysis 3rd ed. Boca Raton: Chapman and Hall/CRC, 2013 -

    -Gelman A, Rubin DB. Inference from iterative simulation using multiple sequences Statistical Science 1992;7: 457-511 -

    -Makowski D, Ben-Shachar MS, Lüdecke D. bayestestR: Describing Effects and their Uncertainty, Existence and Significance within the Bayesian Framework. Journal of Open Source Software 2019;4:1541. doi: 10.21105/joss.01541 - -

    -McElreath R. Statistical Rethinking. A Bayesian Course with Examples in R and Stan Chapman and Hall, 2015

    - -

    Examples

    -
    if (FALSE) { -if (require("rstanarm")) { - fit <- stan_glm(mpg ~ wt + am, data = mtcars, chains = 1) - tidy_stan(fit) - tidy_stan(fit, prob = c(.89, .5)) -}}
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/typical_value.html b/docs/reference/typical_value.html deleted file mode 100644 index ab60fec..0000000 --- a/docs/reference/typical_value.html +++ /dev/null @@ -1,270 +0,0 @@ - - - - - - - - -Return the typical value of a vector — typical_value • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - -
    - -
    -
    - - -
    - -

    This function returns the "typical" value of a variable.

    - -
    - -
    typical_value(x, fun = "mean", weights = NULL, ...)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - -
    x

    A variable.

    fun

    Character vector, naming the function to be applied to -x. Currently, "mean", "weighted.mean", -"median" and "mode" are supported, which call the -corresponding R functions (except "mode", which calls an -internal function to compute the most common value). "zero" -simply returns 0. Note: By default, if x is a factor, -only fun = "mode" is applicable; for all other functions (including -the default, "mean") the reference level of x is returned. -For character vectors, only the mode is returned. You can use a named -vector to apply other different functions to numeric and categorical -x, where factors are first converted to numeric vectors, e.g. -fun = c(numeric = "median", factor = "mean"). See 'Examples'.

    weights

    Name of variable in x that indicated the vector of -weights that will be applied to weight all observations. Default is -NULL, so no weights are used.

    ...

    Further arguments, passed down to fun.

    - -

    Value

    - -

    The "typical" value of x.

    - -

    Details

    - -

    By default, for numeric variables, typical_value() returns the - mean value of x (unless changed with the fun-argument). -

    - For factors, the reference level is returned or the most common value - (if fun = "mode"), unless fun is a named vector. If - fun is a named vector, specify the function for numeric - and categorical variables as element names, e.g. - fun = c(numeric = "median", factor = "mean"). In this case, - factors are converted to numeric values (using to_value) - and the related function is applied. You may abbreviate the names - fun = c(n = "median", f = "mean"). See also 'Examples'. -

    - For character vectors the most common value (mode) is returned.

    - - -

    Examples

    -
    data(iris) -typical_value(iris$Sepal.Length)
    #> [1] 5.843333
    -library(purrr) -map(iris, ~ typical_value(.x))
    #> $Sepal.Length -#> [1] 5.843333 -#> -#> $Sepal.Width -#> [1] 3.057333 -#> -#> $Petal.Length -#> [1] 3.758 -#> -#> $Petal.Width -#> [1] 1.199333 -#> -#> $Species -#> [1] "setosa" -#>
    -# example from ?stats::weighted.mean -wt <- c(5, 5, 4, 1) / 15 -x <- c(3.7, 3.3, 3.5, 2.8) - -typical_value(x, fun = "weighted.mean")
    #> [1] 3.325
    typical_value(x, fun = "weighted.mean", weights = wt)
    #> [1] 3.453333
    -# for factors, return either reference level or mode value -set.seed(123) -x <- sample(iris$Species, size = 30, replace = TRUE) -typical_value(x)
    #> [1] "setosa"
    typical_value(x, fun = "mode")
    #> [1] "virginica"
    -# for factors, use a named vector to apply other functions than "mode" -map(iris, ~ typical_value(.x, fun = c(n = "median", f = "mean")))
    #> $Sepal.Length -#> [1] 5.8 -#> -#> $Sepal.Width -#> [1] 3 -#> -#> $Petal.Length -#> [1] 4.35 -#> -#> $Petal.Width -#> [1] 1.3 -#> -#> $Species -#> [1] 2 -#>
    - -
    -
    - -
    - -
    - - -
    -

    Site built with pkgdown 1.3.0.

    -
    -
    -
    - - - - - - diff --git a/docs/reference/var_pop.html b/docs/reference/var_pop.html deleted file mode 100644 index 3e4770e..0000000 --- a/docs/reference/var_pop.html +++ /dev/null @@ -1,190 +0,0 @@ - - - - - - - - -Calculate population variance and standard deviation — var_pop • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Calculate the population variance or standard deviation of a vector.

    -
    - -
    var_pop(x)
    -
    -sd_pop(x)
    - -

    Arguments

    - - - - - - -
    x

    (Numeric) vector.

    - -

    Value

    - -

    The population variance or standard deviation of x.

    -

    Details

    - -

    Unlike var, which returns the sample variance, - var_pop() returns the population variance. sd_pop() - returns the standard deviation based on the population variance.

    - -

    Examples

    -
    data(efc) - -# sampling variance -var(efc$c12hour, na.rm = TRUE)
    #> [1] 2581.152
    # population variance -var_pop(efc$c12hour)
    #> [1] 2578.291
    -# sampling sd -sd(efc$c12hour, na.rm = TRUE)
    #> [1] 50.80504
    # population sd -sd_pop(efc$c12hour)
    #> [1] 50.77687
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/weight.html b/docs/reference/weight.html deleted file mode 100644 index 3558dfb..0000000 --- a/docs/reference/weight.html +++ /dev/null @@ -1,222 +0,0 @@ - - - - - - - - -Weight a variable — weight • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    These functions weight the variable x by - a specific vector of weights.

    -
    - -
    weight(x, weights, digits = 0)
    -
    -weight2(x, weights)
    - -

    Arguments

    - - - - - - - - - - - - - - -
    x

    (Unweighted) variable.

    weights

    Vector with same length as x, which -contains weight factors. Each value of x has a -specific assigned weight in weights.

    digits

    Numeric value indicating the number of decimal places to be -used for rounding the weighted values. By default, this value is -0, i.e. the returned values are integer values.

    - -

    Value

    - -

    The weighted x.

    -

    Details

    - -

    weight2() sums up all weights values of the associated - categories of x, whereas weight() uses a - xtabs formula to weight cases. Thus, weight() - may return a vector of different length than x.

    -

    Note

    - -

    The values of the returned vector are in sorted order, whereas the values' - order of the original x may be spread randomly. Hence, x can't be - used, for instance, for further cross tabulation. In case you want to have - weighted contingency tables or (grouped) box plots etc., use the weightBy - argument of most functions.

    - -

    Examples

    -
    v <- sample(1:4, 20, TRUE) -table(v)
    #> v -#> 1 2 3 4 -#> 3 6 5 6
    w <- abs(rnorm(20)) -table(weight(v, w))
    #> -#> 1 2 3 4 -#> 2 5 5 3
    table(weight2(v, w))
    #> -#> 1 2 3 4 -#> 2 5 5 3
    -set.seed(1) -x <- sample(letters[1:5], size = 20, replace = TRUE) -w <- runif(n = 20) - -table(x)
    #> x -#> a b c d e -#> 6 4 3 1 6
    table(weight(x, w))
    #> -#> a b c e -#> 3 3 2 3
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/weighted_sd.html b/docs/reference/weighted_sd.html deleted file mode 100644 index 196b69f..0000000 --- a/docs/reference/weighted_sd.html +++ /dev/null @@ -1,402 +0,0 @@ - - - - - - - - -Weighted statistics for tests and variables — survey_median • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Weighted statistics for variables -

    - weighted_sd(), weighted_se(), weighted_mean() and weighted_median() - compute weighted standard deviation, standard error, mean or median for a - variable or for all variables of a data frame. survey_median() computes the - median for a variable in a survey-design (see svydesign). - weighted_correlation() computes a weighted correlation for a two-sided alternative - hypothesis. -

    - Weighted tests -

    - weighted_ttest() computes a weighted t-test, while weighted_mannwhitney() - computes a weighted Mann-Whitney-U test or a Kruskal-Wallis test - (for more than two groups). weighted_chisqtest() computes a weighted - Chi-squared test for contigency tables.

    -
    - -
    survey_median(x, design)
    -
    -weighted_chisqtest(data, ...)
    -
    -# S3 method for default
    -weighted_chisqtest(data, x, y, weights, ...)
    -
    -# S3 method for formula
    -weighted_chisqtest(formula, data, ...)
    -
    -weighted_correlation(data, ...)
    -
    -# S3 method for default
    -weighted_correlation(data, x, y, weights, ci.lvl = 0.95, ...)
    -
    -# S3 method for formula
    -weighted_correlation(formula, data, ci.lvl = 0.95, ...)
    -
    -weighted_mean(x, weights = NULL)
    -
    -weighted_median(x, weights = NULL)
    -
    -weighted_mannwhitney(data, ...)
    -
    -# S3 method for default
    -weighted_mannwhitney(data, x, grp, weights, ...)
    -
    -# S3 method for formula
    -weighted_mannwhitney(formula, data, ...)
    -
    -weighted_sd(x, weights = NULL)
    -
    -wtd_sd(x, weights = NULL)
    -
    -weighted_se(x, weights = NULL)
    -
    -weighted_ttest(data, ...)
    -
    -# S3 method for default
    -weighted_ttest(
    -  data,
    -  x,
    -  y = NULL,
    -  weights,
    -  mu = 0,
    -  paired = FALSE,
    -  ci.lvl = 0.95,
    -  alternative = c("two.sided", "less", "greater"),
    -  ...
    -)
    -
    -# S3 method for formula
    -weighted_ttest(
    -  formula,
    -  data,
    -  mu = 0,
    -  paired = FALSE,
    -  ci.lvl = 0.95,
    -  alternative = c("two.sided", "less", "greater"),
    -  ...
    -)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    x

    (Numeric) vector or a data frame. For survey_median(), weighted_ttest(), -weighted_mannwhitney() and weighted_chisqtest() the bare (unquoted) variable -name, or a character vector with the variable name.

    design

    An object of class svydesign, providing -a specification of the survey design.

    data

    A data frame.

    ...

    For weighted_ttest() and weighted_mannwhitney(), currently not used. -For weighted_chisqtest(), further arguments passed down to -chisq.test.

    y

    Optional, bare (unquoted) variable name, or a character vector with -the variable name.

    weights

    Bare (unquoted) variable name, or a character vector with -the variable name of the numeric vector of weights. If weights = NULL, -unweighted statistic is reported.

    formula

    A formula of the form lhs ~ rhs1 + rhs2 where lhs is a -numeric variable giving the data values and rhs1 a factor with two -levels giving the corresponding groups and rhs2 a variable with weights.

    ci.lvl

    Confidence level of the interval.

    grp

    Bare (unquoted) name of the cross-classifying variable, where -x is grouped into the categories represented by grp, -or a character vector with the variable name.

    mu

    A number indicating the true value of the mean (or difference in -means if you are performing a two sample test).

    paired

    Logical, whether to compute a paired t-test.

    alternative

    A character string specifying the alternative hypothesis, -must be one of "two.sided" (default), "greater" or -"less". You can specify just the initial letter.

    - -

    Value

    - -

    The weighted (test) statistic.

    -

    Note

    - -

    weighted_chisq() is a convenient wrapper for crosstable_statistics. - For a weighted one-way Anova, use means_by_group() with - weights-argument. -

    - weighted_ttest() assumes unequal variance between the two groups.

    - -

    Examples

    -
    # weighted sd and se ---- - -weighted_sd(rnorm(n = 100, mean = 3), runif(n = 100))
    #> [1] 0.8498705
    -data(efc) -weighted_sd(efc[, 1:3], runif(n = nrow(efc)))
    #> c12hour e15relat e16sex -#> 51.7876181 2.0540843 0.4699551
    weighted_se(efc[, 1:3], runif(n = nrow(efc)))
    #> c12hour e15relat e16sex -#> 1.66065784 0.06942749 0.01562877
    -# survey_median ---- - -# median for variables from weighted survey designs -if (require("survey")) { - data(nhanes_sample) - - des <- svydesign( - id = ~SDMVPSU, - strat = ~SDMVSTRA, - weights = ~WTINT2YR, - nest = TRUE, - data = nhanes_sample - ) - - survey_median(total, des) - survey_median("total", des) -}
    #> [1] 6
    -# weighted t-test ---- - -efc$weight <- abs(rnorm(nrow(efc), 1, .3)) -weighted_ttest(efc, e17age, weights = weight)
    #> -#> One Sample t-test (two.sided) -#> # t=292.68 df=890 p-value=0.000 -#> -#> mean of e17age: 79.189 [78.658, 79.720] -#>
    weighted_ttest(efc, e17age, c160age, weights = weight)
    #> -#> Two-Sample t-test (two.sided) -#> -#> # comparison between e17age and c160age -#> # t=49.92 df=1469 p-value=0.000 -#> -#> mean of e17age : 79.187 -#> mean of c160age : 53.208 -#> difference of mean: 25.980 [24.959 27.001] -#>
    weighted_ttest(e17age ~ e16sex + weight, efc)
    #> -#> Two-Sample t-test (two.sided) -#> -#> # comparison of e17age by e16sex -#> # t=-7.46 df=604 p-value=0.000 -#> -#> mean in group [1] male : 76.401 -#> mean in group [2] female: 80.518 -#> difference of mean : -4.117 [-5.201 -3.034] -#>
    -# weighted Mann-Whitney-U-test ---- - -weighted_mannwhitney(c12hour ~ c161sex + weight, efc)
    #> -#> Weighted Mann-Whitney-U test (two.sided) -#> -#> # comparison of c12hour by c161sex -#> # Chisq=3.26 df=899 p-value=0.001 -#> -#> difference in mean rank score: 0.075 -#>
    -# weighted Chi-squared-test ---- - -weighted_chisqtest(efc, c161sex, e16sex, weights = weight, correct = FALSE)
    #> -#> # Measure of Association for Contingency Tables -#> -#> Chi-squared: 2.0566 -#> Phi: 0.0479 -#> p-value: 0.1515
    weighted_chisqtest(c172code ~ c161sex + weight, efc)
    #> -#> # Measure of Association for Contingency Tables -#> -#> Chi-squared: 4.8005 -#> Cramer's V: 0.0758 -#> p-value: 0.0907
    -
    - -
    - - - -
    - - - - - - - - diff --git a/docs/reference/wtd_sd.html b/docs/reference/wtd_sd.html deleted file mode 100644 index 1c55bf4..0000000 --- a/docs/reference/wtd_sd.html +++ /dev/null @@ -1,421 +0,0 @@ - - - - - - - - -Weighted statistics for tests and variables — svy_md • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    Weighted statistics for variables -

    - wtd_sd(), wtd_se(), wtd_mean() and wtd_median() - compute weighted standard deviation, standard error, mean or median for a - variable or for all variables of a data frame. svy_md() computes the - median for a variable in a survey-design (see svydesign). - wtd_cor() computes a weighted correlation for a two-sided alternative - hypothesis. -

    - Weighted tests -

    - wtd_ttest() computes a weighted t-test, while wtd_mwu() - computes a weighted Mann-Whitney-U test or a Kruskal-Wallis test - (for more than two groups). wtd_chisqtest() computes a weighted - Chi-squared test for contigency tables.

    -
    - -
    svy_md(x, design)
    -
    -survey_median(x, design)
    -
    -wtd_chisqtest(data, ...)
    -
    -# S3 method for default
    -wtd_chisqtest(data, x, y, weights, ...)
    -
    -# S3 method for formula
    -wtd_chisqtest(formula, data, ...)
    -
    -wtd_cor(data, ...)
    -
    -# S3 method for default
    -wtd_cor(data, x, y, weights, ci.lvl = 0.95, ...)
    -
    -# S3 method for formula
    -wtd_cor(formula, data, ci.lvl = 0.95, ...)
    -
    -wtd_mean(x, weights = NULL)
    -
    -wtd_median(x, weights = NULL)
    -
    -wtd_mwu(data, ...)
    -
    -# S3 method for default
    -wtd_mwu(data, x, grp, weights, ...)
    -
    -# S3 method for formula
    -wtd_mwu(formula, data, ...)
    -
    -wtd_sd(x, weights = NULL)
    -
    -wtd_se(x, weights = NULL)
    -
    -wtd_ttest(data, ...)
    -
    -# S3 method for default
    -wtd_ttest(
    -  data,
    -  x,
    -  y = NULL,
    -  weights,
    -  mu = 0,
    -  paired = FALSE,
    -  ci.lvl = 0.95,
    -  alternative = c("two.sided", "less", "greater"),
    -  ...
    -)
    -
    -# S3 method for formula
    -wtd_ttest(
    -  formula,
    -  data,
    -  mu = 0,
    -  paired = FALSE,
    -  ci.lvl = 0.95,
    -  alternative = c("two.sided", "less", "greater"),
    -  ...
    -)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    x

    (Numeric) vector or a data frame. For svy_md(), wtd_ttest(), -wtd_mwu() and wtd_chisqtest() the bare (unquoted) variable -name, or a character vector with the variable name.

    design

    An object of class svydesign, providing -a specification of the survey design.

    data

    A data frame.

    ...

    For wtd_ttest() and wtd_mwu(), currently not used. -For wtd_chisqtest(), further arguments passed down to -chisq.test.

    y

    Optional, bare (unquoted) variable name, or a character vector with -the variable name.

    weights

    Bare (unquoted) variable name, or a character vector with -the variable name of the numeric vector of weights. If weights = NULL, -unweighted statistic is reported.

    formula

    A formula of the form lhs ~ rhs1 + rhs2 where lhs is a -numeric variable giving the data values and rhs1 a factor with two -levels giving the corresponding groups and rhs2 a variable with weights.

    ci.lvl

    Confidence level of the interval.

    grp

    Bare (unquoted) name of the cross-classifying variable, where -x is grouped into the categories represented by grp, -or a character vector with the variable name.

    mu

    A number indicating the true value of the mean (or difference in -means if you are performing a two sample test).

    paired

    Logical, whether to compute a paired t-test.

    alternative

    A character string specifying the alternative hypothesis, -must be one of "two.sided" (default), "greater" or -"less". You can specify just the initial letter.

    - -

    Value

    - -

    The weighted (test) statistic.

    -

    Note

    - -

    wtd_chisq() is a convenient wrapper for xtab_statistics. - For a weighted one-way Anova, use grpmean() with - weights-argument. -

    - wtd_ttest() assumes unequal variance between the two groups.

    - -

    Examples

    -
    # weighted sd and se ---- - -wtd_sd(rnorm(n = 100, mean = 3), runif(n = 100))
    #> [1] 0.8498705
    -data(efc) -wtd_sd(efc[, 1:3], runif(n = nrow(efc)))
    #> c12hour e15relat e16sex -#> 51.7876181 2.0540843 0.4699551
    wtd_se(efc[, 1:3], runif(n = nrow(efc)))
    #> c12hour e15relat e16sex -#> 1.66065784 0.06942749 0.01562877
    -# svy_md ---- - -# median for variables from weighted survey designs -library(survey) -data(nhanes_sample) - -des <- svydesign( - id = ~SDMVPSU, - strat = ~SDMVSTRA, - weights = ~WTINT2YR, - nest = TRUE, - data = nhanes_sample -) - -svy_md(total, des)
    #> [1] 6
    svy_md("total", des)
    #> [1] 6
    -# weighted t-test ---- - -efc$weight <- abs(rnorm(nrow(efc), 1, .3)) -wtd_ttest(efc, e17age, weights = weight)
    #> -#> One Sample t-test (two.sided) -#> # t=292.68 df=890 p-value=0.000 -#> -#> mean of e17age: 79.189 [78.658, 79.720] -#>
    wtd_ttest(efc, e17age, c160age, weights = weight)
    #> -#> Two-Sample t-test (two.sided) -#> -#> # comparison between e17age and c160age -#> # t=49.92 df=1469 p-value=0.000 -#> -#> mean of e17age : 79.187 -#> mean of c160age : 53.208 -#> difference of mean: 25.980 [24.959 27.001] -#>
    wtd_ttest(e17age ~ e16sex + weight, efc)
    #> -#> Two-Sample t-test (two.sided) -#> -#> # comparison of e17age by e16sex -#> # t=-7.46 df=604 p-value=0.000 -#> -#> mean in group [1] male : 76.401 -#> mean in group [2] female: 80.518 -#> difference of mean : -4.117 [-5.201 -3.034] -#>
    -# weighted Mann-Whitney-U-test ---- - -wtd_mwu(c12hour ~ c161sex + weight, efc)
    #> -#> Weighted Mann-Whitney-U test (two.sided) -#> -#> # comparison of c12hour by c161sex -#> # Chisq=3.26 df=899 p-value=0.001 -#> -#> difference in mean rank score: 0.075 -#>
    -# weighted Chi-squared-test ---- - -wtd_chisqtest(efc, c161sex, e16sex, weights = weight, correct = FALSE)
    #> -#> # Measure of Association for Contingency Tables -#> -#> Chi-squared: 2.0566 -#> Phi: 0.0479 -#> p-value: 0.1515
    wtd_chisqtest(c172code ~ c161sex + weight, efc)
    #> -#> # Measure of Association for Contingency Tables -#> -#> Chi-squared: 4.8005 -#> Cramer's V: 0.0758 -#> p-value: 0.0907
    -
    -
    - -
    - - -
    - - -
    -

    Site built with pkgdown 1.4.1.

    -
    - -
    -
    - - - - - - - - diff --git a/docs/reference/xtab_statistics.html b/docs/reference/xtab_statistics.html deleted file mode 100644 index b86a8ce..0000000 --- a/docs/reference/xtab_statistics.html +++ /dev/null @@ -1,373 +0,0 @@ - - - - - - - - -Measures of association for contingency tables — cramer • sjstats - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    - - - - -
    - -
    -
    - - -
    -

    This function calculates various measure of association for - contingency tables and returns the statistic and p-value. - Supported measures are Cramer's V, Phi, Spearman's rho, - Kendall's tau and Pearson's r.

    -
    - -
    cramer(tab, ...)
    -
    -# S3 method for formula
    -cramer(
    -  formula,
    -  data,
    -  ci.lvl = NULL,
    -  n = 1000,
    -  method = c("dist", "quantile"),
    -  ...
    -)
    -
    -phi(tab, ...)
    -
    -xtab_statistics(
    -  data,
    -  x1 = NULL,
    -  x2 = NULL,
    -  statistics = c("auto", "cramer", "phi", "spearman", "kendall", "pearson", "fisher"),
    -  weights = NULL,
    -  ...
    -)
    -
    -crosstable_statistics(
    -  data,
    -  x1 = NULL,
    -  x2 = NULL,
    -  statistics = c("auto", "cramer", "phi", "spearman", "kendall", "pearson", "fisher"),
    -  weights = NULL,
    -  ...
    -)
    - -

    Arguments

    - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    tab

    A table or ftable. Tables of class -xtabs and other will be coerced to ftable -objects.

    ...

    Other arguments, passed down to the statistic functions -chisq.test, fisher.test or -cor.test.

    formula

    A formula of the form lhs ~ rhs where lhs is a -numeric variable giving the data values and rhs a factor giving the -corresponding groups.

    data

    A data frame or a table object. If a table object, x1 and -x2 will be ignored. For Kendall's tau, Spearman's rho -or Pearson's product moment correlation coefficient, data needs -to be a data frame. If x1 and x2 are not specified, -the first two columns of the data frames are used as variables -to compute the crosstab.

    ci.lvl

    Scalar between 0 and 1. If not NULL, returns a data -frame including lower and upper confidence intervals.

    n

    Number of bootstraps to be generated.

    method

    Character vector, indicating if confidence intervals should be -based on bootstrap standard error, multiplied by the value of the -quantile function of the t-distribution (default), or on sample -quantiles of the bootstrapped values. See 'Details' in boot_ci(). -May be abbreviated.

    x1

    Name of first variable that should be used to compute the -contingency table. If data is a table object, this argument -will be irgnored.

    x2

    Name of second variable that should be used to compute the -contingency table. If data is a table object, this argument -will be irgnored.

    statistics

    Name of measure of association that should be computed. May -be one of "auto", "cramer", "phi", "spearman", -"kendall", "pearson" or "fisher". See 'Details'.

    weights

    Name of variable in x that indicated the vector of -weights that will be applied to weight all observations. Default is -NULL, so no weights are used.

    - -

    Value

    - -

    For phi(), the table's Phi value. For cramer(), the - table's Cramer's V. -

    - For crosstable_statistics(), a list with following components:

    -
    estimate

    the value of the estimated measure of association.

    -
    p.value

    the p-value for the test.

    -
    statistic

    the value of the test statistic.

    -
    stat.name

    the name of the test statistic.

    -
    stat.html

    if applicable, the name of the test statistic, in HTML-format.

    -
    df

    the degrees of freedom for the contingency table.

    -
    method

    character string indicating the name of the measure of association.

    -
    method.html

    if applicable, the name of the measure of association, in HTML-format.

    -
    method.short

    the short form of association measure, equals the statistics-argument.

    -
    fisher

    logical, if Fisher's exact test was used to calculate the p-value.

    - -
    - -

    Details

    - -

    The p-value for Cramer's V and the Phi coefficient are based - on chisq.test(). If any expected value of a table cell is - smaller than 5, or smaller than 10 and the df is 1, then fisher.test() - is used to compute the p-value, unless statistics = "fisher"; in - this case, the use of fisher.test() is forced to compute the - p-value. The test statistic is calculated with cramer() resp. - phi(). -

    - Both test statistic and p-value for Spearman's rho, Kendall's tau - and Pearson's r are calculated with cor.test(). -

    - When statistics = "auto", only Cramer's V or Phi are calculated, - based on the dimension of the table (i.e. if the table has more than - two rows or columns, Cramer's V is calculated, else Phi).

    - -

    Examples

    -
    # Phi coefficient for 2x2 tables -tab <- table(sample(1:2, 30, TRUE), sample(1:2, 30, TRUE)) -phi(tab)
    #> [1] 0.1443376
    -# Cramer's V for nominal variables with more than 2 categories -tab <- table(sample(1:2, 30, TRUE), sample(1:3, 30, TRUE)) -cramer(tab)
    #> [1] 0.3795188
    -# formula notation -data(efc) -cramer(e16sex ~ c161sex, data = efc)
    #> [1] 0.05258249
    -# bootstrapped confidence intervals -cramer(e16sex ~ c161sex, data = efc, ci.lvl = .95, n = 100)
    #> cramer conf.low conf.high -#> 1 0.05258249 -0.005303236 0.1088931
    -# 2x2 table, compute Phi automatically -crosstable_statistics(efc, e16sex, c161sex)
    #> -#> # Measure of Association for Contingency Tables -#> -#> Chi-squared: 2.2327 -#> Phi: 0.0526 -#> p-value: 0.1351
    -# more dimensions than 2x2, compute Cramer's V automatically -crosstable_statistics(efc, c172code, c161sex)
    #> -#> # Measure of Association for Contingency Tables -#> -#> Chi-squared: 4.1085 -#> Cramer's V: 0.0699 -#> p-value: 0.1282
    -# ordinal data, use Kendall's tau -crosstable_statistics(efc, e42dep, quol_5, statistics = "kendall")
    #> -#> # Measure of Association for Contingency Tables -#> -#> z: -9.5951 -#> Kendall's tau: -0.2496 -#> p-value: <0.001
    -# calcilate Spearman's rho, with continuity correction -crosstable_statistics(efc, - e42dep, - quol_5, - statistics = "spearman", - exact = FALSE, - continuity = TRUE -)
    #> -#> # Measure of Association for Contingency Tables -#> -#> S: 157974157.4198 -#> Spearman's rho: -0.3177 -#> p-value: <0.001
    -
    -
    - -
    - - - -
    - - - - - - - - diff --git a/man/crosstable_statistics.Rd b/man/crosstable_statistics.Rd index 8dd677d..45afaff 100644 --- a/man/crosstable_statistics.Rd +++ b/man/crosstable_statistics.Rd @@ -1,16 +1,19 @@ % Generated by roxygen2: do not edit by hand % Please edit documentation in R/cramer.R, R/phi.R, R/xtab_statistics.R -\name{cramer} +\name{cramers_v} +\alias{cramers_v} \alias{cramer} -\alias{cramer.formula} +\alias{cramers_v.formula} \alias{phi} \alias{crosstable_statistics} \alias{xtab_statistics} \title{Measures of association for contingency tables} \usage{ +cramers_v(tab, ...) + cramer(tab, ...) -\method{cramer}{formula}( +\method{cramers_v}{formula}( formula, data, ci.lvl = NULL, @@ -40,24 +43,21 @@ xtab_statistics( ) } \arguments{ -\item{tab}{A \code{\link{table}} or \code{\link[stats]{ftable}}. Tables of class -\code{\link[stats]{xtabs}} and other will be coerced to \code{ftable} -objects.} +\item{tab}{A \code{\link[=table]{table()}} or \code{\link[=ftable]{ftable()}}. Tables of class \code{\link[=xtabs]{xtabs()}} and +other will be coerced to \code{ftable} objects.} \item{...}{Other arguments, passed down to the statistic functions -\code{\link[stats]{chisq.test}}, \code{\link[stats]{fisher.test}} or -\code{\link[stats]{cor.test}}.} +\code{\link[=chisq.test]{chisq.test()}}, \code{\link[=fisher.test]{fisher.test()}} or \code{\link[=cor.test]{cor.test()}}.} \item{formula}{A formula of the form \code{lhs ~ rhs} where \code{lhs} is a numeric variable giving the data values and \code{rhs} a factor giving the corresponding groups.} \item{data}{A data frame or a table object. If a table object, \code{x1} and -\code{x2} will be ignored. For Kendall's \emph{tau}, Spearman's \emph{rho} -or Pearson's product moment correlation coefficient, \code{data} needs -to be a data frame. If \code{x1} and \code{x2} are not specified, -the first two columns of the data frames are used as variables -to compute the crosstab.} +\code{x2} will be ignored. For Kendall's \emph{tau}, Spearman's \emph{rho} or Pearson's +product moment correlation coefficient, \code{data} needs to be a data frame. +If \code{x1} and \code{x2} are not specified, the first two columns of the data +frames are used as variables to compute the crosstab.} \item{ci.lvl}{Scalar between 0 and 1. If not \code{NULL}, returns a data frame including lower and upper confidence intervals.} @@ -71,37 +71,37 @@ quantiles of the bootstrapped values. See 'Details' in \code{boot_ci()}. May be abbreviated.} \item{x1}{Name of first variable that should be used to compute the -contingency table. If \code{data} is a table object, this argument -will be irgnored.} +contingency table. If \code{data} is a table object, this argument will be +irgnored.} \item{x2}{Name of second variable that should be used to compute the -contingency table. If \code{data} is a table object, this argument -will be irgnored.} +contingency table. If \code{data} is a table object, this argument will be +irgnored.} \item{statistics}{Name of measure of association that should be computed. May -be one of \code{"auto"}, \code{"cramer"}, \code{"phi"}, \code{"spearman"}, -\code{"kendall"}, \code{"pearson"} or \code{"fisher"}. See 'Details'.} +be one of \code{"auto"}, \code{"cramer"}, \code{"phi"}, \code{"spearman"}, \code{"kendall"}, +\code{"pearson"} or \code{"fisher"}. See 'Details'.} \item{weights}{Name of variable in \code{x} that indicated the vector of weights that will be applied to weight all observations. Default is \code{NULL}, so no weights are used.} } \value{ -For \code{phi()}, the table's Phi value. For \code{cramer()}, the +For \code{\link[=phi]{phi()}}, the table's Phi value. For [\verb{cramers_v()]}, the table's Cramer's V. -\cr \cr + For \code{crosstable_statistics()}, a list with following components: -\describe{ -\item{\code{estimate}}{the value of the estimated measure of association.} -\item{\code{p.value}}{the p-value for the test.} -\item{\code{statistic}}{the value of the test statistic.} -\item{\code{stat.name}}{the name of the test statistic.} -\item{\code{stat.html}}{if applicable, the name of the test statistic, in HTML-format.} -\item{\code{df}}{the degrees of freedom for the contingency table.} -\item{\code{method}}{character string indicating the name of the measure of association.} -\item{\code{method.html}}{if applicable, the name of the measure of association, in HTML-format.} -\item{\code{method.short}}{the short form of association measure, equals the \code{statistics}-argument.} -\item{\code{fisher}}{logical, if Fisher's exact test was used to calculate the p-value.} +\itemize{ +\item \code{estimate}: the value of the estimated measure of association. +\item \code{p.value}: the p-value for the test. +\item \code{statistic}: the value of the test statistic. +\item \code{stat.name}: the name of the test statistic. +\item \code{stat.html}: if applicable, the name of the test statistic, in HTML-format. +\item \code{df}: the degrees of freedom for the contingency table. +\item \code{method}: character string indicating the name of the measure of association. +\item \code{method.html}: if applicable, the name of the measure of association, in HTML-format. +\item \code{method.short}: the short form of association measure, equals the \code{statistics}-argument. +\item \code{fisher}: logical, if Fisher's exact test was used to calculate the p-value. } } \description{ @@ -112,19 +112,18 @@ Kendall's tau and Pearson's r. } \details{ The p-value for Cramer's V and the Phi coefficient are based -on \code{chisq.test()}. If any expected value of a table cell is -smaller than 5, or smaller than 10 and the df is 1, then \code{fisher.test()} -is used to compute the p-value, unless \code{statistics = "fisher"}; in -this case, the use of \code{fisher.test()} is forced to compute the -p-value. The test statistic is calculated with \code{cramer()} resp. -\code{phi()}. -\cr \cr -Both test statistic and p-value for Spearman's rho, Kendall's tau -and Pearson's r are calculated with \code{cor.test()}. -\cr \cr -When \code{statistics = "auto"}, only Cramer's V or Phi are calculated, -based on the dimension of the table (i.e. if the table has more than -two rows or columns, Cramer's V is calculated, else Phi). +on \code{chisq.test()}. If any expected value of a table cell is smaller than 5, +or smaller than 10 and the df is 1, then \code{fisher.test()} is used to compute +the p-value, unless \code{statistics = "fisher"}; in this case, the use of +\code{fisher.test()} is forced to compute the p-value. The test statistic is +calculated with \code{cramers_v()} resp. \code{phi()}. + +Both test statistic and p-value for Spearman's rho, Kendall's tau and +Pearson's r are calculated with \code{cor.test()}. + +When \code{statistics = "auto"}, only Cramer's V or Phi are calculated, based on +the dimension of the table (i.e. if the table has more than two rows or +columns, Cramer's V is calculated, else Phi). } \examples{ # Phi coefficient for 2x2 tables diff --git a/man/kruskal_wallis_test.Rd b/man/kruskal_wallis_test.Rd index 75a760b..366a544 100644 --- a/man/kruskal_wallis_test.Rd +++ b/man/kruskal_wallis_test.Rd @@ -27,6 +27,6 @@ This function performs a Kruskal-Wallis rank sum test, see } \examples{ data(efc) -# Mann-Whitney-U-Tests for elder's age by elder's sex. -kruskal_wallis_test(efc, "e17age", by = "e16sex") +# Kruskal-Wallis-Test for elder's age by education +kruskal_wallis_test(efc, "e17age", by = "c172code") } diff --git a/pkgdown/extra.scss b/pkgdown/extra.scss new file mode 100644 index 0000000..a6d0531 --- /dev/null +++ b/pkgdown/extra.scss @@ -0,0 +1,22 @@ +// font size for text +body { + font-size: 0.90rem !important; +} + +// color for links +$link-color: #03638e; + +a { + text-decoration: none !important; + color: $link-color +} + +code a:any-link { + text-decoration: none !important; + color: $link-color +} + +.nav-text.text-default.me-auto { + color: rgba(0,0,0,0.9) !important; +} + From 18f4ffcee7b64ab6d3f0e9f2bfaab5e06ab95083 Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 13:36:36 +0200 Subject: [PATCH 24/34] fix --- R/phi.R | 18 +++++++----------- 1 file changed, 7 insertions(+), 11 deletions(-) diff --git a/R/phi.R b/R/phi.R index f173ce2..48a07f1 100644 --- a/R/phi.R +++ b/R/phi.R @@ -26,17 +26,13 @@ phi.formula <- function(formula, data, ci.lvl = NULL, n = 1000, method = c("dist if (is.null(ci.lvl) || is.na(ci.lvl)) { .cramer(tab) } else { - ci <- data[, terms] %>% - sjstats::bootstrap(n) %>% - dplyr::mutate( - tables = lapply(.data$strap, function(x) { - dat <- as.data.frame(x) - table(dat[[1]], dat[[2]]) - }), - phis = sapply(.data$tables, function(x) .cramer(x)) - ) %>% - dplyr::pull("phis") %>% - boot_ci(ci.lvl = ci.lvl, method = method) + straps <- sjstats::bootstrap(data[terms], n) + tables <- lapply(straps$strap, function(x) { + dat <- as.data.frame(x) + table(dat[[1]], dat[[2]]) + }) + phis <- sapply(tables, function(x) .phi(x)) + ci <- boot_ci(phis, ci.lvl = ci.lvl, method = method) data_frame( phi = .phi(tab), From 947b485301896b2508cd1aad1cbf5957d8ccb5a8 Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 13:43:41 +0200 Subject: [PATCH 25/34] fix --- R/helpfunctions.R | 2 +- R/prop.R | 42 +++++++++++++++++++++++------------------- 2 files changed, 24 insertions(+), 20 deletions(-) diff --git a/R/helpfunctions.R b/R/helpfunctions.R index 2db9790..856f955 100644 --- a/R/helpfunctions.R +++ b/R/helpfunctions.R @@ -34,7 +34,7 @@ get_glm_family <- function(fit) { # create logical for family binom_fam <- fitfam %in% c("binomial", "quasibinomial") poisson_fam <- fitfam %in% c("poisson", "quasipoisson") || - sjmisc::str_contains(fitfam, "negative binomial", ignore.case = T) + sjmisc::str_contains(fitfam, "negative binomial", ignore.case = TRUE) list(is_bin = binom_fam, is_pois = poisson_fam, is_logit = logit_link) } diff --git a/R/prop.R b/R/prop.R index e22e994..167b3cd 100644 --- a/R/prop.R +++ b/R/prop.R @@ -1,11 +1,11 @@ #' @title Proportions of values in a vector #' @name prop #' -#' @description \code{prop()} calculates the proportion of a value or category -#' in a variable. \code{props()} does the same, but allows for +#' @description `prop()` calculates the proportion of a value or category +#' in a variable. `props()` does the same, but allows for #' multiple logical conditions in one statement. It is similar -#' to \code{mean()} with logical predicates, however, both -#' \code{prop()} and \code{props()} work with grouped data frames. +#' to `mean()` with logical predicates, however, both +#' `prop()` and `props()` work with grouped data frames. #' #' @param data A data frame. May also be a grouped data frame (see 'Examples'). #' @param ... One or more value pairs of comparisons (logical predicates). Put @@ -14,17 +14,17 @@ #' 'Examples'. #' @param weights Vector of weights that will be applied to weight all observations. #' Must be a vector of same length as the input vector. Default is -#' \code{NULL}, so no weights are used. +#' `NULL`, so no weights are used. #' @param na.rm Logical, whether to remove NA values from the vector when the -#' proportion is calculated. \code{na.rm = FALSE} gives you the raw -#' percentage of a value in a vector, \code{na.rm = TRUE} the valid +#' proportion is calculated. `na.rm = FALSE` gives you the raw +#' percentage of a value in a vector, `na.rm = TRUE` the valid #' percentage. #' @param digits Amount of digits for returned values. #' -#' @details \code{prop()} only allows one logical statement per comparison, -#' while \code{props()} allows multiple logical statements per comparison. -#' However, \code{prop()} supports weighting of variables before calculating -#' proportions, and comparisons may also be quoted. Hence, \code{prop()} +#' @details `prop()` only allows one logical statement per comparison, +#' while `props()` allows multiple logical statements per comparison. +#' However, `prop()` supports weighting of variables before calculating +#' proportions, and comparisons may also be quoted. Hence, `prop()` #' also processes comparisons, which are passed as character vector #' (see 'Examples'). #' @@ -96,7 +96,9 @@ #' @export prop <- function(data, ..., weights = NULL, na.rm = TRUE, digits = 4) { # check argument - if (!is.data.frame(data)) stop("`data` needs to be a data frame.", call. = F) + if (!is.data.frame(data)) { + insight::format_error("`data` needs to be a data frame.") + } # get dots dots <- match.call(expand.dots = FALSE)$`...` @@ -109,7 +111,9 @@ prop <- function(data, ..., weights = NULL, na.rm = TRUE, digits = 4) { #' @export props <- function(data, ..., na.rm = TRUE, digits = 4) { # check argument - if (!is.data.frame(data)) stop("`data` needs to be a data frame.", call. = F) + if (!is.data.frame(data)) { + insight::format_error("`data` needs to be a data frame.") + } # get dots dots <- match.call(expand.dots = FALSE)$`...` @@ -123,7 +127,7 @@ proportions <- function(data, dots, weight.by, na.rm, digits, multi_logical) { # remember comparisons comparisons <- lapply(dots, function(x) { # to character, and remove spaces and quotes - x <- gsub(" ", "", deparse(x), fixed = T) + x <- gsub(" ", "", deparse(x), fixed = TRUE) x <- gsub("\"", "", x, fixed = TRUE) x }) @@ -188,7 +192,7 @@ proportions <- function(data, dots, weight.by, na.rm, digits, multi_logical) { # order rows by values of grouping variables fr <- fr[do.call(order, reihenfolge), ] - return(fr) + fr } else { # iterate dots (comparing conditions) @@ -206,14 +210,14 @@ proportions <- function(data, dots, weight.by, na.rm, digits, multi_logical) { )) } - return(unlist(result)) + unlist(result) } } get_proportion <- function(x, data, weight.by, na.rm, digits) { # to character, and remove spaces and quotes - x <- gsub(" ", "", deparse(x), fixed = T) + x <- gsub(" ", "", deparse(x), fixed = TRUE) x <- gsub("\"", "", x, fixed = TRUE) # split expression at ==, < or > @@ -252,7 +256,7 @@ get_proportion <- function(x, data, weight.by, na.rm, digits) { if (na.rm) dummy <- na.omit(dummy) # get proportion - round(sum(dummy, na.rm = T) / length(dummy), digits = digits) + round(sum(dummy, na.rm = TRUE) / length(dummy), digits = digits) } @@ -264,5 +268,5 @@ get_multiple_proportion <- function(x, data, na.rm, digits) { if (na.rm) dummy <- na.omit(dummy) # get proportion - round(sum(dummy, na.rm = T) / length(dummy), digits = digits) + round(sum(dummy, na.rm = TRUE) / length(dummy), digits = digits) } From bf5663b0192fcf72569abf6493cc76d6f42fde43 Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 14:32:32 +0200 Subject: [PATCH 26/34] docs --- NAMESPACE | 3 --- R/chi_squared_test.R | 43 ++++++++++++++++++++++++++----- R/find_beta.R | 53 +++++++++++++++++--------------------- R/helpfunctions.R | 1 + R/kruskal_wallis_test.R | 9 +++++-- R/mann_whitney_test.R | 14 +++++++--- man/chi_squared_test.Rd | 28 ++++++++++++++++---- man/find_beta.Rd | 29 ++++++++++----------- man/kruskal_wallis_test.Rd | 10 +++++-- man/mann_whitney_test.Rd | 14 +++++++--- 10 files changed, 133 insertions(+), 71 deletions(-) diff --git a/NAMESPACE b/NAMESPACE index e351d2c..5a224af 100644 --- a/NAMESPACE +++ b/NAMESPACE @@ -167,7 +167,6 @@ importFrom(sjmisc,is_num_fac) importFrom(sjmisc,str_contains) importFrom(sjmisc,trim) importFrom(sjmisc,typical_value) -importFrom(stats,approx) importFrom(stats,as.formula) importFrom(stats,chisq.test) importFrom(stats,coef) @@ -183,12 +182,10 @@ importFrom(stats,model.matrix) importFrom(stats,na.omit) importFrom(stats,na.pass) importFrom(stats,nobs) -importFrom(stats,pbeta) importFrom(stats,pf) importFrom(stats,pnorm) importFrom(stats,predict.glm) importFrom(stats,pt) -importFrom(stats,qcauchy) importFrom(stats,qf) importFrom(stats,qnorm) importFrom(stats,resid) diff --git a/R/chi_squared_test.R b/R/chi_squared_test.R index 960ba31..3fd6932 100644 --- a/R/chi_squared_test.R +++ b/R/chi_squared_test.R @@ -1,9 +1,10 @@ #' @title Chi-Squared Test #' @name chi_squared_test -#' @description This function performs a Mann-Whitney-Test (or Wilcoxon rank -#' sum test for _unpaired_ samples, see [`wilcox.test()`] and [`coin::wilcox_test()`]). -#' -#' The function reports p and Z-values as well as effect size r and group-rank-means. +#' @description This function performs a \eqn{chi}^2 test for contingency +#' tables or tests for given probabilities. The returned effects sizes are +#' Cramer's V for tables with more than two rows and columns, Phi (\eqn{\phi}) +#' for 2x2 tables, and \ifelse{latex}{\eqn{Fei}}{פ (Fei)} for tests against +#' given probabilities (see _Ben-Shachar et al. 2023_). #' #' @param probabilities A numeric vector of probabilities for each cell in the #' contingency table. The length of the vector must match the number of cells @@ -14,7 +15,22 @@ #' @param ... Additional arguments passed down to [`chisq.test()`]. #' @inheritParams mann_whitney_test #' -#' @return A data frame with test results. +#' @return A data frame with test results. The returned effects sizes are +#' Cramer's V for tables with more than two rows and columns, Phi (\eqn{\phi}) +#' for 2x2 tables, and \ifelse{latex}{\eqn{Fei}}{פ (Fei)} for tests against +#' given probabilities. +#' +#' @details The function is a wrapper around [`chisq.test()`] and +#' [`fisher.test()`] (for small expected values) for contingency tables, and +#' `chisq.test()` for given probabilities. When `probabilities` are provided, +#' these are rescaled to sum to 1 (i.e. `rescale.p = TRUE`). When `fisher.test()` +#' is called, simulated p-values are returned (i.e. `simulate.p.value = TRUE`, +#' see `?fisher.test`). +#' +#' @references Ben-Shachar, M.S., Patil, I., Thériault, R., Wiernik, B.M., +#' Lüdecke, D. (2023). Phi, Fei, Fo, Fum: Effect Sizes for Categorical Data +#' That Use the Chi‑Squared Statistic. Mathematics, 11, 1982. +#' \doi{10.3390/math11091982} #' #' @examples #' data(efc) @@ -105,6 +121,7 @@ chi_squared_test <- function(data, ) class(out) <- c("sj_htest_chi", "data.frame") attr(out, "weighted") <- !is.null(weights) + attr(out, "fisher") <- isTRUE(startsWith(htest$method, "Fisher")) attr(out, "caption") <- "Contingency Tables" out } @@ -189,15 +206,27 @@ print.sj_htest_chi <- function(x, ...) { weight_string <- "" } + fisher <- attributes(x)$fisher + # headline insight::print_color(sprintf( - "\n# Chi-Squared Test for %s%s\n\n", + "\n# Chi-Squared Test for %s%s\n", attributes(x)$caption, weight_string ), "blue") + # Fisher's exact test? + if (fisher) { + insight::print_color(" (using Fisher's exact test due to small expected values)\n", "blue") # nolint + } + + cat("\n") + # data info - insight::print_color(sprintf(" Data: %s (n = %i)\n", x$data, round(x$n_obs)), "cyan") + insight::print_color( + sprintf(" Data: %s (n = %i)\n", x$data, round(x$n_obs)), + "cyan" + ) # prepare and align strings eff_symbol <- .format_symbols(x$effect_size_name) diff --git a/R/find_beta.R b/R/find_beta.R index 7e45408..1ae733c 100644 --- a/R/find_beta.R +++ b/R/find_beta.R @@ -1,10 +1,10 @@ #' @title Determining distribution parameters #' @name find_beta #' -#' @description \code{find_beta()}, \code{find_normal()} and \code{find_cauchy()} find the +#' @description `find_beta()`, `find_normal()` and `find_cauchy()` find the #' shape, mean and standard deviation resp. the location and scale parameters #' to describe the beta, normal or cauchy distribution, based on two -#' percentiles. \code{find_beta2()} finds the shape parameters for a Beta +#' percentiles. `find_beta2()` finds the shape parameters for a Beta #' distribution, based on a probability value and its standard error #' or confidence intervals. #' @@ -14,14 +14,14 @@ #' @param p2 Probability of the second percentile. #' @param x Numeric, a probability value between 0 and 1. Typically indicates #' a prevalence rate of an outcome of interest; Or an integer value -#' with the number of observed events. In this case, specify \code{n} +#' with the number of observed events. In this case, specify `n` #' to indicate the toral number of observations. -#' @param se The standard error of \code{x}. Either \code{se} or \code{ci} must +#' @param se The standard error of `x`. Either `se` or `ci` must #' be specified. -#' @param ci The upper limit of the confidence interval of \code{x}. Either -#' \code{se} or \code{ci} must be specified. +#' @param ci The upper limit of the confidence interval of `x`. Either +#' `se` or `ci` must be specified. #' @param n Numeric, number of total observations. Needs to be specified, if -#' \code{x} is an integer (number of observed events), and no +#' `x` is an integer (number of observed events), and no #' probability. See 'Examples'. #' #' @return A list of length two, with the two distribution parameters than can @@ -29,21 +29,20 @@ #' the shape for the given input parameters. #' #' @details These functions can be used to find parameter for various distributions, -#' to define prior probabilities for Bayesian analyses. \code{x1}, -#' \code{p1}, \code{x2} and \code{p2} are parameters that describe two -#' quantiles. Given this knowledge, the distribution parameters are -#' returned. \cr \cr -#' Use \code{find_beta2()}, if the known parameters are, e.g. a prevalence -#' rate or similar probability, and its standard deviation or confidence -#' interval. In this case. \code{x} should be a probability, -#' for example a prevalence rate of a certain event. \code{se} then -#' needs to be the standard error for this probability. Alternatively, -#' \code{ci} can be specified, which should indicate the upper limit -#' of the confidence interval od the probability (prevalence rate) \code{x}. -#' If the number of events out of a total number of trials is known -#' (e.g. 12 heads out of 30 coin tosses), \code{x} can also be the number -#' of observed events, while \code{n} indicates the total amount of trials -#' (in the above example, the function call would be: \code{find_beta2(x = 12, n = 30)}). +#' to define prior probabilities for Bayesian analyses. `x1`, `p1`, `x2` and +#' `p2` are parameters that describe two quantiles. Given this knowledge, the +#' distribution parameters are returned. +#' +#' Use `find_beta2()`, if the known parameters are, e.g. a prevalence rate or +#' similar probability, and its standard deviation or confidence interval. In +#' this case. `x` should be a probability, for example a prevalence rate of a +#' certain event. `se` then needs to be the standard error for this probability. +#' Alternatively, `ci` can be specified, which should indicate the upper limit +#' of the confidence interval od the probability (prevalence rate) `x`. If the +#' number of events out of a total number of trials is known (e.g. 12 heads out +#' of 30 coin tosses), `x` can also be the number of observed events, while `n` +#' indicates the total amount of trials (in the above example, the function +#' call would be: `find_beta2(x = 12, n = 30)`). #' #' @references Cook JD. Determining distribution parameters from quantiles. 2010: Department of Biostatistics, Texas (\href{https://www.johndcook.com/quantiles_parameters.pdf}{PDF}) #' @@ -79,14 +78,12 @@ #' shapes <- find_beta2(x = 3, n = 20) #' curve(dbeta(x, shapes[[1]], shapes[[2]])) #' -#' @importFrom stats pbeta approx -#' @importFrom purrr map_dbl #' @export find_beta <- function(x1, p1, x2, p2) { logK <- seq(-5, 10, length = 200) K <- exp(logK) - m <- purrr::map_dbl(K, ~ betaprior(.x, x1, p1)) + m <- unlist(lapply(K, betaprior, x = x1, p = p1)) prob2 <- stats::pbeta(x2, K * m, K * (1 - m)) ind <- ((prob2 > 0) & (prob2 < 1)) @@ -127,13 +124,13 @@ betaprior <- function(K, x, p) { find_beta2 <- function(x, se, ci, n) { # check if all required arguments are given if (missing(se) && missing(ci) && missing(n)) { - stop("Either `se` or `ci`, or `n` must be specified.", call. = F) + insight::format_error("Either `se` or `ci`, or `n` must be specified.") } # for number of observations, compute variance of beta distribution if (!missing(n)) { if (!is.integer(x) && x < 1) - stop("If `n` is given, x` must be an integer value greater than 0.", call. = F) + insight::format_error("If `n` is given, x` must be an integer value greater than 0.") # compute 2 SD from beta variance bvar <- 2 * sqrt((x * n) / ((x + n)^2 * (x + n + 1))) @@ -164,7 +161,6 @@ find_beta2 <- function(x, se, ci, n) { } -#' @importFrom stats qcauchy #' @rdname find_beta #' @export find_cauchy <- function(x1, p1, x2, p2) { @@ -177,7 +173,6 @@ find_cauchy <- function(x1, p1, x2, p2) { -#' @importFrom stats qnorm #' @rdname find_beta #' @export find_normal <- function(x1, p1, x2, p2) { diff --git a/R/helpfunctions.R b/R/helpfunctions.R index 856f955..ce36f36 100644 --- a/R/helpfunctions.R +++ b/R/helpfunctions.R @@ -84,6 +84,7 @@ get_grouped_data <- function(x) { x <- gsub("Rho", "\u03C1", x, ignore.case = TRUE) x <- gsub("Mu", "\u03BC", x, ignore.case = TRUE) x <- gsub("Theta", "\u03B8", x, ignore.case = TRUE) + x <- gsub("Fei", "\u05E4\u200E", x, ignore.case = TRUE) } x } diff --git a/R/kruskal_wallis_test.R b/R/kruskal_wallis_test.R index 80b6b7b..d004c73 100644 --- a/R/kruskal_wallis_test.R +++ b/R/kruskal_wallis_test.R @@ -1,12 +1,17 @@ #' @title Kruskal-Wallis-Test #' @name kruskal_wallis_test -#' @description This function performs a Kruskal-Wallis rank sum test, see -#' [`kruskal.test()`] and [`coin::kruskal_test()`]). +#' @description This function performs a Kruskal-Wallis rank sum test, to test +#' the null hypothesis that the population median of all of the groups are +#' equal. The alternative is that they differ in at least one. #' #' @inheritParams mann_whitney_test #' #' @return A data frame with test results. #' +#' @details The function simply is a wrapper around [`kruskal.test()`]. The +#' weighted version of the Kruskal-Wallis test is based on the `survey` package, +#' using [`survey::svyranktest()`]. +#' #' @examples #' data(efc) #' # Kruskal-Wallis-Test for elder's age by education diff --git a/R/mann_whitney_test.R b/R/mann_whitney_test.R index 6dbbc25..2473c81 100644 --- a/R/mann_whitney_test.R +++ b/R/mann_whitney_test.R @@ -1,9 +1,11 @@ #' @title Mann-Whitney-Test #' @name mann_whitney_test #' @description This function performs a Mann-Whitney-Test (or Wilcoxon rank -#' sum test for _unpaired_ samples, see [`wilcox.test()`] and [`coin::wilcox_test()`]). +#' sum test for _unpaired_ samples. #' -#' The function reports p and Z-values as well as effect size r and group-rank-means. +#' A Mann-Whitney-Test is a non-parametric test for the null hypothesis that two +#' independent samples have identical continuous distributions. It can be used +#' when the two continuous variables are not normally distributed. #' #' @param data A data frame. #' @param select Name of the dependent variable (as string) to be used for the @@ -16,9 +18,13 @@ #' should be computed. May be one of `"exact"`, `"approximate"` or `"asymptotic"` #' (default). See [`coin::wilcox_test()`] for details. #' -#' @return A data frame with test results. +#' @return A data frame with test results. The function returns p and Z-values +#' as well as effect size r and group-rank-means. +#' +#' @details This function is based on [`wilcox.test()`] and [`coin::wilcox_test()`] +#' (the latter to extract effect sizes). The weighted version of the test is +#' based on [`survey::svyranktest()`]. #' -#' @details This function calls [`coin::wilcox_test()`] to extract effect sizes. #' Interpretation of the effect size **r**, as a rule-of-thumb: #' #' - small effect >= 0.1 diff --git a/man/chi_squared_test.Rd b/man/chi_squared_test.Rd index abb066f..e3196d9 100644 --- a/man/chi_squared_test.Rd +++ b/man/chi_squared_test.Rd @@ -35,13 +35,25 @@ must be \code{NULL}. The probabilities must sum to 1.} \item{...}{Additional arguments passed down to \code{\link[=chisq.test]{chisq.test()}}.} } \value{ -A data frame with test results. +A data frame with test results. The returned effects sizes are +Cramer's V for tables with more than two rows and columns, Phi (\eqn{\phi}) +for 2x2 tables, and \ifelse{latex}{\eqn{Fei}}{פ (Fei)} for tests against +given probabilities. } \description{ -This function performs a Mann-Whitney-Test (or Wilcoxon rank -sum test for \emph{unpaired} samples, see \code{\link[=wilcox.test]{wilcox.test()}} and \code{\link[coin:LocationTests]{coin::wilcox_test()}}). - -The function reports p and Z-values as well as effect size r and group-rank-means. +This function performs a \eqn{chi}^2 test for contingency +tables or tests for given probabilities. The returned effects sizes are +Cramer's V for tables with more than two rows and columns, Phi (\eqn{\phi}) +for 2x2 tables, and \ifelse{latex}{\eqn{Fei}}{פ (Fei)} for tests against +given probabilities (see \emph{Ben-Shachar et al. 2023}). +} +\details{ +The function is a wrapper around \code{\link[=chisq.test]{chisq.test()}} and +\code{\link[=fisher.test]{fisher.test()}} (for small expected values) for contingency tables, and +\code{chisq.test()} for given probabilities. When \code{probabilities} are provided, +these are rescaled to sum to 1 (i.e. \code{rescale.p = TRUE}). When \code{fisher.test()} +is called, simulated p-values are returned (i.e. \code{simulate.p.value = TRUE}, +see \code{?fisher.test}). } \examples{ data(efc) @@ -53,3 +65,9 @@ chi_squared_test(efc, "c161sex", by = "e16sex", weights = "weight") # Chi-squared-test for given probabilities chi_squared_test(efc, "c161sex", probabilities = c(0.3, 0.7)) } +\references{ +Ben-Shachar, M.S., Patil, I., Thériault, R., Wiernik, B.M., +Lüdecke, D. (2023). Phi, Fei, Fo, Fum: Effect Sizes for Categorical Data +That Use the Chi‑Squared Statistic. Mathematics, 11, 1982. +\doi{10.3390/math11091982} +} diff --git a/man/find_beta.Rd b/man/find_beta.Rd index 16d2838..ccd5dcc 100644 --- a/man/find_beta.Rd +++ b/man/find_beta.Rd @@ -54,21 +54,20 @@ or confidence intervals. } \details{ These functions can be used to find parameter for various distributions, -to define prior probabilities for Bayesian analyses. \code{x1}, -\code{p1}, \code{x2} and \code{p2} are parameters that describe two -quantiles. Given this knowledge, the distribution parameters are -returned. \cr \cr -Use \code{find_beta2()}, if the known parameters are, e.g. a prevalence -rate or similar probability, and its standard deviation or confidence -interval. In this case. \code{x} should be a probability, -for example a prevalence rate of a certain event. \code{se} then -needs to be the standard error for this probability. Alternatively, -\code{ci} can be specified, which should indicate the upper limit -of the confidence interval od the probability (prevalence rate) \code{x}. -If the number of events out of a total number of trials is known -(e.g. 12 heads out of 30 coin tosses), \code{x} can also be the number -of observed events, while \code{n} indicates the total amount of trials -(in the above example, the function call would be: \code{find_beta2(x = 12, n = 30)}). +to define prior probabilities for Bayesian analyses. \code{x1}, \code{p1}, \code{x2} and +\code{p2} are parameters that describe two quantiles. Given this knowledge, the +distribution parameters are returned. + +Use \code{find_beta2()}, if the known parameters are, e.g. a prevalence rate or +similar probability, and its standard deviation or confidence interval. In +this case. \code{x} should be a probability, for example a prevalence rate of a +certain event. \code{se} then needs to be the standard error for this probability. +Alternatively, \code{ci} can be specified, which should indicate the upper limit +of the confidence interval od the probability (prevalence rate) \code{x}. If the +number of events out of a total number of trials is known (e.g. 12 heads out +of 30 coin tosses), \code{x} can also be the number of observed events, while \code{n} +indicates the total amount of trials (in the above example, the function +call would be: \code{find_beta2(x = 12, n = 30)}). } \examples{ # example from blogpost: diff --git a/man/kruskal_wallis_test.Rd b/man/kruskal_wallis_test.Rd index 366a544..7be4e2e 100644 --- a/man/kruskal_wallis_test.Rd +++ b/man/kruskal_wallis_test.Rd @@ -22,8 +22,14 @@ not a factor, it will be coerced to a factor. For \code{chi_squared_test()}, if A data frame with test results. } \description{ -This function performs a Kruskal-Wallis rank sum test, see -\code{\link[=kruskal.test]{kruskal.test()}} and \code{\link[coin:LocationTests]{coin::kruskal_test()}}). +This function performs a Kruskal-Wallis rank sum test, to test +the null hypothesis that the population median of all of the groups are +equal. The alternative is that they differ in at least one. +} +\details{ +The function simply is a wrapper around \code{\link[=kruskal.test]{kruskal.test()}}. The +weighted version of the Kruskal-Wallis test is based on the \code{survey} package, +using \code{\link[survey:svyranktest]{survey::svyranktest()}}. } \examples{ data(efc) diff --git a/man/mann_whitney_test.Rd b/man/mann_whitney_test.Rd index 2256df8..1401a04 100644 --- a/man/mann_whitney_test.Rd +++ b/man/mann_whitney_test.Rd @@ -29,16 +29,22 @@ should be computed. May be one of \code{"exact"}, \code{"approximate"} or \code{ (default). See \code{\link[coin:LocationTests]{coin::wilcox_test()}} for details.} } \value{ -A data frame with test results. +A data frame with test results. The function returns p and Z-values +as well as effect size r and group-rank-means. } \description{ This function performs a Mann-Whitney-Test (or Wilcoxon rank -sum test for \emph{unpaired} samples, see \code{\link[=wilcox.test]{wilcox.test()}} and \code{\link[coin:LocationTests]{coin::wilcox_test()}}). +sum test for \emph{unpaired} samples. -The function reports p and Z-values as well as effect size r and group-rank-means. +A Mann-Whitney-Test is a non-parametric test for the null hypothesis that two +independent samples have identical continuous distributions. It can be used +when the two continuous variables are not normally distributed. } \details{ -This function calls \code{\link[coin:LocationTests]{coin::wilcox_test()}} to extract effect sizes. +This function is based on \code{\link[=wilcox.test]{wilcox.test()}} and \code{\link[coin:LocationTests]{coin::wilcox_test()}} +(the latter to extract effect sizes). The weighted version of the test is +based on \code{\link[survey:svyranktest]{survey::svyranktest()}}. + Interpretation of the effect size \strong{r}, as a rule-of-thumb: \itemize{ \item small effect >= 0.1 From f47612db87578aef94dc59945075ea5808a61764 Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 14:34:50 +0200 Subject: [PATCH 27/34] docs --- R/chi_squared_test.R | 3 +++ R/xtab_statistics.R | 5 +++++ man/chi_squared_test.Rd | 3 +++ man/crosstable_statistics.Rd | 6 ++++++ 4 files changed, 17 insertions(+) diff --git a/R/chi_squared_test.R b/R/chi_squared_test.R index 3fd6932..ad898a9 100644 --- a/R/chi_squared_test.R +++ b/R/chi_squared_test.R @@ -27,6 +27,9 @@ #' is called, simulated p-values are returned (i.e. `simulate.p.value = TRUE`, #' see `?fisher.test`). #' +#' The weighted version of the chi-squared test is based on the a weighted +#' table, using [`xtabs()`] as input for `chisq.test()`. +#' #' @references Ben-Shachar, M.S., Patil, I., Thériault, R., Wiernik, B.M., #' Lüdecke, D. (2023). Phi, Fei, Fo, Fum: Effect Sizes for Categorical Data #' That Use the Chi‑Squared Statistic. Mathematics, 11, 1982. diff --git a/R/xtab_statistics.R b/R/xtab_statistics.R index 2e9c619..2182419 100644 --- a/R/xtab_statistics.R +++ b/R/xtab_statistics.R @@ -66,6 +66,11 @@ #' the dimension of the table (i.e. if the table has more than two rows or #' columns, Cramer's V is calculated, else Phi). #' +#' @references Ben-Shachar, M.S., Patil, I., Thériault, R., Wiernik, B.M., +#' Lüdecke, D. (2023). Phi, Fei, Fo, Fum: Effect Sizes for Categorical Data +#' That Use the Chi‑Squared Statistic. Mathematics, 11, 1982. +#' \doi{10.3390/math11091982} +#' #' @examples #' # Phi coefficient for 2x2 tables #' tab <- table(sample(1:2, 30, TRUE), sample(1:2, 30, TRUE)) diff --git a/man/chi_squared_test.Rd b/man/chi_squared_test.Rd index e3196d9..9a6bb90 100644 --- a/man/chi_squared_test.Rd +++ b/man/chi_squared_test.Rd @@ -54,6 +54,9 @@ The function is a wrapper around \code{\link[=chisq.test]{chisq.test()}} and these are rescaled to sum to 1 (i.e. \code{rescale.p = TRUE}). When \code{fisher.test()} is called, simulated p-values are returned (i.e. \code{simulate.p.value = TRUE}, see \code{?fisher.test}). + +The weighted version of the chi-squared test is based on the a weighted +table, using \code{\link[=xtabs]{xtabs()}} as input for \code{chisq.test()}. } \examples{ data(efc) diff --git a/man/crosstable_statistics.Rd b/man/crosstable_statistics.Rd index 45afaff..cbea412 100644 --- a/man/crosstable_statistics.Rd +++ b/man/crosstable_statistics.Rd @@ -159,3 +159,9 @@ crosstable_statistics(efc, continuity = TRUE ) } +\references{ +Ben-Shachar, M.S., Patil, I., Thériault, R., Wiernik, B.M., +Lüdecke, D. (2023). Phi, Fei, Fo, Fum: Effect Sizes for Categorical Data +That Use the Chi‑Squared Statistic. Mathematics, 11, 1982. +\doi{10.3390/math11091982} +} From 7b17a7c372d587d0ffae19c14b14a3984d55a847 Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 15:13:25 +0200 Subject: [PATCH 28/34] update --- .lintr | 17 ++++++++++ R/chi_squared_test.R | 58 +++++++++++++++++++++++----------- R/kruskal_wallis_test.R | 64 +++++++++++++++++++++++++++----------- R/mann_whitney_test.R | 2 +- man/chi_squared_test.Rd | 15 ++++++--- man/kruskal_wallis_test.Rd | 21 ++++++++++--- 6 files changed, 129 insertions(+), 48 deletions(-) create mode 100644 .lintr diff --git a/.lintr b/.lintr new file mode 100644 index 0000000..b0e8abe --- /dev/null +++ b/.lintr @@ -0,0 +1,17 @@ +linters: linters_with_defaults( + absolute_path_linter = NULL, + commented_code_linter = NULL, + cyclocomp_linter = cyclocomp_linter(25), + extraction_operator_linter = NULL, + implicit_integer_linter = NULL, + line_length_linter(120), + namespace_linter = NULL, + nonportable_path_linter = NULL, + object_name_linter = NULL, + object_length_linter(50), + object_usage_linter = NULL, + todo_comment_linter = NULL, + undesirable_function_linter(c("mapply" = NA, "sapply" = NA, "setwd" = NA)), + undesirable_operator_linter = NULL, + defaults = linters_with_tags(tags = NULL) + ) diff --git a/R/chi_squared_test.R b/R/chi_squared_test.R index ad898a9..f47b0a7 100644 --- a/R/chi_squared_test.R +++ b/R/chi_squared_test.R @@ -1,4 +1,4 @@ -#' @title Chi-Squared Test +#' @title Chi-Squared test #' @name chi_squared_test #' @description This function performs a \eqn{chi}^2 test for contingency #' tables or tests for given probabilities. The returned effects sizes are @@ -12,6 +12,8 @@ #' in `select`. If `probabilities` is provided, a chi-squared test for given #' probabilities is conducted. Furthermore, if `probabilities` is given, `by` #' must be `NULL`. The probabilities must sum to 1. +#' @param paired Logical, if `TRUE`, a McNemar test is conducted for 2x2 tables. +#' Note that `paired` only works for 2x2 tables. #' @param ... Additional arguments passed down to [`chisq.test()`]. #' @inheritParams mann_whitney_test #' @@ -25,7 +27,8 @@ #' `chisq.test()` for given probabilities. When `probabilities` are provided, #' these are rescaled to sum to 1 (i.e. `rescale.p = TRUE`). When `fisher.test()` #' is called, simulated p-values are returned (i.e. `simulate.p.value = TRUE`, -#' see `?fisher.test`). +#' see `?fisher.test`). If `paired = TRUE` and a 2x2 table is provided, +#' a McNemar test (see [`mcnemar.test()`]) is conducted. #' #' The weighted version of the chi-squared test is based on the a weighted #' table, using [`xtabs()`] as input for `chisq.test()`. @@ -38,11 +41,11 @@ #' @examples #' data(efc) #' efc$weight <- abs(rnorm(nrow(efc), 1, 0.3)) -#' # Chi-squared-test +#' # Chi-squared test #' chi_squared_test(efc, "c161sex", by = "e16sex") -#' # weighted Chi-squared-test +#' # weighted Chi-squared test #' chi_squared_test(efc, "c161sex", by = "e16sex", weights = "weight") -#' # Chi-squared-test for given probabilities +#' # Chi-squared test for given probabilities #' chi_squared_test(efc, "c161sex", probabilities = c(0.3, 0.7)) #' @export chi_squared_test <- function(data, @@ -50,10 +53,15 @@ chi_squared_test <- function(data, by = NULL, probabilities = NULL, weights = NULL, + paired = FALSE, ...) { if (is.null(probabilities)) { - .calculate_chisq(data, select, by, weights, ...) + .calculate_chisq(data, select, by, weights, paired, ...) } else { + # sanity check - `paired = TRUE` is not available for given probabilities + if (paired) { + insight::format_error("When `probabilities` are provided, `paired = TRUE` is not available.") # nolint + } .calculate_chisq_gof(data, select, probabilities, weights, ...) } } @@ -61,7 +69,7 @@ chi_squared_test <- function(data, # Mann-Whitney-Test for two groups -------------------------------------------- -.calculate_chisq <- function(data, select, by, weights, verbose = TRUE, ...) { +.calculate_chisq <- function(data, select, by, weights, paired = FALSE, ...) { insight::check_if_installed("datawizard") # sanity checks .sanitize_htest_input(data, select, by, weights) @@ -70,6 +78,11 @@ chi_squared_test <- function(data, grp1 <- data[[select]] grp2 <- data[[by]] + # if paired = TRUE, we only allow a 2x2 table + if (paired && (length(stats::na.omit(unique(grp1))) != 2 || length(stats::na.omit(unique(grp2))) != 2)) { + insight::format_error("When `paired = TRUE`, only 2x2 tables are allowed (i.e. both variables must have exactly two levels).") # nolint + } + # create data frame for table x <- data.frame( grp1 = datawizard::to_factor(grp1), @@ -93,13 +106,18 @@ chi_squared_test <- function(data, # expected values, to identify whether Fisher's test is needed expected_values <- as.table(round(as.array(margin.table(tab, 1)) %*% t(as.array(margin.table(tab, 2))) / margin.table(tab))) # nolint - # chi-squared test - htest <- suppressWarnings(stats::chisq.test(tab, ...)) - test_statistic <- htest$statistic - - # need fisher? - if (min(expected_values) < 5 || (min(expected_values) < 10 && htest$parameter == 1)) { - htest <- stats::fisher.test(tab, simulate.p.value = TRUE, ...) + # paired? mc-nemar test + if (paired) { + htest <- suppressWarnings(stats::mcnemar.test(tab, ...)) + test_statistic <- htest$statistic + } else { + # chi-squared test + htest <- suppressWarnings(stats::chisq.test(tab, ...)) + test_statistic <- htest$statistic + # need fisher? + if (min(expected_values) < 5 || (min(expected_values) < 10 && htest$parameter == 1)) { + htest <- stats::fisher.test(tab, simulate.p.value = TRUE, ...) + } } p_value <- htest$p.value @@ -125,7 +143,8 @@ chi_squared_test <- function(data, class(out) <- c("sj_htest_chi", "data.frame") attr(out, "weighted") <- !is.null(weights) attr(out, "fisher") <- isTRUE(startsWith(htest$method, "Fisher")) - attr(out, "caption") <- "Contingency Tables" + attr(out, "mcnemar") <- isTRUE(paired) + attr(out, "caption") <- "contingency tables" out } @@ -192,7 +211,7 @@ chi_squared_test <- function(data, stringsAsFactors = FALSE ) class(out) <- c("sj_htest_chi", "data.frame") - attr(out, "caption") <- "given Probabilities" + attr(out, "caption") <- "given probabilities" attr(out, "weighted") <- !is.null(weights) out } @@ -210,17 +229,20 @@ print.sj_htest_chi <- function(x, ...) { } fisher <- attributes(x)$fisher + mcnemar <- attributes(x)$mcnemar # headline insight::print_color(sprintf( - "\n# Chi-Squared Test for %s%s\n", + "\n# Chi-squared test for %s%s\n", attributes(x)$caption, weight_string ), "blue") # Fisher's exact test? - if (fisher) { + if (isTRUE(fisher)) { insight::print_color(" (using Fisher's exact test due to small expected values)\n", "blue") # nolint + } else if (isTRUE(mcnemar)) { + insight::print_color(" (using McNemar's test for paired data)\n", "blue") # nolint } cat("\n") diff --git a/R/kruskal_wallis_test.R b/R/kruskal_wallis_test.R index d004c73..dd0eebc 100644 --- a/R/kruskal_wallis_test.R +++ b/R/kruskal_wallis_test.R @@ -1,26 +1,31 @@ -#' @title Kruskal-Wallis-Test +#' @title Kruskal-Wallis test #' @name kruskal_wallis_test #' @description This function performs a Kruskal-Wallis rank sum test, to test #' the null hypothesis that the population median of all of the groups are -#' equal. The alternative is that they differ in at least one. +#' equal. The alternative is that they differ in at least one. If `paired = TRUE`, +#' a paired Friedman test is conducted. #' #' @inheritParams mann_whitney_test +#' @param paired Logical, if `TRUE`, a paired Friedman test is conducted (see +#' [`friedman.test()`]). #' #' @return A data frame with test results. #' #' @details The function simply is a wrapper around [`kruskal.test()`]. The #' weighted version of the Kruskal-Wallis test is based on the `survey` package, -#' using [`survey::svyranktest()`]. +#' using [`survey::svyranktest()`]. When `paired = TRUE`, a paired Friedman test +#' is conducted (see [`friedman.test()`]). #' #' @examples #' data(efc) -#' # Kruskal-Wallis-Test for elder's age by education +#' # Kruskal-Wallis test for elder's age by education #' kruskal_wallis_test(efc, "e17age", by = "c172code") #' @export kruskal_wallis_test <- function(data, select = NULL, by = NULL, - weights = NULL) { + weights = NULL, + paired = FALSE) { insight::check_if_installed("datawizard") # sanity checks @@ -38,22 +43,31 @@ kruskal_wallis_test <- function(data, insight::format_error("At least two groups are required, i.e. data must have at least two unique levels in `by` for `kruskal_wallis_test()`.") # nolint } if (is.null(weights)) { - .calculate_kw(dv, grp) + .calculate_kw(dv, grp, paired) } else { - .calculate_weighted_kw(dv, grp, data[[weights]]) + .calculate_weighted_kw(dv, grp, data[[weights]], paired = TRUE) } } # Kruskal-Wallis-Test -------------------------------------------- -.calculate_kw <- function(dv, grp) { +.calculate_kw <- function(dv, grp, paired = FALSE) { # prepare data wcdat <- data.frame(dv, grp) - # perfom wilcox test - wt <- stats::kruskal.test(dv ~ grp, data = wcdat) + if (paired) { + # perfom friedman test for paired data + wt <- stats::friedman.test(table(wcdat)) + } else { + # perfom kruskal wallis test + wt <- stats::kruskal.test(dv ~ grp, data = wcdat) + } # number of groups - n_groups <- vapply(unique(grp), function(g) sum(grp == g, na.rm = TRUE), numeric(1)) + n_groups <- vapply( + stats::na.omit(unique(grp)), + function(g) sum(grp == g, na.rm = TRUE), + numeric(1) + ) out <- data.frame( data = wt$data.name, @@ -64,7 +78,7 @@ kruskal_wallis_test <- function(data, ) attr(out, "n_groups") <- n_groups - attr(out, "method") <- "kruskal" + attr(out, "method") <- ifelse(paired, "friedman", "kruskal") attr(out, "weighted") <- FALSE class(out) <- c("sj_htest_kw", "data.frame") @@ -74,21 +88,28 @@ kruskal_wallis_test <- function(data, # Weighted Mann-Whitney-Test for two groups ---------------------------------- -.calculate_weighted_kw <- function(dv, grp, weights) { +.calculate_weighted_kw <- function(dv, grp, weights, paired = FALSE) { # check if pkg survey is available insight::check_if_installed("survey") dat <- stats::na.omit(data.frame(dv, grp, weights)) colnames(dat) <- c("x", "g", "w") - design <- survey::svydesign(ids = ~0, data = dat, weights = ~w) - result <- survey::svyranktest(formula = x ~ g, design, test = "KruskalWallis") - # number of groups - n_groups <- vapply(unique(grp), function(g) { + n_groups <- vapply(stats::na.omit(unique(grp)), function(g) { sum(dat$w[dat$grp == g], na.rm = TRUE) }, numeric(1)) + if (paired) { + tab <- as.table(round(stats::xtabs(x[[3]] ~ x[[1]] + x[[2]]))) + class(tab) <- "table" + # perfom friedman test for paired data + result <- stats::friedman.test(tab) + } else { + design <- survey::svydesign(ids = ~0, data = dat, weights = ~w) + result <- survey::svyranktest(formula = x ~ g, design, test = "KruskalWallis") + } + out <- data.frame( data = paste(dv, "by", grp), Chi2 = result$statistic, @@ -98,7 +119,7 @@ kruskal_wallis_test <- function(data, ) attr(out, "n_groups") <- n_groups - attr(out, "method") <- "kruskal" + attr(out, "method") <- ifelse(paired, "friedman", "kruskal") attr(out, "weighted") <- TRUE class(out) <- c("sj_htest_kw", "data.frame") @@ -114,6 +135,7 @@ print.sj_htest_kw <- function(x, ...) { # fetch attributes n_groups <- attributes(x)$n_groups weighted <- attributes(x)$weighted + method <- attributes(x)$method if (weighted) { weight_string <- " (weighted)" @@ -122,7 +144,11 @@ print.sj_htest_kw <- function(x, ...) { } # header - insight::print_color(sprintf("# Kruskal-Wallis-Test%s\n\n", weight_string), "blue") + if (identical(method, "kruskal")) { + insight::print_color(sprintf("# Kruskal-Wallis test%s\n\n", weight_string), "blue") + } else { + insight::print_color(sprintf("# Friedman test%s\n\n", weight_string), "blue") + } # data info insight::print_color( diff --git a/R/mann_whitney_test.R b/R/mann_whitney_test.R index 2473c81..d900bf2 100644 --- a/R/mann_whitney_test.R +++ b/R/mann_whitney_test.R @@ -321,7 +321,7 @@ print.sj_htest_mwu <- function(x, ...) { group_labels <- format(group_labels) # header - insight::print_color(sprintf("# Mann-Whitney-Test%s\n\n", weight_string), "blue") + insight::print_color(sprintf("# Mann-Whitney test%s\n\n", weight_string), "blue") # group-1-info insight::print_color( diff --git a/man/chi_squared_test.Rd b/man/chi_squared_test.Rd index 9a6bb90..9bcfd73 100644 --- a/man/chi_squared_test.Rd +++ b/man/chi_squared_test.Rd @@ -2,7 +2,7 @@ % Please edit documentation in R/chi_squared_test.R \name{chi_squared_test} \alias{chi_squared_test} -\title{Chi-Squared Test} +\title{Chi-Squared test} \usage{ chi_squared_test( data, @@ -10,6 +10,7 @@ chi_squared_test( by = NULL, probabilities = NULL, weights = NULL, + paired = FALSE, ... ) } @@ -32,6 +33,9 @@ must be \code{NULL}. The probabilities must sum to 1.} \item{weights}{Name of an (optional) weighting variable to be used for the test.} +\item{paired}{Logical, if \code{TRUE}, a McNemar test is conducted for 2x2 tables. +Note that \code{paired} only works for 2x2 tables.} + \item{...}{Additional arguments passed down to \code{\link[=chisq.test]{chisq.test()}}.} } \value{ @@ -53,7 +57,8 @@ The function is a wrapper around \code{\link[=chisq.test]{chisq.test()}} and \code{chisq.test()} for given probabilities. When \code{probabilities} are provided, these are rescaled to sum to 1 (i.e. \code{rescale.p = TRUE}). When \code{fisher.test()} is called, simulated p-values are returned (i.e. \code{simulate.p.value = TRUE}, -see \code{?fisher.test}). +see \code{?fisher.test}). If \code{paired = TRUE} and a 2x2 table is provided, +a McNemar test (see \code{\link[=mcnemar.test]{mcnemar.test()}}) is conducted. The weighted version of the chi-squared test is based on the a weighted table, using \code{\link[=xtabs]{xtabs()}} as input for \code{chisq.test()}. @@ -61,11 +66,11 @@ table, using \code{\link[=xtabs]{xtabs()}} as input for \code{chisq.test()}. \examples{ data(efc) efc$weight <- abs(rnorm(nrow(efc), 1, 0.3)) -# Chi-squared-test +# Chi-squared test chi_squared_test(efc, "c161sex", by = "e16sex") -# weighted Chi-squared-test +# weighted Chi-squared test chi_squared_test(efc, "c161sex", by = "e16sex", weights = "weight") -# Chi-squared-test for given probabilities +# Chi-squared test for given probabilities chi_squared_test(efc, "c161sex", probabilities = c(0.3, 0.7)) } \references{ diff --git a/man/kruskal_wallis_test.Rd b/man/kruskal_wallis_test.Rd index 7be4e2e..0b30814 100644 --- a/man/kruskal_wallis_test.Rd +++ b/man/kruskal_wallis_test.Rd @@ -2,9 +2,15 @@ % Please edit documentation in R/kruskal_wallis_test.R \name{kruskal_wallis_test} \alias{kruskal_wallis_test} -\title{Kruskal-Wallis-Test} +\title{Kruskal-Wallis test} \usage{ -kruskal_wallis_test(data, select = NULL, by = NULL, weights = NULL) +kruskal_wallis_test( + data, + select = NULL, + by = NULL, + weights = NULL, + paired = FALSE +) } \arguments{ \item{data}{A data frame.} @@ -17,6 +23,9 @@ not a factor, it will be coerced to a factor. For \code{chi_squared_test()}, if \code{probabilities} is provided, \code{by} must be \code{NULL}.} \item{weights}{Name of an (optional) weighting variable to be used for the test.} + +\item{paired}{Logical, if \code{TRUE}, a paired Friedman test is conducted (see +\code{\link[=friedman.test]{friedman.test()}}).} } \value{ A data frame with test results. @@ -24,15 +33,17 @@ A data frame with test results. \description{ This function performs a Kruskal-Wallis rank sum test, to test the null hypothesis that the population median of all of the groups are -equal. The alternative is that they differ in at least one. +equal. The alternative is that they differ in at least one. If \code{paired = TRUE}, +a paired Friedman test is conducted. } \details{ The function simply is a wrapper around \code{\link[=kruskal.test]{kruskal.test()}}. The weighted version of the Kruskal-Wallis test is based on the \code{survey} package, -using \code{\link[survey:svyranktest]{survey::svyranktest()}}. +using \code{\link[survey:svyranktest]{survey::svyranktest()}}. When \code{paired = TRUE}, a paired Friedman test +is conducted (see \code{\link[=friedman.test]{friedman.test()}}). } \examples{ data(efc) -# Kruskal-Wallis-Test for elder's age by education +# Kruskal-Wallis test for elder's age by education kruskal_wallis_test(efc, "e17age", by = "c172code") } From df5a5ee3860b9c9311443f179ae998cebb660f83 Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 17:57:10 +0200 Subject: [PATCH 29/34] update --- R/kruskal_wallis_test.R | 33 ++++++++++++++++++-------------- R/mann_whitney_test.R | 39 +++++++++++++++++++++++++++++--------- man/chi_squared_test.Rd | 5 ++++- man/kruskal_wallis_test.Rd | 22 +++++++-------------- man/mann_whitney_test.Rd | 5 ++++- 5 files changed, 64 insertions(+), 40 deletions(-) diff --git a/R/kruskal_wallis_test.R b/R/kruskal_wallis_test.R index dd0eebc..aca3e98 100644 --- a/R/kruskal_wallis_test.R +++ b/R/kruskal_wallis_test.R @@ -2,19 +2,15 @@ #' @name kruskal_wallis_test #' @description This function performs a Kruskal-Wallis rank sum test, to test #' the null hypothesis that the population median of all of the groups are -#' equal. The alternative is that they differ in at least one. If `paired = TRUE`, -#' a paired Friedman test is conducted. +#' equal. The alternative is that they differ in at least one. #' #' @inheritParams mann_whitney_test -#' @param paired Logical, if `TRUE`, a paired Friedman test is conducted (see -#' [`friedman.test()`]). #' #' @return A data frame with test results. #' #' @details The function simply is a wrapper around [`kruskal.test()`]. The #' weighted version of the Kruskal-Wallis test is based on the `survey` package, -#' using [`survey::svyranktest()`]. When `paired = TRUE`, a paired Friedman test -#' is conducted (see [`friedman.test()`]). +#' using [`survey::svyranktest()`]. #' #' @examples #' data(efc) @@ -24,13 +20,25 @@ kruskal_wallis_test <- function(data, select = NULL, by = NULL, - weights = NULL, - paired = FALSE) { + weights = NULL) { insight::check_if_installed("datawizard") # sanity checks .sanitize_htest_input(data, select, by, weights) + # does select indicate more than one variable? + if (length(select) > 1) { + if (!is.null(by)) { + insight::format_error("If `select` specifies more than one variable, `by` must be `NULL`.") + } + # we convert the data into long format, and create a grouping variable + data <- datawizard::data_to_long(data[select], names_to = "group", values_to = "scale") + by <- select[2] + select <- select[1] + # after converting to long, we have the "grouping" variable first in the data + colnames(data) <- c(by, select) + } + # get data dv <- data[[select]] grp <- data[[by]] @@ -43,9 +51,9 @@ kruskal_wallis_test <- function(data, insight::format_error("At least two groups are required, i.e. data must have at least two unique levels in `by` for `kruskal_wallis_test()`.") # nolint } if (is.null(weights)) { - .calculate_kw(dv, grp, paired) + .calculate_kw(dv, grp) } else { - .calculate_weighted_kw(dv, grp, data[[weights]], paired = TRUE) + .calculate_weighted_kw(dv, grp, data[[weights]]) } } @@ -101,10 +109,7 @@ kruskal_wallis_test <- function(data, }, numeric(1)) if (paired) { - tab <- as.table(round(stats::xtabs(x[[3]] ~ x[[1]] + x[[2]]))) - class(tab) <- "table" - # perfom friedman test for paired data - result <- stats::friedman.test(tab) + ## TODO: paired no working. should call `friedman.test()` } else { design <- survey::svydesign(ids = ~0, data = dat, weights = ~w) result <- survey::svyranktest(formula = x ~ g, design, test = "KruskalWallis") diff --git a/R/mann_whitney_test.R b/R/mann_whitney_test.R index d900bf2..1caf823 100644 --- a/R/mann_whitney_test.R +++ b/R/mann_whitney_test.R @@ -9,7 +9,10 @@ #' #' @param data A data frame. #' @param select Name of the dependent variable (as string) to be used for the -#' test. +#' test. `select` can also be a character vector, specifing the names of +#' multiple continuous variables. In this case, `by` is ignored and variables +#' specified in `select` are used to compute the test. This can be useful if +#' the data is in wide-format and no grouping variable is available. #' @param by Name of the grouping variable to be used for the test. If `by` is #' not a factor, it will be coerced to a factor. For `chi_squared_test()`, if #' `probabilities` is provided, `by` must be `NULL`. @@ -52,6 +55,23 @@ mann_whitney_test <- function(data, # sanity checks .sanitize_htest_input(data, select, by, weights) + # does select indicate more than one variable? + if (length(select) > 1) { + # sanity check - may only specify two variable names + if (length(select) > 2) { + insight::format_error("You may only specify two variables for Mann-Whitney test.") + } + if (!is.null(by)) { + insight::format_error("If `select` specifies more than one variable, `by` must be `NULL`.") + } + # we convert the data into long format, and create a grouping variable + data <- datawizard::data_to_long(data[select], names_to = "group", values_to = "scale") + by <- select[2] + select <- select[1] + # after converting to long, we have the "grouping" variable first in the data + colnames(data) <- c(by, select) + } + # get data dv <- data[[select]] grp <- data[[by]] @@ -61,7 +81,7 @@ mann_whitney_test <- function(data, # only two groups allowed if (insight::n_unique(grp) > 2) { - insight::format_error("Only two groups are allowed for Mann-Whitney-Test. Please use `kruskal_wallis_test()` for more than two groups.") # nolint + insight::format_error("Only two groups are allowed for Mann-Whitney test. Please use `kruskal_wallis_test()` for more than two groups.") # nolint } # value labels @@ -267,21 +287,22 @@ mann_whitney_test <- function(data, } # check if arguments have correct length (length of 1) - if (length(select) != 1 || !is.character(select)) { - insight::format_error("Argument `select` must be the name of a single variable.") + if (!is.character(select)) { + insight::format_error("Argument `select` must be a character string with the name(s) of the variable(s).") } if (length(by) != 1 || !is.character(by)) { - insight::format_error("Argument `by` must be the name of a single variable.") + insight::format_error("Argument `by` must be a character string with the name of a single variable.") } if (!is.null(weights) && length(weights) != 1) { - insight::format_error("Argument `weights` must be the name of a single variable.") + insight::format_error("Argument `weights` must be a character string with the name of a single variable.") } # check if "select" is in data - if (!select %in% colnames(data)) { + if (!all(select %in% colnames(data))) { + not_found <- setdiff(select, colnames(data))[1] insight::format_error( - sprintf("Variable '%s' not found in data frame.", select), - .misspelled_string(colnames(data), select, "Maybe misspelled?") + sprintf("Variable '%s' not found in data frame.", not_found), + .misspelled_string(colnames(data), not_found, "Maybe misspelled?") ) } # check if "by" is in data diff --git a/man/chi_squared_test.Rd b/man/chi_squared_test.Rd index 9bcfd73..71a0b15 100644 --- a/man/chi_squared_test.Rd +++ b/man/chi_squared_test.Rd @@ -18,7 +18,10 @@ chi_squared_test( \item{data}{A data frame.} \item{select}{Name of the dependent variable (as string) to be used for the -test.} +test. \code{select} can also be a character vector, specifing the names of +multiple continuous variables. In this case, \code{by} is ignored and variables +specified in \code{select} are used to compute the test. This can be useful if +the data is in wide-format and no grouping variable is available.} \item{by}{Name of the grouping variable to be used for the test. If \code{by} is not a factor, it will be coerced to a factor. For \code{chi_squared_test()}, if diff --git a/man/kruskal_wallis_test.Rd b/man/kruskal_wallis_test.Rd index 0b30814..6b8bf3f 100644 --- a/man/kruskal_wallis_test.Rd +++ b/man/kruskal_wallis_test.Rd @@ -4,28 +4,22 @@ \alias{kruskal_wallis_test} \title{Kruskal-Wallis test} \usage{ -kruskal_wallis_test( - data, - select = NULL, - by = NULL, - weights = NULL, - paired = FALSE -) +kruskal_wallis_test(data, select = NULL, by = NULL, weights = NULL) } \arguments{ \item{data}{A data frame.} \item{select}{Name of the dependent variable (as string) to be used for the -test.} +test. \code{select} can also be a character vector, specifing the names of +multiple continuous variables. In this case, \code{by} is ignored and variables +specified in \code{select} are used to compute the test. This can be useful if +the data is in wide-format and no grouping variable is available.} \item{by}{Name of the grouping variable to be used for the test. If \code{by} is not a factor, it will be coerced to a factor. For \code{chi_squared_test()}, if \code{probabilities} is provided, \code{by} must be \code{NULL}.} \item{weights}{Name of an (optional) weighting variable to be used for the test.} - -\item{paired}{Logical, if \code{TRUE}, a paired Friedman test is conducted (see -\code{\link[=friedman.test]{friedman.test()}}).} } \value{ A data frame with test results. @@ -33,14 +27,12 @@ A data frame with test results. \description{ This function performs a Kruskal-Wallis rank sum test, to test the null hypothesis that the population median of all of the groups are -equal. The alternative is that they differ in at least one. If \code{paired = TRUE}, -a paired Friedman test is conducted. +equal. The alternative is that they differ in at least one. } \details{ The function simply is a wrapper around \code{\link[=kruskal.test]{kruskal.test()}}. The weighted version of the Kruskal-Wallis test is based on the \code{survey} package, -using \code{\link[survey:svyranktest]{survey::svyranktest()}}. When \code{paired = TRUE}, a paired Friedman test -is conducted (see \code{\link[=friedman.test]{friedman.test()}}). +using \code{\link[survey:svyranktest]{survey::svyranktest()}}. } \examples{ data(efc) diff --git a/man/mann_whitney_test.Rd b/man/mann_whitney_test.Rd index 1401a04..9ab30e6 100644 --- a/man/mann_whitney_test.Rd +++ b/man/mann_whitney_test.Rd @@ -16,7 +16,10 @@ mann_whitney_test( \item{data}{A data frame.} \item{select}{Name of the dependent variable (as string) to be used for the -test.} +test. \code{select} can also be a character vector, specifing the names of +multiple continuous variables. In this case, \code{by} is ignored and variables +specified in \code{select} are used to compute the test. This can be useful if +the data is in wide-format and no grouping variable is available.} \item{by}{Name of the grouping variable to be used for the test. If \code{by} is not a factor, it will be coerced to a factor. For \code{chi_squared_test()}, if From 297780a5daf747c0792688354e01a1c062c244c2 Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 18:01:02 +0200 Subject: [PATCH 30/34] fixes --- R/mann_whitney_test.R | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/R/mann_whitney_test.R b/R/mann_whitney_test.R index 1caf823..0e11055 100644 --- a/R/mann_whitney_test.R +++ b/R/mann_whitney_test.R @@ -282,15 +282,16 @@ mann_whitney_test <- function(data, if (is.null(select)) { insight::format_error("Argument `select` is missing.") } - if (is.null(by)) { + # `by` is only allowed to be NULL if `select` specifies more than one variable + if (is.null(by) && length(select) == 1) { insight::format_error("Arguments `by` is missing.") } - # check if arguments have correct length (length of 1) + # check if arguments have correct length or are of correct type if (!is.character(select)) { insight::format_error("Argument `select` must be a character string with the name(s) of the variable(s).") } - if (length(by) != 1 || !is.character(by)) { + if (!is.null(by) && (length(by) != 1 || !is.character(by))) { insight::format_error("Argument `by` must be a character string with the name of a single variable.") } if (!is.null(weights) && length(weights) != 1) { @@ -306,7 +307,7 @@ mann_whitney_test <- function(data, ) } # check if "by" is in data - if (!by %in% colnames(data)) { + if (!is.null(by) && !by %in% colnames(data)) { insight::format_error( sprintf("Variable '%s' not found in data frame.", by), .misspelled_string(colnames(data), by, "Maybe misspelled?") From 836d04e7e607c9e4666b5ecc3bb4961fd1fecf4b Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 18:36:54 +0200 Subject: [PATCH 31/34] docs --- R/mann_whitney_test.R | 15 ++++++++++++++- man/mann_whitney_test.Rd | 13 +++++++++++++ 2 files changed, 27 insertions(+), 1 deletion(-) diff --git a/R/mann_whitney_test.R b/R/mann_whitney_test.R index 0e11055..30f52a3 100644 --- a/R/mann_whitney_test.R +++ b/R/mann_whitney_test.R @@ -44,6 +44,19 @@ #' data(efc) #' # Mann-Whitney-U-Tests for elder's age by elder's sex. #' mann_whitney_test(efc, "e17age", by = "e16sex") +#' +#' # when data is in wide-format, specify all relevant continuous +#' # variables in `select` and omit `by` +#' set.seed(123) +#' wide_data <- data.frame(scale1 = runif(20), scale2 = runif(20)) +#' mann_whitney_test(wide_data, select = c("scale1", "scale2")) +#' +#' # same as if we had data in long format, with grouping variable +#' long_data <- data.frame( +#' dv = c(wide_data$scale1, wide_data$scale12), +#' grp = rep(c("A", "B"), each = 20) +#' ) +#' mann_whitney_test(long_data, select = "dv", by = "grp") #' @export mann_whitney_test <- function(data, select = NULL, @@ -115,7 +128,7 @@ mann_whitney_test <- function(data, z <- as.numeric(coin::statistic(wt, type = "standardized")) p <- coin::pvalue(wt) r <- abs(z / sqrt(length(dv))) - w <- stats::wilcox.test(dv ~ grp, data = wcdat)$statistic + w <- suppressWarnings(stats::wilcox.test(dv ~ grp, data = wcdat)$statistic) # group means dat_gr1 <- stats::na.omit(dv[grp == group_levels[1]]) diff --git a/man/mann_whitney_test.Rd b/man/mann_whitney_test.Rd index 9ab30e6..6507dc4 100644 --- a/man/mann_whitney_test.Rd +++ b/man/mann_whitney_test.Rd @@ -64,4 +64,17 @@ Interpretation of the effect size \strong{r}, as a rule-of-thumb: data(efc) # Mann-Whitney-U-Tests for elder's age by elder's sex. mann_whitney_test(efc, "e17age", by = "e16sex") + +# when data is in wide-format, specify all relevant continuous +# variables in `select` and omit `by` +set.seed(123) +wide_data <- data.frame(scale1 = runif(20), scale2 = runif(20)) +mann_whitney_test(wide_data, select = c("scale1", "scale2")) + +# same as if we had data in long format, with grouping variable +long_data <- data.frame( + dv = c(wide_data$scale1, wide_data$scale12), + grp = rep(c("A", "B"), each = 20) +) +mann_whitney_test(long_data, select = "dv", by = "grp") } From c58942d7cd21dfe1383d0f852004438e6d0bf993 Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 18:42:11 +0200 Subject: [PATCH 32/34] docs --- R/chi_squared_test.R | 3 +++ R/kruskal_wallis_test.R | 17 +++++++++++++++++ R/mann_whitney_test.R | 6 +++--- man/chi_squared_test.Rd | 3 +++ man/kruskal_wallis_test.Rd | 17 +++++++++++++++++ man/mann_whitney_test.Rd | 6 +++--- 6 files changed, 46 insertions(+), 6 deletions(-) diff --git a/R/chi_squared_test.R b/R/chi_squared_test.R index f47b0a7..426883a 100644 --- a/R/chi_squared_test.R +++ b/R/chi_squared_test.R @@ -41,10 +41,13 @@ #' @examples #' data(efc) #' efc$weight <- abs(rnorm(nrow(efc), 1, 0.3)) +#' #' # Chi-squared test #' chi_squared_test(efc, "c161sex", by = "e16sex") +#' #' # weighted Chi-squared test #' chi_squared_test(efc, "c161sex", by = "e16sex", weights = "weight") +#' #' # Chi-squared test for given probabilities #' chi_squared_test(efc, "c161sex", probabilities = c(0.3, 0.7)) #' @export diff --git a/R/kruskal_wallis_test.R b/R/kruskal_wallis_test.R index aca3e98..8069aca 100644 --- a/R/kruskal_wallis_test.R +++ b/R/kruskal_wallis_test.R @@ -16,6 +16,23 @@ #' data(efc) #' # Kruskal-Wallis test for elder's age by education #' kruskal_wallis_test(efc, "e17age", by = "c172code") +#' +#' # when data is in wide-format, specify all relevant continuous +#' # variables in `select` and omit `by` +#' set.seed(123) +#' wide_data <- data.frame( +#' scale1 = runif(20), +#' scale2 = runif(20), +#' scale3 = runif(20) +#' ) +#' kruskal_wallis_test(wide_data, select = c("scale1", "scale2", "scale3")) +#' +#' # same as if we had data in long format, with grouping variable +#' long_data <- data.frame( +#' scales = c(wide_data$scale1, wide_data$scale2, wide_data$scale3), +#' groups = rep(c("A", "B", "C"), each = 20) +#' ) +#' kruskal_wallis_test(long_data, select = "scales", by = "groups") #' @export kruskal_wallis_test <- function(data, select = NULL, diff --git a/R/mann_whitney_test.R b/R/mann_whitney_test.R index 30f52a3..f7bed50 100644 --- a/R/mann_whitney_test.R +++ b/R/mann_whitney_test.R @@ -53,10 +53,10 @@ #' #' # same as if we had data in long format, with grouping variable #' long_data <- data.frame( -#' dv = c(wide_data$scale1, wide_data$scale12), -#' grp = rep(c("A", "B"), each = 20) +#' scales = c(wide_data$scale1, wide_data$scale2), +#' groups = rep(c("A", "B"), each = 20) #' ) -#' mann_whitney_test(long_data, select = "dv", by = "grp") +#' mann_whitney_test(long_data, select = "scales", by = "groups") #' @export mann_whitney_test <- function(data, select = NULL, diff --git a/man/chi_squared_test.Rd b/man/chi_squared_test.Rd index 71a0b15..10abb7d 100644 --- a/man/chi_squared_test.Rd +++ b/man/chi_squared_test.Rd @@ -69,10 +69,13 @@ table, using \code{\link[=xtabs]{xtabs()}} as input for \code{chisq.test()}. \examples{ data(efc) efc$weight <- abs(rnorm(nrow(efc), 1, 0.3)) + # Chi-squared test chi_squared_test(efc, "c161sex", by = "e16sex") + # weighted Chi-squared test chi_squared_test(efc, "c161sex", by = "e16sex", weights = "weight") + # Chi-squared test for given probabilities chi_squared_test(efc, "c161sex", probabilities = c(0.3, 0.7)) } diff --git a/man/kruskal_wallis_test.Rd b/man/kruskal_wallis_test.Rd index 6b8bf3f..a9b7b5e 100644 --- a/man/kruskal_wallis_test.Rd +++ b/man/kruskal_wallis_test.Rd @@ -38,4 +38,21 @@ using \code{\link[survey:svyranktest]{survey::svyranktest()}}. data(efc) # Kruskal-Wallis test for elder's age by education kruskal_wallis_test(efc, "e17age", by = "c172code") + +# when data is in wide-format, specify all relevant continuous +# variables in `select` and omit `by` +set.seed(123) +wide_data <- data.frame( + scale1 = runif(20), + scale2 = runif(20), + scale3 = runif(20) +) +kruskal_wallis_test(wide_data, select = c("scale1", "scale2", "scale3")) + +# same as if we had data in long format, with grouping variable +long_data <- data.frame( + scales = c(wide_data$scale1, wide_data$scale2, wide_data$scale3), + groups = rep(c("A", "B", "C"), each = 20) +) +kruskal_wallis_test(long_data, select = "scales", by = "groups") } diff --git a/man/mann_whitney_test.Rd b/man/mann_whitney_test.Rd index 6507dc4..9cafad6 100644 --- a/man/mann_whitney_test.Rd +++ b/man/mann_whitney_test.Rd @@ -73,8 +73,8 @@ mann_whitney_test(wide_data, select = c("scale1", "scale2")) # same as if we had data in long format, with grouping variable long_data <- data.frame( - dv = c(wide_data$scale1, wide_data$scale12), - grp = rep(c("A", "B"), each = 20) + scales = c(wide_data$scale1, wide_data$scale2), + groups = rep(c("A", "B"), each = 20) ) -mann_whitney_test(long_data, select = "dv", by = "grp") +mann_whitney_test(long_data, select = "scales", by = "groups") } From 61688e03fcb3d7269d9e92bd94c3d125be4612a2 Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 18:46:59 +0200 Subject: [PATCH 33/34] docs --- R/kruskal_wallis_test.R | 2 +- man/kruskal_wallis_test.Rd | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/R/kruskal_wallis_test.R b/R/kruskal_wallis_test.R index 8069aca..a66e3c1 100644 --- a/R/kruskal_wallis_test.R +++ b/R/kruskal_wallis_test.R @@ -9,7 +9,7 @@ #' @return A data frame with test results. #' #' @details The function simply is a wrapper around [`kruskal.test()`]. The -#' weighted version of the Kruskal-Wallis test is based on the `survey` package, +#' weighted version of the Kruskal-Wallis test is based on the **survey** package, #' using [`survey::svyranktest()`]. #' #' @examples diff --git a/man/kruskal_wallis_test.Rd b/man/kruskal_wallis_test.Rd index a9b7b5e..671a279 100644 --- a/man/kruskal_wallis_test.Rd +++ b/man/kruskal_wallis_test.Rd @@ -31,7 +31,7 @@ equal. The alternative is that they differ in at least one. } \details{ The function simply is a wrapper around \code{\link[=kruskal.test]{kruskal.test()}}. The -weighted version of the Kruskal-Wallis test is based on the \code{survey} package, +weighted version of the Kruskal-Wallis test is based on the \strong{survey} package, using \code{\link[survey:svyranktest]{survey::svyranktest()}}. } \examples{ From 44e998e313ea09c75c887ce3059613b3e4ec6885 Mon Sep 17 00:00:00 2001 From: Daniel Date: Thu, 9 May 2024 18:49:31 +0200 Subject: [PATCH 34/34] docs --- R/mann_whitney_test.R | 2 +- man/chi_squared_test.Rd | 2 +- man/kruskal_wallis_test.Rd | 2 +- man/mann_whitney_test.Rd | 2 +- 4 files changed, 4 insertions(+), 4 deletions(-) diff --git a/R/mann_whitney_test.R b/R/mann_whitney_test.R index f7bed50..d16f178 100644 --- a/R/mann_whitney_test.R +++ b/R/mann_whitney_test.R @@ -9,7 +9,7 @@ #' #' @param data A data frame. #' @param select Name of the dependent variable (as string) to be used for the -#' test. `select` can also be a character vector, specifing the names of +#' test. `select` can also be a character vector, specifying the names of #' multiple continuous variables. In this case, `by` is ignored and variables #' specified in `select` are used to compute the test. This can be useful if #' the data is in wide-format and no grouping variable is available. diff --git a/man/chi_squared_test.Rd b/man/chi_squared_test.Rd index 10abb7d..f6dd5d7 100644 --- a/man/chi_squared_test.Rd +++ b/man/chi_squared_test.Rd @@ -18,7 +18,7 @@ chi_squared_test( \item{data}{A data frame.} \item{select}{Name of the dependent variable (as string) to be used for the -test. \code{select} can also be a character vector, specifing the names of +test. \code{select} can also be a character vector, specifying the names of multiple continuous variables. In this case, \code{by} is ignored and variables specified in \code{select} are used to compute the test. This can be useful if the data is in wide-format and no grouping variable is available.} diff --git a/man/kruskal_wallis_test.Rd b/man/kruskal_wallis_test.Rd index 671a279..d3033c4 100644 --- a/man/kruskal_wallis_test.Rd +++ b/man/kruskal_wallis_test.Rd @@ -10,7 +10,7 @@ kruskal_wallis_test(data, select = NULL, by = NULL, weights = NULL) \item{data}{A data frame.} \item{select}{Name of the dependent variable (as string) to be used for the -test. \code{select} can also be a character vector, specifing the names of +test. \code{select} can also be a character vector, specifying the names of multiple continuous variables. In this case, \code{by} is ignored and variables specified in \code{select} are used to compute the test. This can be useful if the data is in wide-format and no grouping variable is available.} diff --git a/man/mann_whitney_test.Rd b/man/mann_whitney_test.Rd index 9cafad6..ab376af 100644 --- a/man/mann_whitney_test.Rd +++ b/man/mann_whitney_test.Rd @@ -16,7 +16,7 @@ mann_whitney_test( \item{data}{A data frame.} \item{select}{Name of the dependent variable (as string) to be used for the -test. \code{select} can also be a character vector, specifing the names of +test. \code{select} can also be a character vector, specifying the names of multiple continuous variables. In this case, \code{by} is ignored and variables specified in \code{select} are used to compute the test. This can be useful if the data is in wide-format and no grouping variable is available.}