Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add icon name prop #41

Open
ffpetrovic opened this issue Jul 19, 2019 · 19 comments
Open

Add icon name prop #41

ffpetrovic opened this issue Jul 19, 2019 · 19 comments

Comments

@ffpetrovic
Copy link

Does it not make sense to have a reusable icon component of our own that utilizes react-feather, and therefore be able to do something like <Icon icon={this.props.icon} />?

I'm aware that we can always pass an icon like <Camera /> down as a prop, but I'm wondering if it's possible to this just by the name of the icon.

@Kosai106
Copy link

You could always abstract this kind of behavour, but otherwise doing the following would allow you a similar experience (Almost).

const IconTag = Icon[this.props.icon];

<IconTag />

@SeanMcP
Copy link

SeanMcP commented Jul 31, 2019

@Kosai106 I've created a component like that in every project where I've used react-feather. I agree with @ffpetrovic that baking that into the library would be a good idea.

@Kosai106
Copy link

I just realised that my method wouldn't work if you're using TypeScript anyway.

Element implicitly has an 'any' type because type 'typeof import("/Users/oesterkilde/projects/[PROJECT-NAME]/node_modules/react-feather/dist/index")' has no index signature

@LukyVj
Copy link

LukyVj commented Aug 22, 2019

You could always abstract this kind of behavour, but otherwise doing the following would allow you a similar experience (Almost).

const IconTag = Icon[this.props.icon];

<IconTag />

Thanks for the suggestion @Kosai106, this solved my issue :)

@agustingabiola
Copy link

agustingabiola commented May 4, 2020

I know this is a bit old but in case anyone needs it I managed to create a component that lazy loads the icons based on the name (you'll need to use lowercase) and works with Typescript:


interface DpFeatherIconProps {
  icon: string;
}

export default (props: DpFeatherIconProps) => {
  const IconTag = React.lazy(() => import(`react-feather/dist/icons/${props.icon}.js`));

  return (
    <Suspense fallback={null}>
      <IconTag />
    </Suspense>
  );
};

@kungpaogao
Copy link

kungpaogao commented Jul 4, 2020

@agustingabiola Thank you for this workaround. However, the use of lazy/Suspense seems to cause some "blinking" on some loads. Are there any other workarounds?

This is what I've done to prevent layouts from jumping around as much:

<Suspense
  fallback={
    <Loader
      color="white"
      className={props.className}
      size={props.size || 12}
    />
  }
>
...
</Suspense>

@Kosai106
Copy link

Kosai106 commented Jul 4, 2020

@kungpaogao For the fallback, just use an empty div with the same dimensions as your icon - This should prevent any layout shifting.

@kungpaogao
Copy link

@Kosai106 yup, that’s what I’m doing already, but it doesn’t solve the “blinking” that happens when the icon is loaded

@agustingabiola
Copy link

agustingabiola commented Jul 5, 2020

@Kosai106 yup, that’s what I’m doing already, but it doesn’t solve the “blinking” that happens when the icon is loaded

@kungpaogao Use memo for the component alongside with an element with the same dimensions for fallback. For example if you have a blue cross of 16px you can check that if all props are equal return the same rendered component

@kraenhansen
Copy link

kraenhansen commented Jul 11, 2020

For anyone interested, I just wrote this functional component which will render an icon by name (in TypeScript):

import React from "react";
import * as icons from "react-feather";

export type IconName = keyof typeof icons;

export type IconProps = {
  name: IconName;
} & icons.Props;

export function Icon({ name, ...rest }: IconProps) {
  const IconComponent = icons[name];
  return <IconComponent {...rest} />;
}

@kungpaogao
Copy link

@kraenhansen This is exactly what I'm looking for! Thank you!

@osvaldokalvaitir
Copy link

osvaldokalvaitir commented Jul 27, 2020

It would be very interesting to be able to pass on the name as property.

@LukyVj
Copy link

LukyVj commented Nov 5, 2020

Hello 👋

Nice solution from @kraenhansen, however, using this on next@10 & TypeScript result with the following error:

Server Error
Error: Element type is invalid: expected a string (for built-in components) 
or a class/function (for composite components) but got: undefined.
You likely forgot to export your component from the file it's defined in, 
or you might have mixed up default and named imports.
Screenshot 2020-11-05 at 10 49 26

If anyone got a clue or a solution for this.

Below, my current files:

Icon/Icon.tsx

import React from "react";
import * as icons from "react-feather";

export type IconName = keyof typeof icons;

export type IconProps = {
  name: IconName;
} & icons.Props;

export function Icon({ name, ...rest }: IconProps) {
  const IconComponent = icons[name];
  return <IconComponent {...rest} />;
}

And where I use the icon:

import { Icon } from "../Icon/Icon";

...
...
...
{icon && <Icon name={icon} />}

From what I understand, the exports/imports seems to be wrong somehow, I've tried several ways to fix this but none works, any hint would be appreciated 😄

@Kosai106
Copy link

Kosai106 commented Nov 5, 2020

@LukyVj It works just fine in Next.js from what I can tell.
https://codesandbox.io/s/hopeful-franklin-xr1fe?file=/src/pages/index.tsx

From the error message you're sharing it seems to come from somewhere else.

@LukyVj
Copy link

LukyVj commented Nov 5, 2020

@LukyVj It works just fine in Next.js from what I can tell.
codesandbox.io/s/hopeful-franklin-xr1fe?file=/src/pages/index.tsx

From the error message you're sharing it seems to come from somewhere else.

Thanks for answering!

So, yeah, I've tried your code, works well when I pass the name directly as a string Icon name="box" works just fine.

But, when I tried to pass it through a prop, it doesn't work, e.g:

Where I define the needed icon

  <Button
        onClick={() => {
          toggleOpen(!open);
        }}
        primary
        className="mb-16"
        icon="Box"
      />

The button code:

{icon && <Icon name={icon} />}

So, I've somehow understood why this doesn't work based on your comment, turns out my component get rendered several times including one where the icon is undefined, I suppose that's what's causing the error :D

@davidperklin
Copy link

@LukyVj It works just fine in Next.js from what I can tell.
codesandbox.io/s/hopeful-franklin-xr1fe?file=/src/pages/index.tsx
From the error message you're sharing it seems to come from somewhere else.

Thanks for answering!

So, yeah, I've tried your code, works well when I pass the name directly as a string Icon name="box" works just fine.

But, when I tried to pass it through a prop, it doesn't work, e.g:

Where I define the needed icon

  <Button
        onClick={() => {
          toggleOpen(!open);
        }}
        primary
        className="mb-16"
        icon="Box"
      />

The button code:

{icon && <Icon name={icon} />}

So, I've somehow understood why this doesn't work based on your comment, turns out my component get rendered several times including one where the icon is undefined, I suppose that's what's causing the error :D

Have you figured out a solution for this?

@GCobo
Copy link

GCobo commented Dec 13, 2020

@kraenhansen thanks!!

@lawnchamp
Copy link

@kraenhansen doesn't that solution have the problem of all 280 icons being included into your webpack bundle so that you can pick components dynamically? Tree shaking wouldn't be able to work in this case right?

@BenSampo
Copy link

BenSampo commented Dec 9, 2022

For anyone interested, I just wrote this functional component which will render an icon by name (in TypeScript):

import React from "react";
import * as icons from "react-feather";

export type IconName = keyof typeof icons;

export type IconProps = {
  name: IconName;
} & icons.Props;

export function Icon({ name, ...rest }: IconProps) {
  const IconComponent = icons[name];
  return <IconComponent {...rest} />;
}

I have modified as follows, in case it's useful to anyone.

import React from "react";
import * as Icons from "react-feather";

export type IconName = keyof typeof Icons;

export type IconProps = {
    name: IconName;
} & Icons.IconProps;

const Icon: React.FC<IconProps> = ({ name, ...rest }) => {
    const IconComponent = Icons[name];
    return <IconComponent {...rest} />;
};

export default Icon;

Usage is <Icon name="Download" />

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests