Skip to content

Latest commit

 

History

History
831 lines (569 loc) · 38.9 KB

README.md

File metadata and controls

831 lines (569 loc) · 38.9 KB

Code for America, U.S. Web Design System (USWDS) theme

This theme replaces the Honeycrisp Ruby gem but preserves the style and design. The theme uses “design tokens” from the Honeycrisp Design System to make the USWDS look similar to Honeycrisp. Full documentation on using the theme can be referenced on the corresponding documentation site.

Usage

This readme will cover the basics for working with and contributing to the theme codebase. The following notes are a high-level overview of how to install and use the theme. More detailed instructions will be added in the first official release.

npm install @codeforamerica/uswds
  1. Install this package ^ as a project dependency using npm.

  2. Configure Sass include paths.

  3. Add @forward or @use statements to project Sass entrypoint. Example.

  4. Compile Sass to a static assets/css/ directory.

  5. Add import statements to project JavaScript entrypoint. Example.

  6. Compile JavaScript to a static assets/js/ directory.

  7. Copy the static dist/img/ directory to a static assets/img directory.

  8. Fonts do not have to be copied into the static directory since the theme uses the system font but this would be done if desired. Copy the static dist/fonts/ to a static assets/fonts directory.

  9. Link to static stylesheet in the HTML <head> tag and JavaScript file before the closing </body> tag.

The @codeforamerica/uswds-starter can also be used to bootstrap a front-end development environment that uses this theme, taking advantage of the Node.js scripts that compile Sass and JavaScript.

Contributing

Contributing to the theme requires Node.js, NPM, and Ruby on the system level. Installing Node.js will install NPM at the same time. We recommend using Node Version Manager (NVM), but you may also install Node.js directly. Components are written as Thymeleaf fragment and Embedded Ruby (ERB) partial templates. Thymeleaf is included as a project dependency. Ruby, however, is required on the system level to render ERB files. At the time of writing, the MacOS System Ruby 2.6.10p210 was used. Ruby Version Manager (RVM) is available to help manage Ruby versions.

The project primarily uses NPM scripts to run the Node.js libraries Rollup.js, Sass, 11ty (Eleventy), and ThymeleafJS to generate a static site and USWDS assets.

Table of contents

Getting started

Clone the repository and navigate to the project.

git clone https://github.com/codeforamerica/uswds.git codeforamerica/uswds && cd codeforamerica/uswds

If using NVM, you may set the supported Node.js version number first. If not you may skip ahead.

nvm use

Install dependencies,

npm install

Compile and start the development server. View the site locally at http://localhost:8080

npm start

Workflow

Development

Once the development command is running (npm start) 11ty will watch for changes to view and component templates defined in the watchTargets block of config.js. Separate tasks for watching for changes to Sass and JavaScript files are also activated.

Most changes will be made to files within the src/ or packages/ directories. Once a change is made to a source file it will be compiled into the dist/ directory. The local development server will refresh to reflect the change.

Flows and general style guides

Potential changes can be broken down into the following flow categories:

Modifying documentation or content

All content and context (strings, classes, HTML) passed to rendered components occurs in the src/views/ directory. There may be some special cases where a string is hard-coded into a component template but generally this is discouraged. The documentation is written in MarkdownIt + LiquidJS syntax (11ty pre-processes MarkdownIt templates using LiquidJS, hence, the syntax is combined) or Twig. Twig is minimally used and appears in layouts or partials. The language is supported because USWDS component templates are written using Twig.

Pages use front matter, or a YAML block, to define variables for the layout or page contents. This makes it easier to modify short strings deeply nested in the content. Several content blocks for each page are "templatized" to render automatically using variables. The variables are defined in each page, either through front-matter or other, and passed to partials in the views/_partials/ directory. For example, the details section for each component use a partial rendered with the shortcode {% render 'details.md' name: title ... %}.

Component demonstrations in the documentation use context to define their display. Context may be strings containing class names, plain text, or HTML, and booleans. Context is organized using a JSON object set in a capture block. For example, the Accordion component uses the following context:

{% capture context %}{
  "modifier": "cfa-accordion usa-accordion--bordered",
  "multiple": true,
  "items": [
    {
      "expanded": true,
      "controls": "aria-c-{% createId %}",
      "heading": "We'll ask you about",
      "body": "{% md body_accordion_1 %}"
    },
  ...

Which is passed to the figure.md partial. This partial will create the figure that renders the live demonstration of the component paired the HTML source.

{% render 'figure.md', name: 'accordion', nice: 'Bordered', context: context, caption: 'Bordered' %}

On the backend, the context is passed to both the Accordion component's Thymeleaf fragment and Embedded Ruby partial templates (in the packages/cfa-accordion/ directory). It will then render unescaped and escaped HTML and inline it into the static HTML output. In the production build, only the Thymeleaf fragment is displayed.

Built-in Liquid template methods are used to enhance the functionality of Markdown. These are denoted inside the {% ... %} brackets. There are also several custom shortcodes added to the 11ty configuration for this site to assist various parts of the documentation display for the theme.

Config.js

Several packages used by the documentation site are customized in the config.js file.

Modifying or adding a USWDS theme setting

While the src/scss/ directory contains the entry points for Sass, the the main theme-level settings for the USWDS are set in packages/cfa-uswds-theme/_index.scss. These settings are actively extended by the USWDS core in the packages/cfa-uswds/_index.scss. For example, the setting for the theme focus color in cfa-uswds-theme:

$cfa-focus-color: 'gold-30v' !default;

will map to the cfa-uswds package core setting map:

@use 'cfa-uswds-theme' as *;

@use 'uswds-core' with (
  $theme-focus-color: $cfa-focus-color,
  // ...

This set up allows users of the theme only customize the theme settings which is a much smaller subset of the full USWDS settings configuration. If a current setting needs to be modified it can be done in the cfa-uswds-theme package. If a new USWDS setting needs to be added, set the $cfa- version of it in the cfa-uswds-theme package and map it to it's corresponding USWDS setting the cfa-uswds package.

There is a smaller set of package-level settings used by components in this theme. These settings aren’t supported by the USWDS but are used by other component packages. Those are set in the cfa-core package.

All theme and package-level settings are documented here.

Modifying a CfA component

Each component and its relevant code are stored together as packages/ in the directory of the same name. Packages may contain template files for Thymeleaf and ERB to define the markup, Sass to define custom styles that aren't supported by the USWDS configuration, or JavaScript if it's necessary to enable functionality. Each of these concerns are separated into their own file.

— 📂 packages
  └ 📁 cfa-*
    ├ 📄 *.th.html    —  A Thymeleaf template component file for storing component markup.
    ├ 📄 _*.html.erb  —  An Embedded Ruby partial template file for storing component markup.
    ├ 📄 _*.scss      —  A Sass styling file for overriding or extending USWDS styles.
    └ 📄 *.js         —  A JavaScript file for functionality (currently, not to be confused with web components).

In place of the asterisk (*) is the component name. Names are modelled after USWDS counterparts that they represent or override. Names are only customized if they do not exist in the USWDS (ex, details or follow-up-question). It's important that names are consistent across package directories, files, and their corresponding documentation page.

**📄 *.th.html

Thymeleaf fragments are "componentized" templates for the Java template engine. This syntax models "natural templating" or writing thats feels similar to HTML with an added binding syntax through custom HTML attributes. This makes it more formal and less flexible. Passing variables to these bindings fills the standard HTML attributes and string content within HTML tags. To test the rendering of Thymeleaf the theme uses ThymeleafJS. This is important to note because this implementation of the language is limited. Read more about the Thymeleaf template language on the documentation site.

Only USWDS classes prefixed with .usa- are hard-coded into templates. Theme modifiers prefixed with .cfa- are passed to the fragment to achieve theme styling. For example, to append the cfa-accordion class to the usa-accordion, the modifier parameter needs to be set.

<div th:fragment="accordion(modifier, multiple, items)" class="usa-accordion" th:classappend="${modifier}" th:attr="data-allow-multiple=${multiple}" th:each="item: ${items}">
  <!-- ... -->

Fragments will nest other components instead of re-creating them. This enables full customization of every nested component. For example, the Form Group component will pull in the Input component.

<th:block th:replace="~{packages/cfa-input/cfa-input.th :: input(${input}, ${inputGroup})}" />
📄 _*.html.erb

Embedded Ruby partial templates are scripted Ruby files with HTML and Ruby code intermixed. There is no binding syntax for HTML so it is less formal but more flexible. It is much easier to write Ruby code within templates which makes it less "componentized." This application doesn't operate inside of a Ruby on Rails environment so there are no helper methods, most notably the render 'partial-name' method for including other partials. The lowest level and simplest version of Rails render is ERB.new(File.read('partial-name'), 0, 0, [*('a'..'z')].sample(8).join).result_with_hash({key: 'value'}) which can be cumbersome to manage. Read more about the ERB class on the Ruby documentation site.

The same rules for modifiers and nesting other components applies to ERB templates.

<div class="usa-accordion<% if defined?(modifier) %> <%= modifier %><% end %>"<% if defined?(multiple) %> data-allow-multiple<% end %>>
  <!-- ... -->

Nesting another component:

<%= ERB.new(File.read('packages/cfa-input/_cfa-input.html.erb'), 0, 0, '@input')
  .result_with_hash({
    input: input,
    inputGroup: (inputGroup if defined?(inputGroup))
  }.compact) %>

In some cases the string @input in the example above needs to be unique. It can be replaced with [*('a'..'z')].sample(8).join to always create a random string.

📄 _*.scss

Dart Sass modules extend the styling for custom or USWDS components that can't be styled using USWDS settings alone. Where possible, distinguish between theme styles and USWDS styles. Always prefer USWDS Sass theme tokens, utilities, functions, and other mixins over custom methods. This ensures that custom styling remains within the design system.

Use the Block, element, modifier (BEM) naming convention. Always prefix theme classes with .cfa-. Use the same name as the component for the block selector. Read more about the BEM naming convention.

.cfa-block { }
.cfa-block__element { }
.cfa-block--modifier { }

Use the CSS .cfa- prefix for custom styles. This ensures clarity when using custom theme styles versus USWDS defaults. Do not add style overrides dependent on .usa- classes as the root class.

.cfa-button:not(.usa-button--unstyled) {
  // ...

Reuse settings variables defined by theme settings. They are always prefixed by $theme-. Often, they need to be passed to utility functions that will interpret the token value. Read more about design tokens on the USWDS documentation site.

@use 'uswds-core' as *;

.cfa-button.usa-button--big:not(.usa-button--unstyled) {
  font-size: font-size($theme-button-font-family, 'lg');
  // ...

Always use the units() function to convert unit design tokens to the spacing dimensions defined by the system. Never hard-code pixel values. Read more about spacing unit tokens on the USWDS documentation site.

@use 'uswds-core' as *;

.cfa-button:not(.usa-button--unstyled) {
  padding: units(2) units(2.5);

  // ...

Use the set-text-and-bg mixin when defining a background color and foreground text color. This ensures color combinations are run through the USWDS contrast checker. Combinations that don't have sufficient contrast (WCAG AA 2.1) will emit an exception to stdout. Read more about color on the USWDS documentation site.

@use 'uswds-core' as *;

.cfa-button.usa-button--outline:not(.usa-button--unstyled):not([disabled]):not([aria-disabled=true]) {
  @include set-text-and-bg(
    $bg-color: $theme-body-background-color,
    $preferred-text-color: 'ink',
    $context: 'CfA Button'
  );

  // ...

Use theme and state color tokens over system color tokens. This ensures that the color theme has continuity throughout components. Read more about color. Read more about each color token types on the USWDS documentation site:

@use 'uswds-core' as *;

.cfa-button--no {
  > svg {
    fill: color('error');

  // ...

.cfa-button--yes {
  > svg {
    fill: color('success-darker');

  // ...
📄 *.js

ES JavaScript modules provide functionality to custom components or provide a utility wrapper around external libraries. They are written using as standard modules and defined using class templates.

  • Components properties and methods are be encapsulated. Writing properties to global variables, such as the window, is discouraged. Write custom properties to the component instance, instead.
  • All JavaScript should be considered a progressive enhancement. There should be a suitable default state for the component if the page does not invoke JavaScript. ARIA properties, such as aria-hidden will only be toggled when the component is instantiated.
  • Use the data-js to define selectors for components. Use aria-controls to define target element selectors. Follow guidance defined by the ARIA Authoring Practices Guide.

Always use a class to define a component (using the component name, capitalized and camel-cased):

class Component {
  constructor() {

    // ...

export default Component;

Using, or, creating a class instance:

import Component from '@codeforamerica/uswds/packages/cfa-component/cfa-component.js';

new Component();

Classes can be easily extended or have settings configurations passed to them to enable, disable, or modify internal functionality. Methods and properties within them can be overridden or accessed individually. This variety makes them flexible to work with.

import Component from '@codeforamerica/uswds/packages/cfa-component/cfa-component.js';

// Set a component property before it is initialized.
Component.property = 'value';

// Initialize a component with custom values.
let component = new Component({
  property: 'value',
  method: () => {}
});

// Call a component method after initialization.
component.internalMethod();

Static properties used within the class are defined before the export. They are often set for the class within the constructor with an optional settings override that can be passed upon instantiation.

class Component {
  constructor(s = {}) {
    this.selector = s.selector ? s.selector : Component.selector;

    // ...

}

Component.selector = '[data-js="component"]';

// ...

export default Component;

Click, change, or other event listeners are added to the body. This makes functions less dependent on individual elements and more compatible with reactive frameworks.

class Component {
  constructor() {

    // ...

    document.querySelector('body')
      .addEventListener('click', event => {
        if (event.target.matches(this.selector)) {
          this.internalMethod(event.target);
        }
    });

    // ...

Classes may be instantiated once for all elements they concern.

import Component from '@codeforamerica/uswds/packages/cfa-component/cfa-component.js';

new Component();

Or, they may be instantiated per element. However, the main selector for the component should always be stored as a static property.

(elements => {
  for (let i = 0; i < elements.length; i++) {
    new Component(elements[i]);
  }
})(document.querySelectorAll(Component.selector));

Below is the general structure for theme component classes.

'use strict';

class Component {
  /**
   * Component constructor
   *
   * @param   {Object}  s  Optional settings configuration
   *
   * @return  {Object}     Instance of Component
   */
  constructor(s = {}) {
    /**
     * Instance setting overrides. If any of the properties are passed in the
     * s object, they are used by the instance instead of the default properties.
     */

    this.selector = s.selector ? s.selector : Component.selector;

    this.externalMethod = s.externalMethod ? s.externalMethod : Component.externalMethod;

    /**
     * Initialization for the progressively enhanced features
     */

    this.init();

    /**
     * Main event listener for the component
     */

    document.querySelector('body')
      .addEventListener('click', event => {
        if (event.target.matches(this.selector)) {
          this.internalMethod(event.target);
        }
    });

    return this;
  }

  /**
   * Initializes the component, setting progressively enhanced
   * ARIA attributes or other relevant dynamic templates.
   *
   * @return  {Object}  Instance of Component
   */
  init() {

    // ...

    return this;
  }

  /**
   * An internal class method.
   *
   * @param   {Object}  target  Method argument
   *
   * @return  {Object}          Instance of Component
   */
  internalMethod(target) {

    // ...

    return this;
  }
}

/** @type  {String}  The main selector for the Component event listener */
Component.selector = '[data-js="component"]';

/**
 * An external class method
 */
Component.externalMethod = () => {}

export default Component;

Notes on external dependencies

Components may have external dependencies if they provide a very substantial or specific benefit necessary for that component to function.

  • Do not add any design system other than the USWDS (Bootstrap, Material, or other). Design systems are not intended to be used together. The USWDS also provides its own set of CSS utilities so Tailwindcss is also not supported at this time.

  • Do not install jQuery. It is not needed for most modern JS applications and it creates an oversized dependency if one component uses it. Use plain JavaScript.

  • Do not install Underscore or LoDash. Use standard Array or Object prototypes.

  • The use of reactive front-end frameworks are not supported at this time (React.js, Vue.js, or other).

New dependencies should be installed as regular dependencies using npm install so that other projects can inherit them as well.

Modifying the compilation of Sass or JavaScript

The source, distribution, and plugins that transform Sass or JavaScript can be modified in the entrypoints.js file. Each entrypoint is defined as an object {} in the export.

module.exports = [
  {
    /* module settings */
  },
  {
    /* module settings */
  },
  {
    /* module settings */
  }
];

Each object may either be a style or script module. The difference is determined by the extension of the input file.

Style modules

CSS is translated from the source using styles.js. It uses the Dart Sass JavaScript API to compile Sass to CSS. The CSS is then run through a series of PostCSS plugins to optimize the output:

  1. purgecss. Removes unused CSS (currently unused but will be supported).
  2. autoprefixer. Adds vendor prefixes to CSS rules.
  3. mqpacker. Combines all media queries that are the same.
  4. cssnano. Compresses CSS.

The attributes for style modules are documented in the entrypoints.js file in more detail but generally looks like the following:

{
  input: 'src/scss/_styles.scss',
  output: [{
    file:  'dist/css/styles.css',
    options: {
      sourceMap: true,
      loadPaths: [ /* ... */ ]
    }
  }],
  plugins: [ /* ... */ ]
}
Script modules

JavaScript is translated from the source using scripts.js. It uses the Rollup.js to combine concatenate scripts and perform some transformations. The JavaScript is then run through a series of plugins to translate the output:

  1. commonJs. Enables the support for CommonJS module import.
  2. nodeResolve. Adds support for the Node.js module resolution algorithm.
  3. replace. Performs string transformations on the JavaScript output.

The attributes for script modules are documented in the entrypoints.js file in more detail but generally looks like the following:

{
  input: 'src/js/index.js',
  output: [{
    file:  'dist/js/default.js',
    options: {
      sourceMap: true,
      loadPaths: [ /* ... */ ]
    }
  }],
  plugins: [ /* ... */ ]
}
Module plugins

When contributing to the theme, plugins should only be added as default plugins (defined above) if all modules benefit from it. They can be added to either styles.js or scripts.js directly. Which and what order plugins run can be configured for each module object in the plugins attribute. For example, a styles module with this set will run all the default plugins:

plugins: [
  'purgecss',
  'autoprefixer',
  'mqpacker',
  'cssnano'
]

Omit a plugin by commenting it out:

plugins: [
  // 'purgecss',
  'autoprefixer',
  'mqpacker',
  'cssnano'
]
Config.js

Several paths, entrypoint names, distribution strings, and Node module load paths for the theme are defined in the config.js file.

External projects

External projects can use these module scripts to compile their own Sass and JavaScript. This is documented in the @codeforamerica/uswds-starter. They need to set up their own NPM project, entrypoints.js file, and project source. This needs to be kept in mind when updating the theme's module scripts and their configuration.

They can use the plugins attribute in their entrypoints.js file to choose default plugins or configure their own plugins. For example, the purgecss plugin can be configured by swapping the plugin string with an instance of the plugin (using instructions from the purgecss documentation):

plugins: [
  purgecss({
    content: [
      path.join(__dirname, 'dist/**/*.html')
    ],
  }),
  'autoprefixer',
  'mqpacker',
  'cssnano'
]

New plugins should be installed as regular dependencies using npm install so that other projects can inherit them as well.

Committing changes and publishing

After development, changes to the theme need to be staged, committed, tagged, and published to the NPM registry for other projects to inherit. Create a pull request with your changes for review. They will be squashed and merged into the main branch.

Publishing

The simple workflow for this uses two NPM scripts: version and publish. They must be run sequentially. Choose an appropriate semantic version increment: either major, minor, or patch. Then pass that to the version command. This will generate a new build for the package and production documentation site. The force flag, -f will force the staging of changes to the version commit.

npm version {{ major/minor/patch }} -f

Once the build and version commit and tag are complete, run the publish command to push changes to the remote. You will need to be authenticated to NPM to run this command by running the login command first.

npm login

If you are already authenticated you may skip the command above and run:

npm publish

This will push changes to the remote repository, publish the latest version of the documentation site to GitHub pages, and publish the latest package to the NPM registry.


Project directory and file structure

— 📁 dist                —  All static files for the site are distributed here, including HTML, CSS, JavaScript, SVGs, images, and font files.
├ 📂 src                 —  Folder for source files such as Sass and JavaScript modules.
  ├ 📂 views             —  Folder for 11ty template views.
    ├ 📁 _layouts        —  This folder contains layouts extended by other view files.
    ├ 📁 _partials       —  This folder contains partials included in other view files.
     └ 📄 **/*.md        —  View files are usually Markdown files. Directories and file names reflect the URL pattern for the static site pages.
  ├ 📂 scss              —  Folder for Sass entry points.
    ├ 📄 _site.scss      —  Entry point for the site and theme styles are imported and configured.
    └ 📄 _styles.scss    —  Entry point for only the theme styles are imported and configured.
  └ 📂 js                —  Folder for JavaScript entry points.
    └ 📄 index.js        —  Entry point for where site and theme scripts are imported and configured.
├ 📂 packages            —  Folder for CfA theme components, including USWDS component stylesheets and templates for Thymeleaf and ERB.
  ├ 📁 cfa-uswds-theme   —  This package includes the main theme settings for the USWDS.
  ├ 📁 cfa-uswds         —  This package actively extends the USWDS core settings with the theme settings.
  ├ 📁 cfa-core          —  This package stores settings that aren’t supported by the USWDS but are used by other component packages.
  ├ 📁 cfa               —  This package imports all of the theme component packages.
  └ 📁 cfa-*             —  All additional *packages* are theme components or utilities.
    ├ 📄 *.th.html       —  A Thymeleaf template component file for storing component markup.
    ├ 📄 _*.html.erb     —  An Embedded Ruby partial template file for storing component markup.
    ├ 📄 _*.scss         —  A Sass styling file for overriding or extending USWDS styles.
    └ 📄 *.js            —  A JavaScript file for functionality (currently, not to be confused with web components).
├ 📄 config.js           —  Configuration for the theme, including base path definitions for static and source files.
├ 📄 eleventy.config.js  —  Configuration file for the 11ty site.
├ 📄 entrypoints.js      —  Defines the configuration for Sass and JavaScript modules and their distribution.
├ 📄 styles.js           —  Node.js script for generating CSS from Sass using Sass and PostCSS.
├ 📄 scripts.js          —  Node.js script for bundling JavaScript modules using Rollup.js.
├ 📄 package.json        —  Defines the theme package for NPM, manages dependencies, and CLI scripts.
└ 📄 .nvmrc              —  Used to define the Node.js latest version number supported during development.

NPM Scripts

The following NPM scripts, which are CLI commands, are available to run the primary development functions of the application.

Start

Compile styles, scripts, views, and start the 11ty server. Watch for changes to relevant files.

npm run start

Default

Compile styles, scripts, views. Runs once on relevant files.

npm run default

Views

Start the 11ty server. Watches for changes to relevant view and component templates.

npm run views

Scripts

Watch for changes to JavaScript files then compile JavaScript using Rollup.js.

npm run scripts

Styles

Watch for changes to Sass files then compile Sass using its built-in JavaScript API.

npm run styles

Production

Compiles views, styles, and scripts for production.

npm run production

Preview

Compiles views, styles, and scripts for production and run the 11ty server in "production mode." (note, this will pull assets from production).

npm run preview

Version

This script hooks into the default npm version command to compile views, scripts, and styles for production, and staging them for commit before NPM creates a version commit and Git version tag.

This script doesn't need to run on its own, just run npm version {{ semantic version }} when you are ready to commit a new version of the package to the repository before publishing.

npm run version

Pre-publish

This script hooks into the default npm publish command to push local commits and tags to remote before NPM publishes the latest version to its registry.

This script doesn't need to run on its own, just run npm publish when you are ready to publish the latest version of the package.

npm run prepublishOnly

Publish

This script hooks into the default npm publish command to push the dist/ directory to the GitHub Pages branch, gh-pages, which is the "production" environment for the site.

This script doesn't need to run on its own, just run npm publish when you are ready to publish the latest version of the package.

npm run publish

Configuration

The package.json contains configuration for the package name, public homepage URL, design file, and other NPM specifics.

The config.js file contains common configuration for the 11ty site and compilation path definitions for source Sass and JavaScript entry points.

The entrypoints.js contains configuration for source Sass and JavaScript entry point modules. Default plugin settings for PostCSS and Rollup.js are defined in the styles.js and scripts.js files.


11ty

The eleventy.config.js defines the customized functionality available to templates. The main additions are library and plugin configuration, custom shortcodes, global data, and collections. Shortcodes are noted below but documented in more detail in the eleventy.config.js file.

Component

Render a component. The JSON context fills the template slots with the values used by the template. The first boolean value determines whether to return rendered (false, default) or escaped (true) HTML. The second boolean determines wether to return both the rendered default Thymeleaf template (false) or both Thymeleaf and ERB templates (true).

{% component 'name' 'context (JSON String)' false false %}

Markdown

Compile a Markdown string to HTML.

{% md 'String (Markdown String)' %}

Thymeleaf

Retrieve an escaped Thymeleaf template. The boolean value determines whether to return the template fragment contents (false, default) or the template fragment include statement (true).

{% thymeleaf 'name' false %}

Embedded Ruby

Retrieve an escaped Embedded Ruby template partial. The boolean value determines whether to return the template contents (false, default) or ERB partial render statement (true).

{% erb 'name' false %}

Block

Create a code block demonstration from the supplied string. Supported syntax types are defined in the HighlightJs block in the config.js file.

{% block 'type' 'String' %}

Get File

Retrieve a package file path for a package by its type.

{% getFile 'name' 'String (value of thymeleaf, erb, stylesheet, or javascript)' %}

Create ID

Generate a random string to use as a unique ID.

{% createId %}

Create Slug

Generate a slug (URL friendly string) from a human readable string ("My String" would become "my-string").

{% createSlug 'string' %}