2018-08-10 15:01:05 +02:00
# ==================================================================== #
# TITLE #
# Antimicrobial Resistance (AMR) Analysis #
# #
2019-01-02 23:24:07 +01:00
# SOURCE #
2020-07-08 14:48:06 +02:00
# https://github.com/msberends/AMR #
2018-08-10 15:01:05 +02:00
# #
# LICENCE #
2020-01-05 17:22:09 +01:00
# (c) 2018-2020 Berends MS, Luz CF et al. #
2018-08-10 15:01:05 +02:00
# #
2019-01-02 23:24:07 +01:00
# This R package is free software; you can freely use and distribute #
# it for both personal and commercial purposes under the terms of the #
# GNU General Public License version 2.0 (GNU GPL-2), as published by #
# the Free Software Foundation. #
# #
2020-01-05 17:22:09 +01:00
# We created this package for both routine data analysis and academic #
# research and it was publicly released in the hope that it will be #
# useful, but it comes WITHOUT ANY WARRANTY OR LIABILITY. #
2020-07-08 14:48:06 +02:00
# Visit our website for more info: https://msberends.github.io/AMR. #
2018-08-10 15:01:05 +02:00
# ==================================================================== #
2018-08-25 22:01:14 +02:00
#' Class 'rsi'
2018-08-10 15:01:05 +02:00
#'
2020-08-10 11:44:58 +02:00
#' Interpret minimum inhibitory concentration (MIC) values and disk diffusion diameters according to EUCAST or CLSI, or clean up existing R/SI values. This transforms the input to a new class [`rsi`], which is an ordered factor with levels `S < I < R`. Values that cannot be interpreted will be returned as `NA` with a warning.
2020-01-05 17:22:09 +01:00
#' @inheritSection lifecycle Stable lifecycle
2018-08-25 22:01:14 +02:00
#' @rdname as.rsi
2020-02-17 14:38:01 +01:00
#' @param x vector of values (for class [`mic`]: an MIC value in mg/L, for class [`disk`]: a disk diffusion radius in millimetres)
2020-08-10 11:44:58 +02:00
#' @param mo any (vector of) text that can be coerced to a valid microorganism code with [as.mo()], will be determined automatically if the `dplyr` package is installed
2020-02-17 14:38:01 +01:00
#' @param ab any (vector of) text that can be coerced to a valid antimicrobial code with [as.ab()]
2020-02-20 13:19:23 +01:00
#' @param uti (Urinary Tract Infection) A vector with [logical]s (`TRUE` or `FALSE`) to specify whether a UTI specific interpretation from the guideline should be chosen. For using [as.rsi()] on a [data.frame], this can also be a column containing [logical]s or when left blank, the data set will be search for a 'specimen' and rows containing 'urin' in that column will be regarded isolates from a UTI. See *Examples*.
2019-05-10 16:44:59 +02:00
#' @inheritParams first_isolate
2020-05-16 13:05:47 +02:00
#' @param guideline defaults to the latest included EUCAST guideline, see Details for all options
2020-07-29 11:46:59 +02:00
#' @param conserve_capped_values a logical to indicate that MIC values starting with `">"` (but not `">="`) must always return "R" , and that MIC values starting with `"<"` (but not `"<="`) must always return "S"
2019-11-28 22:32:17 +01:00
#' @param threshold maximum fraction of invalid antimicrobial interpretations of `x`, please see *Examples*
2019-05-10 16:44:59 +02:00
#' @param ... parameters passed on to methods
2020-05-16 13:05:47 +02:00
#' @details
2020-08-10 11:44:58 +02:00
#' ## How it works
2020-01-27 11:05:39 +01:00
#'
2020-08-10 11:44:58 +02:00
#' The [as.rsi()] function works in four ways:
2020-05-16 13:05:47 +02:00
#'
2020-08-10 11:44:58 +02:00
#' 1. For **cleaning raw / untransformed data**. The data will be cleaned to only contain values S, I and R and will try its best to determine this with some intelligence. For example, mixed values with R/SI interpretations and MIC values such as `"<0.25; S"` will be coerced to `"S"`. Combined interpretations for multiple test methods (as seen in laboratory records) such as `"S; S"` will be coerced to `"S"`, but a value like `"S; I"` will return `NA` with a warning that the input is unclear.
#'
#' 2. For **interpreting minimum inhibitory concentration (MIC) values** according to EUCAST or CLSI. You must clean your MIC values first using [as.mic()], that also gives your columns the new data class [`mic`]. Also, be sure to have a column with microorganism names or codes. It will be found automatically, but can be set manually using the `mo` parameter.
#' * Using `dplyr`, R/SI interpretation can be done very easily with either:
#' ```
#' your_data %>% mutate_if(is.mic, as.rsi) # until dplyr 1.0.0
#' your_data %>% mutate(across(where(is.mic), as.rsi)) # since dplyr 1.0.0
#' ```
#' * Operators like "<=" will be stripped before interpretation. When using `conserve_capped_values = TRUE`, an MIC value of e.g. ">2" will always return "R", even if the breakpoint according to the chosen guideline is ">=4". This is to prevent that capped values from raw laboratory data would not be treated conservatively. The default behaviour (`conserve_capped_values = FALSE`) considers ">2" to be lower than ">=4" and might in this case return "S" or "I".
#'
#' 3. For **interpreting disk diffusion diameters** according to EUCAST or CLSI. You must clean your disk zones first using [as.disk()], that also gives your columns the new data class [`disk`]. Also, be sure to have a column with microorganism names or codes. It will be found automatically, but can be set manually using the `mo` parameter.
#' * Using `dplyr`, R/SI interpretation can be done very easily with either:
#' ```
#' your_data %>% mutate_if(is.disk, as.rsi) # until dplyr 1.0.0
#' your_data %>% mutate(across(where(is.disk), as.rsi)) # since dplyr 1.0.0
#' ```
#'
#' 4. For **interpreting a complete data set**, with automatic determination of MIC values, disk diffusion diameters, microorganism names or codes, and antimicrobial test results. This is done very simply by running `as.rsi(data)`.
#'
#' ## Supported guidelines
#'
#' For interpreting MIC values as well as disk diffusion diameters, supported guidelines to be used as input for the `guideline` parameter are: `r paste0('"', sort(unique(AMR::rsi_translation$guideline)), '"', collapse = ", ")`.
#'
#' Simply using `"CLSI"` or `"EUCAST"` as input will automatically select the latest version of that guideline.
#'
#' ## After interpretation
#'
#' After using [as.rsi()], you can use the [eucast_rules()] defined by EUCAST to (1) apply inferred susceptibility and resistance based on results of other antimicrobials and (2) apply intrinsic resistance based on taxonomic properties of a microorganism.
#'
#' ## Machine readable interpretation guidelines
2020-07-29 11:46:59 +02:00
#'
2020-07-29 10:33:47 +02:00
#' The repository of this package [contains a machine readable version](https://github.com/msberends/AMR/blob/master/data-raw/rsi_translation.txt) of all guidelines. This is a CSV file consisting of `r format(nrow(AMR::rsi_translation), big.mark = ",")` rows and `r ncol(AMR::rsi_translation)` columns. This file is machine readable, since it contains one row for every unique combination of the test method (MIC or disk diffusion), the antimicrobial agent and the microorganism. **This allows for easy implementation of these rules in laboratory information systems (LIS)**. Note that it only contains interpretation guidelines for humans - interpretation guidelines from CLSI for animals were removed.
2019-04-09 14:59:17 +02:00
#'
2020-08-10 11:44:58 +02:00
#' ## Other
2019-05-10 16:44:59 +02:00
#'
2019-11-28 22:32:17 +01:00
#' The function [is.rsi.eligible()] returns `TRUE` when a columns contains at most 5% invalid antimicrobial interpretations (not S and/or I and/or R), and `FALSE` otherwise. The threshold of 5% can be set with the `threshold` parameter.
2019-11-29 19:43:23 +01:00
#' @section Interpretation of R and S/I:
2020-02-16 22:43:56 +01:00
#' In 2019, the European Committee on Antimicrobial Susceptibility Testing (EUCAST) has decided to change the definitions of susceptibility testing categories R and S/I as shown below (<http://www.eucast.org/newsiandr/>).
2019-05-13 10:10:16 +02:00
#'
2019-11-29 19:43:23 +01:00
#' - **R = Resistant**\cr
#' A microorganism is categorised as *Resistant* when there is a high likelihood of therapeutic failure even when there is increased exposure. Exposure is a function of how the mode of administration, dose, dosing interval, infusion time, as well as distribution and excretion of the antimicrobial agent will influence the infecting organism at the site of infection.
#' - **S = Susceptible**\cr
#' A microorganism is categorised as *Susceptible, standard dosing regimen*, when there is a high likelihood of therapeutic success using a standard dosing regimen of the agent.
#' - **I = Increased exposure, but still susceptible**\cr
#' A microorganism is categorised as *Susceptible, Increased exposure* when there is a high likelihood of therapeutic success because exposure to the agent is increased by adjusting the dosing regimen or by its concentration at the site of infection.
2019-05-13 10:10:16 +02:00
#'
2019-11-28 22:32:17 +01:00
#' This AMR package honours this new insight. Use [susceptibility()] (equal to [proportion_SI()]) to determine antimicrobial susceptibility and [count_susceptible()] (equal to [count_SI()]) to count susceptible isolates.
#' @return Ordered factor with new class [`rsi`]
2019-11-28 23:00:37 +01:00
#' @aliases rsi
2018-08-10 15:01:05 +02:00
#' @export
2020-08-10 11:44:58 +02:00
#' @seealso [as.mic()], [as.disk()], [as.mo()]
2019-01-02 23:24:07 +01:00
#' @inheritSection AMR Read more on our website!
2018-08-25 22:01:14 +02:00
#' @examples
2020-06-22 13:18:40 +02:00
#' summary(example_isolates) # see all R/SI results at a glance
#'
2020-02-16 22:43:56 +01:00
#' # For INTERPRETING disk diffusion and MIC values -----------------------
2020-02-20 13:19:23 +01:00
#'
#' # a whole data set, even with combined MIC values and disk zones
#' df <- data.frame(microorganism = "E. coli",
#' AMP = as.mic(8),
#' CIP = as.mic(0.256),
#' GEN = as.disk(18),
#' TOB = as.disk(16),
#' NIT = as.mic(32))
#' as.rsi(df)
2020-02-16 22:43:56 +01:00
#'
2020-05-16 21:40:50 +02:00
#' \dontrun{
2020-02-21 13:13:34 +01:00
#'
2020-02-20 13:19:23 +01:00
#' # the dplyr way
#' library(dplyr)
2020-08-10 11:44:58 +02:00
#' df %>% mutate_at(vars(AMP:TOB), as.rsi)
#' df %>% mutate(across(AMP:TOB), as.rsi)
2020-08-10 12:46:03 +02:00
#'
2020-02-20 13:19:23 +01:00
#' df %>%
#' mutate_at(vars(AMP:TOB), as.rsi, mo = "E. coli")
#'
#' # to include information about urinary tract infections (UTI)
#' data.frame(mo = "E. coli",
#' NIT = c("<= 2", 32),
#' from_the_bladder = c(TRUE, FALSE)) %>%
#' as.rsi(uti = "from_the_bladder")
#'
#' data.frame(mo = "E. coli",
#' NIT = c("<= 2", 32),
#' specimen = c("urine", "blood")) %>%
#' as.rsi() # automatically determines urine isolates
#'
#' df %>%
2020-02-20 17:21:01 +01:00
#' mutate_at(vars(AMP:NIT), as.rsi, mo = "E. coli", uti = TRUE)
2020-02-21 13:13:34 +01:00
#' }
#'
2020-02-20 13:19:23 +01:00
#' # for single values
2019-05-10 16:44:59 +02:00
#' as.rsi(x = as.mic(2),
#' mo = as.mo("S. pneumoniae"),
2020-02-14 19:54:13 +01:00
#' ab = "AMP",
2019-05-10 16:44:59 +02:00
#' guideline = "EUCAST")
2020-02-14 19:54:13 +01:00
#'
#' as.rsi(x = as.disk(18),
#' mo = "Strep pneu", # `mo` will be coerced with as.mo()
#' ab = "ampicillin", # and `ab` with as.ab()
2019-05-10 16:44:59 +02:00
#' guideline = "EUCAST")
2020-02-16 22:43:56 +01:00
#'
2019-05-10 16:44:59 +02:00
#'
2020-02-16 22:43:56 +01:00
#' # For CLEANING existing R/SI values ------------------------------------
#'
#' as.rsi(c("S", "I", "R", "A", "B", "C"))
#' as.rsi("<= 0.002; S") # will return "S"
#'
#' rsi_data <- as.rsi(c(rep("S", 474), rep("I", 36), rep("R", 370)))
#' is.rsi(rsi_data)
2018-08-25 22:01:14 +02:00
#' plot(rsi_data) # for percentages
#' barplot(rsi_data) # for frequencies
#'
2020-05-16 21:40:50 +02:00
#' \dontrun{
2018-08-25 22:01:14 +02:00
#' library(dplyr)
2019-08-27 16:45:42 +02:00
#' example_isolates %>%
2019-05-10 16:44:59 +02:00
#' mutate_at(vars(PEN:RIF), as.rsi)
2018-12-07 12:04:55 +01:00
#'
2020-02-16 22:43:56 +01:00
#' # fastest way to transform all columns with already valid AMR results to class `rsi`:
2019-08-27 16:45:42 +02:00
#' example_isolates %>%
2020-02-16 22:43:56 +01:00
#' mutate_if(is.rsi.eligible, as.rsi)
#'
#' # note: from dplyr 1.0.0 on, this will be:
#' # example_isolates %>%
#' # mutate(across(is.rsi.eligible, as.rsi))
2019-02-04 12:24:07 +01:00
#'
#' # default threshold of `is.rsi.eligible` is 5%.
#' is.rsi.eligible(WHONET$`First name`) # fails, >80% is invalid
2019-05-10 16:44:59 +02:00
#' is.rsi.eligible(WHONET$`First name`, threshold = 0.99) # succeeds
2020-05-16 21:40:50 +02:00
#' }
2019-05-10 16:44:59 +02:00
as.rsi <- function ( x , ... ) {
UseMethod ( " as.rsi" )
}
2020-07-29 13:48:50 +02:00
#' @rdname as.rsi
#' @export
is.rsi <- function ( x ) {
inherits ( x , " rsi" )
}
#' @rdname as.rsi
#' @export
is.rsi.eligible <- function ( x , threshold = 0.05 ) {
stop_if ( NCOL ( x ) > 1 , " `x` must be a one-dimensional vector." )
if ( any ( c ( " logical" ,
" numeric" ,
" integer" ,
" mo" ,
" Date" ,
" POSIXct" ,
" rsi" ,
" raw" ,
" hms" )
%in% class ( x ) ) ) {
# no transformation needed
FALSE
} else {
x <- x [ ! is.na ( x ) & ! is.null ( x ) & ! identical ( x , " " ) ]
if ( length ( x ) == 0 ) {
return ( FALSE )
}
checked <- suppressWarnings ( as.rsi ( x ) )
outcome <- sum ( is.na ( checked ) ) / length ( x )
outcome <= threshold
}
}
2019-05-10 16:44:59 +02:00
#' @export
as.rsi.default <- function ( x , ... ) {
2018-08-25 22:01:14 +02:00
if ( is.rsi ( x ) ) {
x
2018-12-07 12:04:55 +01:00
} else if ( identical ( levels ( x ) , c ( " S" , " I" , " R" ) ) ) {
2019-10-11 17:21:02 +02:00
structure ( x , class = c ( " rsi" , " ordered" , " factor" ) )
2020-02-20 17:21:01 +01:00
} else if ( inherits ( x , " integer" ) & all ( x %in% c ( 1 : 3 , NA ) ) ) {
2019-11-05 11:28:52 +01:00
x [x == 1 ] <- " S"
x [x == 2 ] <- " I"
x [x == 3 ] <- " R"
structure ( .Data = factor ( x , levels = c ( " S" , " I" , " R" ) , ordered = TRUE ) ,
class = c ( " rsi" , " ordered" , " factor" ) )
2018-08-23 00:40:36 +02:00
} else {
2020-07-13 09:17:24 +02:00
2020-02-20 17:21:01 +01:00
ab <- deparse ( substitute ( x ) )
if ( ! any ( x %like% " (R|S|I)" , na.rm = TRUE ) ) {
if ( ! is.na ( suppressWarnings ( as.ab ( ab ) ) ) ) {
# check if they are actually MICs or disks now that the antibiotic name is valid
if ( all_valid_mics ( x ) ) {
as.rsi ( as.mic ( x ) , ab = ab , ... )
} else if ( all_valid_disks ( x ) ) {
as.rsi ( as.disk ( x ) , ab = ab , ... )
}
}
}
2019-10-23 14:48:25 +02:00
2020-06-26 10:21:22 +02:00
x <- as.character ( unlist ( x ) )
2018-08-25 22:01:14 +02:00
x.bak <- x
2019-10-23 14:48:25 +02:00
2020-06-26 10:21:22 +02:00
na_before <- length ( x [is.na ( x ) | x == " " ] )
2018-08-25 22:01:14 +02:00
# remove all spaces
2019-10-11 17:21:02 +02:00
x <- gsub ( " +" , " " , x )
2018-08-25 22:01:14 +02:00
# remove all MIC-like values: numbers, operators and periods
2019-10-11 17:21:02 +02:00
x <- gsub ( " [0-9.,;:<=>]+" , " " , x )
2019-03-02 22:47:04 +01:00
# remove everything between brackets, and 'high' and 'low'
x <- gsub ( " ([(].*[)])" , " " , x )
x <- gsub ( " (high|low)" , " " , x , ignore.case = TRUE )
2018-08-25 22:01:14 +02:00
# disallow more than 3 characters
x [nchar ( x ) > 3 ] <- NA
# set to capitals
x <- toupper ( x )
# remove all invalid characters
2019-10-11 17:21:02 +02:00
x <- gsub ( " [^RSI]+" , " " , x )
2018-08-25 22:01:14 +02:00
# in cases of "S;S" keep S, but in case of "S;I" make it NA
2019-10-11 17:21:02 +02:00
x <- gsub ( " ^S+$" , " S" , x )
x <- gsub ( " ^I+$" , " I" , x )
x <- gsub ( " ^R+$" , " R" , x )
x [ ! x %in% c ( " S" , " I" , " R" ) ] <- NA
2020-06-26 10:21:22 +02:00
na_after <- length ( x [is.na ( x ) | x == " " ] )
2019-08-09 14:28:46 +02:00
if ( ! isFALSE ( list ( ... ) $ warn ) ) { # so as.rsi(..., warn = FALSE) will never throw a warning
if ( na_before != na_after ) {
2019-10-11 17:21:02 +02:00
list_missing <- x.bak [is.na ( x ) & ! is.na ( x.bak ) & x.bak != " " ] %>%
2019-08-09 14:28:46 +02:00
unique ( ) %>%
sort ( )
2019-10-11 17:21:02 +02:00
list_missing <- paste0 ( ' "' , list_missing , ' "' , collapse = " , " )
warning ( na_after - na_before , " results truncated (" ,
2019-08-09 14:28:46 +02:00
round ( ( ( na_after - na_before ) / length ( x ) ) * 100 ) ,
2019-10-11 17:21:02 +02:00
" %) that were invalid antimicrobial interpretations: " ,
2019-08-09 14:28:46 +02:00
list_missing , call. = FALSE )
}
2018-08-25 22:01:14 +02:00
}
2019-08-09 14:28:46 +02:00
2019-05-10 16:44:59 +02:00
structure ( .Data = factor ( x , levels = c ( " S" , " I" , " R" ) , ordered = TRUE ) ,
2019-10-11 17:21:02 +02:00
class = c ( " rsi" , " ordered" , " factor" ) )
2018-08-23 00:40:36 +02:00
}
2018-08-25 22:01:14 +02:00
}
2018-08-23 00:40:36 +02:00
2019-05-10 16:44:59 +02:00
#' @rdname as.rsi
#' @export
2020-07-29 11:46:59 +02:00
as.rsi.mic <- function ( x ,
2020-08-10 11:44:58 +02:00
mo = NULL ,
2020-07-29 11:46:59 +02:00
ab = deparse ( substitute ( x ) ) ,
guideline = " EUCAST" ,
uti = FALSE ,
conserve_capped_values = FALSE ,
... ) {
2020-08-10 11:44:58 +02:00
# for dplyr's across()
cur_column_dplyr <- import_fn ( " cur_column" , " dplyr" , error_on_fail = FALSE )
if ( ! is.null ( cur_column_dplyr ) ) {
# try to get current column, which will only be available when in across()
ab <- tryCatch ( cur_column_dplyr ( ) ,
error = function ( e ) ab )
}
# for auto-determining mo
mo_var_found <- " "
if ( is.null ( mo ) ) {
peek_mask_dplyr <- import_fn ( " peek_mask" , " dplyr" , error_on_fail = FALSE )
if ( ! is.null ( peek_mask_dplyr ) ) {
try ( {
df <- as.data.frame ( peek_mask_dplyr ( ) $ across_cols ( ) , stringsAsFactors = FALSE )
mo <- suppressMessages ( search_type_in_df ( df , " mo" ) )
if ( ! is.null ( mo ) ) {
mo_var_found <- paste0 ( " based on column `" , font_bold ( mo ) , " `" )
mo <- df [ , mo , drop = TRUE ]
}
} , silent = TRUE )
}
}
if ( is.null ( mo ) ) {
stop_ ( ' No information was supplied about the microorganisms (missing parameter "mo"). See ?as.rsi.\n\n' ,
2020-06-22 11:18:40 +02:00
" To transform certain columns with e.g. mutate_at(), use\n" ,
" `data %>% mutate_at(vars(...), as.rsi, mo = .$x)`, where x is your column with microorganisms.\n\n" ,
" To tranform all MIC variables in a data set, use `as.rsi(data)` or `data %>% as.rsi()`." , call = FALSE )
2020-08-10 11:44:58 +02:00
}
2020-02-20 13:19:23 +01:00
ab_coerced <- suppressWarnings ( as.ab ( ab ) )
mo_coerced <- suppressWarnings ( as.mo ( mo ) )
guideline_coerced <- get_guideline ( guideline )
if ( is.na ( ab_coerced ) ) {
2020-05-16 13:05:47 +02:00
message ( font_red ( paste0 ( " Unknown drug: `" , font_bold ( ab ) , " `. Rename this column to a drug name or code, and check the output with as.ab()." ) ) )
2020-02-20 13:19:23 +01:00
return ( as.rsi ( rep ( NA , length ( x ) ) ) )
}
if ( length ( mo_coerced ) == 1 ) {
mo_coerced <- rep ( mo_coerced , length ( x ) )
}
if ( length ( uti ) == 1 ) {
uti <- rep ( uti , length ( x ) )
}
2020-05-16 13:05:47 +02:00
message ( font_blue ( paste0 ( " => Interpreting MIC values of `" , font_bold ( ab ) , " ` (" ,
2020-07-13 09:17:24 +02:00
ifelse ( ab_coerced != ab , paste0 ( ab_coerced , " , " ) , " " ) ,
2020-08-10 11:44:58 +02:00
ab_name ( ab_coerced , tolower = TRUE ) , " )" , mo_var_found ,
" according to " , font_bold ( guideline_coerced ) , " ... " ) ) ,
2020-02-20 13:19:23 +01:00
appendLF = FALSE )
result <- exec_as.rsi ( method = " mic" ,
x = x ,
mo = mo_coerced ,
ab = ab_coerced ,
guideline = guideline_coerced ,
2020-07-29 11:46:59 +02:00
uti = uti ,
conserve_capped_values = conserve_capped_values ) # exec_as.rsi will return message(font_blue(" OK."))
2020-02-20 13:19:23 +01:00
result
2019-05-10 16:44:59 +02:00
}
#' @rdname as.rsi
#' @export
2020-07-29 11:46:59 +02:00
as.rsi.disk <- function ( x ,
2020-08-10 11:44:58 +02:00
mo = NULL ,
2020-07-29 11:46:59 +02:00
ab = deparse ( substitute ( x ) ) ,
guideline = " EUCAST" ,
uti = FALSE ,
... ) {
2020-08-10 11:44:58 +02:00
# for dplyr's across()
cur_column_dplyr <- import_fn ( " cur_column" , " dplyr" , error_on_fail = FALSE )
if ( ! is.null ( cur_column_dplyr ) ) {
# try to get current column, which will only be available when in across()
ab <- tryCatch ( cur_column_dplyr ( ) ,
error = function ( e ) ab )
}
# for auto-determining mo
mo_var_found <- " "
if ( is.null ( mo ) ) {
peek_mask_dplyr <- import_fn ( " peek_mask" , " dplyr" , error_on_fail = FALSE )
if ( ! is.null ( peek_mask_dplyr ) ) {
try ( {
df <- as.data.frame ( peek_mask_dplyr ( ) $ across_cols ( ) , stringsAsFactors = FALSE )
mo <- suppressMessages ( search_type_in_df ( df , " mo" ) )
if ( ! is.null ( mo ) ) {
mo_var_found <- paste0 ( " based on column `" , font_bold ( mo ) , " `" )
mo <- df [ , mo , drop = TRUE ]
}
} , silent = TRUE )
}
}
if ( is.null ( mo ) ) {
stop_ ( ' No information was supplied about the microorganisms (missing parameter "mo"). See ?as.rsi.\n\n' ,
2020-06-22 11:18:40 +02:00
" To transform certain columns with e.g. mutate_at(), use\n" ,
" `data %>% mutate_at(vars(...), as.rsi, mo = .$x)`, where x is your column with microorganisms.\n\n" ,
" To tranform all disk diffusion zones in a data set, use `as.rsi(data)` or `data %>% as.rsi()`." , call = FALSE )
2020-08-10 11:44:58 +02:00
}
2020-02-20 13:19:23 +01:00
ab_coerced <- suppressWarnings ( as.ab ( ab ) )
mo_coerced <- suppressWarnings ( as.mo ( mo ) )
guideline_coerced <- get_guideline ( guideline )
if ( is.na ( ab_coerced ) ) {
2020-05-16 13:05:47 +02:00
message ( font_red ( paste0 ( " Unknown drug: `" , font_bold ( ab ) , " `. Rename this column to a drug name or code, and check the output with as.ab()." ) ) )
2020-02-20 13:19:23 +01:00
return ( as.rsi ( rep ( NA , length ( x ) ) ) )
}
if ( length ( mo_coerced ) == 1 ) {
mo_coerced <- rep ( mo_coerced , length ( x ) )
}
if ( length ( uti ) == 1 ) {
uti <- rep ( uti , length ( x ) )
}
2020-05-16 13:05:47 +02:00
message ( font_blue ( paste0 ( " => Interpreting disk zones of `" , font_bold ( ab ) , " ` (" ,
2020-07-13 09:17:24 +02:00
ifelse ( ab_coerced != ab , paste0 ( ab_coerced , " , " ) , " " ) ,
ab_name ( ab_coerced , tolower = TRUE ) , " ) using guideline " , font_bold ( guideline_coerced ) , " ... " ) ) ,
2020-02-20 13:19:23 +01:00
appendLF = FALSE )
result <- exec_as.rsi ( method = " disk" ,
x = x ,
mo = mo_coerced ,
ab = ab_coerced ,
guideline = guideline_coerced ,
2020-05-16 13:05:47 +02:00
uti = uti ) # exec_as.rsi will return message(font_blue(" OK."))
2020-02-20 13:19:23 +01:00
result
}
#' @rdname as.rsi
#' @export
2020-07-29 11:46:59 +02:00
as.rsi.data.frame <- function ( x ,
col_mo = NULL ,
guideline = " EUCAST" ,
uti = NULL ,
conserve_capped_values = FALSE ,
... ) {
2020-02-20 13:19:23 +01:00
# try to find columns based on type
# -- mo
if ( is.null ( col_mo ) ) {
col_mo <- search_type_in_df ( x = x , type = " mo" )
2020-06-22 11:18:40 +02:00
stop_if ( is.null ( col_mo ) , " `col_mo` must be set" )
2020-02-20 13:19:23 +01:00
}
2020-06-22 11:18:40 +02:00
2020-02-20 13:19:23 +01:00
# -- UTIs
col_uti <- uti
if ( is.null ( col_uti ) ) {
col_uti <- search_type_in_df ( x = x , type = " uti" )
}
if ( ! is.null ( col_uti ) ) {
if ( is.logical ( col_uti ) ) {
# already a logical vector as input
if ( length ( col_uti ) == 1 ) {
uti <- rep ( col_uti , NROW ( x ) )
} else {
uti <- col_uti
}
} else {
# column found, transform to logical
uti <- as.logical ( x [ , col_uti , drop = TRUE ] )
}
} else {
# look for specimen column and make logicals of the urines
col_specimen <- suppressMessages ( search_type_in_df ( x = x , type = " specimen" ) )
if ( ! is.null ( col_specimen ) ) {
uti <- x [ , col_specimen , drop = TRUE ] %like% " urin"
values <- sort ( unique ( x [uti , col_specimen , drop = TRUE ] ) )
if ( length ( values ) > 1 ) {
plural <- c ( " s" , " " , " " )
} else {
plural <- c ( " " , " s" , " a " )
}
2020-05-16 13:05:47 +02:00
message ( font_blue ( paste0 ( " NOTE: Assuming value" , plural [1 ] , " " ,
2020-07-13 09:17:24 +02:00
paste ( paste0 ( ' "' , values , ' "' ) , collapse = " , " ) ,
" in column `" , font_bold ( col_specimen ) ,
" ` reflect" , plural [2 ] , " " , plural [3 ] , " urinary tract infection" , plural [1 ] , " .\n Use `as.rsi(uti = FALSE)` to prevent this." ) ) )
2020-02-20 13:19:23 +01:00
} else {
# no data about UTI's found
uti <- FALSE
}
}
2020-07-13 09:17:24 +02:00
2020-02-20 13:19:23 +01:00
i <- 0
2020-02-20 20:31:42 +01:00
ab_cols <- colnames ( x ) [sapply ( x , function ( y ) {
2020-02-20 13:19:23 +01:00
i <<- i + 1
check <- is.mic ( y ) | is.disk ( y )
ab <- colnames ( x ) [i ]
ab_coerced <- suppressWarnings ( as.ab ( ab ) )
if ( is.na ( ab_coerced ) ) {
# not even a valid AB code
return ( FALSE )
} else if ( ! check & all_valid_mics ( y ) ) {
2020-05-16 13:05:47 +02:00
message ( font_blue ( paste0 ( " NOTE: Assuming column `" , ab , " ` (" ,
2020-07-13 09:17:24 +02:00
ifelse ( ab_coerced != ab , paste0 ( ab_coerced , " , " ) , " " ) ,
ab_name ( ab_coerced , tolower = TRUE ) , " ) contains MIC values." ) ) )
2020-02-20 13:19:23 +01:00
return ( TRUE )
} else if ( ! check & all_valid_disks ( y ) ) {
2020-05-16 13:05:47 +02:00
message ( font_blue ( paste0 ( " NOTE: Assuming column `" , ab , " ` (" ,
2020-07-13 09:17:24 +02:00
ifelse ( ab_coerced != ab , paste0 ( ab_coerced , " , " ) , " " ) ,
ab_name ( ab_coerced , tolower = TRUE ) , " ) contains disk zones." ) ) )
2020-02-20 13:19:23 +01:00
return ( TRUE )
} else {
return ( check )
}
} ) ]
2020-06-22 11:18:40 +02:00
stop_if ( length ( ab_cols ) == 0 ,
" no columns with MIC values or disk zones found in this data set. Use as.mic() or as.disk() to transform antimicrobial columns." )
2020-02-20 13:19:23 +01:00
# set type per column
types <- character ( length ( ab_cols ) )
types [sapply ( x [ , ab_cols ] , is.mic ) ] <- " mic"
types [types == " " & sapply ( x [ , ab_cols ] , all_valid_mics ) ] <- " mic"
types [sapply ( x [ , ab_cols ] , is.disk ) ] <- " disk"
types [types == " " & sapply ( x [ , ab_cols ] , all_valid_disks ) ] <- " disk"
for ( i in seq_len ( length ( ab_cols ) ) ) {
if ( types [i ] == " mic" ) {
x [ , ab_cols [i ] ] <- as.rsi.mic ( x = x %>% pull ( ab_cols [i ] ) ,
mo = x %>% pull ( col_mo ) ,
ab = ab_cols [i ] ,
guideline = guideline ,
2020-07-29 11:46:59 +02:00
uti = uti ,
conserve_capped_values = conserve_capped_values )
2020-02-20 13:19:23 +01:00
} else if ( types [i ] == " disk" ) {
x [ , ab_cols [i ] ] <- as.rsi.disk ( x = x %>% pull ( ab_cols [i ] ) ,
mo = x %>% pull ( col_mo ) ,
ab = ab_cols [i ] ,
guideline = guideline ,
uti = uti )
}
}
x
2019-05-10 16:44:59 +02:00
}
2019-10-23 14:48:25 +02:00
get_guideline <- function ( guideline ) {
2019-05-13 10:10:16 +02:00
guideline_param <- toupper ( guideline )
if ( guideline_param %in% c ( " CLSI" , " EUCAST" ) ) {
2020-05-16 13:05:47 +02:00
guideline_param <- rev ( sort ( subset ( rsi_translation , guideline %like% guideline_param ) $ guideline ) ) [1L ]
2019-05-10 16:44:59 +02:00
}
2020-02-21 13:13:34 +01:00
if ( ! guideline_param %like% " " ) {
# like 'EUCAST2020', should be 'EUCAST 2020'
guideline_param <- gsub ( " ([a-z]+)([0-9]+)" , " \\1 \\2" , guideline_param , ignore.case = TRUE )
}
2019-10-23 14:48:25 +02:00
2020-06-22 11:18:40 +02:00
stop_ifnot ( guideline_param %in% rsi_translation $ guideline ,
" invalid guideline: '" , guideline ,
" '.\nValid guidelines are: " , paste0 ( " '" , unique ( rsi_translation $ guideline ) , " '" , collapse = " , " ) , call = FALSE )
2019-10-23 14:48:25 +02:00
guideline_param
2020-02-20 13:19:23 +01:00
2019-10-23 14:48:25 +02:00
}
2020-07-29 11:46:59 +02:00
exec_as.rsi <- function ( method , x , mo , ab , guideline , uti , conserve_capped_values ) {
2019-10-23 14:48:25 +02:00
if ( method == " mic" ) {
2020-02-20 13:19:23 +01:00
x <- as.mic ( x ) # when as.rsi.mic is called directly
2019-10-23 14:48:25 +02:00
} else if ( method == " disk" ) {
2020-02-20 13:19:23 +01:00
x <- as.disk ( x ) # when as.rsi.disk is called directly
2019-10-23 14:48:25 +02:00
}
2020-02-20 13:19:23 +01:00
warned <- FALSE
method_param <- toupper ( method )
2019-10-23 14:48:25 +02:00
mo_genus <- as.mo ( mo_genus ( mo ) )
mo_family <- as.mo ( mo_family ( mo ) )
mo_order <- as.mo ( mo_order ( mo ) )
mo_becker <- as.mo ( mo , Becker = TRUE )
mo_lancefield <- as.mo ( mo , Lancefield = TRUE )
2020-04-29 14:33:44 +02:00
mo_other <- as.mo ( rep ( " UNKNOWN" , length ( mo ) ) )
2019-10-23 14:48:25 +02:00
guideline_coerced <- get_guideline ( guideline )
if ( guideline_coerced != guideline ) {
2020-05-16 13:05:47 +02:00
message ( font_blue ( paste0 ( " Note: Using guideline " , font_bold ( guideline_coerced ) , " as input for `guideline`." ) ) )
2019-10-23 14:48:25 +02:00
}
2020-02-17 14:38:01 +01:00
2019-05-10 16:44:59 +02:00
new_rsi <- rep ( NA_character_ , length ( x ) )
2020-02-20 13:19:23 +01:00
ab_param <- ab
2020-02-14 19:54:13 +01:00
trans <- rsi_translation %>%
2020-05-16 13:05:47 +02:00
subset ( guideline == guideline_coerced & method == method_param & ab == ab_param )
trans $ lookup <- paste ( trans $ mo , trans $ ab )
2020-02-17 14:38:01 +01:00
2019-05-10 16:44:59 +02:00
lookup_mo <- paste ( mo , ab )
lookup_genus <- paste ( mo_genus , ab )
lookup_family <- paste ( mo_family , ab )
lookup_order <- paste ( mo_order , ab )
lookup_becker <- paste ( mo_becker , ab )
lookup_lancefield <- paste ( mo_lancefield , ab )
2020-02-14 19:54:13 +01:00
lookup_other <- paste ( mo_other , ab )
2019-10-23 14:48:25 +02:00
2020-02-20 13:19:23 +01:00
if ( all ( trans $ uti == TRUE , na.rm = TRUE ) & all ( uti == FALSE ) ) {
2020-05-16 13:05:47 +02:00
message ( font_red ( " WARNING." ) )
warning ( " Interpretation of " , font_bold ( ab_name ( ab , tolower = TRUE ) ) , " for some microorganisms is only available for (uncomplicated) urinary tract infections (UTI).\n Use parameter 'uti' to set which isolates are from urine. See ?as.rsi." , call. = FALSE )
2020-02-20 13:19:23 +01:00
warned <- TRUE
}
2019-10-11 17:21:02 +02:00
for ( i in seq_len ( length ( x ) ) ) {
2019-05-10 16:44:59 +02:00
get_record <- trans %>%
2020-02-20 13:19:23 +01:00
# no UTI for now
2020-05-16 13:05:47 +02:00
subset ( lookup %in% c ( lookup_mo [i ] ,
2019-05-10 16:44:59 +02:00
lookup_genus [i ] ,
lookup_family [i ] ,
lookup_order [i ] ,
lookup_becker [i ] ,
2020-02-14 19:54:13 +01:00
lookup_lancefield [i ] ,
2020-02-20 13:19:23 +01:00
lookup_other [i ] ) )
if ( isTRUE ( uti [i ] ) ) {
get_record <- get_record %>%
# be as specific as possible (i.e. prefer species over genus):
# desc(uti) = TRUE on top and FALSE on bottom
2020-05-16 13:05:47 +02:00
arrange ( desc ( uti ) , desc ( nchar ( mo ) ) ) # 'uti' is a column in rsi_translation
2020-02-20 13:19:23 +01:00
} else {
get_record <- get_record %>%
filter ( uti == FALSE ) %>% # 'uti' is a column in rsi_translation
2020-05-16 13:05:47 +02:00
arrange ( desc ( nchar ( mo ) ) )
2020-02-20 13:19:23 +01:00
}
2020-05-16 13:05:47 +02:00
get_record <- get_record [1L , ]
2019-10-23 14:48:25 +02:00
2019-05-10 16:44:59 +02:00
if ( NROW ( get_record ) > 0 ) {
2019-10-23 14:48:25 +02:00
if ( is.na ( x [i ] ) ) {
new_rsi [i ] <- NA_character_
} else if ( method == " mic" ) {
2020-02-20 13:19:23 +01:00
mic_input <- x [i ]
mic_S <- as.mic ( get_record $ breakpoint_S )
mic_R <- as.mic ( get_record $ breakpoint_R )
2020-07-29 11:46:59 +02:00
new_rsi [i ] <- ifelse ( isTRUE ( conserve_capped_values ) & mic_input %like% " ^<[0-9]" , " S" ,
ifelse ( isTRUE ( conserve_capped_values ) & mic_input %like% " ^>[0-9]" , " R" ,
ifelse ( isTRUE ( which ( levels ( mic_input ) == mic_input ) <= which ( levels ( mic_S ) == mic_S ) ) , " S" ,
ifelse ( isTRUE ( which ( levels ( mic_input ) == mic_input ) >= which ( levels ( mic_R ) == mic_R ) ) , " R" ,
ifelse ( ! is.na ( get_record $ breakpoint_S ) & ! is.na ( get_record $ breakpoint_R ) , " I" ,
NA_character_ ) ) ) ) )
2019-05-10 16:44:59 +02:00
} else if ( method == " disk" ) {
2020-05-16 13:05:47 +02:00
new_rsi [i ] <- ifelse ( isTRUE ( as.double ( x [i ] ) >= as.double ( get_record $ breakpoint_S ) ) , " S" ,
ifelse ( isTRUE ( as.double ( x [i ] ) <= as.double ( get_record $ breakpoint_R ) ) , " R" ,
ifelse ( ! is.na ( get_record $ breakpoint_S ) & ! is.na ( get_record $ breakpoint_R ) , " I" ,
NA_character_ ) ) )
2019-05-10 16:44:59 +02:00
}
}
}
2020-02-20 13:19:23 +01:00
if ( warned == FALSE ) {
2020-05-16 13:05:47 +02:00
message ( font_green ( " OK." ) )
2020-02-20 13:19:23 +01:00
}
2019-05-10 16:44:59 +02:00
structure ( .Data = factor ( new_rsi , levels = c ( " S" , " I" , " R" ) , ordered = TRUE ) ,
2019-10-11 17:21:02 +02:00
class = c ( " rsi" , " ordered" , " factor" ) )
2019-05-10 16:44:59 +02:00
}
2020-05-28 16:48:55 +02:00
#' @method print rsi
2018-08-25 22:01:14 +02:00
#' @export
#' @noRd
print.rsi <- function ( x , ... ) {
2020-05-27 16:37:49 +02:00
cat ( " Class <rsi>\n" )
2018-08-25 22:01:14 +02:00
print ( as.character ( x ) , quote = FALSE )
}
2020-05-28 16:48:55 +02:00
#' @method droplevels rsi
2018-12-29 22:24:19 +01:00
#' @export
#' @noRd
2019-10-11 17:21:02 +02:00
droplevels.rsi <- function ( x , exclude = if ( anyNA ( levels ( x ) ) ) NULL else NA , ... ) {
2018-12-29 22:24:19 +01:00
x <- droplevels.factor ( x , exclude = exclude , ... )
2019-10-11 17:21:02 +02:00
class ( x ) <- c ( " rsi" , " ordered" , " factor" )
2018-12-29 22:24:19 +01:00
x
}
2020-05-28 16:48:55 +02:00
#' @method summary rsi
2018-08-25 22:01:14 +02:00
#' @export
#' @noRd
summary.rsi <- function ( object , ... ) {
x <- object
2020-06-22 11:18:40 +02:00
n <- sum ( ! is.na ( x ) )
S <- sum ( x == " S" , na.rm = TRUE )
I <- sum ( x == " I" , na.rm = TRUE )
R <- sum ( x == " R" , na.rm = TRUE )
2020-08-10 11:44:58 +02:00
pad <- function ( x ) {
if ( x == " 0%" ) {
x <- " 0.0%"
}
if ( nchar ( x ) < 5 ) {
x <- paste0 ( rep ( " " , 5 - nchar ( x ) ) , x )
}
x
}
2020-07-22 10:24:23 +02:00
value <- c (
2019-10-11 17:21:02 +02:00
" Class" = " rsi" ,
2020-08-10 11:44:58 +02:00
" %R" = paste0 ( pad ( percentage ( R / n , digits = 1 ) ) , " (n=" , R , " )" ) ,
" %SI" = paste0 ( pad ( percentage ( ( S + I ) / n , digits = 1 ) ) , " (n=" , S + I , " )" ) ,
" - %S" = paste0 ( pad ( percentage ( S / n , digits = 1 ) ) , " (n=" , S , " )" ) ,
" - %I" = paste0 ( pad ( percentage ( I / n , digits = 1 ) ) , " (n=" , I , " )" )
2018-08-10 15:01:05 +02:00
)
2020-07-22 10:24:23 +02:00
class ( value ) <- c ( " summaryDefault" , " table" )
value
2018-08-25 22:01:14 +02:00
}
2018-08-10 15:01:05 +02:00
2020-05-28 16:48:55 +02:00
#' @method plot rsi
2018-08-25 22:01:14 +02:00
#' @export
2020-05-28 10:51:56 +02:00
#' @importFrom graphics text axis
2018-08-25 22:01:14 +02:00
#' @noRd
2019-06-16 21:42:40 +02:00
plot.rsi <- function ( x ,
lwd = 2 ,
ylim = NULL ,
2019-10-11 17:21:02 +02:00
ylab = " Percentage" ,
xlab = " Antimicrobial Interpretation" ,
2020-05-28 10:51:56 +02:00
main = paste ( " Resistance Overview of" , deparse ( substitute ( x ) ) ) ,
2019-06-16 21:42:40 +02:00
axes = FALSE ,
... ) {
2020-05-28 10:51:56 +02:00
data <- as.data.frame ( table ( x ) , stringsAsFactors = FALSE )
colnames ( data ) <- c ( " x" , " n" )
data $ s <- round ( ( data $ n / sum ( data $ n ) ) * 100 , 1 )
2020-07-13 09:17:24 +02:00
2019-05-10 16:44:59 +02:00
if ( ! " S" %in% data $ x ) {
data <- rbind ( data , data.frame ( x = " S" , n = 0 , s = 0 ) )
}
if ( ! " I" %in% data $ x ) {
data <- rbind ( data , data.frame ( x = " I" , n = 0 , s = 0 ) )
}
if ( ! " R" %in% data $ x ) {
data <- rbind ( data , data.frame ( x = " R" , n = 0 , s = 0 ) )
}
2019-10-23 14:48:25 +02:00
2020-05-28 10:51:56 +02:00
# don't use as.rsi() here, it will confuse plot()
2019-10-11 17:21:02 +02:00
data $ x <- factor ( data $ x , levels = c ( " S" , " I" , " R" ) , ordered = TRUE )
2019-10-23 14:48:25 +02:00
2018-08-25 22:01:14 +02:00
ymax <- if_else ( max ( data $ s ) > 95 , 105 , 100 )
2019-10-23 14:48:25 +02:00
2020-05-28 10:51:56 +02:00
# get plot() generic; this was moved from the 'graphics' pkg to the 'base' pkg in R 4.0.0
if ( as.integer ( R.Version ( ) $ major ) >= 4 ) {
2020-06-17 15:14:37 +02:00
plot <- import_fn ( " plot" , " base" )
2020-05-28 10:51:56 +02:00
} else {
2020-06-17 15:14:37 +02:00
plot <- import_fn ( " plot" , " graphics" )
2020-05-28 10:51:56 +02:00
}
2018-08-25 22:01:14 +02:00
plot ( x = data $ x ,
y = data $ s ,
2019-06-16 21:42:40 +02:00
lwd = lwd ,
2018-08-25 22:01:14 +02:00
ylim = c ( 0 , ymax ) ,
2019-06-16 21:42:40 +02:00
ylab = ylab ,
xlab = xlab ,
main = main ,
axes = axes ,
2018-08-25 22:01:14 +02:00
... )
# x axis
axis ( side = 1 , at = 1 : n_distinct ( data $ x ) , labels = levels ( data $ x ) , lwd = 0 )
# y axis, 0-100%
axis ( side = 2 , at = seq ( 0 , 100 , 5 ) )
2019-10-23 14:48:25 +02:00
2018-08-25 22:01:14 +02:00
text ( x = data $ x ,
y = data $ s + 4 ,
2019-10-11 17:21:02 +02:00
labels = paste0 ( data $ s , " % (n = " , data $ n , " )" ) )
2018-08-25 22:01:14 +02:00
}
2020-05-28 16:48:55 +02:00
#' @method barplot rsi
2018-08-25 22:01:14 +02:00
#' @export
2019-06-16 22:14:43 +02:00
#' @importFrom graphics barplot axis par
2018-08-25 22:01:14 +02:00
#' @noRd
2019-06-16 21:42:40 +02:00
barplot.rsi <- function ( height ,
2020-02-20 17:21:01 +01:00
col = c ( " chartreuse4" , " chartreuse3" , " brown3" ) ,
2019-10-11 17:21:02 +02:00
xlab = ifelse ( beside , " Antimicrobial Interpretation" , " " ) ,
2020-05-28 10:51:56 +02:00
main = paste ( " Resistance Overview of" , deparse ( substitute ( height ) ) ) ,
2019-10-11 17:21:02 +02:00
ylab = " Frequency" ,
2019-06-16 21:42:40 +02:00
beside = TRUE ,
axes = beside ,
... ) {
2019-10-23 14:48:25 +02:00
2019-06-16 21:42:40 +02:00
if ( axes == TRUE ) {
par ( mar = c ( 5 , 4 , 4 , 2 ) + 0.1 )
} else {
par ( mar = c ( 2 , 4 , 4 , 2 ) + 0.1 )
}
2019-10-23 14:48:25 +02:00
2019-06-16 21:42:40 +02:00
barplot ( as.matrix ( table ( height ) ) ,
col = col ,
xlab = xlab ,
main = main ,
ylab = ylab ,
beside = beside ,
2018-08-25 22:01:14 +02:00
axes = FALSE ,
... )
# y axis, 0-100%
2019-06-16 21:42:40 +02:00
axis ( side = 2 , at = seq ( 0 , max ( table ( height ) ) + max ( table ( height ) ) * 1.1 , by = 25 ) )
if ( axes == TRUE && beside == TRUE ) {
axis ( side = 1 , labels = levels ( height ) , at = c ( 1 , 2 , 3 ) + 0.5 , lwd = 0 )
}
2018-08-10 15:01:05 +02:00
}
2019-08-07 15:37:39 +02:00
2020-05-28 16:48:55 +02:00
#' @method [<- rsi
2020-04-13 21:09:56 +02:00
#' @export
#' @noRd
" [<-.rsi" <- function ( i , j , ... , value ) {
value <- as.rsi ( value )
y <- NextMethod ( )
attributes ( y ) <- attributes ( i )
y
}
2020-05-28 16:48:55 +02:00
#' @method [[<- rsi
2020-04-13 21:09:56 +02:00
#' @export
#' @noRd
" [[<-.rsi" <- function ( i , j , ... , value ) {
value <- as.rsi ( value )
y <- NextMethod ( )
attributes ( y ) <- attributes ( i )
y
}
2020-05-28 16:48:55 +02:00
#' @method c rsi
2020-04-13 21:09:56 +02:00
#' @export
#' @noRd
c.rsi <- function ( x , ... ) {
y <- unlist ( lapply ( list ( ... ) , as.character ) )
x <- as.character ( x )
as.rsi ( c ( x , y ) )
}