0

如何使用样式系统和样式组件创建通用标题组件?

以下有效,但我所有的标题都使用 h1 标签。

import css from "@styled-system/css"
import styled from "styled-components"
import { color, ColorProps, textAlign, TextAlignProps, variant } from "styled-system"

type Props = {
  level: 1 | 2 | 3 | 4 | 5 | 6
} & TextAlignProps &
  ColorProps

const Css = css({
  fontFamily: "heading",
  fontWeight: "heading",
})

const Variant = variant({
  prop: "level",
  variants: {
    1: {
      fontSize: 7,
      lineHeight: 7,
    },
    2: {
      fontSize: 6,
      lineHeight: 6,
    },
    3: {
      fontSize: 5,
      lineHeight: 5,
    },
    4: {
      fontSize: 4,
      lineHeight: 4,
    },
    5: {
      fontSize: 3,
      lineHeight: 3,
    },
    6: {
      fontSize: 2,
      lineHeight: 2,
    },
  },
})

const Heading = styled("h1")<Props>(Css, Variant, textAlign, color)

export default Heading

我试图创建一个 HeadingBase 组件,如:

type Props = {
  level: 1 | 2 | 3 | 4 | 5 | 6,
  as: string | undefined
} & TextAlignProps &
  ColorProps

const HeadingBase = ({ level, as: Component = `h${level}`, ...props }: Props) => <Component {...props} />

const Heading = styled(HeadingBase)<Props>(Css, Variant, textAlign, color)

但我收到一个错误,<Component {...props} />因为它没有用于 TextAlignProps 和 ColorProps 的 API。

4

1 回答 1

1

也许您可以利用as,以便在提供变体时,as密钥会自动更改为变体字符串。这可能通过代码更容易解​​释,看看。

const headingVariant = ({ fontSize }) => {
  let variantConfig = {
    h1: {
      fontSize: [7, 8], //28px 32px
    },
    h2: {
      fontSize: [6, 7], //24px 28px
    },
    h3: {
      fontSize: ['22px', 6], //22px 24px
    },
    h4: {
      fontSize: 5, //20px
    },
    h5: {
      fontSize: 4, //18px
    },
  };

  if (fontSize) {
    variantConfig = Object.keys(variantConfig).reduce((acc, currKey) => {
      acc[currKey] = {
        ...variantConfig[currKey],
        fontSize,
      };
      return acc;
    }, {});
  }
  return variant({
    prop: 'variant',
    variants: variantConfig,
  });
};

const Heading = styled(BaseText).attrs(
  ({ variant: hybridVariant, as: asOverride }) => ({
    as: asOverride ?? hybridVariant,
  })
)(
  ({ isBold, color }) =>
    css({
      fontFamily: `sofia-pro, Helvetica, Arial`,
      lineHeight: '130%',
      color: color || 'text',
      fontWeight: isBold ? 1 : 0,
      fontStyle: 'normal',
    }),
  headingVariant
);

Heading.propTypes = {
  isBold: PropTypes.bool,
  variant: PropTypes.string,
};

Heading.defaultProps = {
  isBold: true,
  variant: 'h2',
};

export default Heading;
于 2021-04-05T13:40:09.480 回答