Apply a Custom Stylesheet
So far we’ve talked about different ways of using the default stylesheet. In place of the default stylesheet, you can instruct Asciidoctor to use a custom stylesheet of your choosing. On this page, you’ll learn how to apply a custom stylesheet and, if necessary, how to get Asciidoctor to copy and link to that stylesheet into the correct location. You’ll also learn about some of the limitations of this feature when processing multiple documents.
Specify the custom stylesheet
Asciidoctor looks for the stylesheet file specified by the stylesheet
document attribute.
If the value of this attribute is empty (the default), Asciidoctor uses the default stylesheet.
Otherwise, Asciidoctor assumes the value is the path of a custom stylesheet file relative to the source document.
All the same behavior described in Stylesheet Modes still applies, except Asciidoctor uses the specified stylesheet instead of the default stylesheet.
To begin, you must create a stylesheet file. For now, create this file adjacent to your AsciiDoc document. To keep it simple, we’ll just define a basic stylesheet that changes all text to red. Create the file my-stylesheet.css the directory with your AsciiDoc source files and populate it with the following contents:
body {
color: #ff0000;
}
Now let’s use the stylesheet
attribute to apply it.
The stylesheet
document attribute must be set by the end of the header to be effective.
One way to do that is to set the attribute in the document header:
= The Dangers of Wolpertingers
:url-wolpertinger: https://en.wikipedia.org/wiki/Wolpertinger
:stylesheet: my-stylesheet.css
Don't worry about gumberoos or splintercats.
Something far more fearsome plagues the days, nights, and inbetweens.
Wolpertingers.
== Origins
Wolpertingers are {url-wolpertinger}[ravenous beasts].
You can also set stylesheet
using the API or CLI (shown here):
$ asciidoctor -a stylesheet=my-stylesheet.css my-document.adoc
When you view the generated HTML file my-document.html in your browser, you’ll see that all the text is red.
If you want to create a custom stylesheet by extending the default stylesheet, see Extend the default stylesheet. |
As with the default stylesheet, you can set the linkcss
document attribute and Asciidoctor will link to your stylesheet instead.
(Note that it doesn’t copy it in this case since it’s already in the same folder as the output file).
$ asciidoctor -a stylesheet=my-stylesheet.css -a linkcss my-document.adoc
You may not want to keep your stylesheet in the same directory as your AsciiDoc documents. Let’s see how to tell Asciidoctor to look for it elsewhere.
Configure the styles directory
When your stylesheet is in a directory below your AsciiDoc document, you need to tell Asciidoctor where to look to find the stylesheet. There are two equivalent ways to do so:
-
Specify the name of the stylesheet file using the
stylesheet
attribute and the directory where it’s located using thestylesdir
attribute -
Include the directory where the stylesheet is located in the value of the
stylesheet
attribute (instead of using thestylesdir
attribute)
The value of the stylesdir
attribute can end with a trailing directory separator (/
), but it’s not required.
If the linkcss
attribute is not set, the styles directory can be either relative or absolute.
The situation gets trickier when linkcss
is set, which we’ll get to later.
Let’s assume you want to put your stylesheet in the my-styles folder relative to your AsciiDoc document(s).
Go ahead and create that folder and move your custom stylesheet into it for this example.
Now we need to tell Asciidoctor where it’s located using stylesdir
.
The stylesdir
document attribute must be set by the end of the header to be effective.
One way to do that is to set the attribute in the document header:
= The Dangers of Wolpertingers
:url-wolpertinger: https://en.wikipedia.org/wiki/Wolpertinger
:stylesheet: my-stylesheet.css
:stylesdir: my-styles
Don't worry about gumberoos or splintercats.
Something far more fearsome plagues the days, nights, and inbetweens.
Wolpertingers.
== Origins
Wolpertingers are {url-wolpertinger}[ravenous beasts].
You can also set stylesdir
using the API or CLI (shown here):
$ asciidoctor -a stylesdir=my-styles -a stylesheet=my-stylesheet.css my-document.adoc
Asciidoctor now looks for the stylesheet file my-styles/my-stylesheet.css relative to the AsciiDoc document and embeds it into the HTML.
You can achieve the same result by including the directory in the value of the stylesheet
attribute:
$ asciidoctor -a stylesheet=my-styles/my-stylesheet.css my-document.adoc
If you set the value of stylesdir
to an absolute directory, then Asciidoctor would still locate the stylesheet and embed it into the HTML.
But this can create problems if you’ve configured the converter to link to the stylesheet instead.
Let’s look at thoses problem and ways to work through them.
Styles directory and linkcss
Using stylesdir
gets tricky when your linking to the stylesheet instead of embedding it.
That’s because we’re now dealing with two different paths.
One is the path where the stylesheet is located on disk (either absolute or relative to the document).
The other is the path the browser uses to access the stylesheet.
First, it’s important to understand that Asciidoctor mirrors the stylesdir
path in the link reference.
Let’s use the previous example to demonstrate.
If you invoke Asciidoctor as follows:
$ asciidoctor -a stylesdir=my-styles -a stylesheet=my-stylesheet.css -a linkcss my-document.adoc
Then when you inspect the HTML, you will see:
<link rel="stylesheet" href="my-styles/asciidoctor.css">
Notice how the value of stylesdir
(my-styles) is included in the referenced path.
Asciidoctor will also preserve this directory if it needs to copy the stylesheet file.
There are certain situations where this isn’t going to work (at least not when publishing the file to the public internet).
Let’s consider one such case.
We’ll specify the location of stylesdir
as an absolute path and generate the output file into a separate output directory.
$ asciidoctor -a stylesdir=`pwd`/my-styles -a stylesheet=my-stylesheet.css -a linkcss -D public my-document.adoc
Asciidoctor generates the HTML file into the public folder, but it does not copy the stylesheet. Furthermore, it uses an absolute path in the link to the stylesheet:
<link rel="stylesheet" href="/path/to/documents/my-styles/asciidoctor.css">
What we want is for Asciidoctor to copy the stylesheet to the output directory and link to it using a relative path.
A similar problem comes up if you want to control the styles directory and stylesheet file referenced by the HTML independently of the location where they are taken.
In brief, we need to be able to decouple the path where the stylesheet is read from the location where the stylesheet is published and referenced.
That’s where the copycss
attribute comes back into play.
Copy from one place, link to another
For complex combinations that involve stylesdir
, linkcss
, and an explicit output directory, the meaning of stylesdir
is too overloaded and needs to be reconciled.
We can turn to the copycss
attribute to clear this situation up.
This situation is unique to when linkcss is set.
It’s not a problem when the converter embeds the stylesheet since there is no secondary reference involved.
|
The copycss
attribute can accepts a value.
Asciidoctor uses that value as an override for where to look for the stylesheet to read.
The converter, on the other hand, does not use this value.
That means we can use stylesdir
and stylesheet
to assemble the path relative to the output directory where Asciidoctor should write and link to the stylesheet independent of the location where it reads the file.
Let’s revisit the broken scenario from the previous section and use copycss
to reconcile the problem:
$ asciidoctor -a stylesdir=css -a stylesheet=default.css \ -a copycss=`pwd`/my-styles/my-stylesheet.css -a linkcss -D public my-document.adoc
Asciidoctor copies the stylesheet from the absolute path specified by the copycss
attribute to the path public/css/default.css and links to it using the path css/default.css.
Notice that we even changed the name of the folder and stylesheet file in the output.
That demonstrates that we have decoupled the path where the stylesheet is read from the location where the stylesheet is published and referenced.
Styles directory and nested documents when linking
When invoking Asciidoctor on a nested set of documents, it’s currently not possible to specify a single relative path for the stylesdir
attribute that works for all of the documents.
This is because the relative depth of the stylesheet’s location differs for the documents in the subdirectories.
One way to solve this problem is to maintain the path to stylesdir
in each document.
Let’s say you have three AsciiDoc documents saved in the following directory structure:
/my-documents a.adoc b.adoc /my-nested-documents c.adoc /my-styles
For a.adoc and b.adoc, set stylesdir
to:
:stylesdir: my-styles
For c.adoc, set stylesdir
to:
:stylesdir: ../my-styles
If you’re serving your documents from a webserver, you can solve this problem by providing an absolute path to the stylesheet.
You can also try to use the copycss
per document to control where Asciidoctor looks for the stylesheet independent of where Asciidoctor copies it and the converter configured the HTML to reference it.