Skip to main content
Dublin Library

The Publishing Project

CSS Paged Media In The Browser

 

When it comes down to printing web pages we haven't had much control over the output. The browser would just render the page as it is and print it. But with the introduction of CSS Paged Media, we can now control how the page is rendered when printed.

This post will cover how to use CSS Paged Media to control the layout of the printed page and some tricks to make the printed page look better.

How does Paged Media Work? #

CSS Paged Media is a module of CSS that defines how documents are formatted for print or other types of paged media. It allows you to control the layout of the printed page, including page size, margins, headers, footers, and page breaks.

It is an enhancement over traditional print stylesheets, which are limited in their ability to control the layout of the printed page. You start by specifyin the page size and margins using the @page at-rule.

This example will set the page size to 8.5 inches by 11 inches (US letter size) with a margin of 0.5 inches on the top and bottom and 1 inch on the left and right.

@page {
  size: 8.5in 11in;
  margin: 0.5in 1in;
}

You can also specify different page sizes and margins for different types of content using the @page at-rule with a selector. For example, you can set the page size and margins for the title page of a book to be different from the rest of the book.

For example, all the chapter pages will have a running header in the bottom center location of the page.

If there is no specific match for the page we're working with, it'll use the default page settings.

@page chapter {
  @bottom-center {
    vertical-align: middle;
    text-align: center;
    content: element(heading);
  }
}

Paged Media provides 16 locations defined in the Page-Margin Boxes section of the Paged Media specification. In these locations you can place content on the page and are defined using their own at rules. The locations are:

  • @top-left-corner
  • @top-left
  • @top-center
  • @top-right
  • @top-right-corner
  • @left-top
  • @left-middle
  • @left-bottom
  • @right-top
  • @right-middle
  • @right-bottom
  • @bottom-left-corner
  • @bottom-left
  • @bottom-center
  • @bottom-right
  • @bottom-right-corner

You can also define different content for right versus left pages using the :right and :left pseudo-classes. For example, you can set the page number on chapter pages to be on the outside corner of the page for right pages and the inside corner for left pages.

@page chapter:right  {
  @bottom-right-corner { content: counter(page) }
  @bottom-left-corner { content: normal }
}

@page chapter:left {
  @bottom-left-corner { content: counter(page) }
  @bottom-right-corner { content: normal }
}

For blank pages we need to reset any numbering or content that might be on the page using the normal value.

@page chapter:blank { /* Need this to clean up page numbers in titlepage in Prince*/
  @bottom-left-corner { content: normal;}
  @bottom-right-corner {content:normal;}
}

To configure the document for print, we do the following:

  • Use data-type attributes to define specific types of content that we can then target with CSS
  • Use the page attribute in CSS to associate types of pages with specific elements on the page

To set up the default for the book, we add the data-type chapter is set to book.

body[data-type="book"] {
  color: cmyk(0%,0%,100%,100%);
  hyphens: auto;
}

We use the section element to separate the different types of content in the book using the data-type attribute.

To create the title page we use data-type="titlepage" and for the chapters we use data-type="chapter".

We also specify that we want page breaks before any new chapter.

section[data-type="titlepage"] {
	page: titlepage;
}

section[data-type="chapter"] {
  page: chapter;
  page-break-before: always;
}

And any specialized type of content that the book might have.

The table of content is a special case, we can use the nav element with the data-type="toc" attribute to define the table of contents.

The content attribute uses the leader function to add a dotted leader to the page number and a cross-reference to the page number.

/* TOC */
nav[data-type="toc"] { page: toc }
nav[data-type="toc"] ol { list-style-type: none }
/* LEADER CONTENT */
nav[data-type="toc"] ol li a:after { content: leader(dotted) " " target-counter(attr(href, url), page); }

In typography, widows and orphans are single words or short lines of text that appear at the top or bottom of a page or column, separated from the rest of the paragraph.

Widows
A line of text at the end of a paragraph that appears at the top of the next page or column.
Orphans
A line of text at the beginning of a paragraph that appears at the bottom of the page or column.

You can control the number of orphans and widows lines via CSS using the orphans and widows properties.

p {
  orphans:4;
  widows:2;
}

To add running headers and footers to the page, you can add the running function to take the value from the flow of the text and place it in a margin box.

p.rh {
  position: running(heading);
  text-align: center;
  font-style: italic;

}

Current status #

Unfortunately the support for CSS Paged Media is not yet widespread and relies on third-party commercial software to generate PDF results.

Third Party Tools #

Most of the results from CSS Paged Media will produce PDF files as the output.

In my experience, the best tools for generating PDFs from HTML and CSS are Prince XML and Antenna House Formatter. Both tools are commercial products and support the full range of CSS Paged Media features and produce high-quality PDF output.

There is also an online service, Doc Raptor that uses Prince XML to generate PDFs from HTML and CSS online from a variety of languages.

While there are open source tools like WeasyPrint that can generate PDFs from HTML and CSS, they are not as feature-rich their commercial equivalents.

Browser Support #

Can I use CSS Paged Media?
Caniuse results for @page

While support for CSS Paged Media is not yet widespread, it is improving. Most modern browsers support the @page at-rule, but support for other features of CSS Paged Media is still limited.

All Modern browsers support the @page at-rule. Since Chrome 131, you can use generated content to target specific margin blocks on each page.

While we may not get the full functionality of the paged media specifications, we can create more plesing print results than we could before and we can better leverage existing web technologies for our printed experiences.

What we can't do yet #

There are still some limitations to what we can do with CSS Paged Media in the browser. Some of the features that, as far as I'm aware, are not yet supported in browsers are listed below and deal mostly with removing content from the flow of text and placing it elsewhere.

These features don't work in the browser but do work in both PrinceXML and Antenna House Formatter.

set strings #

The example below sets the title of the document in the header (h1 element) of the page.

The string-set property contains one or more pairs, each consisting of an custom identifier (the name of the named string) followed by a content-list describing how to construct the value of the named string.

We then place the resulting doctitle string in the desired margin box.

Similar examples can be used to place page numbers and other content in the margin boxes.

h1 {
  string-set: doctitle content();
}

@page :right {
  @top-right {
    content: string(doctitle);
    margin: 30pt 0 10pt 0;
    font-size: 8pt;
  }
}

Footnotes #

Fotnootes in paged media are a bit tricky. We specify the footnote content in the body of the text, inside span elements. Using float: footnote will remove the content from its original location and place it in the footnote area, usually at the bottom of the page.

Footnote numbers are generated using the counter function and placed in the text using the ::footnote-call pseudo-element. We place a period (.) after the number using the ::after pseudo-element.

We call the footnote using the ::footnote-call pseudo-element using the same counter that we used to generate the footnote number.

span.footnote {
  float: footnote;
}

::footnote-marker {
  content: counter(footnote);
  list-style-position: inside;
}

::footnote-marker::after {
  content: '. ';
}

::footnote-call {
  content: counter(footnote);
  vertical-align: super;
  font-size: 65%;
}

Cross References #

Cross-references in paged media are used to reference other parts of the document, such as figures, tables, or sections. We use the ::after pseudo-element to add the cross-reference to the content of the a element.

a.xref[href]::after {
    content: " [See page " target-counter(attr(href), page) "]"
}

Bookmarks #

Bookmarks are PDF-specific and will appear in the navigation menu. We can use the bookmark-level, bookmark-state, and bookmark-label properties to define the level of the bookmark, whether it is open or closed, and the label of the bookmark.

The h1 element shows the level of the bookmark, whether it's open or closed and what content should be displayed in the bookmark.

The h4 element shows a closed bookmark.

When working with bookmarks they will require prefixed vendor prefixed properties, -prince-* for Prince XML and -ah-* for Antenna House Formatter.

section[data-type="chapter"]  h1 {
  bookmark-level: 1;
  bookmark-state: open;
  bookmark-label: content();
}

section[data-type="chapter"] h4 {
  bookmark-level: 4;
}

Final Notes #

CSS Paged Media is a powerful tool for controlling the layout of printed documents.

For example, we could create PDF versions of all blog posts and then link to them from their respective posts or use services like Doc Raptor (that uses the PrinceXML APIs and supported technologies) to generate PDFs on the fly.

Edit on Github