2021-05-22 19:30:11 +02:00
|
|
|
import useLayoutItemSizes, { itemSize, calculateMaxSizeAvailableForItem } from './useLayoutItemSizes';
|
2020-11-13 19:09:28 +02:00
|
|
|
import { LayoutItem, LayoutItemDirection } from './types';
|
|
|
|
import { renderHook } from '@testing-library/react-hooks';
|
|
|
|
import validateLayout from './validateLayout';
|
|
|
|
|
|
|
|
describe('useLayoutItemSizes', () => {
|
|
|
|
|
|
|
|
test('should validate the layout', () => {
|
|
|
|
const layout: LayoutItem = validateLayout({
|
|
|
|
key: 'root',
|
|
|
|
width: 200,
|
|
|
|
height: 100,
|
|
|
|
direction: LayoutItemDirection.Row,
|
|
|
|
children: [
|
|
|
|
{ key: 'col1' },
|
|
|
|
{ key: 'col2' },
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(layout.isRoot).toBe(true);
|
|
|
|
});
|
|
|
|
|
2021-04-30 18:54:55 +02:00
|
|
|
test('should stretch the last visible child item if all siblings have fixed size and the last child is not visible', () => {
|
|
|
|
const layout: LayoutItem = validateLayout({
|
|
|
|
key: 'root',
|
|
|
|
width: 200,
|
|
|
|
height: 100,
|
|
|
|
direction: LayoutItemDirection.Row,
|
|
|
|
children: [
|
|
|
|
{ key: 'col1', width: 50 },
|
|
|
|
{ key: 'col2', width: 50 },
|
|
|
|
{ key: 'col3', width: 70, visible: false },
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
const col1 = layout.children.find(c => c.key === 'col1');
|
|
|
|
expect(col1.width).toBe(50);
|
|
|
|
expect(col1.visible).toBe(true);
|
|
|
|
|
|
|
|
const col2 = layout.children.find(c => c.key === 'col2');
|
|
|
|
expect(col2).not.toHaveProperty('width');
|
|
|
|
expect(col2.visible).toBe(true);
|
|
|
|
|
|
|
|
const col3 = layout.children.find(c => c.key === 'col3');
|
|
|
|
expect(col3.width).toBe(70);
|
|
|
|
expect(col3.visible).toBe(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should stretch the last child item if all siblings have fixed size', () => {
|
|
|
|
const layout: LayoutItem = validateLayout({
|
|
|
|
key: 'root',
|
|
|
|
width: 200,
|
|
|
|
height: 100,
|
|
|
|
direction: LayoutItemDirection.Row,
|
|
|
|
children: [
|
|
|
|
{ key: 'col1', width: 50 },
|
|
|
|
{ key: 'col2', width: 50 },
|
|
|
|
{ key: 'col3', width: 70 },
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
const col1 = layout.children.find(c => c.key === 'col1');
|
|
|
|
expect(col1.width).toBe(50);
|
|
|
|
expect(col1.visible).toBe(true);
|
|
|
|
|
|
|
|
const col2 = layout.children.find(c => c.key === 'col2');
|
|
|
|
expect(col2.width).toBe(50);
|
|
|
|
expect(col2.visible).toBe(true);
|
|
|
|
|
|
|
|
const col3 = layout.children.find(c => c.key === 'col3');
|
|
|
|
expect(col3).not.toHaveProperty('width');
|
|
|
|
expect(col3.visible).toBe(true);
|
|
|
|
});
|
|
|
|
|
2020-11-13 19:09:28 +02:00
|
|
|
test('should give item sizes', () => {
|
|
|
|
const layout: LayoutItem = validateLayout({
|
|
|
|
key: 'root',
|
|
|
|
width: 200,
|
|
|
|
height: 100,
|
|
|
|
direction: LayoutItemDirection.Row,
|
|
|
|
children: [
|
|
|
|
{
|
|
|
|
key: 'col1',
|
|
|
|
width: 50,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'col2',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
const { result } = renderHook(() => useLayoutItemSizes(layout));
|
|
|
|
const sizes = result.current;
|
|
|
|
|
|
|
|
expect(sizes.root).toEqual({ width: 200, height: 100 });
|
|
|
|
expect(sizes.col1).toEqual({ width: 50, height: 100 });
|
|
|
|
expect(sizes.col2).toEqual({ width: 150, height: 100 });
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should leave room for the resizer controls', () => {
|
|
|
|
const layout: LayoutItem = validateLayout({
|
|
|
|
key: 'root',
|
|
|
|
width: 200,
|
|
|
|
height: 100,
|
|
|
|
direction: LayoutItemDirection.Row,
|
|
|
|
children: [
|
|
|
|
{
|
|
|
|
key: 'col1',
|
|
|
|
resizableRight: true,
|
|
|
|
direction: LayoutItemDirection.Column,
|
|
|
|
children: [
|
|
|
|
{ key: 'row1', resizableBottom: true },
|
|
|
|
{ key: 'row2' },
|
|
|
|
],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'col2',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
const { result } = renderHook(() => useLayoutItemSizes(layout));
|
|
|
|
|
|
|
|
const sizes = result.current;
|
|
|
|
|
|
|
|
expect(sizes).toEqual({
|
|
|
|
root: { width: 200, height: 100 },
|
|
|
|
col1: { width: 100, height: 100 },
|
|
|
|
col2: { width: 100, height: 100 },
|
|
|
|
row1: { width: 100, height: 50 },
|
|
|
|
row2: { width: 100, height: 50 },
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(itemSize(layout.children[0], layout, sizes, true)).toEqual({ width: 100, height: 100 });
|
|
|
|
|
|
|
|
const parent = layout.children[0];
|
|
|
|
expect(itemSize(parent.children[0], parent, sizes, false)).toEqual({ width: 95, height: 45 });
|
|
|
|
expect(itemSize(parent.children[1], parent, sizes, false)).toEqual({ width: 95, height: 50 });
|
|
|
|
});
|
|
|
|
|
2021-05-22 19:30:11 +02:00
|
|
|
test('should decrease size of the largest item if the total size would be larger than the container', () => {
|
|
|
|
const layout: LayoutItem = validateLayout({
|
|
|
|
key: 'root',
|
|
|
|
width: 200,
|
|
|
|
height: 100,
|
|
|
|
direction: LayoutItemDirection.Row,
|
|
|
|
children: [
|
|
|
|
{
|
|
|
|
key: 'col1',
|
|
|
|
width: 110,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'col2',
|
|
|
|
width: 100,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'col3',
|
|
|
|
minWidth: 50,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
const { result } = renderHook(() => useLayoutItemSizes(layout));
|
|
|
|
const sizes = result.current;
|
|
|
|
|
|
|
|
expect(sizes.col1.width).toBe(50);
|
|
|
|
expect(sizes.col2.width).toBe(100);
|
|
|
|
expect(sizes.col3.width).toBe(50);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should not allow a minWidth of 0, should still make space for the item', () => {
|
|
|
|
const layout: LayoutItem = validateLayout({
|
|
|
|
key: 'root',
|
|
|
|
width: 200,
|
|
|
|
height: 100,
|
|
|
|
direction: LayoutItemDirection.Row,
|
|
|
|
children: [
|
|
|
|
{
|
|
|
|
key: 'col1',
|
|
|
|
width: 210,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'col2',
|
|
|
|
minWidth: 0,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
const { result } = renderHook(() => useLayoutItemSizes(layout));
|
|
|
|
const sizes = result.current;
|
|
|
|
|
|
|
|
expect(sizes.col1.width).toBe(160);
|
|
|
|
expect(sizes.col2.width).toBe(40); // default minWidth is 40
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should ignore invisible items when counting remaining size', () => {
|
|
|
|
const layout: LayoutItem = validateLayout({
|
|
|
|
key: 'root',
|
|
|
|
width: 200,
|
|
|
|
height: 100,
|
|
|
|
direction: LayoutItemDirection.Row,
|
|
|
|
children: [
|
|
|
|
{
|
|
|
|
key: 'col1',
|
|
|
|
width: 110,
|
|
|
|
visible: false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'col2',
|
|
|
|
width: 100,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'col3',
|
|
|
|
minWidth: 50,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
const { result } = renderHook(() => useLayoutItemSizes(layout));
|
|
|
|
const sizes = result.current;
|
|
|
|
|
|
|
|
expect(sizes.col1.width).toBe(0);
|
|
|
|
expect(sizes.col2.width).toBe(100);
|
|
|
|
expect(sizes.col3.width).toBe(100);
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should ignore invisible items when selecting largest child', () => {
|
|
|
|
const layout: LayoutItem = validateLayout({
|
|
|
|
key: 'root',
|
|
|
|
width: 200,
|
|
|
|
height: 100,
|
|
|
|
direction: LayoutItemDirection.Row,
|
|
|
|
children: [
|
|
|
|
{
|
|
|
|
key: 'col1',
|
|
|
|
width: 110,
|
|
|
|
visible: false,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'col2',
|
|
|
|
width: 100,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'col3',
|
|
|
|
width: 110,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'col4',
|
|
|
|
minWidth: 50,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
const { result } = renderHook(() => useLayoutItemSizes(layout));
|
|
|
|
const sizes = result.current;
|
|
|
|
|
|
|
|
expect(sizes.col1.width).toBe(0);
|
|
|
|
expect(sizes.col2.width).toBe(100);
|
|
|
|
expect(sizes.col3.width).toBe(50);
|
|
|
|
expect(sizes.col4.width).toBe(50);
|
|
|
|
});
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('calculateMaxSizeAvailableForItem', () => {
|
|
|
|
|
|
|
|
test('should give maximum available space this item can take up during resizing', () => {
|
|
|
|
const layout: LayoutItem = validateLayout({
|
|
|
|
key: 'root',
|
|
|
|
width: 200,
|
|
|
|
height: 100,
|
|
|
|
direction: LayoutItemDirection.Row,
|
|
|
|
children: [
|
|
|
|
{
|
|
|
|
key: 'col1',
|
|
|
|
width: 50,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'col2',
|
|
|
|
width: 70,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'col3',
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
const { result } = renderHook(() => useLayoutItemSizes(layout));
|
|
|
|
const sizes = result.current;
|
|
|
|
const maxSize1 = calculateMaxSizeAvailableForItem(layout.children[0], layout, sizes);
|
|
|
|
const maxSize2 = calculateMaxSizeAvailableForItem(layout.children[1], layout, sizes);
|
|
|
|
|
|
|
|
// maxSize = totalSize - ( [size of items with set size, except for the current item] + [minimum size of items with no set size] )
|
|
|
|
expect(maxSize1.width).toBe(90); // 90 = layout.width - (col2.width + col3.minWidth(=40) )
|
|
|
|
expect(maxSize2.width).toBe(110); // 110 = layout.width - (col1.width + col3.minWidth(=40) )
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should respect minimum sizes', () => {
|
|
|
|
const layout: LayoutItem = validateLayout({
|
|
|
|
key: 'root',
|
|
|
|
width: 200,
|
|
|
|
height: 100,
|
|
|
|
direction: LayoutItemDirection.Row,
|
|
|
|
children: [
|
|
|
|
{
|
|
|
|
key: 'col1',
|
|
|
|
width: 50,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'col2',
|
|
|
|
width: 70,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'col3',
|
|
|
|
minWidth: 60,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
const { result } = renderHook(() => useLayoutItemSizes(layout));
|
|
|
|
const sizes = result.current;
|
|
|
|
const maxSize1 = calculateMaxSizeAvailableForItem(layout.children[0], layout, sizes);
|
|
|
|
const maxSize2 = calculateMaxSizeAvailableForItem(layout.children[1], layout, sizes);
|
|
|
|
|
|
|
|
// maxSize = totalSize - ( [size of items with set size, except for the current item] + [minimum size of items with no set size] )
|
|
|
|
expect(maxSize1.width).toBe(70); // 70 = layout.width - (col2.width + col3.minWidth)
|
|
|
|
expect(maxSize2.width).toBe(90); // 90 = layout.width - (col1.width + col3.minWidth)
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should not allow a minWidth of 0, should still leave space for the item', () => {
|
|
|
|
const layout: LayoutItem = validateLayout({
|
|
|
|
key: 'root',
|
|
|
|
width: 200,
|
|
|
|
height: 100,
|
|
|
|
direction: LayoutItemDirection.Row,
|
|
|
|
children: [
|
|
|
|
{
|
|
|
|
key: 'col1',
|
|
|
|
width: 50,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
key: 'col2',
|
|
|
|
minWidth: 0,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
const { result } = renderHook(() => useLayoutItemSizes(layout));
|
|
|
|
const sizes = result.current;
|
|
|
|
const maxSize1 = calculateMaxSizeAvailableForItem(layout.children[0], layout, sizes);
|
|
|
|
|
|
|
|
// maxSize = totalSize - ( [size of items with set size, except for the current item] + [minimum size of items with no set size] )
|
|
|
|
expect(maxSize1.width).toBe(160); // 160 = layout.width - col2.minWidth(=40)
|
|
|
|
});
|
|
|
|
|
2020-11-13 19:09:28 +02:00
|
|
|
});
|