Gesso is a Sass-based starter theme that outputs accessible HTML5 markup. It uses a mobile-first responsive approach and leverages SMACSS to organize styles. This encourages a component-based approach to theming through the creation of discrete, reusable UI elements. Gesso is heavily integrated with Storybook and the Component Libraries module, allowing Drupal and Storybook to share the same markup.
Visit the Gesso Storybook demo site.
For more information, view the Gesso Drupal project page or Gesso GitHub repo. To submit bug reports or feature requests, visit the Gesso issue queue.
The following packages need to be installed on your system in order to compile and use Gesso.
-
Place the Gesso theme in your site’s theme directory. (e.g., themes/gesso) Read documentation on installing themes for more information.
-
Enable the Gesso Helper module. This module comes packaged with the theme, but must be manually enabled for the theme to function.
-
Install the Component Libraries module. Since many of the Drupal templates reference twig files inside Storybook using Twig namespaces, this module is required for the theme to function.
-
Install the Twig Tweak module.
-
Optional: Install the Twig Field Value module. This is not required, but it can make working with Twig templates easier. Please note, however, that using the
|field_value
Twig filter from this module will break Drupal’s QuickEdit functionality. -
Optional: Install the Background Images Formatter module and its Responsive Background Images Formatter submodule. This is not required, but it will allow you to use images uploaded to Drupal as background images, with different image sizes at different breakpoints.
Because Gesso is a starter theme, you may want to rename the Gesso directory or copy its contents to a new custom theme directory based on the name of your project.
The easiest way to accomplish this is to use
Drush. Type drush help gesso
for more
information. If you get an error that the gesso
command is not defined, make
sure you have enabled the Gesso Helper module.
If you can’t use Drush, then manually replace all instances of gesso
within
this directory with a machine-readable name of your choice, including folder
names, filenames, and all occurrences within files. This custom name must start
with a letter and may only contain lowercase letters, numbers, and underscores.
Edit the .info.yml
file and update the theme name and description. You can
also change the screenshot image (images/screenshot.png
) shown on the
Appearance admin page.
For development, you can set the theme up as part of a Drupal site or work only in Storybook. Gesso includes npm tasks to compile design tokens, CSS, JS, Storybook, and the SVG sprite using webpack.
To use these tasks, first run the following npm command in the theme folder to install node dependencies.
npm i
To compile the theme, start Storybook, and watch for changes run the following command in the theme directory:
npm run dev
Open localhost:6006 to view Storybook. If you’re using Docker (or some other container engine) for local development, this might be mapped to a custom domain or a port on a custom domain such as storybook.ddev.site or site.ddev.site:6006.
If you add new SCSS and/or JS files, you will need to restart webpack by
canceling and then re-running npm run dev
. New files will not be processed
until webpack restarts. Errors will also be shown for duplicate filenames.
To initiate the build tasks only (without watching for changes), run the following command in the theme directory:
npm run build
Run npm run component
to create boilerplate files for a new component. This is
the recommended approach as it will set up basic Twig and Storybook files that
you can modify.
Name your stories files [component].stories.jsx
. See menu.stories.jsx
for
an example.
To match Storybook to your site’s branding, change the colors in
.storybook/manager.js
. Any fonts can be added in
.storybook/manager-head.html
. See the Storybook
docs for more
information about and examples of theming.
Sass can be compiled as part of the global styles.css file or to individual CSS files for use in a Drupal library.
@use
is used to import Sass variables, mixins, and/or functions into
individual SCSS files. @import
is discouraged by the Sass team and will
eventually be phased out..
This means that most files will start with @use '00-config' as *;
. This allows
you to use the design token accessor functions without an additional namespace.
Other functions and mixins can be used similarly. Note that to avoid namespace
collisions, only Gesso-related variables, mixins, and functions should be used
with *
.
All Sass files that are compiled to individual CSS files must have a unique filename, even if they are in different directories.
Prefix the name of your Sass file with _
, e.g. _card.scss
. Add it to the
appropriate aggregate file (i.e. _components.scss
).
DO NOT prefix the name of your Sass file with _
, e.g. menu.scss
. Import the
config and global aggregate files. Import your SCSS file at the top of your
Storybook file. See dropdown-menu.stories.jsx
for an example. Don’t forget to
add it to the gesso.libraries.yml
file as well.
Stylelint and Prettier are used to lint CSS and SCSS files. Warnings will break the build, so if you have a valid reason to break Stylelint rules you can have it ignore code in two ways:
-
Add
// stylelint-disable-next-line
to the line just before where the Stylelint warning is triggered. -
To ignore several lines, add
// stylelint-disable
before the code in question and add// stylelint-enable
afterwards.
In both cases above, please add a comment about the valid reason to disable the Stylelint rule(s) in your use case.
The Stylelint rules can be changed in the .stylelintrc.yml
file. By default,
Gesso follows the
sass-guideline.es
and Prettier’s recommended
guidelines, with some
additional customizations.
The Prettier config can be changed in the .prettierrc
file.
JavaScript can be compiled to individual JS files for use in a JavaScript
library or included within a different JS file. JS files that use modern
(ES2015+) syntax must be named [name].es6.js
, but this is not required by the
compiler. JavaScript files should go in the appropriate folder under source
(e.g., source/03-components/menu
for menu-related JavaScript). There is not a
separate folder for JS files as there was in previous versions of Gesso.
All JavaScript files must have a unique filename, even if they are in different directories.
Prefix the name of your JavaScript file with _
, e.g. _Menu.es6.js
. Import it
to the appropriate JavaScript file(s), (i.e. primary-menu.es6.js
).
DO NOT prefix the name of your JS file with _
. Import your JS file at the top
of your Storybook file. See dropdown-menu.stories.jsx
for an example. Don’t
forget to add it to the gesso.libraries.yml
file as well.
Any library you create in gesso.libraries.yml
that includes an individual
component script must include gesso/common
as a dependency. (In most cases, you
will also add core/drupal
as a dependency, if you are using the Drupal
object anywhere in your code.) common.js is generated on production builds
(so you will not notice it missing until you deploy to a staging server) and
contains JavaScript that is shared across two or more components, so that it is
not bundled multiple times on the page. The recommended practice is for each
library to declare its dependencies, even if some of them are repeated across
multiple libraries and/or shared with global. This ensures that Drupal will
always load the dependencies before loading any library that depends on them.
See the dropdown_menu
library in gesso.libraries.yml
as an example.
The common JS file is created using the Webpack SplitChunksPlugin.
To change how it behaves, update webpack.production.js
. You may also need to
update gesso_library_info_build
in libraries.inc
to change what files are
included in the gesso/common
library. We recommend using the default setup
unless you have a specific use case that requires advanced configuration.
ESLint and Prettier are used to lint JavaScript files. If you have a valid reason to break one of the rules, you can ignore a specific line using any of the options in the ESLint documentation.
Please add a comment about the valid reason to disable the ESLint rule(s) in your use case.
The ESLint config can be changed in the .eslintrc.cjs
file. Gesso follows the
Airbnb standards, which are followed
by Drupal
as well.
The Prettier config can be changed in the .prettierrc
file.
Gesso itself does not include any jQuery dependencies and does not ship with jQuery. However, some Drupal modules still rely on jQuery, so you may need to add it if, for example, you need to create and trigger a jQuery event.
To add jQuery to Storybook:
- Install jQuery with
npm i -D jquery @types/jquery
. - Add jQuery to
config.externals
in lines 78-82 of.storybook/main.js
config.externals = { drupal: 'Drupal', drupalSettings: 'drupalSettings', once: 'once', jquery: 'jQuery', };
- Add a jQuery stub similar to
stubs/once.js
and import it in.storybook/preview.js
import jQuery from 'jquery'; window.jQuery = jQuery;
import './stubs/jquery.js'
To add jQuery to Drupal:
- Add jQuery to
externals
in lines 170-174 ofwebpack.common.js
externals: { drupal: 'Drupal', drupalSettings: 'drupalSettings', once: 'once', jquery: 'jQuery' }
- Ensure that
core/jquery
is added a dependency of the appropriate library in gesso.libraries.ymllibrary_name: js: dist/js/file-that-uses-jquery: {} dependencies: - gesso/common - core/drupal - core/once - core/jquery
You can then import jQuery at the top of a file, the same way Drupal
and once
are typically imported, and use it as needed.
Gesso uses the configuration file source/00-config/config.design-tokens.yml
to
manage the theme’s design tokens. The npm build and dev tasks will automatically
generate a global Sass map to easily pull design tokens into individual SCSS
files.
The following Sass functions can be used to access the tokens defined in
config.design-tokens.yml
.
Output a shadow value from the box-shadow token list.
box-shadow: gesso-box-shadow(1);
Output a size value from the breakpoints token list.
@include breakpoint(gesso-breakpoint(desktop)) {
display: flex;
}
@include breakpoint-max(gesso-breakpoint(mobile), true) {
display: none;
}
@include breakpoint-min-max(
gesso-breakpoint(mobile),
gesso-breakpoint(tablet),
true
) {
display: block;
}
Output a color value from the palette brand token list.
color: gesso-brand(blue, light);
Output a color value from the colors token list.
color: gesso-color(text, primary);
Output a size value from the constrains token list.
max-width: gesso-constrain(sm);
Output a timing value from the transitions duration token list.
transition-duration: gesso-duration(short);
Output an easing value from the transitions ease token list.
transition-timing-function: gesso-easing(ease-in-out);
Output a stack value from the font-family token list.
font-family: gesso-font-family(primary);
Output a size value from the font-size token list.
font-size: rem(gesso-font-size(2));
Output a weight value from the font-weight token list.
font-weight: gesso-font-weight(semibold);
Output a color value from the palette grayscale token list.
color: gesso-grayscale(gray-2);
Output a height value from the line-height token list.
line-height: gesso-line-height(tight);
Output a size value from the spacing token list.
margin-bottom: rem(gesso-spacing(md));
Output an index value from the z-index token list.
z-index: gesso-z-index(modal);
The values in Gesso’s configuration file are also exported to JavaScript objects
so that the same values can be used in CSS and JS. The JS objects can be found
in source/00-config/_GESSO.es6.js
. This file is also rebuilt whenever
npm run dev
or npm run build
are run.
For example, to use a breakpoint in a script:
import { BREAKPOINTS } from '../../../00-config/_GESSO.es6';
if (window.matchMedia(`min-width: ${BREAKPOINTS.desktop}`).matches) {
// Some script that should only run on larger screens.
}
This will use the same breakpoint as breakpoint(gesso-breakpoint(desktop))
in
your Sass.
Gesso uses custom mixins to specify viewport width based media queries:
breakpoint
: min-width queriesbreakpoint-max
: max-width queriesbreakpoint-min-max
: queries with both a min and max width
Each mixin takes one or two width parameters, which can be a straight value
(e.g., 800px, 40em) or a design token value called using the gesso-breakpoint
function (e.g., gesso-breakpoint(tablet-lg)
). The breakpoint-max
and
breakpoint-min-max
mixins can also take an optional parameter to subtract one
pixel from the max-width value, which can be useful when you want your query to
go up to the value but not to include it, such as when using Gesso breakpoint
token values.
Output a min-width based media query.
@include breakpoint(800px) {
display: flex;
}
@include breakpoint(gesso-breakpoint(desktop)) {
display: none;
}
Output a max-width based media query. The optional $subtract_1_from_max
parameter will subtract 1px from the width value if set to true
(default:
false
).
@include breakpoint-max(900px) {
display: block;
}
@include breakpoint-max(gesso-breakpoint(mobile), true) {
display: none;
}
Output a media query with both a min-width and max-width. The optional
$subtract_1_from_max parameter will subtract 1px from the max-width value if
set to true
(default: false
).
@include breakpoint-min-max(400px, 700px) {
display: flex;
}
@include breakpoint-min-max(
gesso-breakpoint(mobile),
gesso-breakpoint(tablet),
true
) {
display: block;
}
Gesso uses custom mixins to specify container queries:
container-query
: min-width container queriescontainer-query-max
: max-width container queriescontainer-query-min-max
: container queries with both a min and max width
Each mixin takes one or two width parameters, which can be a straight value
(e.g., 800px, 40em) or a design token value called using the gesso-breakpoint
function (e.g., gesso-breakpoint(tablet-lg)
). The container-max
and
container-min-max
mixins can also take an optional parameter to subtract one
pixel from the max-width value, which can be useful when you want your query to
go up to the value but not to include it, such as when using Gesso breakpoint
token values.
In order for container queries to work, you need to set a containment context on a parent element.
container-type: inline-size;
container: container-name / inline-size;
Output a min-width based media query.
@include container-query(800px) {
display: flex;
}
@include container-query(gesso-breakpoint(desktop)) {
display: none;
}
Output a max-width based container query. The optional $subtract_1_from_max
parameter will subtract 1px from the width value if set to true
(default:
false
).
@include container-query-max(900px) {
display: block;
}
@include container-query-max(gesso-breakpoint(mobile), true) {
display: none;
}
Output a container query with both a min-width and max-width. The optional
$subtract_1_from_max parameter will subtract 1px from the max-width value if
set to true
(default: false
).
@include container-query-min-max(400px, 700px) {
display: flex;
}
@include container-query-min-max(
gesso-breakpoint(mobile),
gesso-breakpoint(tablet),
true
) {
display: block;
}
Gesso includes some additional filters and functions that can be used in Twig templates.
Fork of Drupal Pattern Lab's add_attribute
Twig function.
Allows Twig templates to add attributes that, in Drupal, will be merged with the Drupal attributes object
while also rendering in Storybook.
<div {{ add_attributes(
{
class: 'your-class-one your-class-two',
'data-foo': 'bar'
}
) }}>...</div>
Twig filter to sort an object by key alphabetically.
{% for key, value in your_object|keysort %}
...
{% endfor %}
Twig filter to transform a heading tag to the next level down (h2 -> h3, h3 -> h4, etc.) Used when the parent heading level can vary but, to maintain accessibility, the component's heading or subheading should change accordingly.
{% set subheading_element = title_element|subheading_level %}
<{{ subheading_element|default('h3') }}>...</{{ subheading_element|default('h3') }}>
A static Storybook site can be built with npm run build-storybook
. You will
then be able to view Storybook at
YOUR_URL/themes/gesso/storybook/index.html.
Some aspects of Gesso can be configured in the theme settings. These include the Back to Top component, Breadcrumb options, and Button styles for links.
For the buttons, put the classes that should be added for each button size
and button style on each line, with classes separated with .
, similar to how
you would add custom classes to the WYSIWYG editor.
c-button|Primary
c-button.c-button--secondary|Secondary
c-button.c-button--tertiary|Tertiary
To use these classes, select Gesso Button as the formatter for a link field under the entity's display settings.
Please use the Github issue queue https://github.com/forumone/gesso/issues for discussion, bug reports, feature requests, etc.
Submitted pull requests should be against the latest release candidate branch,
such as 5.x-RC
.
The Gesso theme is maintained by Corey Lafferty, KJ Monahan, and Dan Mouyard (@dcmouyard).