R/latex-tree.R

Defines functions print.latex_tree latex_tree

Documented in latex_tree

#' Inspect the parsed layout of a LaTeX expression
#'
#' Returns the raw draw-record table produced by MicroTeX's layout pass
#' together with the bounding-box metadata. Useful for debugging
#' alignment issues, building custom grobs on top of the layout, or
#' counting glyphs/paths/rules in a formula.
#'
#' @inheritParams latex_grob
#' @return A list with class \code{"latex_tree"} containing:
#'   \describe{
#'     \item{\code{records}}{Data frame of draw records (one row per
#'       glyph, path, line, rect, or text block). Columns include
#'       \code{type}, \code{x}, \code{y}, \code{glyph}, \code{font_size},
#'       \code{color}, \code{text}, \code{codepoint}, \code{font_file}.}
#'     \item{\code{bbox}}{Named numeric vector with \code{width},
#'       \code{height}, \code{depth}, \code{baseline} (all in big points).}
#'     \item{\code{tex}}{The (macro-expanded) input string.}
#'     \item{\code{render_mode}}{Rendering mode used for the layout.}
#'   }
#' @seealso \code{\link{latex_grob}}, \code{\link{latex_dims}}
#' @export
#'
#' @examples
#' \donttest{
#'   tree <- latex_tree("\\frac{a}{b}")
#'   print(tree)
#'   head(tree$records)
#' }
latex_tree <- function(tex, math_font = "", max_width = 0,
                       tex_style = "",
                       input_mode = c("mixed", "math"),
                       render_mode = c("typeface", "path"),
                       gp = grid::gpar()) {
  .apply_opts("math_font", "render_mode", "tex_style", "input_mode")
  render_mode <- match.arg(render_mode)
  input_mode <- match.arg(input_mode)

  parsed <- .parse_from_gp(
    tex = tex, gp = gp, math_font = math_font, max_width = max_width,
    tex_style = tex_style, render_mode = render_mode,
    input_mode = input_mode
  )
  layout <- parsed$layout

  bbox <- c(
    width    = as.numeric(attr(layout, "bbox_width")),
    height   = as.numeric(attr(layout, "bbox_height")),
    depth    = as.numeric(attr(layout, "bbox_depth")),
    baseline = as.numeric(attr(layout, "bbox_baseline"))
  )

  structure(
    list(
      records = layout,
      bbox = bbox,
      tex = parsed$tex,
      render_mode = parsed$render_mode
    ),
    class = "latex_tree"
  )
}

#' @export
print.latex_tree <- function(x, ...) {
  cat("<latex_tree>\n")
  cat("  tex:         ", x$tex, "\n", sep = "")
  cat("  render_mode: ", x$render_mode, "\n", sep = "")
  cat(sprintf(
    "  bbox:        width=%.2f  height=%.2f  depth=%.2f  baseline=%.2f (bigpts)\n",
    x$bbox[["width"]], x$bbox[["height"]],
    x$bbox[["depth"]], x$bbox[["baseline"]]
  ))
  n <- nrow(x$records)
  cat("  records:     ", n, "\n", sep = "")
  if (n > 0) {
    tbl <- table(x$records$type)
    for (nm in names(tbl)) {
      cat(sprintf("    %-10s %d\n", nm, tbl[[nm]]))
    }
  }
  invisible(x)
}

Try the gridmicrotex package in your browser

Any scripts or data that you put into this service are public.

gridmicrotex documentation built on May 16, 2026, 5:06 p.m.