sdapsarray package

This is a base package for “array” like environments. It really is similar to a tabularx environment to some extend. Its purpose is much more specialized compared to tabularx. It is less flexible in the types of layouts that can be realized but a lot more powerful otherwise. The sdapsarray environment has the following features:

  • All sdapsarray environments in the document can be aligned to each other

  • The environment can span multiple pages

  • Headers will be repeated when page splits are encountered

  • The rows/columns can be swapped on the fly

  • Different layouter can be plugged in to modify the rendering

  • Fragile content can be used without further preparation

  • Contained content is executed exactly once (important for metadata generation)

Things that are not possible currently:

  • Row or column backgrounds

  • Grid lines

Warning

The sdapsarray is not a tabular like environment. It behaves in similar ways, but there are fundamental differences, causing some issues:

  • You must not add a trailing \\ to the last row.

The following two \texttt{sdapsarray} environments are almost identical. They
are both aligned to each other because the \texttt{align} option is set to
the same value. In the second environment the rows and columnes are swapped
by setting the \texttt{flip} option.

\begin{sdapsarray}[align=testing]
  row header & colum header & colum header \\
  row header & cell 1 & cell 2 \\
  row header & cell 3 & cell 4
\end{sdapsarray}

\hrule

\begin{sdapsarray}[flip,align=testing]
  row header & colum header & colum header \\
  row header & cell 1 & cell 2 \\
  row header & cell 3 & cell 4
\end{sdapsarray}

The following two \texttt{sdapsarray} environments are almost identical. They
are both aligned to each other because the \texttt{align} option is set to
the same value. In the second environment the rows and columnes are swapped
by setting the \texttt{flip} option.

\begin{sdapsarray}[align=testing]
  row header & colum header & colum header \\
  row header & cell 1 & cell 2 \\
  row header & cell 3 & cell 4
\end{sdapsarray}

\hrule

\begin{sdapsarray}[flip,align=testing]
  row header & colum header & colum header \\
  row header & cell 1 & cell 2 \\
  row header & cell 3 & cell 4
\end{sdapsarray}

Example of a sdapsarray environment

\begin{multicols}{2}
    \begin{sdapsarray}[align=testing,layouter=rotated]
      colum header 0 & colum header 1 & colum header 2 \\
      row header 1 & cell 1 & cell 2 \\
      row header 2 & cell 3 & cell 4 \\
      row header 3 & cell 5 & cell 6 \\
      row header 4 & cell 7 & cell 8 \\
      row header 5 & cell 9 & cell 10 \\
      row header 6 & cell 11 & cell 12
    \end{sdapsarray}
\end{multicols}
Required code in preamble:
\usepackage{multicol}

\begin{multicols}{2}
    \begin{sdapsarray}[align=testing,layouter=rotated]
      colum header 0 & colum header 1 & colum header 2 \\
      row header 1 & cell 1 & cell 2 \\
      row header 2 & cell 3 & cell 4 \\
      row header 3 & cell 5 & cell 6 \\
      row header 4 & cell 7 & cell 8 \\
      row header 5 & cell 9 & cell 10 \\
      row header 6 & cell 11 & cell 12
    \end{sdapsarray}
\end{multicols}

Example of a sdapsarray environment split over two columns using multicols

Layout and formatting considerations

The following hold true inside the environment:

  • The row headers are set into a \vtop with the left over width from the cells. This vertical box is later re-set into a \vbox. The effect is that the interrow skip is calculated between the last element of the previous row and the first element of the next row. this means you must be careful to not insert invisible content at the start of the vertical box. (e.g. by adding a \leavevmode).

  • The exception to the above rule is the start of the environment (i.e. the header row) for which the top baseline information is (currently) discarded!

  • Each cell is set into an \hbox with the last skip in the box removed again (i.e. trailing space). You can use \hfill to align the box to the left/right but need to prevent the \hfill to be removed again for left alignment (e.g. by adding a \kern 0pt).

  • Column headers behave like cells but a special layouter can be assigned to them.

  • Row headers and column headers will usually be set on a common baseline. The exception to this is if the column header contains multiple boxes/lines. In that case the cells will be centered ignoring the baselines of both cells and row header.

  • A penalty of 10 is inserted between rows.

sdapsarray environment

\begin{sdapsarray} [kwargs]
content with cells delimitted with & and \
\end{sdapsarray}
Keyword Arguments
  • flip – Transpose array making rows to columns (default: false)

  • layouter

    The layouter to use. New layouters can be defined, the following exists by default:

    • default: Simple layout centering cells and giving all leftover space to the row header which will line break automatically (this is the default)

    • rotated: Similar to default but rotates the column headers

  • angle – The angle of the header when in rotated mode

  • align – An arbitrary string to align multiple sdapsarray environments to each other. All environments with the same string will be aligned. (default: no alignment)

  • keepenv – Do not modify the parser to consume & and \\ for alignment. Instead, the user must use \sdaps_array_alignment: and \sdaps_array_newline:. This is only useful for writing custom environments which use sdapsarray internally. Normal users should simply put any nested array environment into \sdapsnested to prevent issues (see below).

  • no_header – Disable column header handling and repeating. Note that this setting is independent of whether the flip option is set. As such, one may need to take its value into account when setting it. (default: false)

  • colsep – Spacing added on the left/right of every cell. This defaults to 6pt.

  • rowsep – Extra spacing added between rows. This defaults to 0pt.

The keepenv option should usually not be used by an end user writing a document, it is very useful when writing environments which use sdapsarray internally (like choicearray).

\sdapsnested {content}

Reverts the & and \\ to their original meaning. Content in an sdapsarray environment can be wrapped with this if it requires these characters to be active (i.e. you can use the array environment this way for example).

\sdaps_array_alignment:

Alternative to using the & delimiter between cells. This is useful together with the keepenv kwarg argument. In particular when creating custom environments which use sdapsarray internally.

\sdaps_array_newline:

Alternative to using the \\ delimiter between cells. This is useful together with the keepenv kwarg argument. In particular when creating custom environments which use sdapsarray internally.

\begin{multicols}{2}
    \begin{sdapsarray}
       & col 1 & col 2 \\
      row header 1 & \sdapsnested{$ \begin{array}{cc} a & b \\ c & d \end{array}$} & cell 2 \\
      \verb^row_header^ & cell 3 & cell 4
    \end{sdapsarray}

    \begin{sdapsarray}[keepenv]
       \sdapsalignment col 1 \sdapsalignment col 2 \sdapsnewline
      row header 1 \sdapsalignment $ \begin{array}{cc} a & b \\ c & d \end{array}$ \sdapsalignment cell 2 \sdapsnewline
      \verb^row_header^ \sdapsalignment cell 3 \sdapsalignment cell 4
    \end{sdapsarray}
\end{multicols}
Required code in preamble:
\usepackage{multicol}
% Wrap the commands with _ as we cannot use them directly. This needs to
% be a \def and not a \let because they are redefined dynamically internally.
\ExplSyntaxOn
\def\sdapsalignment{\sdaps_array_alignment:}
\def\sdapsnewline{\sdaps_array_newline:}
\ExplSyntaxOff

\begin{multicols}{2}
    \begin{sdapsarray}
       & col 1 & col 2 \\
      row header 1 & \sdapsnested{$ \begin{array}{cc} a & b \\ c & d \end{array}$} & cell 2 \\
      \verb^row_header^ & cell 3 & cell 4
    \end{sdapsarray}

    \begin{sdapsarray}[keepenv]
       \sdapsalignment col 1 \sdapsalignment col 2 \sdapsnewline
      row header 1 \sdapsalignment $ \begin{array}{cc} a & b \\ c & d \end{array}$ \sdapsalignment cell 2 \sdapsnewline
      \verb^row_header^ \sdapsalignment cell 3 \sdapsalignment cell 4
    \end{sdapsarray}
\end{multicols}

Two sdapsarray environments each with a nested array, in one case using the keepenv option.

Defining a custom layouter

Warning

This is an advanced feature and its use a good or even in depth knowledge of how TeX processes boxes and input!

It is possible to register further layouter which can subsequently used throughout the document. These layouters need to adhere to a number of rules which will not be explained in detail here.

The following code is a copy of the two predefined layouter not showing the implementation of the different macros. Visible here is that they only differ in the method to render the column header colhead, all other methods are identical.

\prop_gput:Nnn \g__sdaps_array_layouter_prop { default } {
  begin = { \_sdaps_array_begin_default: },
  row_start = { \_sdaps_array_row_start_default: },
  rowhead = { \_sdaps_array_rowhead_default:Nw },
  colhead = { \_sdaps_array_cell_default:Nw },
  cell = { \_sdaps_array_cell_default:Nw },
  row = { \_sdaps_array_row_ltr:NNNN },
  end = { \_sdaps_array_end_default: },
}

\prop_gput:Nnn \g__sdaps_array_layouter_prop { rotated } {
  begin = { \_sdaps_array_begin_default: },
  row_start = { \_sdaps_array_row_start_default: },
  rowhead = { \_sdaps_array_rowhead_default:Nw },
  colhead = { \_sdaps_array_cell_rotated:Nw },
  cell = { \_sdaps_array_cell_default:Nw },
  row = { \_sdaps_array_row_ltr:NNNN },
  end = { \_sdaps_array_end_default: },
}

If you consider modifying the layouter, then please have a look at the relevant parts of sdapsarray.dtx. Also, please consider submitting modifications for upstream inclusion so that other people can benefit from new features.