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

refactor(DataSpreadsheet): implement typescript types #5230

Merged
Merged
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,9 @@ import React, {
useState,
useCallback,
useEffect,
ForwardedRef,
MutableRefObject,
LegacyRef,
} from 'react';
import { useBlockLayout, useTable, useColumnOrder } from 'react-table';

Expand Down Expand Up @@ -62,14 +65,102 @@ const defaults = {
theme: 'light',
};

interface Column {
Header?: string;
accessor?: string | (() => void);
Cell?: () => void; // optional cell formatter
}

type Size = 'xs' | 'sm' | 'md' | 'lg';

interface DataSpreadsheetProps {
/**
* Specifies the cell height
*/
cellSize?: Size;

/**
* Provide an optional class to be applied to the containing node.
*/
className?: string;

/**
* The data that will build the column headers
*/
columns?: readonly Column[];

/**
* The spreadsheet data that will be rendered in the body of the spreadsheet component
*/
data?: readonly object[];

/**
* Sets the number of empty rows to be created when there is no data provided
*/
defaultEmptyRowCount?: number;

/**
* The spreadsheet id
*/
id?: number | string;

/**
* The event handler that is called when the active cell changes
*/
onActiveCellChange?: () => void;

/**
* The setter fn for the data prop
*/
onDataUpdate?: ({ ...args }) => void;

/**
* The event handler that is called when the selection area values change
*/
onSelectionAreaChange?: () => void;

/**
* The aria label applied to the Select all button
*/
selectAllAriaLabel: string;

/**
* The aria label applied to the Data spreadsheet component
*/
spreadsheetAriaLabel: string;

/**
* The theme the DataSpreadsheet should use (only used to render active cell/selection area colors on dark theme)
*/
theme?: 'light' | 'dark';

/**
* The total number of columns to be initially visible, additional columns will be rendered and
* visible via horizontal scrollbar
*/
totalVisibleColumns?: number;

/* TODO: add types and DocGen for all props. */
}

type ActiveCellCoordinates = {
column?: string | number;
row?: string | number;
};

type PrevStateType = {
activeCellCoordinates?: ActiveCellCoordinates;
isEditing?: boolean;
};

/**
* DataSpreadsheet: used to organize and display large amounts of structured data, separated by columns and rows in a grid-like format.
*/
export let DataSpreadsheet = React.forwardRef(
(
{
// The component props, in alphabetical order (for consistency).
cellSize = defaults.cellSize,
cellSize = 'sm',
matthewgallo marked this conversation as resolved.
Show resolved Hide resolved
className,
columns = defaults.columns,
data = defaults.data,
Expand All @@ -85,35 +176,37 @@ export let DataSpreadsheet = React.forwardRef(

// Collect any other property values passed in.
...rest
},
ref
}: DataSpreadsheetProps,
ref: ForwardedRef<HTMLDivElement>
) => {
const multiKeyTrackingRef = useRef();
const multiKeyTrackingRef: LegacyRef<HTMLDivElement> = useRef(null);
const localRef = useRef();
const spreadsheetRef = ref || localRef;
const focusedElement = useActiveElement();
const [containerHasFocus, setContainerHasFocus] = useState(false);
const [activeCellCoordinates, setActiveCellCoordinates] = useState(null);
const [selectionAreas, setSelectionAreas] = useState([]);
const [selectionAreaData, setSelectionAreaData] = useState([]);
const [activeCellCoordinates, setActiveCellCoordinates] =
useState<ActiveCellCoordinates | null>(null);
const [selectionAreas, setSelectionAreas] = useState<object[]>([]);
const [selectionAreaData, setSelectionAreaData] = useState<object[]>([]);
const [clickAndHoldActive, setClickAndHoldActive] = useState(false);
const [currentMatcher, setCurrentMatcher] = useState('');
const [isEditing, setIsEditing] = useState(false);
const [cellEditorValue, setCellEditorValue] = useState('');
const [headerCellHoldActive, setHeaderCellHoldActive] = useState(false);
const [isActiveHeaderCellChanged, setIsActiveHeaderCellChanged] =
useState(null);
useState<boolean>(false);
const [activeCellInsideSelectionArea, setActiveCellInsideSelectionArea] =
useState(false);
const previousState = usePreviousValue({
activeCellCoordinates,
isEditing,
});
const previousState: PrevStateType =
usePreviousValue({
activeCellCoordinates,
isEditing,
}) || {};
const cellSizeValue = getCellSize(cellSize);
const cellEditorRef = useRef();
const [activeCellContent, setActiveCellContent] = useState();
const activeCellRef = useRef();
const cellEditorRulerRef = useRef();
const cellEditorRef = useRef<HTMLTextAreaElement>();
const [activeCellContent, setActiveCellContent] = useState(null);
const activeCellRef = useRef<HTMLDivElement | HTMLButtonElement>();
const cellEditorRulerRef = useRef<HTMLPreElement>();
const defaultColumn = useMemo(
() => ({
width: 150,
Expand Down Expand Up @@ -168,16 +261,20 @@ export let DataSpreadsheet = React.forwardRef(

// Removes the active cell element
const removeActiveCell = useCallback(() => {
const activeCellHighlight = spreadsheetRef.current.querySelector(
`.${blockClass}__active-cell--highlight`
);
activeCellHighlight.style.display = 'none';
const activeCellHighlight: HTMLDivElement | null = (
spreadsheetRef as MutableRefObject<HTMLDivElement>
)?.current?.querySelector(`.${blockClass}__active-cell--highlight`);
if (activeCellHighlight) {
activeCellHighlight.style.display = 'none';
}
}, [spreadsheetRef]);

const removeCellEditor = useCallback(() => {
setCellEditorValue('');
setIsEditing(false);
cellEditorRef.current.style.display = 'none';
if (cellEditorRef?.current) {
cellEditorRef.current.style.display = 'none';
}
}, []);

// Remove cell editor if the active cell coordinates change and save with new cell data, this will
Expand All @@ -191,8 +288,10 @@ export let DataSpreadsheet = React.forwardRef(
) {
const cellProps = rows[prevCoords?.row].cells[prevCoords?.column];
removeCellEditor();
updateData(prevCoords?.row, cellProps.column.id);
cellEditorRulerRef.current.textContent = '';
updateData(prevCoords?.row, cellProps.column.id, undefined);
if (cellEditorRulerRef?.current) {
cellEditorRulerRef.current.textContent = '';
}
}
if (
prevCoords?.row !== activeCellCoordinates?.row ||
Expand Down Expand Up @@ -267,7 +366,6 @@ export let DataSpreadsheet = React.forwardRef(
setActiveCellCoordinates,
setSelectionAreas,
removeActiveCell,
removeCellSelections,
setContainerHasFocus,
removeCellEditor,
});
Expand Down Expand Up @@ -331,7 +429,7 @@ export let DataSpreadsheet = React.forwardRef(
column: type === 'Home' ? 0 : columns.length - 1,
},
});
removeCellSelections({ spreadsheetRef });
removeCellSelections({ matcher: undefined, spreadsheetRef });
},
[
activeCellCoordinates,
Expand All @@ -350,7 +448,7 @@ export let DataSpreadsheet = React.forwardRef(

const handleArrowKeyPress = useCallback(
(arrowKey) => {
event.preventDefault();
event?.preventDefault();
handleInitialArrowPress();
const coordinatesClone = { ...activeCellCoordinates };

Expand Down Expand Up @@ -496,23 +594,33 @@ export let DataSpreadsheet = React.forwardRef(
activeCellCoordinates?.column
]
: null;
const activeCellValue = activeCellFullData
? activeCellFullData.row.cells[activeCellCoordinates?.column].value
: null;

let activeCellValue;
if (activeCellFullData && activeCellCoordinates?.column) {
activeCellValue = activeCellFullData
? activeCellFullData.row.cells?.[activeCellCoordinates?.column]?.value
: null;
}

setCellEditorValue(activeCellValue || '');
cellEditorRulerRef.current.textContent = activeCellValue;
cellEditorRef.current.style.width = activeCellRef?.current.style.width;
if (cellEditorRulerRef?.current) {
cellEditorRulerRef.current.textContent = activeCellValue;
}
if (cellEditorRef?.current && activeCellRef?.current) {
cellEditorRef.current.style.width =
activeCellRef?.current?.style?.width;
}
};

// Sets the initial placement of the cell editor cursor at the end of the text area
// this is not done for us by default in Safari
useEffect(() => {
if (isEditing && !previousState?.isEditing) {
cellEditorRef.current.setSelectionRange(
cellEditorRulerRef.current.textContent.length,
cellEditorRulerRef.current.textContent.length
cellEditorRef?.current?.setSelectionRange(
Number(cellEditorRulerRef?.current?.textContent?.length),
Number(cellEditorRulerRef?.current?.textContent?.length)
);
cellEditorRef.current.focus();
cellEditorRef?.current?.focus();
}
}, [isEditing, previousState?.isEditing]);

Expand All @@ -531,7 +639,10 @@ export let DataSpreadsheet = React.forwardRef(
) {
return;
}
handleRowColumnHeaderClick({ isKeyboard: false, index: indexValue });
handleRowColumnHeaderClick({
isKeyboard: false,
index: Number(indexValue),
});
}
return;
};
Expand All @@ -549,7 +660,7 @@ export let DataSpreadsheet = React.forwardRef(
) {
const tempMatcher = uuidv4();
setClickAndHoldActive(true);
removeCellSelections({ spreadsheetRef });
removeCellSelections({ matcher: null, spreadsheetRef });
setSelectionAreas([
{ point1: activeCellCoordinates, matcher: tempMatcher },
]);
Expand Down Expand Up @@ -579,19 +690,21 @@ export let DataSpreadsheet = React.forwardRef(
}
};

const handleRowColumnHeaderClick = ({ isKeyboard, index = null }) => {
const handleRowColumnHeaderClick = ({ isKeyboard, index = -1 }) => {
const handleHeaderCellProps = {
activeCellCoordinates,
rows,
columns,
currentMatcher,
setActiveCellCoordinates,
setCurrentMatcher,
setSelectionAreas,
spreadsheetRef,
index,
isKeyboard,
setSelectionAreaData,
index,
currentMatcher,
isHoldingCommandKey: null,
isHoldingShiftKey: null,
};
// Select an entire column
if (
Expand Down Expand Up @@ -720,7 +833,7 @@ export let DataSpreadsheet = React.forwardRef(
<div ref={multiKeyTrackingRef}>
{/* HEADER */}
<DataSpreadsheetHeader
ref={spreadsheetRef}
ref={spreadsheetRef as LegacyRef<HTMLDivElement>}
activeCellCoordinates={activeCellCoordinates}
cellSize={cellSize}
columns={columns}
Expand All @@ -746,15 +859,14 @@ export let DataSpreadsheet = React.forwardRef(
<DataSpreadsheetBody
activeCellRef={activeCellRef}
activeCellCoordinates={activeCellCoordinates}
ref={spreadsheetRef}
ref={spreadsheetRef as LegacyRef<HTMLDivElement>}
clickAndHoldActive={clickAndHoldActive}
setClickAndHoldActive={setClickAndHoldActive}
currentMatcher={currentMatcher}
setCurrentMatcher={setCurrentMatcher}
setContainerHasFocus={setContainerHasFocus}
selectionAreas={selectionAreas}
setSelectionAreas={setSelectionAreas}
cellSize={cellSize}
headerGroups={headerGroups}
defaultColumn={defaultColumn}
getTableBodyProps={getTableBodyProps}
Expand Down Expand Up @@ -784,7 +896,7 @@ export let DataSpreadsheet = React.forwardRef(
onKeyDown={handleActiveCellKeyDown}
onDoubleClick={handleActiveCellDoubleClick}
onMouseEnter={handleActiveCellMouseEnter}
ref={activeCellRef}
ref={activeCellRef as LegacyRef<HTMLButtonElement>}
className={cx(
`${blockClass}--interactive-cell-element`,
`${blockClass}__active-cell--highlight`,
Expand Down Expand Up @@ -815,13 +927,15 @@ export let DataSpreadsheet = React.forwardRef(
})}
onChange={(event) => {
setCellEditorValue(event.target.value);
cellEditorRulerRef.current.textContent = event.target.value;
if (cellEditorRulerRef?.current) {
cellEditorRulerRef.current.textContent = event.target.value;
}
}}
ref={cellEditorRef}
ref={cellEditorRef as LegacyRef<HTMLTextAreaElement>}
aria-labelledby={
activeCellCoordinates
? `${blockClass}__cell--${activeCellCoordinates?.row}--${activeCellCoordinates?.column}`
: null
: ''
}
className={cx(
`${blockClass}__cell-editor`,
Expand All @@ -834,7 +948,7 @@ export let DataSpreadsheet = React.forwardRef(
/>
<pre
aria-hidden
ref={cellEditorRulerRef}
ref={cellEditorRulerRef as LegacyRef<HTMLPreElement>}
className={`${blockClass}__cell-editor-ruler`}
/>
</div>
Expand Down Expand Up @@ -867,6 +981,7 @@ DataSpreadsheet.propTypes = {
/**
* The data that will build the column headers
*/
/**@ts-ignore */
columns: PropTypes.arrayOf(
PropTypes.shape({
Header: PropTypes.string,
Expand All @@ -878,6 +993,7 @@ DataSpreadsheet.propTypes = {
/**
* The spreadsheet data that will be rendered in the body of the spreadsheet component
*/
/**@ts-ignore */
data: PropTypes.arrayOf(PropTypes.shape),

/**
Expand Down
Loading
Loading