R/tecanInfiniteFunctions.R

Defines functions infinite_parse next_filled next_blank

Documented in infinite_parse next_blank next_filled

#' Find next blank line
#'
#' @param start_idx
#' @param data
#'
#' @return row index of next blank line
next_blank <- function(start_idx, data){
  next_start_idx <- start_idx
  while (!is.na(data[next_start_idx, 1])) {
    next_start_idx <- next_start_idx + 1
  }
  return(next_start_idx)
}

#' Find next non-blank line
#'
#' @param start_idx
#' @param data
#'
#' @return row index of next non-blank line
next_filled <- function(start_idx, data){
  next_start_idx <- start_idx
  while (is.na(data[next_start_idx, 1])) {
    next_start_idx <- next_start_idx + 1
  }
  return(next_start_idx)
}

#' Parser for Tecan Infinite plate reader data
#'
#' @param data_csv path to .csv, .xls or .xlsx file from Tecan Infinit plate reader
#' @param layout_csv path to csv file containing plate layout information
#' @param timeseries Boolean flag indicating whether the data is a timeseries or
#' single recording. The Tecan software outputs the two scenarios differently.
#'
#' @return a data.frame containing the parsed plate reader data
#' @export
#' @importFrom rlang .data
#'
infinite_parse <- function(data_csv, layout_csv, timeseries=F) {

  if(stringr::str_ends(data_csv, ".xlsx") | stringr::str_ends(data_csv, ".xls")){
    data <- as.data.frame(readxl::read_excel(path = data_csv,
                                             col_names = F,
                                             col_types = "text"))
  } else if(stringr::str_ends(data_csv, ".csv")){
    data <- utils::read.table(data_csv, sep = ",",
                              na.strings = c(""),
                              header = F,
                              stringsAsFactors = F)
  } else {
    stop("data_csv is must be a .csv, .xls or .xlsx file.")
  }

  plate_layout <- utils::read.csv(layout_csv)

  if(timeseries == TRUE){
    start_time_idx <- which(data[, 1] == "Start Time:")
    if (length(start_time_idx) > 1) {
      start_time_idx <- start_time_idx[length(start_time_idx)]
    }

    # find where the next block starts
    next_block_start_idx <- next_filled(start_time_idx + 1, data)

    end_of_file <- F
    all_data <- c()
    while (!end_of_file) {
      # find what is being measured
      block_name <- data[next_block_start_idx, 1]

      # check if we are at the end of the file
      if (block_name == "End Time:") {
        end_of_file <- T
        break
      }

      # find where the end of the current measurement block is
      block_end_idx <- next_blank(next_block_start_idx, data)

      # grab the data only for that measurement
      new_block <- data[(next_block_start_idx + 1):(block_end_idx - 1), ]
      # new_block <- new_block[-c(1,3), ]  # remove cycle no. and temp.

      # trim unnecessary readings i.e. temp and cycle number
      # and rename columns
      times <- as.character(new_block[2,])
      new_block <- new_block[-c(1:3), ]
      names(new_block) <- times
      names(new_block)[1] <- "well"
      new_block <- new_block[,names(new_block) != "NA"]  # remove erroneous NA columns

      # reshape time into long data
      new_block <- new_block %>%
        tidyr::pivot_longer(cols = 2:ncol(new_block),
                            names_to = "time", names_transform = as.numeric,
                            values_to = "value", values_transform = as.numeric)

      # add info for each well
      joined_block <- dplyr::full_join(plate_layout, new_block)
      joined_block$measure <- block_name

      #
      all_data <- rbind(all_data, joined_block)

      #
      next_block_start_idx <- next_filled(block_end_idx + 1, data)
    }

    # rearrange data ----------------------------------------------------------
    layout_cols <- ncol(plate_layout)
    out_data <- all_data %>%
      tidyr::pivot_wider(names_from = .data$measure, values_from = .data$value) %>%  # reshape so we have a column for each measurement type
      dplyr::mutate(row = substr(x = .data$well, start = 1, stop = 1)) %>%  # make a "row" column from the "well" column
      dplyr::mutate(column = as.numeric(substr(x = .data$well, start = 2,  # and make a "column" column
                                               stop = nchar(.data$well)))) %>%
      dplyr::arrange_at(dplyr::vars(.data$time,  # order the rows
                                    .data$row,
                                    .data$column))

    # write parsed data to csv ------------------------------------------------
    out_name <- gsub(".csv", "_parsed.csv", data_csv)
    out_name <- gsub(".xlsx", "_parsed.csv", out_name)
    utils::write.csv(x = out_data, file = out_name, row.names = FALSE)

    return(out_data)
  }
  # else if (timeseries == FALSE){
  #   start_time_idx <- which(data[, 1] == "Start Time")  # get start and end time ids
  #   end_idx <- which(data[, 1] == "End Time")
  #   names_idx <- which(data[, 1] == "Name")
  #   names_idx <- names_idx[2:length(names_idx)]  # remove the first start time entry which just details plate type
  #
  #   all_data <- c()
  #   for (i in seq_len(length(start_time_idx))) {
  #     block_name <- data[names_idx[i], 2]  # record name of what is being measured
  #
  #     block_start <- start_time_idx[i] + 4  # find start and end of measurement block
  #     block_end_idx <- end_idx[i] - 3
  #
  #     new_block <- data[(block_start):(block_end_idx), 1:2]  # grab and name the data
  #     names(new_block)[1] <- "well"
  #     names(new_block)[2] <- "value"
  #
  #     new_block$value <- as.numeric(new_block$value)
  #
  #     joined_block <- dplyr::full_join(plate_layout, new_block)  # join to plate layout csv, add measurement category
  #     joined_block$measure <- block_name
  #
  #     all_data <- rbind(all_data, joined_block)  # add to all data
  #   }
  #
  #   # rearrange data ----------------------------------------------------------
  #   spread_data <- tidyr::pivot_wider(all_data, names_from = .data$measure,
  #                                     values_from = .data$value)
  #   spread_data$row <- substr(x = spread_data$well, start = 1, stop = 1)
  #   spread_data$column <- as.numeric(substr(x = spread_data$well, start = 2,
  #                                           stop = nchar(spread_data$well)))
  #   spread_data <- dplyr::arrange_at(spread_data, dplyr::vars(.data$row,
  #                                                             .data$column))
  #
  #   # write parsed data to csv ------------------------------------------------
  #   out_name <- gsub(".csv", "_parsed.csv", data_csv)
  #   utils::write.csv(x = spread_data, file = out_name, row.names = FALSE)
  #
  #   return(spread_data)
  # }
}
ucl-cssb/flopr documentation built on Jan. 27, 2024, 11:49 a.m.