[[_themes]] == Themes {project_name} provides theme support for web pages and emails. This allows customizing the look and feel of end-user facing pages so they can be integrated with your applications. image::images/login-sunrise.png[caption="",title="Login page with sunrise example theme"] === Theme types A theme can provide one or more types to customize different aspects of {project_name}. The types available are: * Account - Account Console * Admin - Admin Console * Email - Emails * Login - Login forms * Welcome - Welcome page === Configuring a theme All theme types, except welcome, are configured through the Admin Console. .Procedure . Log into the Admin Console. . Select your realm from the drop-down box in the top left corner. . Click *Realm Settings* from the menu. . Click the *Themes* tab. + NOTE: To set the theme for the `master` Admin Console you need to set the Admin Console theme for the `master` realm. + . To see the changes to the Admin Console refresh the page. . Change the welcome theme by using the `spi-theme-welcome-theme` option. . For example: + [source,bash] ---- bin/kc.[sh|bat] start --spi-theme-welcome-theme=custom-theme ---- [[_default-themes]] === Default themes {project_name} comes bundled with default themes in the JAR file `keycloak-themes-{project_versionMvn}.jar` inside the server distribution. The server's root `themes` directory does not contain any themes by default, but it contains a README file with some additional details about the default themes. To simplify upgrading, do not edit the bundled themes directly. Instead create your own theme that extends one of the bundled themes. === Creating a theme A theme consists of: * HTML templates (https://freemarker.apache.org/[Freemarker Templates]) * Images * Message bundles * Stylesheets * Scripts * Theme properties Unless you plan to replace every single page you should extend another theme. Most likely you will want to extend some existing theme. Alternatively, if you intend to provide your own implementation of the admin or account console, consider extending the `base` theme. The `base` theme consists of a message bundle and therefore such implementation needs to start from scratch, including implementation of the main `index.ftl` Freemarker template, but it can leverage existing translations from the message bundle. When extending a theme you can override individual resources (templates, stylesheets, etc.). If you decide to override HTML templates bear in mind that you may need to update your custom template when upgrading to a new release. While creating a theme it's a good idea to disable caching as this makes it possible to edit theme resources directly from the `themes` directory without restarting {project_name}. .Procedure . Run Keycloak with the following options: + [source,bash] ---- bin/kc.[sh|bat] start --spi-theme-static-max-age=-1 --spi-theme-cache-themes=false --spi-theme-cache-templates=false ---- . Create a directory in the `themes` directory. + The name of the directory becomes the name of the theme. For example to create a theme called `mytheme` create the directory `themes/mytheme`. . Inside the theme directory, create a directory for each of the types your theme is going to provide. + For example, to add the login type to the `mytheme` theme, create the directory `themes/mytheme/login`. . For each type create a file `theme.properties` which allows setting some configuration for the theme. + For example, to configure the theme `themes/mytheme/login` to extend the `base` theme and import some common resources, create the file `themes/mytheme/login/theme.properties` with following contents: + [source] ---- parent=base import=common/keycloak ---- + You have now created a theme with support for the login type. . Log into the Admin Console to check out your new theme . Select your realm . Click *Realm Settings* from the menu. . Click on the *Themes* tab. . For *Login Theme* select *mytheme* and click *Save*. . Open the login page for the realm. + You can do this either by logging in through your application or by opening the Account Console (`/realms/{realm name}/account`). . To see the effect of changing the parent theme, set `parent=keycloak` in `theme.properties` and refresh the login page. [NOTE] ==== Be sure to re-enable caching in production as it will significantly impact performance. ==== [NOTE] ==== If you want to manually delete the content of the themes cache, you can do so by deleting the `data/tmp/kc-gzip-cache` directory of the server distribution. It can be useful for instance if you redeployed custom providers or custom themes without disabling themes caching in the previous server executions. ==== ==== Theme properties Theme properties are set in the file `/theme.properties` in the theme directory. * parent - Parent theme to extend * import - Import resources from another theme * common - Override the common resource path. The default value is `common/keycloak` when not specified. This value would be used as value of suffix of `${url.resourcesCommonPath}`, which is used typically in freemarker templates (prefix of `${url.resoucesCommonPath}` value is theme root uri). * styles - Space-separated list of styles to include * locales - Comma-separated list of supported locales There are a list of properties that can be used to change the css class used for certain element types. For a list of these properties look at the theme.properties file in the corresponding type of the keycloak theme (`themes/keycloak//theme.properties`). You can also add your own custom properties and use them from custom templates. When doing so, you can substitute system properties or environment variables by using these formats: * `${some.system.property}` - for system properties * `${env.ENV_VAR}` - for environment variables. A default value can also be provided in case the system property or the environment variable is not found with `${foo:defaultValue}`. NOTE: If no default value is provided and there's no corresponding system property or environment variable, then nothing is replaced and you end up with the format in your template. Here's an example of what is possible: [source,properties] ---- javaVersion=${java.version} unixHome=${env.HOME:Unix home not found} windowsHome=${env.HOMEPATH:Windows home not found} ---- [[_theme_stylesheet]] ==== Add a stylesheet to a theme You can add one or more stylesheets to a theme. .Procedure . Create a file in the `/resources/css` directory of your theme. . Add this file to the `styles` property in `theme.properties`. + For example, to add `styles.css` to the `mytheme`, create `themes/mytheme/login/resources/css/styles.css` with the following content: + [source,css] ---- .login-pf body { background: DimGrey none; } ---- . Edit `themes/mytheme/login/theme.properties` and add: + [source] ---- styles=css/styles.css ---- . To see the changes, open the login page for your realm. + You will notice that the only styles being applied are those from your custom stylesheet. . To include the styles from the parent theme, load the styles from that theme. Edit `themes/mytheme/login/theme.properties` and change `styles` to: + [source] ---- styles=css/login.css css/styles.css ---- + NOTE: To override styles from the parent stylesheets, ensure that your stylesheet is listed last. ==== Adding a script to a theme You can add one or more scripts to a theme. .Procedure . Create a file in the `/resources/js` directory of your theme. . Add the file to the `scripts` property in `theme.properties`. + For example, to add `script.js` to the `mytheme`, create `themes/mytheme/login/resources/js/script.js` with the following content: + [source,javascript] ---- alert('Hello'); ---- + Then edit `themes/mytheme/login/theme.properties` and add: + [source] ---- scripts=js/script.js ---- ==== Adding an image to a theme To make images available to the theme add them to the `/resources/img` directory of your theme. These can be used from within stylesheets or directly in HTML templates. For example to add an image to the `mytheme` copy an image to `themes/mytheme/login/resources/img/image.jpg`. You can then use this image from within a custom stylesheet with: [source,css] ---- body { background-image: url('../img/image.jpg'); background-size: cover; } ---- Or to use directly in HTML templates add the following to a custom HTML template: [source,html] ---- My image description ---- [[_theme_custom_footer]] ==== Adding a custom footer to a login theme In order to use a custom footer, create a `footer.ftl` file in your custom login theme with the desired content. An example for a custom `footer.ftl` may look like this: ``` <#macro content> <#-- footer at the end of the login box -->
``` ==== Adding an image to a email theme To make images available to the theme add them to the `/email/resources/img` directory of your theme. These can be used from within directly in HTML templates. For example to add an image to the `mytheme` copy an image to `themes/mytheme/email/resources/img/logo.jpg`. To use directly in HTML templates add the following to a custom HTML template: [source,html] ---- My image description ---- ==== Messages Text in the templates is loaded from message bundles. A theme that extends another theme will inherit all messages from the parent's message bundle and you can override individual messages by adding `/messages/messages_en.properties` to your theme. For example to replace `Username` on the login form with `Your Username` for the `mytheme` create the file `themes/mytheme/login/messages/messages_en.properties` with the following content: [source] ---- usernameOrEmail=Your Username ---- Within a message values like `{0}` and `{1}` are replaced with arguments when the message is used. For example {0} in `Log in to {0}` is replaced with the name of the realm. Texts of these message bundles can be overwritten by realm-specific values. The realm-specific values are manageable via UI and API. ==== Adding a language to a realm .Prerequisites * To enable internationalization for a realm, see the {adminguide_link}[{adminguide_name}]. .Procedure . Create the file `/messages/messages_.properties` in the directory of your theme. . Add this file to the `locales` property in `/theme.properties`. For a language to be available to users the realms `login`, `account` and `email`, the theme has to support the language, so you need to add your language for those theme types. + For example, to add Norwegian translations to the `mytheme` theme create the file `themes/mytheme/login/messages/messages_no.properties` with the following content: + [source] ---- usernameOrEmail=Brukernavn password=Passord ---- + If you omit a translation for messages, they will use English. . Edit `themes/mytheme/login/theme.properties` and add: + [source] ---- locales=en,no ---- . Add the same for the `account` and `email` theme types. To do this create `themes/mytheme/account/messages/messages_no.properties` and `themes/mytheme/email/messages/messages_no.properties`. Leaving these files empty will result in the English messages being used. . Copy `themes/mytheme/login/theme.properties` to `themes/mytheme/account/theme.properties` and `themes/mytheme/email/theme.properties`. . Add a translation for the language selector. This is done by adding a message to the English translation. To do this add the following to `themes/mytheme/account/messages/messages_en.properties` and `themes/mytheme/login/messages/messages_en.properties`: + [source] ---- locale_no=Norsk ---- By default, message properties files should be encoded using UTF-8. Keycloak falls back to ISO-8859-1 handling if it can't read the contents as UTF-8. Unicode characters can be escaped as described in Java's documentation for https://docs.oracle.com/en/java/javase/17/docs/api/java.base/java/util/PropertyResourceBundle.html[PropertyResourceBundle]. Previous versions of Keycloak supported specifying the encoding in the first line with a comment like `# encoding: UTF-8`, which is no longer supported. [role="_additional-resources"] .Additional resources * See <<_locale_selector,Locale Selector>> for details on how the current locale is selected. [[custom-identity-providers-icons]] ==== Adding custom Identity Providers icons {project_name} supports adding icons for custom Identity providers, which are displayed on the login screen. .Procedure . Define icon classes in your login `theme.properties` file (for example, `themes/mytheme/login/theme.properties`) with key pattern `kcLogoIdP-`. . For an Identity Provider with an alias `myProvider`, you may add a line to `theme.properties` file of your custom theme. For example: + [source] ---- kcLogoIdP-myProvider = fa fa-lock ---- All icons are available on the official website of PatternFly4. Icons for social providers are already defined in `base` login theme properties (`themes/keycloak/login/theme.properties`), where you can inspire yourself. ==== Creating a custom HTML template {project_name} uses https://freemarker.apache.org/[Apache Freemarker] templates to generate HTML and render pages. Although it is possible to create custom templates to change completely how pages are rendered, the recommendation is to leverage the built-in templates as much as possible. The reasons are: * During upgrades, you might be forced to update your custom templates to get the latest updates from newer versions * link:#_theme_stylesheet[Configuring CSS styles] to your themes allows you to adapt the UI to match your UI design standards and guidelines. * link:{adminguide_link}#user-profile[User Profile] allows you to support custom user attributes and configure how they are rendered. In most cases, you won't need to change templates to adapt {project_name} to your needs, but you can override individual templates in your own theme by creating `/