Skip to content Skip to sidebar Skip to footer

React Styled Component: How To Add Css Styles Based On Props?

I've been trying to find the solution to my problem. I have several Heading Tags (H1, H2 etc) each in their own file. I would like to add some css when calling them based on a prop

Solution 1:

Something like this works:

constHeadingH2 = styled.h2`
  position: ${props => props.relative && 'relative'};
  padding: ${props => props.paddingBottom ? '0 0 20px 0' : '0'};
}
`;

Then use like this:

<HeadingH2relativepaddingBottom />

Solution 2:

You can also use css helper from styled-components to create a SharedStyles.js file.

In the demo you can see it in action. Just using it in a style of an inherited component is not working as expected. If I'm adding it to StyledBase then the variables are not correctly added afterwards (hover style override stops working). That's why I copied ${borderBottom} to each styled component Heading1 / Heading2 instead of adding it to StyledBase.

I think having a level prop for the heading is a good idea but I would handle it differently by creating a HeadingBase component and add your styles to a StyledBase component (also see code in the demo).

The HeadingBase code looks like this:

constHeadingBase = ({ className, children, level = 1 }) =>
    React.createElement(`h${level}`, { className }, children);

It's a component that renders h1,h2,... tags based on the prop level passed (defaults to h1). The h-tag receives className as props (needed for styled-components) and contains the children passed to the component.

SharedStyles.js

import { css } from"styled-components";

exportconst borderBottom = css`
    &:after{
        content: "";
        display: block;
        height: 3px;
        width: 200px;
        background-color: ${props => props.color || "black"};
        /* position */position: absolute;
        bottom: 0;
        left: 50%;
        transform: translateX(-50%);
`;

Then you can import it with import { borderBottom } from "./SharedStyles"; and add it to your styled-component like following:

constHeading1= styled.h1`
    ${borderBottom}
`;

Solution 3:

Possible answer:

I add the following CSS rules in a separate file like so. I create a function which returns a string of text. I can import this function and add those rules to any component I wish.

exportconstborderBottom=()=> {
  return`position:relative;text-transform:uppercase;text-align:center;&:after{
    content:"";display:block;height:3px;width:45px;background-color:currentColor;/*position*/position:absolute;bottom:0;left:50%;transform:translateX(-50%);
  } 
`
 }

And use it like so on any heading or component that I wish:

import styled from'styled-components';
import colors from'../../../../colors';
import fonts from'../../../../fonts';
import fontWeights from'../../../../fontWeights';
import {borderBottom} from'../../../../css';

constHeadingH5 = styled.h5`
  color: ${colors.black};
  font-family: ${fonts.montSerrat};
  font-size: 1em;
  font-weight: ${fontWeights.light};
  letter-spacing: 0.1em;
  padding-bottom: 0.45em;
  margin-bottom: 25px;
  ${borderBottom}
`  
;

exportdefaultHeadingH5;

This works for me. Any other suggestions are welcome.

Solution 4:

You should definely check this: typestyle

the best way you can write dynamic css (for me). Works perfectly with react, even with ssr if you need it...

Solution 5:

Why not just have a headingLevel prop? and then pass it into the styled component? And just have one StyledHeader styled component as I'm guessing the code is the mostly the same in all the heading styles files? Which is a big no no, you want to always try not to repeat your code.

constFeatures = () => {
  return(
    <SectionContainerid={"what"}><StyledHeaderheadingLevel={headingLevel}
      >
        What We Do
      </StyledHeader><HeadingBaseline>
      {Text.headingBaseline}
    </HeadingBaseline><FeaturesContainer><Featuresrc={feature1}headingText={Text.feature1.heading}paragraph={Text.feature1.paragraph}
      /><Featuresrc={feature2}headingText={Text.feature2.heading}paragraph={Text.feature2.paragraph}
      /><Featuresrc={feature3}headingText={Text.feature3.heading}paragraph={Text.feature3.paragraph}
      /></FeaturesContainer></SectionContainer>
)
};

exportdefaultFeatures;

And in your StyledHeader file

The below function will take your passed in heading level of h1, h2, h3 and will apply a border, if not the above 3 heading level it will be 0 value. I'd do some checks to ensure the value is lower case e.g. toLowerCase()

constgetBorder=({headingLevel})=> {
  constheadingLevelMap= {
    h1:0.7,
    h2:0.6,
    h3:0.6,
  };returnheadingLevelMap[headingLevel] ||0;
}

constHeadingH2=styled.headingLevel`color:${colors.text};font-family:${fonts.montSerrat};font-size:1.6em;font-weight:${fontWeights.light};letter-spacing:0.2em;padding-bottom:0.7em;border-bottom:${getCorrectBorderBottom}emsolidblackposition:relative;text-transform:uppercase;text-align:center;&:after{content:"";display:block;height:3px;width:45px;background-color:currentColor;/*position*/position:absolute;bottom:0;left:50%;transform:translateX(-50%);}`;

I'd also check that if the value of the passed in headingLevel is not any of the 6 heading levels it should have a default value of whatever you want.

The above was just quick pseudo code, but hopefully get the general idea here? Let me know it comments if not.

I'd also recommend that you split your Title component out into a separate component.

Post a Comment for "React Styled Component: How To Add Css Styles Based On Props?"