Add tests for VirtualizedListBox
Signed-off-by: danny-may <dannyjmay97@gmail.com>
This commit is contained in:
@@ -0,0 +1,5 @@
|
||||
---
|
||||
'@backstage/plugin-scaffolder': patch
|
||||
---
|
||||
|
||||
Add an extra bit of height to the EntityPicker dropdown to make it clear there are more options to select from, and to remove the scroll bar when there is less than 10 options
|
||||
@@ -0,0 +1,277 @@
|
||||
/*
|
||||
* Copyright 2024 The Backstage Authors
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
import React from 'react';
|
||||
import { VirtualizedListbox } from './VirtualizedListbox';
|
||||
import { renderInTestApp } from '@backstage/test-utils';
|
||||
|
||||
describe('<VirtualizedListbox />', () => {
|
||||
it('Should forward additional props to the outer div', async () => {
|
||||
const { baseElement } = await renderInTestApp(
|
||||
<VirtualizedListbox
|
||||
className="MuiAutocomplete-root MuiAutocomplete-hasClearIcon MuiAutocomplete-hasPopupIcon"
|
||||
aria-expanded="true"
|
||||
role="combobox"
|
||||
/>,
|
||||
);
|
||||
|
||||
expect(baseElement.children[0]).toMatchInlineSnapshot(`
|
||||
<div>
|
||||
<div>
|
||||
<div
|
||||
aria-expanded="true"
|
||||
class="MuiAutocomplete-root MuiAutocomplete-hasClearIcon MuiAutocomplete-hasPopupIcon"
|
||||
role="combobox"
|
||||
style="position: relative; height: 18px; width: 100%; overflow: auto; will-change: transform; direction: ltr;"
|
||||
>
|
||||
<div
|
||||
style="height: 0px; width: 100%;"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
`);
|
||||
});
|
||||
it('Should render with no items', async () => {
|
||||
const { baseElement } = await renderInTestApp(<VirtualizedListbox />);
|
||||
|
||||
expect(baseElement.children[0]).toMatchInlineSnapshot(`
|
||||
<div>
|
||||
<div>
|
||||
<div
|
||||
style="position: relative; height: 18px; width: 100%; overflow: auto; will-change: transform; direction: ltr;"
|
||||
>
|
||||
<div
|
||||
style="height: 0px; width: 100%;"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
`);
|
||||
});
|
||||
it('Should render with one item', async () => {
|
||||
const { baseElement } = await renderInTestApp(
|
||||
<VirtualizedListbox>
|
||||
<span>Item 1</span>
|
||||
</VirtualizedListbox>,
|
||||
);
|
||||
|
||||
expect(baseElement.children[0]).toMatchInlineSnapshot(`
|
||||
<div>
|
||||
<div>
|
||||
<div
|
||||
style="position: relative; height: 54px; width: 100%; overflow: auto; will-change: transform; direction: ltr;"
|
||||
>
|
||||
<div
|
||||
style="height: 36px; width: 100%;"
|
||||
>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 0px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item 1
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
`);
|
||||
});
|
||||
it('Should render with 10 items', async () => {
|
||||
const { baseElement } = await renderInTestApp(
|
||||
<VirtualizedListbox>
|
||||
{[...new Array(10)].map((_, i) => (
|
||||
<span key={i}>Item {i}</span>
|
||||
))}
|
||||
</VirtualizedListbox>,
|
||||
);
|
||||
|
||||
expect(baseElement.children[0]).toMatchInlineSnapshot(`
|
||||
<div>
|
||||
<div>
|
||||
<div
|
||||
style="position: relative; height: 378px; width: 100%; overflow: auto; will-change: transform; direction: ltr;"
|
||||
>
|
||||
<div
|
||||
style="height: 360px; width: 100%;"
|
||||
>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 0px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
0
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 36px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
1
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 72px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
2
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 108px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
3
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 144px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
4
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 180px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
5
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 216px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
6
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 252px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
7
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 288px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
8
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 324px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
9
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
`);
|
||||
});
|
||||
it('Should render up to 10.5 items (+ 2 buffer) even when there are many more', async () => {
|
||||
const { baseElement } = await renderInTestApp(
|
||||
<VirtualizedListbox>
|
||||
{[...new Array(100)].map((_, i) => (
|
||||
<span key={i}>Item {i}</span>
|
||||
))}
|
||||
</VirtualizedListbox>,
|
||||
);
|
||||
|
||||
expect(baseElement.children[0]).toMatchInlineSnapshot(`
|
||||
<div>
|
||||
<div>
|
||||
<div
|
||||
style="position: relative; height: 378px; width: 100%; overflow: auto; will-change: transform; direction: ltr;"
|
||||
>
|
||||
<div
|
||||
style="height: 3600px; width: 100%;"
|
||||
>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 0px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
0
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 36px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
1
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 72px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
2
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 108px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
3
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 144px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
4
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 180px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
5
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 216px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
6
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 252px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
7
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 288px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
8
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 324px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
9
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 360px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
10
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 396px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
11
|
||||
</span>
|
||||
<span
|
||||
style="position: absolute; left: 0px; top: 432px; height: 36px; width: 100%;"
|
||||
>
|
||||
Item
|
||||
12
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
`);
|
||||
});
|
||||
});
|
||||
@@ -17,22 +17,26 @@
|
||||
import React from 'react';
|
||||
import { FixedSizeList, ListChildComponentProps } from 'react-window';
|
||||
|
||||
type HTMLDivProps = React.HTMLAttributes<HTMLDivElement>;
|
||||
|
||||
const renderRow = (props: ListChildComponentProps) => {
|
||||
const { data, index, style } = props;
|
||||
return React.cloneElement(data[index], { style });
|
||||
};
|
||||
|
||||
// Context needed to keep Autocomplete working correctly : https://v4.mui.com/components/autocomplete/#Virtualize.tsx
|
||||
const OuterElementContext = React.createContext({});
|
||||
const OuterElementContext = React.createContext<HTMLDivProps>({});
|
||||
|
||||
const OuterElementType = React.forwardRef<HTMLDivElement>((props, ref) => {
|
||||
const outerProps = React.useContext(OuterElementContext);
|
||||
return <div ref={ref} {...props} {...outerProps} />;
|
||||
});
|
||||
const OuterElementType = React.forwardRef<HTMLDivElement, HTMLDivProps>(
|
||||
(props, ref) => {
|
||||
const outerProps = React.useContext(OuterElementContext);
|
||||
return <div ref={ref} {...props} {...outerProps} />;
|
||||
},
|
||||
);
|
||||
|
||||
export const VirtualizedListbox = React.forwardRef<
|
||||
HTMLDivElement,
|
||||
{ children?: React.ReactNode }
|
||||
HTMLDivProps
|
||||
>((props, ref) => {
|
||||
const { children, ...other } = props;
|
||||
const itemData = React.Children.toArray(children);
|
||||
@@ -40,7 +44,7 @@ export const VirtualizedListbox = React.forwardRef<
|
||||
|
||||
const itemSize = 36;
|
||||
|
||||
const itemsToShow = Math.min(10, itemCount);
|
||||
const itemsToShow = Math.min(10, itemCount) + 0.5;
|
||||
const height = itemsToShow * itemSize;
|
||||
|
||||
return (
|
||||
|
||||
Reference in New Issue
Block a user