File: groff.info, Node: Writing Macros, Next: Page Motions, Prev: Conditionals and Loops, Up: GNU troff Reference 5.24 Writing Macros =================== A "macro" is a stored collection of text and control lines that can be interpolated multiple times. Use macros to define common operations. Macros are called in the same way that requests are invoked. While requests exist for the purpose of creating macros, simply calling an undefined macro, or interpolating it as a string, will cause it to be defined as empty. *Note Identifiers::. -- Request: .de name [end] Define a macro NAME, replacing the definition of any existing request, macro, string, or diversion called NAME. If NAME already exists as an alias, the target of the alias is redefined; recall *note Strings::. GNU 'troff' enters copy mode,(1) (*note Writing Macros-Footnote-1::) storing subsequent input lines as the macro definition. If the optional second argument is not specified, the definition ends with the control line '..' (two dots). Alternatively, END identifies a macro whose call syntax at the start of a control line ends the definition of NAME; END is then called normally. A macro definition must end in the same conditional block (if any) in which it began (*note Conditional Blocks::). Spaces or tabs are permitted after the control character in the line containing this ending token (either '.' or 'END'), but a tab immediately after the token prevents its recognition as the end of a macro definition. The macro END can be called with arguments.(2) (*note Writing Macros-Footnote-2::) Here is a small example macro called 'P' that causes a break and inserts some vertical space. It could be used to separate paragraphs. .de P . br . sp .8v .. We can define one macro within another. Attempting to nest '..' naïvely will end the outer definition because the inner definition isn't interpreted as such until the outer macro is later interpolated. We can use an end macro instead. Each level of nesting should use a unique end macro. An end macro need not be defined until it is called. This fact enables a nested macro definition to begin inside one macro and end inside another. Consider the following example.(3) (*note Writing Macros-Footnote-3::) .de m1 . de m2 m3 you .. .de m3 Hello, Joe. .. .de m4 do .. .m1 know? . m3 What .m4 .m2 => Hello, Joe. What do you know? A nested macro definition _can_ be terminated with '..' and nested macros _can_ reuse end macros, but these control lines must be escaped multiple times for each level of nesting. The necessity of this escaping and the utility of nested macro definitions will become clearer when we employ macro parameters and consider the behavior of copy mode in detail. 'de' defines a macro that inherits the compatibility mode enablement status of its context (*note Implementation Differences::). Often it is desirable to make a macro that uses 'groff' features callable from contexts where compatibility mode is on; for instance, when writing extensions to a historical macro package. To achieve this, compatibility mode needs to be switched off while such a macro is interpreted--without disturbing that state when it is finished. -- Request: .de1 name [end] The 'de1' request defines a macro to be interpreted with compatibility mode disabled. When NAME is called, compatibility mode enablement status is saved; it is restored when the call completes. Observe the extra backlash before the interpolation of register 'xxx'; we'll explore this subject in *note Copy Mode::. .nr xxx 12345 .de aa The value of xxx is \\n[xxx]. . br .. .de1 bb The value of xxx is \\n[xxx]. .. .cp 1 .aa error-> warning: register '[' not defined => The value of xxx is 0xxx]. .bb => The value of xxx is 12345. -- Request: .dei name [end] -- Request: .dei1 name [end] The 'dei' request defines a macro with its name and end macro indirected through strings. That is, it interpolates strings named NAME and END before performing the definition. The following examples are equivalent. .ds xx aa .ds yy bb .dei xx yy .de aa bb The 'dei1' request bears the same relationship to 'dei' as 'de1' does to 'de'; it temporarily turns compatibility mode off when NAME is called. -- Request: .am name [end] -- Request: .am1 name [end] -- Request: .ami name [end] -- Request: .ami1 name [end] 'am' appends subsequent input lines to macro NAME, extending its definition, and otherwise working as 'de' does. To make the previously defined 'P' macro set indented instead of block paragraphs, add the necessary code to the existing macro. .am P .ti +5n .. The other requests are analogous to their 'de' counterparts. The 'am1' request turns off compatibility mode during interpretation of the appendment. The 'ami' request appends indirectly, meaning that strings NAME and END are interpolated with the resulting names used before appending. The 'ami1' request is similar to 'ami', disabling compatibility mode during interpretation of the appended lines. Using 'trace.tmac', you can trace calls to 'de', 'de1', 'am', and 'am1'. You can also use the 'backtrace' request at any point desired to troubleshoot tricky spots (*note Debugging::). *Note Strings::, for the 'als', 'rm', and 'rn' requests to create an alias of, remove, and rename a macro, respectively. Macro identifiers share their name space with requests, strings, and diversions; see *note Identifiers::. The 'am', 'as', 'da', 'de', 'di', and 'ds' requests (together with their variants) create a new object only if the name of the macro, diversion, or string is currently undefined or if it is defined as a request; normally, they modify the value of an existing object. *Note the description of the 'als' request: als, for pitfalls when redefining a macro that is aliased. -- Request: .return [anything] Exit a macro, immediately returning to the caller. If called with an argument ANYTHING, exit twice--the current macro and the macro one level higher. This is used to define a wrapper macro for 'return' in 'trace.tmac'. * Menu: * Parameters:: * Copy Mode::