Sometimes we have some styles based on component's props or state, or dynamic in some way. If you use the styled
helper with React, this is automatically handled using CSS custom properties. But we cannot do the same for css
tags since they aren't linked to any component, and so we don't have access to state and props. However, there are some approaches to tackle this, each with their own limitations.
Inline styles are the most straightforward way to use dynamic styles. Pass a style
object with the dynamic styles, and you're done.
import React from 'react';
export function Pager({ index, children }) {
return (
<div style={{ transform: `translateX(${index * 100}%)` }}>
{children}
</div>
);
}
However, it's not possible to use inline styles with pseudo-selectors or media queries.
CSS custom properties can be used to expose dynamic properties to the CSS.
import React from 'react';
import { css } from 'linaria';
const box = css`
height: var(--box-size);
width: var(--box-size);
`;
export function Box({ size }) {
return (
<div
className={box}
style={{ '--box-size': size }}
/>
);
}
The browser support for CSS custom properties is limited, and it's not polyfilled. Therefore it's not a viable approach if you need to support older browsers. Worth noting that custom properties cascade, so if you don't override the value for the current element, and a custom property with the same name exists for a parent element, it'll be used instead.
In cases where you know the values ahead of time, you can use data attributes to dynamically switch the styles that are applied.
import React from 'react';
import { css } from 'linaria';
const box = css`
&[data-valid] {
color: yellow;
}
&[data-valid="invalid"] {
color: red;
}
&[data-valid="valid"] {
color: green;
}
`
export function Box({ color, valid }) {
return (
<div
className={box}
data-valid={valid ? 'valid' : 'invalid'}
/>
);
}
For color values, browsers support a currentColor
property which points to the text color of the element. It is well supported in all browsers.
import React from 'react';
import { css } from 'linaria';
const box = css`
background-color: currentColor;
`;
const content = css`
color: white;
`;
export function Box({ color }) {
return (
<div
className={box}
style={{ color }}
>
<span className={content}>
¯\_(ツ)_/¯
</span>
</div>
);
}
You cannot use this approach if the dynamic value is not a color, or the element contains some text which needs to be styled with a different color. If the element has children, you will need to reset the color
property for the text.