* Fix note formatting Signed-off-by: Emma Indal <emma.indahl@gmail.com> * Update app-custom-theme.md Signed-off-by: Emma Indal <emma.indahl@gmail.com>
9.5 KiB
id, title, description
| id | title | description |
|---|---|---|
| app-custom-theme | Customize the look-and-feel of your App | Documentation on Customizing look and feel of the App |
Backstage ships with a default theme with a light and dark mode variant. The themes are provided as a part of the @backstage/theme package, which also includes utilities for customizing the default theme, or creating completely new themes.
Creating a Custom Theme
The easiest way to create a new theme is to use the createTheme function
exported by the
@backstage/theme package. You
can use it to override so basic parameters of the default theme such as the
color palette and font.
For example, you can create a new theme based on the default light theme like this:
import { createTheme, lightTheme } from '@backstage/theme';
const myTheme = createTheme({
palette: lightTheme.palette,
fontFamily: 'Comic Sans MS',
defaultPageTheme: 'home',
});
If you want more control over the theme, and for example customize font sizes
and margins, you can use the lower-level createThemeOverrides function
exported by @backstage/theme
in combination with
createTheme
from @material-ui/core. See
the "Overriding Backstage and Material UI css rules" section below.
You can also create a theme from scratch that matches the BackstageTheme type
exported by @backstage/theme.
See the
material-ui docs on theming
for more information about how that can be done.
Using your Custom Theme
To add a custom theme to your Backstage app, you pass it as configuration to
createApp.
For example, adding the theme that we created in the previous section can be done like this:
import { createApp } from '@backstage/app-defaults';
import { ThemeProvider } from '@material-ui/core/styles';
import CssBaseline from '@material-ui/core/CssBaseline';
const app = createApp({
apis: ...,
plugins: ...,
themes: [{
id: 'my-theme',
title: 'My Custom Theme',
variant: 'light',
Provider: ({ children }) => (
<ThemeProvider theme={myTheme}>
<CssBaseline>{children}</CssBaseline>
</ThemeProvider>
),
}]
})
Note that your list of custom themes overrides the default themes. If you still
want to use the default themes, they are exported as lightTheme and
darkTheme from
@backstage/theme.
Example of a custom theme
import {
createTheme,
genPageTheme,
lightTheme,
shapes,
} from '@backstage/theme';
const myTheme = createTheme({
palette: {
...lightTheme.palette,
primary: {
main: '#343b58',
},
secondary: {
main: '#565a6e',
},
error: {
main: '#8c4351',
},
warning: {
main: '#8f5e15',
},
info: {
main: '#34548a',
},
success: {
main: '#485e30',
},
background: {
default: '#d5d6db',
paper: '#d5d6db',
},
banner: {
info: '#34548a',
error: '#8c4351',
text: '#343b58',
link: '#565a6e',
},
errorBackground: '#8c4351',
warningBackground: '#8f5e15',
infoBackground: '#343b58',
navigation: {
background: '#343b58',
indicator: '#8f5e15',
color: '#d5d6db',
selectedColor: '#ffffff',
},
},
defaultPageTheme: 'home',
fontFamily: 'Comic Sans MS',
/* below drives the header colors */
pageTheme: {
home: genPageTheme(['#8c4351', '#343b58'], shapes.wave),
documentation: genPageTheme(['#8c4351', '#343b58'], shapes.wave2),
tool: genPageTheme(['#8c4351', '#343b58'], shapes.round),
service: genPageTheme(['#8c4351', '#343b58'], shapes.wave),
website: genPageTheme(['#8c4351', '#343b58'], shapes.wave),
library: genPageTheme(['#8c4351', '#343b58'], shapes.wave),
other: genPageTheme(['#8c4351', '#343b58'], shapes.wave),
app: genPageTheme(['#8c4351', '#343b58'], shapes.wave),
apis: genPageTheme(['#8c4351', '#343b58'], shapes.wave),
},
});
Overriding Backstage and Material UI components styles
When creating a custom theme you would be applying different values to component's css rules that use the theme object. For example, a Backstage component's styles might look like this:
const useStyles = makeStyles<BackstageTheme>(
theme => ({
header: {
padding: theme.spacing(3),
boxShadow: '0 0 8px 3px rgba(20, 20, 20, 0.3)',
backgroundImage: theme.page.backgroundImage,
},
}),
{ name: 'BackstageHeader' },
);
Notice how the padding is getting its value from theme.spacing, that means
that setting a value for spacing in your custom theme would affect this
component padding property and the same goes for backgroundImage which uses
theme.page.backgroundImage. However, the boxShadow property doesn't
reference any value from the theme, that means that creating a custom theme
wouldn't be enough to alter the box-shadow property or to add css rules that
aren't already defined like a margin. For these cases you should also create an
override.
import { createApp } from '@backstage/core-app-api';
import { BackstageTheme, lightTheme } from '@backstage/theme';
/**
* The `@backstage/core-components` package exposes this type that
* contains all Backstage and `material-ui` components that can be
* overridden along with the classes key those components use.
*/
import { BackstageOverrides } from '@backstage/core-components';
export const createCustomThemeOverrides = (
theme: BackstageTheme,
): BackstageOverrides => {
return {
BackstageHeader: {
header: {
width: 'auto',
margin: '20px',
boxShadow: 'none',
borderBottom: `4px solid ${theme.palette.primary.main}`,
},
},
};
};
const customTheme: BackstageTheme = {
...lightTheme,
overrides: {
// These are the overrides that Backstage applies to `material-ui` components
...lightTheme.overrides,
// These are your custom overrides, either to `material-ui` or Backstage components.
...createCustomThemeOverrides(lightTheme),
},
};
const app = createApp({
apis: ...,
plugins: ...,
themes: [{
id: 'my-theme',
title: 'My Custom Theme',
variant: 'light',
Provider: ({ children }) => (
<ThemeProvider theme={customTheme}>
<CssBaseline>{children}</CssBaseline>
</ThemeProvider>
),
}]
});
Custom Logo
In addition to a custom theme, you can also customize the logo displayed at the far top left of the site.
In your frontend app, locate src/components/Root/ folder. You'll find two
components:
LogoFull.tsx- A larger logo used when the Sidebar navigation is opened.LogoIcon.tsx- A smaller logo used when the sidebar navigation is closed.
To replace the images, you can simply replace the relevant code in those components with raw SVG definitions.
You can also use another web image format such as PNG by importing it. To do
this, place your new image into a new subdirectory such as
src/components/Root/logo/my-company-logo.png, and then add this code:
import MyCustomLogoFull from './logo/my-company-logo.png';
//...
const LogoFull = () => {
return <img src={MyCustomLogoFull} />;
};
Custom Homepage
In addition to a custom theme, a custom logo, you can also customize the homepage of your app. To do that we need to go through a few steps.
Setting up the Home Page
- Create a Home Page Component that will be used for composition.
packages/app/src/components/home/HomePage.tsx
import React from 'react';
export const HomePage = () => {
return {
/* TODO: Compose a Home Page here */
};
};
- Add a route where the homepage will live, presumably
/.
packages/app/src/App.tsx
import { HomepageCompositionRoot } from '@backstage/plugin-home';
import { HomePage } from './components/home/HomePage';
// ...
<Route path="/" element={<HomepageCompositionRoot />}>
<HomePage />
</Route>;
// ...
Composing your Home Page
Composing a Home Page is no different from creating a regular React Component, i.e. the App Integrator is free to include whatever content they like. However, there are components developed with the Home Page in mind. If you are looking for components to use when composing your homepage, you can take a look at the collection of Homepage components in storybook. If you don't find a component that suits your needs but want to contribute, check the Contributing documentation.
If you want to use one of the available homepage templates you can find the templates in the storybook under the "Home" plugin. And if you would like to contribute a template, please see the Contributing documentation
import React from 'react';
import Grid from '@material-ui/core/Grid';
import { HomePageCompanyLogo } from '@backstage/plugin-home';
export const HomePage = () => {
return (
<Grid container spacing={3}>
<Grid item xs={12} md={4}>
<HomePageCompanyLogo className={container} />
</Grid>
</Grid>
);
};