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

feat(scroll btn): Add SelectScrollButton for scrolling dropdown, visual overflow indication #7579

Open
wants to merge 1 commit into
base: main
Choose a base branch
from

Conversation

thitt7
Copy link

@thitt7 thitt7 commented Mar 28, 2025

Description

This PR introduces a new component SelectScrollButton imported into the Select component that tracks the content/scrollability of the Select component and conditionally renders a scrolldown button that scrolls the select element down when hovered over as well as serving as a visual indicator of overflow since certain browsers opt to hide scrollbars. I tried first to use the ScrollDownButton component as suggested by user 'shadowspawn' in the issue thread, but it seems that it's not designed to be used alongside the component ScrollArea(ScrollPrimitive) while inside the Select component and does not render anything in that scenario. The behavior of the new component is closely matched to that of Radix UI's Select.ScrollDownButton. Currently it only works for scrolling down and I wanted to make sure this approach is acceptable before moving forward and potentially giving it the ability to stick to the top of the container and scroll up as well as down. This is my first contribution to this repository, please let me know if I missed anything or if you think it's necessary for me to add tests for this component, etc.

Validation

I manually tested the dropdowns on the downloads page on my (windows 10) desktop on firefox, chrome and edge as well as my android phone and it appears to function as intended.

Screenshot (2)

Related Issues

#7468

Check List

  • I have read the Contributing Guidelines and made commit messages that follow the guideline.
  • I have run npm run format to ensure the code follows the style guide.
  • I have run npm run test to check if all tests are passing.
  • I have run npx turbo build to check if the website builds without errors.
  • I've covered new added functionality with unit tests if necessary.

@thitt7 thitt7 requested a review from a team as a code owner March 28, 2025 04:10
Copy link

vercel bot commented Mar 28, 2025

The latest updates on your projects. Learn more about Vercel for Git ↗︎

Name Status Preview Updated (UTC)
nodejs-org ✅ Ready (Inspect) Visit Preview Mar 28, 2025 4:11am

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could you add stories for this component ?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

(Keep in mind, that, ideally, the stories will show this component working within a Select component, and not on its own)

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

we do it for pagination

@@ -0,0 +1,130 @@
import { ChevronDownIcon, ChevronUpIcon } from '@heroicons/react/24/outline';
import { useEffect, useRef, useState } from 'react';
import { type FC, type RefObject } from 'react';
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
import { type FC, type RefObject } from 'react';
import type { FC, RefObject } from 'react';

import { useEffect, useRef, useState } from 'react';
import { type FC, type RefObject } from 'react';

import styles from '@node-core/ui-components/Common/Select/index.module.css';
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
import styles from '@node-core/ui-components/Common/Select/index.module.css';
import styles from '@node-core/ui-components/Common/Select/SelectScrollButton/index.module.css';

Maybe use dedicate css-module to simplify code

direction,
selectContentRef,
scrollAmount = 35,
scrollInterval = 50,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
scrollInterval = 50,
scrollInterval = 500,

for moment it's to quick

Comment on lines +153 to +180
.scrollBtn {
@apply sticky
z-10
flex
w-full
cursor-pointer
justify-center
bg-white
p-1
transition-colors
hover:bg-neutral-100
dark:bg-neutral-950
dark:hover:bg-neutral-900;
}

.scrollBtn[data-direction='down'] {
bottom: 0;
}

.scrollBtn[data-direction='up'] {
top: 0;
}

.scrollBtnIcon {
@apply size-5
text-neutral-600
dark:text-neutral-400;
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
.scrollBtn {
@apply sticky
z-10
flex
w-full
cursor-pointer
justify-center
bg-white
p-1
transition-colors
hover:bg-neutral-100
dark:bg-neutral-950
dark:hover:bg-neutral-900;
}
.scrollBtn[data-direction='down'] {
bottom: 0;
}
.scrollBtn[data-direction='up'] {
top: 0;
}
.scrollBtnIcon {
@apply size-5
text-neutral-600
dark:text-neutral-400;
}
.scrollBtn {
@apply sticky
z-10
flex
w-full
cursor-pointer
justify-center
bg-white
p-1
transition-colors
hover:bg-neutral-100
dark:bg-neutral-950
dark:hover:bg-neutral-900;
&[data-direction='down'] {
@apply bottom-0;
}
&[data-direction='up'] {
@apply top-0
}
.scrollBtnIcon {
@apply size-5
text-neutral-600
dark:text-neutral-400;
}
}

you can use nested css to make it easier to maintain

Comment on lines +44 to +64
if (direction === 'down') {
container.scrollBy({ top: scrollAmount, behavior: 'smooth' });

if (
container.scrollTop >=
container.scrollHeight - container.clientHeight
) {
clearScrollInterval();
setIsVisible(false);
}
} else {
container.scrollBy({
top: -Math.abs(scrollAmount),
behavior: 'smooth',
});

if (container.scrollTop <= 0) {
clearScrollInterval();
setIsVisible(false);
}
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is an extremely extremely minor nitpick, but this could probably be simplified into a helper function?

(But this is such a tiny nitpick that you really don't have to do anything)

justify-center
bg-white
p-1
transition-colors
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
transition-colors
motion-safe:transition-colors

scrollInterval?: number;
};

const SelectScrollButton: FC<SelectScrollButtonProps> = ({
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Primitive radix select already has ScrollUpButton and ScrollDownButton Instead of adding these features, you can use what radix provides directly. Here is an example of primitive select anatomy; https://www.radix-ui.com/primitives/docs/components/select#anatomy

@avivkeller
Copy link
Member

avivkeller commented Mar 28, 2025

The up chevron doesn't ever appear to show, can you investigate? It looks like only the down arrow shows.

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

Successfully merging this pull request may close these issues.

No visible cue on Mac that dropdown has hidden scrollable items, affecting downloads page et al
4 participants