R/zzz.R

Defines functions stringify_params get_description

Documented in get_description

#' Get struct object help description
#'
#' This
#' function is to help developers including struct objects in their own R
#' packages, and isnt intended for general use.
#' Use with roxygen 2 `@eval` tags this function generates a detailed
#' description of a struct object generated by extracting names,
#' descriptions etc from slots in a suitable format.
#'
#' @param id (character) the name of a struct object to generate documentation for
#' @return a character string of roxygen formatted documentation for the object
#' @examples
#' get_description('example_model')
#'
#' @export
get_description=function(id) {

    # object template
    M=new_struct(id)

    # title
    str=paste0('@title ', M$name)

    # get description
    str=c(str,paste0('@description ',M$description))

    # add inheritance
    i=.class2(M)
    w=which(i=='struct_class')
    str = c(str,
            '@section ','Inheritance: ',
            paste0('A `', class(M),
                   '` object inherits the following `struct` classes: \\cr\\cr'),
            paste0('`[',i[1:w],']`',collapse = ' >> '))


    # citations
    cits=citations(M)
    cits[length(cits)]=NULL
    for (k in seq_along(cits)) {
        cit=format(cits[[k]],style='text')
        str=c(str,paste0('@references ',cit))
    }

    # get libraries
    if (length(M$libraries)>0) {
        str2=paste0('@details ',
                    'This object makes use of functionality from the following packages:',
                    '\\itemize{ '
        )
        for (k in seq_along(M$libraries)) {
            str2=paste0(str2,' \\item{\\code{',M$libraries[k],'}}')
        }
        str2=paste0(str2,'}')
        str=c(str,str2)
    }

    # parameters
    P=formals(id)

    # for each parameter generate some text
    D=list()
    for (k in seq_along(P)) {

        # skip if ellipsis
        if (names(P)[k]=='...') {
            D[[k]]= paste0('@param ',names(P)[k],
                           ' Additional slots and values passed to \\code{struct_class}.')
            next
        }
        #g=param_obj(M,names(P)[k])

        D[[k]]=stringify_params(M,names(P)[k],type='param',val=P[[k]])
    }
    str=c(str,D)

    # add outputs and descriptions to value
    O = output_list(M)

    D=list()
    for (k in seq_along(O)) {
        D[[k]]=stringify_params(M,names(O)[k],type='output',val=O[[k]])
    }

    if (length(D)>0){
        str=c(str,
              paste0('@return ','A  \\code{', class(M)[1],'} object with the following \\code{output} slots:' ),
              '\\tabular{ll}{',
              D,
              '}')
    } else{
        str=c(str,'@return ','A  \\code{', class(M)[1],'} object. This object has no \\code{output} slots.')
        if (is(M,'chart')) {
            str=c(str, 'See \\code{\\link[struct]{chart_plot}} in the \\code{struct} package to plot this chart object.')
        }

    }

    # basic example
    eg = '@examples'
    cd = as.code(M,mode='neat',quiet=TRUE)
    cd = gsub('[a list]','list()',cd,fixed = TRUE)
    cd = gsub('[a function]','function(){}',cd,fixed = TRUE)
    cd = gsub('[a annotation_database]','annotation_database()',cd,fixed = TRUE)
    cd = gsub('[a logical]','FALSE',cd,fixed = TRUE)
    cd = gsub('[a data.frame]','data.frame(id=NA)',cd,fixed = TRUE)
    cd = gsub('[a call]','call("example")',cd,fixed = TRUE)
    cd = gsub('[a ANY]','"ANY"',cd,fixed = TRUE)

    str=c(str,eg,cd,'')

    return(unlist(str))
}



# in internal function used by get_desciption
stringify_params = function(M,P,type='param',val=NULL) {

    # get parameter as an object
    if (type=='param') {
        p = param_obj(M,P)
    } else {
        p=output_obj(M,P)
    }

    # if its an entity object then get its description
    if (is(p,'entity')) {
        d = p$description
        # ensure first character is upper case and last character is a fullstop.
        d=unlist(lapply(d,function(x){
            # first is upper
            substr(x,1,1) = toupper(substr(x,1,1))
            # last is .
            if (substr(x,nchar(x),nchar(x)) != '.') {
                x=paste0(x,'.')
            }
            return(x)
        }))

        # if d has more than one entry and is a named vector then...
        if (length(d)>1) {
            # if it has names then
            if (!is.null(names(d))) {
                # create a named list
                it_list='\\itemize{ '
                for (j in seq_along(d)) {
                    it_list=paste0(it_list,'\\item{\\code{"',names(d)[j],'"}: ',d[j],'}')
                }
            } else {
                # no names so use a bulleted list
                it_list='\\itemize{'
                for (j in seq_along(d)) {
                    it_list=paste0(it_list,'\\item{',d[j],'}')
                }
            }

            # add list
            it_list=paste0(it_list,'}')
            d=paste0(p$name,'. Allowed values are limited to the following: ',it_list)
        }

        # add the allowed types
        t = p$type

    } else {
        # if not an entity then there is no description
        d = ''
        t=class(val)[[1]]
    }
    # collapse if more than 1
    t=paste0(t,collapse=', ')
    # enclose in brackets
    t=paste0('(',t,') ')
    # add to description
    d=paste0(t,d)

    # if the parameter has a default, then add on the text.
    if ( (!is(val,'name')) & type=='param'){
        d=paste0(d, ' The default is ')
        if (length(val)>1 & !is.call(val)) {
            d=paste0(d,'\\code{',capture.output(val)[1],'}.')
        } else {
            if (is.null(val)) {
                d=paste0(d,'\\code{NULL}.')
            } else if (is(val,'character')) {
                d=paste0(d,'\\code{"',val,'"}.')
            } else if (is.function(val) | is.call(val)) {
                d=paste0(d,
                         '\\code{',gsub('}','\\}',
                         paste0(trimws(deparse1(val)),collapse='')),'}.')
                #d=paste0(d,'\\code{some_function()}.')
            } else if (is(val,'formula')) {
                d = paste0('\\code{',deparse1(val),'}.')
            } else {

                d=paste0(d,'\\code{',val,'}.\\cr')
            }
        }
    } else {
        # no default is provided
    }
    if (type=='param') {
        OUT=paste0('@', type, ' ', P, ' ', d)
    } else {
        OUT=paste0('\\code{',P,'} \\tab          ',d,' \\cr')
    }
}
computational-metabolomics/struct documentation built on March 27, 2024, 4:26 p.m.