mirror of
https://github.com/Kingsrook/qqq-frontend-material-dashboard.git
synced 2025-07-21 22:58:43 +00:00
320 lines
12 KiB
TypeScript
320 lines
12 KiB
TypeScript
/*
|
|
* QQQ - Low-code Application Framework for Engineers.
|
|
* Copyright (C) 2021-2024. Kingsrook, LLC
|
|
* 651 N Broad St Ste 205 # 6917 | Middletown DE 19709 | United States
|
|
* contact@kingsrook.com
|
|
* https://github.com/Kingsrook/
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as
|
|
* published by the Free Software Foundation, either version 3 of the
|
|
* License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Affero General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
|
|
import {QFieldMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QFieldMetaData";
|
|
import {QFieldType} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QFieldType";
|
|
import {QInstance} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QInstance";
|
|
import {QTableMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QTableMetaData";
|
|
import Autocomplete from "@mui/material/Autocomplete";
|
|
import Box from "@mui/material/Box";
|
|
import Button from "@mui/material/Button";
|
|
import Icon from "@mui/material/Icon";
|
|
import TextField from "@mui/material/TextField";
|
|
import type {Identifier, XYCoord} from "dnd-core";
|
|
import colors from "qqq/assets/theme/base/colors";
|
|
import FieldAutoComplete from "qqq/components/misc/FieldAutoComplete";
|
|
import {DragItemTypes, fieldAutoCompleteTextFieldSX, getSelectedFieldForAutoComplete, xIconButtonSX} from "qqq/components/widgets/misc/PivotTableSetupWidget";
|
|
import {functionsPerFieldType, PivotTableDefinition, pivotTableFunctionLabels, PivotTableValue} from "qqq/models/misc/PivotTableDefinitionModels";
|
|
import React, {FC, useReducer, useRef, useState} from "react";
|
|
import {useDrag, useDrop} from "react-dnd";
|
|
|
|
|
|
/*******************************************************************************
|
|
** component props
|
|
*******************************************************************************/
|
|
export interface PivotTableValueElementProps
|
|
{
|
|
id: string;
|
|
index: number;
|
|
dragCallback: (dragIndex: number, hoverIndex: number) => void;
|
|
metaData: QInstance;
|
|
tableMetaData: QTableMetaData;
|
|
pivotTableDefinition: PivotTableDefinition;
|
|
availableFieldNames: string[];
|
|
usedGroupByFieldNames: string[];
|
|
isEditable: boolean;
|
|
value: PivotTableValue;
|
|
callback: () => void;
|
|
attemptedSubmit?: boolean;
|
|
}
|
|
|
|
|
|
/*******************************************************************************
|
|
** item to support react-dnd
|
|
*******************************************************************************/
|
|
interface DragItem
|
|
{
|
|
index: number;
|
|
id: string;
|
|
type: string;
|
|
}
|
|
|
|
|
|
/*******************************************************************************
|
|
** Element to render 1 pivot-table value.
|
|
*******************************************************************************/
|
|
export const PivotTableValueElement: FC<PivotTableValueElementProps> = ({id, index, dragCallback, metaData, tableMetaData, pivotTableDefinition, availableFieldNames, usedGroupByFieldNames, value, isEditable, callback, attemptedSubmit}) =>
|
|
{
|
|
const [defaultFunctionValue, setDefaultFunctionValue] = useState(null);
|
|
const [, forceUpdate] = useReducer((x) => x + 1, 0);
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
// credit: https://react-dnd.github.io/react-dnd/examples/sortable/simple //
|
|
////////////////////////////////////////////////////////////////////////////
|
|
const ref = useRef<HTMLDivElement>(null);
|
|
const [{handlerId}, drop] = useDrop<DragItem, void, { handlerId: Identifier | null }>(
|
|
{
|
|
accept: DragItemTypes.VALUE,
|
|
collect(monitor)
|
|
{
|
|
return {
|
|
handlerId: monitor.getHandlerId(),
|
|
};
|
|
},
|
|
hover(item: DragItem, monitor)
|
|
{
|
|
if (!ref.current)
|
|
{
|
|
return;
|
|
}
|
|
const dragIndex = item.index;
|
|
const hoverIndex = index;
|
|
|
|
// Don't replace items with themselves
|
|
if (dragIndex === hoverIndex)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Determine rectangle on screen
|
|
const hoverBoundingRect = ref.current?.getBoundingClientRect();
|
|
|
|
// Get vertical middle
|
|
const hoverMiddleY = (hoverBoundingRect.bottom - hoverBoundingRect.top) / 2;
|
|
|
|
// Determine mouse position
|
|
const clientOffset = monitor.getClientOffset();
|
|
|
|
// Get pixels to the top
|
|
const hoverClientY = (clientOffset as XYCoord).y - hoverBoundingRect.top;
|
|
|
|
// Only perform the move when the mouse has crossed half of the items height
|
|
// When dragging downwards, only move when the cursor is below 50%
|
|
// When dragging upwards, only move when the cursor is above 50%
|
|
|
|
// Dragging downwards
|
|
if (dragIndex < hoverIndex && hoverClientY < hoverMiddleY)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Dragging upwards
|
|
if (dragIndex > hoverIndex && hoverClientY > hoverMiddleY)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Time to actually perform the action
|
|
dragCallback(dragIndex, hoverIndex);
|
|
|
|
// Note: we're mutating the monitor item here! Generally it's better to avoid mutations,
|
|
// but it's good here for the sake of performance to avoid expensive index searches.
|
|
item.index = hoverIndex;
|
|
},
|
|
});
|
|
|
|
const [{isDragging}, drag] = useDrag({
|
|
type: DragItemTypes.VALUE,
|
|
item: () =>
|
|
{
|
|
return {id, index};
|
|
},
|
|
collect: (monitor: any) => ({
|
|
isDragging: monitor.isDragging(),
|
|
}),
|
|
});
|
|
|
|
|
|
/*******************************************************************************
|
|
**
|
|
*******************************************************************************/
|
|
function getFunctionsForField(field: QFieldMetaData)
|
|
{
|
|
if(field)
|
|
{
|
|
let type = field.type;
|
|
if (field.possibleValueSourceName)
|
|
{
|
|
type = QFieldType.STRING;
|
|
}
|
|
|
|
if(functionsPerFieldType[type])
|
|
{
|
|
return (functionsPerFieldType[type]);
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////
|
|
// return broadest list if no field //
|
|
//////////////////////////////////////
|
|
return (functionsPerFieldType[QFieldType.INTEGER]);
|
|
}
|
|
|
|
|
|
/*******************************************************************************
|
|
** event handler for user selecting a field
|
|
*******************************************************************************/
|
|
function handleFieldChange(event: any, newValue: any, reason: string)
|
|
{
|
|
value.fieldName = newValue ? newValue.fieldName : null;
|
|
|
|
if(newValue)
|
|
{
|
|
/////////////////////////////////////////////////////////////////////////////////////////
|
|
// if newly selected field doesn't have the currently selected function, then clear it //
|
|
/////////////////////////////////////////////////////////////////////////////////////////
|
|
const newSelectedField = getSelectedFieldForAutoComplete(tableMetaData, newValue.fieldName);
|
|
if (newSelectedField)
|
|
{
|
|
if(getFunctionsForField(newSelectedField.field).indexOf(value.function) == -1)
|
|
{
|
|
setDefaultFunctionValue(null);
|
|
handleFunctionChange(null, null, null);
|
|
forceUpdate();
|
|
}
|
|
}
|
|
}
|
|
|
|
callback();
|
|
}
|
|
|
|
|
|
/*******************************************************************************
|
|
** event handler for user selecting a function
|
|
*******************************************************************************/
|
|
function handleFunctionChange(event: any, newValue: any, reason: string)
|
|
{
|
|
value.function = newValue ? newValue.id : null;
|
|
callback();
|
|
}
|
|
|
|
|
|
/*******************************************************************************
|
|
** event handler for clicking remove button
|
|
*******************************************************************************/
|
|
function removeValue(index: number)
|
|
{
|
|
pivotTableDefinition.values.splice(index, 1);
|
|
callback();
|
|
}
|
|
|
|
const selectedField = getSelectedFieldForAutoComplete(tableMetaData, value.fieldName);
|
|
|
|
/////////////////////////////////////////////////////////////////////
|
|
// if we're not on an edit screen, return a simpler read-only view //
|
|
/////////////////////////////////////////////////////////////////////
|
|
if (!isEditable)
|
|
{
|
|
let label = "--";
|
|
if (selectedField && value.function)
|
|
{
|
|
label = pivotTableFunctionLabels[value.function] + " of " + (selectedField.table.name == tableMetaData.name ? selectedField.field.label : selectedField.table.label + ": " + selectedField.field.label);
|
|
}
|
|
|
|
return (<Box><Box display="inline-block" mr="0.375rem" mb="0.5rem" border={`1px solid ${colors.grayLines.main}`} borderRadius="0.75rem" p="0.25rem 0.75rem">{label}</Box></Box>);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// figure out functions to display in drop down, plus selected/default value //
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
const functionOptions: any[] = [];
|
|
const availableFunctions = getFunctionsForField(selectedField?.field);
|
|
for (let pivotTableFunction of availableFunctions)
|
|
{
|
|
const label = pivotTableFunctionLabels[pivotTableFunction];
|
|
const option = {id: pivotTableFunction, label: label};
|
|
functionOptions.push(option);
|
|
|
|
if (option.id == value.function && JSON.stringify(option) != JSON.stringify(defaultFunctionValue))
|
|
{
|
|
setDefaultFunctionValue(option);
|
|
}
|
|
}
|
|
|
|
drag(drop(ref));
|
|
|
|
const showValueError = attemptedSubmit && !value.fieldName;
|
|
const showFunctionError = attemptedSubmit && !value.function;
|
|
|
|
return (<Box ref={ref} display="flex" p="0.5rem" pl="0" gap="0.5rem" alignItems="center" sx={{backgroundColor: "white", opacity: isDragging ? 0 : 1}} data-handler-id={handlerId}>
|
|
<Box>
|
|
<Icon sx={{cursor: "ns-resize"}}>drag_indicator</Icon>
|
|
</Box>
|
|
<Box width="100%">
|
|
<FieldAutoComplete
|
|
id={`values-field-${index}`}
|
|
label={null}
|
|
variant="outlined"
|
|
textFieldSX={fieldAutoCompleteTextFieldSX}
|
|
metaData={metaData}
|
|
tableMetaData={tableMetaData}
|
|
handleFieldChange={handleFieldChange}
|
|
availableFieldNames={availableFieldNames}
|
|
hiddenFieldNames={usedGroupByFieldNames}
|
|
defaultValue={selectedField}
|
|
hasError={showValueError}
|
|
noOptionsText="There are no fields available."
|
|
/>
|
|
</Box>
|
|
<Box width="370px">
|
|
<Autocomplete
|
|
id={`values-function-${index}`}
|
|
renderInput={(params) =>
|
|
{
|
|
const inputProps = params.InputProps;
|
|
const originalEndAdornment = inputProps.endAdornment;
|
|
inputProps.endAdornment = <Box>
|
|
{showFunctionError && <Icon color="error">error_outline</Icon>}
|
|
{originalEndAdornment}
|
|
</Box>;
|
|
|
|
return (<TextField {...params} label={null} variant="outlined" sx={fieldAutoCompleteTextFieldSX} autoComplete="off" type="search" InputProps={inputProps} />)
|
|
}}
|
|
// @ts-ignore
|
|
value={defaultFunctionValue}
|
|
inputValue={defaultFunctionValue?.label ?? ""}
|
|
options={functionOptions}
|
|
onChange={handleFunctionChange}
|
|
isOptionEqualToValue={(option, value) => option.id === value.id}
|
|
getOptionLabel={(option) => option.label}
|
|
autoSelect={true}
|
|
autoHighlight={true}
|
|
disableClearable
|
|
/>
|
|
</Box>
|
|
<Box>
|
|
<Button sx={xIconButtonSX} onClick={() => removeValue(index)}><Icon>clear</Icon></Button>
|
|
</Box>
|
|
</Box>);
|
|
|
|
};
|