Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/jakobhoeg/enhanced-button
An enhanced version of the default shadcn-button component
https://github.com/jakobhoeg/enhanced-button
button component component-library nextjs nextjs13 nextjs14 shadcn shadcn-ui shadcn-ui-button typescript ui-component ui-library
Last synced: about 10 hours ago
JSON representation
An enhanced version of the default shadcn-button component
- Host: GitHub
- URL: https://github.com/jakobhoeg/enhanced-button
- Owner: jakobhoeg
- License: mit
- Created: 2024-02-01T17:31:31.000Z (12 months ago)
- Default Branch: master
- Last Pushed: 2024-02-13T17:15:09.000Z (11 months ago)
- Last Synced: 2024-10-18T19:37:53.875Z (3 months ago)
- Topics: button, component, component-library, nextjs, nextjs13, nextjs14, shadcn, shadcn-ui, shadcn-ui-button, typescript, ui-component, ui-library
- Language: TypeScript
- Homepage: https://enhanced-button.vercel.app
- Size: 6.32 MB
- Stars: 435
- Watchers: 2
- Forks: 15
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- my-awesome-list - enhanced-button - button component | jakobhoeg | 673 | (TypeScript)
README
[](https://enhanced-button.vercel.app/)
# enhanced-button
An enhanced version of the default **shadcn-button** component.
Expands the default component by adding new beautiful button styles and features with minimal code, so you no longer have to create and manage multiple button components for your projects.
[Demo](https://enhanced-button.vercel.app/) • [Preview](#Preview) • [Requisites](#Requisites) • [Installation](#Installation) • [Usage](#Usage)
# Preview
https://github.com/jakobhoeg/enhanced-button/assets/114422072/ee05475f-6502-4e7a-8bde-ae5144ece85e
# Requisites
[shadcn-ui](https://ui.shadcn.com/docs/installation) and [shadcn-ui button component](https://ui.shadcn.com/docs/components/button) must be installed in your project.
# Installation
All it takes is **two** copy & paste and you're ready to go.
**_1. Copy the `button.tsx` component below and replace it with your existing one in `src/components/ui/button.tsx`_**
**_2. Copy the lines from `tailwind.config.ts` to your existing file._**
button.tsx
```tsx
import * as React from 'react';
import { Slot, Slottable } from '@radix-ui/react-slot';
import { cva, type VariantProps } from 'class-variance-authority';
import { cn } from '@/lib/utils';const buttonVariants = cva(
'inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium ring-offset-background transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:size-4 [&_svg]:shrink-0',
{
variants: {
variant: {
default: 'bg-primary text-primary-foreground hover:bg-primary/90',
destructive: 'bg-destructive text-destructive-foreground hover:bg-destructive/90',
outline: 'border border-input bg-background hover:bg-accent hover:text-accent-foreground',
secondary: 'bg-secondary text-secondary-foreground hover:bg-secondary/80',
ghost: 'hover:bg-accent hover:text-accent-foreground',
link: 'text-primary underline-offset-4 hover:underline',
},
effect: {
expandIcon: 'group gap-0 relative',
ringHover: 'transition-all duration-300 hover:ring-2 hover:ring-primary/90 hover:ring-offset-2',
shine:
'before:animate-shine relative overflow-hidden before:absolute before:inset-0 before:rounded-[inherit] before:bg-[linear-gradient(45deg,transparent_25%,rgba(255,255,255,0.5)_50%,transparent_75%,transparent_100%)] before:bg-[length:250%_250%,100%_100%] before:bg-no-repeat background-position_0s_ease',
shineHover:
'relative overflow-hidden before:absolute before:inset-0 before:rounded-[inherit] before:bg-[linear-gradient(45deg,transparent_25%,rgba(255,255,255,0.5)_50%,transparent_75%,transparent_100%)] before:bg-[length:250%_250%,100%_100%] before:bg-[position:200%_0,0_0] before:bg-no-repeat before:transition-[background-position_0s_ease] hover:before:bg-[position:-100%_0,0_0] before:duration-1000',
gooeyRight:
'relative z-0 overflow-hidden transition-all duration-500 before:absolute before:inset-0 before:-z-10 before:translate-x-[150%] before:translate-y-[150%] before:scale-[2.5] before:rounded-[100%] before:bg-gradient-to-r from-white/40 before:transition-transform before:duration-1000 hover:before:translate-x-[0%] hover:before:translate-y-[0%]',
gooeyLeft:
'relative z-0 overflow-hidden transition-all duration-500 after:absolute after:inset-0 after:-z-10 after:translate-x-[-150%] after:translate-y-[150%] after:scale-[2.5] after:rounded-[100%] after:bg-gradient-to-l from-white/40 after:transition-transform after:duration-1000 hover:after:translate-x-[0%] hover:after:translate-y-[0%]',
underline:
'relative !no-underline after:absolute after:bg-primary after:bottom-2 after:h-[1px] after:w-2/3 after:origin-bottom-left after:scale-x-100 hover:after:origin-bottom-right hover:after:scale-x-0 after:transition-transform after:ease-in-out after:duration-300',
hoverUnderline:
'relative !no-underline after:absolute after:bg-primary after:bottom-2 after:h-[1px] after:w-2/3 after:origin-bottom-right after:scale-x-0 hover:after:origin-bottom-left hover:after:scale-x-100 after:transition-transform after:ease-in-out after:duration-300',
},
size: {
default: 'h-10 px-4 py-2',
sm: 'h-9 rounded-md px-3',
lg: 'h-11 rounded-md px-8',
icon: 'h-10 w-10',
},
},
defaultVariants: {
variant: 'default',
size: 'default',
},
}
);interface IconProps {
icon: React.ElementType;
iconPlacement: 'left' | 'right';
}interface IconRefProps {
icon?: never;
iconPlacement?: undefined;
}export interface ButtonProps extends React.ButtonHTMLAttributes, VariantProps {
asChild?: boolean;
}export type ButtonIconProps = IconProps | IconRefProps;
const Button = React.forwardRef(
({ className, variant, effect, size, icon: Icon, iconPlacement, asChild = false, ...props }, ref) => {
const Comp = asChild ? Slot : 'button';
return (
{Icon &&
iconPlacement === 'left' &&
(effect === 'expandIcon' ? (
) : (
))}
{props.children}
{Icon &&
iconPlacement === 'right' &&
(effect === 'expandIcon' ? (
) : (
))}
);
}
);
Button.displayName = 'Button';export { Button, buttonVariants };
```tailwind.config.ts
```diff
const config = {
extend: {
keyframes: {
+ shine: {
+ '0%': { backgroundPosition: '200% 0' },
+ '25%': { backgroundPosition: '-200% 0' },
+ '100%': { backgroundPosition: '-200% 0' },
+ },
},
animation: {
+ shine: 'shine 3s ease-out infinite',
},
},
}
```# Usage
Example usage:
```tsx
import { Button } from './ui/button';Icon right
;
``````tsx
import { Button } from './ui/button';Gooey right;
```Mix with other variants:
```tsx
import { Button } from './ui/button';Outline with shine hover
;
```Check out the [demo](https://enhanced-button.vercel.app/) to see **all** the different styles.