最新消息:雨落星辰是一个专注网站SEO优化、网站SEO诊断、搜索引擎研究、网络营销推广、网站策划运营及站长类的自媒体原创博客

javascript - How to pass styles via props with styled components without having to create the component inside the component - S

programmeradmin2浏览0评论

I cannot find another way to give my props dynamic styles without creating the styled-ponent inside the Text ponent. For performance I'd like to create the ponent outside of the Text ponent.

Styled Component

import React, { ReactChildren } from 'react'
import styled from 'styled-ponents'


export const Text = ({ as = 'p', children, styles = {} }) => {
  const newStyles = Object.entries(styles).reduce((acc, [key, value]) => {
    return `${acc}${key}: ${value};`
  }, '')
  const Component = styled.p`
    ${newStyles}
    color: #000;
  `
  return <Component as={as}>{children}</Component>
}

Desired Usage

<Text styles={{ height: "10px" }} />

Output HTML

<p styles="height: 10px" />

I cannot find another way to give my props dynamic styles without creating the styled-ponent inside the Text ponent. For performance I'd like to create the ponent outside of the Text ponent.

Styled Component

import React, { ReactChildren } from 'react'
import styled from 'styled-ponents'


export const Text = ({ as = 'p', children, styles = {} }) => {
  const newStyles = Object.entries(styles).reduce((acc, [key, value]) => {
    return `${acc}${key}: ${value};`
  }, '')
  const Component = styled.p`
    ${newStyles}
    color: #000;
  `
  return <Component as={as}>{children}</Component>
}

Desired Usage

<Text styles={{ height: "10px" }} />

Output HTML

<p styles="height: 10px" />

Share Improve this question edited Jun 16, 2021 at 15:59 Jamie Hutber asked Jun 15, 2021 at 16:55 Jamie HutberJamie Hutber 28.1k54 gold badges194 silver badges313 bronze badges 2
  • Couldn't you just do the Object.entries bit outside the styled-ponent and pass it in? – serraosays Commented Jun 15, 2021 at 17:04
  • Ye, but the issue is having the Component inside the Text ponent. So if I were to take both out I wouldn't be able to return a ponent from a function. But maybe I can and I just cannot see it :p – Jamie Hutber Commented Jun 15, 2021 at 17:15
Add a ment  | 

2 Answers 2

Reset to default 9

The above code is confusing in that you want to apply styles to the DOM style property, but you're also applying them as CSS styles to a classname. There's no need to create this posed ponent because styled-ponents already handles as, children, and style properties without posing:

Example 1:

import React from "react";
import styled, { Interpolation } from "styled-ponents";

export type TextProps = {
  as?: string | React.ComponentType<any>;
  children: React.ReactNode;
  styles?: Interpolation<React.CSSProperties>;
};

const Component = styled.p<TextProps>`
  ${({ styles }) => styles}
  color: #000;
`;

export const Text = ({
  as,
  children,
  styles
}: TextProps): React.ReactElement => (
  <Component styles={styles} as={as}>
    {children}
  </Component>
);

export default Text;

Example 2:

import styled from "styled-ponents";

const Text = styled.p`
  color: #000;
`;

export default Text;

Example 3:

import * as React from "react";
import styled from "styled-ponents";

export type TextComponentProps = {
  className?: string;
  children: React.ReactNode;
  styles?: React.CSSProperties;
};

const TextComponent = ({
  children,
  className,
  styles
}: TextComponentProps): React.ReactElement => (
  <p className={className} style={styles}>
    {children}
  </p>
);

const Text = styled(TextComponent)`
  color: #000;
`;

export default Text;

Usage:

import * as React from "react";
import Box from "./Box";
import Text from "./Text";
import Text2 from "./Text2";
import Text3 from "./Text3";
import "./styles.css";

const App = (): React.ReactElement => (
  <div className="app">
    <h1>Example 1 - "styles" as CSS Styles</h1>
    <Box>
      <Text styles={{ height: "10px" }}>Hello</Text>
      <Text as="h1">Goodbye</Text>
    </Box>
    <hr />
    <h1>Example 2 - "style" as DOM styles</h1>
    <Box>
      <Text2 style={{ height: "10px" }}>Hello</Text2>
      <Text2 as="h1">Goodbye</Text2>
    </Box>
    <hr />
    <h1>Example 3 - "styles" as DOM styles</h1>
    <Box>
      <Text3 styles={{ height: "10px" }}>Hello</Text3>
      <Text3 as="h1">Goodbye</Text3>
    </Box>
  </div>
);

export default App;

Output:

On that note, it sounds like you might be trying to do something I did with posabled-styled-ponents, although I wouldn't remend it because it doesn't work with SSR apps.

You could create a ref on the styled ponent in order to set the style in an effect like this:

const Component = styled.p`
  background-color: red;
`;

const Text = ({ as = 'p', children, styles = {} }) => {
  const newStyles = Object.entries(styles).reduce((acc, [key, value]) => {
    return `${acc}${key}: ${value};`
  }, '')
  
  const ref = React.createRef();

  React.useEffect(() => {
    ref.current.style = newStyles;
  }, []);
  
  return <Component ref={ref} as={as}>{children}</Component>
}

ReactDOM.render(<Text styles={{ height: "10px" }}>This paragraph is 10px high</Text>, document.getElementById('root'));
<script crossorigin src="https://unpkg./react@17/umd/react.production.min.js"></script>
<script crossorigin src="https://unpkg./react-dom@17/umd/react-dom.production.min.js"></script></script>
<script src="https://unpkg./react-is@17/umd/react-is.production.min.js"></script>
<script src="https://unpkg./styled-ponents/dist/styled-ponents.min.js"></script>

<div id="root"></div>

与本文相关的文章

发布评论

评论列表(0)

  1. 暂无评论