mirror of
https://github.com/Kingsrook/qqq-frontend-material-dashboard.git
synced 2025-07-19 13:50:43 +00:00
Merged feature/CE-881-create-basic-saved-reports into integration/sprint-40
This commit is contained in:
@ -41,6 +41,8 @@
|
|||||||
"react-ace": "10.1.0",
|
"react-ace": "10.1.0",
|
||||||
"react-chartjs-2": "3.0.4",
|
"react-chartjs-2": "3.0.4",
|
||||||
"react-cookie": "4.1.1",
|
"react-cookie": "4.1.1",
|
||||||
|
"react-dnd": "16.0.1",
|
||||||
|
"react-dnd-html5-backend": "16.0.1",
|
||||||
"react-dom": "18.0.0",
|
"react-dom": "18.0.0",
|
||||||
"react-github-btn": "1.2.1",
|
"react-github-btn": "1.2.1",
|
||||||
"react-google-drive-picker": "^1.2.0",
|
"react-google-drive-picker": "^1.2.0",
|
||||||
|
@ -22,6 +22,7 @@
|
|||||||
package com.kingsrook.qqq.frontend.materialdashboard.model.metadata;
|
package com.kingsrook.qqq.frontend.materialdashboard.model.metadata;
|
||||||
|
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
@ -30,6 +31,8 @@ import com.kingsrook.qqq.backend.core.model.metadata.QInstance;
|
|||||||
import com.kingsrook.qqq.backend.core.model.metadata.tables.QSupplementalTableMetaData;
|
import com.kingsrook.qqq.backend.core.model.metadata.tables.QSupplementalTableMetaData;
|
||||||
import com.kingsrook.qqq.backend.core.model.metadata.tables.QTableMetaData;
|
import com.kingsrook.qqq.backend.core.model.metadata.tables.QTableMetaData;
|
||||||
import com.kingsrook.qqq.backend.core.utils.CollectionUtils;
|
import com.kingsrook.qqq.backend.core.utils.CollectionUtils;
|
||||||
|
import com.kingsrook.qqq.backend.core.utils.StringUtils;
|
||||||
|
import com.kingsrook.qqq.frontend.materialdashboard.model.metadata.fieldrules.FieldRule;
|
||||||
|
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
@ -37,8 +40,11 @@ import com.kingsrook.qqq.backend.core.utils.CollectionUtils;
|
|||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
public class MaterialDashboardTableMetaData extends QSupplementalTableMetaData
|
public class MaterialDashboardTableMetaData extends QSupplementalTableMetaData
|
||||||
{
|
{
|
||||||
|
public static final String TYPE = "materialDashboard";
|
||||||
|
|
||||||
private List<List<String>> gotoFieldNames;
|
private List<List<String>> gotoFieldNames;
|
||||||
private List<String> defaultQuickFilterFieldNames;
|
private List<String> defaultQuickFilterFieldNames;
|
||||||
|
private List<FieldRule> fieldRules;
|
||||||
|
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
@ -58,10 +64,25 @@ public class MaterialDashboardTableMetaData extends QSupplementalTableMetaData
|
|||||||
@Override
|
@Override
|
||||||
public String getType()
|
public String getType()
|
||||||
{
|
{
|
||||||
return ("materialDashboard");
|
return (TYPE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
public static MaterialDashboardTableMetaData ofOrWithNew(QTableMetaData table)
|
||||||
|
{
|
||||||
|
MaterialDashboardTableMetaData supplementalMetaData = (MaterialDashboardTableMetaData) table.getSupplementalMetaData(TYPE);
|
||||||
|
if(supplementalMetaData == null)
|
||||||
|
{
|
||||||
|
supplementalMetaData = new MaterialDashboardTableMetaData();
|
||||||
|
table.withSupplementalMetaData(supplementalMetaData);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (supplementalMetaData);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
** Getter for gotoFieldNames
|
** Getter for gotoFieldNames
|
||||||
@ -110,6 +131,22 @@ public class MaterialDashboardTableMetaData extends QSupplementalTableMetaData
|
|||||||
validateListOfFieldNames(tableMetaData, gotoFieldNameSubList, qInstanceValidator, prefix + "gotoFieldNames: ");
|
validateListOfFieldNames(tableMetaData, gotoFieldNameSubList, qInstanceValidator, prefix + "gotoFieldNames: ");
|
||||||
}
|
}
|
||||||
validateListOfFieldNames(tableMetaData, defaultQuickFilterFieldNames, qInstanceValidator, prefix + "defaultQuickFilterFieldNames: ");
|
validateListOfFieldNames(tableMetaData, defaultQuickFilterFieldNames, qInstanceValidator, prefix + "defaultQuickFilterFieldNames: ");
|
||||||
|
|
||||||
|
for(FieldRule fieldRule : CollectionUtils.nonNullList(fieldRules))
|
||||||
|
{
|
||||||
|
qInstanceValidator.assertCondition(fieldRule.getTrigger() != null, prefix + "has a fieldRule without a trigger");
|
||||||
|
qInstanceValidator.assertCondition(fieldRule.getAction() != null, prefix + "has a fieldRule without an action");
|
||||||
|
|
||||||
|
if(qInstanceValidator.assertCondition(StringUtils.hasContent(fieldRule.getSourceField()), prefix + "has a fieldRule without a sourceField"))
|
||||||
|
{
|
||||||
|
qInstanceValidator.assertNoException(() -> tableMetaData.getField(fieldRule.getSourceField()), prefix + "has a fieldRule with an unrecognized sourceField: " + fieldRule.getSourceField());
|
||||||
|
}
|
||||||
|
|
||||||
|
if(qInstanceValidator.assertCondition(StringUtils.hasContent(fieldRule.getTargetField()), prefix + "has a fieldRule without a targetField"))
|
||||||
|
{
|
||||||
|
qInstanceValidator.assertNoException(() -> tableMetaData.getField(fieldRule.getTargetField()), prefix + "has a fieldRule with an unrecognized targetField: " + fieldRule.getTargetField());
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -124,7 +161,7 @@ public class MaterialDashboardTableMetaData extends QSupplementalTableMetaData
|
|||||||
{
|
{
|
||||||
if(qInstanceValidator.assertNoException(() -> tableMetaData.getField(fieldName), prefix + " unrecognized field name: " + fieldName))
|
if(qInstanceValidator.assertNoException(() -> tableMetaData.getField(fieldName), prefix + " unrecognized field name: " + fieldName))
|
||||||
{
|
{
|
||||||
qInstanceValidator.assertCondition(!usedNames.contains(fieldName), prefix + " has a duplicated field name: " + fieldName);
|
qInstanceValidator.assertCondition(!usedNames.contains(fieldName), prefix + "has a duplicated field name: " + fieldName);
|
||||||
usedNames.add(fieldName);
|
usedNames.add(fieldName);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -161,4 +198,51 @@ public class MaterialDashboardTableMetaData extends QSupplementalTableMetaData
|
|||||||
return (this);
|
return (this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Getter for fieldRules
|
||||||
|
*******************************************************************************/
|
||||||
|
public List<FieldRule> getFieldRules()
|
||||||
|
{
|
||||||
|
return (this.fieldRules);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Setter for fieldRules
|
||||||
|
*******************************************************************************/
|
||||||
|
public void setFieldRules(List<FieldRule> fieldRules)
|
||||||
|
{
|
||||||
|
this.fieldRules = fieldRules;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Fluent setter for fieldRules
|
||||||
|
*******************************************************************************/
|
||||||
|
public MaterialDashboardTableMetaData withFieldRules(List<FieldRule> fieldRules)
|
||||||
|
{
|
||||||
|
this.fieldRules = fieldRules;
|
||||||
|
return (this);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Fluent setter for fieldRules
|
||||||
|
*******************************************************************************/
|
||||||
|
public MaterialDashboardTableMetaData withFieldRule(FieldRule fieldRule)
|
||||||
|
{
|
||||||
|
if(this.fieldRules == null)
|
||||||
|
{
|
||||||
|
this.fieldRules = new ArrayList<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
this.fieldRules.add(fieldRule);
|
||||||
|
|
||||||
|
return (this);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,165 @@
|
|||||||
|
/*
|
||||||
|
* 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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.kingsrook.qqq.frontend.materialdashboard.model.metadata.fieldrules;
|
||||||
|
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** definition of rules for how UI fields should behave.
|
||||||
|
**
|
||||||
|
** e.g., one field being changed causing different things to be needed in another
|
||||||
|
** field.
|
||||||
|
*******************************************************************************/
|
||||||
|
public class FieldRule implements Serializable
|
||||||
|
{
|
||||||
|
private FieldRuleTrigger trigger;
|
||||||
|
private String sourceField;
|
||||||
|
private FieldRuleAction action;
|
||||||
|
private String targetField;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Getter for trigger
|
||||||
|
*******************************************************************************/
|
||||||
|
public FieldRuleTrigger getTrigger()
|
||||||
|
{
|
||||||
|
return (this.trigger);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Setter for trigger
|
||||||
|
*******************************************************************************/
|
||||||
|
public void setTrigger(FieldRuleTrigger trigger)
|
||||||
|
{
|
||||||
|
this.trigger = trigger;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Fluent setter for trigger
|
||||||
|
*******************************************************************************/
|
||||||
|
public FieldRule withTrigger(FieldRuleTrigger trigger)
|
||||||
|
{
|
||||||
|
this.trigger = trigger;
|
||||||
|
return (this);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Getter for sourceField
|
||||||
|
*******************************************************************************/
|
||||||
|
public String getSourceField()
|
||||||
|
{
|
||||||
|
return (this.sourceField);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Setter for sourceField
|
||||||
|
*******************************************************************************/
|
||||||
|
public void setSourceField(String sourceField)
|
||||||
|
{
|
||||||
|
this.sourceField = sourceField;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Fluent setter for sourceField
|
||||||
|
*******************************************************************************/
|
||||||
|
public FieldRule withSourceField(String sourceField)
|
||||||
|
{
|
||||||
|
this.sourceField = sourceField;
|
||||||
|
return (this);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Getter for action
|
||||||
|
*******************************************************************************/
|
||||||
|
public FieldRuleAction getAction()
|
||||||
|
{
|
||||||
|
return (this.action);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Setter for action
|
||||||
|
*******************************************************************************/
|
||||||
|
public void setAction(FieldRuleAction action)
|
||||||
|
{
|
||||||
|
this.action = action;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Fluent setter for action
|
||||||
|
*******************************************************************************/
|
||||||
|
public FieldRule withAction(FieldRuleAction action)
|
||||||
|
{
|
||||||
|
this.action = action;
|
||||||
|
return (this);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Getter for targetField
|
||||||
|
*******************************************************************************/
|
||||||
|
public String getTargetField()
|
||||||
|
{
|
||||||
|
return (this.targetField);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Setter for targetField
|
||||||
|
*******************************************************************************/
|
||||||
|
public void setTargetField(String targetField)
|
||||||
|
{
|
||||||
|
this.targetField = targetField;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Fluent setter for targetField
|
||||||
|
*******************************************************************************/
|
||||||
|
public FieldRule withTargetField(String targetField)
|
||||||
|
{
|
||||||
|
this.targetField = targetField;
|
||||||
|
return (this);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,31 @@
|
|||||||
|
/*
|
||||||
|
* 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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.kingsrook.qqq.frontend.materialdashboard.model.metadata.fieldrules;
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** possible actions associated with field rules
|
||||||
|
*******************************************************************************/
|
||||||
|
public enum FieldRuleAction
|
||||||
|
{
|
||||||
|
CLEAR_TARGET_FIELD
|
||||||
|
}
|
@ -0,0 +1,31 @@
|
|||||||
|
/*
|
||||||
|
* 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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.kingsrook.qqq.frontend.materialdashboard.model.metadata.fieldrules;
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** possible triggers associated with field rules
|
||||||
|
*******************************************************************************/
|
||||||
|
public enum FieldRuleTrigger
|
||||||
|
{
|
||||||
|
ON_CHANGE
|
||||||
|
}
|
@ -0,0 +1,59 @@
|
|||||||
|
/*
|
||||||
|
* 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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.kingsrook.qqq.frontend.materialdashboard.savedreports;
|
||||||
|
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import com.kingsrook.qqq.backend.core.model.metadata.tables.QTableMetaData;
|
||||||
|
import com.kingsrook.qqq.frontend.materialdashboard.model.metadata.MaterialDashboardTableMetaData;
|
||||||
|
import com.kingsrook.qqq.frontend.materialdashboard.model.metadata.fieldrules.FieldRule;
|
||||||
|
import com.kingsrook.qqq.frontend.materialdashboard.model.metadata.fieldrules.FieldRuleAction;
|
||||||
|
import com.kingsrook.qqq.frontend.materialdashboard.model.metadata.fieldrules.FieldRuleTrigger;
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Add frontend material dashboard enhacements to saved report table
|
||||||
|
*******************************************************************************/
|
||||||
|
public class SavedReportTableFrontendMaterialDashboardEnricher
|
||||||
|
{
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
public static void enrich(QTableMetaData tableMetaData)
|
||||||
|
{
|
||||||
|
MaterialDashboardTableMetaData materialDashboardTableMetaData = MaterialDashboardTableMetaData.ofOrWithNew(tableMetaData);
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////////
|
||||||
|
// make changes to the tableName field clear the value in these fields //
|
||||||
|
/////////////////////////////////////////////////////////////////////////
|
||||||
|
for(String targetField : List.of("queryFilterJson", "columnsJson", "pivotTableJson"))
|
||||||
|
{
|
||||||
|
materialDashboardTableMetaData.withFieldRule(new FieldRule()
|
||||||
|
.withSourceField("tableName")
|
||||||
|
.withTrigger(FieldRuleTrigger.ON_CHANGE)
|
||||||
|
.withAction(FieldRuleAction.CLEAR_TARGET_FIELD)
|
||||||
|
.withTargetField(targetField));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -43,7 +43,10 @@ import DynamicFormUtils from "qqq/components/forms/DynamicFormUtils";
|
|||||||
import MDTypography from "qqq/components/legacy/MDTypography";
|
import MDTypography from "qqq/components/legacy/MDTypography";
|
||||||
import HelpContent from "qqq/components/misc/HelpContent";
|
import HelpContent from "qqq/components/misc/HelpContent";
|
||||||
import QRecordSidebar from "qqq/components/misc/RecordSidebar";
|
import QRecordSidebar from "qqq/components/misc/RecordSidebar";
|
||||||
|
import PivotTableSetupWidget from "qqq/components/widgets/misc/PivotTableSetupWidget";
|
||||||
import RecordGridWidget, {ChildRecordListData} from "qqq/components/widgets/misc/RecordGridWidget";
|
import RecordGridWidget, {ChildRecordListData} from "qqq/components/widgets/misc/RecordGridWidget";
|
||||||
|
import ReportSetupWidget from "qqq/components/widgets/misc/ReportSetupWidget";
|
||||||
|
import {FieldRule, FieldRuleAction, FieldRuleTrigger} from "qqq/models/fields/FieldRules";
|
||||||
import HtmlUtils from "qqq/utils/HtmlUtils";
|
import HtmlUtils from "qqq/utils/HtmlUtils";
|
||||||
import Client from "qqq/utils/qqq/Client";
|
import Client from "qqq/utils/qqq/Client";
|
||||||
import TableUtils from "qqq/utils/qqq/TableUtils";
|
import TableUtils from "qqq/utils/qqq/TableUtils";
|
||||||
@ -77,6 +80,15 @@ EntityForm.defaultProps = {
|
|||||||
onSubmitCallback: null,
|
onSubmitCallback: null,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////////////////
|
||||||
|
// define a function that we can make referenes to, which we'll overwrite //
|
||||||
|
// with formik's setFieldValue function, once we're inside formik. //
|
||||||
|
////////////////////////////////////////////////////////////////////////////
|
||||||
|
let formikSetFieldValueFunction = (field: string, value: any, shouldValidate?: boolean): void =>
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
function EntityForm(props: Props): JSX.Element
|
function EntityForm(props: Props): JSX.Element
|
||||||
{
|
{
|
||||||
const qController = Client.getInstance();
|
const qController = Client.getInstance();
|
||||||
@ -97,6 +109,7 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
|
|
||||||
const [asyncLoadInited, setAsyncLoadInited] = useState(false);
|
const [asyncLoadInited, setAsyncLoadInited] = useState(false);
|
||||||
const [tableMetaData, setTableMetaData] = useState(null as QTableMetaData);
|
const [tableMetaData, setTableMetaData] = useState(null as QTableMetaData);
|
||||||
|
const [fieldRules, setFieldRules] = useState([] as FieldRule[]);
|
||||||
const [metaData, setMetaData] = useState(null as QInstance);
|
const [metaData, setMetaData] = useState(null as QInstance);
|
||||||
const [record, setRecord] = useState(null as QRecord);
|
const [record, setRecord] = useState(null as QRecord);
|
||||||
const [tableSections, setTableSections] = useState(null as QTableSection[]);
|
const [tableSections, setTableSections] = useState(null as QTableSection[]);
|
||||||
@ -108,6 +121,9 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
|
|
||||||
const [notAllowedError, setNotAllowedError] = useState(null as string);
|
const [notAllowedError, setNotAllowedError] = useState(null as string);
|
||||||
|
|
||||||
|
const [formValuesJSON, setFormValuesJSON] = useState("");
|
||||||
|
const [formValues, setFormValues] = useState({} as {[name: string]: any});
|
||||||
|
|
||||||
const {pageHeader, setPageHeader} = useContext(QContext);
|
const {pageHeader, setPageHeader} = useContext(QContext);
|
||||||
|
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
@ -269,6 +285,21 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Watch the record values - if they change, re-render widgets
|
||||||
|
*******************************************************************************/
|
||||||
|
useEffect(() =>
|
||||||
|
{
|
||||||
|
const newRenderedWidgetSections: {[name: string]: JSX.Element} = {};
|
||||||
|
for (let widgetName in renderedWidgetSections)
|
||||||
|
{
|
||||||
|
const widgetMetaData = metaData.widgets.get(widgetName);
|
||||||
|
newRenderedWidgetSections[widgetName] = getWidgetSection(widgetMetaData, childListWidgetData[widgetName]);
|
||||||
|
}
|
||||||
|
setRenderedWidgetSections(newRenderedWidgetSections);
|
||||||
|
}, [formValuesJSON]);
|
||||||
|
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
** render a section (full of fields) as a form
|
** render a section (full of fields) as a form
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
@ -319,25 +350,66 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** if we have a widget that wants to set form-field values, they can take this
|
||||||
|
** function in as a callback, and then call it with their values.
|
||||||
|
*******************************************************************************/
|
||||||
|
function setFormFieldValuesFromWidget(values: {[name: string]: any})
|
||||||
|
{
|
||||||
|
for (let key in values)
|
||||||
|
{
|
||||||
|
formikSetFieldValueFunction(key, values[key]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
** render a section as a widget
|
** render a section as a widget
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
function getWidgetSection(widgetMetaData: QWidgetMetaData, widgetData: any): JSX.Element
|
function getWidgetSection(widgetMetaData: QWidgetMetaData, widgetData: any): JSX.Element
|
||||||
{
|
{
|
||||||
widgetData.viewAllLink = null;
|
if(widgetMetaData.type == "childRecordList")
|
||||||
widgetMetaData.showExportButton = false;
|
{
|
||||||
|
widgetData.viewAllLink = null;
|
||||||
|
widgetMetaData.showExportButton = false;
|
||||||
|
|
||||||
return <RecordGridWidget
|
return <RecordGridWidget
|
||||||
key={new Date().getTime()} // added so that editing values actually re-renders...
|
key={new Date().getTime()} // added so that editing values actually re-renders...
|
||||||
widgetMetaData={widgetMetaData}
|
widgetMetaData={widgetMetaData}
|
||||||
data={widgetData}
|
data={widgetData}
|
||||||
disableRowClick
|
disableRowClick
|
||||||
allowRecordEdit
|
allowRecordEdit
|
||||||
allowRecordDelete
|
allowRecordDelete
|
||||||
addNewRecordCallback={() => openAddChildRecord(widgetMetaData.name, widgetData)}
|
addNewRecordCallback={() => openAddChildRecord(widgetMetaData.name, widgetData)}
|
||||||
editRecordCallback={(rowIndex) => openEditChildRecord(widgetMetaData.name, widgetData, rowIndex)}
|
editRecordCallback={(rowIndex) => openEditChildRecord(widgetMetaData.name, widgetData, rowIndex)}
|
||||||
deleteRecordCallback={(rowIndex) => deleteChildRecord(widgetMetaData.name, widgetData, rowIndex)}
|
deleteRecordCallback={(rowIndex) => deleteChildRecord(widgetMetaData.name, widgetData, rowIndex)}
|
||||||
/>;
|
/>;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(widgetMetaData.type == "reportSetup")
|
||||||
|
{
|
||||||
|
return <ReportSetupWidget
|
||||||
|
key={formValues["tableName"]} // todo, is this good? it was added so that editing values actually re-renders...
|
||||||
|
isEditable={true}
|
||||||
|
widgetMetaData={widgetMetaData}
|
||||||
|
recordValues={formValues}
|
||||||
|
onSaveCallback={setFormFieldValuesFromWidget}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
|
||||||
|
if(widgetMetaData.type == "pivotTableSetup")
|
||||||
|
{
|
||||||
|
return <PivotTableSetupWidget
|
||||||
|
key={formValues["tableName"]} // todo, is this good? it was added so that editing values actually re-renders...
|
||||||
|
isEditable={true}
|
||||||
|
widgetMetaData={widgetMetaData}
|
||||||
|
recordValues={formValues}
|
||||||
|
onSaveCallback={setFormFieldValuesFromWidget}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
|
||||||
|
return (<Box>Unsupported widget type: {widgetMetaData.type}</Box>)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -357,6 +429,32 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function setupFieldRules(tableMetaData: QTableMetaData)
|
||||||
|
{
|
||||||
|
const mdbMetaData = tableMetaData?.supplementalTableMetaData?.get("materialDashboard");
|
||||||
|
if(!mdbMetaData)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(mdbMetaData.fieldRules)
|
||||||
|
{
|
||||||
|
const newFieldRules: FieldRule[] = [];
|
||||||
|
for (let i = 0; i < mdbMetaData.fieldRules.length; i++)
|
||||||
|
{
|
||||||
|
newFieldRules.push(mdbMetaData.fieldRules[i]);
|
||||||
|
}
|
||||||
|
setFieldRules(newFieldRules);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//////////////////
|
||||||
|
// initial load //
|
||||||
|
//////////////////
|
||||||
if (!asyncLoadInited)
|
if (!asyncLoadInited)
|
||||||
{
|
{
|
||||||
setAsyncLoadInited(true);
|
setAsyncLoadInited(true);
|
||||||
@ -365,6 +463,8 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
const tableMetaData = await qController.loadTableMetaData(tableName);
|
const tableMetaData = await qController.loadTableMetaData(tableName);
|
||||||
setTableMetaData(tableMetaData);
|
setTableMetaData(tableMetaData);
|
||||||
|
|
||||||
|
setupFieldRules(tableMetaData);
|
||||||
|
|
||||||
const metaData = await qController.loadMetaData();
|
const metaData = await qController.loadMetaData();
|
||||||
setMetaData(metaData);
|
setMetaData(metaData);
|
||||||
|
|
||||||
@ -373,7 +473,21 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
/////////////////////////////////////////////////
|
/////////////////////////////////////////////////
|
||||||
const tableSections = TableUtils.getSectionsForRecordSidebar(tableMetaData, [...tableMetaData.fields.keys()], (section: QTableSection) =>
|
const tableSections = TableUtils.getSectionsForRecordSidebar(tableMetaData, [...tableMetaData.fields.keys()], (section: QTableSection) =>
|
||||||
{
|
{
|
||||||
return section.widgetName && metaData.widgets.get(section.widgetName)?.type == "childRecordList" && metaData.widgets.get(section.widgetName)?.defaultValues?.has("manageAssociationName");
|
const widget = metaData.widgets.get(section.widgetName);
|
||||||
|
if(widget)
|
||||||
|
{
|
||||||
|
if(widget.type == "childRecordList" && widget.defaultValues?.has("manageAssociationName"))
|
||||||
|
{
|
||||||
|
return (true);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(widget.type == "reportSetup" || widget.type == "pivotTableSetup")
|
||||||
|
{
|
||||||
|
return (true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (false);
|
||||||
});
|
});
|
||||||
setTableSections(tableSections);
|
setTableSections(tableSections);
|
||||||
|
|
||||||
@ -549,13 +663,7 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
}
|
}
|
||||||
|
|
||||||
const hasFields = section.fieldNames && section.fieldNames.length > 0;
|
const hasFields = section.fieldNames && section.fieldNames.length > 0;
|
||||||
const hasChildRecordListWidget = section.widgetName && metaData.widgets.get(section.widgetName)?.type == "childRecordList";
|
if(hasFields)
|
||||||
if (!hasFields && !hasChildRecordListWidget)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (hasFields)
|
|
||||||
{
|
{
|
||||||
for (let j = 0; j < section.fieldNames.length; j++)
|
for (let j = 0; j < section.fieldNames.length; j++)
|
||||||
{
|
{
|
||||||
@ -599,6 +707,7 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
newRenderedWidgetSections[section.widgetName] = getWidgetSection(widgetMetaData, widgetData);
|
newRenderedWidgetSections[section.widgetName] = getWidgetSection(widgetMetaData, widgetData);
|
||||||
newChildListWidgetData[section.widgetName] = widgetData;
|
newChildListWidgetData[section.widgetName] = widgetData;
|
||||||
}
|
}
|
||||||
|
|
||||||
//////////////////////////////////////
|
//////////////////////////////////////
|
||||||
// capture the tier1 section's name //
|
// capture the tier1 section's name //
|
||||||
//////////////////////////////////////
|
//////////////////////////////////////
|
||||||
@ -849,6 +958,27 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
})();
|
})();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** process a form-field having a changed value (e.g., apply field rules).
|
||||||
|
*******************************************************************************/
|
||||||
|
function handleChangedFieldValue(fieldName: string, oldValue: any, newValue: any, valueChangesToMake: {[fieldName: string]: any})
|
||||||
|
{
|
||||||
|
for (let fieldRule of fieldRules)
|
||||||
|
{
|
||||||
|
if(fieldRule.trigger == FieldRuleTrigger.ON_CHANGE && fieldRule.sourceField == fieldName)
|
||||||
|
{
|
||||||
|
switch (fieldRule.action)
|
||||||
|
{
|
||||||
|
case FieldRuleAction.CLEAR_TARGET_FIELD:
|
||||||
|
console.log(`Clearing value from [${fieldRule.targetField}] due to change in [${fieldName}]`);
|
||||||
|
valueChangesToMake[fieldRule.targetField] = null;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const formId = props.id != null ? `edit-${tableMetaData?.name}-form` : `create-${tableMetaData?.name}-form`;
|
const formId = props.id != null ? `edit-${tableMetaData?.name}-form` : `create-${tableMetaData?.name}-form`;
|
||||||
|
|
||||||
let body;
|
let body;
|
||||||
@ -887,7 +1017,7 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
body = (
|
body = (
|
||||||
<Box mb={3}>
|
<Box mb={3} className="entityForm">
|
||||||
{
|
{
|
||||||
(alertContent || warningContent) &&
|
(alertContent || warningContent) &&
|
||||||
<Grid container spacing={3}>
|
<Grid container spacing={3}>
|
||||||
@ -924,51 +1054,116 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
errors,
|
errors,
|
||||||
touched,
|
touched,
|
||||||
isSubmitting,
|
isSubmitting,
|
||||||
}) => (
|
setFieldValue,
|
||||||
<Form id={formId} autoComplete="off">
|
dirty
|
||||||
<ScrollToFirstError />
|
}) =>
|
||||||
|
{
|
||||||
|
/////////////////////////////////////////////////
|
||||||
|
// if we have values from formik, look at them //
|
||||||
|
/////////////////////////////////////////////////
|
||||||
|
if(values)
|
||||||
|
{
|
||||||
|
////////////////////////////////////////////////////////////////////////
|
||||||
|
// use stringified values as cheap/easy way to see if any are changed //
|
||||||
|
////////////////////////////////////////////////////////////////////////
|
||||||
|
const newFormValuesJSON = JSON.stringify(values);
|
||||||
|
if(formValuesJSON != newFormValuesJSON)
|
||||||
|
{
|
||||||
|
const valueChangesToMake: {[fieldName: string]: any} = {};
|
||||||
|
|
||||||
<Box pb={3} pt={0}>
|
////////////////////////////////////////////////////////////////////
|
||||||
<Card id={`${t1sectionName}`} sx={{overflow: "visible", pb: 2, scrollMarginTop: "100px"}} elevation={cardElevation}>
|
// if the form is dirty (e.g., we're not doing the initial load), //
|
||||||
<Box display="flex" p={3} pb={1}>
|
// then process rules for any changed fields //
|
||||||
<Box mr={1.5}>
|
////////////////////////////////////////////////////////////////////
|
||||||
<Avatar sx={{bgcolor: accentColor}}>
|
if(dirty)
|
||||||
<Icon>
|
{
|
||||||
{tableMetaData?.iconName}
|
for (let fieldName in values)
|
||||||
</Icon>
|
|
||||||
</Avatar>
|
|
||||||
</Box>
|
|
||||||
<Box display="flex" alignItems="center">
|
|
||||||
<MDTypography variant="h5">{formTitle}</MDTypography>
|
|
||||||
</Box>
|
|
||||||
</Box>
|
|
||||||
{t1section && getSectionHelp(t1section)}
|
|
||||||
{
|
{
|
||||||
t1sectionName && formFields ? (
|
if (formValues[fieldName] != values[fieldName])
|
||||||
<Box px={3}>
|
{
|
||||||
<Box pb={"0.25rem"} width="100%">
|
handleChangedFieldValue(fieldName, formValues[fieldName], values[fieldName], valueChangesToMake);
|
||||||
{getFormSection(t1section, values, touched, formFields.get(t1sectionName), errors, true)}
|
}
|
||||||
</Box>
|
formValues[fieldName] = values[fieldName];
|
||||||
</Box>
|
|
||||||
) : null
|
|
||||||
}
|
}
|
||||||
</Card>
|
}
|
||||||
</Box>
|
else
|
||||||
{formFields && nonT1Sections.length ? nonT1Sections.map((section: QTableSection) => (
|
{
|
||||||
<Box key={`edit-card-${section.name}`} pb={3}>
|
/////////////////////////////////////////////////////////////////////////////////////
|
||||||
{renderSection(section, values, touched, formFields, errors)}
|
// if the form is clean, make sure the formValues object has all form values in it //
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
for (let fieldName in values)
|
||||||
|
{
|
||||||
|
formValues[fieldName] = values[fieldName];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
// if there were any changes to be made from the rule evaluation, //
|
||||||
|
// make those changes in the formValues map, and in formik (setFieldValue) //
|
||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
for (let fieldName in valueChangesToMake)
|
||||||
|
{
|
||||||
|
formValues[fieldName] = valueChangesToMake[fieldName];
|
||||||
|
setFieldValue(fieldName, valueChangesToMake[fieldName], false);
|
||||||
|
}
|
||||||
|
|
||||||
|
setFormValues(formValues)
|
||||||
|
setFormValuesJSON(JSON.stringify(values));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////
|
||||||
|
// once we're in the formik form, use its setFieldValue function //
|
||||||
|
// over top of the default one we created globally //
|
||||||
|
///////////////////////////////////////////////////////////////////
|
||||||
|
formikSetFieldValueFunction = setFieldValue;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Form id={formId} autoComplete="off">
|
||||||
|
<ScrollToFirstError />
|
||||||
|
|
||||||
|
<Box pb={3} pt={0}>
|
||||||
|
<Card id={`${t1sectionName}`} sx={{overflow: "visible", pb: 2, scrollMarginTop: "100px"}} elevation={cardElevation}>
|
||||||
|
<Box display="flex" p={3} pb={1}>
|
||||||
|
<Box mr={1.5}>
|
||||||
|
<Avatar sx={{bgcolor: accentColor}}>
|
||||||
|
<Icon>
|
||||||
|
{tableMetaData?.iconName}
|
||||||
|
</Icon>
|
||||||
|
</Avatar>
|
||||||
|
</Box>
|
||||||
|
<Box display="flex" alignItems="center">
|
||||||
|
<MDTypography variant="h5">{formTitle}</MDTypography>
|
||||||
|
</Box>
|
||||||
|
</Box>
|
||||||
|
{t1section && getSectionHelp(t1section)}
|
||||||
|
{
|
||||||
|
t1sectionName && formFields ? (
|
||||||
|
<Box px={3}>
|
||||||
|
<Box pb={"0.25rem"} width="100%">
|
||||||
|
{getFormSection(t1section, values, touched, formFields.get(t1sectionName), errors, true)}
|
||||||
|
</Box>
|
||||||
|
</Box>
|
||||||
|
) : null
|
||||||
|
}
|
||||||
|
</Card>
|
||||||
</Box>
|
</Box>
|
||||||
)) : null}
|
{formFields && nonT1Sections.length ? nonT1Sections.map((section: QTableSection) => (
|
||||||
|
<Box key={`edit-card-${section.name}`} pb={3}>
|
||||||
|
{renderSection(section, values, touched, formFields, errors)}
|
||||||
|
</Box>
|
||||||
|
)) : null}
|
||||||
|
|
||||||
<Box component="div" p={3}>
|
<Box component="div" p={3}>
|
||||||
<Grid container justifyContent="flex-end" spacing={3}>
|
<Grid container justifyContent="flex-end" spacing={3}>
|
||||||
<QCancelButton onClickHandler={props.isModal ? props.closeModalHandler : handleCancelClicked} disabled={isSubmitting} />
|
<QCancelButton onClickHandler={props.isModal ? props.closeModalHandler : handleCancelClicked} disabled={isSubmitting} />
|
||||||
<QSaveButton disabled={isSubmitting} />
|
<QSaveButton disabled={isSubmitting} />
|
||||||
</Grid>
|
</Grid>
|
||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
</Form>
|
</Form>
|
||||||
)}
|
);
|
||||||
|
}}
|
||||||
</Formik>
|
</Formik>
|
||||||
|
|
||||||
{
|
{
|
||||||
|
@ -23,9 +23,11 @@
|
|||||||
import {QFieldMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QFieldMetaData";
|
import {QFieldMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QFieldMetaData";
|
||||||
import {QInstance} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QInstance";
|
import {QInstance} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QInstance";
|
||||||
import {QTableMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QTableMetaData";
|
import {QTableMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QTableMetaData";
|
||||||
|
import {Box} from "@mui/material";
|
||||||
import Autocomplete, {AutocompleteRenderOptionState} from "@mui/material/Autocomplete";
|
import Autocomplete, {AutocompleteRenderOptionState} from "@mui/material/Autocomplete";
|
||||||
|
import Icon from "@mui/material/Icon";
|
||||||
import TextField from "@mui/material/TextField";
|
import TextField from "@mui/material/TextField";
|
||||||
import React, {ReactNode} from "react";
|
import React, {ReactNode, useState} from "react";
|
||||||
|
|
||||||
interface FieldAutoCompleteProps
|
interface FieldAutoCompleteProps
|
||||||
{
|
{
|
||||||
@ -33,10 +35,16 @@ interface FieldAutoCompleteProps
|
|||||||
metaData: QInstance;
|
metaData: QInstance;
|
||||||
tableMetaData: QTableMetaData;
|
tableMetaData: QTableMetaData;
|
||||||
handleFieldChange: (event: any, newValue: any, reason: string) => void;
|
handleFieldChange: (event: any, newValue: any, reason: string) => void;
|
||||||
defaultValue?: {field: QFieldMetaData, table: QTableMetaData, fieldName: string};
|
defaultValue?: { field: QFieldMetaData, table: QTableMetaData, fieldName: string };
|
||||||
autoFocus?: boolean;
|
autoFocus?: boolean;
|
||||||
forceOpen?: boolean;
|
forceOpen?: boolean;
|
||||||
hiddenFieldNames?: string[];
|
hiddenFieldNames?: string[];
|
||||||
|
availableFieldNames?: string[];
|
||||||
|
variant?: "standard" | "filled" | "outlined";
|
||||||
|
label?: string;
|
||||||
|
textFieldSX?: any;
|
||||||
|
autocompleteSlotProps?: any;
|
||||||
|
hasError?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
FieldAutoComplete.defaultProps =
|
FieldAutoComplete.defaultProps =
|
||||||
@ -44,17 +52,28 @@ FieldAutoComplete.defaultProps =
|
|||||||
defaultValue: null,
|
defaultValue: null,
|
||||||
autoFocus: false,
|
autoFocus: false,
|
||||||
forceOpen: null,
|
forceOpen: null,
|
||||||
hiddenFieldNames: []
|
hiddenFieldNames: [],
|
||||||
|
availableFieldNames: [],
|
||||||
|
variant: "standard",
|
||||||
|
label: "Field",
|
||||||
|
textFieldSX: null,
|
||||||
|
autocompleteSlotProps: null,
|
||||||
|
hasError: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
function makeFieldOptionsForTable(tableMetaData: QTableMetaData, fieldOptions: any[], isJoinTable: boolean, hiddenFieldNames: string[])
|
function makeFieldOptionsForTable(tableMetaData: QTableMetaData, fieldOptions: any[], isJoinTable: boolean, hiddenFieldNames: string[], availableFieldNames: string[], selectedFieldName: string)
|
||||||
{
|
{
|
||||||
const sortedFields = [...tableMetaData.fields.values()].sort((a, b) => a.label.localeCompare(b.label));
|
const sortedFields = [...tableMetaData.fields.values()].sort((a, b) => a.label.localeCompare(b.label));
|
||||||
for (let i = 0; i < sortedFields.length; i++)
|
for (let i = 0; i < sortedFields.length; i++)
|
||||||
{
|
{
|
||||||
const fieldName = isJoinTable ? `${tableMetaData.name}.${sortedFields[i].name}` : sortedFields[i].name;
|
const fieldName = isJoinTable ? `${tableMetaData.name}.${sortedFields[i].name}` : sortedFields[i].name;
|
||||||
|
|
||||||
if(hiddenFieldNames && hiddenFieldNames.indexOf(fieldName) > -1)
|
if (hiddenFieldNames && hiddenFieldNames.indexOf(fieldName) > -1 && fieldName != selectedFieldName)
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (availableFieldNames?.length && availableFieldNames.indexOf(fieldName) == -1)
|
||||||
{
|
{
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -63,10 +82,16 @@ function makeFieldOptionsForTable(tableMetaData: QTableMetaData, fieldOptions: a
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function FieldAutoComplete({id, metaData, tableMetaData, handleFieldChange, defaultValue, autoFocus, forceOpen, hiddenFieldNames}: FieldAutoCompleteProps): JSX.Element
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Component for rendering a list of field names from a table as an auto-complete.
|
||||||
|
*******************************************************************************/
|
||||||
|
export default function FieldAutoComplete({id, metaData, tableMetaData, handleFieldChange, defaultValue, autoFocus, forceOpen, hiddenFieldNames, availableFieldNames, variant, label, textFieldSX, autocompleteSlotProps, hasError}: FieldAutoCompleteProps): JSX.Element
|
||||||
{
|
{
|
||||||
|
const [selectedFieldName, setSelectedFieldName] = useState(defaultValue ? defaultValue.fieldName : null);
|
||||||
|
|
||||||
const fieldOptions: any[] = [];
|
const fieldOptions: any[] = [];
|
||||||
makeFieldOptionsForTable(tableMetaData, fieldOptions, false, hiddenFieldNames);
|
makeFieldOptionsForTable(tableMetaData, fieldOptions, false, hiddenFieldNames, availableFieldNames, selectedFieldName);
|
||||||
let fieldsGroupBy = null;
|
let fieldsGroupBy = null;
|
||||||
|
|
||||||
if (tableMetaData.exposedJoins && tableMetaData.exposedJoins.length > 0)
|
if (tableMetaData.exposedJoins && tableMetaData.exposedJoins.length > 0)
|
||||||
@ -77,7 +102,7 @@ export default function FieldAutoComplete({id, metaData, tableMetaData, handleFi
|
|||||||
if (metaData.tables.has(exposedJoin.joinTable.name))
|
if (metaData.tables.has(exposedJoin.joinTable.name))
|
||||||
{
|
{
|
||||||
fieldsGroupBy = (option: any) => `${option.table.label} fields`;
|
fieldsGroupBy = (option: any) => `${option.table.label} fields`;
|
||||||
makeFieldOptionsForTable(exposedJoin.joinTable, fieldOptions, true, hiddenFieldNames);
|
makeFieldOptionsForTable(exposedJoin.joinTable, fieldOptions, true, hiddenFieldNames, availableFieldNames, selectedFieldName);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -130,27 +155,47 @@ export default function FieldAutoComplete({id, metaData, tableMetaData, handleFi
|
|||||||
// seems like, if we always add the open attribute, then if its false or null, then the autocomplete //
|
// seems like, if we always add the open attribute, then if its false or null, then the autocomplete //
|
||||||
// doesn't open at all... so, only add the attribute at all, if forceOpen is true //
|
// doesn't open at all... so, only add the attribute at all, if forceOpen is true //
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
const alsoOpen: {[key: string]: any} = {}
|
const alsoOpen: { [key: string]: any } = {};
|
||||||
if(forceOpen)
|
if (forceOpen)
|
||||||
{
|
{
|
||||||
alsoOpen["open"] = forceOpen;
|
alsoOpen["open"] = forceOpen;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function onChange(event: any, newValue: any, reason: string)
|
||||||
|
{
|
||||||
|
setSelectedFieldName(newValue ? newValue.fieldName : null);
|
||||||
|
handleFieldChange(event, newValue, reason);
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Autocomplete
|
<Autocomplete
|
||||||
id={id}
|
id={id}
|
||||||
renderInput={(params) => (<TextField {...params} autoFocus={autoFocus} label={"Field"} variant="standard" autoComplete="off" type="search" InputProps={{...params.InputProps}} />)}
|
renderInput={(params) =>
|
||||||
|
{
|
||||||
|
const inputProps = params.InputProps;
|
||||||
|
const originalEndAdornment = inputProps.endAdornment;
|
||||||
|
inputProps.endAdornment = <Box>
|
||||||
|
{hasError && <Icon color="error">error_outline</Icon>}
|
||||||
|
{originalEndAdornment}
|
||||||
|
</Box>;
|
||||||
|
|
||||||
|
return (<TextField {...params} autoFocus={autoFocus} label={label} variant={variant} sx={textFieldSX} autoComplete="off" type="search" InputProps={inputProps} />)
|
||||||
|
}}
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
defaultValue={defaultValue}
|
defaultValue={defaultValue}
|
||||||
options={fieldOptions}
|
options={fieldOptions}
|
||||||
onChange={handleFieldChange}
|
onChange={onChange}
|
||||||
isOptionEqualToValue={(option, value) => isFieldOptionEqual(option, value)}
|
isOptionEqualToValue={(option, value) => isFieldOptionEqual(option, value)}
|
||||||
groupBy={fieldsGroupBy}
|
groupBy={fieldsGroupBy}
|
||||||
getOptionLabel={(option) => getFieldOptionLabel(option)}
|
getOptionLabel={(option) => getFieldOptionLabel(option)}
|
||||||
renderOption={(props, option, state) => renderFieldOption(props, option, state)}
|
renderOption={(props, option, state) => renderFieldOption(props, option, state)}
|
||||||
autoSelect={true}
|
autoSelect={true}
|
||||||
autoHighlight={true}
|
autoHighlight={true}
|
||||||
slotProps={{popper: {className: "filterCriteriaRowColumnPopper", style: {padding: 0, width: "250px"}}}}
|
slotProps={autocompleteSlotProps ?? {}}
|
||||||
{...alsoOpen}
|
{...alsoOpen}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
@ -44,6 +44,7 @@ import QContext from "QContext";
|
|||||||
import colors from "qqq/assets/theme/base/colors";
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
import {QCancelButton, QDeleteButton, QSaveButton} from "qqq/components/buttons/DefaultButtons";
|
import {QCancelButton, QDeleteButton, QSaveButton} from "qqq/components/buttons/DefaultButtons";
|
||||||
import RecordQueryView from "qqq/models/query/RecordQueryView";
|
import RecordQueryView from "qqq/models/query/RecordQueryView";
|
||||||
|
import {QueryScreenUsage} from "qqq/pages/records/query/RecordQuery";
|
||||||
import FilterUtils from "qqq/utils/qqq/FilterUtils";
|
import FilterUtils from "qqq/utils/qqq/FilterUtils";
|
||||||
import {SavedViewUtils} from "qqq/utils/qqq/SavedViewUtils";
|
import {SavedViewUtils} from "qqq/utils/qqq/SavedViewUtils";
|
||||||
import React, {useContext, useEffect, useRef, useState} from "react";
|
import React, {useContext, useEffect, useRef, useState} from "react";
|
||||||
@ -60,9 +61,10 @@ interface Props
|
|||||||
viewAsJson?: string;
|
viewAsJson?: string;
|
||||||
viewOnChangeCallback?: (selectedSavedViewId: number) => void;
|
viewOnChangeCallback?: (selectedSavedViewId: number) => void;
|
||||||
loadingSavedView: boolean
|
loadingSavedView: boolean
|
||||||
|
queryScreenUsage: QueryScreenUsage;
|
||||||
}
|
}
|
||||||
|
|
||||||
function SavedViews({qController, metaData, tableMetaData, currentSavedView, tableDefaultView, view, viewAsJson, viewOnChangeCallback, loadingSavedView}: Props): JSX.Element
|
function SavedViews({qController, metaData, tableMetaData, currentSavedView, tableDefaultView, view, viewAsJson, viewOnChangeCallback, loadingSavedView, queryScreenUsage}: Props): JSX.Element
|
||||||
{
|
{
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
@ -91,6 +93,14 @@ function SavedViews({qController, metaData, tableMetaData, currentSavedView, tab
|
|||||||
|
|
||||||
const {accentColor, accentColorLight} = useContext(QContext);
|
const {accentColor, accentColorLight} = useContext(QContext);
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// this component is used by <RecordQuery> - but that component has different usages - //
|
||||||
|
// e.g., the full-fledged query screen, but also, within other screens (e.g., a modal //
|
||||||
|
// under the ReportSetupWidget). So, there are some behaviors we only want when we're //
|
||||||
|
// on the full-fledged query screen, such as changing the URL with saved view ids. //
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
const isQueryScreen = queryScreenUsage == "queryScreen";
|
||||||
|
|
||||||
const openSavedViewsMenu = (event: any) => setSavedViewsMenu(event.currentTarget);
|
const openSavedViewsMenu = (event: any) => setSavedViewsMenu(event.currentTarget);
|
||||||
const closeSavedViewsMenu = () => setSavedViewsMenu(null);
|
const closeSavedViewsMenu = () => setSavedViewsMenu(null);
|
||||||
|
|
||||||
@ -142,7 +152,10 @@ function SavedViews({qController, metaData, tableMetaData, currentSavedView, tab
|
|||||||
setSaveFilterPopupOpen(false);
|
setSaveFilterPopupOpen(false);
|
||||||
closeSavedViewsMenu();
|
closeSavedViewsMenu();
|
||||||
viewOnChangeCallback(record.values.get("id"));
|
viewOnChangeCallback(record.values.get("id"));
|
||||||
navigate(`${metaData.getTablePathByName(tableMetaData.name)}/savedView/${record.values.get("id")}`);
|
if(isQueryScreen)
|
||||||
|
{
|
||||||
|
navigate(`${metaData.getTablePathByName(tableMetaData.name)}/savedView/${record.values.get("id")}`);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -175,7 +188,10 @@ function SavedViews({qController, metaData, tableMetaData, currentSavedView, tab
|
|||||||
case CLEAR_OPTION:
|
case CLEAR_OPTION:
|
||||||
setSaveFilterPopupOpen(false)
|
setSaveFilterPopupOpen(false)
|
||||||
viewOnChangeCallback(null);
|
viewOnChangeCallback(null);
|
||||||
navigate(metaData.getTablePathByName(tableMetaData.name));
|
if(isQueryScreen)
|
||||||
|
{
|
||||||
|
navigate(metaData.getTablePathByName(tableMetaData.name));
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case RENAME_OPTION:
|
case RENAME_OPTION:
|
||||||
if(currentSavedView != null)
|
if(currentSavedView != null)
|
||||||
@ -415,11 +431,14 @@ function SavedViews({qController, metaData, tableMetaData, currentSavedView, tab
|
|||||||
open={Boolean(savedViewsMenu)}
|
open={Boolean(savedViewsMenu)}
|
||||||
onClose={closeSavedViewsMenu}
|
onClose={closeSavedViewsMenu}
|
||||||
keepMounted
|
keepMounted
|
||||||
PaperProps={{style: {maxHeight: "calc(100vh - 200px)", minHeight: "200px"}}}
|
PaperProps={{style: {maxHeight: "calc(100vh - 200px)", minWidth: "300px"}}}
|
||||||
>
|
>
|
||||||
<MenuItem sx={{width: "300px"}} disabled style={{"opacity": "initial"}}><b>View Actions</b></MenuItem>
|
|
||||||
{
|
{
|
||||||
hasStorePermission &&
|
isQueryScreen &&
|
||||||
|
<MenuItem sx={{width: "300px"}} disabled style={{"opacity": "initial"}}><b>View Actions</b></MenuItem>
|
||||||
|
}
|
||||||
|
{
|
||||||
|
isQueryScreen && hasStorePermission &&
|
||||||
<Tooltip {...menuTooltipAttribs} title={<>Save your current filters, columns and settings, for quick re-use at a later time.<br /><br />You will be prompted to enter a name if you choose this option.</>}>
|
<Tooltip {...menuTooltipAttribs} title={<>Save your current filters, columns and settings, for quick re-use at a later time.<br /><br />You will be prompted to enter a name if you choose this option.</>}>
|
||||||
<MenuItem onClick={() => handleDropdownOptionClick(SAVE_OPTION)}>
|
<MenuItem onClick={() => handleDropdownOptionClick(SAVE_OPTION)}>
|
||||||
<ListItemIcon><Icon>save</Icon></ListItemIcon>
|
<ListItemIcon><Icon>save</Icon></ListItemIcon>
|
||||||
@ -428,7 +447,7 @@ function SavedViews({qController, metaData, tableMetaData, currentSavedView, tab
|
|||||||
</Tooltip>
|
</Tooltip>
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
hasStorePermission && currentSavedView != null &&
|
isQueryScreen && hasStorePermission && currentSavedView != null &&
|
||||||
<Tooltip {...menuTooltipAttribs} title="Change the name for this saved view.">
|
<Tooltip {...menuTooltipAttribs} title="Change the name for this saved view.">
|
||||||
<MenuItem disabled={currentSavedView === null} onClick={() => handleDropdownOptionClick(RENAME_OPTION)}>
|
<MenuItem disabled={currentSavedView === null} onClick={() => handleDropdownOptionClick(RENAME_OPTION)}>
|
||||||
<ListItemIcon><Icon>edit</Icon></ListItemIcon>
|
<ListItemIcon><Icon>edit</Icon></ListItemIcon>
|
||||||
@ -437,7 +456,7 @@ function SavedViews({qController, metaData, tableMetaData, currentSavedView, tab
|
|||||||
</Tooltip>
|
</Tooltip>
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
hasStorePermission && currentSavedView != null &&
|
isQueryScreen && hasStorePermission && currentSavedView != null &&
|
||||||
<Tooltip {...menuTooltipAttribs} title="Save a new copy this view, with a different name, separate from the original.">
|
<Tooltip {...menuTooltipAttribs} title="Save a new copy this view, with a different name, separate from the original.">
|
||||||
<MenuItem disabled={currentSavedView === null} onClick={() => handleDropdownOptionClick(DUPLICATE_OPTION)}>
|
<MenuItem disabled={currentSavedView === null} onClick={() => handleDropdownOptionClick(DUPLICATE_OPTION)}>
|
||||||
<ListItemIcon><Icon>content_copy</Icon></ListItemIcon>
|
<ListItemIcon><Icon>content_copy</Icon></ListItemIcon>
|
||||||
@ -446,7 +465,7 @@ function SavedViews({qController, metaData, tableMetaData, currentSavedView, tab
|
|||||||
</Tooltip>
|
</Tooltip>
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
hasDeletePermission && currentSavedView != null &&
|
isQueryScreen && hasDeletePermission && currentSavedView != null &&
|
||||||
<Tooltip {...menuTooltipAttribs} title="Delete this saved view.">
|
<Tooltip {...menuTooltipAttribs} title="Delete this saved view.">
|
||||||
<MenuItem disabled={currentSavedView === null} onClick={() => handleDropdownOptionClick(DELETE_OPTION)}>
|
<MenuItem disabled={currentSavedView === null} onClick={() => handleDropdownOptionClick(DELETE_OPTION)}>
|
||||||
<ListItemIcon><Icon>delete</Icon></ListItemIcon>
|
<ListItemIcon><Icon>delete</Icon></ListItemIcon>
|
||||||
@ -455,6 +474,7 @@ function SavedViews({qController, metaData, tableMetaData, currentSavedView, tab
|
|||||||
</Tooltip>
|
</Tooltip>
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
|
isQueryScreen &&
|
||||||
<Tooltip {...menuTooltipAttribs} title="Create a new view of this table, resetting the filters and columns to their defaults.">
|
<Tooltip {...menuTooltipAttribs} title="Create a new view of this table, resetting the filters and columns to their defaults.">
|
||||||
<MenuItem onClick={() => handleDropdownOptionClick(CLEAR_OPTION)}>
|
<MenuItem onClick={() => handleDropdownOptionClick(CLEAR_OPTION)}>
|
||||||
<ListItemIcon><Icon>monitor</Icon></ListItemIcon>
|
<ListItemIcon><Icon>monitor</Icon></ListItemIcon>
|
||||||
@ -463,7 +483,7 @@ function SavedViews({qController, metaData, tableMetaData, currentSavedView, tab
|
|||||||
</Tooltip>
|
</Tooltip>
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
hasSavedReportsPermission &&
|
isQueryScreen && hasSavedReportsPermission &&
|
||||||
<Tooltip {...menuTooltipAttribs} title="Create a new Saved Report using your current view of this table as a starting point.">
|
<Tooltip {...menuTooltipAttribs} title="Create a new Saved Report using your current view of this table as a starting point.">
|
||||||
<MenuItem onClick={() => handleDropdownOptionClick(NEW_REPORT_OPTION)}>
|
<MenuItem onClick={() => handleDropdownOptionClick(NEW_REPORT_OPTION)}>
|
||||||
<ListItemIcon><Icon>article</Icon></ListItemIcon>
|
<ListItemIcon><Icon>article</Icon></ListItemIcon>
|
||||||
@ -471,7 +491,9 @@ function SavedViews({qController, metaData, tableMetaData, currentSavedView, tab
|
|||||||
</MenuItem>
|
</MenuItem>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
}
|
}
|
||||||
<Divider/>
|
{
|
||||||
|
isQueryScreen && <Divider/>
|
||||||
|
}
|
||||||
<MenuItem disabled style={{"opacity": "initial"}}><b>Your Saved Views</b></MenuItem>
|
<MenuItem disabled style={{"opacity": "initial"}}><b>Your Saved Views</b></MenuItem>
|
||||||
{
|
{
|
||||||
savedViews && savedViews.length > 0 ? (
|
savedViews && savedViews.length > 0 ? (
|
||||||
@ -481,7 +503,7 @@ function SavedViews({qController, metaData, tableMetaData, currentSavedView, tab
|
|||||||
</MenuItem>
|
</MenuItem>
|
||||||
)
|
)
|
||||||
): (
|
): (
|
||||||
<MenuItem>
|
<MenuItem disabled sx={{opacity: "1 !important"}}>
|
||||||
<i>You do not have any saved views for this table.</i>
|
<i>You do not have any saved views for this table.</i>
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
)
|
)
|
||||||
@ -580,25 +602,29 @@ function SavedViews({qController, metaData, tableMetaData, currentSavedView, tab
|
|||||||
<Box pl={2} pr={2} sx={{display: "flex", alignItems: "center"}}>
|
<Box pl={2} pr={2} sx={{display: "flex", alignItems: "center"}}>
|
||||||
{
|
{
|
||||||
!currentSavedView && viewIsModified && <>
|
!currentSavedView && viewIsModified && <>
|
||||||
<Tooltip {...tooltipMaxWidth("24rem")} sx={{cursor: "pointer"}} title={<>
|
{
|
||||||
<b>Unsaved Changes</b>
|
isQueryScreen && <>
|
||||||
<ul style={{padding: "0.5rem 1rem"}}>
|
<Tooltip {...tooltipMaxWidth("24rem")} sx={{cursor: "pointer"}} title={<>
|
||||||
{
|
<b>Unsaved Changes</b>
|
||||||
viewDiffs.map((s: string, i: number) => <li key={i}>{s}</li>)
|
<ul style={{padding: "0.5rem 1rem"}}>
|
||||||
}
|
{
|
||||||
</ul>
|
viewDiffs.map((s: string, i: number) => <li key={i}>{s}</li>)
|
||||||
</>}>
|
}
|
||||||
<Button disableRipple={true} sx={linkButtonStyle} onClick={() => handleDropdownOptionClick(SAVE_OPTION)}>Save View As…</Button>
|
</ul>
|
||||||
</Tooltip>
|
</>}>
|
||||||
|
<Button disableRipple={true} sx={linkButtonStyle} onClick={() => handleDropdownOptionClick(SAVE_OPTION)}>Save View As…</Button>
|
||||||
|
</Tooltip>
|
||||||
|
|
||||||
{/* vertical rule */}
|
{/* vertical rule */}
|
||||||
<Box display="inline-block" borderLeft={`1px solid ${colors.grayLines.main}`} height="1rem" width="1px" position="relative" />
|
<Box display="inline-block" borderLeft={`1px solid ${colors.grayLines.main}`} height="1rem" width="1px" position="relative" />
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
|
||||||
<Button disableRipple={true} sx={{color: colors.gray.main, ... linkButtonStyle}} onClick={() => handleDropdownOptionClick(CLEAR_OPTION)}>Reset All Changes</Button>
|
<Button disableRipple={true} sx={{color: colors.gray.main, ... linkButtonStyle}} onClick={() => handleDropdownOptionClick(CLEAR_OPTION)}>Reset All Changes</Button>
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
currentSavedView && viewIsModified && <>
|
isQueryScreen && currentSavedView && viewIsModified && <>
|
||||||
<Tooltip {...tooltipMaxWidth("24rem")} sx={{cursor: "pointer"}} title={<>
|
<Tooltip {...tooltipMaxWidth("24rem")} sx={{cursor: "pointer"}} title={<>
|
||||||
<b>Unsaved Changes</b>
|
<b>Unsaved Changes</b>
|
||||||
<ul style={{padding: "0.5rem 1rem"}}>
|
<ul style={{padding: "0.5rem 1rem"}}>
|
||||||
@ -617,6 +643,34 @@ function SavedViews({qController, metaData, tableMetaData, currentSavedView, tab
|
|||||||
<Button disableRipple={true} sx={{color: colors.gray.main, ... linkButtonStyle}} onClick={() => handleSavedViewRecordOnClick(currentSavedView)}>Reset All Changes</Button>
|
<Button disableRipple={true} sx={{color: colors.gray.main, ... linkButtonStyle}} onClick={() => handleSavedViewRecordOnClick(currentSavedView)}>Reset All Changes</Button>
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
|
{
|
||||||
|
!isQueryScreen && currentSavedView &&
|
||||||
|
<Box>
|
||||||
|
<Box display="inline-block" fontSize="0.875rem" fontWeight="500" sx={{position: "relative", top: "-1px"}}>
|
||||||
|
{currentSavedView.values.get("label")}
|
||||||
|
</Box>
|
||||||
|
|
||||||
|
{
|
||||||
|
viewIsModified &&
|
||||||
|
<>
|
||||||
|
<Tooltip {...tooltipMaxWidth("24rem")} sx={{cursor: "pointer"}} title={<>
|
||||||
|
<b>Changes</b>
|
||||||
|
<ul style={{padding: "0.5rem 1rem"}}>
|
||||||
|
{
|
||||||
|
viewDiffs.map((s: string, i: number) => <li key={i}>{s}</li>)
|
||||||
|
}
|
||||||
|
</ul></>}>
|
||||||
|
<Box display="inline" ml="0.25rem" mr="0.25rem" sx={{...linkButtonStyle, p: 0, cursor: "default", position: "relative", top: "-1px"}}>with {viewDiffs.length} Change{viewDiffs.length == 1 ? "" : "s"}</Box>
|
||||||
|
</Tooltip>
|
||||||
|
<Button disableRipple={true} sx={{color: colors.gray.main, ... linkButtonStyle}} onClick={() => handleSavedViewRecordOnClick(currentSavedView)}>Reset Changes</Button>
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
|
||||||
|
{/* vertical rule */}
|
||||||
|
<Box display="inline-block" ml="0.25rem" borderLeft={`1px solid ${colors.grayLines.main}`} height="1rem" width="1px" position="relative" />
|
||||||
|
<Button disableRipple={true} sx={{color: colors.gray.main, ... linkButtonStyle}} onClick={() => handleDropdownOptionClick(CLEAR_OPTION)}>Reset to New View</Button>
|
||||||
|
</Box>
|
||||||
|
}
|
||||||
</Box>
|
</Box>
|
||||||
</Box>
|
</Box>
|
||||||
{
|
{
|
||||||
|
153
src/qqq/components/query/AdvancedQueryPreview.tsx
Normal file
153
src/qqq/components/query/AdvancedQueryPreview.tsx
Normal file
@ -0,0 +1,153 @@
|
|||||||
|
/*
|
||||||
|
* 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 {QTableMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QTableMetaData";
|
||||||
|
import {QQueryFilter} from "@kingsrook/qqq-frontend-core/lib/model/query/QQueryFilter";
|
||||||
|
import Box from "@mui/material/Box";
|
||||||
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
|
import {validateCriteria} from "qqq/components/query/FilterCriteriaRow";
|
||||||
|
import XIcon from "qqq/components/query/XIcon";
|
||||||
|
import FilterUtils from "qqq/utils/qqq/FilterUtils";
|
||||||
|
import React, {useState} from "react";
|
||||||
|
|
||||||
|
interface AdvancedQueryPreviewProps
|
||||||
|
{
|
||||||
|
tableMetaData: QTableMetaData;
|
||||||
|
queryFilter: QQueryFilter;
|
||||||
|
isEditable: boolean;
|
||||||
|
isQueryTooComplex: boolean;
|
||||||
|
removeCriteriaByIndexCallback: (index: number) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Box shown on query screen (and more??) to preview what a query looks like,
|
||||||
|
** as an "advanced" style/precursor-to-writing-your-own-query thing.
|
||||||
|
*******************************************************************************/
|
||||||
|
export default function AdvancedQueryPreview({tableMetaData, queryFilter, isEditable, isQueryTooComplex, removeCriteriaByIndexCallback}: AdvancedQueryPreviewProps): JSX.Element
|
||||||
|
{
|
||||||
|
const [mouseOverElement, setMouseOverElement] = useState(null as string);
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function handleMouseOverElement(name: string)
|
||||||
|
{
|
||||||
|
setMouseOverElement(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function handleMouseOutElement()
|
||||||
|
{
|
||||||
|
setMouseOverElement(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** format the current query as a string for showing on-screen as a preview.
|
||||||
|
*******************************************************************************/
|
||||||
|
const queryToAdvancedString = (thisQueryFilter: QQueryFilter) =>
|
||||||
|
{
|
||||||
|
if (queryFilter == null || !queryFilter.criteria)
|
||||||
|
{
|
||||||
|
return (<span></span>);
|
||||||
|
}
|
||||||
|
|
||||||
|
let counter = 0;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<React.Fragment>
|
||||||
|
{thisQueryFilter.criteria?.map((criteria, i) =>
|
||||||
|
{
|
||||||
|
const {criteriaIsValid} = validateCriteria(criteria, null);
|
||||||
|
if (criteriaIsValid)
|
||||||
|
{
|
||||||
|
counter++;
|
||||||
|
return (
|
||||||
|
<span key={i} style={{marginBottom: "0.125rem"}} onMouseOver={() => handleMouseOverElement(`queryPreview-${i}`)} onMouseOut={() => handleMouseOutElement()}>
|
||||||
|
{counter > 1 ? <span style={{marginLeft: "0.25rem", marginRight: "0.25rem"}}>{thisQueryFilter.booleanOperator} </span> : <span />}
|
||||||
|
{FilterUtils.criteriaToHumanString(tableMetaData, criteria, true)}
|
||||||
|
{isEditable && !isQueryTooComplex && (
|
||||||
|
mouseOverElement == `queryPreview-${i}` && <span className={`advancedQueryPreviewX-${counter - 1}`}>
|
||||||
|
<XIcon position="forAdvancedQueryPreview" onClick={() => removeCriteriaByIndexCallback(i)} /></span>
|
||||||
|
)}
|
||||||
|
{counter > 1 && i == thisQueryFilter.criteria?.length - 1 && thisQueryFilter.subFilters?.length > 0 ? <span style={{marginLeft: "0.25rem", marginRight: "0.25rem"}}>{thisQueryFilter.booleanOperator} </span> : <span />}
|
||||||
|
</span>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return (<span />);
|
||||||
|
}
|
||||||
|
})}
|
||||||
|
|
||||||
|
{thisQueryFilter.subFilters?.length > 0 && (thisQueryFilter.subFilters.map((filter: QQueryFilter, j) =>
|
||||||
|
{
|
||||||
|
return (
|
||||||
|
<React.Fragment key={j}>
|
||||||
|
{j > 0 ? <span style={{marginLeft: "0.25rem", marginRight: "0.25rem"}}>{thisQueryFilter.booleanOperator} </span> : <span></span>}
|
||||||
|
<span style={{display: "flex", marginRight: "0.20rem"}}>(</span>
|
||||||
|
{queryToAdvancedString(filter)}
|
||||||
|
<span style={{display: "flex", marginRight: "0.20rem"}}>)</span>
|
||||||
|
</React.Fragment>
|
||||||
|
);
|
||||||
|
}))}
|
||||||
|
</React.Fragment>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const moreSX = isEditable ?
|
||||||
|
{
|
||||||
|
borderTop: `1px solid ${colors.grayLines.main}`,
|
||||||
|
boxShadow: "inset 0px 0px 4px 2px #EFEFED",
|
||||||
|
borderRadius: "0 0 0.75rem 0.75rem",
|
||||||
|
} :
|
||||||
|
{
|
||||||
|
borderRadius: "0.75rem",
|
||||||
|
border: `1px solid ${colors.grayLines.main}`,
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Box whiteSpace="nowrap" display="flex" flexShrink={1} flexGrow={1} alignItems="center">
|
||||||
|
{
|
||||||
|
<Box
|
||||||
|
className="advancedQueryString"
|
||||||
|
display="inline-block"
|
||||||
|
width="100%"
|
||||||
|
sx={{fontSize: "1rem", background: "#FFFFFF"}}
|
||||||
|
minHeight={"2.5rem"}
|
||||||
|
p={"0.5rem"}
|
||||||
|
pb={"0.125rem"}
|
||||||
|
{...moreSX}
|
||||||
|
>
|
||||||
|
<Box display="flex" flexWrap="wrap" fontSize="0.875rem">
|
||||||
|
{queryToAdvancedString(queryFilter)}
|
||||||
|
</Box>
|
||||||
|
</Box>
|
||||||
|
}
|
||||||
|
</Box>
|
||||||
|
)
|
||||||
|
}
|
@ -44,11 +44,13 @@ import {GridApiPro} from "@mui/x-data-grid-pro/models/gridApiPro";
|
|||||||
import QContext from "QContext";
|
import QContext from "QContext";
|
||||||
import colors from "qqq/assets/theme/base/colors";
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
import {QCancelButton, QSaveButton} from "qqq/components/buttons/DefaultButtons";
|
import {QCancelButton, QSaveButton} from "qqq/components/buttons/DefaultButtons";
|
||||||
|
import AdvancedQueryPreview from "qqq/components/query/AdvancedQueryPreview";
|
||||||
import {QFilterCriteriaWithId} from "qqq/components/query/CustomFilterPanel";
|
import {QFilterCriteriaWithId} from "qqq/components/query/CustomFilterPanel";
|
||||||
import FieldListMenu from "qqq/components/query/FieldListMenu";
|
import FieldListMenu from "qqq/components/query/FieldListMenu";
|
||||||
import {validateCriteria} from "qqq/components/query/FilterCriteriaRow";
|
import {validateCriteria} from "qqq/components/query/FilterCriteriaRow";
|
||||||
import QuickFilter, {quickFilterButtonStyles} from "qqq/components/query/QuickFilter";
|
import QuickFilter, {quickFilterButtonStyles} from "qqq/components/query/QuickFilter";
|
||||||
import XIcon from "qqq/components/query/XIcon";
|
import XIcon from "qqq/components/query/XIcon";
|
||||||
|
import {QueryScreenUsage} from "qqq/pages/records/query/RecordQuery";
|
||||||
import FilterUtils from "qqq/utils/qqq/FilterUtils";
|
import FilterUtils from "qqq/utils/qqq/FilterUtils";
|
||||||
import TableUtils from "qqq/utils/qqq/TableUtils";
|
import TableUtils from "qqq/utils/qqq/TableUtils";
|
||||||
import React, {forwardRef, useContext, useImperativeHandle, useReducer, useState} from "react";
|
import React, {forwardRef, useContext, useImperativeHandle, useReducer, useState} from "react";
|
||||||
@ -75,6 +77,8 @@ interface BasicAndAdvancedQueryControlsProps
|
|||||||
/////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
queryFilterJSON: string;
|
queryFilterJSON: string;
|
||||||
|
|
||||||
|
queryScreenUsage: QueryScreenUsage;
|
||||||
|
|
||||||
mode: string;
|
mode: string;
|
||||||
setMode: (mode: string) => void;
|
setMode: (mode: string) => void;
|
||||||
}
|
}
|
||||||
@ -397,60 +401,6 @@ const BasicAndAdvancedQueryControls = forwardRef((props: BasicAndAdvancedQueryCo
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
/*******************************************************************************
|
|
||||||
** format the current query as a string for showing on-screen as a preview.
|
|
||||||
*******************************************************************************/
|
|
||||||
const queryToAdvancedString = (thisQueryFilter: QQueryFilter) =>
|
|
||||||
{
|
|
||||||
if (queryFilter == null || !queryFilter.criteria)
|
|
||||||
{
|
|
||||||
return (<span></span>);
|
|
||||||
}
|
|
||||||
|
|
||||||
let counter = 0;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<React.Fragment>
|
|
||||||
{thisQueryFilter.criteria?.map((criteria, i) =>
|
|
||||||
{
|
|
||||||
const {criteriaIsValid} = validateCriteria(criteria, null);
|
|
||||||
if (criteriaIsValid)
|
|
||||||
{
|
|
||||||
counter++;
|
|
||||||
return (
|
|
||||||
<span key={i} style={{marginBottom: "0.125rem"}} onMouseOver={() => handleMouseOverElement(`queryPreview-${i}`)} onMouseOut={() => handleMouseOutElement()}>
|
|
||||||
{counter > 1 ? <span style={{marginLeft: "0.25rem", marginRight: "0.25rem"}}>{thisQueryFilter.booleanOperator} </span> : <span />}
|
|
||||||
{FilterUtils.criteriaToHumanString(tableMetaData, criteria, true)}
|
|
||||||
{!isQueryTooComplex && (
|
|
||||||
mouseOverElement == `queryPreview-${i}` && <span className={`advancedQueryPreviewX-${counter - 1}`}>
|
|
||||||
<XIcon position="forAdvancedQueryPreview" onClick={() => removeCriteriaByIndex(i)} /></span>
|
|
||||||
)}
|
|
||||||
{counter > 1 && i == thisQueryFilter.criteria?.length - 1 && thisQueryFilter.subFilters?.length > 0 ? <span style={{marginLeft: "0.25rem", marginRight: "0.25rem"}}>{thisQueryFilter.booleanOperator} </span> : <span />}
|
|
||||||
</span>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
return (<span />);
|
|
||||||
}
|
|
||||||
})}
|
|
||||||
|
|
||||||
{thisQueryFilter.subFilters?.length > 0 && (thisQueryFilter.subFilters.map((filter: QQueryFilter, j) =>
|
|
||||||
{
|
|
||||||
return (
|
|
||||||
<React.Fragment key={j}>
|
|
||||||
{j > 0 ? <span style={{marginLeft: "0.25rem", marginRight: "0.25rem"}}>{thisQueryFilter.booleanOperator} </span> : <span></span>}
|
|
||||||
<span style={{display: "flex", marginRight: "0.20rem"}}>(</span>
|
|
||||||
{queryToAdvancedString(filter)}
|
|
||||||
<span style={{display: "flex", marginRight: "0.20rem"}}>)</span>
|
|
||||||
</React.Fragment>
|
|
||||||
);
|
|
||||||
}))}
|
|
||||||
</React.Fragment>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
** event handler for toggling between modes - basic & advanced.
|
** event handler for toggling between modes - basic & advanced.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
@ -807,26 +757,7 @@ const BasicAndAdvancedQueryControls = forwardRef((props: BasicAndAdvancedQueryCo
|
|||||||
{sortMenuComponent}
|
{sortMenuComponent}
|
||||||
</Box>
|
</Box>
|
||||||
</Box>
|
</Box>
|
||||||
<Box whiteSpace="nowrap" display="flex" flexShrink={1} flexGrow={1} alignItems="center">
|
<AdvancedQueryPreview tableMetaData={tableMetaData} queryFilter={queryFilter} isEditable={true} isQueryTooComplex={isQueryTooComplex} removeCriteriaByIndexCallback={removeCriteriaByIndex} />
|
||||||
{
|
|
||||||
<Box
|
|
||||||
className="advancedQueryString"
|
|
||||||
display="inline-block"
|
|
||||||
borderTop={`1px solid ${borderGray}`}
|
|
||||||
borderRadius="0 0 0.75rem 0.75rem"
|
|
||||||
width="100%"
|
|
||||||
sx={{fontSize: "1rem", background: "#FFFFFF"}}
|
|
||||||
minHeight={"2.375rem"}
|
|
||||||
p={"0.5rem"}
|
|
||||||
pb={"0.125rem"}
|
|
||||||
boxShadow={"inset 0px 0px 4px 2px #EFEFED"}
|
|
||||||
>
|
|
||||||
<Box display="flex" flexWrap="wrap" fontSize="0.875rem">
|
|
||||||
{queryToAdvancedString(queryFilter)}
|
|
||||||
</Box>
|
|
||||||
</Box>
|
|
||||||
}
|
|
||||||
</Box>
|
|
||||||
</Box>
|
</Box>
|
||||||
}
|
}
|
||||||
</Box>
|
</Box>
|
||||||
|
@ -33,10 +33,10 @@ import MenuItem from "@mui/material/MenuItem";
|
|||||||
import Select, {SelectChangeEvent} from "@mui/material/Select/Select";
|
import Select, {SelectChangeEvent} from "@mui/material/Select/Select";
|
||||||
import TextField from "@mui/material/TextField";
|
import TextField from "@mui/material/TextField";
|
||||||
import Tooltip from "@mui/material/Tooltip";
|
import Tooltip from "@mui/material/Tooltip";
|
||||||
import React, {ReactNode, SyntheticEvent, useState} from "react";
|
|
||||||
import FieldAutoComplete from "qqq/components/misc/FieldAutoComplete";
|
import FieldAutoComplete from "qqq/components/misc/FieldAutoComplete";
|
||||||
import FilterCriteriaRowValues from "qqq/components/query/FilterCriteriaRowValues";
|
import FilterCriteriaRowValues from "qqq/components/query/FilterCriteriaRowValues";
|
||||||
import FilterUtils from "qqq/utils/qqq/FilterUtils";
|
import FilterUtils from "qqq/utils/qqq/FilterUtils";
|
||||||
|
import React, {ReactNode, SyntheticEvent, useState} from "react";
|
||||||
|
|
||||||
|
|
||||||
export enum ValueMode
|
export enum ValueMode
|
||||||
@ -484,7 +484,9 @@ export function FilterCriteriaRow({id, index, tableMetaData, metaData, criteria,
|
|||||||
: <span />}
|
: <span />}
|
||||||
</Box>
|
</Box>
|
||||||
<Box display="inline-block" width={250} className="fieldColumn">
|
<Box display="inline-block" width={250} className="fieldColumn">
|
||||||
<FieldAutoComplete id={`field-${id}`} metaData={metaData} tableMetaData={tableMetaData} defaultValue={defaultFieldValue} handleFieldChange={handleFieldChange} />
|
<FieldAutoComplete id={`field-${id}`} metaData={metaData} tableMetaData={tableMetaData} defaultValue={defaultFieldValue} handleFieldChange={handleFieldChange}
|
||||||
|
autocompleteSlotProps={{popper: {className: "filterCriteriaRowColumnPopper", style: {padding: 0, width: "250px"}}}}
|
||||||
|
/>
|
||||||
</Box>
|
</Box>
|
||||||
<Box display="inline-block" width={200} className="operatorColumn">
|
<Box display="inline-block" width={200} className="operatorColumn">
|
||||||
<Tooltip title={criteria.fieldName == null ? "You must select a field before you can select an operator" : null} enterDelay={tooltipEnterDelay}>
|
<Tooltip title={criteria.fieldName == null ? "You must select a field before you can select an operator" : null} enterDelay={tooltipEnterDelay}>
|
||||||
|
@ -19,6 +19,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import {QWidgetMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QWidgetMetaData";
|
import {QWidgetMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QWidgetMetaData";
|
||||||
|
import {QRecord} from "@kingsrook/qqq-frontend-core/lib/model/QRecord";
|
||||||
import {Alert, Skeleton} from "@mui/material";
|
import {Alert, Skeleton} from "@mui/material";
|
||||||
import Box from "@mui/material/Box";
|
import Box from "@mui/material/Box";
|
||||||
import Grid from "@mui/material/Grid";
|
import Grid from "@mui/material/Grid";
|
||||||
@ -38,8 +39,10 @@ import CompositeWidget from "qqq/components/widgets/CompositeWidget";
|
|||||||
import DataBagViewer from "qqq/components/widgets/misc/DataBagViewer";
|
import DataBagViewer from "qqq/components/widgets/misc/DataBagViewer";
|
||||||
import DividerWidget from "qqq/components/widgets/misc/Divider";
|
import DividerWidget from "qqq/components/widgets/misc/Divider";
|
||||||
import FieldValueListWidget from "qqq/components/widgets/misc/FieldValueListWidget";
|
import FieldValueListWidget from "qqq/components/widgets/misc/FieldValueListWidget";
|
||||||
|
import PivotTableSetupWidget from "qqq/components/widgets/misc/PivotTableSetupWidget";
|
||||||
import QuickSightChart from "qqq/components/widgets/misc/QuickSightChart";
|
import QuickSightChart from "qqq/components/widgets/misc/QuickSightChart";
|
||||||
import RecordGridWidget from "qqq/components/widgets/misc/RecordGridWidget";
|
import RecordGridWidget from "qqq/components/widgets/misc/RecordGridWidget";
|
||||||
|
import ReportSetupWidget from "qqq/components/widgets/misc/ReportSetupWidget";
|
||||||
import ScriptViewer from "qqq/components/widgets/misc/ScriptViewer";
|
import ScriptViewer from "qqq/components/widgets/misc/ScriptViewer";
|
||||||
import StepperCard from "qqq/components/widgets/misc/StepperCard";
|
import StepperCard from "qqq/components/widgets/misc/StepperCard";
|
||||||
import USMapWidget from "qqq/components/widgets/misc/USMapWidget";
|
import USMapWidget from "qqq/components/widgets/misc/USMapWidget";
|
||||||
@ -61,6 +64,7 @@ interface Props
|
|||||||
widgetMetaDataList: QWidgetMetaData[];
|
widgetMetaDataList: QWidgetMetaData[];
|
||||||
tableName?: string;
|
tableName?: string;
|
||||||
entityPrimaryKey?: string;
|
entityPrimaryKey?: string;
|
||||||
|
record?: QRecord;
|
||||||
omitWrappingGridContainer: boolean;
|
omitWrappingGridContainer: boolean;
|
||||||
areChildren?: boolean;
|
areChildren?: boolean;
|
||||||
childUrlParams?: string;
|
childUrlParams?: string;
|
||||||
@ -79,7 +83,7 @@ DashboardWidgets.defaultProps = {
|
|||||||
wrapWidgetsInTabPanels: false,
|
wrapWidgetsInTabPanels: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
function DashboardWidgets({widgetMetaDataList, tableName, entityPrimaryKey, omitWrappingGridContainer, areChildren, childUrlParams, parentWidgetMetaData, wrapWidgetsInTabPanels}: Props): JSX.Element
|
function DashboardWidgets({widgetMetaDataList, tableName, entityPrimaryKey, record, omitWrappingGridContainer, areChildren, childUrlParams, parentWidgetMetaData, wrapWidgetsInTabPanels}: Props): JSX.Element
|
||||||
{
|
{
|
||||||
const [widgetData, setWidgetData] = useState([] as any[]);
|
const [widgetData, setWidgetData] = useState([] as any[]);
|
||||||
const [widgetCounter, setWidgetCounter] = useState(0);
|
const [widgetCounter, setWidgetCounter] = useState(0);
|
||||||
@ -248,6 +252,23 @@ function DashboardWidgets({widgetMetaDataList, tableName, entityPrimaryKey, omit
|
|||||||
|
|
||||||
const widgetCount = widgetMetaDataList ? widgetMetaDataList.length : 0;
|
const widgetCount = widgetMetaDataList ? widgetMetaDataList.length : 0;
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** helper function, to convert values from a QRecord values map to a regular old
|
||||||
|
** js object
|
||||||
|
*******************************************************************************/
|
||||||
|
function convertQRecordValuesFromMapToObject(record: QRecord): {[name: string]: any}
|
||||||
|
{
|
||||||
|
const rs: {[name: string]: any} = {};
|
||||||
|
|
||||||
|
if(record.values)
|
||||||
|
{
|
||||||
|
record.values.forEach((value, key) => rs[key] = value);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (rs);
|
||||||
|
}
|
||||||
|
|
||||||
const renderWidget = (widgetMetaData: QWidgetMetaData, i: number): JSX.Element =>
|
const renderWidget = (widgetMetaData: QWidgetMetaData, i: number): JSX.Element =>
|
||||||
{
|
{
|
||||||
const labelAdditionalComponentsRight: LabelComponent[] = [];
|
const labelAdditionalComponentsRight: LabelComponent[] = [];
|
||||||
@ -561,6 +582,20 @@ function DashboardWidgets({widgetMetaDataList, tableName, entityPrimaryKey, omit
|
|||||||
</Widget>
|
</Widget>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
{
|
||||||
|
widgetMetaData.type === "reportSetup" && (
|
||||||
|
widgetData && widgetData[i] && widgetData[i].queryParams &&
|
||||||
|
<ReportSetupWidget isEditable={false} widgetMetaData={widgetMetaData} recordValues={convertQRecordValuesFromMapToObject(record)} onSaveCallback={() =>
|
||||||
|
{}} />
|
||||||
|
)
|
||||||
|
}
|
||||||
|
{
|
||||||
|
widgetMetaData.type === "pivotTableSetup" && (
|
||||||
|
widgetData && widgetData[i] && widgetData[i].queryParams &&
|
||||||
|
<PivotTableSetupWidget isEditable={false} widgetMetaData={widgetMetaData} recordValues={convertQRecordValuesFromMapToObject(record)} onSaveCallback={() =>
|
||||||
|
{}} />
|
||||||
|
)
|
||||||
|
}
|
||||||
</Box>
|
</Box>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@ -21,10 +21,12 @@
|
|||||||
|
|
||||||
import {QTableMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QTableMetaData";
|
import {QTableMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QTableMetaData";
|
||||||
import {QWidgetMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QWidgetMetaData";
|
import {QWidgetMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QWidgetMetaData";
|
||||||
|
import {InputLabel} from "@mui/material";
|
||||||
import Box from "@mui/material/Box";
|
import Box from "@mui/material/Box";
|
||||||
import Button from "@mui/material/Button";
|
import Button from "@mui/material/Button";
|
||||||
import Card from "@mui/material/Card";
|
import Card from "@mui/material/Card";
|
||||||
import Icon from "@mui/material/Icon";
|
import Icon from "@mui/material/Icon";
|
||||||
|
import Switch from "@mui/material/Switch";
|
||||||
import Tooltip from "@mui/material/Tooltip/Tooltip";
|
import Tooltip from "@mui/material/Tooltip/Tooltip";
|
||||||
import Typography from "@mui/material/Typography";
|
import Typography from "@mui/material/Typography";
|
||||||
import parse from "html-react-parser";
|
import parse from "html-react-parser";
|
||||||
@ -60,6 +62,7 @@ interface Props
|
|||||||
labelAdditionalComponentsLeft: LabelComponent[];
|
labelAdditionalComponentsLeft: LabelComponent[];
|
||||||
labelAdditionalElementsLeft: JSX.Element[];
|
labelAdditionalElementsLeft: JSX.Element[];
|
||||||
labelAdditionalComponentsRight: LabelComponent[];
|
labelAdditionalComponentsRight: LabelComponent[];
|
||||||
|
labelAdditionalElementsRight: JSX.Element[];
|
||||||
labelBoxAdditionalSx?: any;
|
labelBoxAdditionalSx?: any;
|
||||||
widgetMetaData?: QWidgetMetaData;
|
widgetMetaData?: QWidgetMetaData;
|
||||||
widgetData?: WidgetData;
|
widgetData?: WidgetData;
|
||||||
@ -80,6 +83,7 @@ Widget.defaultProps = {
|
|||||||
labelAdditionalComponentsLeft: [],
|
labelAdditionalComponentsLeft: [],
|
||||||
labelAdditionalElementsLeft: [],
|
labelAdditionalElementsLeft: [],
|
||||||
labelAdditionalComponentsRight: [],
|
labelAdditionalComponentsRight: [],
|
||||||
|
labelAdditionalElementsRight: [],
|
||||||
labelBoxAdditionalSx: {},
|
labelBoxAdditionalSx: {},
|
||||||
omitPadding: false,
|
omitPadding: false,
|
||||||
};
|
};
|
||||||
@ -160,6 +164,79 @@ export class HeaderIcon extends LabelComponent
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** a link (actually a button) for in a widget's header
|
||||||
|
*******************************************************************************/
|
||||||
|
interface HeaderLinkButtonComponentProps
|
||||||
|
{
|
||||||
|
label: string;
|
||||||
|
onClickCallback: () => void;
|
||||||
|
disabled?: boolean;
|
||||||
|
disabledTooltip?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
HeaderLinkButtonComponent.defaultProps = {
|
||||||
|
disabled: false,
|
||||||
|
disabledTooltip: null
|
||||||
|
};
|
||||||
|
|
||||||
|
export function HeaderLinkButtonComponent({label, onClickCallback, disabled, disabledTooltip}: HeaderLinkButtonComponentProps): JSX.Element
|
||||||
|
{
|
||||||
|
return (
|
||||||
|
<Tooltip title={disabledTooltip}>
|
||||||
|
<span>
|
||||||
|
<Button disabled={disabled} onClick={() => onClickCallback()} sx={{p: 0}} disableRipple>
|
||||||
|
<Typography display="inline" textTransform="none" fontSize={"1.125rem"}>
|
||||||
|
{label}
|
||||||
|
</Typography>
|
||||||
|
</Button>
|
||||||
|
</span>
|
||||||
|
</Tooltip>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
interface HeaderToggleComponentProps
|
||||||
|
{
|
||||||
|
label: string;
|
||||||
|
getValue: () => boolean;
|
||||||
|
onClickCallback: () => void;
|
||||||
|
disabled?: boolean;
|
||||||
|
disabledTooltip?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
HeaderToggleComponent.defaultProps = {
|
||||||
|
disabled: false,
|
||||||
|
disabledTooltip: null
|
||||||
|
};
|
||||||
|
|
||||||
|
export function HeaderToggleComponent({label, getValue, onClickCallback, disabled, disabledTooltip}: HeaderToggleComponentProps): JSX.Element
|
||||||
|
{
|
||||||
|
const onClick = () =>
|
||||||
|
{
|
||||||
|
onClickCallback();
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Box alignItems="baseline" mr="-0.75rem">
|
||||||
|
<Tooltip title={disabledTooltip}>
|
||||||
|
<span>
|
||||||
|
<InputLabel sx={{fontSize: "1.125rem", px: "0 !important", cursor: disabled ? "default" : "pointer", opacity: disabled ? 0.65 : 1}} unselectable="on">
|
||||||
|
{label} <Switch disabled={disabled} checked={getValue()} onClick={onClick} />
|
||||||
|
</InputLabel>
|
||||||
|
</span>
|
||||||
|
</Tooltip>
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
**
|
**
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
@ -573,6 +650,8 @@ function Widget(props: React.PropsWithChildren<Props>): JSX.Element
|
|||||||
localLabelAdditionalElementsLeft.push(WidgetUtils.generateExportButton(onExportClick));
|
localLabelAdditionalElementsLeft.push(WidgetUtils.generateExportButton(onExportClick));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let localLabelAdditionalElementsRight = [...props.labelAdditionalElementsRight];
|
||||||
|
|
||||||
const hasPermission = props.widgetData?.hasPermission === undefined || props.widgetData?.hasPermission === true;
|
const hasPermission = props.widgetData?.hasPermission === undefined || props.widgetData?.hasPermission === true;
|
||||||
|
|
||||||
const isSet = (v: any): boolean =>
|
const isSet = (v: any): boolean =>
|
||||||
@ -589,6 +668,7 @@ function Widget(props: React.PropsWithChildren<Props>): JSX.Element
|
|||||||
needLabelBox ||= (labelComponentsLeft && labelComponentsLeft.length > 0);
|
needLabelBox ||= (labelComponentsLeft && labelComponentsLeft.length > 0);
|
||||||
needLabelBox ||= (localLabelAdditionalElementsLeft && localLabelAdditionalElementsLeft.length > 0);
|
needLabelBox ||= (localLabelAdditionalElementsLeft && localLabelAdditionalElementsLeft.length > 0);
|
||||||
needLabelBox ||= (labelComponentsRight && labelComponentsRight.length > 0);
|
needLabelBox ||= (labelComponentsRight && labelComponentsRight.length > 0);
|
||||||
|
needLabelBox ||= (localLabelAdditionalElementsRight && localLabelAdditionalElementsRight.length > 0);
|
||||||
needLabelBox ||= isSet(props.widgetData?.icon);
|
needLabelBox ||= isSet(props.widgetData?.icon);
|
||||||
needLabelBox ||= isSet(props.widgetData?.label);
|
needLabelBox ||= isSet(props.widgetData?.label);
|
||||||
needLabelBox ||= isSet(props.widgetMetaData?.label);
|
needLabelBox ||= isSet(props.widgetMetaData?.label);
|
||||||
@ -720,6 +800,7 @@ function Widget(props: React.PropsWithChildren<Props>): JSX.Element
|
|||||||
})
|
})
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
{localLabelAdditionalElementsRight}
|
||||||
</Box>
|
</Box>
|
||||||
</Box>
|
</Box>
|
||||||
}
|
}
|
||||||
|
208
src/qqq/components/widgets/misc/PivotTableGroupByElement.tsx
Normal file
208
src/qqq/components/widgets/misc/PivotTableGroupByElement.tsx
Normal file
@ -0,0 +1,208 @@
|
|||||||
|
/*
|
||||||
|
* 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 {QInstance} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QInstance";
|
||||||
|
import {QTableMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QTableMetaData";
|
||||||
|
import Box from "@mui/material/Box";
|
||||||
|
import Button from "@mui/material/Button";
|
||||||
|
import Icon from "@mui/material/Icon";
|
||||||
|
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 {PivotTableDefinition, PivotTableGroupBy} from "qqq/models/misc/PivotTableDefinitionModels";
|
||||||
|
import React, {FC, useRef} from "react";
|
||||||
|
import {useDrag, useDrop} from "react-dnd";
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** component props
|
||||||
|
*******************************************************************************/
|
||||||
|
export interface PivotTableGroupByElementProps
|
||||||
|
{
|
||||||
|
id: string;
|
||||||
|
index: number;
|
||||||
|
dragCallback: (rowsOrColumns: "rows" | "columns", dragIndex: number, hoverIndex: number) => void;
|
||||||
|
metaData: QInstance;
|
||||||
|
tableMetaData: QTableMetaData;
|
||||||
|
pivotTableDefinition: PivotTableDefinition;
|
||||||
|
usedGroupByFieldNames: string[];
|
||||||
|
availableFieldNames: string[];
|
||||||
|
isEditable: boolean;
|
||||||
|
groupBy: PivotTableGroupBy;
|
||||||
|
rowsOrColumns: "rows" | "columns";
|
||||||
|
callback: () => void;
|
||||||
|
attemptedSubmit?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** item to support react-dnd
|
||||||
|
*******************************************************************************/
|
||||||
|
interface DragItem
|
||||||
|
{
|
||||||
|
index: number;
|
||||||
|
id: string;
|
||||||
|
type: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
export const PivotTableGroupByElement: FC<PivotTableGroupByElementProps> = ({id, index, dragCallback, rowsOrColumns, metaData, tableMetaData, pivotTableDefinition, groupBy, usedGroupByFieldNames, availableFieldNames, isEditable, callback, attemptedSubmit}) =>
|
||||||
|
{
|
||||||
|
////////////////////////////////////////////////////////////////////////////
|
||||||
|
// 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: rowsOrColumns == "rows" ? DragItemTypes.ROW : DragItemTypes.COLUMN,
|
||||||
|
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(rowsOrColumns, 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, preview] = useDrag({
|
||||||
|
type: rowsOrColumns == "rows" ? DragItemTypes.ROW : DragItemTypes.COLUMN,
|
||||||
|
item: () =>
|
||||||
|
{
|
||||||
|
return {id, index};
|
||||||
|
},
|
||||||
|
collect: (monitor: any) => ({
|
||||||
|
isDragging: monitor.isDragging(),
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
const handleFieldChange = (event: any, newValue: any, reason: string) =>
|
||||||
|
{
|
||||||
|
groupBy.fieldName = newValue ? newValue.fieldName : null;
|
||||||
|
callback();
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function removeGroupBy(index: number, rowsOrColumns: "rows" | "columns")
|
||||||
|
{
|
||||||
|
pivotTableDefinition[rowsOrColumns].splice(index, 1);
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isEditable)
|
||||||
|
{
|
||||||
|
const selectedField = getSelectedFieldForAutoComplete(tableMetaData, groupBy.fieldName);
|
||||||
|
if (selectedField)
|
||||||
|
{
|
||||||
|
const label = 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>);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (<React.Fragment />);
|
||||||
|
}
|
||||||
|
|
||||||
|
preview(drop(ref));
|
||||||
|
|
||||||
|
const showError = attemptedSubmit && !groupBy.fieldName;
|
||||||
|
|
||||||
|
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 ref={drag} sx={{cursor: "ns-resize"}}>drag_indicator</Icon>
|
||||||
|
</Box>
|
||||||
|
<Box width="100%">
|
||||||
|
<FieldAutoComplete
|
||||||
|
id={`${rowsOrColumns}-${index}`}
|
||||||
|
label={null}
|
||||||
|
variant="outlined"
|
||||||
|
textFieldSX={fieldAutoCompleteTextFieldSX}
|
||||||
|
metaData={metaData}
|
||||||
|
tableMetaData={tableMetaData}
|
||||||
|
handleFieldChange={handleFieldChange}
|
||||||
|
hiddenFieldNames={usedGroupByFieldNames}
|
||||||
|
availableFieldNames={availableFieldNames}
|
||||||
|
defaultValue={getSelectedFieldForAutoComplete(tableMetaData, groupBy.fieldName)}
|
||||||
|
hasError={showError}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
<Box>
|
||||||
|
<Button sx={xIconButtonSX} onClick={() => removeGroupBy(index, rowsOrColumns)}><Icon>clear</Icon></Button>
|
||||||
|
</Box>
|
||||||
|
</Box>);
|
||||||
|
};
|
849
src/qqq/components/widgets/misc/PivotTableSetupWidget.tsx
Normal file
849
src/qqq/components/widgets/misc/PivotTableSetupWidget.tsx
Normal file
@ -0,0 +1,849 @@
|
|||||||
|
/*
|
||||||
|
* 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 {QInstance} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QInstance";
|
||||||
|
import {QTableMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QTableMetaData";
|
||||||
|
import {QWidgetMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QWidgetMetaData";
|
||||||
|
import Alert from "@mui/material/Alert";
|
||||||
|
import Autocomplete from "@mui/material/Autocomplete";
|
||||||
|
import Box from "@mui/material/Box";
|
||||||
|
import Button from "@mui/material/Button";
|
||||||
|
import Card from "@mui/material/Card";
|
||||||
|
import Grid from "@mui/material/Grid";
|
||||||
|
import Icon from "@mui/material/Icon";
|
||||||
|
import Modal from "@mui/material/Modal";
|
||||||
|
import TextField from "@mui/material/TextField";
|
||||||
|
import Tooltip from "@mui/material/Tooltip/Tooltip";
|
||||||
|
import Typography from "@mui/material/Typography";
|
||||||
|
import QContext from "QContext";
|
||||||
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
|
import {QCancelButton, QSaveButton} from "qqq/components/buttons/DefaultButtons";
|
||||||
|
import FieldAutoComplete from "qqq/components/misc/FieldAutoComplete";
|
||||||
|
import HelpContent, {hasHelpContent} from "qqq/components/misc/HelpContent";
|
||||||
|
import {PivotTableGroupByElement} from "qqq/components/widgets/misc/PivotTableGroupByElement";
|
||||||
|
import {PivotTableValueElement} from "qqq/components/widgets/misc/PivotTableValueElement";
|
||||||
|
import {buttonSX, unborderedButtonSX} from "qqq/components/widgets/misc/ReportSetupWidget";
|
||||||
|
import Widget, {HeaderToggleComponent} from "qqq/components/widgets/Widget";
|
||||||
|
import {PivotObjectKey, PivotTableDefinition, PivotTableFunction, pivotTableFunctionLabels, PivotTableGroupBy, PivotTableValue} from "qqq/models/misc/PivotTableDefinitionModels";
|
||||||
|
import QQueryColumns from "qqq/models/query/QQueryColumns";
|
||||||
|
import Client from "qqq/utils/qqq/Client";
|
||||||
|
import FilterUtils from "qqq/utils/qqq/FilterUtils";
|
||||||
|
import React, {useCallback, useContext, useEffect, useReducer, useState} from "react";
|
||||||
|
import {DndProvider} from "react-dnd";
|
||||||
|
import {HTML5Backend} from "react-dnd-html5-backend";
|
||||||
|
|
||||||
|
export const DragItemTypes =
|
||||||
|
{
|
||||||
|
ROW: "row",
|
||||||
|
COLUMN: "column",
|
||||||
|
VALUE: "value"
|
||||||
|
};
|
||||||
|
|
||||||
|
export const xIconButtonSX =
|
||||||
|
{
|
||||||
|
border: `1px solid ${colors.grayLines.main} !important`,
|
||||||
|
borderRadius: "0.75rem",
|
||||||
|
textTransform: "none",
|
||||||
|
fontSize: "1rem",
|
||||||
|
fontWeight: "400",
|
||||||
|
width: "40px",
|
||||||
|
minWidth: "40px",
|
||||||
|
paddingLeft: 0,
|
||||||
|
paddingRight: 0,
|
||||||
|
color: colors.error.main,
|
||||||
|
"&:hover": {color: colors.error.main},
|
||||||
|
"&:focus": {color: colors.error.main},
|
||||||
|
"&:focus:not(:hover)": {color: colors.error.main},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const fieldAutoCompleteTextFieldSX =
|
||||||
|
{
|
||||||
|
"& .MuiInputBase-input": {fontSize: "1rem", padding: "0 !important"}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
export function getSelectedFieldForAutoComplete(tableMetaData: QTableMetaData, fieldName: string)
|
||||||
|
{
|
||||||
|
if (fieldName)
|
||||||
|
{
|
||||||
|
let [field, fieldTable] = FilterUtils.getField(tableMetaData, fieldName);
|
||||||
|
if (field && fieldTable)
|
||||||
|
{
|
||||||
|
return ({field: field, table: fieldTable, fieldName: fieldName});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (null);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** component props
|
||||||
|
*******************************************************************************/
|
||||||
|
interface PivotTableSetupWidgetProps
|
||||||
|
{
|
||||||
|
isEditable: boolean;
|
||||||
|
widgetMetaData: QWidgetMetaData;
|
||||||
|
recordValues: { [name: string]: any };
|
||||||
|
onSaveCallback?: (values: { [name: string]: any }) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** default values for props
|
||||||
|
*******************************************************************************/
|
||||||
|
PivotTableSetupWidget.defaultProps = {
|
||||||
|
onSaveCallback: null
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
const qController = Client.getInstance();
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Component to edit the setup of a Pivot Table - rows, columns, values!
|
||||||
|
*******************************************************************************/
|
||||||
|
export default function PivotTableSetupWidget({isEditable, widgetMetaData, recordValues, onSaveCallback}: PivotTableSetupWidgetProps): JSX.Element
|
||||||
|
{
|
||||||
|
const [metaData, setMetaData] = useState(null as QInstance);
|
||||||
|
const [tableMetaData, setTableMetaData] = useState(null as QTableMetaData);
|
||||||
|
|
||||||
|
const [modalOpen, setModalOpen] = useState(false);
|
||||||
|
const [enabled, setEnabled] = useState(!!recordValues["usePivotTable"]);
|
||||||
|
const [attemptedSubmit, setAttemptedSubmit] = useState(false);
|
||||||
|
const [errorAlert, setErrorAlert] = useState(null as string);
|
||||||
|
|
||||||
|
const [pivotTableDefinition, setPivotTableDefinition] = useState(null as PivotTableDefinition);
|
||||||
|
|
||||||
|
const [, forceUpdate] = useReducer((x) => x + 1, 0);
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// this is a copy of pivotTableDefinition, that we'll render in the modal. //
|
||||||
|
// then on-save, we'll move it to pivotTableDefinition, e.g., the actual record. //
|
||||||
|
///////////////////////////////////////////////////////////////////////////////////
|
||||||
|
const [modalPivotTableDefinition, setModalPivotTableDefinition] = useState(null as PivotTableDefinition);
|
||||||
|
|
||||||
|
const [usedGroupByFieldNames, setUsedGroupByFieldNames] = useState([] as string[]);
|
||||||
|
const [availableFieldNames, setAvailableFieldNames] = useState([] as string[]);
|
||||||
|
|
||||||
|
const {helpHelpActive} = useContext(QContext);
|
||||||
|
|
||||||
|
//////////////////
|
||||||
|
// initial load //
|
||||||
|
//////////////////
|
||||||
|
useEffect(() =>
|
||||||
|
{
|
||||||
|
if (!pivotTableDefinition)
|
||||||
|
{
|
||||||
|
let originalPivotTableDefinition = recordValues["pivotTableJson"] && JSON.parse(recordValues["pivotTableJson"]) as PivotTableDefinition;
|
||||||
|
if (originalPivotTableDefinition)
|
||||||
|
{
|
||||||
|
setEnabled(true);
|
||||||
|
}
|
||||||
|
else if (!originalPivotTableDefinition)
|
||||||
|
{
|
||||||
|
originalPivotTableDefinition = new PivotTableDefinition();
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let i = 0; i < originalPivotTableDefinition?.rows?.length; i++)
|
||||||
|
{
|
||||||
|
if (!originalPivotTableDefinition?.rows[i].key)
|
||||||
|
{
|
||||||
|
originalPivotTableDefinition.rows[i].key = PivotObjectKey.next();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let i = 0; i < originalPivotTableDefinition?.columns?.length; i++)
|
||||||
|
{
|
||||||
|
if (!originalPivotTableDefinition?.columns[i].key)
|
||||||
|
{
|
||||||
|
originalPivotTableDefinition.columns[i].key = PivotObjectKey.next();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let i = 0; i < originalPivotTableDefinition?.values?.length; i++)
|
||||||
|
{
|
||||||
|
if (!originalPivotTableDefinition?.values[i].key)
|
||||||
|
{
|
||||||
|
originalPivotTableDefinition.values[i].key = PivotObjectKey.next();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setPivotTableDefinition(originalPivotTableDefinition);
|
||||||
|
updateUsedGroupByFieldNames(originalPivotTableDefinition);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (recordValues["columnsJson"])
|
||||||
|
{
|
||||||
|
updateAvailableFieldNames(JSON.parse(recordValues["columnsJson"]) as QQueryColumns);
|
||||||
|
}
|
||||||
|
|
||||||
|
(async () =>
|
||||||
|
{
|
||||||
|
setMetaData(await qController.loadMetaData());
|
||||||
|
})();
|
||||||
|
});
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////
|
||||||
|
// handle the table name changing - load current table's meta-data //
|
||||||
|
/////////////////////////////////////////////////////////////////////
|
||||||
|
useEffect(() =>
|
||||||
|
{
|
||||||
|
if (recordValues["tableName"] && (tableMetaData == null || tableMetaData.name != recordValues["tableName"]))
|
||||||
|
{
|
||||||
|
(async () =>
|
||||||
|
{
|
||||||
|
const tableMetaData = await qController.loadTableMetaData(recordValues["tableName"]);
|
||||||
|
setTableMetaData(tableMetaData);
|
||||||
|
})();
|
||||||
|
}
|
||||||
|
}, [recordValues]);
|
||||||
|
|
||||||
|
|
||||||
|
const helpRoles = isEditable ? [recordValues["id"] ? "EDIT_SCREEN" : "INSERT_SCREEN", "WRITE_SCREENS", "ALL_SCREENS"] : ["VIEW_SCREEN", "READ_SCREENS", "ALL_SCREENS"];
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function showHelp(slot: string)
|
||||||
|
{
|
||||||
|
return (helpHelpActive || hasHelpContent(widgetMetaData?.helpContent?.get(slot), helpRoles));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function getHelpContent(slot: string)
|
||||||
|
{
|
||||||
|
const key = `widget:${widgetMetaData.name};slot:${slot}`;
|
||||||
|
return <HelpContent helpContents={widgetMetaData?.helpContent?.get(slot)} roles={helpRoles} helpContentKey={key} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function toggleEnabled()
|
||||||
|
{
|
||||||
|
const newEnabled = !!!getEnabled();
|
||||||
|
setEnabled(newEnabled);
|
||||||
|
onSaveCallback({usePivotTable: newEnabled});
|
||||||
|
|
||||||
|
if (!newEnabled)
|
||||||
|
{
|
||||||
|
onSaveCallback({pivotTableJson: null});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function getEnabled()
|
||||||
|
{
|
||||||
|
return (enabled);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function addGroupBy(rowsOrColumns: "rows" | "columns")
|
||||||
|
{
|
||||||
|
if (!modalPivotTableDefinition[rowsOrColumns])
|
||||||
|
{
|
||||||
|
modalPivotTableDefinition[rowsOrColumns] = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
modalPivotTableDefinition[rowsOrColumns].push(new PivotTableGroupBy());
|
||||||
|
validateForm()
|
||||||
|
forceUpdate();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function groupByChangedCallback()
|
||||||
|
{
|
||||||
|
updateUsedGroupByFieldNames(modalPivotTableDefinition);
|
||||||
|
validateForm()
|
||||||
|
forceUpdate();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function addValue()
|
||||||
|
{
|
||||||
|
if (!modalPivotTableDefinition.values)
|
||||||
|
{
|
||||||
|
modalPivotTableDefinition.values = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
modalPivotTableDefinition.values.push(new PivotTableValue());
|
||||||
|
validateForm()
|
||||||
|
forceUpdate();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function removeValue(index: number)
|
||||||
|
{
|
||||||
|
modalPivotTableDefinition.values.splice(index, 1);
|
||||||
|
validateForm()
|
||||||
|
forceUpdate();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function updateUsedGroupByFieldNames(ptd: PivotTableDefinition = pivotTableDefinition)
|
||||||
|
{
|
||||||
|
const usedFieldNames: string[] = [];
|
||||||
|
|
||||||
|
for (let i = 0; i < ptd?.rows?.length; i++)
|
||||||
|
{
|
||||||
|
usedFieldNames.push(ptd?.rows[i].fieldName);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let i = 0; i < ptd?.columns?.length; i++)
|
||||||
|
{
|
||||||
|
usedFieldNames.push(ptd?.columns[i].fieldName);
|
||||||
|
}
|
||||||
|
|
||||||
|
setUsedGroupByFieldNames(usedFieldNames);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function updateAvailableFieldNames(columns: QQueryColumns)
|
||||||
|
{
|
||||||
|
const fieldNames: string[] = [];
|
||||||
|
for (let i = 0; i < columns?.columns?.length; i++)
|
||||||
|
{
|
||||||
|
if (columns.columns[i].isVisible)
|
||||||
|
{
|
||||||
|
fieldNames.push(columns.columns[i].name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
setAvailableFieldNames(fieldNames);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function renderOneValue(value: PivotTableValue, index: number)
|
||||||
|
{
|
||||||
|
if (!isEditable)
|
||||||
|
{
|
||||||
|
const selectedField = getSelectedFieldForAutoComplete(tableMetaData, value.fieldName);
|
||||||
|
if (selectedField && value.function)
|
||||||
|
{
|
||||||
|
const label = selectedField.table.name == tableMetaData.name ? selectedField.field.label : selectedField.table.label + ": " + selectedField.field.label;
|
||||||
|
return (<Box mr="0.375rem" mb="0.5rem" border={`1px solid ${colors.grayLines.main}`} borderRadius="0.75rem" p="0.25rem 0.75rem">{pivotTableFunctionLabels[value.function]} of {label}</Box>);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (<React.Fragment />);
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleFieldChange = (event: any, newValue: any, reason: string) =>
|
||||||
|
{
|
||||||
|
value.fieldName = newValue ? newValue.fieldName : null;
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleFunctionChange = (event: any, newValue: any, reason: string) =>
|
||||||
|
{
|
||||||
|
value.function = newValue ? newValue.id : null;
|
||||||
|
};
|
||||||
|
|
||||||
|
const functionOptions: any[] = [];
|
||||||
|
let defaultFunctionValue = null;
|
||||||
|
for (let pivotTableFunctionKey in PivotTableFunction)
|
||||||
|
{
|
||||||
|
// @ts-ignore any?
|
||||||
|
const label = "" + pivotTableFunctionLabels[pivotTableFunctionKey];
|
||||||
|
const option = {id: pivotTableFunctionKey, label: label};
|
||||||
|
functionOptions.push(option);
|
||||||
|
|
||||||
|
if (option.id == value.function)
|
||||||
|
{
|
||||||
|
defaultFunctionValue = option;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// maybe cursor:grab (and then change to "grabbing")
|
||||||
|
return (<Box display="flex" p="0.5rem" pl="0" gap="0.5rem" alignItems="center">
|
||||||
|
<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}
|
||||||
|
defaultValue={getSelectedFieldForAutoComplete(tableMetaData, value.fieldName)}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
<Box width="330px">
|
||||||
|
<Autocomplete
|
||||||
|
id={`values-field-${index}`}
|
||||||
|
renderInput={(params) => (<TextField {...params} label={null} variant="outlined" sx={fieldAutoCompleteTextFieldSX} autoComplete="off" type="search" InputProps={{...params.InputProps}} />)}
|
||||||
|
// @ts-ignore
|
||||||
|
defaultValue={defaultFunctionValue}
|
||||||
|
options={functionOptions}
|
||||||
|
onChange={handleFunctionChange}
|
||||||
|
isOptionEqualToValue={(option, value) => option.id === value.id}
|
||||||
|
getOptionLabel={(option) => option.label}
|
||||||
|
// todo? renderOption={(props, option, state) => renderFieldOption(props, option, state)}
|
||||||
|
autoSelect={true}
|
||||||
|
autoHighlight={true}
|
||||||
|
disableClearable
|
||||||
|
// slotProps={{popper: {className: "filterCriteriaRowColumnPopper", style: {padding: 0, width: "250px"}}}}
|
||||||
|
// {...alsoOpen}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
<Box>
|
||||||
|
<Button sx={xIconButtonSX} onClick={() => removeValue(index)}><Icon>clear</Icon></Button>
|
||||||
|
</Box>
|
||||||
|
</Box>);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** drag & drop callback to move one of the pivot-table group-bys (rows/columns)
|
||||||
|
*******************************************************************************/
|
||||||
|
const moveGroupBy = useCallback((rowsOrColumns: "rows" | "columns", dragIndex: number, hoverIndex: number) =>
|
||||||
|
{
|
||||||
|
const array = modalPivotTableDefinition[rowsOrColumns];
|
||||||
|
const dragItem = array[dragIndex];
|
||||||
|
array.splice(dragIndex, 1);
|
||||||
|
array.splice(hoverIndex, 0, dragItem);
|
||||||
|
|
||||||
|
forceUpdate();
|
||||||
|
}, [modalPivotTableDefinition]);
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** drag & drop callback to move one of the pivot-table values
|
||||||
|
*******************************************************************************/
|
||||||
|
const moveValue = useCallback((dragIndex: number, hoverIndex: number) =>
|
||||||
|
{
|
||||||
|
const array = modalPivotTableDefinition.values;
|
||||||
|
const dragItem = array[dragIndex];
|
||||||
|
array.splice(dragIndex, 1);
|
||||||
|
array.splice(hoverIndex, 0, dragItem);
|
||||||
|
|
||||||
|
forceUpdate();
|
||||||
|
}, [modalPivotTableDefinition]);
|
||||||
|
|
||||||
|
|
||||||
|
const noTable = (tableMetaData == null);
|
||||||
|
const noColumns = (!availableFieldNames || availableFieldNames.length == 0);
|
||||||
|
|
||||||
|
const selectTableFirstTooltipTitle = noTable ? "You must select a table before you can set up your pivot table" : null;
|
||||||
|
const selectColumnsFirstTooltipTitle = noColumns ? "You must set up your report's Columns before you can set up your Pivot Table" : null;
|
||||||
|
const editPopupDisabled = noTable || noColumns;
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////
|
||||||
|
// add toggle component to widget header for editable mode //
|
||||||
|
/////////////////////////////////////////////////////////////
|
||||||
|
const labelAdditionalElementsRight: JSX.Element[] = [];
|
||||||
|
if (isEditable)
|
||||||
|
{
|
||||||
|
labelAdditionalElementsRight.push(<HeaderToggleComponent disabled={editPopupDisabled} disabledTooltip={selectTableFirstTooltipTitle ?? selectColumnsFirstTooltipTitle} label="Use Pivot Table?" getValue={() => enabled} onClickCallback={toggleEnabled} />);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** render a group-by (row or column)
|
||||||
|
*******************************************************************************/
|
||||||
|
const renderGroupBy = useCallback((groupBy: PivotTableGroupBy, rowsOrColumns: "rows" | "columns", index: number, forModal: boolean) =>
|
||||||
|
{
|
||||||
|
return (
|
||||||
|
<PivotTableGroupByElement
|
||||||
|
key={groupBy.fieldName}
|
||||||
|
index={index}
|
||||||
|
id={`${groupBy.key}`}
|
||||||
|
dragCallback={moveGroupBy}
|
||||||
|
metaData={metaData}
|
||||||
|
tableMetaData={tableMetaData}
|
||||||
|
pivotTableDefinition={forModal ? modalPivotTableDefinition : pivotTableDefinition}
|
||||||
|
usedGroupByFieldNames={usedGroupByFieldNames}
|
||||||
|
availableFieldNames={availableFieldNames}
|
||||||
|
isEditable={isEditable && forModal}
|
||||||
|
groupBy={groupBy}
|
||||||
|
rowsOrColumns={rowsOrColumns}
|
||||||
|
callback={groupByChangedCallback}
|
||||||
|
attemptedSubmit={attemptedSubmit}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
[tableMetaData, usedGroupByFieldNames, availableFieldNames],
|
||||||
|
);
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** render a pivot-table value (row or column)
|
||||||
|
*******************************************************************************/
|
||||||
|
const renderValue = useCallback((value: PivotTableValue, index: number, forModal: boolean) =>
|
||||||
|
{
|
||||||
|
return (
|
||||||
|
<PivotTableValueElement
|
||||||
|
key={value.key}
|
||||||
|
index={index}
|
||||||
|
id={`${value.key}`}
|
||||||
|
dragCallback={moveValue}
|
||||||
|
metaData={metaData}
|
||||||
|
tableMetaData={tableMetaData}
|
||||||
|
pivotTableDefinition={forModal ? modalPivotTableDefinition : pivotTableDefinition}
|
||||||
|
availableFieldNames={availableFieldNames}
|
||||||
|
isEditable={isEditable && forModal}
|
||||||
|
value={value}
|
||||||
|
callback={groupByChangedCallback}
|
||||||
|
attemptedSubmit={attemptedSubmit}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
[tableMetaData, usedGroupByFieldNames, availableFieldNames],
|
||||||
|
);
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function openEditor()
|
||||||
|
{
|
||||||
|
if (recordValues["tableName"])
|
||||||
|
{
|
||||||
|
setModalPivotTableDefinition(Object.assign({}, pivotTableDefinition));
|
||||||
|
setModalOpen(true);
|
||||||
|
setAttemptedSubmit(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function closeEditor(event?: {}, reason?: "backdropClick" | "escapeKeyDown")
|
||||||
|
{
|
||||||
|
if (reason == "backdropClick" || reason == "escapeKeyDown")
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setModalOpen(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function renderGroupBys(forModal: boolean, rowsOrColumns: "rows" | "columns")
|
||||||
|
{
|
||||||
|
const ptd = forModal ? modalPivotTableDefinition : pivotTableDefinition;
|
||||||
|
|
||||||
|
return <>
|
||||||
|
<h5>{rowsOrColumns == "rows" ? "Rows" : "Columns"}</h5>
|
||||||
|
<Box fontSize="1rem">
|
||||||
|
{
|
||||||
|
tableMetaData && (<div>{ptd[rowsOrColumns]?.map((groupBy, i) => renderGroupBy(groupBy, rowsOrColumns, i, forModal))}</div>)
|
||||||
|
}
|
||||||
|
</Box>
|
||||||
|
{
|
||||||
|
(forModal || (isEditable && !ptd[rowsOrColumns]?.length)) &&
|
||||||
|
<Box mt={forModal ? "0.5rem" : "0"} mb="1rem">
|
||||||
|
<Tooltip title={selectTableFirstTooltipTitle ?? selectColumnsFirstTooltipTitle}>
|
||||||
|
<span><Button disabled={editPopupDisabled} sx={forModal ? buttonSX : unborderedButtonSX} onClick={() => forModal ? addGroupBy(rowsOrColumns) : openEditor()}>+ Add new {rowsOrColumns == "rows" ? "row" : "column"}</Button></span>
|
||||||
|
</Tooltip>
|
||||||
|
</Box>
|
||||||
|
}
|
||||||
|
{
|
||||||
|
!isEditable && !forModal && !ptd[rowsOrColumns]?.length &&
|
||||||
|
<Box color={colors.gray.main} fontSize="1rem">Your pivot table has no {rowsOrColumns}.</Box>
|
||||||
|
}
|
||||||
|
</>;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function renderValues(forModal: boolean)
|
||||||
|
{
|
||||||
|
const ptd = forModal ? modalPivotTableDefinition : pivotTableDefinition;
|
||||||
|
|
||||||
|
return <>
|
||||||
|
<h5>Values</h5>
|
||||||
|
<Box fontSize="1rem">
|
||||||
|
{
|
||||||
|
tableMetaData && (<div>{ptd?.values?.map((value, i) => renderValue(value, i, forModal))}</div>)
|
||||||
|
}
|
||||||
|
</Box>
|
||||||
|
{
|
||||||
|
(forModal || (isEditable && !ptd?.values?.length)) &&
|
||||||
|
<Box mt={forModal ? "0.5rem" : "0"} mb="1rem">
|
||||||
|
<Tooltip title={selectTableFirstTooltipTitle ?? selectColumnsFirstTooltipTitle}>
|
||||||
|
<span><Button disabled={editPopupDisabled} sx={forModal ? buttonSX : unborderedButtonSX} onClick={() => forModal ? addValue() : openEditor()}>+ Add new value</Button></span>
|
||||||
|
</Tooltip>
|
||||||
|
</Box>
|
||||||
|
}
|
||||||
|
{
|
||||||
|
!isEditable && !forModal && !ptd?.values?.length &&
|
||||||
|
<Box color={colors.gray.main} fontSize="1rem">Your pivot table has no values.</Box>
|
||||||
|
}
|
||||||
|
</>;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function validateForm(submitting: boolean = false)
|
||||||
|
{
|
||||||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// if this isn't a call from the on-submit handler, and we haven't previously attempted a submit, then return w/o setting any alerts //
|
||||||
|
// this is like a version of considering "touched"... //
|
||||||
|
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
if(!submitting && !attemptedSubmit)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let missingValues = 0;
|
||||||
|
|
||||||
|
for (let i = 0; i < modalPivotTableDefinition?.rows?.length; i++)
|
||||||
|
{
|
||||||
|
if (!modalPivotTableDefinition.rows[i].fieldName)
|
||||||
|
{
|
||||||
|
missingValues++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let i = 0; i < modalPivotTableDefinition?.columns?.length; i++)
|
||||||
|
{
|
||||||
|
if (!modalPivotTableDefinition.columns[i].fieldName)
|
||||||
|
{
|
||||||
|
missingValues++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let i = 0; i < modalPivotTableDefinition?.values?.length; i++)
|
||||||
|
{
|
||||||
|
if (!modalPivotTableDefinition.values[i].fieldName)
|
||||||
|
{
|
||||||
|
missingValues++;
|
||||||
|
}
|
||||||
|
if (!modalPivotTableDefinition.values[i].function)
|
||||||
|
{
|
||||||
|
missingValues++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (missingValues == 0)
|
||||||
|
{
|
||||||
|
setErrorAlert(null);
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// this is to catch the case of - user attempted to submit, and there were errors //
|
||||||
|
// now they've fixed 'em - so go back to a 'clean' state - so if they add more //
|
||||||
|
// boxes, they won't immediately show errors, until a re-submit //
|
||||||
|
////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
if(attemptedSubmit)
|
||||||
|
{
|
||||||
|
setAttemptedSubmit(false);
|
||||||
|
}
|
||||||
|
return (false);
|
||||||
|
}
|
||||||
|
|
||||||
|
setErrorAlert(`Missing value in ${missingValues} field${missingValues == 1 ? "" : "s"}.`);
|
||||||
|
return (true);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function saveClicked()
|
||||||
|
{
|
||||||
|
setAttemptedSubmit(true);
|
||||||
|
|
||||||
|
if (validateForm(true))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!onSaveCallback)
|
||||||
|
{
|
||||||
|
console.log("onSaveCallback was not defined");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setPivotTableDefinition(Object.assign({}, modalPivotTableDefinition));
|
||||||
|
updateUsedGroupByFieldNames(modalPivotTableDefinition);
|
||||||
|
|
||||||
|
onSaveCallback({pivotTableJson: JSON.stringify(modalPivotTableDefinition)});
|
||||||
|
|
||||||
|
closeEditor();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
////////////
|
||||||
|
// render //
|
||||||
|
////////////
|
||||||
|
return (<Widget widgetMetaData={widgetMetaData} labelAdditionalElementsRight={labelAdditionalElementsRight}>
|
||||||
|
{
|
||||||
|
<React.Fragment>
|
||||||
|
<DndProvider backend={HTML5Backend}>
|
||||||
|
{
|
||||||
|
enabled &&
|
||||||
|
<Box display="flex" justifyContent="space-between">
|
||||||
|
<Box>
|
||||||
|
{
|
||||||
|
showHelp("sectionSubhead") &&
|
||||||
|
<Box color={colors.gray.main} pb={"0.5rem"} fontSize={"0.875rem"}>
|
||||||
|
{getHelpContent("sectionSubhead")}
|
||||||
|
</Box>
|
||||||
|
}
|
||||||
|
</Box>
|
||||||
|
{
|
||||||
|
isEditable &&
|
||||||
|
<Tooltip title={selectTableFirstTooltipTitle ?? selectColumnsFirstTooltipTitle}>
|
||||||
|
<span>
|
||||||
|
<Button disabled={editPopupDisabled} onClick={() => openEditor()} sx={{p: 0}} disableRipple>
|
||||||
|
<Typography display="inline" textTransform="none" fontSize={"1.125rem"}>
|
||||||
|
Edit Pivot Table
|
||||||
|
</Typography>
|
||||||
|
</Button>
|
||||||
|
</span>
|
||||||
|
</Tooltip>
|
||||||
|
}
|
||||||
|
</Box>
|
||||||
|
}
|
||||||
|
{
|
||||||
|
(!enabled || !pivotTableDefinition) && !isEditable &&
|
||||||
|
<Box fontSize="1rem">Your report does not use a Pivot Table.</Box>
|
||||||
|
}
|
||||||
|
{
|
||||||
|
enabled && pivotTableDefinition &&
|
||||||
|
<>
|
||||||
|
<Grid container spacing="16">
|
||||||
|
|
||||||
|
<Grid item lg={4} md={6} xs={12}>{renderGroupBys(false, "rows")}</Grid>
|
||||||
|
<Grid item lg={4} md={6} xs={12}>{renderGroupBys(false, "columns")}</Grid>
|
||||||
|
<Grid item lg={4} md={6} xs={12}>{renderValues(false)}</Grid>
|
||||||
|
|
||||||
|
</Grid>
|
||||||
|
{
|
||||||
|
modalOpen &&
|
||||||
|
<Modal open={modalOpen} onClose={(event, reason) => closeEditor(event, reason)}>
|
||||||
|
<div>
|
||||||
|
<Box sx={{position: "absolute", width: "100%"}}>
|
||||||
|
<Card sx={{m: "2rem", p: "2rem", overflowY: "auto", height: "calc(100vh - 4rem)"}}>
|
||||||
|
<h3>Edit Pivot Table</h3>
|
||||||
|
{
|
||||||
|
showHelp("modalSubheader") &&
|
||||||
|
<Box color={colors.gray.main}>
|
||||||
|
{getHelpContent("modalSubheader")}
|
||||||
|
</Box>
|
||||||
|
}
|
||||||
|
{
|
||||||
|
errorAlert && <Alert icon={<Icon>error_outline</Icon>} color="error" onClose={() => setErrorAlert(null)}>{errorAlert}</Alert>
|
||||||
|
}
|
||||||
|
<Grid container spacing="16" overflow="auto" mt="0.5rem" mb="1rem" height="100%">
|
||||||
|
|
||||||
|
<Grid item lg={4} md={6} xs={12}>{renderGroupBys(true, "rows")}</Grid>
|
||||||
|
<Grid item lg={4} md={6} xs={12}>{renderGroupBys(true, "columns")}</Grid>
|
||||||
|
<Grid item lg={4} md={6} xs={12}>{renderValues(true)}</Grid>
|
||||||
|
|
||||||
|
</Grid>
|
||||||
|
<Box>
|
||||||
|
<Box display="flex" justifyContent="flex-end">
|
||||||
|
<QCancelButton disabled={false} onClickHandler={closeEditor} />
|
||||||
|
<QSaveButton label="OK" iconName="check" disabled={false} onClickHandler={saveClicked} />
|
||||||
|
</Box>
|
||||||
|
</Box>
|
||||||
|
</Card>
|
||||||
|
</Box>
|
||||||
|
</div>
|
||||||
|
</Modal>
|
||||||
|
}
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
</DndProvider>
|
||||||
|
</React.Fragment>
|
||||||
|
}
|
||||||
|
</Widget>);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* this was a rough-draft of what a preview of a pivot could look like...
|
||||||
|
<Box mt={"1rem"}>
|
||||||
|
<h5>Preview</h5>
|
||||||
|
<table>
|
||||||
|
<tr>
|
||||||
|
<th style={{textAlign: "left", fontSize: "0.875rem"}}></th>
|
||||||
|
<th style={{textAlign: "left", fontSize: "0.875rem"}}>Column Labels</th>
|
||||||
|
</tr>
|
||||||
|
{
|
||||||
|
pivotTableDefinition?.columns?.map((column, i) =>
|
||||||
|
(
|
||||||
|
<tr key={column.key}>
|
||||||
|
<th style={{textAlign: "left", fontSize: "0.875rem"}}></th>
|
||||||
|
<th style={{textAlign: "left", fontSize: "0.875rem"}}>{column.fieldName}</th>
|
||||||
|
</tr>
|
||||||
|
))
|
||||||
|
}
|
||||||
|
<tr>
|
||||||
|
<th style={{textAlign: "left", fontSize: "0.875rem"}}>Row Labels</th>
|
||||||
|
{
|
||||||
|
pivotTableDefinition?.values?.map((value, i) =>
|
||||||
|
(
|
||||||
|
<th key={value.key} style={{textAlign: "left", fontSize: "0.875rem"}}>{value.function} of {value.fieldName}</th>
|
||||||
|
))
|
||||||
|
}
|
||||||
|
</tr>
|
||||||
|
{
|
||||||
|
pivotTableDefinition?.rows?.map((row, i) =>
|
||||||
|
(
|
||||||
|
<tr key={row.key}>
|
||||||
|
<th style={{textAlign: "left", fontSize: "0.875rem", paddingLeft: (i * 1) + "rem"}}>{row.fieldName}</th>
|
||||||
|
</tr>
|
||||||
|
))
|
||||||
|
}
|
||||||
|
</table>
|
||||||
|
</Box>
|
||||||
|
*/
|
316
src/qqq/components/widgets/misc/PivotTableValueElement.tsx
Normal file
316
src/qqq/components/widgets/misc/PivotTableValueElement.tsx
Normal file
@ -0,0 +1,316 @@
|
|||||||
|
/*
|
||||||
|
* 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[];
|
||||||
|
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, 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}
|
||||||
|
defaultValue={selectedField}
|
||||||
|
hasError={showValueError}
|
||||||
|
/>
|
||||||
|
</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>);
|
||||||
|
|
||||||
|
};
|
348
src/qqq/components/widgets/misc/ReportSetupWidget.tsx
Normal file
348
src/qqq/components/widgets/misc/ReportSetupWidget.tsx
Normal file
@ -0,0 +1,348 @@
|
|||||||
|
/*
|
||||||
|
* 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 {QTableMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QTableMetaData";
|
||||||
|
import {QWidgetMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QWidgetMetaData";
|
||||||
|
import {QQueryFilter} from "@kingsrook/qqq-frontend-core/lib/model/query/QQueryFilter";
|
||||||
|
import {Alert, Collapse} from "@mui/material";
|
||||||
|
import Box from "@mui/material/Box";
|
||||||
|
import Button from "@mui/material/Button";
|
||||||
|
import Card from "@mui/material/Card";
|
||||||
|
import Modal from "@mui/material/Modal";
|
||||||
|
import Tooltip from "@mui/material/Tooltip/Tooltip";
|
||||||
|
import QContext from "QContext";
|
||||||
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
|
import {QCancelButton, QSaveButton} from "qqq/components/buttons/DefaultButtons";
|
||||||
|
import HelpContent, {hasHelpContent} from "qqq/components/misc/HelpContent";
|
||||||
|
import AdvancedQueryPreview from "qqq/components/query/AdvancedQueryPreview";
|
||||||
|
import Widget, {HeaderLinkButtonComponent} from "qqq/components/widgets/Widget";
|
||||||
|
import QQueryColumns, {Column} from "qqq/models/query/QQueryColumns";
|
||||||
|
import RecordQuery from "qqq/pages/records/query/RecordQuery";
|
||||||
|
import Client from "qqq/utils/qqq/Client";
|
||||||
|
import FilterUtils from "qqq/utils/qqq/FilterUtils";
|
||||||
|
import React, {useContext, useEffect, useRef, useState} from "react";
|
||||||
|
|
||||||
|
interface ReportSetupWidgetProps
|
||||||
|
{
|
||||||
|
isEditable: boolean;
|
||||||
|
widgetMetaData: QWidgetMetaData;
|
||||||
|
recordValues: {[name: string]: any};
|
||||||
|
onSaveCallback?: (values: {[name: string]: any}) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
ReportSetupWidget.defaultProps = {
|
||||||
|
onSaveCallback: null
|
||||||
|
};
|
||||||
|
|
||||||
|
export const buttonSX =
|
||||||
|
{
|
||||||
|
border: `1px solid ${colors.grayLines.main} !important`,
|
||||||
|
borderRadius: "0.75rem",
|
||||||
|
textTransform: "none",
|
||||||
|
fontSize: "1rem",
|
||||||
|
fontWeight: "400",
|
||||||
|
paddingLeft: "1rem",
|
||||||
|
paddingRight: "1rem",
|
||||||
|
opacity: "1",
|
||||||
|
color: colors.dark.main,
|
||||||
|
"&:hover": {color: colors.dark.main},
|
||||||
|
"&:focus": {color: colors.dark.main},
|
||||||
|
"&:focus:not(:hover)": {color: colors.dark.main},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const unborderedButtonSX = Object.assign({}, buttonSX);
|
||||||
|
unborderedButtonSX.border = "none !important";
|
||||||
|
unborderedButtonSX.opacity = "0.7";
|
||||||
|
|
||||||
|
|
||||||
|
const qController = Client.getInstance();
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Component for editing the main setup of a report - that is: filter & columns
|
||||||
|
*******************************************************************************/
|
||||||
|
export default function ReportSetupWidget({isEditable, widgetMetaData, recordValues, onSaveCallback}: ReportSetupWidgetProps): JSX.Element
|
||||||
|
{
|
||||||
|
const [modalOpen, setModalOpen] = useState(false);
|
||||||
|
const [tableMetaData, setTableMetaData] = useState(null as QTableMetaData);
|
||||||
|
|
||||||
|
const [alertContent, setAlertContent] = useState(null as string);
|
||||||
|
|
||||||
|
const {helpHelpActive} = useContext(QContext);
|
||||||
|
|
||||||
|
const recordQueryRef = useRef();
|
||||||
|
|
||||||
|
|
||||||
|
/////////////////////////////
|
||||||
|
// load values from record //
|
||||||
|
/////////////////////////////
|
||||||
|
let queryFilter = recordValues["queryFilterJson"] && JSON.parse(recordValues["queryFilterJson"]) as QQueryFilter;
|
||||||
|
if(!queryFilter)
|
||||||
|
{
|
||||||
|
queryFilter = new QQueryFilter();
|
||||||
|
}
|
||||||
|
|
||||||
|
let columns = recordValues["columnsJson"] && JSON.parse(recordValues["columnsJson"]) as QQueryColumns;
|
||||||
|
if(!columns)
|
||||||
|
{
|
||||||
|
columns = new QQueryColumns();
|
||||||
|
}
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////
|
||||||
|
// load tableMetaData initially, and if/when selected table changes //
|
||||||
|
//////////////////////////////////////////////////////////////////////
|
||||||
|
useEffect(() =>
|
||||||
|
{
|
||||||
|
if (recordValues["tableName"] && (tableMetaData == null || tableMetaData.name != recordValues["tableName"]))
|
||||||
|
{
|
||||||
|
(async () =>
|
||||||
|
{
|
||||||
|
const tableMetaData = await qController.loadTableMetaData(recordValues["tableName"])
|
||||||
|
setTableMetaData(tableMetaData);
|
||||||
|
})();
|
||||||
|
}
|
||||||
|
}, [recordValues]);
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function openEditor()
|
||||||
|
{
|
||||||
|
if(recordValues["tableName"])
|
||||||
|
{
|
||||||
|
setModalOpen(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function saveClicked()
|
||||||
|
{
|
||||||
|
if(!onSaveCallback)
|
||||||
|
{
|
||||||
|
console.log("onSaveCallback was not defined");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// @ts-ignore possibly 'undefined'.
|
||||||
|
const view = recordQueryRef?.current?.getCurrentView();
|
||||||
|
|
||||||
|
view.queryColumns.sortColumnsFixingPinPositions();
|
||||||
|
|
||||||
|
onSaveCallback({queryFilterJson: JSON.stringify(view.queryFilter), columnsJson: JSON.stringify(view.queryColumns)});
|
||||||
|
|
||||||
|
closeEditor();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function closeEditor(event?: {}, reason?: "backdropClick" | "escapeKeyDown")
|
||||||
|
{
|
||||||
|
if(reason == "backdropClick" || reason == "escapeKeyDown")
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setModalOpen(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function renderColumn(column: Column): JSX.Element
|
||||||
|
{
|
||||||
|
const [field, table] = FilterUtils.getField(tableMetaData, column.name)
|
||||||
|
|
||||||
|
if(!column || !column.isVisible || column.name == "__check__" || !field)
|
||||||
|
{
|
||||||
|
return (<React.Fragment />);
|
||||||
|
}
|
||||||
|
|
||||||
|
const tableLabelPart = table.name != tableMetaData.name ? table.label + ": " : "";
|
||||||
|
|
||||||
|
return (<Box mr="0.375rem" mb="0.5rem" border={`1px solid ${colors.grayLines.main}`} borderRadius="0.75rem" p="0.25rem 0.75rem">
|
||||||
|
{tableLabelPart}{field.label}
|
||||||
|
</Box>);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function mayShowQueryPreview(): boolean
|
||||||
|
{
|
||||||
|
if(tableMetaData)
|
||||||
|
{
|
||||||
|
if(queryFilter?.criteria?.length > 0 || queryFilter?.subFilters?.length > 0)
|
||||||
|
{
|
||||||
|
return (true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function mayShowColumnsPreview(): boolean
|
||||||
|
{
|
||||||
|
if(tableMetaData)
|
||||||
|
{
|
||||||
|
for(let i = 0; i<columns?.columns?.length; i++)
|
||||||
|
{
|
||||||
|
if(columns.columns[i].isVisible && columns.columns[i].name != "__check__")
|
||||||
|
{
|
||||||
|
return (true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (false);
|
||||||
|
}
|
||||||
|
|
||||||
|
const helpRoles = isEditable ? [recordValues["id"] ? "EDIT_SCREEN" : "INSERT_SCREEN", "WRITE_SCREENS", "ALL_SCREENS"] : ["VIEW_SCREEN", "READ_SCREENS", "ALL_SCREENS"];
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function showHelp(slot: string)
|
||||||
|
{
|
||||||
|
return (helpHelpActive || hasHelpContent(widgetMetaData?.helpContent?.get(slot), helpRoles));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
function getHelpContent(slot: string)
|
||||||
|
{
|
||||||
|
const key = `widget:${widgetMetaData.name};slot:${slot}`;
|
||||||
|
return <HelpContent helpContents={widgetMetaData?.helpContent?.get(slot)} roles={helpRoles} helpContentKey={key} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////
|
||||||
|
// add link to widget header for opening modal //
|
||||||
|
/////////////////////////////////////////////////
|
||||||
|
const selectTableFirstTooltipTitle = tableMetaData ? null : "You must select a table before you can set up your report filters and columns";
|
||||||
|
const labelAdditionalElementsRight: JSX.Element[] = []
|
||||||
|
if(isEditable)
|
||||||
|
{
|
||||||
|
labelAdditionalElementsRight.push(<HeaderLinkButtonComponent label="Edit Filters and Columns" onClickCallback={openEditor} disabled={tableMetaData == null} disabledTooltip={selectTableFirstTooltipTitle} />)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
return (<Widget widgetMetaData={widgetMetaData} labelAdditionalElementsRight={labelAdditionalElementsRight}>
|
||||||
|
<React.Fragment>
|
||||||
|
{
|
||||||
|
showHelp("sectionSubhead") &&
|
||||||
|
<Box color={colors.gray.main} pb={"0.5rem"} fontSize={"0.875rem"}>
|
||||||
|
{getHelpContent("sectionSubhead")}
|
||||||
|
</Box>
|
||||||
|
}
|
||||||
|
<Collapse in={Boolean(alertContent)}>
|
||||||
|
<Alert severity="error" sx={{mt: 1.5, mb: 0.5}} onClose={() => setAlertContent(null)}>{alertContent}</Alert>
|
||||||
|
</Collapse>
|
||||||
|
<Box pt="0.5rem">
|
||||||
|
<h5>Query Filter</h5>
|
||||||
|
{
|
||||||
|
mayShowQueryPreview() &&
|
||||||
|
<AdvancedQueryPreview tableMetaData={tableMetaData} queryFilter={queryFilter} isEditable={false} isQueryTooComplex={queryFilter.subFilters?.length > 0} removeCriteriaByIndexCallback={null} />
|
||||||
|
}
|
||||||
|
{
|
||||||
|
!mayShowQueryPreview() &&
|
||||||
|
<Box width="100%" sx={{fontSize: "1rem", background: "#FFFFFF"}} minHeight={"2.5rem"} p={"0.5rem"} pb={"0.125rem"} borderRadius="0.75rem" border={`1px solid ${colors.grayLines.main}`}>
|
||||||
|
{
|
||||||
|
isEditable &&
|
||||||
|
<Tooltip title={selectTableFirstTooltipTitle}>
|
||||||
|
<span><Button disabled={!recordValues["tableName"]} sx={{mb: "0.125rem", ...unborderedButtonSX}} onClick={openEditor}>+ Add Filters</Button></span>
|
||||||
|
</Tooltip>
|
||||||
|
}
|
||||||
|
{
|
||||||
|
!isEditable && <Box color={colors.gray.main}>Your report has no filters.</Box>
|
||||||
|
}
|
||||||
|
</Box>
|
||||||
|
}
|
||||||
|
</Box>
|
||||||
|
<Box pt="1rem">
|
||||||
|
<h5>Columns</h5>
|
||||||
|
<Box display="flex" flexWrap="wrap" fontSize="1rem">
|
||||||
|
{
|
||||||
|
mayShowColumnsPreview() &&
|
||||||
|
columns.columns.map((column, i) => <React.Fragment key={i}>{renderColumn(column)}</React.Fragment>)
|
||||||
|
}
|
||||||
|
{
|
||||||
|
!mayShowColumnsPreview() &&
|
||||||
|
<Box width="100%" sx={{fontSize: "1rem", background: "#FFFFFF"}} minHeight={"2.375rem"} p={"0.5rem"} pb={"0.125rem"}>
|
||||||
|
{
|
||||||
|
isEditable &&
|
||||||
|
<Tooltip title={selectTableFirstTooltipTitle}>
|
||||||
|
<span><Button disabled={!recordValues["tableName"]} sx={unborderedButtonSX} onClick={openEditor}>+ Add Columns</Button></span>
|
||||||
|
</Tooltip>
|
||||||
|
}
|
||||||
|
{
|
||||||
|
!isEditable && <Box color={colors.gray.main}>Your report has no columns.</Box>
|
||||||
|
}
|
||||||
|
</Box>
|
||||||
|
}
|
||||||
|
</Box>
|
||||||
|
</Box>
|
||||||
|
{
|
||||||
|
modalOpen &&
|
||||||
|
<Modal open={modalOpen} onClose={(event, reason) => closeEditor(event, reason)}>
|
||||||
|
<div>
|
||||||
|
<Box sx={{position: "absolute", overflowY: "auto", maxHeight: "100%", width: "100%"}}>
|
||||||
|
<Card sx={{m: "2rem", p: "2rem"}}>
|
||||||
|
<h3>Edit Filters and Columns</h3>
|
||||||
|
{
|
||||||
|
showHelp("modalSubheader") &&
|
||||||
|
<Box color={colors.gray.main} pb={"0.5rem"}>
|
||||||
|
{getHelpContent("modalSubheader")}
|
||||||
|
</Box>
|
||||||
|
}
|
||||||
|
{
|
||||||
|
tableMetaData && <RecordQuery
|
||||||
|
ref={recordQueryRef}
|
||||||
|
table={tableMetaData}
|
||||||
|
usage="reportSetup"
|
||||||
|
isModal={true} />
|
||||||
|
}
|
||||||
|
|
||||||
|
<Box>
|
||||||
|
<Box display="flex" justifyContent="flex-end">
|
||||||
|
<QCancelButton disabled={false} onClickHandler={closeEditor} />
|
||||||
|
<QSaveButton label="OK" iconName="check" disabled={false} onClickHandler={saveClicked} />
|
||||||
|
</Box>
|
||||||
|
</Box>
|
||||||
|
</Card>
|
||||||
|
</Box>
|
||||||
|
</div>
|
||||||
|
</Modal>
|
||||||
|
}
|
||||||
|
</React.Fragment>
|
||||||
|
</Widget>);
|
||||||
|
}
|
50
src/qqq/models/fields/FieldRules.ts
Normal file
50
src/qqq/models/fields/FieldRules.ts
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
/*
|
||||||
|
* 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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
export interface FieldRule
|
||||||
|
{
|
||||||
|
trigger: FieldRuleTrigger;
|
||||||
|
sourceField: string;
|
||||||
|
action: FieldRuleAction;
|
||||||
|
targetField: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
export enum FieldRuleTrigger
|
||||||
|
{
|
||||||
|
ON_CHANGE = "ON_CHANGE"
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
export enum FieldRuleAction
|
||||||
|
{
|
||||||
|
CLEAR_TARGET_FIELD = "CLEAR_TARGET_FIELD"
|
||||||
|
}
|
159
src/qqq/models/misc/PivotTableDefinitionModels.ts
Normal file
159
src/qqq/models/misc/PivotTableDefinitionModels.ts
Normal file
@ -0,0 +1,159 @@
|
|||||||
|
/*
|
||||||
|
* 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 {QFieldType} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QFieldType";
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** put a unique key value in all the pivot table group-by and value objects,
|
||||||
|
** to help react rendering be sane.
|
||||||
|
*******************************************************************************/
|
||||||
|
export class PivotObjectKey
|
||||||
|
{
|
||||||
|
private static value = new Date().getTime();
|
||||||
|
|
||||||
|
static next(): number
|
||||||
|
{
|
||||||
|
return PivotObjectKey.value++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Full definition of pivot table
|
||||||
|
*******************************************************************************/
|
||||||
|
export class PivotTableDefinition
|
||||||
|
{
|
||||||
|
rows: PivotTableGroupBy[];
|
||||||
|
columns: PivotTableGroupBy[];
|
||||||
|
values: PivotTableValue[];
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** A field that the pivot table is grouped by, either as a row or column
|
||||||
|
*******************************************************************************/
|
||||||
|
export class PivotTableGroupBy
|
||||||
|
{
|
||||||
|
fieldName: string;
|
||||||
|
key: number;
|
||||||
|
|
||||||
|
constructor()
|
||||||
|
{
|
||||||
|
this.key = PivotObjectKey.next();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** A field & function that serves as the computed values in the pivot table
|
||||||
|
*******************************************************************************/
|
||||||
|
export class PivotTableValue
|
||||||
|
{
|
||||||
|
fieldName: string;
|
||||||
|
function: PivotTableFunction;
|
||||||
|
|
||||||
|
key: number;
|
||||||
|
|
||||||
|
constructor()
|
||||||
|
{
|
||||||
|
this.key = PivotObjectKey.next();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** Functions that can be applied to pivot table values
|
||||||
|
*******************************************************************************/
|
||||||
|
export enum PivotTableFunction
|
||||||
|
{
|
||||||
|
SUM = "SUM",
|
||||||
|
COUNT = "COUNT",
|
||||||
|
AVERAGE = "AVERAGE",
|
||||||
|
MAX = "MAX",
|
||||||
|
MIN = "MIN",
|
||||||
|
PRODUCT = "PRODUCT",
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
// i don't think we have a useful version of count-nums --unless we allowed //
|
||||||
|
// it on string fields, and counted if they looked like numbers? is that //
|
||||||
|
// what we should do? ... leave here as zombie in case that request comes in //
|
||||||
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
// COUNT_NUMS = "COUNT_NUMS",
|
||||||
|
|
||||||
|
STD_DEV = "STD_DEV",
|
||||||
|
STD_DEVP = "STD_DEVP",
|
||||||
|
VAR = "VAR",
|
||||||
|
VARP = "VARP",
|
||||||
|
}
|
||||||
|
|
||||||
|
const allFunctions = [
|
||||||
|
PivotTableFunction.SUM,
|
||||||
|
PivotTableFunction.COUNT,
|
||||||
|
PivotTableFunction.AVERAGE,
|
||||||
|
PivotTableFunction.MAX,
|
||||||
|
PivotTableFunction.MIN,
|
||||||
|
PivotTableFunction.PRODUCT,
|
||||||
|
// PivotTableFunction.COUNT_NUMS,
|
||||||
|
PivotTableFunction.STD_DEV,
|
||||||
|
PivotTableFunction.STD_DEVP,
|
||||||
|
PivotTableFunction.VAR,
|
||||||
|
PivotTableFunction.VARP
|
||||||
|
];
|
||||||
|
|
||||||
|
const onlyCount = [PivotTableFunction.COUNT];
|
||||||
|
|
||||||
|
const functionsForDates = [PivotTableFunction.COUNT, PivotTableFunction.AVERAGE, PivotTableFunction.MAX, PivotTableFunction.MIN];
|
||||||
|
|
||||||
|
export const functionsPerFieldType: { [type: string]: PivotTableFunction[] } = {};
|
||||||
|
functionsPerFieldType[QFieldType.STRING] = onlyCount;
|
||||||
|
functionsPerFieldType[QFieldType.BOOLEAN] = onlyCount;
|
||||||
|
functionsPerFieldType[QFieldType.BLOB] = onlyCount;
|
||||||
|
functionsPerFieldType[QFieldType.HTML] = onlyCount;
|
||||||
|
functionsPerFieldType[QFieldType.PASSWORD] = onlyCount;
|
||||||
|
functionsPerFieldType[QFieldType.TEXT] = onlyCount;
|
||||||
|
functionsPerFieldType[QFieldType.TIME] = onlyCount;
|
||||||
|
|
||||||
|
functionsPerFieldType[QFieldType.INTEGER] = allFunctions;
|
||||||
|
functionsPerFieldType[QFieldType.DECIMAL] = allFunctions;
|
||||||
|
// functionsPerFieldType[QFieldType.LONG] = allFunctions;
|
||||||
|
|
||||||
|
functionsPerFieldType[QFieldType.DATE] = functionsForDates;
|
||||||
|
functionsPerFieldType[QFieldType.DATE_TIME] = functionsForDates;
|
||||||
|
|
||||||
|
|
||||||
|
//////////////////////////////////////
|
||||||
|
// labels for pivot table functions //
|
||||||
|
//////////////////////////////////////
|
||||||
|
export const pivotTableFunctionLabels =
|
||||||
|
{
|
||||||
|
"SUM": "Sum",
|
||||||
|
"COUNT": "Count",
|
||||||
|
"AVERAGE": "Average",
|
||||||
|
"MAX": "Max",
|
||||||
|
"MIN": "Min",
|
||||||
|
"PRODUCT": "Product",
|
||||||
|
// "COUNT_NUMS": "Count Numbers",
|
||||||
|
"STD_DEV": "StdDev",
|
||||||
|
"STD_DEVP": "StdDevp",
|
||||||
|
"VAR": "Var",
|
||||||
|
"VARP": "Varp"
|
||||||
|
};
|
@ -23,14 +23,13 @@
|
|||||||
import {QFieldMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QFieldMetaData";
|
import {QFieldMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QFieldMetaData";
|
||||||
import {QTableMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QTableMetaData";
|
import {QTableMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QTableMetaData";
|
||||||
import {GridPinnedColumns} from "@mui/x-data-grid-pro";
|
import {GridPinnedColumns} from "@mui/x-data-grid-pro";
|
||||||
import quickSightChart from "qqq/components/widgets/misc/QuickSightChart";
|
|
||||||
import DataGridUtils from "qqq/utils/DataGridUtils";
|
import DataGridUtils from "qqq/utils/DataGridUtils";
|
||||||
import TableUtils from "qqq/utils/qqq/TableUtils";
|
import TableUtils from "qqq/utils/qqq/TableUtils";
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
** member object
|
** member object
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
interface Column
|
export interface Column
|
||||||
{
|
{
|
||||||
name: string;
|
name: string;
|
||||||
isVisible: boolean;
|
isVisible: boolean;
|
||||||
@ -81,11 +80,19 @@ export default class QQueryColumns
|
|||||||
fields.forEach((field) =>
|
fields.forEach((field) =>
|
||||||
{
|
{
|
||||||
const column: Column = {name: field.name, isVisible: true, width: DataGridUtils.getColumnWidthForField(field, table)};
|
const column: Column = {name: field.name, isVisible: true, width: DataGridUtils.getColumnWidthForField(field, table)};
|
||||||
queryColumns.columns.push(column);
|
|
||||||
|
|
||||||
if (field.name == table.primaryKeyField)
|
if (field.name == table.primaryKeyField)
|
||||||
{
|
{
|
||||||
column.pinned = "left";
|
column.pinned = "left";
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////
|
||||||
|
// insert the primary key field after __check__ //
|
||||||
|
//////////////////////////////////////////////////
|
||||||
|
queryColumns.columns.splice(1, 0, column);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
queryColumns.columns.push(column);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -393,6 +400,42 @@ export default class QQueryColumns
|
|||||||
return columnVisibilityModel;
|
return columnVisibilityModel;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** sort the columns list, so that pinned columns go to the front (left) or back
|
||||||
|
** (right) of the list.
|
||||||
|
*******************************************************************************/
|
||||||
|
public sortColumnsFixingPinPositions = (): void =>
|
||||||
|
{
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// do a sort to push pinned-left columns to the start, and pinned-right columns to the end //
|
||||||
|
// and otherwise, leave everything alone //
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
this.columns = this.columns.sort((a: Column, b: Column) =>
|
||||||
|
{
|
||||||
|
if(a.pinned == "left" && b.pinned != "left")
|
||||||
|
{
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
else if(b.pinned == "left" && a.pinned != "left")
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
else if(a.pinned == "right" && b.pinned != "right")
|
||||||
|
{
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
else if(b.pinned == "right" && a.pinned != "right")
|
||||||
|
{
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return (0);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -76,7 +76,7 @@ import ProcessUtils from "qqq/utils/qqq/ProcessUtils";
|
|||||||
import {SavedViewUtils} from "qqq/utils/qqq/SavedViewUtils";
|
import {SavedViewUtils} from "qqq/utils/qqq/SavedViewUtils";
|
||||||
import TableUtils from "qqq/utils/qqq/TableUtils";
|
import TableUtils from "qqq/utils/qqq/TableUtils";
|
||||||
import ValueUtils from "qqq/utils/qqq/ValueUtils";
|
import ValueUtils from "qqq/utils/qqq/ValueUtils";
|
||||||
import React, {forwardRef, useContext, useEffect, useReducer, useRef, useState} from "react";
|
import React, {forwardRef, useContext, useEffect, useImperativeHandle, useReducer, useRef, useState} from "react";
|
||||||
import {useLocation, useNavigate, useSearchParams} from "react-router-dom";
|
import {useLocation, useNavigate, useSearchParams} from "react-router-dom";
|
||||||
|
|
||||||
const CURRENT_SAVED_VIEW_ID_LOCAL_STORAGE_KEY_ROOT = "qqq.currentSavedViewId";
|
const CURRENT_SAVED_VIEW_ID_LOCAL_STORAGE_KEY_ROOT = "qqq.currentSavedViewId";
|
||||||
@ -84,18 +84,16 @@ const DENSITY_LOCAL_STORAGE_KEY_ROOT = "qqq.density";
|
|||||||
const VIEW_LOCAL_STORAGE_KEY_ROOT = "qqq.recordQueryView";
|
const VIEW_LOCAL_STORAGE_KEY_ROOT = "qqq.recordQueryView";
|
||||||
|
|
||||||
export const TABLE_VARIANT_LOCAL_STORAGE_KEY_ROOT = "qqq.tableVariant";
|
export const TABLE_VARIANT_LOCAL_STORAGE_KEY_ROOT = "qqq.tableVariant";
|
||||||
|
export type QueryScreenUsage = "queryScreen" | "reportSetup"
|
||||||
|
|
||||||
interface Props
|
interface Props
|
||||||
{
|
{
|
||||||
table?: QTableMetaData;
|
table?: QTableMetaData;
|
||||||
launchProcess?: QProcessMetaData;
|
launchProcess?: QProcessMetaData;
|
||||||
|
usage?: QueryScreenUsage;
|
||||||
|
isModal?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
RecordQuery.defaultProps = {
|
|
||||||
table: null,
|
|
||||||
launchProcess: null
|
|
||||||
};
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////
|
||||||
// define possible values for our pageState variable //
|
// define possible values for our pageState variable //
|
||||||
///////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////
|
||||||
@ -107,8 +105,13 @@ const qController = Client.getInstance();
|
|||||||
** function to produce standard version of the screen while we're "loading"
|
** function to produce standard version of the screen while we're "loading"
|
||||||
** like the main table meta data etc.
|
** like the main table meta data etc.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
const getLoadingScreen = () =>
|
const getLoadingScreen = (isModal: boolean) =>
|
||||||
{
|
{
|
||||||
|
if(isModal)
|
||||||
|
{
|
||||||
|
return (<Box> </Box>);
|
||||||
|
}
|
||||||
|
|
||||||
return (<BaseLayout>
|
return (<BaseLayout>
|
||||||
|
|
||||||
</BaseLayout>);
|
</BaseLayout>);
|
||||||
@ -120,7 +123,7 @@ const getLoadingScreen = () =>
|
|||||||
**
|
**
|
||||||
** Yuge component. The best. Lots of very smart people are saying so.
|
** Yuge component. The best. Lots of very smart people are saying so.
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
function RecordQuery({table, launchProcess}: Props): JSX.Element
|
const RecordQuery = forwardRef(({table, usage, isModal}: Props, ref) =>
|
||||||
{
|
{
|
||||||
const tableName = table.name;
|
const tableName = table.name;
|
||||||
const [searchParams] = useSearchParams();
|
const [searchParams] = useSearchParams();
|
||||||
@ -136,6 +139,16 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
const [firstRender, setFirstRender] = useState(true);
|
const [firstRender, setFirstRender] = useState(true);
|
||||||
const [isFirstRenderAfterChangingTables, setIsFirstRenderAfterChangingTables] = useState(false);
|
const [isFirstRenderAfterChangingTables, setIsFirstRenderAfterChangingTables] = useState(false);
|
||||||
|
|
||||||
|
useImperativeHandle(ref, () =>
|
||||||
|
{
|
||||||
|
return {
|
||||||
|
getCurrentView(): RecordQueryView
|
||||||
|
{
|
||||||
|
return view;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// manage "state" being passed from some screens (like delete) into query screen - by grabbing, and then deleting //
|
// manage "state" being passed from some screens (like delete) into query screen - by grabbing, and then deleting //
|
||||||
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
@ -688,8 +701,11 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
{
|
{
|
||||||
if (localStorage.getItem(currentSavedViewLocalStorageKey))
|
if (localStorage.getItem(currentSavedViewLocalStorageKey))
|
||||||
{
|
{
|
||||||
currentSavedViewId = Number.parseInt(localStorage.getItem(currentSavedViewLocalStorageKey));
|
if(usage == "queryScreen")
|
||||||
navigate(`${metaData.getTablePathByName(tableName)}/savedView/${currentSavedViewId}`);
|
{
|
||||||
|
currentSavedViewId = Number.parseInt(localStorage.getItem(currentSavedViewLocalStorageKey));
|
||||||
|
navigate(`${metaData.getTablePathByName(tableName)}/savedView/${currentSavedViewId}`);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -943,7 +959,7 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
console.log(`Received error for query ${thisQueryId}`);
|
console.log(`Received error for query ${thisQueryId}`);
|
||||||
console.log(error);
|
console.log(error);
|
||||||
|
|
||||||
var errorMessage;
|
let errorMessage;
|
||||||
if (error && error.message)
|
if (error && error.message)
|
||||||
{
|
{
|
||||||
errorMessage = error.message;
|
errorMessage = error.message;
|
||||||
@ -2183,27 +2199,30 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
<GridToolbarDensitySelector nonce={undefined} />
|
<GridToolbarDensitySelector nonce={undefined} />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div style={{zIndex: 10}}>
|
{
|
||||||
<MenuButton label="Selection" iconName={selectedIds.length == 0 ? "check_box_outline_blank" : "check_box"} disabled={totalRecords == 0} options={selectionMenuOptions} callback={selectionMenuCallback} />
|
usage == "queryScreen" &&
|
||||||
<SelectionSubsetDialog isOpen={selectionSubsetSizePromptOpen} initialValue={selectionSubsetSize} closeHandler={(value) =>
|
<div style={{zIndex: 10}}>
|
||||||
{
|
<MenuButton label="Selection" iconName={selectedIds.length == 0 ? "check_box_outline_blank" : "check_box"} disabled={totalRecords == 0} options={selectionMenuOptions} callback={selectionMenuCallback} />
|
||||||
setSelectionSubsetSizePromptOpen(false);
|
<SelectionSubsetDialog isOpen={selectionSubsetSizePromptOpen} initialValue={selectionSubsetSize} closeHandler={(value) =>
|
||||||
|
|
||||||
if (value !== undefined)
|
|
||||||
{
|
{
|
||||||
if (typeof value === "number" && value > 0)
|
setSelectionSubsetSizePromptOpen(false);
|
||||||
|
|
||||||
|
if (value !== undefined)
|
||||||
{
|
{
|
||||||
programmaticallySelectSomeOrAllRows(value);
|
if (typeof value === "number" && value > 0)
|
||||||
setSelectionSubsetSize(value);
|
{
|
||||||
setSelectFullFilterState("filterSubset");
|
programmaticallySelectSomeOrAllRows(value);
|
||||||
|
setSelectionSubsetSize(value);
|
||||||
|
setSelectFullFilterState("filterSubset");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
setAlertContent("Unexpected value: " + value);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
}} />
|
||||||
{
|
</div>
|
||||||
setAlertContent("Unexpected value: " + value);
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
}} />
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
{
|
{
|
||||||
@ -2301,7 +2320,9 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
|
|
||||||
if (pageState == "ready")
|
if (pageState == "ready")
|
||||||
{
|
{
|
||||||
const newFilterHash = JSON.stringify(prepQueryFilterForBackend(queryFilter));
|
const filterForBackend = prepQueryFilterForBackend(queryFilter);
|
||||||
|
|
||||||
|
const newFilterHash = JSON.stringify(filterForBackend);
|
||||||
if (filterHash != newFilterHash)
|
if (filterHash != newFilterHash)
|
||||||
{
|
{
|
||||||
setFilterHash(newFilterHash);
|
setFilterHash(newFilterHash);
|
||||||
@ -2474,7 +2495,10 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
{
|
{
|
||||||
const currentSavedViewId = Number.parseInt(localStorage.getItem(currentSavedViewLocalStorageKey));
|
const currentSavedViewId = Number.parseInt(localStorage.getItem(currentSavedViewLocalStorageKey));
|
||||||
console.log(`returning to previously active saved view ${currentSavedViewId}`);
|
console.log(`returning to previously active saved view ${currentSavedViewId}`);
|
||||||
navigate(`${metaData.getTablePathByName(tableName)}/savedView/${currentSavedViewId}`);
|
if(usage == "queryScreen")
|
||||||
|
{
|
||||||
|
navigate(`${metaData.getTablePathByName(tableName)}/savedView/${currentSavedViewId}`);
|
||||||
|
}
|
||||||
setViewIdInLocation(currentSavedViewId);
|
setViewIdInLocation(currentSavedViewId);
|
||||||
|
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
@ -2530,7 +2554,7 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
promptForTableVariantSelection();
|
promptForTableVariantSelection();
|
||||||
}
|
}
|
||||||
|
|
||||||
return (getLoadingScreen());
|
return (getLoadingScreen(isModal));
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////
|
||||||
@ -2564,7 +2588,7 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
setRows([]);
|
setRows([]);
|
||||||
setIsFirstRenderAfterChangingTables(true);
|
setIsFirstRenderAfterChangingTables(true);
|
||||||
|
|
||||||
return (getLoadingScreen());
|
return (getLoadingScreen(isModal));
|
||||||
}
|
}
|
||||||
|
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
@ -2608,7 +2632,7 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
if (pageState != "ready")
|
if (pageState != "ready")
|
||||||
{
|
{
|
||||||
console.log(`page state is ${pageState}... no-op while those complete async's run...`);
|
console.log(`page state is ${pageState}... no-op while those complete async's run...`);
|
||||||
return (getLoadingScreen());
|
return (getLoadingScreen(isModal));
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////////////
|
||||||
@ -2617,13 +2641,13 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
///////////////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////////////
|
||||||
if (!tableMetaData)
|
if (!tableMetaData)
|
||||||
{
|
{
|
||||||
return (getLoadingScreen());
|
return (getLoadingScreen(isModal));
|
||||||
}
|
}
|
||||||
|
|
||||||
let savedViewsComponent = null;
|
let savedViewsComponent = null;
|
||||||
if (metaData && metaData.processes.has("querySavedView"))
|
if (metaData && metaData.processes.has("querySavedView"))
|
||||||
{
|
{
|
||||||
savedViewsComponent = (<SavedViews qController={qController} metaData={metaData} tableMetaData={tableMetaData} view={view} viewAsJson={viewAsJson} currentSavedView={currentSavedView} tableDefaultView={tableDefaultView} viewOnChangeCallback={handleSavedViewChange} loadingSavedView={loadingSavedView} />);
|
savedViewsComponent = (<SavedViews qController={qController} metaData={metaData} tableMetaData={tableMetaData} view={view} viewAsJson={viewAsJson} currentSavedView={currentSavedView} tableDefaultView={tableDefaultView} viewOnChangeCallback={handleSavedViewChange} loadingSavedView={loadingSavedView} queryScreenUsage={usage} />);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -2700,7 +2724,7 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
};
|
};
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// these numbers help set the height of the grid (so page won't scroll) based on spcae above & below it //
|
// these numbers help set the height of the grid (so page won't scroll) based on space above & below it //
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
let spaceBelowGrid = 40;
|
let spaceBelowGrid = 40;
|
||||||
let spaceAboveGrid = 205;
|
let spaceAboveGrid = 205;
|
||||||
@ -2714,40 +2738,48 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
spaceAboveGrid += 60;
|
spaceAboveGrid += 60;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(isModal)
|
||||||
|
{
|
||||||
|
spaceAboveGrid += 130;
|
||||||
|
}
|
||||||
|
|
||||||
////////////////////////
|
////////////////////////
|
||||||
// main screen render //
|
// main screen render //
|
||||||
////////////////////////
|
////////////////////////
|
||||||
return (
|
const body = (
|
||||||
<BaseLayout>
|
<React.Fragment>
|
||||||
<Box display="flex" justifyContent="space-between">
|
<Box display="flex" justifyContent="space-between">
|
||||||
<Box>
|
<Box>
|
||||||
<Typography textTransform="capitalize" variant="h3">
|
<Typography textTransform="capitalize" variant="h3">
|
||||||
{pageLoadingState.isLoading() && ""}
|
{pageLoadingState.isLoading() && ""}
|
||||||
{pageLoadingState.isLoadingSlow() && "Loading..."}
|
{pageLoadingState.isLoadingSlow() && "Loading..."}
|
||||||
{pageLoadingState.isNotLoading() && getPageHeader(tableMetaData, visibleJoinTables, tableVariant)}
|
{pageLoadingState.isNotLoading() && !isModal && getPageHeader(tableMetaData, visibleJoinTables, tableVariant)}
|
||||||
</Typography>
|
</Typography>
|
||||||
</Box>
|
</Box>
|
||||||
<Box whiteSpace="nowrap">
|
{
|
||||||
<GotoRecordButton metaData={metaData} tableMetaData={tableMetaData} />
|
!isModal &&
|
||||||
<Box display="inline-block" width="150px">
|
<Box whiteSpace="nowrap">
|
||||||
|
<GotoRecordButton metaData={metaData} tableMetaData={tableMetaData} />
|
||||||
|
<Box display="inline-block" width="150px">
|
||||||
|
{
|
||||||
|
tableMetaData &&
|
||||||
|
<QueryScreenActionMenu
|
||||||
|
metaData={metaData}
|
||||||
|
tableMetaData={tableMetaData}
|
||||||
|
tableProcesses={tableProcesses}
|
||||||
|
bulkLoadClicked={bulkLoadClicked}
|
||||||
|
bulkEditClicked={bulkEditClicked}
|
||||||
|
bulkDeleteClicked={bulkDeleteClicked}
|
||||||
|
processClicked={processClicked}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
</Box>
|
||||||
{
|
{
|
||||||
tableMetaData &&
|
table.capabilities.has(Capability.TABLE_INSERT) && table.insertPermission &&
|
||||||
<QueryScreenActionMenu
|
<QCreateNewButton tablePath={metaData?.getTablePathByName(tableName)} />
|
||||||
metaData={metaData}
|
|
||||||
tableMetaData={tableMetaData}
|
|
||||||
tableProcesses={tableProcesses}
|
|
||||||
bulkLoadClicked={bulkLoadClicked}
|
|
||||||
bulkEditClicked={bulkEditClicked}
|
|
||||||
bulkDeleteClicked={bulkDeleteClicked}
|
|
||||||
processClicked={processClicked}
|
|
||||||
/>
|
|
||||||
}
|
}
|
||||||
</Box>
|
</Box>
|
||||||
{
|
}
|
||||||
table.capabilities.has(Capability.TABLE_INSERT) && table.insertPermission &&
|
|
||||||
<QCreateNewButton tablePath={metaData?.getTablePathByName(tableName)} />
|
|
||||||
}
|
|
||||||
</Box>
|
|
||||||
</Box>
|
</Box>
|
||||||
<div className="recordQuery">
|
<div className="recordQuery">
|
||||||
{/*
|
{/*
|
||||||
@ -2801,6 +2833,7 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
setQuickFilterFieldNames={doSetQuickFilterFieldNames}
|
setQuickFilterFieldNames={doSetQuickFilterFieldNames}
|
||||||
gridApiRef={gridApiRef}
|
gridApiRef={gridApiRef}
|
||||||
mode={mode}
|
mode={mode}
|
||||||
|
queryScreenUsage={usage}
|
||||||
setMode={doSetMode}
|
setMode={doSetMode}
|
||||||
savedViewsComponent={savedViewsComponent}
|
savedViewsComponent={savedViewsComponent}
|
||||||
columnMenuComponent={buildColumnMenu()}
|
columnMenuComponent={buildColumnMenu()}
|
||||||
@ -2841,7 +2874,7 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
sortingMode="server"
|
sortingMode="server"
|
||||||
filterMode="server"
|
filterMode="server"
|
||||||
page={pageNumber}
|
page={pageNumber}
|
||||||
checkboxSelection
|
checkboxSelection={usage == "queryScreen"}
|
||||||
disableSelectionOnClick
|
disableSelectionOnClick
|
||||||
autoHeight={false}
|
autoHeight={false}
|
||||||
rows={rows}
|
rows={rows}
|
||||||
@ -2850,7 +2883,7 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
rowBuffer={10}
|
rowBuffer={10}
|
||||||
rowCount={totalRecords === null || totalRecords === undefined ? 0 : totalRecords}
|
rowCount={totalRecords === null || totalRecords === undefined ? 0 : totalRecords}
|
||||||
onPageSizeChange={handleRowsPerPageChange}
|
onPageSizeChange={handleRowsPerPageChange}
|
||||||
onRowClick={handleRowClick}
|
onRowClick={usage == "queryScreen" ? handleRowClick : null}
|
||||||
onStateChange={handleStateChange}
|
onStateChange={handleStateChange}
|
||||||
density={density}
|
density={density}
|
||||||
loading={loading}
|
loading={loading}
|
||||||
@ -2908,8 +2941,26 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
</Modal>
|
</Modal>
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
</BaseLayout>
|
</React.Fragment>
|
||||||
);
|
);
|
||||||
}
|
|
||||||
|
if(isModal)
|
||||||
|
{
|
||||||
|
return body;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<BaseLayout>{body}</BaseLayout>
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
RecordQuery.defaultProps = {
|
||||||
|
table: null,
|
||||||
|
usage: "queryScreen",
|
||||||
|
launchProcess: null,
|
||||||
|
isModal: false,
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
export default RecordQuery;
|
export default RecordQuery;
|
||||||
|
@ -505,7 +505,7 @@ function RecordView({table, launchProcess}: Props): JSX.Element
|
|||||||
sectionFieldElements.set(section.name,
|
sectionFieldElements.set(section.name,
|
||||||
<Grid id={section.name} key={section.name} item lg={widgetMetaData.gridColumns ? widgetMetaData.gridColumns : 12} xs={12} sx={{display: "flex", alignItems: "stretch", flexGrow: 1, scrollMarginTop: "100px"}}>
|
<Grid id={section.name} key={section.name} item lg={widgetMetaData.gridColumns ? widgetMetaData.gridColumns : 12} xs={12} sx={{display: "flex", alignItems: "stretch", flexGrow: 1, scrollMarginTop: "100px"}}>
|
||||||
<Box width="100%" flexGrow={1} alignItems="stretch">
|
<Box width="100%" flexGrow={1} alignItems="stretch">
|
||||||
<DashboardWidgets key={section.name} tableName={tableMetaData.name} widgetMetaDataList={[widgetMetaData]} entityPrimaryKey={record.values.get(tableMetaData.primaryKeyField)} omitWrappingGridContainer={true} />
|
<DashboardWidgets key={section.name} tableName={tableMetaData.name} widgetMetaDataList={[widgetMetaData]} record={record} entityPrimaryKey={record.values.get(tableMetaData.primaryKeyField)} omitWrappingGridContainer={true} />
|
||||||
</Box>
|
</Box>
|
||||||
</Grid>
|
</Grid>
|
||||||
);
|
);
|
||||||
@ -845,7 +845,7 @@ function RecordView({table, launchProcess}: Props): JSX.Element
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<BaseLayout>
|
<BaseLayout>
|
||||||
<Box>
|
<Box className="recordView">
|
||||||
<Grid container>
|
<Grid container>
|
||||||
<Grid item xs={12}>
|
<Grid item xs={12}>
|
||||||
<Box mb={3}>
|
<Box mb={3}>
|
||||||
|
@ -658,3 +658,9 @@ input[type="search"]::-webkit-search-results-decoration { display: none; }
|
|||||||
{
|
{
|
||||||
border: none;
|
border: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.entityForm h5,
|
||||||
|
.recordView h5
|
||||||
|
{
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
Reference in New Issue
Block a user