Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Load a theme via a theme key in volto.config.js or in package.json #4625

Merged
merged 19 commits into from
Apr 11, 2023
Merged
Show file tree
Hide file tree
Changes from 7 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
33 changes: 33 additions & 0 deletions addon-registry.js
Original file line number Diff line number Diff line change
Expand Up @@ -130,6 +130,10 @@ class AddonConfigurationRegistry {
this.packages = {};
this.customizations = new Map();

// Theme from a package.json key or from volto.config.js
// Programatically via volto.config.js wins
this.theme = packageJson.theme || this.voltoConfigJS.theme;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I might want an addon to provide the theme.

The volto.config.js implies that the project is no longer throw-away, or at least there's a standard "distribution-like" volto.config.js that I use in all my projects.

Copy link
Member Author

@sneridagh sneridagh Mar 30, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I agree that would be desirable, but the current state is that you need to define the enabled addons list in volto.config.js or in package.json anyways, so adding the theme is nothing that we "loose"...

In fact, these (addons and theme) are the only things that you need to "save" in order of a project to be expendable. Same applies for the dockerized approach... as long as you have a volto.config.js that you can forward to your expendable project, you are safe.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would add an option to define a theme name from an environment var. We have it already for addons iirc, so it would be a nice complement, and a very very nice addition for docker users :)


this.initDevelopmentPackages();
this.initPublishedPackages();
this.initAddonsFromEnvVar();
Expand Down Expand Up @@ -354,6 +358,35 @@ class AddonConfigurationRegistry {
.filter((e) => e);
}

getCustomThemeAddons() {
const customThemeAddonsInfo = {
variables: [],
main: [],
};

this.getAddonDependencies().forEach((addon) => {
const normalizedAddonName = addon.split(':')[0];
// We have two possible insertion points, variables and main

const customThemeVariables = `${this.packages[normalizedAddonName].modulePath}/theme/_variables.scss`;
const customThemeMain = `${this.packages[normalizedAddonName].modulePath}/theme/_main.scss`;
if (
fs.existsSync(customThemeVariables) &&
normalizedAddonName !== this.theme
) {
customThemeAddonsInfo.variables.push(normalizedAddonName);
}
if (
fs.existsSync(customThemeMain) &&
normalizedAddonName !== this.theme
) {
customThemeAddonsInfo.main.push(normalizedAddonName);
}
});

return customThemeAddonsInfo;
}

/**
* Returns a mapping name:diskpath to be uses in webpack's resolve aliases
*/
Expand Down
79 changes: 79 additions & 0 deletions create-theme-addons-loader.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,79 @@
const path = require('path');
const fs = require('fs');
const tmp = require('tmp');
const cryptoRandomString = require('crypto-random-string');

const titleCase = (w) => w.slice(0, 1).toUpperCase() + w.slice(1, w.length);

/*
* Transforms a package name to javascript variable name
*/
function nameFromPackage(name) {
name =
name.replace(/[@~./\\:\s]/gi, '') ||
cryptoRandomString({ length: 10, characters: 'abcdefghijk' });
return name
.split('-')
.map((w, i) => (i > 0 ? titleCase(w) : w))
.join('');
}

/*
* Creates a static file with code necessary to load the addons configuration
*
*/
function getAddonsLoaderCode(name, customThemeAddons = []) {
let buf = `/*
This file is autogenerated. Don't change it directly.
Add a ./theme/_${name}.scss in your add-on to load your theme customizations in the current theme.
*/

`;
customThemeAddons.forEach((addon) => {
const customization = `${addon}/theme/${name}`;
const line = `@import '${customization}';\n`;
buf += line;
});

return buf;
}

module.exports = ({ main, variables }) => {
// const addonsThemeLoaderVariablesPath = path.join(
// process.cwd(),
// 'src',
// '_variables.scss',
// );
// const addonsThemeLoaderMainPath = path.join(
// process.cwd(),
// 'src',
// '_main.scss',
// );

// const addonsThemeLoaderVariablesPath = path.join(
// process.cwd(),
// 'src',
// '_variables.scss',
// );
// const addonsThemeLoaderMainPath = path.join(
// process.cwd(),
// 'src',
// '_main.scss',
// );

const addonsThemeLoaderVariablesPath = tmp.tmpNameSync({ postfix: '.scss' });
const addonsThemeLoaderMainPath = tmp.tmpNameSync({ postfix: '.scss' });
fs.writeFileSync(
addonsThemeLoaderVariablesPath,
new Buffer.from(getAddonsLoaderCode('variables', variables)),
);
fs.writeFileSync(
addonsThemeLoaderMainPath,
new Buffer.from(getAddonsLoaderCode('main', main)),
);

return [addonsThemeLoaderVariablesPath, addonsThemeLoaderMainPath];
};

module.exports.getAddonsLoaderCode = getAddonsLoaderCode;
module.exports.nameFromPackage = nameFromPackage;
200 changes: 200 additions & 0 deletions docs/source/addons/theme.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,200 @@
---
myst:
html_meta:
"description": "Create a theme add-on"
"property=og:description": "Create a theme add-on"
"property=og:title": "Create a theme add-on"
"keywords": "Volto, Plone, Semantic UI, CSS, Volto theme"
---

# Create a theme add-on

We can create a Volto Add-on that acts as a theme Add-on, so we can detach it from the project.
The advantage is that you can deploy the same theme in different projects, or have themes depending on conditions that you could inject on build time.

1. Add a `theme` key in your `volto.config.js` file in the root of your project:
ksuess marked this conversation as resolved.
Show resolved Hide resolved

```js
module.exports = {
addons: [],
theme: 'volto-my-theme'
};
```
or add a key in your `package.json` project:

```json
"theme": "volto-my-theme"
```

2. Create a directory `src/theme` in your add-on, then add this file `theme.config`, replacing `<name_of_your_theme>` with your add-on name:

```less
/*******************************
Theme Selection
*******************************/

/* To override a theme for an individual element specify theme name below */

/* Global */
@site : 'pastanaga';
@reset : 'pastanaga';

/* Elements */
@button : 'pastanaga';
@container : 'pastanaga';
@divider : 'pastanaga';
@flag : 'pastanaga';
@header : 'pastanaga';
@icon : 'pastanaga';
@image : 'pastanaga';
@input : 'pastanaga';
@label : 'pastanaga';
@list : 'pastanaga';
@loader : 'pastanaga';
@placeholder : 'pastanaga';
@rail : 'pastanaga';
@reveal : 'pastanaga';
@segment : 'pastanaga';
@step : 'pastanaga';

/* Collections */
@breadcrumb : 'pastanaga';
@form : 'pastanaga';
@grid : 'pastanaga';
@menu : 'pastanaga';
@message : 'pastanaga';
@table : 'pastanaga';

/* Modules */
@accordion : 'pastanaga';
@checkbox : 'pastanaga';
@dimmer : 'pastanaga';
@dropdown : 'pastanaga';
@embed : 'pastanaga';
@modal : 'pastanaga';
@nag : 'pastanaga';
@popup : 'pastanaga';
@progress : 'pastanaga';
@rating : 'pastanaga';
@search : 'pastanaga';
@shape : 'pastanaga';
@sidebar : 'pastanaga';
@sticky : 'pastanaga';
@tab : 'pastanaga';
@transition : 'pastanaga';

/* Views */
@ad : 'pastanaga';
@card : 'pastanaga';
@comment : 'pastanaga';
@feed : 'pastanaga';
@item : 'pastanaga';
@statistic : 'pastanaga';

/* Extras */
@main : 'pastanaga';
@custom : 'pastanaga';

/*******************************
Folders
*******************************/

/* Path to theme packages */
@themesFolder : '~volto-themes';

/* Path to site override folder */
@siteFolder : "~<name_of_your_theme>/theme";
sneridagh marked this conversation as resolved.
Show resolved Hide resolved

/*******************************
Import Theme
*******************************/

@import (multiple) "~semantic-ui-less/theme.less";
@fontPath : "~volto-themes/@{theme}/assets/fonts";

.loadAddonOverrides() {
@import (optional) "@{siteFolder}/@{addon}/@{addontype}s/@{addonelement}.overrides";
}

/* End Config */
```

3. You also have to declare it as an add-on, so don't forget to add it in the `addons` key in `volto.config.js` or in `package.json`.
sneridagh marked this conversation as resolved.
Show resolved Hide resolved
4. After starting Volto, the theme should be active and you can add now overrides to the default theme in `src/theme` as you would do it in a project.
sneridagh marked this conversation as resolved.
Show resolved Hide resolved
5. You can safely delete now your project's `theme` folder, since the one in the add-on will take precedence.
sneridagh marked this conversation as resolved.
Show resolved Hide resolved

## Using your own scape hatch
sneridagh marked this conversation as resolved.
Show resolved Hide resolved

Volto theming uses SemanticUI theming capabilities to define and extend a theme for your site. However, while maintaining and playing well with the Semantic UI Volto base, using a traditional CSS approach can be done using the less-based `extras` scape hatch.
sneridagh marked this conversation as resolved.
Show resolved Hide resolved

At the same time, one can entirely bail off the extras scape hatch and add your own, by customizing the `theme.js` module in Volto.
sneridagh marked this conversation as resolved.
Show resolved Hide resolved

```js
import 'semantic-ui-less/semantic.less';
import '@plone/volto/../theme/themes/pastanaga/extras/extras.less';

// You can add more entry points for theming
import '@kitconcept/volto-light-theme/theme/main.scss';
```

While building your own scape hatch for theming, you can use the preprocessor of your choice (in the example, scss) while maintaining the "base" Volto theme, but customizing it using the resultant CSS.
sneridagh marked this conversation as resolved.
Show resolved Hide resolved

You can see an example of such a theme in: https://github.com/kitconcept/volto-light-theme

## Modify a custom theme using from another add-on
sneridagh marked this conversation as resolved.
Show resolved Hide resolved

Sometimes you have a custom theme that you want to reuse through all your projects, but with some differences, maintaining the base.
Usually, the only option would be to use an addon that adds more CSS to the base theme, using imports that will load after the theme.
sneridagh marked this conversation as resolved.
Show resolved Hide resolved
However, there is a problem with this approach.
You cannot use existing theme variables (like breakpoints, etc) on these new CSS.
sneridagh marked this conversation as resolved.
Show resolved Hide resolved
The same way, it gets somewhat detached from the normal flow of the loaded theme.
sneridagh marked this conversation as resolved.
Show resolved Hide resolved
Same applies for add-ons, they are detached from the current theme.
sneridagh marked this conversation as resolved.
Show resolved Hide resolved
One could use a SemanticUI approach for making this work, but it's SemanticUI bound.

```{warning}
This is only possible when using your own scape hatch, and works only with scss-based themes, and not with SemanticUI themes, since it enables a couple of entry points that only support scss files. For an example of how it could be used, see: https://github.com/kitconcept/volto-light-theme
sneridagh marked this conversation as resolved.
Show resolved Hide resolved
```

If your custom scape hatch defines a custom theme using scss, you can take advantage of this feature.
sneridagh marked this conversation as resolved.
Show resolved Hide resolved
Although not limited to this, it would be possible to extend this feature to add more entry points, use another preprocessor or theming approach.
sneridagh marked this conversation as resolved.
Show resolved Hide resolved

This feature enables two entry points: variables and main.
From your add-on code, you can extend an existing theme in use by creating a file corresponding to each entry point:
sneridagh marked this conversation as resolved.
Show resolved Hide resolved

* `./src/theme/_variables.scss`
* `./src/theme/_main.scss`

### Variables (`addonsThemeCustomizationsVariables`)

Use this entrypoint file to modify the original variables of the current loaded theme (by adding the entrypoint after your own variable definitions in the theme). In the theme, it should be imported like this:
sneridagh marked this conversation as resolved.
Show resolved Hide resolved

```scss hl_lines="2"
@import 'variables';
@import 'addonsThemeCustomizationsVariables';
pnicolli marked this conversation as resolved.
Show resolved Hide resolved
@import 'typography';
@import 'utils';
@import 'layout';
```

Volto will not only load your add-on entrypoints files but it will also detect all the add-ons that have these entrypoint files and import them grouped under a single file, and will add automatically an `addonsThemeCustomizationsVariables` alias that can be referenced from the theme as shown above.
sneridagh marked this conversation as resolved.
Show resolved Hide resolved

### Main (`addonsThemeCustomizationsMain`)

This entrypoint is intended to add your own styling definitions, complementing the theme ones. You should add it after all the CSS of your theme:
sneridagh marked this conversation as resolved.
Show resolved Hide resolved

```scss hl_lines="6"
@import 'blocks/search';
@import 'blocks/listing';

@import 'temp';

@import 'addonsThemeCustomizationsMain';

/* No CSS beyond this point */
```

Volto will also detect all the add-ons that have this entrypoint files and import them grouped under a single file, and will add automatically an `addonsThemeCustomizationsMain` alias that can be referenced from the theme as shown above.
sneridagh marked this conversation as resolved.
Show resolved Hide resolved

```{note}
It will only work in combination with the theme declaration in `volto.config.js` or in `package.json`.
```
2 changes: 2 additions & 0 deletions news/4625.feature
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
Support for declaring a theme in `volto.config.js` or in `package.json`
Add two entrypoints for allowing extending a theme from other add-ons. @sneridagh
sneridagh marked this conversation as resolved.
Show resolved Hide resolved
23 changes: 23 additions & 0 deletions razzle.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ const fs = require('fs');
const RootResolverPlugin = require('./webpack-plugins/webpack-root-resolver');
const RelativeResolverPlugin = require('./webpack-plugins/webpack-relative-resolver');
const createAddonsLoader = require('./create-addons-loader');
const createThemeAddonsLoader = require('./create-theme-addons-loader');
const AddonConfigurationRegistry = require('./addon-registry');
const CircularDependencyPlugin = require('circular-dependency-plugin');
const TerserPlugin = require('terser-webpack-plugin');
Expand Down Expand Up @@ -245,6 +246,28 @@ const defaultModify = ({
'lodash-es': path.dirname(require.resolve('lodash')),
};

const [
addonsThemeLoaderVariablesPath,
addonsThemeLoaderMainPath,
] = createThemeAddonsLoader(registry.getCustomThemeAddons());

// Automatic Theme Loading
if (registry.theme) {
// The themes should be located in `src/theme`
const themePath = registry.packages[registry.theme].modulePath;
const themeConfigPath = `${themePath}/theme/theme.config`;
config.resolve.alias['../../theme.config$'] = themeConfigPath;
config.resolve.alias['../../theme.config'] = themeConfigPath;

// We create an alias for each custom theme insertion point (variables, main)
config.resolve.alias[
'addonsThemeCustomizationsVariables'
] = addonsThemeLoaderVariablesPath;
config.resolve.alias[
'addonsThemeCustomizationsMain'
] = addonsThemeLoaderMainPath;
}

config.performance = {
maxAssetSize: 10000000,
maxEntrypointSize: 10000000,
Expand Down
3 changes: 2 additions & 1 deletion requirements-docs.txt
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,8 @@ lesscpy
linkify-it-py
myst-parser
sphinx-autobuild
sphinx-book-theme
pydata-sphinx-theme<=0.8.99
sphinx-book-theme==0.3.3
sphinx-copybutton
sphinx-sitemap
sphinx-togglebutton
Expand Down