-
-
Notifications
You must be signed in to change notification settings - Fork 497
Styling
const para = new Paragraph("To whom it may concern:")
.heading2()
.center();
const name = new TextRun("Name:")
.bold()
.font("Calibri")
.allCaps();
- For run formatting:
-
.bold()
,.italic()
,.smallCaps()
,.allCaps()
,.strike()
,.doubleStrike()
,.subScript()
,.superScript()
: Set the formatting property to true -
.underline(style="single", color=null)
: Set the underline style and color -
.color(color)
: Set the text color, using 6 hex characters for RRGGBB (no leading#
) -
.size(halfPts)
: Set the font size, measured in half-points -
.font(name)
: Set the run's font -
.style(name)
: Apply a named run style
-
- For paragraph formatting:
-
.heading1()
,.heading2()
,.heading3()
,.heading4()
,.heading5()
,.title()
: apply the appropriate style to the paragraph -
.left()
,.center()
,.right()
,.justified()
: set the paragraph's alignment -
.thematicBreak()
,.pageBreak()
: Insert a thick rule or a page break beneath the paragraph -
.leftTabStop(position)
: Add a left tab stop (measured in TWIPs from the left) -
.maxRightTabStop()
: Add a right tab stop at the far right -
.bullet()
: Use the default bullet style -
.setNumbering(numbering, indentLevel)
: Use a custom numbering format for the paragraph -
.style(name)
: Apply a named paragraph style -
.indent(start, hanging=0)
: Set the paragraph's indent level (in TWIPs) -
.spacing({before=0, after=0, line=0})
: Set the line and before/after on the paragraph. Before/after is measured in TWIPs, line is measured in 240ths of a line
-
Paragraph styles have all the run formatting methods, except style()
, and .left()
, .center()
, .right()
, .justified()
, .thematicBreak()
, .leftTabStop(position)
, .maxRightTabStop()
, .indent(start, hanging=0)
, and .spacing({before=0, after=0, line=0})
methods.
There are 4 items in DOCX that can be styled:
- Characters: Attributes that can change within a paragraph. e.g., bold, italics, etc.
- Paragraphs: Attributes like indent, text alignment, line spacing, etc.
- Tables: Border styles, table formats, etc.
- List items: These are the numbers and bullets that are automatically inserted
There are a few different ways of styling this content in DOCX, which somewhat resemble the HTML/CSS approach. In order of greatest to lowest priority:
- Direct formatting (AKA inline formatting)
- Centrally defined styles (similar to external CSS)
- Document defaults (similar to a
*
rule in CSS)
Unlike CSS, less specific rules don't necessarily override parent rules. The rules are a bit wonky, but if you're interested, see the [advanced formatting section](#Advanced formatting).
This is the type of formatting that your uncle uses when he types out documents: N ... a ... m ... e ... : Then he grabs the mouse, highlights Name: and moves over to the B for bold. This manner of formatting results in markup that is similar to writing <span style="bold: true">Name:</span>
if you were typing out HTML. DOCX (the format) allows you to specify this for any of the four types of items. docx
(the library) only supports this type of formatting for paragraphs and characters, using a fluent api. Thus you could do:
const name = new TextRun("Name:")
.bold()
.font("Calibri")
.allCaps();
Or for paragraph formatting:
const para = new Paragraph("To whom it may concern:")
.heading2()
.center();
DOCX files contain a styles section separate from the main content, much like how HTML includes CSS files. Unlike CSS, DOCX distinguishes between styles meant for tables (which show up in the table formatting toolbar), styles for lists (which show up under bullets and numbering), and styles for runs and paragraphs, which show up as dropdowns offering standard styles, like "Heading 1", "Caption", or any custom styles defined in that document. . docx
allows you to define these styles using a fluent interface as well.
There are three parts to using custom styles with docx
:
-
Create a container object for the style definitions:
const myStyles = new docx.Styles();
-
Define your custom styles, similar to the way you would format a paragraph or run
// The first argument is an ID you use to apply the style to paragraphs // The second argument is a human-friendly name to show in the UI myStyles.createParagraphStyle('myWonkyStyle', 'My Wonky Style') .basedOn('Normal') .next('Normal') .color('999999') .italics() .indent(720) // 720 TWIP === 720 / 20 pt === .5 in .spacing({line: 276}); // 276 / 240 = 1.15x line spacing myStyles.createParagraphStyle('Heading2', 'Heading 2') .basedOn("Normal") .next("Normal") .quickFormat() .size(26) // 26 half-points === 13pt font .bold() .underline('double', 'FF0000') .spacing({before: 240, after: 120}); // TWIP for both
-
When you generate your document, make sure to pass the
styles
container to thePacker
:const packer = new Packer(doc, myStyles); packer.pack(myOutStream);
Note: If you are using the .headingX
or .title
methods of paragraphs, you must make sure to define HeadingX
or Title
styles for these. Otherwise they'll show up unstyled :(. If you are using the .bullet
or .setNumbering
methods, you need to define a ListParagraph
style or the numbers may not show up.
Setting document defaults acts like a *
rule in CSS: it applies to every paragraph and run in the document, but at a low priority level. Other styles affecting this property will override these defaults.
Styles may define a basedOn
attribute that references another style of the same type. In this case, any unspecified properties are inherited from the parent style.
In addition to the 3-layer hierarchy spelled above, there is some interaction between the 4 items that you can style. For instance numbering styles may also specify some styling for paragraphs (typically indentation and tab stops); paragraphs may specify character formatting (e.g., heading font sizes); etc.
The elements that each style may affect are summarized in the table below. So, e.g., table styles may specify table formatting, paragraph formatting, and character formatting.
Style type | Table | Paragraph | List item | Characters |
---|---|---|---|---|
Document defaults | X | X | ||
Table | X | X | X | |
Paragraph | X | X | X | |
Numbering | X | X | ||
Character | X | |||
Direct formatting | X | X | X | X |
To determine the value of a styling property, you must first identify whether it's a table, paragraph, list item, or character property. E.g., numbering definition is a list item property. Then you need to find the last row in the table for which that property has an "X" and the document has formatting of that type. So if a particular run was in a paragraph whose style specified color as FF0000
, but it also had a character style specifying color as 00DD00
, then the character style (lower down on the table) would trump, and the character would have color 00DD00
.
The following properties are treated in a special manner; they're called toggle properties:
- Bold
- All caps
- Small caps
- Italics
- Single strike-through
- Hidden
- Imprint
- Emboss
- Character outline
- Character shadow
For these properties, the rules state the following conflict resolution in case the property is specified at multiple points for the same item:
- Direct formatting trumps all if specified (either true or false)
- Otherwise, if the property is true in document defaults, the property is set to true
- Otherwise, the property's value is an XOR of its effective table, paragraph, and character values. (So specifying bold
true
on a table style and a paragraph style would result in non-bold text if a paragraph inside the table had that style)