In if
, ie
, and while
requests, in addition to the
numeric expressions described in Numeric Expressions, several
Boolean operators are available; the members of this expanded class are
termed conditional expressions.
c glyph
True if glyph is available, where glyph is an ordinary
character, a special character ‘\(xx’ or ‘\[xxx]’,
‘\N'xxx'’, or has been defined by any of the char
,
fchar
, fschar
, or schar
requests.
d name
True if a string, macro, diversion, or request called name exists.
e
True if the current page is even-numbered.
F font
True if font exists. font is handled as if it were opened
with the ft
request (that is, font translation and styles are
applied), without actually mounting it.
m color
True if color is defined.
n
¶True if the document is being processed in nroff
mode.
See troff
and nroff
Modes.
o
True if the current page is odd-numbered.
r register
True if register exists.
S style
True if style is available for the current font family. Font translation is applied.
t
True if the document is being processed in troff
mode.
See troff
and nroff
Modes.
v
Always false. This condition is recognized only for compatibility with
certain other troff
implementations.88
If the first argument to an if
, ie
, or while
request begins with a non-alphanumeric character apart from !
(see below); it performs an output comparison test.
89
'
xxx'
yyy'
True if formatting the comparands xxx and yyy produces the same output commands. The delimiter need not be a neutral apostrophe: the output comparison operator accepts the same delimiters as most escape sequences; see Delimiters. This output comparison operator formats xxx and yyy in separate environments; after the comparison, the resulting data are discarded.
.ie "|"\fR|\fP" true .el false ⇒ true
The resulting glyph properties, including font family, style, size, and
slant, must match, but not necessarily the requests and/or escape
sequences used to obtain them. In the previous example, ‘|’ and
‘\fR|\fP’ result in ‘|’ glyphs in the same typefaces at the
same positions, so the comparands are equal. If ‘.ft I’ had
been added before the ‘.ie’, they would differ: the first ‘|’
would produce an italic ‘|’, not a roman one. Motions must match
in orientation and magnitude to within the applicable horizontal and
vertical motion quanta of the device, after rounding. ‘.if
"\u\d"\v'0'"’ is false even though both comparands result in zero net
motion, because motions are not interpreted or optimized but sent as-is
to the output.90 On the other hand, ‘.if "\d"\v'0.5m'"’ is true, because
\d
is defined as a downward motion of one-half em.91
Surround the comparands with \?
to avoid formatting them; this
causes them to be compared character by character, as with string
comparisons in other programming languages.
.ie "\?|\?"\?\fR|\fP\?" true .el false ⇒ false
Since comparands protected with \?
are read in copy mode
(see Copy Mode), they need not even be valid groff
syntax.
The escape character is still lexically recognized, however, and
consumes the next character.
.ds a \[ .ds b \[ .if '\?\*a\?'\?\*b\?' a and b equivalent .if '\?\\?'\?\\?' backslashes equivalent ⇒ a and b equivalent
The above operators can’t be combined with most others, but a leading ‘!’, not followed immediately by spaces or tabs, complements an expression.
.nr x 1 .ie !r x register x is not defined .el register x is defined ⇒ register x is defined
Spaces and tabs are optional immediately after the ‘c’, ‘d’, ‘F’, ‘m’, ‘r’, and ‘S’ operators, but right after ‘!’, they end the predicate and the conditional evaluates true.92
.nr x 1 .ie ! r x register x is not defined .el register x is defined ⇒ r x register x is not defined
The unexpected ‘r x’ in the output is a clue that our conditional was not interpreted as we planned, but matters may not always be so obvious.