-
-
Notifications
You must be signed in to change notification settings - Fork 28
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
Website: Implement glowing cursor effect #1009
base: main
Are you sure you want to change the base?
Conversation
The latest updates on your projects. Learn more about Vercel for Git ↗︎ 1 Skipped Deployment
|
📝 WalkthroughWalkthroughThe pull request introduces a glow hover effect system for UI components. It adds a new Changes
Sequence DiagramsequenceDiagram
participant User
participant Component
participant GlowHoverHook
participant GlowHoverEffect
User->>Component: Hover/Interact
Component->>GlowHoverHook: Apply Hover Effect
GlowHoverHook->>GlowHoverEffect: Initialize Effect
GlowHoverEffect-->>Component: Update Visual Styling
GlowHoverEffect->>GlowHoverEffect: Animate Glow
Possibly related issues
✨ Finishing Touches
Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media? 🪧 TipsChatThere are 3 ways to chat with CodeRabbit:
Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments. CodeRabbit Commands (Invoked using PR comments)
Other keywords and placeholders
CodeRabbit Configuration File (
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 3
🧹 Nitpick comments (3)
ui/src/components/use-glow-hover/glow-hover-effect.ts (1)
124-124
: Avoid assignments within arrow function expressionsUsing assignments within arrow function expressions without braces can be confusing and reduce code readability.
Apply the following diff to improve clarity:
-onIdUpdate: (newId) => (lightSizeEnterAnimationId = newId), +onIdUpdate: (newId) => { + lightSizeEnterAnimationId = newId; +},Repeat this change for lines 158, 177, and 196.
Also applies to: 158-158, 177-177, 196-196
🧰 Tools
🪛 Biome (1.9.4)
[error] 124-124: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.(lint/suspicious/noAssignInExpressions)
ui/src/components/use-glow-hover/use-glow-hover.ts (1)
13-13
: OptimizeuseEffect
dependencies to prevent unnecessary re-rendersUsing
Object.values(options)
in the dependency array may cause the effect to run on every render ifoptions
is not memoized, leading to performance issues.Consider memoizing the
options
object or explicitly listing each option in the dependency array.ui/src/components/use-glow-hover/linear-animation.ts (1)
14-15
: ClampinitialProgress
to the range [0, 1]Currently,
initialProgress
can be outside the range[0, 1]
, which may cause unexpected behavior in the animation.Apply this diff to ensure
initialProgress
is within the valid range:}: LinearAnimationParams) => { + initialProgress = Math.min(Math.max(initialProgress, 0), 1); if (time === 0) {
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (7)
ui/src/components/containers/base-container.tsx
(1 hunks)ui/src/components/use-glow-hover/glow-hover-effect.ts
(1 hunks)ui/src/components/use-glow-hover/index.ts
(1 hunks)ui/src/components/use-glow-hover/linear-animation.ts
(1 hunks)ui/src/components/use-glow-hover/use-glow-hover.ts
(1 hunks)ui/src/index.ts
(1 hunks)website/src/app/[lang]/[region]/(website)/(home)/(sections)/overview.tsx
(2 hunks)
✅ Files skipped from review due to trivial changes (1)
- ui/src/components/use-glow-hover/index.ts
🧰 Additional context used
🪛 GitHub Actions: Website
ui/src/components/containers/base-container.tsx
[error] 14-14: Property 'wrapperClassName' does not exist on type '{ backgroundColor?: BackgroundColor | undefined; wrapperRef?: Ref | undefined; } & HTMLAttributes'
🪛 Biome (1.9.4)
ui/src/components/use-glow-hover/glow-hover-effect.ts
[error] 124-124: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 158-158: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 177-177: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 196-196: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
🔇 Additional comments (2)
ui/src/index.ts (1)
30-30
: LGTM!The new export follows the existing pattern and maintains alphabetical order.
website/src/app/[lang]/[region]/(website)/(home)/(sections)/overview.tsx (1)
14-26
: Remove temporary example code.The TODO comment indicates this code should be removed before merge. Please remove this example usage of GLowHoverContainer.
export const GLowHoverContainer = React.forwardRef<HTMLDivElement, Omit<BaseContainerProps, 'wrapperRef'>>( | ||
({ className, ...props }, ref) => { | ||
const refCard = useGlowHover({ lightColor: '#CEFF00' }); | ||
|
||
return ( | ||
<BaseContainer | ||
wrapperClassName={twMerge('theme-blue', className)} | ||
ref={ref} | ||
{...props} | ||
wrapperRef={refCard as React.Ref<HTMLDivElement>} | ||
/> | ||
); | ||
}, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Improve type safety and configuration of GLowHoverContainer.
Two suggestions for improvement:
- Avoid type casting of refCard
- Make the light color configurable
-export const GLowHoverContainer = React.forwardRef<HTMLDivElement, Omit<BaseContainerProps, 'wrapperRef'>>(
+type GlowHoverContainerProps = Omit<BaseContainerProps, 'wrapperRef'> & {
+ lightColor?: string;
+};
+
+export const GLowHoverContainer = React.forwardRef<HTMLDivElement, GlowHoverContainerProps>(
- ({ className, ...props }, ref) => {
+ ({ className, lightColor = '#CEFF00', ...props }, ref) => {
- const refCard = useGlowHover({ lightColor: '#CEFF00' });
+ const refCard = useGlowHover({ lightColor });
return (
<BaseContainer
wrapperClassName={twMerge('theme-blue', className)}
ref={ref}
{...props}
- wrapperRef={refCard as React.Ref<HTMLDivElement>}
+ wrapperRef={refCard}
/>
);
},
);
📝 Committable suggestion
‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.
export const GLowHoverContainer = React.forwardRef<HTMLDivElement, Omit<BaseContainerProps, 'wrapperRef'>>( | |
({ className, ...props }, ref) => { | |
const refCard = useGlowHover({ lightColor: '#CEFF00' }); | |
return ( | |
<BaseContainer | |
wrapperClassName={twMerge('theme-blue', className)} | |
ref={ref} | |
{...props} | |
wrapperRef={refCard as React.Ref<HTMLDivElement>} | |
/> | |
); | |
}, | |
type GlowHoverContainerProps = Omit<BaseContainerProps, 'wrapperRef'> & { | |
lightColor?: string; | |
}; | |
export const GLowHoverContainer = React.forwardRef<HTMLDivElement, GlowHoverContainerProps>( | |
({ className, lightColor = '#CEFF00', ...props }, ref) => { | |
const refCard = useGlowHover({ lightColor }); | |
return ( | |
<BaseContainer | |
wrapperClassName={twMerge('theme-blue', className)} | |
ref={ref} | |
{...props} | |
wrapperRef={refCard} | |
/> | |
); | |
}, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@ssandino hi!
Could you please tell if you need the glowing color to be configurable?
As I can see now, the blue color mostly used throughout the site to make banners and accent sections and yellow cursor glow fits perfectly here. However, there's no problem adding prop for a custom color.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Seems like the humans are having a chat. I'll hop back into my burrow for now. If you need me again, just tag @coderabbitai in a new comment, and I'll come hopping out!
<GLowHoverContainer className="mb-16 mt-8 flex flex-col items-center justify-center md:mb-48 md:mt-16"> | ||
<Typography size="3xl" weight="medium" className="my-10 text-center"> | ||
{translator.t('section-2.title-1')} | ||
</Typography> | ||
<div className="mb-8 text-center"> | ||
{translator.t<{ text: string; color?: FontColor }[]>('section-2.title-2').map((title, index) => ( | ||
<Typography as="span" size="3xl" weight="medium" color={title.color} key={index}> | ||
{title.text}{' '} | ||
</Typography> | ||
))} | ||
</div> | ||
</GLowHoverContainer> | ||
|
||
<BaseContainer className="mx-auto mb-16 mt-8 flex w-4/5 flex-col items-center justify-center md:mb-48 md:mt-16 lg:w-3/5"> | ||
<Typography size="3xl" weight="medium" className="my-10 text-center"> | ||
{translator.t('section-2.title-1')} | ||
</Typography> | ||
<div className="mb-8 text-center"> | ||
{translator.t<{ text: string; color?: FontColor }[]>('section-2.title-2').map((title, index) => ( | ||
<Typography as="span" size="3xl" weight="medium" color={title.color} key={index}> | ||
{title.text}{' '} | ||
</Typography> | ||
))} | ||
</div> | ||
<Typography size="3xl" weight="medium" className="mb-5 mt-12 text-center"> | ||
{translator.t('section-2.title-3')} | ||
</Typography> | ||
<ol className="mb-8 list-decimal"> | ||
<li> | ||
<Typography as="span">{translator.t('section-2.text-3.1')}</Typography> | ||
</li> | ||
<li> | ||
<Typography as="span">{translator.t('section-2.text-3.2')}</Typography> | ||
</li> | ||
</ol> | ||
<Typography size="3xl" weight="medium" className="mb-5 mt-12 text-center"> | ||
{translator.t('section-2.title-4')} | ||
</Typography> | ||
<Typography className="mb-8">{translator.t('section-2.text-4')}</Typography> | ||
<Typography size="3xl" weight="medium" className="mb-5 mt-12 text-center"> | ||
{translator.t('section-2.title-5')} | ||
</Typography> | ||
<Typography className="w-full">{translator.t('section-2.text-5')}</Typography> | ||
</BaseContainer> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Avoid content duplication.
The same content is duplicated between GLowHoverContainer and BaseContainer. Choose one container based on the intended design.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
JFI: this part of duplicated content will be removed before merge
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@IPogorelova Thanks for clarifying! I see the TODO comment now, and understand this is just a temporary example.
If you found this review helpful, would you consider giving us a shout-out on X?
Thank you for using CodeRabbit!
Visit the preview URL for this PR (updated for commit b3b8e29): https://si-admin-staging--pr1009-implement-glowing-cu-0qtybgf8.web.app (expires Tue, 21 Jan 2025 18:13:21 GMT) 🔥 via Firebase Hosting GitHub Action 🌎 Sign: b7b0969384059dce6ea8fad1ee1d1737e54e6676 |
The deployment on Vercel preview throws following error:
And the co-pilot bug report says: |
… into implement-glowing-cursor-effect
…-effect-wip-fix' into sandino/implement-glowing-cursor-effect-wip-fix # Conflicts: # ui/src/components/use-glow-hover/glow-hover-effect.ts
…-effect-wip-fix' into sandino/implement-glowing-cursor-effect-wip-fix
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actionable comments posted: 1
🧹 Nitpick comments (3)
ui/src/components/use-glow-hover/presets.ts (1)
1-11
: Add documentation for preset values.Consider adding JSDoc comments to explain:
- The significance of the chosen color value
- The reasoning behind animation timing values
- The impact of each configuration option
ui/src/components/use-glow-hover/glow-hover-effect.ts (2)
32-48
: Optimize color parsing implementation.The current implementation creates and removes DOM elements for each color parse, which is inefficient. Consider:
- Using a regex-only approach for common color formats
- Caching parsed results for repeated colors
+const colorCache = new Map<string, number[]>(); + export function parseColor(colorToParse: string) { + if (colorCache.has(colorToParse)) { + return colorCache.get(colorToParse)!; + } const div = document.createElement('div'); // ... rest of the implementation document.body.removeChild(div); if (parsedColor) { const alpha = typeof parsedColor[4] === 'undefined' ? 1 : parsedColor[4]; - return [parsedColor[1], parsedColor[2], parsedColor[3], alpha]; + const result = [parsedColor[1], parsedColor[2], parsedColor[3], alpha]; + colorCache.set(colorToParse, result); + return result; } else { console.error(`Color ${colorToParse} could not be parsed.`); return [0, 0, 0, 0]; } }
108-219
: Improve event handler implementation.
- Extract duplicated position update logic into a shared function
- Consider using AbortController for cleanup during unmount
+const updateMousePosition = (e: MouseEvent, el: HTMLElement) => { + const curBox = el.getBoundingClientRect(); + return { + mousePos: { x: e.clientX, y: e.clientY }, + elPos: { x: curBox.left, y: curBox.top } + }; +};🧰 Tools
🪛 Biome (1.9.4)
[error] 129-129: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.(lint/suspicious/noAssignInExpressions)
[error] 171-171: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.(lint/suspicious/noAssignInExpressions)
[error] 194-194: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.(lint/suspicious/noAssignInExpressions)
[error] 217-217: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.(lint/suspicious/noAssignInExpressions)
📜 Review details
Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro
📒 Files selected for processing (3)
ui/src/components/use-glow-hover/glow-hover-effect.ts
(1 hunks)ui/src/components/use-glow-hover/linear-animation.ts
(1 hunks)ui/src/components/use-glow-hover/presets.ts
(1 hunks)
🚧 Files skipped from review as they are similar to previous changes (1)
- ui/src/components/use-glow-hover/linear-animation.ts
🧰 Additional context used
🪛 Biome (1.9.4)
ui/src/components/use-glow-hover/glow-hover-effect.ts
[error] 129-129: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 171-171: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 194-194: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
[error] 217-217: The assignment should not be in an expression.
The use of assignments in expressions is confusing.
Expressions are often considered as side-effect free.
(lint/suspicious/noAssignInExpressions)
⏰ Context from checks skipped due to timeout of 90000ms (2)
- GitHub Check: Prettify
- GitHub Check: Security checks (typescript)
🔇 Additional comments (1)
ui/src/components/use-glow-hover/glow-hover-effect.ts (1)
221-265
: Well-implemented cleanup handling!The code properly manages:
- Event listener cleanup
- Animation frame cancellation
- ResizeObserver disconnection
export const glowHoverEffect = (el: HTMLElement, { preset, ...options }: GlowHoverOptions) => { | ||
if (!el) { | ||
return () => {}; | ||
} | ||
|
||
const lightColor = options.lightColor ?? '#CEFF00'; | ||
const lightSize = options.lightSize ?? 130; | ||
const lightSizeEnterAnimationTime = options.lightSizeEnterAnimationTime ?? 100; | ||
const lightSizeLeaveAnimationTime = options.lightSizeLeaveAnimationTime ?? 50; | ||
const isElementMovable = options.isElementMovable ?? false; | ||
const customStaticBg = options.customStaticBg ?? null; | ||
|
||
const enableBurst = options.enableBurst ?? false; | ||
|
||
const getResolvedHoverBg = () => getComputedStyle(el).backgroundColor; | ||
|
||
let resolvedHoverBg = getResolvedHoverBg(); | ||
|
||
// default bg (if not defined) is rgba(0, 0, 0, 0) which is bugged in gradients in Safari | ||
// so we use transparent lightColor instead | ||
const parsedLightColor = parseColor(lightColor); | ||
const parsedLightColorRGBString = parsedLightColor.slice(0, 3).join(','); | ||
const resolvedGradientBg = `rgba(${parsedLightColorRGBString},0)`; | ||
|
||
let isMouseInside = false; | ||
let currentLightSize = 0; | ||
let blownSize = 0; | ||
let lightSizeEnterAnimationId: number | undefined = undefined; | ||
let lightSizeLeaveAnimationId: number | undefined = undefined; | ||
let blownSizeIncreaseAnimationId: number | undefined = undefined; | ||
let blownSizeDecreaseAnimationId: number | undefined = undefined; | ||
let lastMousePos: Coords; | ||
const defaultBox = el.getBoundingClientRect(); | ||
let lastElPos: Coords = { x: defaultBox.left, y: defaultBox.top }; | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Refactor animation ID management.
- Address the assignments in expressions flagged by static analysis
- Consider centralizing animation cleanup
+const cancelAnimations = (state: {
+ lightSizeEnterAnimationId?: number;
+ lightSizeLeaveAnimationId?: number;
+ blownSizeIncreaseAnimationId?: number;
+ blownSizeDecreaseAnimationId?: number;
+}) => {
+ Object.entries(state).forEach(([_, id]) => {
+ if (id !== undefined) {
+ window.cancelAnimationFrame(id);
+ }
+ });
+};
Committable suggestion skipped: line range outside the PR's diff.
This PR introduces a new component,
GlowHoverContainer
, which extendsBaseContainer
with a glowing cursor effect.Issue: #987
Changes:
GlowHoverContainer
: a new component that wrapsBaseContainer
and adds the glowing cursor effect.BaseContainer
: minor adjustments to enable the glowing cursor functionality:backgroundColor
property, which was previously used on both the parent and child<div>
s within the component. This change appears to have no impact on existing functionality, as I've found only 1 usage ofbackgroundColor
property in the project (on the /arts page) and it works fine after component's change.GlowHoverContainer
usage on the main page below the Hero section. (TODO: Remove this example before merging)Note:
npm run format-code
produced unexpected changes across all repo files, primarily related to line breaks. To avoid unintended consequences, I did not commit these changes.I understand that consistent code formatting is important. However, I'm encountering difficulties with the current formatting command locally. If you have any suggestions or have ever encountered the same issue, I'd really appreciate any advice.
Hope this helps!
Summary by CodeRabbit
New Features
GLowHoverContainer
component for enhanced visual interaction.Improvements
Technical Enhancements