mirror of
https://github.com/Kingsrook/qqq-frontend-material-dashboard.git
synced 2025-07-22 07:08:44 +00:00
Compare commits
92 Commits
snapshot-f
...
snapshot-f
Author | SHA1 | Date | |
---|---|---|---|
be393884cc | |||
fc5637b133 | |||
8c7a7ae43e | |||
ca83dbd83b | |||
68f652f3f3 | |||
adb2b4613d | |||
c94f518422 | |||
d293db5136 | |||
2ace79c08a | |||
c31db7ac32 | |||
dd887037c2 | |||
92516a2eb0 | |||
13a918441c | |||
9e1c68b1fd | |||
d5c6985bc4 | |||
b8be374a01 | |||
2ef118a433 | |||
b1d685b5b1 | |||
87edebb79f | |||
c722081ae7 | |||
ca52466b79 | |||
dd45079ecd | |||
aa7f9e93f1 | |||
c899e5712b | |||
2a8bed1093 | |||
627dd3c9f5 | |||
40ac89dac3 | |||
e9223a1c23 | |||
eeb121ff12 | |||
455869c96b | |||
90861b33a4 | |||
6be18627a7 | |||
2255451745 | |||
f9b29e932a | |||
c86bfcff4d | |||
c8fe46c5bf | |||
8de2f2ce33 | |||
036c2253b1 | |||
6c6c1cfe3d | |||
754010df3d | |||
9d7315e773 | |||
12f13983ea | |||
ab0fb977fb | |||
7cb3f2284d | |||
6bdd8ed935 | |||
4f0469a04c | |||
f1c3b93049 | |||
1c1cfc6d75 | |||
87d0c7d478 | |||
fbcee2b819 | |||
c1065099e5 | |||
adcfa86f73 | |||
ad306728c2 | |||
5969f1a6ba | |||
6c3bfa776a | |||
924e657531 | |||
b52d0977cb | |||
ea728d3cf0 | |||
e5430101fa | |||
0879cb4f80 | |||
f1b0618e9d | |||
95f1fa83bb | |||
4b0e12ba47 | |||
6cfb1e04ed | |||
0d763cbfc8 | |||
279840e77a | |||
51b2f5bb5a | |||
02fe351084 | |||
25fa2e82ea | |||
24b4674208 | |||
04630fd154 | |||
1503e2a1d5 | |||
3d7502531d | |||
d0a7db28fe | |||
95244a8aba | |||
45f247785c | |||
9e6d5c10fb | |||
4e0b13ad02 | |||
7f57a11e00 | |||
83da3a3a0a | |||
b59ed8c8c1 | |||
7101420124 | |||
b903e6bef9 | |||
970c9f262c | |||
9313988f9b | |||
123d1742e7 | |||
47fca52437 | |||
44b92690ab | |||
64fe2305ad | |||
91d38a1d15 | |||
60a8baff35 | |||
81b46408b4 |
@ -2,7 +2,7 @@ version: 2.1
|
|||||||
|
|
||||||
orbs:
|
orbs:
|
||||||
node: circleci/node@5.1.0
|
node: circleci/node@5.1.0
|
||||||
browser-tools: circleci/browser-tools@1.4.5
|
browser-tools: circleci/browser-tools@1.4.6
|
||||||
|
|
||||||
executors:
|
executors:
|
||||||
java17:
|
java17:
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
"@auth0/auth0-react": "1.10.2",
|
"@auth0/auth0-react": "1.10.2",
|
||||||
"@emotion/react": "11.7.1",
|
"@emotion/react": "11.7.1",
|
||||||
"@emotion/styled": "11.6.0",
|
"@emotion/styled": "11.6.0",
|
||||||
"@kingsrook/qqq-frontend-core": "1.0.82",
|
"@kingsrook/qqq-frontend-core": "1.0.85",
|
||||||
"@mui/icons-material": "5.4.1",
|
"@mui/icons-material": "5.4.1",
|
||||||
"@mui/material": "5.11.1",
|
"@mui/material": "5.11.1",
|
||||||
"@mui/styles": "5.11.1",
|
"@mui/styles": "5.11.1",
|
||||||
@ -42,6 +42,7 @@
|
|||||||
"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",
|
||||||
|
"react-markdown": "9.0.1",
|
||||||
"react-router-dom": "6.2.1",
|
"react-router-dom": "6.2.1",
|
||||||
"react-router-hash-link": "2.4.3",
|
"react-router-hash-link": "2.4.3",
|
||||||
"react-table": "7.7.0",
|
"react-table": "7.7.0",
|
||||||
|
6
pom.xml
6
pom.xml
@ -29,7 +29,7 @@
|
|||||||
<packaging>jar</packaging>
|
<packaging>jar</packaging>
|
||||||
|
|
||||||
<properties>
|
<properties>
|
||||||
<revision>0.19.0-SNAPSHOT</revision>
|
<revision>0.20.0-SNAPSHOT</revision>
|
||||||
|
|
||||||
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
|
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
|
||||||
<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
|
<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
|
||||||
@ -77,13 +77,13 @@
|
|||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.seleniumhq.selenium</groupId>
|
<groupId>org.seleniumhq.selenium</groupId>
|
||||||
<artifactId>selenium-java</artifactId>
|
<artifactId>selenium-java</artifactId>
|
||||||
<version>4.10.0</version>
|
<version>4.15.0</version>
|
||||||
<scope>test</scope>
|
<scope>test</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>io.github.bonigarcia</groupId>
|
<groupId>io.github.bonigarcia</groupId>
|
||||||
<artifactId>webdrivermanager</artifactId>
|
<artifactId>webdrivermanager</artifactId>
|
||||||
<version>5.4.1</version>
|
<version>5.6.2</version>
|
||||||
<scope>test</scope>
|
<scope>test</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
|
@ -36,7 +36,7 @@ import {LicenseInfo} from "@mui/x-license-pro";
|
|||||||
import jwt_decode from "jwt-decode";
|
import jwt_decode from "jwt-decode";
|
||||||
import React, {JSXElementConstructor, Key, ReactElement, useEffect, useState,} from "react";
|
import React, {JSXElementConstructor, Key, ReactElement, useEffect, useState,} from "react";
|
||||||
import {useCookies} from "react-cookie";
|
import {useCookies} from "react-cookie";
|
||||||
import {Navigate, Route, Routes, useLocation,} from "react-router-dom";
|
import {Navigate, Route, Routes, useLocation, useSearchParams,} from "react-router-dom";
|
||||||
import {Md5} from "ts-md5/dist/md5";
|
import {Md5} from "ts-md5/dist/md5";
|
||||||
import CommandMenu from "CommandMenu";
|
import CommandMenu from "CommandMenu";
|
||||||
import QContext from "QContext";
|
import QContext from "QContext";
|
||||||
@ -226,6 +226,7 @@ export default function App()
|
|||||||
const {miniSidenav, direction, layout, openConfigurator, sidenavColor} = controller;
|
const {miniSidenav, direction, layout, openConfigurator, sidenavColor} = controller;
|
||||||
const [onMouseEnter, setOnMouseEnter] = useState(false);
|
const [onMouseEnter, setOnMouseEnter] = useState(false);
|
||||||
const {pathname} = useLocation();
|
const {pathname} = useLocation();
|
||||||
|
const [queryParams] = useSearchParams();
|
||||||
|
|
||||||
const [needToLoadRoutes, setNeedToLoadRoutes] = useState(true);
|
const [needToLoadRoutes, setNeedToLoadRoutes] = useState(true);
|
||||||
const [sideNavRoutes, setSideNavRoutes] = useState([]);
|
const [sideNavRoutes, setSideNavRoutes] = useState([]);
|
||||||
@ -517,7 +518,7 @@ export default function App()
|
|||||||
name: loggedInUser?.name ?? "Anonymous",
|
name: loggedInUser?.name ?? "Anonymous",
|
||||||
key: "username",
|
key: "username",
|
||||||
noCollapse: true,
|
noCollapse: true,
|
||||||
icon: <Avatar src={profilePicture} alt="{user?.name}" />,
|
icon: <Avatar src={profilePicture} alt="{loggedInUser?.name}" />,
|
||||||
};
|
};
|
||||||
setProfileRoutes(profileRoutes);
|
setProfileRoutes(profileRoutes);
|
||||||
|
|
||||||
@ -659,6 +660,8 @@ export default function App()
|
|||||||
const [tableProcesses, setTableProcesses] = useState(null);
|
const [tableProcesses, setTableProcesses] = useState(null);
|
||||||
const [dotMenuOpen, setDotMenuOpen] = useState(false);
|
const [dotMenuOpen, setDotMenuOpen] = useState(false);
|
||||||
const [keyboardHelpOpen, setKeyboardHelpOpen] = useState(false);
|
const [keyboardHelpOpen, setKeyboardHelpOpen] = useState(false);
|
||||||
|
const [helpHelpActive] = useState(queryParams.has("helpHelp"));
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|
||||||
appRoutes && (
|
appRoutes && (
|
||||||
@ -669,6 +672,7 @@ export default function App()
|
|||||||
tableProcesses: tableProcesses,
|
tableProcesses: tableProcesses,
|
||||||
dotMenuOpen: dotMenuOpen,
|
dotMenuOpen: dotMenuOpen,
|
||||||
keyboardHelpOpen: keyboardHelpOpen,
|
keyboardHelpOpen: keyboardHelpOpen,
|
||||||
|
helpHelpActive: helpHelpActive,
|
||||||
setPageHeader: (header: string | JSX.Element) => setPageHeader(header),
|
setPageHeader: (header: string | JSX.Element) => setPageHeader(header),
|
||||||
setAccentColor: (accentColor: string) => setAccentColor(accentColor),
|
setAccentColor: (accentColor: string) => setAccentColor(accentColor),
|
||||||
setTableMetaData: (tableMetaData: QTableMetaData) => setTableMetaData(tableMetaData),
|
setTableMetaData: (tableMetaData: QTableMetaData) => setTableMetaData(tableMetaData),
|
||||||
|
@ -51,6 +51,7 @@ interface QContext
|
|||||||
///////////////////////////////////
|
///////////////////////////////////
|
||||||
pathToLabelMap?: {[path: string]: string};
|
pathToLabelMap?: {[path: string]: string};
|
||||||
branding?: QBrandingMetaData;
|
branding?: QBrandingMetaData;
|
||||||
|
helpHelpActive?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
const defaultState = {
|
const defaultState = {
|
||||||
@ -59,6 +60,7 @@ const defaultState = {
|
|||||||
dotMenuOpen: false,
|
dotMenuOpen: false,
|
||||||
keyboardHelpOpen: false,
|
keyboardHelpOpen: false,
|
||||||
pathToLabelMap: {},
|
pathToLabelMap: {},
|
||||||
|
helpHelpActive: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
const QContext = createContext<QContext>(defaultState);
|
const QContext = createContext<QContext>(defaultState);
|
||||||
|
@ -0,0 +1,31 @@
|
|||||||
|
/*
|
||||||
|
* QQQ - Low-code Application Framework for Engineers.
|
||||||
|
* Copyright (C) 2021-2023. 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;
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
public interface MaterialDashboardIconRoleNames
|
||||||
|
{
|
||||||
|
String TOP_RIGHT_INSIDE_CARD = "topRightInsideCard";
|
||||||
|
}
|
@ -149,7 +149,7 @@ interface Types {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const baseProperties = {
|
const baseProperties = {
|
||||||
fontFamily: '"Roboto", "Helvetica", "Arial", sans-serif',
|
fontFamily: '"SF Pro Display", "Roboto", "Helvetica", "Arial", sans-serif',
|
||||||
fontWeightLighter: 100,
|
fontWeightLighter: 100,
|
||||||
fontWeightLight: 300,
|
fontWeightLight: 300,
|
||||||
fontWeightRegular: 400,
|
fontWeightRegular: 400,
|
||||||
|
@ -78,6 +78,19 @@ interface Types
|
|||||||
light: string;
|
light: string;
|
||||||
main: string;
|
main: string;
|
||||||
focus: string;
|
focus: string;
|
||||||
|
}
|
||||||
|
blueGray:
|
||||||
|
| {
|
||||||
|
main: string;
|
||||||
|
}
|
||||||
|
gray:
|
||||||
|
| {
|
||||||
|
main: string;
|
||||||
|
focus: string;
|
||||||
|
}
|
||||||
|
grayLines:
|
||||||
|
| {
|
||||||
|
main: string;
|
||||||
}
|
}
|
||||||
| any;
|
| any;
|
||||||
primary: ColorsTypes | any;
|
primary: ColorsTypes | any;
|
||||||
@ -174,6 +187,19 @@ const colors: Types = {
|
|||||||
focus: "#ffffff",
|
focus: "#ffffff",
|
||||||
},
|
},
|
||||||
|
|
||||||
|
blueGray: {
|
||||||
|
main: "#546E7A"
|
||||||
|
},
|
||||||
|
|
||||||
|
gray: {
|
||||||
|
main: "#757575",
|
||||||
|
focus: "#757575",
|
||||||
|
},
|
||||||
|
|
||||||
|
grayLines: {
|
||||||
|
main: "#D6D6D6"
|
||||||
|
},
|
||||||
|
|
||||||
black: {
|
black: {
|
||||||
light: "#000000",
|
light: "#000000",
|
||||||
main: "#000000",
|
main: "#000000",
|
||||||
@ -216,7 +242,7 @@ const colors: Types = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
dark: {
|
dark: {
|
||||||
main: "#344767",
|
main: "#212121",
|
||||||
focus: "#2c3c58",
|
focus: "#2c3c58",
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -149,7 +149,7 @@ interface Types {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const baseProperties = {
|
const baseProperties = {
|
||||||
fontFamily: '"Roboto", "Helvetica", "Arial", sans-serif',
|
fontFamily: '"SF Pro Display", "Roboto", "Helvetica", "Arial", sans-serif',
|
||||||
fontWeightLighter: 100,
|
fontWeightLighter: 100,
|
||||||
fontWeightLight: 300,
|
fontWeightLight: 300,
|
||||||
fontWeightRegular: 400,
|
fontWeightRegular: 400,
|
||||||
@ -199,9 +199,10 @@ const typography: Types = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
h3: {
|
h3: {
|
||||||
fontSize: pxToRem(30),
|
fontSize: "1.75rem",
|
||||||
lineHeight: 1.375,
|
lineHeight: 1.375,
|
||||||
...baseHeadingProperties,
|
...baseHeadingProperties,
|
||||||
|
fontWeight: 600
|
||||||
},
|
},
|
||||||
|
|
||||||
h4: {
|
h4: {
|
||||||
@ -217,9 +218,10 @@ const typography: Types = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
h6: {
|
h6: {
|
||||||
fontSize: pxToRem(16),
|
fontSize: "1.125rem",
|
||||||
lineHeight: 1.625,
|
lineHeight: 1.625,
|
||||||
...baseHeadingProperties,
|
...baseHeadingProperties,
|
||||||
|
fontWeight: 500
|
||||||
},
|
},
|
||||||
|
|
||||||
subtitle1: {
|
subtitle1: {
|
||||||
|
@ -31,7 +31,7 @@ type Types = any;
|
|||||||
|
|
||||||
const card: Types = {
|
const card: Types = {
|
||||||
defaultProps: {
|
defaultProps: {
|
||||||
elevation: 3
|
elevation: 0
|
||||||
},
|
},
|
||||||
styleOverrides: {
|
styleOverrides: {
|
||||||
root: {
|
root: {
|
||||||
@ -42,7 +42,7 @@ const card: Types = {
|
|||||||
wordWrap: "break-word",
|
wordWrap: "break-word",
|
||||||
backgroundColor: white.main,
|
backgroundColor: white.main,
|
||||||
backgroundClip: "border-box",
|
backgroundClip: "border-box",
|
||||||
border: `${borderWidth[0]} solid ${rgba(black.main, 0.125)}`,
|
border: `${borderWidth[1]} solid ${colors.grayLines.main}`,
|
||||||
borderRadius: borderRadius.xl,
|
borderRadius: borderRadius.xl,
|
||||||
overflow: "visible",
|
overflow: "visible",
|
||||||
},
|
},
|
||||||
|
@ -15,15 +15,11 @@ Coded by www.creative-tim.com
|
|||||||
|
|
||||||
// Material Dashboard 2 PRO React TS Base Styles
|
// Material Dashboard 2 PRO React TS Base Styles
|
||||||
import colors from "qqq/assets/theme/base/colors";
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
import borders from "qqq/assets/theme/base/borders";
|
|
||||||
import boxShadows from "qqq/assets/theme/base/boxShadows";
|
|
||||||
|
|
||||||
// Material Dashboard 2 PRO React TS Helper Functions
|
// Material Dashboard 2 PRO React TS Helper Functions
|
||||||
import pxToRem from "qqq/assets/theme/functions/pxToRem";
|
import pxToRem from "qqq/assets/theme/functions/pxToRem";
|
||||||
|
|
||||||
const {grey, white} = colors;
|
const {grey, white} = colors;
|
||||||
const { borderRadius } = borders;
|
|
||||||
const { tabsBoxShadow } = boxShadows;
|
|
||||||
|
|
||||||
// types
|
// types
|
||||||
type Types = any;
|
type Types = any;
|
||||||
@ -32,15 +28,25 @@ const tabs: Types = {
|
|||||||
styleOverrides: {
|
styleOverrides: {
|
||||||
root: {
|
root: {
|
||||||
position: "relative",
|
position: "relative",
|
||||||
backgroundColor: grey[100],
|
borderRadius: 0,
|
||||||
borderRadius: borderRadius.xl,
|
borderBottom: "1px solid",
|
||||||
|
borderBottomColor: grey[400],
|
||||||
minHeight: "unset",
|
minHeight: "unset",
|
||||||
padding: pxToRem(4),
|
padding: "0",
|
||||||
|
margin: "0",
|
||||||
|
"& button": {
|
||||||
|
fontWeight: 500
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
scroller: {
|
||||||
|
marginLeft: "0.5rem"
|
||||||
},
|
},
|
||||||
|
|
||||||
flexContainer: {
|
flexContainer: {
|
||||||
height: "100%",
|
height: "100%",
|
||||||
position: "relative",
|
position: "relative",
|
||||||
|
width: "fit-content",
|
||||||
zIndex: 10,
|
zIndex: 10,
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -57,9 +63,10 @@ const tabs: Types = {
|
|||||||
|
|
||||||
indicator: {
|
indicator: {
|
||||||
height: "100%",
|
height: "100%",
|
||||||
borderRadius: borderRadius.lg,
|
borderRadius: 0,
|
||||||
backgroundColor: white.main,
|
backgroundColor: white.main,
|
||||||
boxShadow: tabsBoxShadow.indicator,
|
borderBottom: "2px solid",
|
||||||
|
borderBottomColor: colors.info.main,
|
||||||
transition: "all 500ms ease",
|
transition: "all 500ms ease",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
@ -43,8 +43,10 @@ const tab: Types = {
|
|||||||
fontWeight: fontWeightRegular,
|
fontWeight: fontWeightRegular,
|
||||||
textTransform: "none",
|
textTransform: "none",
|
||||||
lineHeight: "inherit",
|
lineHeight: "inherit",
|
||||||
padding: pxToRem(4),
|
padding: "0.75rem 0.5rem 0.5rem",
|
||||||
borderRadius: borderRadius.lg,
|
margin: "0 0.5rem",
|
||||||
|
borderRadius: 0,
|
||||||
|
border: 0,
|
||||||
color: `${dark.main} !important`,
|
color: `${dark.main} !important`,
|
||||||
opacity: "1 !important",
|
opacity: "1 !important",
|
||||||
|
|
||||||
|
@ -24,7 +24,7 @@ import borders from "qqq/assets/theme/base/borders";
|
|||||||
// Material Dashboard 2 PRO React TS Helper Functions
|
// Material Dashboard 2 PRO React TS Helper Functions
|
||||||
import pxToRem from "qqq/assets/theme/functions/pxToRem";
|
import pxToRem from "qqq/assets/theme/functions/pxToRem";
|
||||||
|
|
||||||
const { black, light } = colors;
|
const { black, light, white, dark } = colors;
|
||||||
const { size, fontWeightRegular } = typography;
|
const { size, fontWeightRegular } = typography;
|
||||||
const { borderRadius } = borders;
|
const { borderRadius } = borders;
|
||||||
|
|
||||||
@ -39,19 +39,20 @@ const tooltip: Types = {
|
|||||||
|
|
||||||
styleOverrides: {
|
styleOverrides: {
|
||||||
tooltip: {
|
tooltip: {
|
||||||
maxWidth: pxToRem(200),
|
maxWidth: pxToRem(300),
|
||||||
backgroundColor: black.main,
|
backgroundColor: white.main,
|
||||||
color: light.main,
|
color: dark.main,
|
||||||
fontSize: size.sm,
|
fontSize: size.sm,
|
||||||
fontWeight: fontWeightRegular,
|
fontWeight: fontWeightRegular,
|
||||||
textAlign: "center",
|
textAlign: "left",
|
||||||
borderRadius: borderRadius.md,
|
borderRadius: borderRadius.md,
|
||||||
opacity: 0.7,
|
opacity: 0.7,
|
||||||
padding: `${pxToRem(5)} ${pxToRem(8)} ${pxToRem(4)}`,
|
padding: "1rem",
|
||||||
|
boxShadow: "0px 0px 12px rgba(128, 128, 128, 0.40)"
|
||||||
},
|
},
|
||||||
|
|
||||||
arrow: {
|
arrow: {
|
||||||
color: black.main,
|
color: white.main,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
@ -402,7 +402,8 @@ function AuditBody({tableMetaData, recordId, record}: Props): JSX.Element
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<Box key={audit0.values.get("id")} className="auditGroupBlock">
|
<Box key={audit0.values.get("id")} className="auditGroupBlock">
|
||||||
<Box display="flex" flexDirection="row" justifyContent="center" fontSize={14}>
|
<Box position="sticky" top="0" zIndex={3}>
|
||||||
|
<Box display="flex" flexDirection="row" justifyContent="center" fontSize={14} position={"relative"} top={"-1px"} pb={"6px"} sx={{backgroundImage: "linear-gradient(to bottom, rgba(255,255,255,1), rgba(255,255,255,1) 80%, rgba(255,255,255,0))"}}>
|
||||||
<Box borderTop={1} mt={1.25} mr={1} width="100%" borderColor="#B0B0B0" />
|
<Box borderTop={1} mt={1.25} mr={1} width="100%" borderColor="#B0B0B0" />
|
||||||
<Box whiteSpace="nowrap">
|
<Box whiteSpace="nowrap">
|
||||||
{ValueUtils.getFullWeekday(audit0.values.get("timestamp"))} {timestampParts[0]}
|
{ValueUtils.getFullWeekday(audit0.values.get("timestamp"))} {timestampParts[0]}
|
||||||
@ -411,6 +412,7 @@ function AuditBody({tableMetaData, recordId, record}: Props): JSX.Element
|
|||||||
</Box>
|
</Box>
|
||||||
<Box borderTop={1} mt={1.25} ml={1} width="100%" borderColor="#B0B0B0" />
|
<Box borderTop={1} mt={1.25} ml={1} width="100%" borderColor="#B0B0B0" />
|
||||||
</Box>
|
</Box>
|
||||||
|
</Box>
|
||||||
|
|
||||||
{
|
{
|
||||||
audits.map((audit) =>
|
audits.map((audit) =>
|
||||||
|
@ -19,17 +19,18 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {InputLabel} from "@mui/material";
|
import {Box, InputLabel} from "@mui/material";
|
||||||
import Stack from "@mui/material/Stack";
|
import Stack from "@mui/material/Stack";
|
||||||
import {styled} from "@mui/material/styles";
|
import {styled} from "@mui/material/styles";
|
||||||
import Switch from "@mui/material/Switch";
|
import Switch from "@mui/material/Switch";
|
||||||
import Typography from "@mui/material/Typography";
|
import Typography from "@mui/material/Typography";
|
||||||
import {useFormikContext} from "formik";
|
import {useFormikContext} from "formik";
|
||||||
import React, {SyntheticEvent} from "react";
|
import React, {SyntheticEvent} from "react";
|
||||||
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
|
|
||||||
const AntSwitch = styled(Switch)(({theme}) => ({
|
const AntSwitch = styled(Switch)(({theme}) => ({
|
||||||
width: 28,
|
width: 32,
|
||||||
height: 16,
|
height: 20,
|
||||||
padding: 0,
|
padding: 0,
|
||||||
display: "flex",
|
display: "flex",
|
||||||
"&:active": {
|
"&:active": {
|
||||||
@ -53,15 +54,19 @@ const AntSwitch = styled(Switch)(({theme}) => ({
|
|||||||
},
|
},
|
||||||
"& .MuiSwitch-thumb": {
|
"& .MuiSwitch-thumb": {
|
||||||
boxShadow: "0 2px 4px 0 rgb(0 35 11 / 20%)",
|
boxShadow: "0 2px 4px 0 rgb(0 35 11 / 20%)",
|
||||||
width: 12,
|
width: 16,
|
||||||
height: 12,
|
height: 16,
|
||||||
borderRadius: 6,
|
borderRadius: 8,
|
||||||
transition: theme.transitions.create([ "width" ], {
|
transition: theme.transitions.create([ "width" ], {
|
||||||
duration: 200,
|
duration: 200,
|
||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
|
"&.nullSwitch .MuiSwitch-thumb": {
|
||||||
|
width: 28,
|
||||||
|
},
|
||||||
"& .MuiSwitch-track": {
|
"& .MuiSwitch-track": {
|
||||||
borderRadius: 16 / 2,
|
height: 20,
|
||||||
|
borderRadius: 20 / 2,
|
||||||
opacity: 1,
|
opacity: 1,
|
||||||
backgroundColor:
|
backgroundColor:
|
||||||
theme.palette.mode === "dark" ? "rgba(255,255,255,.35)" : "rgba(0,0,0,.25)",
|
theme.palette.mode === "dark" ? "rgba(255,255,255,.35)" : "rgba(0,0,0,.25)",
|
||||||
@ -78,6 +83,7 @@ interface Props
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
function BooleanFieldSwitch({name, label, value, isDisabled}: Props) : JSX.Element
|
function BooleanFieldSwitch({name, label, value, isDisabled}: Props) : JSX.Element
|
||||||
{
|
{
|
||||||
const {setFieldValue} = useFormikContext();
|
const {setFieldValue} = useFormikContext();
|
||||||
@ -96,27 +102,29 @@ function BooleanFieldSwitch({name, label, value, isDisabled}: Props) : JSX.Eleme
|
|||||||
setFieldValue(name, !value);
|
setFieldValue(name, !value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const classNullSwitch = (value === null || value == undefined || `${value}` == "") ? "nullSwitch" : "";
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<Box bgcolor={isDisabled ? colors.grey[200] : ""}>
|
||||||
<InputLabel shrink={true}>{label}</InputLabel>
|
<InputLabel shrink={true}>{label}</InputLabel>
|
||||||
<Stack direction="row" spacing={1} alignItems="center">
|
<Stack direction="row" spacing={1} alignItems="center" height="37px">
|
||||||
<Typography
|
<Typography
|
||||||
fontSize="0.875rem"
|
fontSize="1rem"
|
||||||
color={value === false ? "auto" : "#bfbfbf" }
|
color={value === false ? "auto" : "#bfbfbf" }
|
||||||
onClick={(e) => setSwitch(e, false)}
|
onClick={(e) => setSwitch(e, false)}
|
||||||
sx={{cursor: value === false || isDisabled ? "inherit" : "pointer"}}>
|
sx={{cursor: value === false || isDisabled ? "inherit" : "pointer"}}>
|
||||||
No
|
No
|
||||||
</Typography>
|
</Typography>
|
||||||
<AntSwitch name={name} checked={value} onClick={toggleSwitch} disabled={isDisabled} />
|
<AntSwitch className={classNullSwitch} name={name} checked={value} onClick={toggleSwitch} disabled={isDisabled} />
|
||||||
<Typography
|
<Typography
|
||||||
fontSize="0.875rem"
|
fontSize="1rem"
|
||||||
color={value === true ? "auto" : "#bfbfbf"}
|
color={value === true ? "auto" : "#bfbfbf"}
|
||||||
onClick={(e) => setSwitch(e, true)}
|
onClick={(e) => setSwitch(e, true)}
|
||||||
sx={{cursor: value === true || isDisabled ? "inherit" : "pointer"}}>
|
sx={{cursor: value === true || isDisabled ? "inherit" : "pointer"}}>
|
||||||
Yes
|
Yes
|
||||||
</Typography>
|
</Typography>
|
||||||
</Stack>
|
</Stack>
|
||||||
</>
|
</Box>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -32,6 +32,7 @@ import React, {useState} from "react";
|
|||||||
import QDynamicFormField from "qqq/components/forms/DynamicFormField";
|
import QDynamicFormField from "qqq/components/forms/DynamicFormField";
|
||||||
import DynamicSelect from "qqq/components/forms/DynamicSelect";
|
import DynamicSelect from "qqq/components/forms/DynamicSelect";
|
||||||
import MDTypography from "qqq/components/legacy/MDTypography";
|
import MDTypography from "qqq/components/legacy/MDTypography";
|
||||||
|
import HelpContent from "qqq/components/misc/HelpContent";
|
||||||
import ValueUtils from "qqq/utils/qqq/ValueUtils";
|
import ValueUtils from "qqq/utils/qqq/ValueUtils";
|
||||||
|
|
||||||
interface Props
|
interface Props
|
||||||
@ -41,16 +42,13 @@ interface Props
|
|||||||
bulkEditMode?: boolean;
|
bulkEditMode?: boolean;
|
||||||
bulkEditSwitchChangeHandler?: any;
|
bulkEditSwitchChangeHandler?: any;
|
||||||
record?: QRecord;
|
record?: QRecord;
|
||||||
|
helpRoles?: string[];
|
||||||
|
helpContentKeyPrefix?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
function QDynamicForm(props: Props): JSX.Element
|
function QDynamicForm({formData, formLabel, bulkEditMode, bulkEditSwitchChangeHandler, record, helpRoles, helpContentKeyPrefix}: Props): JSX.Element
|
||||||
{
|
{
|
||||||
const {
|
const {formFields, values, errors, touched} = formData;
|
||||||
formData, formLabel, bulkEditMode, bulkEditSwitchChangeHandler,
|
|
||||||
} = props;
|
|
||||||
const {
|
|
||||||
formFields, values, errors, touched,
|
|
||||||
} = formData;
|
|
||||||
|
|
||||||
const formikProps = useFormikContext();
|
const formikProps = useFormikContext();
|
||||||
const [fileName, setFileName] = useState(null as string);
|
const [fileName, setFileName] = useState(null as string);
|
||||||
@ -70,8 +68,8 @@ function QDynamicForm(props: Props): JSX.Element
|
|||||||
{
|
{
|
||||||
setFileName(null);
|
setFileName(null);
|
||||||
formikProps.setFieldValue(fieldName, null);
|
formikProps.setFieldValue(fieldName, null);
|
||||||
props.record?.values.delete(fieldName)
|
record?.values.delete(fieldName)
|
||||||
props.record?.displayValues.delete(fieldName)
|
record?.displayValues.delete(fieldName)
|
||||||
};
|
};
|
||||||
|
|
||||||
const bulkEditSwitchChanged = (name: string, value: boolean) =>
|
const bulkEditSwitchChanged = (name: string, value: boolean) =>
|
||||||
@ -79,6 +77,7 @@ function QDynamicForm(props: Props): JSX.Element
|
|||||||
bulkEditSwitchChangeHandler(name, value);
|
bulkEditSwitchChangeHandler(name, value);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box>
|
<Box>
|
||||||
<Box lineHeight={0}>
|
<Box lineHeight={0}>
|
||||||
@ -96,29 +95,38 @@ function QDynamicForm(props: Props): JSX.Element
|
|||||||
&& Object.keys(formFields).map((fieldName: any) =>
|
&& Object.keys(formFields).map((fieldName: any) =>
|
||||||
{
|
{
|
||||||
const field = formFields[fieldName];
|
const field = formFields[fieldName];
|
||||||
|
if (field.omitFromQDynamicForm)
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
if (values[fieldName] === undefined)
|
if (values[fieldName] === undefined)
|
||||||
{
|
{
|
||||||
values[fieldName] = "";
|
values[fieldName] = "";
|
||||||
}
|
}
|
||||||
|
|
||||||
if (field.omitFromQDynamicForm)
|
let formattedHelpContent = <HelpContent helpContents={field.fieldMetaData.helpContents} roles={helpRoles} helpContentKey={`${helpContentKeyPrefix ?? ""}field:${fieldName}`} />;
|
||||||
|
if(formattedHelpContent)
|
||||||
{
|
{
|
||||||
return null;
|
formattedHelpContent = <Box color="#757575" fontSize="0.875rem" mt="-0.25rem">{formattedHelpContent}</Box>
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const labelElement = <Box fontSize="1rem" fontWeight="500" marginBottom="0.25rem">
|
||||||
|
<label htmlFor={field.name}>{field.label}</label>
|
||||||
|
</Box>
|
||||||
|
|
||||||
if (field.type === "file")
|
if (field.type === "file")
|
||||||
{
|
{
|
||||||
const pseudoField = new QFieldMetaData({name: fieldName, type: QFieldType.BLOB});
|
const pseudoField = new QFieldMetaData({name: fieldName, type: QFieldType.BLOB});
|
||||||
return (
|
return (
|
||||||
<Grid item xs={12} sm={6} key={fieldName}>
|
<Grid item xs={12} sm={6} key={fieldName}>
|
||||||
<Box mb={1.5}>
|
<Box mb={1.5}>
|
||||||
|
{labelElement}
|
||||||
<InputLabel shrink={true}>{field.label}</InputLabel>
|
|
||||||
{
|
{
|
||||||
props.record && props.record.values.get(fieldName) && <Box fontSize="0.875rem" pb={1}>
|
record && record.values.get(fieldName) && <Box fontSize="0.875rem" pb={1}>
|
||||||
Current File:
|
Current File:
|
||||||
<Box display="inline-flex" pl={1}>
|
<Box display="inline-flex" pl={1}>
|
||||||
{ValueUtils.getDisplayValue(pseudoField, props.record, "view")}
|
{ValueUtils.getDisplayValue(pseudoField, record, "view")}
|
||||||
<Tooltip placement="bottom" title="Remove current file">
|
<Tooltip placement="bottom" title="Remove current file">
|
||||||
<Icon className="blobIcon" fontSize="small" onClick={(e) => removeFile(fieldName)}>delete</Icon>
|
<Icon className="blobIcon" fontSize="small" onClick={(e) => removeFile(fieldName)}>delete</Icon>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
@ -162,18 +170,20 @@ function QDynamicForm(props: Props): JSX.Element
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<Grid item xs={12} sm={6} key={fieldName}>
|
<Grid item xs={12} sm={6} key={fieldName}>
|
||||||
|
{labelElement}
|
||||||
<DynamicSelect
|
<DynamicSelect
|
||||||
tableName={field.possibleValueProps.tableName}
|
tableName={field.possibleValueProps.tableName}
|
||||||
processName={field.possibleValueProps.processName}
|
processName={field.possibleValueProps.processName}
|
||||||
fieldName={fieldName}
|
fieldName={fieldName}
|
||||||
isEditable={field.isEditable}
|
isEditable={field.isEditable}
|
||||||
fieldLabel={field.label}
|
fieldLabel=""
|
||||||
initialValue={values[fieldName]}
|
initialValue={values[fieldName]}
|
||||||
initialDisplayValue={field.possibleValueProps.initialDisplayValue}
|
initialDisplayValue={field.possibleValueProps.initialDisplayValue}
|
||||||
bulkEditMode={bulkEditMode}
|
bulkEditMode={bulkEditMode}
|
||||||
bulkEditSwitchChangeHandler={bulkEditSwitchChanged}
|
bulkEditSwitchChangeHandler={bulkEditSwitchChanged}
|
||||||
otherValues={otherValuesMap}
|
otherValues={otherValuesMap}
|
||||||
/>
|
/>
|
||||||
|
{formattedHelpContent}
|
||||||
</Grid>
|
</Grid>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -182,9 +192,11 @@ function QDynamicForm(props: Props): JSX.Element
|
|||||||
// todo? placeholder={password.placeholder}
|
// todo? placeholder={password.placeholder}
|
||||||
return (
|
return (
|
||||||
<Grid item xs={12} sm={6} key={fieldName}>
|
<Grid item xs={12} sm={6} key={fieldName}>
|
||||||
|
{labelElement}
|
||||||
<QDynamicFormField
|
<QDynamicFormField
|
||||||
|
id={field.name}
|
||||||
type={field.type}
|
type={field.type}
|
||||||
label={field.label}
|
label=""
|
||||||
isEditable={field.isEditable}
|
isEditable={field.isEditable}
|
||||||
name={fieldName}
|
name={fieldName}
|
||||||
displayFormat={field.displayFormat}
|
displayFormat={field.displayFormat}
|
||||||
@ -195,6 +207,7 @@ function QDynamicForm(props: Props): JSX.Element
|
|||||||
success={`${values[fieldName]}` !== "" && !errors[fieldName] && touched[fieldName]}
|
success={`${values[fieldName]}` !== "" && !errors[fieldName] && touched[fieldName]}
|
||||||
formFieldObject={field}
|
formFieldObject={field}
|
||||||
/>
|
/>
|
||||||
|
{formattedHelpContent}
|
||||||
</Grid>
|
</Grid>
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
@ -207,6 +220,7 @@ function QDynamicForm(props: Props): JSX.Element
|
|||||||
QDynamicForm.defaultProps = {
|
QDynamicForm.defaultProps = {
|
||||||
formLabel: undefined,
|
formLabel: undefined,
|
||||||
bulkEditMode: false,
|
bulkEditMode: false,
|
||||||
|
helpRoles: ["ALL_SCREENS"],
|
||||||
bulkEditSwitchChangeHandler: () =>
|
bulkEditSwitchChangeHandler: () =>
|
||||||
{
|
{
|
||||||
},
|
},
|
||||||
|
@ -25,6 +25,7 @@ import Switch from "@mui/material/Switch";
|
|||||||
import {ErrorMessage, Field, useFormikContext} from "formik";
|
import {ErrorMessage, Field, useFormikContext} from "formik";
|
||||||
import React, {useState} from "react";
|
import React, {useState} from "react";
|
||||||
import AceEditor from "react-ace";
|
import AceEditor from "react-ace";
|
||||||
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
import BooleanFieldSwitch from "qqq/components/forms/BooleanFieldSwitch";
|
import BooleanFieldSwitch from "qqq/components/forms/BooleanFieldSwitch";
|
||||||
import MDInput from "qqq/components/legacy/MDInput";
|
import MDInput from "qqq/components/legacy/MDInput";
|
||||||
import MDTypography from "qqq/components/legacy/MDTypography";
|
import MDTypography from "qqq/components/legacy/MDTypography";
|
||||||
@ -52,6 +53,7 @@ function QDynamicFormField({
|
|||||||
{
|
{
|
||||||
const [switchChecked, setSwitchChecked] = useState(false);
|
const [switchChecked, setSwitchChecked] = useState(false);
|
||||||
const [isDisabled, setIsDisabled] = useState(!isEditable || bulkEditMode);
|
const [isDisabled, setIsDisabled] = useState(!isEditable || bulkEditMode);
|
||||||
|
const {inputBorderColor} = colors;
|
||||||
|
|
||||||
const {setFieldValue} = useFormikContext();
|
const {setFieldValue} = useFormikContext();
|
||||||
|
|
||||||
@ -88,7 +90,14 @@ function QDynamicFormField({
|
|||||||
if (type === "checkbox")
|
if (type === "checkbox")
|
||||||
{
|
{
|
||||||
getsBulkEditHtmlLabel = false;
|
getsBulkEditHtmlLabel = false;
|
||||||
field = (<BooleanFieldSwitch name={name} label={label} value={value} isDisabled={isDisabled} />);
|
field = (<>
|
||||||
|
<BooleanFieldSwitch name={name} label={label} value={value} isDisabled={isDisabled} />
|
||||||
|
<Box mt={0.75}>
|
||||||
|
<MDTypography component="div" variant="caption" color="error" fontWeight="regular">
|
||||||
|
{!isDisabled && <div className="fieldErrorMessage"><ErrorMessage name={name} render={msg => <span data-field-error="true">{msg}</span>} /></div>}
|
||||||
|
</MDTypography>
|
||||||
|
</Box>
|
||||||
|
</>);
|
||||||
}
|
}
|
||||||
else if (type === "ace")
|
else if (type === "ace")
|
||||||
{
|
{
|
||||||
@ -115,7 +124,7 @@ function QDynamicFormField({
|
|||||||
width="100%"
|
width="100%"
|
||||||
height="300px"
|
height="300px"
|
||||||
value={value}
|
value={value}
|
||||||
style={{border: "1px solid gray"}}
|
style={{border: `1px solid ${inputBorderColor}`, borderRadius: "0.75rem"}}
|
||||||
/>
|
/>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
@ -124,7 +133,7 @@ function QDynamicFormField({
|
|||||||
{
|
{
|
||||||
field = (
|
field = (
|
||||||
<>
|
<>
|
||||||
<Field {...rest} onWheel={handleOnWheel} name={name} type={type} as={MDInput} variant="standard" label={label} InputLabelProps={inputLabelProps} InputProps={inputProps} fullWidth disabled={isDisabled}
|
<Field {...rest} onWheel={handleOnWheel} name={name} type={type} as={MDInput} variant="outlined" label={label} InputLabelProps={inputLabelProps} InputProps={inputProps} fullWidth disabled={isDisabled}
|
||||||
onKeyPress={(e: any) =>
|
onKeyPress={(e: any) =>
|
||||||
{
|
{
|
||||||
if (e.key === "Enter")
|
if (e.key === "Enter")
|
||||||
@ -164,6 +173,14 @@ function QDynamicFormField({
|
|||||||
id={`bulkEditSwitch-${name}`}
|
id={`bulkEditSwitch-${name}`}
|
||||||
checked={switchChecked}
|
checked={switchChecked}
|
||||||
onClick={bulkEditSwitchChanged}
|
onClick={bulkEditSwitchChanged}
|
||||||
|
sx={{top: "-4px",
|
||||||
|
"& .MuiSwitch-track": {
|
||||||
|
height: 20,
|
||||||
|
borderRadius: 10,
|
||||||
|
top: -3,
|
||||||
|
position: "relative"
|
||||||
|
}
|
||||||
|
}}
|
||||||
/>
|
/>
|
||||||
</Box>
|
</Box>
|
||||||
<Box width="100%" sx={{background: (type == "checkbox" && isDisabled) ? "#f0f2f5!important" : "initial"}}>
|
<Box width="100%" sx={{background: (type == "checkbox" && isDisabled) ? "#f0f2f5!important" : "initial"}}>
|
||||||
|
@ -89,6 +89,7 @@ class DynamicFormUtils
|
|||||||
label += field.isRequired ? " *" : "";
|
label += field.isRequired ? " *" : "";
|
||||||
|
|
||||||
return ({
|
return ({
|
||||||
|
fieldMetaData: field,
|
||||||
name: field.name,
|
name: field.name,
|
||||||
label: label,
|
label: label,
|
||||||
isRequired: field.isRequired,
|
isRequired: field.isRequired,
|
||||||
|
@ -29,6 +29,7 @@ import Switch from "@mui/material/Switch";
|
|||||||
import TextField from "@mui/material/TextField";
|
import TextField from "@mui/material/TextField";
|
||||||
import {ErrorMessage, useFormikContext} from "formik";
|
import {ErrorMessage, useFormikContext} from "formik";
|
||||||
import React, {useEffect, useState} from "react";
|
import React, {useEffect, useState} from "react";
|
||||||
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
import MDTypography from "qqq/components/legacy/MDTypography";
|
import MDTypography from "qqq/components/legacy/MDTypography";
|
||||||
import Client from "qqq/utils/qqq/Client";
|
import Client from "qqq/utils/qqq/Client";
|
||||||
|
|
||||||
@ -49,6 +50,7 @@ interface Props
|
|||||||
bulkEditMode?: boolean;
|
bulkEditMode?: boolean;
|
||||||
bulkEditSwitchChangeHandler?: any;
|
bulkEditSwitchChangeHandler?: any;
|
||||||
otherValues?: Map<string, any>;
|
otherValues?: Map<string, any>;
|
||||||
|
variant: "standard" | "outlined";
|
||||||
}
|
}
|
||||||
|
|
||||||
DynamicSelect.defaultProps = {
|
DynamicSelect.defaultProps = {
|
||||||
@ -63,6 +65,7 @@ DynamicSelect.defaultProps = {
|
|||||||
isMultiple: false,
|
isMultiple: false,
|
||||||
bulkEditMode: false,
|
bulkEditMode: false,
|
||||||
otherValues: new Map<string, any>(),
|
otherValues: new Map<string, any>(),
|
||||||
|
variant: "outlined",
|
||||||
bulkEditSwitchChangeHandler: () =>
|
bulkEditSwitchChangeHandler: () =>
|
||||||
{
|
{
|
||||||
},
|
},
|
||||||
@ -70,12 +73,13 @@ DynamicSelect.defaultProps = {
|
|||||||
|
|
||||||
const qController = Client.getInstance();
|
const qController = Client.getInstance();
|
||||||
|
|
||||||
function DynamicSelect({tableName, processName, fieldName, overrideId, fieldLabel, inForm, initialValue, initialDisplayValue, initialValues, onChange, isEditable, isMultiple, bulkEditMode, bulkEditSwitchChangeHandler, otherValues}: Props)
|
function DynamicSelect({tableName, processName, fieldName, overrideId, fieldLabel, inForm, initialValue, initialDisplayValue, initialValues, onChange, isEditable, isMultiple, bulkEditMode, bulkEditSwitchChangeHandler, otherValues, variant}: Props)
|
||||||
{
|
{
|
||||||
const [open, setOpen] = useState(false);
|
const [open, setOpen] = useState(false);
|
||||||
const [options, setOptions] = useState<readonly QPossibleValue[]>([]);
|
const [options, setOptions] = useState<readonly QPossibleValue[]>([]);
|
||||||
const [searchTerm, setSearchTerm] = useState(null);
|
const [searchTerm, setSearchTerm] = useState(null);
|
||||||
const [firstRender, setFirstRender] = useState(true);
|
const [firstRender, setFirstRender] = useState(true);
|
||||||
|
const {inputBorderColor} = colors;
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// default value - needs to be an array (from initialValues (array) prop) for multiple mode - //
|
// default value - needs to be an array (from initialValues (array) prop) for multiple mode - //
|
||||||
@ -230,7 +234,7 @@ function DynamicSelect({tableName, processName, fieldName, overrideId, fieldLabe
|
|||||||
// attributes. so, doing this, w/ key=id, seemed to fix it. //
|
// attributes. so, doing this, w/ key=id, seemed to fix it. //
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
return (
|
return (
|
||||||
<li {...props} key={option.id}>
|
<li {...props} key={option.id} style={{fontSize: "1rem"}}>
|
||||||
{content}
|
{content}
|
||||||
</li>
|
</li>
|
||||||
);
|
);
|
||||||
@ -244,13 +248,35 @@ function DynamicSelect({tableName, processName, fieldName, overrideId, fieldLabe
|
|||||||
bulkEditSwitchChangeHandler(fieldName, newSwitchValue);
|
bulkEditSwitchChangeHandler(fieldName, newSwitchValue);
|
||||||
};
|
};
|
||||||
|
|
||||||
// console.log(`default value: ${JSON.stringify(defaultValue)}`);
|
////////////////////////////////////////////
|
||||||
|
// for outlined style, adjust some styles //
|
||||||
|
////////////////////////////////////////////
|
||||||
|
let autocompleteSX = {};
|
||||||
|
if (variant == "outlined")
|
||||||
|
{
|
||||||
|
autocompleteSX = {
|
||||||
|
"& .MuiOutlinedInput-root": {
|
||||||
|
borderRadius: "0.75rem",
|
||||||
|
},
|
||||||
|
"& .MuiInputBase-root": {
|
||||||
|
padding: "0.5rem",
|
||||||
|
background: isDisabled ? "#f0f2f5!important" : "initial",
|
||||||
|
},
|
||||||
|
"& .MuiOutlinedInput-root .MuiAutocomplete-input": {
|
||||||
|
padding: "0",
|
||||||
|
fontSize: "1rem"
|
||||||
|
},
|
||||||
|
"& .Mui-disabled .MuiOutlinedInput-notchedOutline": {
|
||||||
|
borderColor: inputBorderColor
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const autocomplete = (
|
const autocomplete = (
|
||||||
<Box>
|
<Box>
|
||||||
<Autocomplete
|
<Autocomplete
|
||||||
id={overrideId ?? fieldName}
|
id={overrideId ?? fieldName}
|
||||||
sx={{background: isDisabled ? "#f0f2f5!important" : "initial"}}
|
sx={autocompleteSX}
|
||||||
open={open}
|
open={open}
|
||||||
fullWidth
|
fullWidth
|
||||||
onOpen={() =>
|
onOpen={() =>
|
||||||
@ -305,7 +331,7 @@ function DynamicSelect({tableName, processName, fieldName, overrideId, fieldLabe
|
|||||||
<TextField
|
<TextField
|
||||||
{...params}
|
{...params}
|
||||||
label={fieldLabel}
|
label={fieldLabel}
|
||||||
variant="standard"
|
variant={variant}
|
||||||
autoComplete="off"
|
autoComplete="off"
|
||||||
type="search"
|
type="search"
|
||||||
InputProps={{
|
InputProps={{
|
||||||
@ -341,6 +367,14 @@ function DynamicSelect({tableName, processName, fieldName, overrideId, fieldLabe
|
|||||||
id={`bulkEditSwitch-${fieldName}`}
|
id={`bulkEditSwitch-${fieldName}`}
|
||||||
checked={switchChecked}
|
checked={switchChecked}
|
||||||
onClick={bulkEditSwitchChanged}
|
onClick={bulkEditSwitchChanged}
|
||||||
|
sx={{top: "-4px",
|
||||||
|
"& .MuiSwitch-track": {
|
||||||
|
height: 20,
|
||||||
|
borderRadius: 10,
|
||||||
|
top: -3,
|
||||||
|
position: "relative"
|
||||||
|
}
|
||||||
|
}}
|
||||||
/>
|
/>
|
||||||
</Box>
|
</Box>
|
||||||
<Box width="100%">
|
<Box width="100%">
|
||||||
|
@ -37,10 +37,12 @@ import React, {useContext, useEffect, useReducer, useState} from "react";
|
|||||||
import {useLocation, useNavigate, useParams} from "react-router-dom";
|
import {useLocation, useNavigate, useParams} from "react-router-dom";
|
||||||
import * as Yup from "yup";
|
import * as Yup from "yup";
|
||||||
import QContext from "QContext";
|
import QContext from "QContext";
|
||||||
|
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 QDynamicForm from "qqq/components/forms/DynamicForm";
|
import QDynamicForm from "qqq/components/forms/DynamicForm";
|
||||||
import DynamicFormUtils from "qqq/components/forms/DynamicFormUtils";
|
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 QRecordSidebar from "qqq/components/misc/RecordSidebar";
|
import QRecordSidebar from "qqq/components/misc/RecordSidebar";
|
||||||
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";
|
||||||
@ -79,6 +81,7 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
const [validations, setValidations] = useState({});
|
const [validations, setValidations] = useState({});
|
||||||
const [initialValues, setInitialValues] = useState({} as { [key: string]: any });
|
const [initialValues, setInitialValues] = useState({} as { [key: string]: any });
|
||||||
const [formFields, setFormFields] = useState(null as Map<string, any>);
|
const [formFields, setFormFields] = useState(null as Map<string, any>);
|
||||||
|
const [t1section, setT1Section] = useState(null as QTableSection);
|
||||||
const [t1sectionName, setT1SectionName] = useState(null as string);
|
const [t1sectionName, setT1SectionName] = useState(null as string);
|
||||||
const [nonT1Sections, setNonT1Sections] = useState([] as QTableSection[]);
|
const [nonT1Sections, setNonT1Sections] = useState([] as QTableSection[]);
|
||||||
|
|
||||||
@ -151,7 +154,9 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
{
|
{
|
||||||
return <div>Loading...</div>;
|
return <div>Loading...</div>;
|
||||||
}
|
}
|
||||||
return <QDynamicForm formData={formData} record={record} />;
|
|
||||||
|
const helpRoles = [props.id ? "EDIT_SCREEN" : "INSERT_SCREEN", "WRITE_SCREENS", "ALL_SCREENS"]
|
||||||
|
return <QDynamicForm formData={formData} record={record} helpRoles={helpRoles} helpContentKeyPrefix={`table:${tableName};`} />;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!asyncLoadInited)
|
if (!asyncLoadInited)
|
||||||
@ -330,6 +335,7 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
/////////////////////////////////////
|
/////////////////////////////////////
|
||||||
const dynamicFormFieldsBySection = new Map<string, any>();
|
const dynamicFormFieldsBySection = new Map<string, any>();
|
||||||
let t1sectionName;
|
let t1sectionName;
|
||||||
|
let t1section;
|
||||||
const nonT1Sections: QTableSection[] = [];
|
const nonT1Sections: QTableSection[] = [];
|
||||||
for (let i = 0; i < tableSections.length; i++)
|
for (let i = 0; i < tableSections.length; i++)
|
||||||
{
|
{
|
||||||
@ -382,6 +388,7 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
if (section.tier === "T1")
|
if (section.tier === "T1")
|
||||||
{
|
{
|
||||||
t1sectionName = section.name;
|
t1sectionName = section.name;
|
||||||
|
t1section = section;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -389,6 +396,7 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
setT1SectionName(t1sectionName);
|
setT1SectionName(t1sectionName);
|
||||||
|
setT1Section(t1section);
|
||||||
setNonT1Sections(nonT1Sections);
|
setNonT1Sections(nonT1Sections);
|
||||||
setFormFields(dynamicFormFieldsBySection);
|
setFormFields(dynamicFormFieldsBySection);
|
||||||
setValidations(Yup.object().shape(formValidations));
|
setValidations(Yup.object().shape(formValidations));
|
||||||
@ -426,6 +434,11 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
actions.setSubmitting(true);
|
actions.setSubmitting(true);
|
||||||
await (async () =>
|
await (async () =>
|
||||||
{
|
{
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// we will be manipulating the values sent to the backend, so clone values so they remained unchanged for the form widgets //
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
const valuesToPost = JSON.parse(JSON.stringify(values));
|
||||||
|
|
||||||
for(let fieldName of tableMetaData.fields.keys())
|
for(let fieldName of tableMetaData.fields.keys())
|
||||||
{
|
{
|
||||||
const fieldMetaData = tableMetaData.fields.get(fieldName);
|
const fieldMetaData = tableMetaData.fields.get(fieldName);
|
||||||
@ -438,17 +451,17 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
// changing from, say, 12:15:30 to just 12:15:00... this seems to get around that, for cases when the //
|
// changing from, say, 12:15:30 to just 12:15:00... this seems to get around that, for cases when the //
|
||||||
// user didn't change the value in the field (but if the user did change the value, then we will submit it) //
|
// user didn't change the value in the field (but if the user did change the value, then we will submit it) //
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
if(fieldMetaData.type === QFieldType.DATE_TIME && values[fieldName])
|
if(fieldMetaData.type === QFieldType.DATE_TIME && valuesToPost[fieldName])
|
||||||
{
|
{
|
||||||
console.log(`DateTime ${fieldName}: Initial value: [${initialValues[fieldName]}] -> [${values[fieldName]}]`)
|
console.log(`DateTime ${fieldName}: Initial value: [${initialValues[fieldName]}] -> [${valuesToPost[fieldName]}]`)
|
||||||
if (initialValues[fieldName] == values[fieldName])
|
if (initialValues[fieldName] == valuesToPost[fieldName])
|
||||||
{
|
{
|
||||||
console.log(" - Is the same, so, deleting from the post");
|
console.log(" - Is the same, so, deleting from the post");
|
||||||
delete (values[fieldName]);
|
delete (valuesToPost[fieldName]);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
values[fieldName] = ValueUtils.frontendLocalZoneDateTimeStringToUTCStringForBackend(values[fieldName]);
|
valuesToPost[fieldName] = ValueUtils.frontendLocalZoneDateTimeStringToUTCStringForBackend(valuesToPost[fieldName]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -461,10 +474,14 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
if(fieldMetaData.type === QFieldType.BLOB)
|
if(fieldMetaData.type === QFieldType.BLOB)
|
||||||
{
|
{
|
||||||
if(typeof values[fieldName] === "string")
|
if(typeof valuesToPost[fieldName] === "string")
|
||||||
{
|
{
|
||||||
console.log(`${fieldName} value was a string, so, we're deleting it from the values array, to not submit it to the backend, to not change it.`);
|
console.log(`${fieldName} value was a string, so, we're deleting it from the values array, to not submit it to the backend, to not change it.`);
|
||||||
delete(values[fieldName]);
|
delete(valuesToPost[fieldName]);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
valuesToPost[fieldName] = values[fieldName];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -473,7 +490,7 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
{
|
{
|
||||||
// todo - audit that it's a dupe
|
// todo - audit that it's a dupe
|
||||||
await qController
|
await qController
|
||||||
.update(tableName, props.id, values)
|
.update(tableName, props.id, valuesToPost)
|
||||||
.then((record) =>
|
.then((record) =>
|
||||||
{
|
{
|
||||||
if (props.isModal)
|
if (props.isModal)
|
||||||
@ -506,7 +523,7 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
await qController
|
await qController
|
||||||
.create(tableName, values)
|
.create(tableName, valuesToPost)
|
||||||
.then((record) =>
|
.then((record) =>
|
||||||
{
|
{
|
||||||
if (props.isModal)
|
if (props.isModal)
|
||||||
@ -543,6 +560,19 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
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;
|
||||||
|
|
||||||
|
const getSectionHelp = (section: QTableSection) =>
|
||||||
|
{
|
||||||
|
const helpRoles = [props.id ? "EDIT_SCREEN" : "INSERT_SCREEN", "WRITE_SCREENS", "ALL_SCREENS"]
|
||||||
|
const formattedHelpContent = <HelpContent helpContents={section.helpContents} roles={helpRoles} helpContentKey={`table:${tableMetaData.name};section:${section.name}`} />;
|
||||||
|
|
||||||
|
return formattedHelpContent && (
|
||||||
|
<Box px={"1.5rem"} fontSize={"0.875rem"}>
|
||||||
|
{formattedHelpContent}
|
||||||
|
</Box>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
if (notAllowedError)
|
if (notAllowedError)
|
||||||
{
|
{
|
||||||
body = (
|
body = (
|
||||||
@ -564,9 +594,11 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
const cardElevation = props.isModal ? 3 : 1;
|
const cardElevation = props.isModal ? 3 : 0;
|
||||||
body = (
|
body = (
|
||||||
<Box mb={3}>
|
<Box mb={3}>
|
||||||
|
{
|
||||||
|
(alertContent || warningContent) &&
|
||||||
<Grid container spacing={3}>
|
<Grid container spacing={3}>
|
||||||
<Grid item xs={12}>
|
<Grid item xs={12}>
|
||||||
{alertContent ? (
|
{alertContent ? (
|
||||||
@ -581,6 +613,7 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
) : ("")}
|
) : ("")}
|
||||||
</Grid>
|
</Grid>
|
||||||
</Grid>
|
</Grid>
|
||||||
|
}
|
||||||
<Grid container spacing={3}>
|
<Grid container spacing={3}>
|
||||||
{
|
{
|
||||||
!props.isModal &&
|
!props.isModal &&
|
||||||
@ -618,10 +651,11 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
<MDTypography variant="h5">{formTitle}</MDTypography>
|
<MDTypography variant="h5">{formTitle}</MDTypography>
|
||||||
</Box>
|
</Box>
|
||||||
</Box>
|
</Box>
|
||||||
|
{t1section && getSectionHelp(t1section)}
|
||||||
{
|
{
|
||||||
t1sectionName && formFields ? (
|
t1sectionName && formFields ? (
|
||||||
<Box pb={1} px={3}>
|
<Box px={3}>
|
||||||
<Box p={3} width="100%">
|
<Box pb={"0.25rem"} width="100%">
|
||||||
{getFormSection(values, touched, formFields.get(t1sectionName), errors)}
|
{getFormSection(values, touched, formFields.get(t1sectionName), errors)}
|
||||||
</Box>
|
</Box>
|
||||||
</Box>
|
</Box>
|
||||||
@ -635,8 +669,9 @@ function EntityForm(props: Props): JSX.Element
|
|||||||
<MDTypography variant="h6" p={3} pb={1}>
|
<MDTypography variant="h6" p={3} pb={1}>
|
||||||
{section.label}
|
{section.label}
|
||||||
</MDTypography>
|
</MDTypography>
|
||||||
|
{getSectionHelp(section)}
|
||||||
<Box pb={1} px={3}>
|
<Box pb={1} px={3}>
|
||||||
<Box p={3} width="100%">
|
<Box pb={"0.75rem"} width="100%">
|
||||||
{getFormSection(values, touched, formFields.get(section.name), errors)}
|
{getFormSection(values, touched, formFields.get(section.name), errors)}
|
||||||
</Box>
|
</Box>
|
||||||
</Box>
|
</Box>
|
||||||
|
@ -25,6 +25,7 @@ import Icon from "@mui/material/Icon";
|
|||||||
import {ReactNode, useContext} from "react";
|
import {ReactNode, useContext} from "react";
|
||||||
import {Link} from "react-router-dom";
|
import {Link} from "react-router-dom";
|
||||||
import QContext from "QContext";
|
import QContext from "QContext";
|
||||||
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
import MDTypography from "qqq/components/legacy/MDTypography";
|
import MDTypography from "qqq/components/legacy/MDTypography";
|
||||||
|
|
||||||
interface Props
|
interface Props
|
||||||
@ -112,43 +113,35 @@ function QBreadcrumbs({icon, title, route, light}: Props): JSX.Element
|
|||||||
<Box mr={{xs: 0, xl: 8}}>
|
<Box mr={{xs: 0, xl: 8}}>
|
||||||
<MuiBreadcrumbs
|
<MuiBreadcrumbs
|
||||||
sx={{
|
sx={{
|
||||||
|
fontSize: "1.125rem",
|
||||||
|
fontWeight: "500",
|
||||||
|
color: colors.dark.main,
|
||||||
|
"& li": {
|
||||||
|
lineHeight: "unset!important"
|
||||||
|
},
|
||||||
|
"& a": {
|
||||||
|
color: colors.gray.main
|
||||||
|
},
|
||||||
"& .MuiBreadcrumbs-separator": {
|
"& .MuiBreadcrumbs-separator": {
|
||||||
color: ({palette: {white, grey}}) => (light ? white.main : grey[600]),
|
fontSize: "1.125rem",
|
||||||
|
fontWeight: "500",
|
||||||
|
color: colors.dark.main
|
||||||
},
|
},
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Link to="/">
|
<Link to="/">
|
||||||
<MDTypography
|
<Icon sx={{fontSize: "1.25rem!important", position: "relative", top: "0.25rem"}}>{icon}</Icon>
|
||||||
component="span"
|
|
||||||
variant="body2"
|
|
||||||
color={light ? "white" : "dark"}
|
|
||||||
opacity={light ? 0.8 : 0.5}
|
|
||||||
sx={{lineHeight: 0}}
|
|
||||||
>
|
|
||||||
<Icon>{icon}</Icon>
|
|
||||||
</MDTypography>
|
|
||||||
</Link>
|
</Link>
|
||||||
{fullRoutes.map((fullRoute: string) => (
|
{fullRoutes.map((fullRoute: string) => (
|
||||||
<Link to={fullRoute} key={fullRoute}>
|
<Link to={fullRoute} key={fullRoute}>
|
||||||
<MDTypography
|
|
||||||
component="span"
|
|
||||||
variant="button"
|
|
||||||
fontWeight="regular"
|
|
||||||
textTransform="capitalize"
|
|
||||||
color={light ? "white" : "dark"}
|
|
||||||
opacity={light ? 0.8 : 0.5}
|
|
||||||
sx={{lineHeight: 0}}
|
|
||||||
>
|
|
||||||
{fullPathToLabel(fullRoute, fullRoute.replace(/.*\//, ""))}
|
{fullPathToLabel(fullRoute, fullRoute.replace(/.*\//, ""))}
|
||||||
</MDTypography>
|
|
||||||
</Link>
|
</Link>
|
||||||
))}
|
))}
|
||||||
</MuiBreadcrumbs>
|
</MuiBreadcrumbs>
|
||||||
<MDTypography
|
<MDTypography
|
||||||
pt={1}
|
pt={1}
|
||||||
fontWeight="bold"
|
|
||||||
textTransform="capitalize"
|
textTransform="capitalize"
|
||||||
variant="h5"
|
variant="h3"
|
||||||
color={light ? "white" : "dark"}
|
color={light ? "white" : "dark"}
|
||||||
noWrap
|
noWrap
|
||||||
>
|
>
|
||||||
|
@ -19,7 +19,7 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import {Popper} from "@mui/material";
|
import {Popper, InputAdornment} from "@mui/material";
|
||||||
import AppBar from "@mui/material/AppBar";
|
import AppBar from "@mui/material/AppBar";
|
||||||
import Autocomplete from "@mui/material/Autocomplete";
|
import Autocomplete from "@mui/material/Autocomplete";
|
||||||
import Badge from "@mui/material/Badge";
|
import Badge from "@mui/material/Badge";
|
||||||
@ -34,8 +34,8 @@ import React, {useContext, useEffect, useState} from "react";
|
|||||||
import {useLocation, useNavigate} from "react-router-dom";
|
import {useLocation, useNavigate} from "react-router-dom";
|
||||||
import QContext from "QContext";
|
import QContext from "QContext";
|
||||||
import QBreadcrumbs, {routeToLabel} from "qqq/components/horseshoe/Breadcrumbs";
|
import QBreadcrumbs, {routeToLabel} from "qqq/components/horseshoe/Breadcrumbs";
|
||||||
import {navbar, navbarContainer, navbarIconButton, navbarRow,} from "qqq/components/horseshoe/Styles";
|
import {navbar, navbarContainer, navbarRow, navbarMobileMenu, recentlyViewedMenu,} from "qqq/components/horseshoe/Styles";
|
||||||
import {setTransparentNavbar, useMaterialUIController,} from "qqq/context";
|
import {setTransparentNavbar, useMaterialUIController, setMiniSidenav} from "qqq/context";
|
||||||
import HistoryUtils from "qqq/utils/HistoryUtils";
|
import HistoryUtils from "qqq/utils/HistoryUtils";
|
||||||
|
|
||||||
// Declaring prop types for NavBar
|
// Declaring prop types for NavBar
|
||||||
@ -57,7 +57,7 @@ function NavBar({absolute, light, isMini}: Props): JSX.Element
|
|||||||
{
|
{
|
||||||
const [navbarType, setNavbarType] = useState<"fixed" | "absolute" | "relative" | "static" | "sticky">();
|
const [navbarType, setNavbarType] = useState<"fixed" | "absolute" | "relative" | "static" | "sticky">();
|
||||||
const [controller, dispatch] = useMaterialUIController();
|
const [controller, dispatch] = useMaterialUIController();
|
||||||
const {transparentNavbar, fixedNavbar, darkMode,} = controller;
|
const {miniSidenav, transparentNavbar, fixedNavbar, darkMode,} = controller;
|
||||||
const [openMenu, setOpenMenu] = useState<any>(false);
|
const [openMenu, setOpenMenu] = useState<any>(false);
|
||||||
const [history, setHistory] = useState<any>([] as HistoryEntry[]);
|
const [history, setHistory] = useState<any>([] as HistoryEntry[]);
|
||||||
const [autocompleteValue, setAutocompleteValue] = useState<any>(null);
|
const [autocompleteValue, setAutocompleteValue] = useState<any>(null);
|
||||||
@ -105,6 +105,8 @@ function NavBar({absolute, light, isMini}: Props): JSX.Element
|
|||||||
return () => window.removeEventListener("scroll", handleTransparentNavbar);
|
return () => window.removeEventListener("scroll", handleTransparentNavbar);
|
||||||
}, [dispatch, fixedNavbar]);
|
}, [dispatch, fixedNavbar]);
|
||||||
|
|
||||||
|
const handleMiniSidenav = () => setMiniSidenav(dispatch, !miniSidenav);
|
||||||
|
|
||||||
const goToHistory = (path: string) =>
|
const goToHistory = (path: string) =>
|
||||||
{
|
{
|
||||||
navigate(path);
|
navigate(path);
|
||||||
@ -157,12 +159,20 @@ function NavBar({absolute, light, isMini}: Props): JSX.Element
|
|||||||
options={history}
|
options={history}
|
||||||
autoHighlight
|
autoHighlight
|
||||||
blurOnSelect
|
blurOnSelect
|
||||||
style={{width: "200px"}}
|
style={{width: "16rem"}}
|
||||||
onOpen={handleHistoryOnOpen}
|
onOpen={handleHistoryOnOpen}
|
||||||
onChange={handleAutocompleteOnChange}
|
onChange={handleAutocompleteOnChange}
|
||||||
PopperComponent={CustomPopper}
|
PopperComponent={CustomPopper}
|
||||||
isOptionEqualToValue={(option, value) => option.id === value.id}
|
isOptionEqualToValue={(option, value) => option.id === value.id}
|
||||||
renderInput={(params) => <TextField {...params} label="Recently Viewed Records" />}
|
sx={recentlyViewedMenu}
|
||||||
|
renderInput={(params) => <TextField {...params} label="Recently Viewed Records" InputProps={{
|
||||||
|
...params.InputProps,
|
||||||
|
endAdornment: (
|
||||||
|
<InputAdornment position="end">
|
||||||
|
<Icon sx={{position: "relative", right: "-1rem"}}>keyboard_arrow_down</Icon>
|
||||||
|
</InputAdornment>
|
||||||
|
)
|
||||||
|
}} />}
|
||||||
renderOption={(props, option: HistoryEntry) => (
|
renderOption={(props, option: HistoryEntry) => (
|
||||||
<Box {...props} component="li" key={option.id} sx={{width: "auto"}}>
|
<Box {...props} component="li" key={option.id} sx={{width: "auto"}}>
|
||||||
<Box sx={{width: "auto", px: "8px", whiteSpace: "overflow"}} key={option.id}>
|
<Box sx={{width: "auto", px: "8px", whiteSpace: "overflow"}} key={option.id}>
|
||||||
@ -175,22 +185,6 @@ function NavBar({absolute, light, isMini}: Props): JSX.Element
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Render the notifications menu
|
|
||||||
const renderMenu = () => (
|
|
||||||
<Menu
|
|
||||||
anchorEl={openMenu}
|
|
||||||
anchorReference={null}
|
|
||||||
anchorOrigin={{
|
|
||||||
vertical: "bottom",
|
|
||||||
horizontal: "left",
|
|
||||||
}}
|
|
||||||
open={Boolean(openMenu)}
|
|
||||||
onClose={handleCloseMenu}
|
|
||||||
sx={{mt: 2}}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
|
|
||||||
// Styles for the navbar icons
|
// Styles for the navbar icons
|
||||||
const iconsStyle = ({
|
const iconsStyle = ({
|
||||||
palette: {dark, white, text},
|
palette: {dark, white, text},
|
||||||
@ -240,26 +234,22 @@ function NavBar({absolute, light, isMini}: Props): JSX.Element
|
|||||||
>
|
>
|
||||||
<Toolbar sx={navbarContainer}>
|
<Toolbar sx={navbarContainer}>
|
||||||
<Box color="inherit" mb={{xs: 1, md: 0}} sx={(theme) => navbarRow(theme, {isMini})}>
|
<Box color="inherit" mb={{xs: 1, md: 0}} sx={(theme) => navbarRow(theme, {isMini})}>
|
||||||
|
<IconButton
|
||||||
|
size="small"
|
||||||
|
disableRipple
|
||||||
|
color="inherit"
|
||||||
|
sx={navbarMobileMenu}
|
||||||
|
onClick={handleMiniSidenav}
|
||||||
|
>
|
||||||
|
<Icon sx={iconsStyle} fontSize="large">menu</Icon>
|
||||||
|
</IconButton>
|
||||||
<QBreadcrumbs icon="home" title={breadcrumbTitle} route={route} light={light} />
|
<QBreadcrumbs icon="home" title={breadcrumbTitle} route={route} light={light} />
|
||||||
</Box>
|
</Box>
|
||||||
{isMini ? null : (
|
{isMini ? null : (
|
||||||
<Box sx={(theme) => navbarRow(theme, {isMini})}>
|
<Box sx={(theme) => navbarRow(theme, {isMini})}>
|
||||||
<Box pr={1}>
|
<Box pr={0} mr={-2} mt={-4}>
|
||||||
{renderHistory()}
|
{renderHistory()}
|
||||||
</Box>
|
</Box>
|
||||||
<Box color={light ? "white" : "inherit"}>
|
|
||||||
<IconButton
|
|
||||||
size="small"
|
|
||||||
color="inherit"
|
|
||||||
sx={navbarIconButton}
|
|
||||||
onClick={handleOpenMenu}
|
|
||||||
>
|
|
||||||
<Badge badgeContent={0} color="error" variant="dot">
|
|
||||||
<Icon sx={iconsStyle}>notifications</Icon>
|
|
||||||
</Badge>
|
|
||||||
</IconButton>
|
|
||||||
{renderMenu()}
|
|
||||||
</Box>
|
|
||||||
</Box>
|
</Box>
|
||||||
)}
|
)}
|
||||||
</Toolbar>
|
</Toolbar>
|
||||||
|
@ -20,6 +20,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import {Theme} from "@mui/material/styles";
|
import {Theme} from "@mui/material/styles";
|
||||||
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
|
|
||||||
function navbar(theme: Theme | any, ownerState: any)
|
function navbar(theme: Theme | any, ownerState: any)
|
||||||
{
|
{
|
||||||
@ -110,11 +111,10 @@ const navbarContainer = ({breakpoints}: Theme): any => ({
|
|||||||
const navbarRow = ({breakpoints}: Theme, {isMini}: any) => ({
|
const navbarRow = ({breakpoints}: Theme, {isMini}: any) => ({
|
||||||
display: "flex",
|
display: "flex",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
justifyContent: "space-between",
|
|
||||||
width: "100%",
|
width: "100%",
|
||||||
|
|
||||||
[breakpoints.up("md")]: {
|
[breakpoints.up("md")]: {
|
||||||
justifyContent: isMini ? "space-between" : "stretch",
|
justifyContent: "stretch",
|
||||||
width: isMini ? "100%" : "max-content",
|
width: isMini ? "100%" : "max-content",
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -146,12 +146,37 @@ const navbarDesktopMenu = ({breakpoints}: Theme) => ({
|
|||||||
display: "none !important",
|
display: "none !important",
|
||||||
cursor: "pointer",
|
cursor: "pointer",
|
||||||
|
|
||||||
[breakpoints.up("xl")]: {
|
[breakpoints.down("sm")]: {
|
||||||
display: "inline-block !important",
|
display: "inline-block !important",
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const recentlyViewedMenu = ({breakpoints}: Theme) => ({
|
||||||
|
"& .MuiInputLabel-root": {
|
||||||
|
color: colors.gray.main,
|
||||||
|
fontWeight: "500",
|
||||||
|
fontSize: "1rem"
|
||||||
|
},
|
||||||
|
"& .MuiInputAdornment-root": {
|
||||||
|
marginTop: "0.5rem",
|
||||||
|
color: colors.gray.main,
|
||||||
|
fontSize: "1rem"
|
||||||
|
},
|
||||||
|
"& .MuiOutlinedInput-root": {
|
||||||
|
borderRadius: "0",
|
||||||
|
padding: "0"
|
||||||
|
},
|
||||||
|
"& .MuiOutlinedInput-root .MuiOutlinedInput-notchedOutline": {
|
||||||
|
border: "0"
|
||||||
|
},
|
||||||
|
display: "block",
|
||||||
|
[breakpoints.down("md")]: {
|
||||||
|
display: "none !important",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
const navbarMobileMenu = ({breakpoints}: Theme) => ({
|
const navbarMobileMenu = ({breakpoints}: Theme) => ({
|
||||||
|
left: "-0.75rem",
|
||||||
display: "inline-block",
|
display: "inline-block",
|
||||||
lineHeight: 0,
|
lineHeight: 0,
|
||||||
|
|
||||||
@ -167,4 +192,5 @@ export {
|
|||||||
navbarIconButton,
|
navbarIconButton,
|
||||||
navbarDesktopMenu,
|
navbarDesktopMenu,
|
||||||
navbarMobileMenu,
|
navbarMobileMenu,
|
||||||
|
recentlyViewedMenu
|
||||||
};
|
};
|
||||||
|
@ -27,7 +27,7 @@ export default styled(Drawer)(({theme, ownerState}: { theme?: Theme | any; owner
|
|||||||
const {palette, boxShadows, transitions, breakpoints, functions} = theme;
|
const {palette, boxShadows, transitions, breakpoints, functions} = theme;
|
||||||
const {transparentSidenav, whiteSidenav, miniSidenav, darkMode} = ownerState;
|
const {transparentSidenav, whiteSidenav, miniSidenav, darkMode} = ownerState;
|
||||||
|
|
||||||
const sidebarWidth = 250;
|
const sidebarWidth = 245;
|
||||||
const {transparent, gradients, white, background} = palette;
|
const {transparent, gradients, white, background} = palette;
|
||||||
const {xxl} = boxShadows;
|
const {xxl} = boxShadows;
|
||||||
const {pxToRem, linearGradient} = functions;
|
const {pxToRem, linearGradient} = functions;
|
||||||
@ -94,6 +94,9 @@ export default styled(Drawer)(({theme, ownerState}: { theme?: Theme | any; owner
|
|||||||
"& .MuiDrawer-paper": {
|
"& .MuiDrawer-paper": {
|
||||||
boxShadow: xxl,
|
boxShadow: xxl,
|
||||||
border: "none",
|
border: "none",
|
||||||
|
margin: "0",
|
||||||
|
borderRadius: "0",
|
||||||
|
height: "100%",
|
||||||
|
|
||||||
...(miniSidenav ? drawerCloseStyles() : drawerOpenStyles()),
|
...(miniSidenav ? drawerCloseStyles() : drawerOpenStyles()),
|
||||||
},
|
},
|
||||||
|
@ -64,7 +64,8 @@ function collapseItem(theme: Theme, ownerState: any)
|
|||||||
borderRadius: borderRadius.md,
|
borderRadius: borderRadius.md,
|
||||||
cursor: "pointer",
|
cursor: "pointer",
|
||||||
userSelect: "none",
|
userSelect: "none",
|
||||||
whiteSpace: "nowrap",
|
whiteSpace: "wrap",
|
||||||
|
overflow: "hidden",
|
||||||
boxShadow: active && !whiteSidenav && !darkMode && !transparentSidenav ? md : "none",
|
boxShadow: active && !whiteSidenav && !darkMode && !transparentSidenav ? md : "none",
|
||||||
[breakpoints.up("xl")]: {
|
[breakpoints.up("xl")]: {
|
||||||
transition: transitions.create(["box-shadow", "background-color"], {
|
transition: transitions.create(["box-shadow", "background-color"], {
|
||||||
@ -73,6 +74,10 @@ function collapseItem(theme: Theme, ownerState: any)
|
|||||||
}),
|
}),
|
||||||
},
|
},
|
||||||
|
|
||||||
|
"& .MuiListItemText-primary": {
|
||||||
|
lineHeight: "revert"
|
||||||
|
},
|
||||||
|
|
||||||
"&:hover, &:focus": {
|
"&:hover, &:focus": {
|
||||||
backgroundColor:
|
backgroundColor:
|
||||||
transparentSidenav && !darkMode
|
transparentSidenav && !darkMode
|
||||||
|
@ -69,7 +69,15 @@ export default styled(TextField)(({theme, ownerState}: { theme?: Theme; ownerSta
|
|||||||
});
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
"& .MuiInputBase-root": {
|
||||||
backgroundColor: disabled ? `${grey[200]} !important` : transparent.main,
|
backgroundColor: disabled ? `${grey[200]} !important` : transparent.main,
|
||||||
|
borderRadius: "0.75rem",
|
||||||
|
},
|
||||||
|
"& input": {
|
||||||
|
backgroundColor: `${transparent.main}!important`,
|
||||||
|
padding: "0.5rem",
|
||||||
|
fontSize: "1rem",
|
||||||
|
},
|
||||||
pointerEvents: disabled ? "none" : "auto",
|
pointerEvents: disabled ? "none" : "auto",
|
||||||
...(error && errorStyles()),
|
...(error && errorStyles()),
|
||||||
...(success && successStyles()),
|
...(success && successStyles()),
|
||||||
|
@ -149,7 +149,7 @@ interface Types {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const baseProperties = {
|
const baseProperties = {
|
||||||
fontFamily: '"Roboto", "Helvetica", "Arial", sans-serif',
|
fontFamily: '"SF Pro Display", "Roboto", "Helvetica", "Arial", sans-serif',
|
||||||
fontWeightLighter: 100,
|
fontWeightLighter: 100,
|
||||||
fontWeightLight: 300,
|
fontWeightLight: 300,
|
||||||
fontWeightRegular: 400,
|
fontWeightRegular: 400,
|
||||||
|
@ -153,7 +153,7 @@ interface Types
|
|||||||
}
|
}
|
||||||
|
|
||||||
const baseProperties = {
|
const baseProperties = {
|
||||||
fontFamily: "\"Roboto\", \"Helvetica\", \"Arial\", sans-serif",
|
fontFamily: "\"SF Pro Display\", \"Roboto\", \"Helvetica\", \"Arial\", sans-serif",
|
||||||
fontWeightLighter: 100,
|
fontWeightLighter: 100,
|
||||||
fontWeightLight: 300,
|
fontWeightLight: 300,
|
||||||
fontWeightRegular: 400,
|
fontWeightRegular: 400,
|
||||||
|
139
src/qqq/components/misc/HelpContent.tsx
Normal file
139
src/qqq/components/misc/HelpContent.tsx
Normal file
@ -0,0 +1,139 @@
|
|||||||
|
/*
|
||||||
|
* QQQ - Low-code Application Framework for Engineers.
|
||||||
|
* Copyright (C) 2021-2023. 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 {QHelpContent} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QHelpContent";
|
||||||
|
import Box from "@mui/material/Box";
|
||||||
|
import parse from "html-react-parser";
|
||||||
|
import React, {useContext} from "react";
|
||||||
|
import Markdown from "react-markdown";
|
||||||
|
import QContext from "QContext";
|
||||||
|
|
||||||
|
interface Props
|
||||||
|
{
|
||||||
|
helpContents: QHelpContent[];
|
||||||
|
roles: string[];
|
||||||
|
heading?: string;
|
||||||
|
helpContentKey?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
HelpContent.defaultProps = {};
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** format some content - meaning, change it from string to JSX element(s) or string.
|
||||||
|
** does a parse() for HTML, and a <Markdown> for markdown, else just text.
|
||||||
|
*******************************************************************************/
|
||||||
|
const formatHelpContent = (content: string, format: string): string | JSX.Element | JSX.Element[] =>
|
||||||
|
{
|
||||||
|
if (format == "HTML")
|
||||||
|
{
|
||||||
|
return parse(content);
|
||||||
|
}
|
||||||
|
else if (format == "MARKDOWN")
|
||||||
|
{
|
||||||
|
return (<Markdown>{content}</Markdown>)
|
||||||
|
}
|
||||||
|
|
||||||
|
return content;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** return the first help content from the list that matches the first role
|
||||||
|
** in the roles list.
|
||||||
|
*******************************************************************************/
|
||||||
|
const getMatchingHelpContent = (helpContents: QHelpContent[], roles: string[]): QHelpContent =>
|
||||||
|
{
|
||||||
|
if (helpContents)
|
||||||
|
{
|
||||||
|
if (helpContents.length == 1 && helpContents[0].roles.size == 0)
|
||||||
|
{
|
||||||
|
//////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// if there's only 1 entry, and it has no roles, then assume user wanted it globally and use it //
|
||||||
|
//////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
return (helpContents[0]);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for (let i = 0; i < roles.length; i++)
|
||||||
|
{
|
||||||
|
for (let j = 0; j < helpContents.length; j++)
|
||||||
|
{
|
||||||
|
if (helpContents[j].roles.has(roles[i]))
|
||||||
|
{
|
||||||
|
return(helpContents[j])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (null);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** test if a list of help contents would find any matches from a list of roles.
|
||||||
|
*******************************************************************************/
|
||||||
|
export const hasHelpContent = (helpContents: QHelpContent[], roles: string[]) =>
|
||||||
|
{
|
||||||
|
return getMatchingHelpContent(helpContents, roles) != null;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** component that renders a box of formatted help content, from a list of
|
||||||
|
** helpContents (from meta-data), and for a list of roles (based on what screen
|
||||||
|
*******************************************************************************/
|
||||||
|
function HelpContent({helpContents, roles, heading, helpContentKey}: Props): JSX.Element
|
||||||
|
{
|
||||||
|
const {helpHelpActive} = useContext(QContext);
|
||||||
|
let selectedHelpContent = getMatchingHelpContent(helpContents, roles);
|
||||||
|
|
||||||
|
let content = null;
|
||||||
|
if (helpHelpActive)
|
||||||
|
{
|
||||||
|
if (!selectedHelpContent)
|
||||||
|
{
|
||||||
|
selectedHelpContent = new QHelpContent({content: ""});
|
||||||
|
}
|
||||||
|
content = selectedHelpContent.content + ` [${helpContentKey ?? "?"}]`;
|
||||||
|
}
|
||||||
|
else if(selectedHelpContent)
|
||||||
|
{
|
||||||
|
content = selectedHelpContent.content;
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////
|
||||||
|
// if content was found, format it and return it //
|
||||||
|
///////////////////////////////////////////////////
|
||||||
|
if (content)
|
||||||
|
{
|
||||||
|
return <Box display="inline" className="helpContent">
|
||||||
|
{heading && <span className="header">{heading}</span>}
|
||||||
|
{formatHelpContent(content, selectedHelpContent.format)}
|
||||||
|
</Box>;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (null);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default HelpContent;
|
@ -76,7 +76,7 @@ function QRecordSidebar({tableSections, widgetMetaDataList, light, stickyTop}: P
|
|||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card sx={{borderRadius: ({borders: {borderRadius}}) => borderRadius.lg, position: "sticky", top: stickyTop}}>
|
<Card sx={{borderRadius: "0.75rem", position: "sticky", top: stickyTop, overflow: "auto", maxHeight: "calc(100vh - 200px)"}}>
|
||||||
<Box component="ul" display="flex" flexDirection="column" p={2} m={0} sx={{listStyle: "none"}}>
|
<Box component="ul" display="flex" flexDirection="column" p={2} m={0} sx={{listStyle: "none"}}>
|
||||||
{
|
{
|
||||||
sidebarEntries ? sidebarEntries.map((entry: SidebarEntry, key: number) => (
|
sidebarEntries ? sidebarEntries.map((entry: SidebarEntry, key: number) => (
|
||||||
|
@ -28,11 +28,12 @@ interface TabPanelProps
|
|||||||
children?: React.ReactNode;
|
children?: React.ReactNode;
|
||||||
index: number;
|
index: number;
|
||||||
value: number;
|
value: number;
|
||||||
|
style?: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function TabPanel(props: TabPanelProps)
|
export default function TabPanel(props: TabPanelProps)
|
||||||
{
|
{
|
||||||
const {children, value, index, ...other} = props;
|
const {children, value, index, style, ...other} = props;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
@ -40,6 +41,7 @@ export default function TabPanel(props: TabPanelProps)
|
|||||||
hidden={value !== index}
|
hidden={value !== index}
|
||||||
id={`simple-tabpanel-${index}`}
|
id={`simple-tabpanel-${index}`}
|
||||||
aria-labelledby={`simple-tab-${index}`}
|
aria-labelledby={`simple-tab-${index}`}
|
||||||
|
style={style}
|
||||||
{...other}
|
{...other}
|
||||||
>
|
>
|
||||||
{value === index && (
|
{value === index && (
|
||||||
|
@ -155,7 +155,7 @@ function ValidationReview({
|
|||||||
"false",
|
"false",
|
||||||
"Skip Validation. Submit the records for immediate processing", (
|
"Skip Validation. Submit the records for immediate processing", (
|
||||||
<div>
|
<div>
|
||||||
If you choose this option, the records input records will immediately be processed.
|
If you choose this option, the input records will immediately be processed.
|
||||||
You will be told how many records were successfully processed, and which ones had issues after the processing is completed.
|
You will be told how many records were successfully processed, and which ones had issues after the processing is completed.
|
||||||
<br />
|
<br />
|
||||||
<br />
|
<br />
|
||||||
|
@ -215,6 +215,7 @@ function FilterCriteriaRowValues({operatorOption, criteria, field, table, valueC
|
|||||||
initialDisplayValue={selectedPossibleValue?.label}
|
initialDisplayValue={selectedPossibleValue?.label}
|
||||||
inForm={false}
|
inForm={false}
|
||||||
onChange={(value: any) => valueChangeHandler(null, 0, value)}
|
onChange={(value: any) => valueChangeHandler(null, 0, value)}
|
||||||
|
variant="standard"
|
||||||
/>
|
/>
|
||||||
</Box>;
|
</Box>;
|
||||||
case ValueMode.PVS_MULTI:
|
case ValueMode.PVS_MULTI:
|
||||||
@ -242,6 +243,7 @@ function FilterCriteriaRowValues({operatorOption, criteria, field, table, valueC
|
|||||||
initialValues={initialValues}
|
initialValues={initialValues}
|
||||||
inForm={false}
|
inForm={false}
|
||||||
onChange={(value: any) => valueChangeHandler(null, "all", value)}
|
onChange={(value: any) => valueChangeHandler(null, "all", value)}
|
||||||
|
variant="standard"
|
||||||
/>
|
/>
|
||||||
</Box>;
|
</Box>;
|
||||||
}
|
}
|
||||||
|
@ -22,11 +22,13 @@ import {QWidgetMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/Q
|
|||||||
import {Skeleton} from "@mui/material";
|
import {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";
|
||||||
|
import Tab from "@mui/material/Tab";
|
||||||
|
import Tabs from "@mui/material/Tabs";
|
||||||
import parse from "html-react-parser";
|
import parse from "html-react-parser";
|
||||||
import React, {useContext, useEffect, useReducer, useState} from "react";
|
import React, {useContext, useEffect, useReducer, useState} from "react";
|
||||||
import {useLocation} from "react-router-dom";
|
|
||||||
import QContext from "QContext";
|
import QContext from "QContext";
|
||||||
import MDTypography from "qqq/components/legacy/MDTypography";
|
import MDTypography from "qqq/components/legacy/MDTypography";
|
||||||
|
import TabPanel from "qqq/components/misc/TabPanel";
|
||||||
import BarChart from "qqq/components/widgets/charts/barchart/BarChart";
|
import BarChart from "qqq/components/widgets/charts/barchart/BarChart";
|
||||||
import HorizontalBarChart from "qqq/components/widgets/charts/barchart/HorizontalBarChart";
|
import HorizontalBarChart from "qqq/components/widgets/charts/barchart/HorizontalBarChart";
|
||||||
import DefaultLineChart from "qqq/components/widgets/charts/linechart/DefaultLineChart";
|
import DefaultLineChart from "qqq/components/widgets/charts/linechart/DefaultLineChart";
|
||||||
@ -44,7 +46,7 @@ import USMapWidget from "qqq/components/widgets/misc/USMapWidget";
|
|||||||
import ParentWidget from "qqq/components/widgets/ParentWidget";
|
import ParentWidget from "qqq/components/widgets/ParentWidget";
|
||||||
import MultiStatisticsCard from "qqq/components/widgets/statistics/MultiStatisticsCard";
|
import MultiStatisticsCard from "qqq/components/widgets/statistics/MultiStatisticsCard";
|
||||||
import StatisticsCard from "qqq/components/widgets/statistics/StatisticsCard";
|
import StatisticsCard from "qqq/components/widgets/statistics/StatisticsCard";
|
||||||
import Widget, {WIDGET_DROPDOWN_SELECTION_LOCAL_STORAGE_KEY_ROOT} from "qqq/components/widgets/Widget";
|
import Widget, {HeaderIcon, WIDGET_DROPDOWN_SELECTION_LOCAL_STORAGE_KEY_ROOT, LabelComponent} from "qqq/components/widgets/Widget";
|
||||||
import ProcessRun from "qqq/pages/processes/ProcessRun";
|
import ProcessRun from "qqq/pages/processes/ProcessRun";
|
||||||
import Client from "qqq/utils/qqq/Client";
|
import Client from "qqq/utils/qqq/Client";
|
||||||
import TableWidget from "./tables/TableWidget";
|
import TableWidget from "./tables/TableWidget";
|
||||||
@ -58,9 +60,10 @@ interface Props
|
|||||||
tableName?: string;
|
tableName?: string;
|
||||||
entityPrimaryKey?: string;
|
entityPrimaryKey?: string;
|
||||||
omitWrappingGridContainer: boolean;
|
omitWrappingGridContainer: boolean;
|
||||||
areChildren?: boolean
|
areChildren?: boolean;
|
||||||
childUrlParams?: string
|
childUrlParams?: string;
|
||||||
parentWidgetMetaData?: QWidgetMetaData
|
parentWidgetMetaData?: QWidgetMetaData;
|
||||||
|
wrapWidgetsInTabPanels: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
DashboardWidgets.defaultProps = {
|
DashboardWidgets.defaultProps = {
|
||||||
@ -70,12 +73,12 @@ DashboardWidgets.defaultProps = {
|
|||||||
omitWrappingGridContainer: false,
|
omitWrappingGridContainer: false,
|
||||||
areChildren: false,
|
areChildren: false,
|
||||||
childUrlParams: "",
|
childUrlParams: "",
|
||||||
parentWidgetMetaData: null
|
parentWidgetMetaData: null,
|
||||||
|
wrapWidgetsInTabPanels: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
function DashboardWidgets({widgetMetaDataList, tableName, entityPrimaryKey, omitWrappingGridContainer, areChildren, childUrlParams, parentWidgetMetaData}: Props): JSX.Element
|
function DashboardWidgets({widgetMetaDataList, tableName, entityPrimaryKey, omitWrappingGridContainer, areChildren, childUrlParams, parentWidgetMetaData, wrapWidgetsInTabPanels}: Props): JSX.Element
|
||||||
{
|
{
|
||||||
const location = useLocation();
|
|
||||||
const [widgetData, setWidgetData] = useState([] as any[]);
|
const [widgetData, setWidgetData] = useState([] as any[]);
|
||||||
const [widgetCounter, setWidgetCounter] = useState(0);
|
const [widgetCounter, setWidgetCounter] = useState(0);
|
||||||
const [, forceUpdate] = useReducer((x) => x + 1, 0);
|
const [, forceUpdate] = useReducer((x) => x + 1, 0);
|
||||||
@ -84,6 +87,24 @@ function DashboardWidgets({widgetMetaDataList, tableName, entityPrimaryKey, omit
|
|||||||
const [haveLoadedParams, setHaveLoadedParams] = useState(false);
|
const [haveLoadedParams, setHaveLoadedParams] = useState(false);
|
||||||
const {accentColor} = useContext(QContext);
|
const {accentColor} = useContext(QContext);
|
||||||
|
|
||||||
|
let initialSelectedTab = 0;
|
||||||
|
let selectedTabKey: string = null;
|
||||||
|
if(parentWidgetMetaData && wrapWidgetsInTabPanels)
|
||||||
|
{
|
||||||
|
selectedTabKey = `qqq.widgets.selectedTabs.${parentWidgetMetaData.name}`
|
||||||
|
if (localStorage.getItem(selectedTabKey))
|
||||||
|
{
|
||||||
|
initialSelectedTab = Number(localStorage.getItem(selectedTabKey));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const [selectedTab, setSelectedTab] = useState(initialSelectedTab);
|
||||||
|
|
||||||
|
const changeTab = (newValue: number) =>
|
||||||
|
{
|
||||||
|
setSelectedTab(newValue);
|
||||||
|
localStorage.setItem(selectedTabKey, String(newValue));
|
||||||
|
};
|
||||||
|
|
||||||
useEffect(() =>
|
useEffect(() =>
|
||||||
{
|
{
|
||||||
setWidgetData([]);
|
setWidgetData([]);
|
||||||
@ -151,7 +172,7 @@ function DashboardWidgets({widgetMetaDataList, tableName, entityPrimaryKey, omit
|
|||||||
|
|
||||||
forceUpdate();
|
forceUpdate();
|
||||||
})();
|
})();
|
||||||
}
|
};
|
||||||
|
|
||||||
function getQueryParams(widgetMetaData: QWidgetMetaData, extraParams: string): string
|
function getQueryParams(widgetMetaData: QWidgetMetaData, extraParams: string): string
|
||||||
{
|
{
|
||||||
@ -227,6 +248,16 @@ function DashboardWidgets({widgetMetaDataList, tableName, entityPrimaryKey, omit
|
|||||||
|
|
||||||
const renderWidget = (widgetMetaData: QWidgetMetaData, i: number): JSX.Element =>
|
const renderWidget = (widgetMetaData: QWidgetMetaData, i: number): JSX.Element =>
|
||||||
{
|
{
|
||||||
|
const labelAdditionalComponentsRight: LabelComponent[] = [];
|
||||||
|
if (widgetMetaData && widgetMetaData.icons)
|
||||||
|
{
|
||||||
|
const topRightInsideCardIcon = widgetMetaData.icons.get("topRightInsideCard");
|
||||||
|
if (topRightInsideCardIcon)
|
||||||
|
{
|
||||||
|
labelAdditionalComponentsRight.push(new HeaderIcon(topRightInsideCardIcon.name, topRightInsideCardIcon.path, topRightInsideCardIcon.color));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box key={`${widgetMetaData.name}-${i}`} sx={{alignItems: "stretch", flexGrow: 1, display: "flex", marginTop: "0px", paddingTop: "0px", width: "100%", height: "100%"}}>
|
<Box key={`${widgetMetaData.name}-${i}`} sx={{alignItems: "stretch", flexGrow: 1, display: "flex", marginTop: "0px", paddingTop: "0px", width: "100%", height: "100%"}}>
|
||||||
{
|
{
|
||||||
@ -238,7 +269,7 @@ function DashboardWidgets({widgetMetaDataList, tableName, entityPrimaryKey, omit
|
|||||||
widgetIndex={i}
|
widgetIndex={i}
|
||||||
widgetMetaData={widgetMetaData}
|
widgetMetaData={widgetMetaData}
|
||||||
data={widgetData[i]}
|
data={widgetData[i]}
|
||||||
reloadWidgetCallback={reloadWidget}
|
reloadWidgetCallback={(data) => reloadWidget(i, data)}
|
||||||
storeDropdownSelections={widgetMetaData.storeDropdownSelections}
|
storeDropdownSelections={widgetMetaData.storeDropdownSelections}
|
||||||
/>
|
/>
|
||||||
)
|
)
|
||||||
@ -270,8 +301,9 @@ function DashboardWidgets({widgetMetaDataList, tableName, entityPrimaryKey, omit
|
|||||||
widgetData={widgetData[i]}
|
widgetData={widgetData[i]}
|
||||||
reloadWidgetCallback={(data) => reloadWidget(i, data)}
|
reloadWidgetCallback={(data) => reloadWidget(i, data)}
|
||||||
isChild={areChildren}
|
isChild={areChildren}
|
||||||
|
labelAdditionalComponentsRight={labelAdditionalComponentsRight}
|
||||||
>
|
>
|
||||||
<StackedBarChart data={widgetData[i]?.chartData}/>
|
<StackedBarChart data={widgetData[i]?.chartData} chartSubheaderData={widgetData[i]?.chartSubheaderData} />
|
||||||
</Widget>
|
</Widget>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@ -311,7 +343,7 @@ function DashboardWidgets({widgetMetaDataList, tableName, entityPrimaryKey, omit
|
|||||||
reloadWidgetCallback={(data) => reloadWidget(i, data)}
|
reloadWidgetCallback={(data) => reloadWidget(i, data)}
|
||||||
widgetData={widgetData[i]}
|
widgetData={widgetData[i]}
|
||||||
>
|
>
|
||||||
<Box px={3} pt={0} pb={2}>
|
<Box>
|
||||||
<MDTypography component="div" variant="button" color="text" fontWeight="light">
|
<MDTypography component="div" variant="button" color="text" fontWeight="light">
|
||||||
{
|
{
|
||||||
widgetData && widgetData[i] && widgetData[i].html ? (
|
widgetData && widgetData[i] && widgetData[i].html ? (
|
||||||
@ -381,10 +413,12 @@ function DashboardWidgets({widgetMetaDataList, tableName, entityPrimaryKey, omit
|
|||||||
widgetData={widgetData[i]}
|
widgetData={widgetData[i]}
|
||||||
reloadWidgetCallback={(data) => reloadWidget(i, data)}
|
reloadWidgetCallback={(data) => reloadWidget(i, data)}
|
||||||
isChild={areChildren}
|
isChild={areChildren}
|
||||||
|
labelAdditionalComponentsRight={labelAdditionalComponentsRight}
|
||||||
>
|
>
|
||||||
<div>
|
<div>
|
||||||
<PieChart
|
<PieChart
|
||||||
chartData={widgetData[i]?.chartData}
|
chartData={widgetData[i]?.chartData}
|
||||||
|
chartSubheaderData={widgetData[i]?.chartSubheaderData}
|
||||||
description={widgetData[i]?.description}
|
description={widgetData[i]?.description}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
@ -461,32 +495,76 @@ function DashboardWidgets({widgetMetaDataList, tableName, entityPrimaryKey, omit
|
|||||||
}
|
}
|
||||||
</Box>
|
</Box>
|
||||||
);
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
if(wrapWidgetsInTabPanels)
|
||||||
|
{
|
||||||
|
omitWrappingGridContainer = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
const body: JSX.Element =
|
const body: JSX.Element =
|
||||||
(
|
(
|
||||||
<>
|
<>
|
||||||
{
|
{
|
||||||
widgetMetaDataList.map((widgetMetaData, i) => (
|
widgetMetaDataList.map((widgetMetaData, i) =>
|
||||||
omitWrappingGridContainer
|
{
|
||||||
? widgetMetaData && renderWidget(widgetMetaData, i)
|
let renderedWidget = widgetMetaData ? renderWidget(widgetMetaData, i) : (<></>);
|
||||||
:
|
|
||||||
widgetMetaData && <Grid id={widgetMetaData.name} key={`${widgetMetaData.name}-${i}`} item lg={widgetMetaData.gridColumns ? widgetMetaData.gridColumns : 12} xs={12} sx={{display: "flex", alignItems: "stretch", scrollMarginTop: "100px"}}>
|
if (!omitWrappingGridContainer)
|
||||||
{renderWidget(widgetMetaData, i)}
|
{
|
||||||
</Grid>
|
// @ts-ignore
|
||||||
))
|
renderedWidget = (<Grid id={widgetMetaData.name} item xxl={widgetMetaData.gridColumns ? widgetMetaData.gridColumns : 12} xs={12} sx={{display: "flex", alignItems: "stretch", scrollMarginTop: "100px"}}>
|
||||||
|
{renderedWidget}
|
||||||
|
</Grid>);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (wrapWidgetsInTabPanels)
|
||||||
|
{
|
||||||
|
renderedWidget = (<TabPanel index={i} value={selectedTab} style={{
|
||||||
|
padding: 0,
|
||||||
|
margin: "-1rem",
|
||||||
|
marginBottom: "-3.5rem",
|
||||||
|
width: "calc(100% + 2rem)"
|
||||||
|
}}>
|
||||||
|
{renderedWidget}
|
||||||
|
</TabPanel>);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (<React.Fragment key={`${widgetMetaData.name}-${i}`}>{renderedWidget}</React.Fragment>)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const tabs = widgetMetaDataList && wrapWidgetsInTabPanels ?
|
||||||
|
<Tabs
|
||||||
|
sx={{m: 0, mb: 1.5, ml: -2, mr: -2, mt: -3,
|
||||||
|
"& .MuiTabs-scroller": {
|
||||||
|
ml: 0
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
value={selectedTab}
|
||||||
|
onChange={(event, newValue) => changeTab(newValue)}
|
||||||
|
variant="standard"
|
||||||
|
>
|
||||||
|
{widgetMetaDataList.map((widgetMetaData, i) => (
|
||||||
|
<Tab key={widgetMetaData.name} label={widgetMetaData.label} />
|
||||||
|
))}
|
||||||
|
</Tabs>
|
||||||
|
: <></>
|
||||||
|
|
||||||
return (
|
return (
|
||||||
widgetCount > 0 ? (
|
widgetCount > 0 ? (
|
||||||
omitWrappingGridContainer ? body :
|
<>
|
||||||
(
|
{tabs}
|
||||||
<Grid container spacing={3} pb={4}>
|
{
|
||||||
|
omitWrappingGridContainer ? body : (
|
||||||
|
<Grid container spacing={2.5}>
|
||||||
{body}
|
{body}
|
||||||
</Grid>
|
</Grid>
|
||||||
)
|
)
|
||||||
|
}
|
||||||
|
</>
|
||||||
) : null
|
) : null
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -43,6 +43,7 @@ export interface ParentWidgetData
|
|||||||
dropdownNeedsSelectedText?: string;
|
dropdownNeedsSelectedText?: string;
|
||||||
storeDropdownSelections?: boolean;
|
storeDropdownSelections?: boolean;
|
||||||
icon?: string;
|
icon?: string;
|
||||||
|
layoutType: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -55,7 +56,7 @@ interface Props
|
|||||||
widgetMetaData?: QWidgetMetaData;
|
widgetMetaData?: QWidgetMetaData;
|
||||||
widgetIndex: number;
|
widgetIndex: number;
|
||||||
data: ParentWidgetData;
|
data: ParentWidgetData;
|
||||||
reloadWidgetCallback?: (widgetIndex: number, params: string) => void;
|
reloadWidgetCallback?: (params: string) => void;
|
||||||
entityPrimaryKey?: string;
|
entityPrimaryKey?: string;
|
||||||
tableName?: string;
|
tableName?: string;
|
||||||
storeDropdownSelections?: boolean;
|
storeDropdownSelections?: boolean;
|
||||||
@ -91,18 +92,29 @@ function ParentWidget({urlParams, widgetMetaData, widgetIndex, data, reloadWidge
|
|||||||
}
|
}
|
||||||
}, [qInstance, data, childUrlParams]);
|
}, [qInstance, data, childUrlParams]);
|
||||||
|
|
||||||
|
useEffect(() =>
|
||||||
|
{
|
||||||
|
setChildUrlParams(urlParams)
|
||||||
|
}, [urlParams]);
|
||||||
|
|
||||||
const parentReloadWidgetCallback = (data: string) =>
|
const parentReloadWidgetCallback = (data: string) =>
|
||||||
{
|
{
|
||||||
setChildUrlParams(data);
|
setChildUrlParams(data);
|
||||||
reloadWidgetCallback(widgetIndex, data);
|
reloadWidgetCallback(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// if this parent widget is in card form, and its children are too, then we need some px //
|
// if this parent widget is in card form, and its children are too, then we need some px //
|
||||||
///////////////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////////////
|
||||||
const px = (widgetMetaData && widgetMetaData.isCard && widgets && widgets[0] && widgets[0].isCard) ? 3 : 0;
|
const parentIsCard = widgetMetaData && widgetMetaData.isCard;
|
||||||
|
const childrenAreCards = widgetMetaData && widgets && widgets[0] && widgets[0].isCard;
|
||||||
|
const px = (parentIsCard && childrenAreCards) ? 3 : 0;
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// if this is a parent, which is not a card, then we need to omit the padding, i think, on the Widget that ultimately gets rendered //
|
||||||
|
//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
const omitPadding = !parentIsCard;
|
||||||
|
|
||||||
// @ts-ignore
|
|
||||||
return (
|
return (
|
||||||
qInstance && data ? (
|
qInstance && data ? (
|
||||||
<Widget
|
<Widget
|
||||||
@ -110,9 +122,10 @@ function ParentWidget({urlParams, widgetMetaData, widgetIndex, data, reloadWidge
|
|||||||
widgetData={data}
|
widgetData={data}
|
||||||
storeDropdownSelections={storeDropdownSelections}
|
storeDropdownSelections={storeDropdownSelections}
|
||||||
reloadWidgetCallback={parentReloadWidgetCallback}
|
reloadWidgetCallback={parentReloadWidgetCallback}
|
||||||
|
omitPadding={omitPadding}
|
||||||
>
|
>
|
||||||
<Box sx={{height: "100%", width: "100%"}} px={px}>
|
<Box sx={{height: "100%", width: "100%"}} px={px}>
|
||||||
<DashboardWidgets widgetMetaDataList={widgets} entityPrimaryKey={entityPrimaryKey} tableName={tableName} childUrlParams={childUrlParams} areChildren={true} parentWidgetMetaData={widgetMetaData}/>
|
<DashboardWidgets widgetMetaDataList={widgets} entityPrimaryKey={entityPrimaryKey} tableName={tableName} childUrlParams={childUrlParams} areChildren={true} parentWidgetMetaData={widgetMetaData} wrapWidgetsInTabPanels={data.layoutType == "TABS"}/>
|
||||||
</Box>
|
</Box>
|
||||||
</Widget>
|
</Widget>
|
||||||
) : null
|
) : null
|
||||||
|
@ -25,14 +25,13 @@ 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 LinearProgress from "@mui/material/LinearProgress";
|
|
||||||
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";
|
||||||
import React, {useEffect, useState} from "react";
|
import React, {useEffect, useState} from "react";
|
||||||
import {NavigateFunction, useNavigate} from "react-router-dom";
|
import {NavigateFunction, useNavigate} from "react-router-dom";
|
||||||
import colors from "qqq/components/legacy/colors";
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
import DropdownMenu, {DropdownOption} from "qqq/components/widgets/components/DropdownMenu";
|
import WidgetDropdownMenu, {DropdownOption} from "qqq/components/widgets/components/WidgetDropdownMenu";
|
||||||
|
|
||||||
export interface WidgetData
|
export interface WidgetData
|
||||||
{
|
{
|
||||||
@ -43,6 +42,7 @@ export interface WidgetData
|
|||||||
id: string,
|
id: string,
|
||||||
label: string
|
label: string
|
||||||
}[][];
|
}[][];
|
||||||
|
dropdownDefaultValueList?: string[];
|
||||||
dropdownNeedsSelectedText?: string;
|
dropdownNeedsSelectedText?: string;
|
||||||
hasPermission?: boolean;
|
hasPermission?: boolean;
|
||||||
errorLoading?: boolean;
|
errorLoading?: boolean;
|
||||||
@ -56,6 +56,7 @@ interface Props
|
|||||||
labelAdditionalComponentsLeft: LabelComponent[];
|
labelAdditionalComponentsLeft: LabelComponent[];
|
||||||
labelAdditionalElementsLeft: JSX.Element[];
|
labelAdditionalElementsLeft: JSX.Element[];
|
||||||
labelAdditionalComponentsRight: LabelComponent[];
|
labelAdditionalComponentsRight: LabelComponent[];
|
||||||
|
labelBoxAdditionalSx?: any;
|
||||||
widgetMetaData?: QWidgetMetaData;
|
widgetMetaData?: QWidgetMetaData;
|
||||||
widgetData?: WidgetData;
|
widgetData?: WidgetData;
|
||||||
children: JSX.Element;
|
children: JSX.Element;
|
||||||
@ -64,6 +65,7 @@ interface Props
|
|||||||
isChild?: boolean;
|
isChild?: boolean;
|
||||||
footerHTML?: string;
|
footerHTML?: string;
|
||||||
storeDropdownSelections?: boolean;
|
storeDropdownSelections?: boolean;
|
||||||
|
omitPadding: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget.defaultProps = {
|
Widget.defaultProps = {
|
||||||
@ -74,6 +76,8 @@ Widget.defaultProps = {
|
|||||||
labelAdditionalComponentsLeft: [],
|
labelAdditionalComponentsLeft: [],
|
||||||
labelAdditionalElementsLeft: [],
|
labelAdditionalElementsLeft: [],
|
||||||
labelAdditionalComponentsRight: [],
|
labelAdditionalComponentsRight: [],
|
||||||
|
labelBoxAdditionalSx: {},
|
||||||
|
omitPadding: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -96,6 +100,54 @@ export class LabelComponent
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
**
|
||||||
|
*******************************************************************************/
|
||||||
|
export class HeaderIcon extends LabelComponent
|
||||||
|
{
|
||||||
|
iconName: string;
|
||||||
|
iconPath: string;
|
||||||
|
color: string;
|
||||||
|
coloredBG: boolean;
|
||||||
|
|
||||||
|
iconColor: string;
|
||||||
|
bgColor: string;
|
||||||
|
|
||||||
|
constructor(iconName: string, iconPath: string, color: string, coloredBG: boolean = true)
|
||||||
|
{
|
||||||
|
super();
|
||||||
|
this.iconName = iconName;
|
||||||
|
this.iconPath = iconPath;
|
||||||
|
this.color = color;
|
||||||
|
this.coloredBG = coloredBG;
|
||||||
|
|
||||||
|
this.iconColor = this.coloredBG ? "#FFFFFF" : this.color;
|
||||||
|
this.bgColor = this.coloredBG ? this.color : "none";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
render = (args: LabelComponentRenderArgs): JSX.Element =>
|
||||||
|
{
|
||||||
|
const styles = {
|
||||||
|
width: "1.75rem",
|
||||||
|
height: "1.75rem",
|
||||||
|
color: this.iconColor,
|
||||||
|
backgroundColor: this.bgColor,
|
||||||
|
borderRadius: "0.25rem"
|
||||||
|
};
|
||||||
|
|
||||||
|
if (this.iconPath)
|
||||||
|
{
|
||||||
|
return (<Box sx={{textAlign: "center", ...styles}}><img src={this.iconPath} width="16" height="16" /></Box>);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return (<Icon sx={{padding: "0.25rem", ...styles}} fontSize="small">{this.iconName}</Icon>);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
**
|
**
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
@ -124,7 +176,7 @@ export class AddNewRecordButton extends LabelComponent
|
|||||||
render = (args: LabelComponentRenderArgs): JSX.Element =>
|
render = (args: LabelComponentRenderArgs): JSX.Element =>
|
||||||
{
|
{
|
||||||
return (
|
return (
|
||||||
<Typography variant="body2" p={2} pr={0} display="inline" position="relative" top="0.25rem">
|
<Typography variant="body2" p={2} pr={0} display="inline" position="relative" top="-0.5rem">
|
||||||
<Button sx={{mt: 0.75}} onClick={() => this.openEditForm(args.navigate, this.table, null, this.defaultValues, this.disabledFields)}>{this.label}</Button>
|
<Button sx={{mt: 0.75}} onClick={() => this.openEditForm(args.navigate, this.table, null, this.defaultValues, this.disabledFields)}>{this.label}</Button>
|
||||||
</Typography>
|
</Typography>
|
||||||
);
|
);
|
||||||
@ -138,41 +190,111 @@ export class AddNewRecordButton extends LabelComponent
|
|||||||
export class Dropdown extends LabelComponent
|
export class Dropdown extends LabelComponent
|
||||||
{
|
{
|
||||||
label: string;
|
label: string;
|
||||||
|
dropdownMetaData: any;
|
||||||
options: DropdownOption[];
|
options: DropdownOption[];
|
||||||
|
dropdownDefaultValue?: string;
|
||||||
dropdownName: string;
|
dropdownName: string;
|
||||||
onChangeCallback: any;
|
onChangeCallback: any;
|
||||||
|
|
||||||
constructor(label: string, options: DropdownOption[], dropdownName: string, onChangeCallback: any)
|
constructor(label: string, dropdownMetaData: any, options: DropdownOption[], dropdownDefaultValue: string, dropdownName: string, onChangeCallback: any)
|
||||||
{
|
{
|
||||||
super();
|
super();
|
||||||
this.label = label;
|
this.label = label;
|
||||||
|
this.dropdownMetaData = dropdownMetaData;
|
||||||
this.options = options;
|
this.options = options;
|
||||||
|
this.dropdownDefaultValue = dropdownDefaultValue;
|
||||||
this.dropdownName = dropdownName;
|
this.dropdownName = dropdownName;
|
||||||
this.onChangeCallback = onChangeCallback;
|
this.onChangeCallback = onChangeCallback;
|
||||||
}
|
}
|
||||||
|
|
||||||
render = (args: LabelComponentRenderArgs): JSX.Element =>
|
render = (args: LabelComponentRenderArgs): JSX.Element =>
|
||||||
{
|
{
|
||||||
|
const label = `Select ${this.label}`;
|
||||||
let defaultValue = null;
|
let defaultValue = null;
|
||||||
const localStorageKey = `${WIDGET_DROPDOWN_SELECTION_LOCAL_STORAGE_KEY_ROOT}.${args.widgetProps.widgetMetaData.name}.${this.dropdownName}`;
|
const localStorageKey = `${WIDGET_DROPDOWN_SELECTION_LOCAL_STORAGE_KEY_ROOT}.${args.widgetProps.widgetMetaData.name}.${this.dropdownName}`;
|
||||||
if (args.widgetProps.storeDropdownSelections)
|
if (args.widgetProps.storeDropdownSelections)
|
||||||
{
|
{
|
||||||
///////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// see if an existing value is stored in local storage, and if so set it in dropdown //
|
// see if an existing value is stored in local storage, and if so set it in dropdown //
|
||||||
///////////////////////////////////////////////////////////////////////////////////////
|
// originally we used the full object from localStorage - but - in case the label //
|
||||||
defaultValue = JSON.parse(localStorage.getItem(localStorageKey));
|
// changed since it was stored, we'll instead just find the option by id (or in case that //
|
||||||
|
// option isn't available anymore, then we'll select nothing instead of a missing value //
|
||||||
|
////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
try
|
||||||
|
{
|
||||||
|
const localStorageOption = JSON.parse(localStorage.getItem(localStorageKey));
|
||||||
|
if(localStorageOption)
|
||||||
|
{
|
||||||
|
const id = localStorageOption.id;
|
||||||
|
for (let i = 0; i < this.options.length; i++)
|
||||||
|
{
|
||||||
|
if (this.options[i].id == id)
|
||||||
|
{
|
||||||
|
defaultValue = this.options[i]
|
||||||
args.dropdownData[args.componentIndex] = defaultValue?.id;
|
args.dropdownData[args.componentIndex] = defaultValue?.id;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch(e)
|
||||||
|
{
|
||||||
|
console.log(`Error getting default value for dropdown [${this.dropdownName}] from local storage`, e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// if there wasn't a value selected, but there is a default from the backend, then use it. //
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
if (defaultValue == null && this.dropdownDefaultValue != null)
|
||||||
|
{
|
||||||
|
for (let i = 0; i < this.options.length; i++)
|
||||||
|
{
|
||||||
|
if(this.options[i].id == this.dropdownDefaultValue)
|
||||||
|
{
|
||||||
|
defaultValue = this.options[i];
|
||||||
|
args.dropdownData[args.componentIndex] = defaultValue?.id;
|
||||||
|
|
||||||
|
if (args.widgetProps.storeDropdownSelections)
|
||||||
|
{
|
||||||
|
localStorage.setItem(localStorageKey, JSON.stringify(defaultValue));
|
||||||
|
}
|
||||||
|
|
||||||
|
this.onChangeCallback(label, defaultValue);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
// if there's a 'label for null value' (and no default from the backend), //
|
||||||
|
// then add that as an option (and select it if nothing else was selected) //
|
||||||
|
/////////////////////////////////////////////////////////////////////////////
|
||||||
|
let options = this.options;
|
||||||
|
if (this.dropdownMetaData.labelForNullValue && !this.dropdownDefaultValue)
|
||||||
|
{
|
||||||
|
const nullOption = {id: null as string, label: this.dropdownMetaData.labelForNullValue};
|
||||||
|
options = [nullOption, ...this.options];
|
||||||
|
|
||||||
|
if (!defaultValue)
|
||||||
|
{
|
||||||
|
defaultValue = nullOption;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box my={2} sx={{float: "right"}}>
|
<Box mb={2} sx={{float: "right"}}>
|
||||||
<DropdownMenu
|
<WidgetDropdownMenu
|
||||||
name={this.dropdownName}
|
name={this.dropdownName}
|
||||||
defaultValue={defaultValue}
|
defaultValue={defaultValue}
|
||||||
sx={{width: 200, marginLeft: "15px"}}
|
sx={{marginLeft: "1rem"}}
|
||||||
label={`Select ${this.label}`}
|
label={label}
|
||||||
dropdownOptions={this.options}
|
startIcon={this.dropdownMetaData.startIconName}
|
||||||
|
allowBackAndForth={this.dropdownMetaData.allowBackAndForth}
|
||||||
|
backAndForthInverted={this.dropdownMetaData.backAndForthInverted}
|
||||||
|
disableClearable={this.dropdownMetaData.disableClearable}
|
||||||
|
dropdownOptions={options}
|
||||||
onChangeCallback={this.onChangeCallback}
|
onChangeCallback={this.onChangeCallback}
|
||||||
|
width={this.dropdownMetaData.width ?? 225}
|
||||||
/>
|
/>
|
||||||
</Box>
|
</Box>
|
||||||
);
|
);
|
||||||
@ -196,8 +318,8 @@ export class ReloadControl extends LabelComponent
|
|||||||
render = (args: LabelComponentRenderArgs): JSX.Element =>
|
render = (args: LabelComponentRenderArgs): JSX.Element =>
|
||||||
{
|
{
|
||||||
return (
|
return (
|
||||||
<Typography variant="body2" py={2} px={0} display="inline" position="relative" top="-0.375rem">
|
<Typography variant="body2" py={2} px={0} display="inline" position="relative" top="-0.175rem">
|
||||||
<Tooltip title="Refresh"><Button sx={{px: 1, py: 0, minWidth: "initial"}} onClick={() => this.callback()}><Icon>refresh</Icon></Button></Tooltip>
|
<Tooltip title="Refresh"><Button sx={{px: 1, py: 0, minWidth: "initial"}} onClick={() => this.callback()}><Icon sx={{color: colors.gray.main, fontSize: 1.125}}>refresh</Icon></Button></Tooltip>
|
||||||
</Typography>
|
</Typography>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
@ -282,7 +404,12 @@ function Widget(props: React.PropsWithChildren<Props>): JSX.Element
|
|||||||
props.widgetData.dropdownDataList?.map((dropdownData: any, index: number) =>
|
props.widgetData.dropdownDataList?.map((dropdownData: any, index: number) =>
|
||||||
{
|
{
|
||||||
// console.log(`${props.widgetMetaData.name} building a Dropdown, data is: ${dropdownData}`);
|
// console.log(`${props.widgetMetaData.name} building a Dropdown, data is: ${dropdownData}`);
|
||||||
updatedStateLabelComponentsRight.push(new Dropdown(props.widgetData.dropdownLabelList[index], dropdownData, props.widgetData.dropdownNameList[index], handleDataChange));
|
let defaultValue = null;
|
||||||
|
if(props.widgetData.dropdownDefaultValueList && props.widgetData.dropdownDefaultValueList.length >= index)
|
||||||
|
{
|
||||||
|
defaultValue = props.widgetData.dropdownDefaultValueList[index];
|
||||||
|
}
|
||||||
|
updatedStateLabelComponentsRight.push(new Dropdown(props.widgetData.dropdownLabelList[index], props.widgetMetaData.dropdowns[index], dropdownData, defaultValue, props.widgetData.dropdownNameList[index], handleDataChange));
|
||||||
});
|
});
|
||||||
setLabelComponentsRight(updatedStateLabelComponentsRight);
|
setLabelComponentsRight(updatedStateLabelComponentsRight);
|
||||||
}
|
}
|
||||||
@ -406,20 +533,33 @@ function Widget(props: React.PropsWithChildren<Props>): JSX.Element
|
|||||||
needLabelBox ||= isSet(props.widgetMetaData?.label);
|
needLabelBox ||= isSet(props.widgetMetaData?.label);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// first look for a label in the widget data, which would override that in the metadata //
|
||||||
|
// note - previously this had a ?: and one was pl={2}, the other was pl={3}... //
|
||||||
|
//////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
const labelToUse = props.widgetData?.label ?? props.widgetMetaData?.label;
|
||||||
|
let labelElement = (
|
||||||
|
<Typography sx={{cursor: "default", pl: "auto", pt: props.widgetMetaData.type == "parentWidget" ? "1rem" : "auto", fontWeight: 600}} variant="h6" display="inline">
|
||||||
|
{labelToUse}
|
||||||
|
</Typography>
|
||||||
|
);
|
||||||
|
|
||||||
|
if (props.widgetMetaData.tooltip)
|
||||||
|
{
|
||||||
|
labelElement = <Tooltip title={props.widgetMetaData.tooltip} arrow={false} followCursor={true} placement="bottom-start">{labelElement}</Tooltip>;
|
||||||
|
}
|
||||||
|
|
||||||
const errorLoading = props.widgetData?.errorLoading !== undefined && props.widgetData?.errorLoading === true;
|
const errorLoading = props.widgetData?.errorLoading !== undefined && props.widgetData?.errorLoading === true;
|
||||||
const widgetContent =
|
const widgetContent =
|
||||||
<Box sx={{width: "100%", height: "100%", minHeight: props.widgetMetaData?.minHeight ? props.widgetMetaData?.minHeight : "initial"}}>
|
<Box sx={{width: "100%", height: "100%", minHeight: props.widgetMetaData?.minHeight ? props.widgetMetaData?.minHeight : "initial"}}>
|
||||||
{
|
{
|
||||||
needLabelBox &&
|
needLabelBox &&
|
||||||
<Box pr={2} display="flex" justifyContent="space-between" alignItems="flex-start" sx={{width: "100%"}} height={"3.5rem"}>
|
<Box display="flex" justifyContent="space-between" alignItems="flex-start" sx={{width: "100%", ...props.labelBoxAdditionalSx}} minHeight={"2.5rem"}>
|
||||||
<Box pt={2} pb={1}>
|
<Box>
|
||||||
{
|
{
|
||||||
hasPermission ?
|
hasPermission ?
|
||||||
props.widgetMetaData?.icon && (
|
props.widgetMetaData?.icon && (
|
||||||
<Box
|
<Box ml={1} mr={2} mt={-4} sx={{
|
||||||
ml={3}
|
|
||||||
mt={-4}
|
|
||||||
sx={{
|
|
||||||
display: "flex",
|
display: "flex",
|
||||||
justifyContent: "center",
|
justifyContent: "center",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
@ -437,10 +577,7 @@ function Widget(props: React.PropsWithChildren<Props>): JSX.Element
|
|||||||
</Box>
|
</Box>
|
||||||
) :
|
) :
|
||||||
(
|
(
|
||||||
<Box
|
<Box ml={3} mt={-4} sx={{
|
||||||
ml={3}
|
|
||||||
mt={-4}
|
|
||||||
sx={{
|
|
||||||
display: "flex",
|
display: "flex",
|
||||||
justifyContent: "center",
|
justifyContent: "center",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
@ -457,20 +594,7 @@ function Widget(props: React.PropsWithChildren<Props>): JSX.Element
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////
|
hasPermission && labelToUse && (labelElement)
|
||||||
// first look for a label in the widget data, which would override that in the metadata //
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
hasPermission && props.widgetData?.label ? (
|
|
||||||
<Typography sx={{position: "relative", top: -4}} variant="h6" fontWeight="medium" pl={2} display="inline-block">
|
|
||||||
{props.widgetData.label}
|
|
||||||
</Typography>
|
|
||||||
) : (
|
|
||||||
hasPermission && props.widgetMetaData?.label && (
|
|
||||||
<Typography sx={{position: "relative", top: -4}} variant="h6" fontWeight="medium" pl={3} display="inline-block">
|
|
||||||
{props.widgetMetaData.label}
|
|
||||||
</Typography>
|
|
||||||
)
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
hasPermission && (
|
hasPermission && (
|
||||||
@ -495,7 +619,12 @@ function Widget(props: React.PropsWithChildren<Props>): JSX.Element
|
|||||||
</Box>
|
</Box>
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
props.widgetMetaData?.isCard && (reloading ? <LinearProgress color="info" sx={{overflow: "hidden", borderRadius: "0"}} /> : <Box height="0.375rem" />)
|
///////////////////////////////////////////////////////////////////
|
||||||
|
// turning this off... for now. maybe make a property in future //
|
||||||
|
///////////////////////////////////////////////////////////////////
|
||||||
|
/*
|
||||||
|
props.widgetMetaData?.isCard && (reloading ? <LinearProgress color="info" sx={{overflow: "hidden", borderRadius: "0", mx:-2}} /> : <Box height="0.375rem" />)
|
||||||
|
*/
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
errorLoading ? (
|
errorLoading ? (
|
||||||
@ -505,7 +634,7 @@ function Widget(props: React.PropsWithChildren<Props>): JSX.Element
|
|||||||
</Box>
|
</Box>
|
||||||
) : (
|
) : (
|
||||||
hasPermission && props.widgetData?.dropdownNeedsSelectedText ? (
|
hasPermission && props.widgetData?.dropdownNeedsSelectedText ? (
|
||||||
<Box pb={3} pr={3} sx={{width: "100%", textAlign: "right"}}>
|
<Box pb={3} sx={{width: "100%", textAlign: "right"}}>
|
||||||
<Typography variant="body2">
|
<Typography variant="body2">
|
||||||
{props.widgetData?.dropdownNeedsSelectedText}
|
{props.widgetData?.dropdownNeedsSelectedText}
|
||||||
</Typography>
|
</Typography>
|
||||||
@ -526,11 +655,12 @@ function Widget(props: React.PropsWithChildren<Props>): JSX.Element
|
|||||||
}
|
}
|
||||||
</Box>;
|
</Box>;
|
||||||
|
|
||||||
|
const padding = props.omitPadding ? "auto" : "24px 16px";
|
||||||
return props.widgetMetaData?.isCard
|
return props.widgetMetaData?.isCard
|
||||||
? <Card sx={{marginTop: props.widgetMetaData?.icon ? 2 : 0, width: "100%"}} className={fullScreenWidgetClassName}>
|
? <Card sx={{marginTop: props.widgetMetaData?.icon ? 2 : 0, width: "100%", p: padding}} className={fullScreenWidgetClassName}>
|
||||||
{widgetContent}
|
{widgetContent}
|
||||||
</Card>
|
</Card>
|
||||||
: widgetContent;
|
: <span style={{width: "100%", padding: padding}}>{widgetContent}</span>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default Widget;
|
export default Widget;
|
||||||
|
@ -28,6 +28,7 @@ import {Bar} from "react-chartjs-2";
|
|||||||
import {useNavigate} from "react-router-dom";
|
import {useNavigate} from "react-router-dom";
|
||||||
import colors from "qqq/assets/theme/base/colors";
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
import {chartColors, DefaultChartData} from "qqq/components/widgets/charts/DefaultChartData";
|
import {chartColors, DefaultChartData} from "qqq/components/widgets/charts/DefaultChartData";
|
||||||
|
import ChartSubheaderWithData, {ChartSubheaderData} from "qqq/components/widgets/components/ChartSubheaderWithData";
|
||||||
|
|
||||||
ChartJS.register(
|
ChartJS.register(
|
||||||
CategoryScale,
|
CategoryScale,
|
||||||
@ -39,18 +40,61 @@ ChartJS.register(
|
|||||||
);
|
);
|
||||||
|
|
||||||
export const options = {
|
export const options = {
|
||||||
|
maintainAspectRatio: false,
|
||||||
responsive: true,
|
responsive: true,
|
||||||
animation: {
|
animation: {
|
||||||
duration: 0
|
duration: 0
|
||||||
},
|
},
|
||||||
|
elements: {
|
||||||
|
bar: {
|
||||||
|
borderRadius: 4
|
||||||
|
}
|
||||||
|
},
|
||||||
|
plugins: {
|
||||||
|
tooltip: {
|
||||||
|
// todo - some configs around this
|
||||||
|
callbacks: {
|
||||||
|
title: function(context: any)
|
||||||
|
{
|
||||||
|
return ("");
|
||||||
|
},
|
||||||
|
label: function(context: any)
|
||||||
|
{
|
||||||
|
if(context.dataset.label.startsWith(context.label))
|
||||||
|
{
|
||||||
|
return `${context.label}: ${context.formattedValue}`;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return ("");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
legend: {
|
||||||
|
position: "bottom",
|
||||||
|
labels: {
|
||||||
|
usePointStyle: true,
|
||||||
|
pointStyle: "circle",
|
||||||
|
boxHeight: 6,
|
||||||
|
boxWidth: 6,
|
||||||
|
padding: 12,
|
||||||
|
font: {
|
||||||
|
size: 14
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
scales: {
|
scales: {
|
||||||
x: {
|
x: {
|
||||||
stacked: true,
|
stacked: true,
|
||||||
grid: {offset: false},
|
grid: {display: false},
|
||||||
ticks: {autoSkip: false, maxRotation: 90}
|
ticks: {autoSkip: false, maxRotation: 90}
|
||||||
},
|
},
|
||||||
y: {
|
y: {
|
||||||
stacked: true,
|
stacked: true,
|
||||||
|
position: "right",
|
||||||
|
ticks: {precision: 0}
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
@ -58,10 +102,12 @@ export const options = {
|
|||||||
interface Props
|
interface Props
|
||||||
{
|
{
|
||||||
data: DefaultChartData;
|
data: DefaultChartData;
|
||||||
|
chartSubheaderData?: ChartSubheaderData;
|
||||||
}
|
}
|
||||||
|
|
||||||
const {gradients} = colors;
|
const {gradients} = colors;
|
||||||
function StackedBarChart({data}: Props): JSX.Element
|
|
||||||
|
function StackedBarChart({data, chartSubheaderData}: Props): JSX.Element
|
||||||
{
|
{
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
@ -76,7 +122,7 @@ function StackedBarChart({data}: Props): JSX.Element
|
|||||||
navigate(data.urls[e[0]["index"]]);
|
navigate(data.urls[e[0]["index"]]);
|
||||||
}
|
}
|
||||||
console.log(e);
|
console.log(e);
|
||||||
}
|
};
|
||||||
|
|
||||||
useEffect(() =>
|
useEffect(() =>
|
||||||
{
|
{
|
||||||
@ -85,9 +131,16 @@ function StackedBarChart({data}: Props): JSX.Element
|
|||||||
data?.datasets.forEach((dataset: any, index: number) =>
|
data?.datasets.forEach((dataset: any, index: number) =>
|
||||||
{
|
{
|
||||||
if (!dataset.backgroundColor)
|
if (!dataset.backgroundColor)
|
||||||
|
{
|
||||||
|
if (gradients[chartColors[index]])
|
||||||
{
|
{
|
||||||
dataset.backgroundColor = gradients[chartColors[index]].state;
|
dataset.backgroundColor = gradients[chartColors[index]].state;
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
dataset.backgroundColor = chartColors[index];
|
||||||
|
}
|
||||||
|
}
|
||||||
});
|
});
|
||||||
setStateData(stateData);
|
setStateData(stateData);
|
||||||
}
|
}
|
||||||
@ -95,7 +148,12 @@ function StackedBarChart({data}: Props): JSX.Element
|
|||||||
|
|
||||||
|
|
||||||
return data ? (
|
return data ? (
|
||||||
<Box p={3}><Bar data={data} options={options} getElementsAtEvent={handleClick} /></Box>
|
<Box>
|
||||||
|
{chartSubheaderData && (<ChartSubheaderWithData chartSubheaderData={chartSubheaderData} />)}
|
||||||
|
<Box width="100%" height="300px">
|
||||||
|
<Bar data={data} options={options} getElementsAtEvent={handleClick} />
|
||||||
|
</Box>
|
||||||
|
</Box>
|
||||||
) : <Skeleton sx={{marginLeft: "20px", marginRight: "20px", height: "200px"}} />;
|
) : <Skeleton sx={{marginLeft: "20px", marginRight: "20px", height: "200px"}} />;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -63,7 +63,7 @@ const options = {
|
|||||||
font: {
|
font: {
|
||||||
size: 14,
|
size: 14,
|
||||||
weight: 300,
|
weight: 300,
|
||||||
family: "Roboto",
|
family: "SF Pro Display,Roboto",
|
||||||
style: "normal",
|
style: "normal",
|
||||||
lineHeight: 2,
|
lineHeight: 2,
|
||||||
},
|
},
|
||||||
@ -86,7 +86,7 @@ const options = {
|
|||||||
font: {
|
font: {
|
||||||
size: 14,
|
size: 14,
|
||||||
weight: 300,
|
weight: 300,
|
||||||
family: "Roboto",
|
family: "SF Pro Display,Roboto",
|
||||||
style: "normal",
|
style: "normal",
|
||||||
lineHeight: 2,
|
lineHeight: 2,
|
||||||
},
|
},
|
||||||
|
@ -67,7 +67,7 @@ const options = {
|
|||||||
font: {
|
font: {
|
||||||
size: 14,
|
size: 14,
|
||||||
weight: 300,
|
weight: 300,
|
||||||
family: "Roboto",
|
family: "SF Pro Display,Roboto",
|
||||||
style: "normal",
|
style: "normal",
|
||||||
lineHeight: 2,
|
lineHeight: 2,
|
||||||
},
|
},
|
||||||
@ -88,7 +88,7 @@ const options = {
|
|||||||
font: {
|
font: {
|
||||||
size: 14,
|
size: 14,
|
||||||
weight: 300,
|
weight: 300,
|
||||||
family: "Roboto",
|
family: "SF Pro Display,Roboto",
|
||||||
style: "normal",
|
style: "normal",
|
||||||
lineHeight: 2,
|
lineHeight: 2,
|
||||||
},
|
},
|
||||||
|
@ -81,7 +81,7 @@ const options = {
|
|||||||
font: {
|
font: {
|
||||||
size: 14,
|
size: 14,
|
||||||
weight: 300,
|
weight: 300,
|
||||||
family: "Roboto",
|
family: "SF Pro Display,Roboto",
|
||||||
style: "normal",
|
style: "normal",
|
||||||
lineHeight: 2,
|
lineHeight: 2,
|
||||||
},
|
},
|
||||||
@ -107,7 +107,7 @@ const options = {
|
|||||||
font: {
|
font: {
|
||||||
size: 14,
|
size: 14,
|
||||||
weight: 300,
|
weight: 300,
|
||||||
family: "Roboto",
|
family: "SF Pro Display,Roboto",
|
||||||
style: "normal",
|
style: "normal",
|
||||||
lineHeight: 2,
|
lineHeight: 2,
|
||||||
},
|
},
|
||||||
|
@ -69,7 +69,7 @@ function configs(labels: any, datasets: any)
|
|||||||
font: {
|
font: {
|
||||||
size: 14,
|
size: 14,
|
||||||
weight: 300,
|
weight: 300,
|
||||||
family: "Roboto",
|
family: "SF Pro Display,Roboto",
|
||||||
style: "normal",
|
style: "normal",
|
||||||
lineHeight: 2,
|
lineHeight: 2,
|
||||||
},
|
},
|
||||||
@ -90,7 +90,7 @@ function configs(labels: any, datasets: any)
|
|||||||
font: {
|
font: {
|
||||||
size: 14,
|
size: 14,
|
||||||
weight: 300,
|
weight: 300,
|
||||||
family: "Roboto",
|
family: "SF Pro Display,Roboto",
|
||||||
style: "normal",
|
style: "normal",
|
||||||
lineHeight: 2,
|
lineHeight: 2,
|
||||||
},
|
},
|
||||||
|
@ -30,6 +30,7 @@ import {useNavigate} from "react-router-dom";
|
|||||||
import MDTypography from "qqq/components/legacy/MDTypography";
|
import MDTypography from "qqq/components/legacy/MDTypography";
|
||||||
import {chartColors} from "qqq/components/widgets/charts/DefaultChartData";
|
import {chartColors} from "qqq/components/widgets/charts/DefaultChartData";
|
||||||
import configs from "qqq/components/widgets/charts/piechart/PieChartConfigs";
|
import configs from "qqq/components/widgets/charts/piechart/PieChartConfigs";
|
||||||
|
import ChartSubheaderWithData, {ChartSubheaderData} from "qqq/components/widgets/components/ChartSubheaderWithData";
|
||||||
|
|
||||||
//////////////////////////////////////////
|
//////////////////////////////////////////
|
||||||
// structure of expected bar chart data //
|
// structure of expected bar chart data //
|
||||||
@ -51,20 +52,24 @@ interface Props
|
|||||||
{
|
{
|
||||||
description?: string;
|
description?: string;
|
||||||
chartData: PieChartData;
|
chartData: PieChartData;
|
||||||
|
chartSubheaderData?: ChartSubheaderData;
|
||||||
|
|
||||||
[key: string]: any;
|
[key: string]: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
function PieChart({description, chartData}: Props): JSX.Element
|
function PieChart({description, chartData, chartSubheaderData}: Props): JSX.Element
|
||||||
{
|
{
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
const [dataLoaded, setDataLoaded] = useState(false);
|
const [dataLoaded, setDataLoaded] = useState(false);
|
||||||
|
|
||||||
if (chartData && chartData.dataset)
|
if (chartData && chartData.dataset)
|
||||||
|
{
|
||||||
|
if(!chartData.dataset.backgroundColors)
|
||||||
{
|
{
|
||||||
chartData.dataset.backgroundColors = chartColors;
|
chartData.dataset.backgroundColors = chartColors;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
const {data, options} = configs(chartData?.labels || [], chartData?.dataset || {});
|
const {data, options} = configs(chartData?.labels || [], chartData?.dataset || {});
|
||||||
|
|
||||||
useEffect(() =>
|
useEffect(() =>
|
||||||
@ -82,14 +87,15 @@ function PieChart({description, chartData}: Props): JSX.Element
|
|||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
navigate(chartData.dataset.urls[e[0]["index"]]);
|
navigate(chartData.dataset.urls[e[0]["index"]]);
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card sx={{minHeight: "400px", boxShadow: "none", height: "100%", width: "100%", display: "flex", flexGrow: 1}}>
|
<Card sx={{boxShadow: "none", height: "100%", width: "100%", display: "flex", flexGrow: 1, border: 0}}>
|
||||||
<Box mt={3}>
|
<Box>
|
||||||
<Grid container alignItems="center">
|
<Box>
|
||||||
<Grid item xs={12} justifyContent="center">
|
{chartSubheaderData && (<ChartSubheaderWithData chartSubheaderData={chartSubheaderData} />)}
|
||||||
<Box width="100%" height="80%" py={2} pr={2} pl={2}>
|
</Box>
|
||||||
|
<Box width="100%" height="300px">
|
||||||
{useMemo(
|
{useMemo(
|
||||||
() => (
|
() => (
|
||||||
<Pie data={data} options={options} getElementsAtEvent={handleClick} />
|
<Pie data={data} options={options} getElementsAtEvent={handleClick} />
|
||||||
@ -105,25 +111,22 @@ function PieChart({description, chartData}: Props): JSX.Element
|
|||||||
left: "50%",
|
left: "50%",
|
||||||
transform: "translate(-50%, -50%)",
|
transform: "translate(-50%, -50%)",
|
||||||
display: "flex",
|
display: "flex",
|
||||||
justifyContent: "center"}}>
|
justifyContent: "center"
|
||||||
|
}}>
|
||||||
<Skeleton sx={{width: "150px", height: "150px"}} variant="circular" />
|
<Skeleton sx={{width: "150px", height: "150px"}} variant="circular" />
|
||||||
</Box>
|
</Box>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
</Grid>
|
|
||||||
</Grid>
|
|
||||||
<Divider />
|
|
||||||
{
|
{
|
||||||
description && (
|
description && (
|
||||||
<Grid container>
|
<>
|
||||||
<Grid item xs={12}>
|
<Divider />
|
||||||
<Box pb={2} px={2} display="flex" flexDirection={{xs: "column", sm: "row"}} mt="auto">
|
<Box display="flex" flexDirection={{xs: "column", sm: "row"}} mt="auto">
|
||||||
<MDTypography variant="button" color="text" fontWeight="light">
|
<MDTypography variant="button" color="text" fontWeight="light">
|
||||||
{parse(description)}
|
{parse(description)}
|
||||||
</MDTypography>
|
</MDTypography>
|
||||||
</Box>
|
</Box>
|
||||||
</Grid>
|
</>
|
||||||
</Grid>
|
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
</Box>
|
</Box>
|
||||||
|
@ -30,10 +30,16 @@ function configs(labels: any, datasets: any)
|
|||||||
if (datasets.backgroundColors)
|
if (datasets.backgroundColors)
|
||||||
{
|
{
|
||||||
datasets.backgroundColors.forEach((color: string) =>
|
datasets.backgroundColors.forEach((color: string) =>
|
||||||
gradients[color]
|
{
|
||||||
? backgroundColors.push(gradients[color].state)
|
if (gradients[color])
|
||||||
: backgroundColors.push(dark.main)
|
{
|
||||||
);
|
backgroundColors.push(gradients[color].state);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
backgroundColors.push(color);
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -58,12 +64,49 @@ function configs(labels: any, datasets: any)
|
|||||||
],
|
],
|
||||||
},
|
},
|
||||||
options: {
|
options: {
|
||||||
maintainAspectRatio: true,
|
maintainAspectRatio: false,
|
||||||
responsive: true,
|
responsive: true,
|
||||||
aspectRatio: 2,
|
|
||||||
plugins: {
|
plugins: {
|
||||||
|
tooltip: {
|
||||||
|
callbacks: {
|
||||||
|
label: function(context: any)
|
||||||
|
{
|
||||||
|
let percentSuffix = "";
|
||||||
|
try
|
||||||
|
{
|
||||||
|
//////////////////////////////////////////////////////////////////////////
|
||||||
|
// make percent by dividing this slice's value by the sum of all values //
|
||||||
|
//////////////////////////////////////////////////////////////////////////
|
||||||
|
const thisSlice = context.dataset.data[context.dataIndex];
|
||||||
|
const sum = context.dataset.data.reduce((acc: number, val: number) => acc + val, 0);
|
||||||
|
percentSuffix = " (" + Number(100 * thisSlice / sum).toFixed(1) + "%)";
|
||||||
|
}
|
||||||
|
catch(e)
|
||||||
|
{
|
||||||
|
// leave percentSuffix empty
|
||||||
|
}
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// our labels already have the value in them - so just use the label in the //
|
||||||
|
// tooltip (lib by default puts label + value, so we were duplicating value!) //
|
||||||
|
// oh, and we add percent if we can //
|
||||||
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
return context.label + percentSuffix;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
legend: {
|
legend: {
|
||||||
position: "bottom",
|
position: "bottom",
|
||||||
|
labels: {
|
||||||
|
usePointStyle: true,
|
||||||
|
pointStyle: "circle",
|
||||||
|
padding: 12,
|
||||||
|
boxHeight: 8,
|
||||||
|
boxWidth: 8,
|
||||||
|
font: {
|
||||||
|
size: 14
|
||||||
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
scales: {
|
scales: {
|
||||||
|
105
src/qqq/components/widgets/components/ChartSubheaderWithData.tsx
Normal file
105
src/qqq/components/widgets/components/ChartSubheaderWithData.tsx
Normal file
@ -0,0 +1,105 @@
|
|||||||
|
/*
|
||||||
|
* QQQ - Low-code Application Framework for Engineers.
|
||||||
|
* Copyright (C) 2021-2022. 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 {Skeleton} from "@mui/material";
|
||||||
|
import Box from "@mui/material/Box";
|
||||||
|
import Icon from "@mui/material/Icon";
|
||||||
|
import Typography from "@mui/material/Typography";
|
||||||
|
import React from "react";
|
||||||
|
import {Link} from "react-router-dom";
|
||||||
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
|
import ValueUtils from "qqq/utils/qqq/ValueUtils";
|
||||||
|
|
||||||
|
export interface ChartSubheaderData
|
||||||
|
{
|
||||||
|
mainNumber: number;
|
||||||
|
vsPreviousPercent: number;
|
||||||
|
vsPreviousNumber: number;
|
||||||
|
isUpVsPrevious: boolean;
|
||||||
|
isGoodVsPrevious: boolean;
|
||||||
|
vsDescription: string;
|
||||||
|
mainNumberUrl: string;
|
||||||
|
previousNumberUrl: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface Props
|
||||||
|
{
|
||||||
|
chartSubheaderData: ChartSubheaderData;
|
||||||
|
}
|
||||||
|
|
||||||
|
const GOOD_COLOR = colors.success.main;
|
||||||
|
const BAD_COLOR = colors.error.main;
|
||||||
|
const UP_ICON = "arrow_drop_up";
|
||||||
|
const DOWN_ICON = "arrow_drop_down";
|
||||||
|
|
||||||
|
function StackedBarChart({chartSubheaderData}: Props): JSX.Element
|
||||||
|
{
|
||||||
|
let color = "black";
|
||||||
|
if (chartSubheaderData && chartSubheaderData.isGoodVsPrevious != null)
|
||||||
|
{
|
||||||
|
color = chartSubheaderData.isGoodVsPrevious ? GOOD_COLOR : BAD_COLOR;
|
||||||
|
}
|
||||||
|
|
||||||
|
let iconName: string = null;
|
||||||
|
if (chartSubheaderData && chartSubheaderData.isUpVsPrevious != null)
|
||||||
|
{
|
||||||
|
iconName = chartSubheaderData.isUpVsPrevious ? UP_ICON : DOWN_ICON;
|
||||||
|
}
|
||||||
|
|
||||||
|
let mainNumberElement = <Typography variant="h3" display="inline">{ValueUtils.getFormattedNumber(chartSubheaderData.mainNumber)}</Typography>;
|
||||||
|
if(chartSubheaderData.mainNumberUrl)
|
||||||
|
{
|
||||||
|
mainNumberElement = <Link to={chartSubheaderData.mainNumberUrl}>{mainNumberElement}</Link>
|
||||||
|
}
|
||||||
|
mainNumberElement = <Box pr={1}>{mainNumberElement}</Box>
|
||||||
|
|
||||||
|
let previousNumberElement = (
|
||||||
|
<>
|
||||||
|
<Typography display="block" variant="body2" sx={{color: colors.gray.main, fontSize: ".875rem", fontWeight: 500}}>
|
||||||
|
{chartSubheaderData.vsDescription}
|
||||||
|
{chartSubheaderData.vsPreviousNumber && (<> ({ValueUtils.getFormattedNumber(chartSubheaderData.vsPreviousNumber)})</>)}
|
||||||
|
</Typography>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
|
||||||
|
if(chartSubheaderData.previousNumberUrl)
|
||||||
|
{
|
||||||
|
previousNumberElement = <Link to={chartSubheaderData.previousNumberUrl}>{previousNumberElement}</Link>
|
||||||
|
}
|
||||||
|
|
||||||
|
return chartSubheaderData ? (
|
||||||
|
<Box display="inline-flex" alignItems="flex-end" flexWrap="wrap">
|
||||||
|
{mainNumberElement}
|
||||||
|
{
|
||||||
|
chartSubheaderData.vsPreviousPercent != null && iconName != null && (
|
||||||
|
<Box display="inline-flex" alignItems="baseline" pb={0.5} ml={-0.5}>
|
||||||
|
<Icon fontSize="medium" sx={{color: color, alignSelf: "flex-end"}}>{iconName}</Icon>
|
||||||
|
<Typography display="inline" variant="body2" sx={{color: color, fontSize: ".875rem", fontWeight: 500}}>{chartSubheaderData.vsPreviousPercent}%</Typography>
|
||||||
|
{previousNumberElement}
|
||||||
|
</Box>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
</Box>
|
||||||
|
) : <Skeleton sx={{marginLeft: "20px", marginRight: "20px", height: "12px"}} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default StackedBarChart;
|
@ -1,179 +0,0 @@
|
|||||||
/*
|
|
||||||
* QQQ - Low-code Application Framework for Engineers.
|
|
||||||
* Copyright (C) 2021-2022. 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 {Collapse, Theme} from "@mui/material";
|
|
||||||
import Autocomplete from "@mui/material/Autocomplete";
|
|
||||||
import Box from "@mui/material/Box";
|
|
||||||
import TextField from "@mui/material/TextField";
|
|
||||||
import {SxProps} from "@mui/system";
|
|
||||||
import {Field, Form, Formik} from "formik";
|
|
||||||
import React, {useState} from "react";
|
|
||||||
import MDInput from "qqq/components/legacy/MDInput";
|
|
||||||
import FilterUtils from "qqq/utils/qqq/FilterUtils";
|
|
||||||
import ValueUtils from "qqq/utils/qqq/ValueUtils";
|
|
||||||
|
|
||||||
|
|
||||||
export interface DropdownOption
|
|
||||||
{
|
|
||||||
id: string;
|
|
||||||
label: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
/////////////////////////
|
|
||||||
// inputs and defaults //
|
|
||||||
/////////////////////////
|
|
||||||
interface Props
|
|
||||||
{
|
|
||||||
name: string;
|
|
||||||
defaultValue?: any;
|
|
||||||
label?: string;
|
|
||||||
dropdownOptions?: DropdownOption[];
|
|
||||||
onChangeCallback?: (dropdownLabel: string, data: any) => void;
|
|
||||||
sx?: SxProps<Theme>;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface StartAndEndDate
|
|
||||||
{
|
|
||||||
startDate?: string,
|
|
||||||
endDate?: string
|
|
||||||
}
|
|
||||||
|
|
||||||
function parseCustomTimeValuesFromDefaultValue(defaultValue: any): StartAndEndDate
|
|
||||||
{
|
|
||||||
const customTimeValues: StartAndEndDate = {};
|
|
||||||
if(defaultValue && defaultValue.id)
|
|
||||||
{
|
|
||||||
var parts = defaultValue.id.split(",");
|
|
||||||
if(parts.length >= 2)
|
|
||||||
{
|
|
||||||
customTimeValues["startDate"] = ValueUtils.formatDateTimeValueForForm(parts[1]);
|
|
||||||
}
|
|
||||||
if(parts.length >= 3)
|
|
||||||
{
|
|
||||||
customTimeValues["endDate"] = ValueUtils.formatDateTimeValueForForm(parts[2]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return (customTimeValues);
|
|
||||||
}
|
|
||||||
|
|
||||||
function makeBackendValuesFromFrontendValues(frontendDefaultValues: StartAndEndDate): StartAndEndDate
|
|
||||||
{
|
|
||||||
const backendTimeValues: StartAndEndDate = {};
|
|
||||||
if(frontendDefaultValues && frontendDefaultValues.startDate)
|
|
||||||
{
|
|
||||||
backendTimeValues.startDate = ValueUtils.frontendLocalZoneDateTimeStringToUTCStringForBackend(frontendDefaultValues.startDate);
|
|
||||||
}
|
|
||||||
if(frontendDefaultValues && frontendDefaultValues.endDate)
|
|
||||||
{
|
|
||||||
backendTimeValues.endDate = ValueUtils.frontendLocalZoneDateTimeStringToUTCStringForBackend(frontendDefaultValues.endDate);
|
|
||||||
}
|
|
||||||
return (backendTimeValues);
|
|
||||||
}
|
|
||||||
|
|
||||||
function DropdownMenu({name, defaultValue, label, dropdownOptions, onChangeCallback, sx}: Props): JSX.Element
|
|
||||||
{
|
|
||||||
const [customTimesVisible, setCustomTimesVisible] = useState(defaultValue && defaultValue.id && defaultValue.id.startsWith("custom,"));
|
|
||||||
const [customTimeValuesFrontend, setCustomTimeValuesFrontend] = useState(parseCustomTimeValuesFromDefaultValue(defaultValue) as StartAndEndDate);
|
|
||||||
const [customTimeValuesBackend, setCustomTimeValuesBackend] = useState(makeBackendValuesFromFrontendValues(customTimeValuesFrontend) as StartAndEndDate);
|
|
||||||
const [debounceTimeout, setDebounceTimeout] = useState(null as any);
|
|
||||||
|
|
||||||
const handleOnChange = (event: any, newValue: any, reason: string) =>
|
|
||||||
{
|
|
||||||
const isTimeframeCustom = name == "timeframe" && newValue && newValue.id == "custom"
|
|
||||||
setCustomTimesVisible(isTimeframeCustom);
|
|
||||||
|
|
||||||
if(isTimeframeCustom)
|
|
||||||
{
|
|
||||||
callOnChangeCallbackIfCustomTimeframeHasDateValues();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
onChangeCallback(label, newValue);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const callOnChangeCallbackIfCustomTimeframeHasDateValues = () =>
|
|
||||||
{
|
|
||||||
if(customTimeValuesBackend["startDate"] && customTimeValuesBackend["endDate"])
|
|
||||||
{
|
|
||||||
onChangeCallback(label, {id: `custom,${customTimeValuesBackend["startDate"]},${customTimeValuesBackend["endDate"]}`, label: "Custom"});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let customTimes = <></>;
|
|
||||||
if (name == "timeframe")
|
|
||||||
{
|
|
||||||
const handleSubmit = async (values: any, actions: any) =>
|
|
||||||
{
|
|
||||||
};
|
|
||||||
|
|
||||||
const dateChanged = (fieldName: "startDate" | "endDate", event: any) =>
|
|
||||||
{
|
|
||||||
customTimeValuesFrontend[fieldName] = event.target.value;
|
|
||||||
customTimeValuesBackend[fieldName] = ValueUtils.frontendLocalZoneDateTimeStringToUTCStringForBackend(event.target.value);
|
|
||||||
|
|
||||||
clearTimeout(debounceTimeout);
|
|
||||||
const newDebounceTimeout = setTimeout(() =>
|
|
||||||
{
|
|
||||||
callOnChangeCallbackIfCustomTimeframeHasDateValues();
|
|
||||||
}, 500);
|
|
||||||
setDebounceTimeout(newDebounceTimeout);
|
|
||||||
};
|
|
||||||
|
|
||||||
customTimes = <Box sx={{display: "inline-block", position: "relative", top: "-7px"}}>
|
|
||||||
<Collapse orientation="horizontal" in={customTimesVisible}>
|
|
||||||
<Formik initialValues={customTimeValuesFrontend} onSubmit={handleSubmit}>
|
|
||||||
{({}) => (
|
|
||||||
<Form id="timeframe-form" autoComplete="off">
|
|
||||||
<Field name="startDate" type="datetime-local" as={MDInput} variant="standard" label="Custom Timeframe Start" InputLabelProps={{shrink: true}} InputProps={{size: "small"}} sx={{ml: 2, width: 198}} onChange={(event: any) => dateChanged("startDate", event)} />
|
|
||||||
<Field name="endDate" type="datetime-local" as={MDInput} variant="standard" label="Custom Timeframe End" InputLabelProps={{shrink: true}} InputProps={{size: "small"}} sx={{ml: 2, width: 198}} onChange={(event: any) => dateChanged("endDate", event)} />
|
|
||||||
</Form>
|
|
||||||
)}
|
|
||||||
</Formik>
|
|
||||||
</Collapse>
|
|
||||||
</Box>;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
dropdownOptions ? (
|
|
||||||
<span style={{whiteSpace: "nowrap", display: "flex"}} className="dashboardDropdownMenu">
|
|
||||||
<Autocomplete
|
|
||||||
defaultValue={defaultValue}
|
|
||||||
size="small"
|
|
||||||
disablePortal
|
|
||||||
id={`${label}-combo-box`}
|
|
||||||
options={dropdownOptions}
|
|
||||||
sx={{...sx, cursor: "pointer", display: "inline-block"}}
|
|
||||||
onChange={handleOnChange}
|
|
||||||
isOptionEqualToValue={(option, value) => option.id === value.id}
|
|
||||||
renderInput={(params: any) => <TextField {...params} label={label} />}
|
|
||||||
renderOption={(props, option: DropdownOption) => (
|
|
||||||
<li {...props} style={{whiteSpace: "normal"}}>{option.label}</li>
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
{customTimes}
|
|
||||||
</span>
|
|
||||||
) : null
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export default DropdownMenu;
|
|
335
src/qqq/components/widgets/components/WidgetDropdownMenu.tsx
Normal file
335
src/qqq/components/widgets/components/WidgetDropdownMenu.tsx
Normal file
@ -0,0 +1,335 @@
|
|||||||
|
/*
|
||||||
|
* QQQ - Low-code Application Framework for Engineers.
|
||||||
|
* Copyright (C) 2021-2022. 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 {Collapse, Theme, InputAdornment} from "@mui/material";
|
||||||
|
import Autocomplete from "@mui/material/Autocomplete";
|
||||||
|
import Box from "@mui/material/Box";
|
||||||
|
import Icon from "@mui/material/Icon";
|
||||||
|
import IconButton from "@mui/material/IconButton";
|
||||||
|
import TextField from "@mui/material/TextField";
|
||||||
|
import {SxProps} from "@mui/system";
|
||||||
|
import {Field, Form, Formik} from "formik";
|
||||||
|
import React, {useState} from "react";
|
||||||
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
|
import MDInput from "qqq/components/legacy/MDInput";
|
||||||
|
import ValueUtils from "qqq/utils/qqq/ValueUtils";
|
||||||
|
|
||||||
|
|
||||||
|
export interface DropdownOption
|
||||||
|
{
|
||||||
|
id: string;
|
||||||
|
label: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/////////////////////////
|
||||||
|
// inputs and defaults //
|
||||||
|
/////////////////////////
|
||||||
|
interface Props
|
||||||
|
{
|
||||||
|
name: string;
|
||||||
|
defaultValue?: any;
|
||||||
|
label?: string;
|
||||||
|
startIcon?: string;
|
||||||
|
width?: number;
|
||||||
|
disableClearable?: boolean;
|
||||||
|
allowBackAndForth?: boolean;
|
||||||
|
backAndForthInverted?: boolean;
|
||||||
|
dropdownOptions?: DropdownOption[];
|
||||||
|
onChangeCallback?: (dropdownLabel: string, data: any) => void;
|
||||||
|
sx?: SxProps<Theme>;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface StartAndEndDate
|
||||||
|
{
|
||||||
|
startDate?: string,
|
||||||
|
endDate?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseCustomTimeValuesFromDefaultValue(defaultValue: any): StartAndEndDate
|
||||||
|
{
|
||||||
|
const customTimeValues: StartAndEndDate = {};
|
||||||
|
if (defaultValue && defaultValue.id)
|
||||||
|
{
|
||||||
|
var parts = defaultValue.id.split(",");
|
||||||
|
if (parts.length >= 2)
|
||||||
|
{
|
||||||
|
customTimeValues["startDate"] = ValueUtils.formatDateTimeValueForForm(parts[1]);
|
||||||
|
}
|
||||||
|
if (parts.length >= 3)
|
||||||
|
{
|
||||||
|
customTimeValues["endDate"] = ValueUtils.formatDateTimeValueForForm(parts[2]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (customTimeValues);
|
||||||
|
}
|
||||||
|
|
||||||
|
function makeBackendValuesFromFrontendValues(frontendDefaultValues: StartAndEndDate): StartAndEndDate
|
||||||
|
{
|
||||||
|
const backendTimeValues: StartAndEndDate = {};
|
||||||
|
if (frontendDefaultValues && frontendDefaultValues.startDate)
|
||||||
|
{
|
||||||
|
backendTimeValues.startDate = ValueUtils.frontendLocalZoneDateTimeStringToUTCStringForBackend(frontendDefaultValues.startDate);
|
||||||
|
}
|
||||||
|
if (frontendDefaultValues && frontendDefaultValues.endDate)
|
||||||
|
{
|
||||||
|
backendTimeValues.endDate = ValueUtils.frontendLocalZoneDateTimeStringToUTCStringForBackend(frontendDefaultValues.endDate);
|
||||||
|
}
|
||||||
|
return (backendTimeValues);
|
||||||
|
}
|
||||||
|
|
||||||
|
function WidgetDropdownMenu({name, defaultValue, label, startIcon, width, disableClearable, allowBackAndForth, backAndForthInverted, dropdownOptions, onChangeCallback, sx}: Props): JSX.Element
|
||||||
|
{
|
||||||
|
const [customTimesVisible, setCustomTimesVisible] = useState(defaultValue && defaultValue.id && defaultValue.id.startsWith("custom,"));
|
||||||
|
const [customTimeValuesFrontend, setCustomTimeValuesFrontend] = useState(parseCustomTimeValuesFromDefaultValue(defaultValue) as StartAndEndDate);
|
||||||
|
const [customTimeValuesBackend, setCustomTimeValuesBackend] = useState(makeBackendValuesFromFrontendValues(customTimeValuesFrontend) as StartAndEndDate);
|
||||||
|
const [debounceTimeout, setDebounceTimeout] = useState(null as any);
|
||||||
|
|
||||||
|
const [isOpen, setIsOpen] = useState(false);
|
||||||
|
const [value, setValue] = useState(defaultValue);
|
||||||
|
const [inputValue, setInputValue] = useState("");
|
||||||
|
|
||||||
|
const [backDisabled, setBackDisabled] = useState(false);
|
||||||
|
const [forthDisabled, setForthDisabled] = useState(false);
|
||||||
|
|
||||||
|
const doForceOpen = (event: React.MouseEvent<HTMLDivElement>) =>
|
||||||
|
{
|
||||||
|
setIsOpen(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
function getSelectedIndex(value: DropdownOption)
|
||||||
|
{
|
||||||
|
let currentIndex = null;
|
||||||
|
for (let i = 0; i < dropdownOptions.length; i++)
|
||||||
|
{
|
||||||
|
if (value && dropdownOptions[i].id == value.id)
|
||||||
|
{
|
||||||
|
currentIndex = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return currentIndex;
|
||||||
|
}
|
||||||
|
|
||||||
|
const navigateBackAndForth = (event: React.MouseEvent, direction: -1 | 1) =>
|
||||||
|
{
|
||||||
|
event.stopPropagation();
|
||||||
|
let currentIndex = getSelectedIndex(value);
|
||||||
|
|
||||||
|
if (currentIndex == null)
|
||||||
|
{
|
||||||
|
console.log("No current value.... TODO");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (currentIndex == 0 && direction == -1)
|
||||||
|
{
|
||||||
|
console.log("Can't go -1");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (currentIndex == dropdownOptions.length - 1 && direction == 1)
|
||||||
|
{
|
||||||
|
console.log("Can't go +1");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
handleOnChange(event, dropdownOptions[currentIndex + direction], "navigatedBackAndForth");
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
const handleOnChange = (event: any, newValue: any, reason: string) =>
|
||||||
|
{
|
||||||
|
setValue(newValue);
|
||||||
|
|
||||||
|
const isTimeframeCustom = name == "timeframe" && newValue && newValue.id == "custom";
|
||||||
|
setCustomTimesVisible(isTimeframeCustom);
|
||||||
|
|
||||||
|
if (isTimeframeCustom)
|
||||||
|
{
|
||||||
|
callOnChangeCallbackIfCustomTimeframeHasDateValues();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
onChangeCallback(label, newValue);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* this had bugs (seemed to not take immediate effect?), so don't use for now.
|
||||||
|
let currentIndex = getSelectedIndex(value);
|
||||||
|
if(currentIndex == 0)
|
||||||
|
{
|
||||||
|
backAndForthInverted ? setForthDisabled(true) : setBackDisabled(true);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
backAndForthInverted ? setForthDisabled(false) : setBackDisabled(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (currentIndex == dropdownOptions.length - 1)
|
||||||
|
{
|
||||||
|
backAndForthInverted ? setBackDisabled(true) : setForthDisabled(true);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
backAndForthInverted ? setBackDisabled(false) : setForthDisabled(false);
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleOnInputChange = (event: any, newValue: any, reason: string) =>
|
||||||
|
{
|
||||||
|
setInputValue(newValue);
|
||||||
|
};
|
||||||
|
|
||||||
|
const callOnChangeCallbackIfCustomTimeframeHasDateValues = () =>
|
||||||
|
{
|
||||||
|
if (customTimeValuesBackend["startDate"] && customTimeValuesBackend["endDate"])
|
||||||
|
{
|
||||||
|
onChangeCallback(label, {id: `custom,${customTimeValuesBackend["startDate"]},${customTimeValuesBackend["endDate"]}`, label: "Custom"});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let customTimes = <></>;
|
||||||
|
if (name == "timeframe")
|
||||||
|
{
|
||||||
|
const handleSubmit = async (values: any, actions: any) =>
|
||||||
|
{
|
||||||
|
};
|
||||||
|
|
||||||
|
const dateChanged = (fieldName: "startDate" | "endDate", event: any) =>
|
||||||
|
{
|
||||||
|
customTimeValuesFrontend[fieldName] = event.target.value;
|
||||||
|
customTimeValuesBackend[fieldName] = ValueUtils.frontendLocalZoneDateTimeStringToUTCStringForBackend(event.target.value);
|
||||||
|
|
||||||
|
clearTimeout(debounceTimeout);
|
||||||
|
const newDebounceTimeout = setTimeout(() =>
|
||||||
|
{
|
||||||
|
callOnChangeCallbackIfCustomTimeframeHasDateValues();
|
||||||
|
}, 500);
|
||||||
|
setDebounceTimeout(newDebounceTimeout);
|
||||||
|
};
|
||||||
|
|
||||||
|
customTimes = <Box sx={{display: "inline-block", position: "relative", top: "-7px"}}>
|
||||||
|
<Collapse orientation="horizontal" in={customTimesVisible}>
|
||||||
|
<Formik initialValues={customTimeValuesFrontend} onSubmit={handleSubmit}>
|
||||||
|
{({}) => (
|
||||||
|
<Form id="timeframe-form" autoComplete="off">
|
||||||
|
<Field name="startDate" type="datetime-local" as={MDInput} variant="standard" label="Custom Timeframe Start" InputLabelProps={{shrink: true}} InputProps={{size: "small"}} sx={{ml: 2, width: 198}} onChange={(event: any) => dateChanged("startDate", event)} />
|
||||||
|
<Field name="endDate" type="datetime-local" as={MDInput} variant="standard" label="Custom Timeframe End" InputLabelProps={{shrink: true}} InputProps={{size: "small"}} sx={{ml: 2, width: 198}} onChange={(event: any) => dateChanged("endDate", event)} />
|
||||||
|
</Form>
|
||||||
|
)}
|
||||||
|
</Formik>
|
||||||
|
</Collapse>
|
||||||
|
</Box>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const startAdornment = startIcon ? <Icon sx={{fontSize: "1.25rem!important", color: colors.gray.main, paddingLeft: allowBackAndForth ? "auto" : "0.25rem", width: allowBackAndForth ? "1.5rem" : "1.75rem"}}>{startIcon}</Icon> : null;
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// we tried this end-adornment, for a different style of down-arrow - but by using it, we then messed something else up (i forget what), so... not used right now //
|
||||||
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
const endAdornment = <InputAdornment position="end" sx={{position: "absolute", right: allowBackAndForth ? "-0.5rem" : "0.5rem"}}><Icon sx={{fontSize: "1.75rem!important", color: colors.gray.main}}>keyboard_arrow_down</Icon></InputAdornment>;
|
||||||
|
|
||||||
|
const fontSize = "1rem";
|
||||||
|
let optionPaddingLeftRems = 0.75;
|
||||||
|
if(startIcon)
|
||||||
|
{
|
||||||
|
optionPaddingLeftRems += allowBackAndForth ? 1.5 : 1.75
|
||||||
|
}
|
||||||
|
if(allowBackAndForth)
|
||||||
|
{
|
||||||
|
optionPaddingLeftRems += 2.5;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
dropdownOptions ? (
|
||||||
|
<Box sx={{whiteSpace: "nowrap", display: "flex",
|
||||||
|
"& .MuiPopperUnstyled-root": {
|
||||||
|
border: `1px solid ${colors.grayLines.main}`,
|
||||||
|
borderTop: "none",
|
||||||
|
borderRadius: "0 0 0.75rem 0.75rem",
|
||||||
|
padding: 0,
|
||||||
|
}, "& .MuiPaper-rounded": {
|
||||||
|
borderRadius: "0 0 0.75rem 0.75rem",
|
||||||
|
}
|
||||||
|
}} className="dashboardDropdownMenu">
|
||||||
|
<Autocomplete
|
||||||
|
id={`${label}-combo-box`}
|
||||||
|
|
||||||
|
defaultValue={defaultValue}
|
||||||
|
value={value}
|
||||||
|
onChange={handleOnChange}
|
||||||
|
inputValue={inputValue}
|
||||||
|
onInputChange={handleOnInputChange}
|
||||||
|
|
||||||
|
isOptionEqualToValue={(option, value) => option.id === value.id}
|
||||||
|
|
||||||
|
open={isOpen}
|
||||||
|
onOpen={() => setIsOpen(true)}
|
||||||
|
onClose={() => setIsOpen(false)}
|
||||||
|
|
||||||
|
size="small"
|
||||||
|
disablePortal
|
||||||
|
disableClearable={disableClearable}
|
||||||
|
options={dropdownOptions}
|
||||||
|
sx={{
|
||||||
|
...sx,
|
||||||
|
cursor: "pointer",
|
||||||
|
display: "inline-block",
|
||||||
|
"& .MuiOutlinedInput-notchedOutline": {
|
||||||
|
border: "none"
|
||||||
|
},
|
||||||
|
}}
|
||||||
|
renderInput={(params: any) =>
|
||||||
|
<>
|
||||||
|
<Box sx={{width: `${width}px`, background: "white", borderRadius: isOpen ? "0.75rem 0.75rem 0 0" : "0.75rem", border: `1px solid ${colors.grayLines.main}`, "& *": {cursor: "pointer"}}} display="flex" alignItems="center" onClick={(event) => doForceOpen(event)}>
|
||||||
|
{allowBackAndForth && <IconButton onClick={(event) => navigateBackAndForth(event, backAndForthInverted ? 1 : -1)} disabled={backDisabled}><Icon>navigate_before</Icon></IconButton>}
|
||||||
|
<TextField {...params} placeholder={label} sx={{
|
||||||
|
"& .MuiInputBase-input": {
|
||||||
|
fontSize: fontSize
|
||||||
|
}
|
||||||
|
}} InputProps={{...params.InputProps, startAdornment: startAdornment/*, endAdornment: endAdornment*/}}
|
||||||
|
/>
|
||||||
|
{allowBackAndForth && <IconButton onClick={(event) => navigateBackAndForth(event, backAndForthInverted ? -1 : 1)} disabled={forthDisabled}><Icon>navigate_next</Icon></IconButton>}
|
||||||
|
</Box>
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
renderOption={(props, option: DropdownOption) => (
|
||||||
|
<li {...props} style={{whiteSpace: "normal", fontSize: fontSize, paddingLeft: `${optionPaddingLeftRems}rem`}}>{option.label}</li>
|
||||||
|
)}
|
||||||
|
|
||||||
|
noOptionsText={<Box fontSize={fontSize}>No options found</Box>}
|
||||||
|
|
||||||
|
slotProps={{
|
||||||
|
popper: {
|
||||||
|
sx: {
|
||||||
|
width: `${width}px!important`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
{customTimes}
|
||||||
|
</Box>
|
||||||
|
) : null
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default WidgetDropdownMenu;
|
@ -286,18 +286,15 @@ export default function DataBagViewer({dataBagId}: Props): JSX.Element
|
|||||||
<Grid container spacing={3}>
|
<Grid container spacing={3}>
|
||||||
<Grid item xs={12}>
|
<Grid item xs={12}>
|
||||||
<>
|
<>
|
||||||
<Box display="flex" alignItems="center" justifyContent="space-between" gap={2} mt={-6}>
|
|
||||||
<Typography variant="h5" p={2}></Typography>
|
|
||||||
<Tabs
|
<Tabs
|
||||||
sx={{m: 1}}
|
sx={{m: 0, mb: 1, mt: -3}}
|
||||||
value={selectedTab}
|
value={selectedTab}
|
||||||
onChange={(event, newValue) => changeTab(newValue)}
|
onChange={(event, newValue) => changeTab(newValue)}
|
||||||
variant="standard"
|
variant="standard"
|
||||||
>
|
>
|
||||||
<Tab label="Raw Data" id="simple-tab-0" aria-controls="simple-tabpanel-0" sx={{width: "150px"}} />
|
<Tab label="Raw Data" id="simple-tab-0" aria-controls="simple-tabpanel-0" />
|
||||||
<Tab label="Data Preview" id="simple-tab-1" aria-controls="simple-tabpanel-1" sx={{width: "150px"}} />
|
<Tab label="Data Preview" id="simple-tab-1" aria-controls="simple-tabpanel-1" />
|
||||||
</Tabs>
|
</Tabs>
|
||||||
</Box>
|
|
||||||
|
|
||||||
<TabPanel index={0} value={selectedTab}>
|
<TabPanel index={0} value={selectedTab}>
|
||||||
<Grid container>
|
<Grid container>
|
||||||
|
@ -22,6 +22,7 @@
|
|||||||
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 {QRecord} from "@kingsrook/qqq-frontend-core/lib/model/QRecord";
|
import {QRecord} from "@kingsrook/qqq-frontend-core/lib/model/QRecord";
|
||||||
|
import Box from "@mui/material/Box";
|
||||||
import Button from "@mui/material/Button";
|
import Button from "@mui/material/Button";
|
||||||
import Icon from "@mui/material/Icon";
|
import Icon from "@mui/material/Icon";
|
||||||
import Tooltip from "@mui/material/Tooltip/Tooltip";
|
import Tooltip from "@mui/material/Tooltip/Tooltip";
|
||||||
@ -134,7 +135,7 @@ function RecordGridWidget({widgetMetaData, data}: Props): JSX.Element
|
|||||||
if(data && data.viewAllLink)
|
if(data && data.viewAllLink)
|
||||||
{
|
{
|
||||||
labelAdditionalElementsLeft.push(
|
labelAdditionalElementsLeft.push(
|
||||||
<Typography variant="body2" p={2} display="inline" fontSize=".875rem" pt="0" position="relative" top="-0.25rem">
|
<Typography variant="body2" p={2} display="inline" fontSize=".875rem" pt="0" position="relative">
|
||||||
<Link to={data.viewAllLink}>View All</Link>
|
<Link to={data.viewAllLink}>View All</Link>
|
||||||
</Typography>
|
</Typography>
|
||||||
)
|
)
|
||||||
@ -174,8 +175,8 @@ function RecordGridWidget({widgetMetaData, data}: Props): JSX.Element
|
|||||||
if(widgetMetaData?.showExportButton)
|
if(widgetMetaData?.showExportButton)
|
||||||
{
|
{
|
||||||
labelAdditionalElementsLeft.push(
|
labelAdditionalElementsLeft.push(
|
||||||
<Typography key={1} variant="body2" py={2} px={0} display="inline" position="relative" top="-0.375rem">
|
<Typography key={1} variant="body2" py={2} px={0} display="inline" position="relative">
|
||||||
<Tooltip title={tooltipTitle}><Button sx={{px: 1, py: 0, minWidth: "initial"}} onClick={onExportClick} disabled={isExportDisabled}><Icon>save_alt</Icon></Button></Tooltip>
|
<Tooltip title={tooltipTitle}><Button sx={{px: 1, py: 0, minWidth: "initial"}} onClick={onExportClick} disabled={isExportDisabled}><Icon sx={{color: "#757575", fontSize: 1.25}}>save_alt</Icon></Button></Tooltip>
|
||||||
</Typography>
|
</Typography>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -215,9 +216,16 @@ function RecordGridWidget({widgetMetaData, data}: Props): JSX.Element
|
|||||||
widgetData={data}
|
widgetData={data}
|
||||||
labelAdditionalElementsLeft={labelAdditionalElementsLeft}
|
labelAdditionalElementsLeft={labelAdditionalElementsLeft}
|
||||||
labelAdditionalComponentsRight={labelAdditionalComponentsRight}
|
labelAdditionalComponentsRight={labelAdditionalComponentsRight}
|
||||||
|
labelBoxAdditionalSx={{position: "relative", top: "-0.375rem"}}
|
||||||
>
|
>
|
||||||
|
<Box mx={-2} mb={-3}>
|
||||||
<DataGridPro
|
<DataGridPro
|
||||||
autoHeight
|
autoHeight
|
||||||
|
sx={{
|
||||||
|
borderBottom: "none",
|
||||||
|
borderLeft: "none",
|
||||||
|
borderRight: "none"
|
||||||
|
}}
|
||||||
rows={rows}
|
rows={rows}
|
||||||
disableSelectionOnClick
|
disableSelectionOnClick
|
||||||
columns={columns}
|
columns={columns}
|
||||||
@ -250,6 +258,7 @@ function RecordGridWidget({widgetMetaData, data}: Props): JSX.Element
|
|||||||
// sortingOrder={[ "asc", "desc" ]}
|
// sortingOrder={[ "asc", "desc" ]}
|
||||||
// sortModel={columnSortModel}
|
// sortModel={columnSortModel}
|
||||||
/>
|
/>
|
||||||
|
</Box>
|
||||||
</Widget>
|
</Widget>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -392,14 +392,8 @@ export default function ScriptViewer({scriptId, associatedScriptTableName, assoc
|
|||||||
return JSON.stringify(new QQueryFilter([new QFilterCriteria("scriptRevisionId", QCriteriaOperator.EQUALS, [scriptRevisionId])]));
|
return JSON.stringify(new QQueryFilter([new QFilterCriteria("scriptRevisionId", QCriteriaOperator.EQUALS, [scriptRevisionId])]));
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
position: relative;
|
|
||||||
left: -356px;
|
|
||||||
width: calc(100% + 380px);
|
|
||||||
*/
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Grid container className="scriptViewer">
|
<Grid container className="scriptViewer" m={-2} pt={4} width={"calc(100% + 2rem)"}>
|
||||||
<Grid item xs={12}>
|
<Grid item xs={12}>
|
||||||
<Box>
|
<Box>
|
||||||
{
|
{
|
||||||
@ -430,20 +424,17 @@ export default function ScriptViewer({scriptId, associatedScriptTableName, assoc
|
|||||||
<Grid container spacing={3}>
|
<Grid container spacing={3}>
|
||||||
<Grid item xs={12}>
|
<Grid item xs={12}>
|
||||||
<>
|
<>
|
||||||
<Box display="flex" alignItems="center" justifyContent="space-between" gap={2} mt={-6}>
|
|
||||||
<Typography variant="h5" p={2}></Typography>
|
|
||||||
<Tabs
|
<Tabs
|
||||||
sx={{m: 1}}
|
sx={{m: 0, mb: 1, mt: -3}}
|
||||||
value={selectedTab}
|
value={selectedTab}
|
||||||
onChange={(event, newValue) => changeTab(newValue)}
|
onChange={(event, newValue) => changeTab(newValue)}
|
||||||
variant="standard"
|
variant="standard"
|
||||||
>
|
>
|
||||||
<Tab label="Code" id="simple-tab-0" aria-controls="simple-tabpanel-0" sx={{width: "100px"}} />
|
<Tab label="Code" id="simple-tab-0" aria-controls="simple-tabpanel-0" />
|
||||||
<Tab label="Logs" id="simple-tab-1" aria-controls="simple-tabpanel-1" sx={{width: "100px"}} />
|
<Tab label="Logs" id="simple-tab-1" aria-controls="simple-tabpanel-1" />
|
||||||
<Tab label="Test" id="simple-tab-1" aria-controls="simple-tabpanel-2" sx={{width: "100px"}} />
|
<Tab label="Test" id="simple-tab-1" aria-controls="simple-tabpanel-2" />
|
||||||
<Tab label="Docs" id="simple-tab-1" aria-controls="simple-tabpanel-3" sx={{width: "100px"}} />
|
<Tab label="Docs" id="simple-tab-1" aria-controls="simple-tabpanel-3" />
|
||||||
</Tabs>
|
</Tabs>
|
||||||
</Box>
|
|
||||||
|
|
||||||
<TabPanel index={0} value={selectedTab}>
|
<TabPanel index={0} value={selectedTab}>
|
||||||
<Grid container>
|
<Grid container>
|
||||||
@ -498,7 +489,7 @@ export default function ScriptViewer({scriptId, associatedScriptTableName, assoc
|
|||||||
editorProps={{$blockScrolling: true}}
|
editorProps={{$blockScrolling: true}}
|
||||||
setOptions={{useWorker: false}}
|
setOptions={{useWorker: false}}
|
||||||
width="100%"
|
width="100%"
|
||||||
height="368px"
|
height="400px"
|
||||||
value={getSelectedFileCode()}
|
value={getSelectedFileCode()}
|
||||||
style={{borderTop: "1px solid lightgray", borderBottomRightRadius: "1rem"}}
|
style={{borderTop: "1px solid lightgray", borderBottomRightRadius: "1rem"}}
|
||||||
/>
|
/>
|
||||||
|
@ -30,7 +30,8 @@ import TableRow from "@mui/material/TableRow";
|
|||||||
import Tooltip from "@mui/material/Tooltip";
|
import Tooltip from "@mui/material/Tooltip";
|
||||||
import parse from "html-react-parser";
|
import parse from "html-react-parser";
|
||||||
import {useEffect, useMemo, useState} from "react";
|
import {useEffect, useMemo, useState} from "react";
|
||||||
import {useAsyncDebounce, useGlobalFilter, usePagination, useSortBy, useTable} from "react-table";
|
import {useAsyncDebounce, useGlobalFilter, usePagination, useSortBy, useTable, useExpanded} from "react-table";
|
||||||
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
import MDInput from "qqq/components/legacy/MDInput";
|
import MDInput from "qqq/components/legacy/MDInput";
|
||||||
import MDPagination from "qqq/components/legacy/MDPagination";
|
import MDPagination from "qqq/components/legacy/MDPagination";
|
||||||
import MDTypography from "qqq/components/legacy/MDTypography";
|
import MDTypography from "qqq/components/legacy/MDTypography";
|
||||||
@ -47,6 +48,8 @@ interface Props
|
|||||||
canSearch?: boolean;
|
canSearch?: boolean;
|
||||||
showTotalEntries?: boolean;
|
showTotalEntries?: boolean;
|
||||||
hidePaginationDropdown?: boolean;
|
hidePaginationDropdown?: boolean;
|
||||||
|
fixedStickyLastRow?: boolean;
|
||||||
|
fixedHeight?: number;
|
||||||
table: TableDataInput;
|
table: TableDataInput;
|
||||||
pagination?: {
|
pagination?: {
|
||||||
variant: "contained" | "gradient";
|
variant: "contained" | "gradient";
|
||||||
@ -56,6 +59,18 @@ interface Props
|
|||||||
noEndBorder?: boolean;
|
noEndBorder?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
DataTable.defaultProps = {
|
||||||
|
entriesPerPage: 10,
|
||||||
|
entriesPerPageOptions: ["5", "10", "15", "20", "25"],
|
||||||
|
canSearch: false,
|
||||||
|
showTotalEntries: true,
|
||||||
|
fixedStickyLastRow: false,
|
||||||
|
fixedHeight: null,
|
||||||
|
pagination: {variant: "gradient", color: "info"},
|
||||||
|
isSorted: true,
|
||||||
|
noEndBorder: false,
|
||||||
|
};
|
||||||
|
|
||||||
const NoMaxWidthTooltip = styled(({className, ...props}: TooltipProps) => (
|
const NoMaxWidthTooltip = styled(({className, ...props}: TooltipProps) => (
|
||||||
<Tooltip {...props} classes={{popper: className}} />
|
<Tooltip {...props} classes={{popper: className}} />
|
||||||
))({
|
))({
|
||||||
@ -71,6 +86,8 @@ function DataTable({
|
|||||||
hidePaginationDropdown,
|
hidePaginationDropdown,
|
||||||
canSearch,
|
canSearch,
|
||||||
showTotalEntries,
|
showTotalEntries,
|
||||||
|
fixedStickyLastRow,
|
||||||
|
fixedHeight,
|
||||||
table,
|
table,
|
||||||
pagination,
|
pagination,
|
||||||
isSorted,
|
isSorted,
|
||||||
@ -83,8 +100,77 @@ function DataTable({
|
|||||||
defaultValue = (entriesPerPage) ? entriesPerPage : "10";
|
defaultValue = (entriesPerPage) ? entriesPerPage : "10";
|
||||||
entries = entriesPerPageOptions ? entriesPerPageOptions : ["10", "25", "50", "100"];
|
entries = entriesPerPageOptions ? entriesPerPageOptions : ["10", "25", "50", "100"];
|
||||||
|
|
||||||
const columns = useMemo<any>(() => table.columns, [table]);
|
let widths = [];
|
||||||
|
for(let i = 0; i<table.columns.length; i++)
|
||||||
|
{
|
||||||
|
const column = table.columns[i];
|
||||||
|
if(column.type !== "hidden")
|
||||||
|
{
|
||||||
|
widths.push(table.columns[i].width ?? "1fr");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let showExpandColumn = false;
|
||||||
|
if(table.rows)
|
||||||
|
{
|
||||||
|
for (let i = 0; i < table.rows.length; i++)
|
||||||
|
{
|
||||||
|
if (table.rows[i].subRows)
|
||||||
|
{
|
||||||
|
showExpandColumn = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const columnsToMemo = [...table.columns];
|
||||||
|
if(showExpandColumn)
|
||||||
|
{
|
||||||
|
widths.push("60px");
|
||||||
|
columnsToMemo.push(
|
||||||
|
{
|
||||||
|
///////////////////////////////
|
||||||
|
// Build our expander column //
|
||||||
|
///////////////////////////////
|
||||||
|
id: "__expander",
|
||||||
|
width: 60,
|
||||||
|
|
||||||
|
////////////////////////////////////////////////
|
||||||
|
// use this block if we want to do expand-all //
|
||||||
|
////////////////////////////////////////////////
|
||||||
|
// @ts-ignore
|
||||||
|
// header: ({getToggleAllRowsExpandedProps, isAllRowsExpanded}) => (
|
||||||
|
// <span {...getToggleAllRowsExpandedProps()}>
|
||||||
|
// {isAllRowsExpanded ? "yes" : "no"}
|
||||||
|
// </span>
|
||||||
|
// ),
|
||||||
|
header: () => (<span />),
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
cell: ({row}) =>
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Use the row.canExpand and row.getToggleRowExpandedProps prop getter to build the toggle for expanding a row //
|
||||||
|
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
row.canExpand ? (
|
||||||
|
<span
|
||||||
|
{...row.getToggleRowExpandedProps({
|
||||||
|
//////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// We could use the row.depth property and paddingLeft to indicate the depth of the row //
|
||||||
|
//////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// style: {paddingLeft: `${row.depth * 2}rem`,},
|
||||||
|
})}
|
||||||
|
>
|
||||||
|
{/* float this icon to keep it "out of the flow" - in other words, to keep it from making the row taller than it otherwise would be... */}
|
||||||
|
<Icon fontSize="medium" sx={{float: "left"}}>{row.isExpanded ? "expand_less" : "chevron_right"}</Icon>
|
||||||
|
</span>
|
||||||
|
) : null,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const columns = useMemo<any>(() => columnsToMemo, [table]);
|
||||||
const data = useMemo<any>(() => table.rows, [table]);
|
const data = useMemo<any>(() => table.rows, [table]);
|
||||||
|
const gridTemplateColumns = widths.join(" ");
|
||||||
|
|
||||||
if (!columns || !data)
|
if (!columns || !data)
|
||||||
{
|
{
|
||||||
@ -95,6 +181,7 @@ function DataTable({
|
|||||||
{columns, data, initialState: {pageIndex: 0}},
|
{columns, data, initialState: {pageIndex: 0}},
|
||||||
useGlobalFilter,
|
useGlobalFilter,
|
||||||
useSortBy,
|
useSortBy,
|
||||||
|
useExpanded,
|
||||||
usePagination
|
usePagination
|
||||||
);
|
);
|
||||||
|
|
||||||
@ -113,7 +200,7 @@ function DataTable({
|
|||||||
previousPage,
|
previousPage,
|
||||||
setPageSize,
|
setPageSize,
|
||||||
setGlobalFilter,
|
setGlobalFilter,
|
||||||
state: {pageIndex, pageSize, globalFilter},
|
state: {pageIndex, pageSize, globalFilter, expanded},
|
||||||
}: any = tableInstance;
|
}: any = tableInstance;
|
||||||
|
|
||||||
// Set the default value for the entries per page when component mounts
|
// Set the default value for the entries per page when component mounts
|
||||||
@ -193,6 +280,126 @@ function DataTable({
|
|||||||
entriesEnd = pageSize * (pageIndex + 1);
|
entriesEnd = pageSize * (pageIndex + 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getTable(includeHead: boolean, rows: any, isFooter: boolean)
|
||||||
|
{
|
||||||
|
let boxStyle = {};
|
||||||
|
if(fixedStickyLastRow)
|
||||||
|
{
|
||||||
|
boxStyle = isFooter
|
||||||
|
? {borderTop: `0.0625rem solid ${colors.grayLines.main};`, overflow: "auto", scrollbarGutter: "stable"}
|
||||||
|
: {height: fixedHeight ? `${fixedHeight}px` : "360px", overflowY: "scroll", scrollbarGutter: "stable", marginBottom: "-1px"};
|
||||||
|
}
|
||||||
|
|
||||||
|
return <Box sx={boxStyle}>
|
||||||
|
<Table {...getTableProps()}>
|
||||||
|
{
|
||||||
|
includeHead && (
|
||||||
|
<Box component="thead" sx={{position: "sticky", top: 0, background: "white"}}>
|
||||||
|
{headerGroups.map((headerGroup: any, i: number) => (
|
||||||
|
<TableRow key={i} {...headerGroup.getHeaderGroupProps()} sx={{display: "grid", gridTemplateColumns: gridTemplateColumns}}>
|
||||||
|
{headerGroup.headers.map((column: any) => (
|
||||||
|
column.type !== "hidden" && (
|
||||||
|
<DataTableHeadCell
|
||||||
|
key={i++}
|
||||||
|
{...column.getHeaderProps(isSorted && column.getSortByToggleProps())}
|
||||||
|
align={column.align ? column.align : "left"}
|
||||||
|
sorted={setSortedValue(column)}
|
||||||
|
>
|
||||||
|
{column.render("header")}
|
||||||
|
</DataTableHeadCell>
|
||||||
|
)
|
||||||
|
))}
|
||||||
|
</TableRow>
|
||||||
|
))}
|
||||||
|
</Box>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
<TableBody {...getTableBodyProps()}>
|
||||||
|
{rows.map((row: any, key: any) =>
|
||||||
|
{
|
||||||
|
prepareRow(row);
|
||||||
|
|
||||||
|
let overrideNoEndBorder = false;
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////////////
|
||||||
|
// don't do an end-border on nested rows - unless they're the last one in a set //
|
||||||
|
//////////////////////////////////////////////////////////////////////////////////
|
||||||
|
if(row.depth > 0)
|
||||||
|
{
|
||||||
|
overrideNoEndBorder = true;
|
||||||
|
if(key + 1 < rows.length && rows[key + 1].depth == 0)
|
||||||
|
{
|
||||||
|
overrideNoEndBorder = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////
|
||||||
|
// don't do end-border on the footer //
|
||||||
|
///////////////////////////////////////
|
||||||
|
if(isFooter)
|
||||||
|
{
|
||||||
|
overrideNoEndBorder = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<TableRow sx={{verticalAlign: "top", display: "grid", gridTemplateColumns: gridTemplateColumns, background: (row.depth > 0 ? "#FAFAFA" : "initial")}} key={key} {...row.getRowProps()}>
|
||||||
|
{row.cells.map((cell: any) => (
|
||||||
|
cell.column.type !== "hidden" && (
|
||||||
|
<DataTableBodyCell
|
||||||
|
key={key}
|
||||||
|
noBorder={noEndBorder || overrideNoEndBorder}
|
||||||
|
depth={row.depth}
|
||||||
|
align={cell.column.align ? cell.column.align : "left"}
|
||||||
|
{...cell.getCellProps()}
|
||||||
|
>
|
||||||
|
{
|
||||||
|
cell.column.type === "default" && (
|
||||||
|
cell.value && "number" === typeof cell.value ? (
|
||||||
|
<DefaultCell isFooter={isFooter}>{cell.value.toLocaleString()}</DefaultCell>
|
||||||
|
) : (<DefaultCell isFooter={isFooter}>{cell.render("Cell")}</DefaultCell>)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
{
|
||||||
|
cell.column.type === "htmlAndTooltip" && (
|
||||||
|
<DefaultCell isFooter={isFooter}>
|
||||||
|
<NoMaxWidthTooltip title={parse(row.values["tooltip"])}>
|
||||||
|
<Box>
|
||||||
|
{parse(cell.value)}
|
||||||
|
</Box>
|
||||||
|
</NoMaxWidthTooltip>
|
||||||
|
</DefaultCell>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
{
|
||||||
|
cell.column.type === "html" && (
|
||||||
|
<DefaultCell isFooter={isFooter}>{parse(cell.value)}</DefaultCell>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
{
|
||||||
|
cell.column.type === "image" && row.values["imageTotal"] && (
|
||||||
|
<ImageCell imageUrl={row.values["imageUrl"]} label={row.values["imageLabel"]} total={row.values["imageTotal"].toLocaleString()} totalType={row.values["imageTotalType"]} />
|
||||||
|
)
|
||||||
|
}
|
||||||
|
{
|
||||||
|
cell.column.type === "image" && !row.values["imageTotal"] && (
|
||||||
|
<ImageCell imageUrl={row.values["imageUrl"]} label={row.values["imageLabel"]} />
|
||||||
|
)
|
||||||
|
}
|
||||||
|
{
|
||||||
|
(cell.column.id === "__expander") && cell.render("cell")
|
||||||
|
}
|
||||||
|
</DataTableBodyCell>
|
||||||
|
)
|
||||||
|
))}
|
||||||
|
</TableRow>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
|
||||||
|
</TableBody>
|
||||||
|
</Table>
|
||||||
|
</Box>
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<TableContainer sx={{boxShadow: "none"}}>
|
<TableContainer sx={{boxShadow: "none"}}>
|
||||||
{entriesPerPage && ((hidePaginationDropdown !== undefined && !hidePaginationDropdown) || canSearch) ? (
|
{entriesPerPage && ((hidePaginationDropdown !== undefined && !hidePaginationDropdown) || canSearch) ? (
|
||||||
@ -240,82 +447,15 @@ function DataTable({
|
|||||||
)}
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
) : null}
|
) : null}
|
||||||
<Table {...getTableProps()}>
|
|
||||||
<Box component="thead">
|
|
||||||
{headerGroups.map((headerGroup: any, i: number) => (
|
|
||||||
<TableRow key={i} {...headerGroup.getHeaderGroupProps()}>
|
|
||||||
{headerGroup.headers.map((column: any) => (
|
|
||||||
column.type !== "hidden" && (
|
|
||||||
<DataTableHeadCell
|
|
||||||
key={i++}
|
|
||||||
{...column.getHeaderProps(isSorted && column.getSortByToggleProps())}
|
|
||||||
width={column.width ? column.width : "auto"}
|
|
||||||
align={column.align ? column.align : "left"}
|
|
||||||
sorted={setSortedValue(column)}
|
|
||||||
>
|
|
||||||
{column.render("header")}
|
|
||||||
</DataTableHeadCell>
|
|
||||||
)
|
|
||||||
))}
|
|
||||||
</TableRow>
|
|
||||||
))}
|
|
||||||
</Box>
|
|
||||||
<TableBody {...getTableBodyProps()}>
|
|
||||||
{page.map((row: any, key: any) =>
|
|
||||||
{
|
|
||||||
prepareRow(row);
|
|
||||||
return (
|
|
||||||
<TableRow sx={{verticalAlign: "top"}} key={key} {...row.getRowProps()}>
|
|
||||||
{row.cells.map((cell: any) => (
|
|
||||||
cell.column.type !== "hidden" && (
|
|
||||||
<DataTableBodyCell
|
|
||||||
key={key}
|
|
||||||
noBorder={noEndBorder && rows.length - 1 === key}
|
|
||||||
align={cell.column.align ? cell.column.align : "left"}
|
|
||||||
{...cell.getCellProps()}
|
|
||||||
>
|
|
||||||
{
|
|
||||||
cell.column.type === "default" && (
|
|
||||||
cell.value && "number" === typeof cell.value ? (
|
|
||||||
<DefaultCell>{cell.value.toLocaleString()}</DefaultCell>
|
|
||||||
) : (<DefaultCell>{cell.render("Cell")}</DefaultCell>)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
{
|
|
||||||
cell.column.type === "htmlAndTooltip" && (
|
|
||||||
<DefaultCell>
|
|
||||||
|
|
||||||
<NoMaxWidthTooltip title={parse(row.values["tooltip"])}>
|
|
||||||
<Box>
|
|
||||||
{parse(cell.value)}
|
|
||||||
</Box>
|
|
||||||
</NoMaxWidthTooltip>
|
|
||||||
</DefaultCell>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
{
|
{
|
||||||
cell.column.type === "html" && (
|
fixedStickyLastRow ? (
|
||||||
<DefaultCell>{parse(cell.value)}</DefaultCell>
|
<>
|
||||||
)
|
{getTable(true, page.slice(0, page.length -1), false)}
|
||||||
|
{getTable(false, page.slice(page.length-1), true)}
|
||||||
|
</>
|
||||||
|
) : getTable(true, page, false)
|
||||||
}
|
}
|
||||||
{
|
|
||||||
cell.column.type === "image" && row.values["imageTotal"] && (
|
|
||||||
<ImageCell imageUrl={row.values["imageUrl"]} label={row.values["imageLabel"]} total={row.values["imageTotal"].toLocaleString()} totalType={row.values["imageTotalType"]} />
|
|
||||||
)
|
|
||||||
}
|
|
||||||
{
|
|
||||||
cell.column.type === "image" && !row.values["imageTotal"] && (
|
|
||||||
<ImageCell imageUrl={row.values["imageUrl"]} label={row.values["imageLabel"]} />
|
|
||||||
)
|
|
||||||
}
|
|
||||||
</DataTableBodyCell>
|
|
||||||
)
|
|
||||||
))}
|
|
||||||
</TableRow>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</TableBody>
|
|
||||||
</Table>
|
|
||||||
|
|
||||||
<Box
|
<Box
|
||||||
display="flex"
|
display="flex"
|
||||||
@ -368,15 +508,4 @@ function DataTable({
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Declaring default props for DataTable
|
|
||||||
DataTable.defaultProps = {
|
|
||||||
entriesPerPage: 10,
|
|
||||||
entriesPerPageOptions: ["5", "10", "15", "20", "25"],
|
|
||||||
canSearch: false,
|
|
||||||
showTotalEntries: true,
|
|
||||||
pagination: {variant: "gradient", color: "info"},
|
|
||||||
isSorted: true,
|
|
||||||
noEndBorder: false,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default DataTable;
|
export default DataTable;
|
||||||
|
@ -54,11 +54,13 @@ interface Props
|
|||||||
noRowsFoundHTML?: string;
|
noRowsFoundHTML?: string;
|
||||||
rowsPerPage?: number;
|
rowsPerPage?: number;
|
||||||
hidePaginationDropdown?: boolean;
|
hidePaginationDropdown?: boolean;
|
||||||
|
fixedStickyLastRow?: boolean;
|
||||||
|
fixedHeight?: number;
|
||||||
data: TableDataInput;
|
data: TableDataInput;
|
||||||
}
|
}
|
||||||
|
|
||||||
const qController = Client.getInstance();
|
const qController = Client.getInstance();
|
||||||
function TableCard({noRowsFoundHTML, data, rowsPerPage, hidePaginationDropdown}: Props): JSX.Element
|
function TableCard({noRowsFoundHTML, data, rowsPerPage, hidePaginationDropdown, fixedStickyLastRow, fixedHeight}: Props): JSX.Element
|
||||||
{
|
{
|
||||||
const [qInstance, setQInstance] = useState(null as QInstance);
|
const [qInstance, setQInstance] = useState(null as QInstance);
|
||||||
|
|
||||||
@ -72,16 +74,17 @@ function TableCard({noRowsFoundHTML, data, rowsPerPage, hidePaginationDropdown}:
|
|||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box py={1}>
|
<Box py={1} mx={-2}>
|
||||||
{
|
{
|
||||||
data && data.columns && !noRowsFoundHTML ?
|
data && data.columns && !noRowsFoundHTML ?
|
||||||
<DataTable
|
<DataTable
|
||||||
table={data}
|
table={data}
|
||||||
entriesPerPage={rowsPerPage}
|
entriesPerPage={rowsPerPage}
|
||||||
hidePaginationDropdown={hidePaginationDropdown}
|
hidePaginationDropdown={hidePaginationDropdown}
|
||||||
|
fixedStickyLastRow={fixedStickyLastRow}
|
||||||
|
fixedHeight={fixedHeight}
|
||||||
showTotalEntries={false}
|
showTotalEntries={false}
|
||||||
isSorted={false}
|
isSorted={false}
|
||||||
noEndBorder
|
|
||||||
/>
|
/>
|
||||||
: noRowsFoundHTML ?
|
: noRowsFoundHTML ?
|
||||||
<Box p={3} pt={1} pb={1} sx={{textAlign: "center"}}>
|
<Box p={3} pt={1} pb={1} sx={{textAlign: "center"}}>
|
||||||
@ -114,7 +117,7 @@ function TableCard({noRowsFoundHTML, data, rowsPerPage, hidePaginationDropdown}:
|
|||||||
<TableRow sx={{verticalAlign: "top"}} key={`row-${i}`}>
|
<TableRow sx={{verticalAlign: "top"}} key={`row-${i}`}>
|
||||||
{Array(8).fill(0).map((_, j) =>
|
{Array(8).fill(0).map((_, j) =>
|
||||||
<DataTableBodyCell key={`cell-${i}-${j}`} align="center">
|
<DataTableBodyCell key={`cell-${i}-${j}`} align="center">
|
||||||
<DefaultCell><Skeleton /></DefaultCell>
|
<DefaultCell isFooter={false}><Skeleton /></DefaultCell>
|
||||||
</DataTableBodyCell>
|
</DataTableBodyCell>
|
||||||
)}
|
)}
|
||||||
</TableRow>
|
</TableRow>
|
||||||
|
@ -28,6 +28,7 @@ import Typography from "@mui/material/Typography";
|
|||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
import {htmlToText} from "html-to-text";
|
import {htmlToText} from "html-to-text";
|
||||||
import React, {useEffect, useState} from "react";
|
import React, {useEffect, useState} from "react";
|
||||||
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
import TableCard from "qqq/components/widgets/tables/TableCard";
|
import TableCard from "qqq/components/widgets/tables/TableCard";
|
||||||
import Widget, {WidgetData} from "qqq/components/widgets/Widget";
|
import Widget, {WidgetData} from "qqq/components/widgets/Widget";
|
||||||
import HtmlUtils from "qqq/utils/HtmlUtils";
|
import HtmlUtils from "qqq/utils/HtmlUtils";
|
||||||
@ -124,8 +125,8 @@ function TableWidget(props: Props): JSX.Element
|
|||||||
if(props.widgetMetaData?.showExportButton)
|
if(props.widgetMetaData?.showExportButton)
|
||||||
{
|
{
|
||||||
labelAdditionalElementsLeft.push(
|
labelAdditionalElementsLeft.push(
|
||||||
<Typography key={1} variant="body2" py={2} px={0} display="inline" position="relative" top="-0.375rem">
|
<Typography key={1} variant="body2" py={2} px={0} display="inline" position="relative" top="-0.25rem">
|
||||||
<Tooltip title="Export"><Button sx={{px: 1, py: 0, minWidth: "initial"}} onClick={onExportClick} disabled={false}><Icon>save_alt</Icon></Button></Tooltip>
|
<Tooltip title="Export"><Button sx={{px: 1, py: 0, minWidth: "initial"}} onClick={onExportClick} disabled={false}><Icon sx={{color: colors.gray.main, fontSize: 1.125}}>save_alt</Icon></Button></Tooltip>
|
||||||
</Typography>
|
</Typography>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -143,6 +144,8 @@ function TableWidget(props: Props): JSX.Element
|
|||||||
noRowsFoundHTML={props.widgetData?.noRowsFoundHTML}
|
noRowsFoundHTML={props.widgetData?.noRowsFoundHTML}
|
||||||
rowsPerPage={props.widgetData?.rowsPerPage}
|
rowsPerPage={props.widgetData?.rowsPerPage}
|
||||||
hidePaginationDropdown={props.widgetData?.hidePaginationDropdown}
|
hidePaginationDropdown={props.widgetData?.hidePaginationDropdown}
|
||||||
|
fixedStickyLastRow={props.widgetData?.fixedStickyLastRow}
|
||||||
|
fixedHeight={props.widgetData?.fixedHeight}
|
||||||
data={{columns: props.widgetData?.columns, rows: props.widgetData?.rows}}
|
data={{columns: props.widgetData?.columns, rows: props.widgetData?.rows}}
|
||||||
/>
|
/>
|
||||||
</Widget>
|
</Widget>
|
||||||
|
@ -22,6 +22,7 @@
|
|||||||
import Box from "@mui/material/Box";
|
import Box from "@mui/material/Box";
|
||||||
import {Theme} from "@mui/material/styles";
|
import {Theme} from "@mui/material/styles";
|
||||||
import {ReactNode} from "react";
|
import {ReactNode} from "react";
|
||||||
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
|
|
||||||
// Declaring prop types for DataTableBodyCell
|
// Declaring prop types for DataTableBodyCell
|
||||||
interface Props
|
interface Props
|
||||||
@ -40,14 +41,26 @@ function DataTableBodyCell({noBorder, align, children}: Props): JSX.Element
|
|||||||
py={1.5}
|
py={1.5}
|
||||||
px={3}
|
px={3}
|
||||||
sx={({palette: {light}, typography: {size}, borders: {borderWidth}}: Theme) => ({
|
sx={({palette: {light}, typography: {size}, borders: {borderWidth}}: Theme) => ({
|
||||||
fontSize: size.sm,
|
borderBottom: noBorder ? "none" : `${borderWidth[1]} solid ${colors.grayLines.main}`,
|
||||||
borderBottom: noBorder ? "none" : `${borderWidth[1]} solid ${light.main}`,
|
fontSize: "0.875rem",
|
||||||
|
"@media (min-width: 1440px)": {
|
||||||
|
fontSize: "1rem"
|
||||||
|
},
|
||||||
|
"@media (max-width: 1440px)": {
|
||||||
|
fontSize: "0.875rem"
|
||||||
|
},
|
||||||
|
"&:nth-child(1)": {
|
||||||
|
paddingLeft: "1rem"
|
||||||
|
},
|
||||||
|
"&:last-child": {
|
||||||
|
paddingRight: "1rem"
|
||||||
|
}
|
||||||
})}
|
})}
|
||||||
>
|
>
|
||||||
<Box
|
<Box
|
||||||
display="initial"
|
display="initial"
|
||||||
width="max-content"
|
width="max-content"
|
||||||
color="text"
|
color={colors.dark.main}
|
||||||
>
|
>
|
||||||
{children}
|
{children}
|
||||||
</Box>
|
</Box>
|
||||||
|
@ -23,6 +23,7 @@ import Box from "@mui/material/Box";
|
|||||||
import Icon from "@mui/material/Icon";
|
import Icon from "@mui/material/Icon";
|
||||||
import {Theme} from "@mui/material/styles";
|
import {Theme} from "@mui/material/styles";
|
||||||
import {ReactNode} from "react";
|
import {ReactNode} from "react";
|
||||||
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
import {useMaterialUIController} from "qqq/context";
|
import {useMaterialUIController} from "qqq/context";
|
||||||
|
|
||||||
// Declaring props types for DataTableHeadCell
|
// Declaring props types for DataTableHeadCell
|
||||||
@ -46,18 +47,28 @@ function DataTableHeadCell({width, children, sorted, align, ...rest}: Props): JS
|
|||||||
py={1.5}
|
py={1.5}
|
||||||
px={3}
|
px={3}
|
||||||
sx={({palette: {light}, borders: {borderWidth}}: Theme) => ({
|
sx={({palette: {light}, borders: {borderWidth}}: Theme) => ({
|
||||||
borderBottom: `${borderWidth[1]} solid ${light.main}`,
|
borderBottom: `${borderWidth[1]} solid ${colors.grayLines.main}`,
|
||||||
|
"&:nth-child(1)": {
|
||||||
|
paddingLeft: "1rem"
|
||||||
|
},
|
||||||
|
"&:last-child": {
|
||||||
|
paddingRight: "1rem"
|
||||||
|
},
|
||||||
})}
|
})}
|
||||||
>
|
>
|
||||||
<Box
|
<Box
|
||||||
{...rest}
|
{...rest}
|
||||||
sx={({typography: {size, fontWeightBold}}: Theme) => ({
|
sx={({typography: {size, fontWeightBold}}: Theme) => ({
|
||||||
position: "relative",
|
position: "relative",
|
||||||
opacity: "0.7",
|
color: colors.grey[700],
|
||||||
textAlign: align,
|
textAlign: align,
|
||||||
fontSize: size.xxs,
|
"@media (min-width: 1440px)": {
|
||||||
fontWeight: fontWeightBold,
|
fontSize: "1rem"
|
||||||
textTransform: "uppercase",
|
},
|
||||||
|
"@media (max-width: 1440px)": {
|
||||||
|
fontSize: "0.875rem"
|
||||||
|
},
|
||||||
|
fontWeight: 600,
|
||||||
cursor: sorted && "pointer",
|
cursor: sorted && "pointer",
|
||||||
userSelect: sorted && "none",
|
userSelect: sorted && "none",
|
||||||
})}
|
})}
|
||||||
|
@ -14,12 +14,31 @@ Coded by www.creative-tim.com
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import {ReactNode} from "react";
|
import {ReactNode} from "react";
|
||||||
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
import MDTypography from "qqq/components/legacy/MDTypography";
|
import MDTypography from "qqq/components/legacy/MDTypography";
|
||||||
|
|
||||||
function DefaultCell({children}: { children: ReactNode }): JSX.Element
|
interface Props
|
||||||
|
{
|
||||||
|
isFooter: boolean
|
||||||
|
children: ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function DefaultCell({isFooter, children}: Props): JSX.Element
|
||||||
{
|
{
|
||||||
return (
|
return (
|
||||||
<MDTypography variant="button" fontWeight="regular" color="text">
|
<MDTypography variant="button" color={colors.dark.main} sx={{
|
||||||
|
fontWeight: isFooter ? 600 : 500,
|
||||||
|
"@media (min-width: 1440px)": {
|
||||||
|
fontSize: "1rem"
|
||||||
|
},
|
||||||
|
"@media (max-width: 1440px)": {
|
||||||
|
fontSize: "0.875rem"
|
||||||
|
},
|
||||||
|
"& a": {
|
||||||
|
color: colors.blueGray.main
|
||||||
|
}
|
||||||
|
}}>
|
||||||
{children}
|
{children}
|
||||||
</MDTypography>
|
</MDTypography>
|
||||||
);
|
);
|
||||||
|
@ -38,11 +38,11 @@ function DashboardLayout({children}: { children: ReactNode }): JSX.Element
|
|||||||
return (
|
return (
|
||||||
<Box
|
<Box
|
||||||
sx={({breakpoints, transitions, functions: {pxToRem}}) => ({
|
sx={({breakpoints, transitions, functions: {pxToRem}}) => ({
|
||||||
p: 3,
|
p: "20px",
|
||||||
position: "relative",
|
position: "relative",
|
||||||
|
|
||||||
[breakpoints.up("xl")]: {
|
[breakpoints.up("xl")]: {
|
||||||
marginLeft: miniSidenav ? pxToRem(120) : pxToRem(274),
|
marginLeft: miniSidenav ? pxToRem(120) : pxToRem(245),
|
||||||
transition: transitions.create(["margin-left", "margin-right"], {
|
transition: transitions.create(["margin-left", "margin-right"], {
|
||||||
easing: transitions.easing.easeInOut,
|
easing: transitions.easing.easeInOut,
|
||||||
duration: transitions.duration.standard,
|
duration: transitions.duration.standard,
|
||||||
|
@ -204,7 +204,9 @@ function AppHome({app}: Props): JSX.Element
|
|||||||
<BaseLayout>
|
<BaseLayout>
|
||||||
<Box>
|
<Box>
|
||||||
{app.widgets && (
|
{app.widgets && (
|
||||||
|
<Box pb={app.sections ? 2.375 : 0}>
|
||||||
<DashboardWidgets widgetMetaDataList={widgets} />
|
<DashboardWidgets widgetMetaDataList={widgets} />
|
||||||
|
</Box>
|
||||||
)}
|
)}
|
||||||
<Grid container spacing={3}>
|
<Grid container spacing={3}>
|
||||||
{
|
{
|
||||||
|
@ -323,7 +323,7 @@ function ProcessRun({process, table, defaultProcessValues, isModal, isWidget, is
|
|||||||
<Grid m={3} mt={9} container>
|
<Grid m={3} mt={9} container>
|
||||||
<Grid item xs={0} lg={3} />
|
<Grid item xs={0} lg={3} />
|
||||||
<Grid item xs={12} lg={6}>
|
<Grid item xs={12} lg={6}>
|
||||||
<Card elevation={5}>
|
<Card>
|
||||||
<Box p={3}>
|
<Box p={3}>
|
||||||
<MDTypography variant="h5" component="div">
|
<MDTypography variant="h5" component="div">
|
||||||
Working
|
Working
|
||||||
@ -414,7 +414,15 @@ function ProcessRun({process, table, defaultProcessValues, isModal, isWidget, is
|
|||||||
//////////////////////////////////////////////////
|
//////////////////////////////////////////////////
|
||||||
// render all of the components for this screen //
|
// render all of the components for this screen //
|
||||||
//////////////////////////////////////////////////
|
//////////////////////////////////////////////////
|
||||||
step.components && (step.components.map((component: QFrontendComponent, index: number) => (
|
step.components && (step.components.map((component: QFrontendComponent, index: number) =>
|
||||||
|
{
|
||||||
|
let helpRoles = ["PROCESS_SCREEN", "ALL_SCREENS"]
|
||||||
|
if(component.type == QComponentType.BULK_EDIT_FORM)
|
||||||
|
{
|
||||||
|
helpRoles = ["EDIT_SCREEN", "WRITE_SCREENS", "ALL_SCREENS"]
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
<div key={index}>
|
<div key={index}>
|
||||||
{
|
{
|
||||||
component.type === QComponentType.HELP_TEXT && (
|
component.type === QComponentType.HELP_TEXT && (
|
||||||
@ -454,8 +462,8 @@ function ProcessRun({process, table, defaultProcessValues, isModal, isWidget, is
|
|||||||
}
|
}
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid item xs={12} lg={9}>
|
<Grid item xs={12} lg={9}>
|
||||||
|
{
|
||||||
{localTableSections.map((section: QTableSection, index: number) =>
|
localTableSections.map((section: QTableSection, index: number) =>
|
||||||
{
|
{
|
||||||
const name = section.name;
|
const name = section.name;
|
||||||
|
|
||||||
@ -491,7 +499,7 @@ function ProcessRun({process, table, defaultProcessValues, isModal, isWidget, is
|
|||||||
{section.label}
|
{section.label}
|
||||||
</MDTypography>
|
</MDTypography>
|
||||||
<Box px={2}>
|
<Box px={2}>
|
||||||
<QDynamicForm formData={sectionFormData} bulkEditMode bulkEditSwitchChangeHandler={bulkEditSwitchChanged} />
|
<QDynamicForm formData={sectionFormData} bulkEditMode bulkEditSwitchChangeHandler={bulkEditSwitchChanged} helpRoles={helpRoles} helpContentKeyPrefix={`table:${tableMetaData?.name};`} />
|
||||||
</Box>
|
</Box>
|
||||||
</Card>
|
</Card>
|
||||||
</Box>
|
</Box>
|
||||||
@ -501,16 +509,16 @@ function ProcessRun({process, table, defaultProcessValues, isModal, isWidget, is
|
|||||||
{
|
{
|
||||||
return (<br />);
|
return (<br />);
|
||||||
}
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
)}
|
|
||||||
</Grid>
|
</Grid>
|
||||||
</Grid>
|
</Grid>
|
||||||
: <QDynamicForm formData={formData} bulkEditMode bulkEditSwitchChangeHandler={bulkEditSwitchChanged} />
|
: <QDynamicForm formData={formData} bulkEditMode bulkEditSwitchChangeHandler={bulkEditSwitchChanged} helpRoles={helpRoles} helpContentKeyPrefix={`table:${tableMetaData?.name};`} />
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
component.type === QComponentType.EDIT_FORM && (
|
component.type === QComponentType.EDIT_FORM && (
|
||||||
<QDynamicForm formData={formData} />
|
<QDynamicForm formData={formData} helpRoles={helpRoles} helpContentKeyPrefix={`process:${processName};`} />
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
@ -635,7 +643,9 @@ function ProcessRun({process, table, defaultProcessValues, isModal, isWidget, is
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
</div>
|
</div>
|
||||||
)))}
|
);
|
||||||
|
}))
|
||||||
|
}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
@ -1278,6 +1288,7 @@ function ProcessRun({process, table, defaultProcessValues, isModal, isWidget, is
|
|||||||
{
|
{
|
||||||
mainCardStyles.background = "none";
|
mainCardStyles.background = "none";
|
||||||
mainCardStyles.boxShadow = "none";
|
mainCardStyles.boxShadow = "none";
|
||||||
|
mainCardStyles.border = "none";
|
||||||
mainCardStyles.minHeight = "";
|
mainCardStyles.minHeight = "";
|
||||||
mainCardStyles.alignItems = "stretch";
|
mainCardStyles.alignItems = "stretch";
|
||||||
mainCardStyles.flexGrow = 1;
|
mainCardStyles.flexGrow = 1;
|
||||||
|
@ -34,12 +34,8 @@ function EntityCreate({table}: Props): JSX.Element
|
|||||||
{
|
{
|
||||||
return (
|
return (
|
||||||
<BaseLayout>
|
<BaseLayout>
|
||||||
<Box mt={4}>
|
<Box mb={3}>
|
||||||
<Grid container spacing={3}>
|
|
||||||
<Grid item xs={12} lg={12}>
|
|
||||||
<EntityForm table={table} />
|
<EntityForm table={table} />
|
||||||
</Grid>
|
|
||||||
</Grid>
|
|
||||||
</Box>
|
</Box>
|
||||||
</BaseLayout>
|
</BaseLayout>
|
||||||
);
|
);
|
||||||
|
@ -43,18 +43,8 @@ function EntityEdit({table, isCopy}: Props): JSX.Element
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<BaseLayout>
|
<BaseLayout>
|
||||||
<Box mt={4}>
|
|
||||||
<Grid container spacing={3}>
|
|
||||||
<Grid item xs={12} lg={12}>
|
|
||||||
<Box mb={3}>
|
<Box mb={3}>
|
||||||
<Grid container spacing={3}>
|
|
||||||
<Grid item xs={12}>
|
|
||||||
<EntityForm table={table} id={id} isCopy={isCopy} />
|
<EntityForm table={table} id={id} isCopy={isCopy} />
|
||||||
</Grid>
|
|
||||||
</Grid>
|
|
||||||
</Box>
|
|
||||||
</Grid>
|
|
||||||
</Grid>
|
|
||||||
</Box>
|
</Box>
|
||||||
</BaseLayout>
|
</BaseLayout>
|
||||||
);
|
);
|
||||||
|
@ -1118,7 +1118,7 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
exportWindow.document.write(`<html lang="en">
|
exportWindow.document.write(`<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<style>
|
<style>
|
||||||
* { font-family: "Roboto","Helvetica","Arial",sans-serif; }
|
* { font-family: "SF Pro Display","Roboto","Helvetica","Arial",sans-serif; }
|
||||||
</style>
|
</style>
|
||||||
<title>${filename}</title>
|
<title>${filename}</title>
|
||||||
<script>
|
<script>
|
||||||
@ -2017,7 +2017,7 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
page={pageNumber}
|
page={pageNumber}
|
||||||
checkboxSelection
|
checkboxSelection
|
||||||
disableSelectionOnClick
|
disableSelectionOnClick
|
||||||
autoHeight
|
autoHeight={false}
|
||||||
rows={rows}
|
rows={rows}
|
||||||
// getRowHeight={() => "auto"} // maybe nice? wraps values in cells...
|
// getRowHeight={() => "auto"} // maybe nice? wraps values in cells...
|
||||||
columns={columnsModel}
|
columns={columnsModel}
|
||||||
@ -2041,6 +2041,7 @@ function RecordQuery({table, launchProcess}: Props): JSX.Element
|
|||||||
getRowId={(row) => row.__rowIndex}
|
getRowId={(row) => row.__rowIndex}
|
||||||
selectionModel={rowSelectionModel}
|
selectionModel={rowSelectionModel}
|
||||||
hideFooterSelectedRowCount={true}
|
hideFooterSelectedRowCount={true}
|
||||||
|
sx={{border: 0, height: "calc(100vh - 250px)"}}
|
||||||
/>
|
/>
|
||||||
</Box>
|
</Box>
|
||||||
</Card>
|
</Card>
|
||||||
|
@ -190,7 +190,7 @@ function RecordDeveloperView({table}: Props): JSX.Element
|
|||||||
return (
|
return (
|
||||||
<div key={fieldName}>
|
<div key={fieldName}>
|
||||||
<Card sx={{mb: 3}}>
|
<Card sx={{mb: 3}}>
|
||||||
<Typography variant="h6" p={2} pl={3} pb={1}>{field?.label}</Typography>
|
<Typography variant="h6" p={2} pl={3} pb={3}>{field?.label}</Typography>
|
||||||
|
|
||||||
<Box display="flex" alignItems="center" justifyContent="space-between" gap={2}>
|
<Box display="flex" alignItems="center" justifyContent="space-between" gap={2}>
|
||||||
{scriptId ?
|
{scriptId ?
|
||||||
|
@ -45,13 +45,16 @@ import ListItemIcon from "@mui/material/ListItemIcon";
|
|||||||
import Menu from "@mui/material/Menu";
|
import Menu from "@mui/material/Menu";
|
||||||
import MenuItem from "@mui/material/MenuItem";
|
import MenuItem from "@mui/material/MenuItem";
|
||||||
import Modal from "@mui/material/Modal";
|
import Modal from "@mui/material/Modal";
|
||||||
|
import Tooltip from "@mui/material/Tooltip/Tooltip";
|
||||||
import React, {useContext, useEffect, useState} from "react";
|
import React, {useContext, useEffect, useState} from "react";
|
||||||
import {useLocation, useNavigate, useParams} from "react-router-dom";
|
import {useLocation, useNavigate, useParams} from "react-router-dom";
|
||||||
import QContext from "QContext";
|
import QContext from "QContext";
|
||||||
|
import colors from "qqq/assets/theme/base/colors";
|
||||||
import AuditBody from "qqq/components/audits/AuditBody";
|
import AuditBody from "qqq/components/audits/AuditBody";
|
||||||
import {QActionsMenuButton, QCancelButton, QDeleteButton, QEditButton} from "qqq/components/buttons/DefaultButtons";
|
import {QActionsMenuButton, QCancelButton, QDeleteButton, QEditButton} from "qqq/components/buttons/DefaultButtons";
|
||||||
import EntityForm from "qqq/components/forms/EntityForm";
|
import EntityForm from "qqq/components/forms/EntityForm";
|
||||||
import {GotoRecordButton} from "qqq/components/misc/GotoRecordDialog";
|
import {GotoRecordButton} from "qqq/components/misc/GotoRecordDialog";
|
||||||
|
import HelpContent, {hasHelpContent} from "qqq/components/misc/HelpContent";
|
||||||
import QRecordSidebar from "qqq/components/misc/RecordSidebar";
|
import QRecordSidebar from "qqq/components/misc/RecordSidebar";
|
||||||
import DashboardWidgets from "qqq/components/widgets/DashboardWidgets";
|
import DashboardWidgets from "qqq/components/widgets/DashboardWidgets";
|
||||||
import BaseLayout from "qqq/layouts/BaseLayout";
|
import BaseLayout from "qqq/layouts/BaseLayout";
|
||||||
@ -98,6 +101,7 @@ function RecordView({table, launchProcess}: Props): JSX.Element
|
|||||||
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([] as QTableSection[]);
|
const [tableSections, setTableSections] = useState([] as QTableSection[]);
|
||||||
|
const [t1Section, setT1Section] = useState(null as QTableSection);
|
||||||
const [t1SectionName, setT1SectionName] = useState(null as string);
|
const [t1SectionName, setT1SectionName] = useState(null as string);
|
||||||
const [t1SectionElement, setT1SectionElement] = useState(null as JSX.Element);
|
const [t1SectionElement, setT1SectionElement] = useState(null as JSX.Element);
|
||||||
const [nonT1TableSections, setNonT1TableSections] = useState([] as QTableSection[]);
|
const [nonT1TableSections, setNonT1TableSections] = useState([] as QTableSection[]);
|
||||||
@ -117,7 +121,7 @@ function RecordView({table, launchProcess}: Props): JSX.Element
|
|||||||
const openActionsMenu = (event: any) => setActionsMenu(event.currentTarget);
|
const openActionsMenu = (event: any) => setActionsMenu(event.currentTarget);
|
||||||
const closeActionsMenu = () => setActionsMenu(null);
|
const closeActionsMenu = () => setActionsMenu(null);
|
||||||
|
|
||||||
const {accentColor, setPageHeader, tableMetaData, setTableMetaData, tableProcesses, setTableProcesses, dotMenuOpen, keyboardHelpOpen} = useContext(QContext);
|
const {accentColor, setPageHeader, tableMetaData, setTableMetaData, tableProcesses, setTableProcesses, dotMenuOpen, keyboardHelpOpen, helpHelpActive} = useContext(QContext);
|
||||||
|
|
||||||
if (localStorage.getItem(tableVariantLocalStorageKey))
|
if (localStorage.getItem(tableVariantLocalStorageKey))
|
||||||
{
|
{
|
||||||
@ -351,6 +355,23 @@ function RecordView({table, launchProcess}: Props): JSX.Element
|
|||||||
return (visibleJoinTables);
|
return (visibleJoinTables);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************************
|
||||||
|
** get an element (or empty) to use as help content for a section
|
||||||
|
*******************************************************************************/
|
||||||
|
const getSectionHelp = (section: QTableSection) =>
|
||||||
|
{
|
||||||
|
const helpRoles = ["VIEW_SCREEN", "READ_SCREENS", "ALL_SCREENS"]
|
||||||
|
const formattedHelpContent = <HelpContent helpContents={section.helpContents} roles={helpRoles} helpContentKey={`table:${tableName};section:${section.name}`} />;
|
||||||
|
|
||||||
|
return formattedHelpContent && (
|
||||||
|
<Box px={"1.5rem"} fontSize={"0.875rem"} color={colors.blueGray.main}>
|
||||||
|
{formattedHelpContent}
|
||||||
|
</Box>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
if (!asyncLoadInited)
|
if (!asyncLoadInited)
|
||||||
{
|
{
|
||||||
setAsyncLoadInited(true);
|
setAsyncLoadInited(true);
|
||||||
@ -502,15 +523,24 @@ function RecordView({table, launchProcess}: Props): JSX.Element
|
|||||||
{
|
{
|
||||||
let [field, tableForField] = TableUtils.getFieldAndTable(tableMetaData, fieldName);
|
let [field, tableForField] = TableUtils.getFieldAndTable(tableMetaData, fieldName);
|
||||||
let label = field.label;
|
let label = field.label;
|
||||||
|
|
||||||
|
const helpRoles = ["VIEW_SCREEN", "READ_SCREENS", "ALL_SCREENS"]
|
||||||
|
const showHelp = helpHelpActive || hasHelpContent(field.helpContents, helpRoles);
|
||||||
|
const formattedHelpContent = <HelpContent helpContents={field.helpContents} roles={helpRoles} heading={label} helpContentKey={`table:${tableName};field:${fieldName}`} />;
|
||||||
|
|
||||||
|
const labelElement = <Typography variant="button" textTransform="none" fontWeight="bold" pr={1} color="rgb(52, 71, 103)" sx={{cursor: "default"}}>{label}:</Typography>
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box key={fieldName} flexDirection="row" pr={2}>
|
<Box key={fieldName} flexDirection="row" pr={2}>
|
||||||
<Typography variant="button" textTransform="none" fontWeight="bold" pr={1} color="rgb(52, 71, 103)">
|
<>
|
||||||
{label}:
|
{
|
||||||
|
showHelp && formattedHelpContent ? <Tooltip title={formattedHelpContent}>{labelElement}</Tooltip> : labelElement
|
||||||
|
}
|
||||||
<div style={{display: "inline-block", width: 0}}> </div>
|
<div style={{display: "inline-block", width: 0}}> </div>
|
||||||
</Typography>
|
|
||||||
<Typography variant="button" textTransform="none" fontWeight="regular" color="rgb(123, 128, 154)">
|
<Typography variant="button" textTransform="none" fontWeight="regular" color="rgb(123, 128, 154)">
|
||||||
{ValueUtils.getDisplayValue(field, record, "view", fieldName)}
|
{ValueUtils.getDisplayValue(field, record, "view", fieldName)}
|
||||||
</Typography>
|
</Typography>
|
||||||
|
</>
|
||||||
</Box>
|
</Box>
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
@ -531,6 +561,7 @@ function RecordView({table, launchProcess}: Props): JSX.Element
|
|||||||
<Typography variant="h6" p={3} pb={1}>
|
<Typography variant="h6" p={3} pb={1}>
|
||||||
{section.label}
|
{section.label}
|
||||||
</Typography>
|
</Typography>
|
||||||
|
{getSectionHelp(section)}
|
||||||
<Box p={3} pt={0} flexDirection="column">
|
<Box p={3} pt={0} flexDirection="column">
|
||||||
{fields}
|
{fields}
|
||||||
</Box>
|
</Box>
|
||||||
@ -549,6 +580,7 @@ function RecordView({table, launchProcess}: Props): JSX.Element
|
|||||||
{
|
{
|
||||||
setT1SectionElement(sectionFieldElements.get(section.name));
|
setT1SectionElement(sectionFieldElements.get(section.name));
|
||||||
setT1SectionName(section.name);
|
setT1SectionName(section.name);
|
||||||
|
setT1Section(section);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -879,6 +911,7 @@ function RecordView({table, launchProcess}: Props): JSX.Element
|
|||||||
{renderActionsMenu}
|
{renderActionsMenu}
|
||||||
</Box>
|
</Box>
|
||||||
</Box>
|
</Box>
|
||||||
|
{t1Section && getSectionHelp(t1Section)}
|
||||||
{t1SectionElement ? (<Box p={3} pt={0}>{t1SectionElement}</Box>) : null}
|
{t1SectionElement ? (<Box p={3} pt={0}>{t1SectionElement}</Box>) : null}
|
||||||
</Card>
|
</Card>
|
||||||
</Grid>
|
</Grid>
|
||||||
|
@ -100,9 +100,12 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* move the green check / red x down to align with the calendar icon */
|
/* move the green check / red x down to align with the calendar icon */
|
||||||
.MuiFormControl-root
|
.MuiFormControl-root:has(input[type="datetime-local"]),
|
||||||
|
.MuiFormControl-root:has(input[type="date"]),
|
||||||
|
.MuiFormControl-root:has(input[type="time"]),
|
||||||
|
.MuiFormControl-root:has(.MuiInputBase-inputAdornedEnd)
|
||||||
{
|
{
|
||||||
background-position-y: 1.4rem !important;
|
background-position: right 2rem center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.MuiInputAdornment-sizeMedium *
|
.MuiInputAdornment-sizeMedium *
|
||||||
@ -564,3 +567,33 @@ input[type="search"]::-webkit-search-results-decoration { display: none; }
|
|||||||
display: inline;
|
display: inline;
|
||||||
right: .5rem
|
right: .5rem
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* help-content */
|
||||||
|
.helpContent
|
||||||
|
{
|
||||||
|
color: #757575;
|
||||||
|
}
|
||||||
|
|
||||||
|
.helpContent .header
|
||||||
|
{
|
||||||
|
color: #212121;
|
||||||
|
font-weight: 500;
|
||||||
|
display: block;
|
||||||
|
margin-bottom: 0.25rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.MuiTooltip-tooltip .helpContent P + P
|
||||||
|
{
|
||||||
|
margin-top: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.helpContent UL
|
||||||
|
{
|
||||||
|
margin-left: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* for query screen column-header tooltips, move them up a little bit, to be more closely attached to the text. */
|
||||||
|
.dataGridHeaderTooltip
|
||||||
|
{
|
||||||
|
top: -1.25rem;
|
||||||
|
}
|
@ -25,10 +25,13 @@ import {QFieldType} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QField
|
|||||||
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 {QRecord} from "@kingsrook/qqq-frontend-core/lib/model/QRecord";
|
import {QRecord} from "@kingsrook/qqq-frontend-core/lib/model/QRecord";
|
||||||
|
import Tooltip from "@mui/material/Tooltip/Tooltip";
|
||||||
import {GridColDef, GridFilterItem, GridRowsProp} from "@mui/x-data-grid-pro";
|
import {GridColDef, GridFilterItem, GridRowsProp} from "@mui/x-data-grid-pro";
|
||||||
import {GridFilterOperator} from "@mui/x-data-grid/models/gridFilterOperator";
|
import {GridFilterOperator} from "@mui/x-data-grid/models/gridFilterOperator";
|
||||||
|
import {GridColumnHeaderParams} from "@mui/x-data-grid/models/params/gridColumnHeaderParams";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import {Link} from "react-router-dom";
|
import {Link} from "react-router-dom";
|
||||||
|
import HelpContent, {hasHelpContent} from "qqq/components/misc/HelpContent";
|
||||||
import {buildQGridPvsOperators, QGridBlobOperators, QGridBooleanOperators, QGridNumericOperators, QGridStringOperators} from "qqq/pages/records/query/GridFilterOperators";
|
import {buildQGridPvsOperators, QGridBlobOperators, QGridBooleanOperators, QGridNumericOperators, QGridStringOperators} from "qqq/pages/records/query/GridFilterOperators";
|
||||||
import ValueUtils from "qqq/utils/qqq/ValueUtils";
|
import ValueUtils from "qqq/utils/qqq/ValueUtils";
|
||||||
|
|
||||||
@ -310,6 +313,20 @@ export default class DataGridUtils
|
|||||||
(cellValues.value)
|
(cellValues.value)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const helpRoles = ["QUERY_SCREEN", "READ_SCREENS", "ALL_SCREENS"]
|
||||||
|
const showHelp = hasHelpContent(field.helpContents, helpRoles); // todo - maybe - take helpHelpActive from context all the way down to here?
|
||||||
|
if(showHelp)
|
||||||
|
{
|
||||||
|
const formattedHelpContent = <HelpContent helpContents={field.helpContents} roles={helpRoles} heading={headerName} helpContentKey={`table:${tableMetaData.name};field:${fieldName}`} />;
|
||||||
|
column.renderHeader = (params: GridColumnHeaderParams) => (
|
||||||
|
<Tooltip title={formattedHelpContent}>
|
||||||
|
<div className="MuiDataGrid-columnHeaderTitle" style={{lineHeight: "initial"}}>
|
||||||
|
{headerName}
|
||||||
|
</div>
|
||||||
|
</Tooltip>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
return (column);
|
return (column);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -134,7 +134,7 @@ export default class HtmlUtils
|
|||||||
openInWindow.document.write(`<html lang="en">
|
openInWindow.document.write(`<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<style>
|
<style>
|
||||||
* { font-family: "Roboto","Helvetica","Arial",sans-serif; }
|
* { font-family: "SF Pro Display","Roboto","Helvetica","Arial",sans-serif; }
|
||||||
</style>
|
</style>
|
||||||
<title>${filename}</title>
|
<title>${filename}</title>
|
||||||
<script>
|
<script>
|
||||||
|
@ -46,7 +46,7 @@ public class QBaseSeleniumTest
|
|||||||
String headless = System.getenv("QQQ_SELENIUM_HEADLESS");
|
String headless = System.getenv("QQQ_SELENIUM_HEADLESS");
|
||||||
if("true".equals(headless))
|
if("true".equals(headless))
|
||||||
{
|
{
|
||||||
chromeOptions.setHeadless(true);
|
chromeOptions.addArguments("--headless=new");
|
||||||
}
|
}
|
||||||
|
|
||||||
WebDriverManager.chromiumdriver().setup();
|
WebDriverManager.chromiumdriver().setup();
|
||||||
|
@ -1,3 +1,24 @@
|
|||||||
|
/*
|
||||||
|
* QQQ - Low-code Application Framework for Engineers.
|
||||||
|
* Copyright (C) 2021-2023. 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.materialdashboard.lib;
|
package com.kingsrook.qqq.materialdashboard.lib;
|
||||||
|
|
||||||
|
|
||||||
@ -7,7 +28,7 @@ package com.kingsrook.qqq.materialdashboard.lib;
|
|||||||
public interface QQQMaterialDashboardSelectors
|
public interface QQQMaterialDashboardSelectors
|
||||||
{
|
{
|
||||||
String SIDEBAR_ITEM = ".MuiDrawer-paperAnchorDockedLeft li.MuiListItem-root";
|
String SIDEBAR_ITEM = ".MuiDrawer-paperAnchorDockedLeft li.MuiListItem-root";
|
||||||
String BREADCRUMB_HEADER = ".MuiToolbar-root h5";
|
String BREADCRUMB_HEADER = ".MuiToolbar-root h3";
|
||||||
|
|
||||||
String QUERY_GRID_CELL = ".MuiDataGrid-root .MuiDataGrid-cellContent";
|
String QUERY_GRID_CELL = ".MuiDataGrid-root .MuiDataGrid-cellContent";
|
||||||
String QUERY_FILTER_INPUT = ".customFilterPanel input.MuiInput-input";
|
String QUERY_FILTER_INPUT = ".customFilterPanel input.MuiInput-input";
|
||||||
|
@ -30,7 +30,6 @@ import com.kingsrook.qqq.materialdashboard.lib.javalin.QSeleniumJavalin;
|
|||||||
import org.apache.commons.io.FileUtils;
|
import org.apache.commons.io.FileUtils;
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
import org.openqa.selenium.By;
|
import org.openqa.selenium.By;
|
||||||
import static org.assertj.core.api.Assertions.assertThat;
|
|
||||||
import static org.junit.jupiter.api.Assertions.assertEquals;
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||||
|
|
||||||
|
|
||||||
@ -95,8 +94,8 @@ public class DashboardTableWidgetExportTest extends QBaseSeleniumTest
|
|||||||
.click();
|
.click();
|
||||||
|
|
||||||
qSeleniumLib.waitForCondition("Should have downloaded 1 file", () -> getDownloadedFiles().size() == 1);
|
qSeleniumLib.waitForCondition("Should have downloaded 1 file", () -> getDownloadedFiles().size() == 1);
|
||||||
|
qSeleniumLib.waitForCondition("Expected file name", () -> getDownloadedFiles().get(0).getName().matches("Sample Table Widget.*.csv"));
|
||||||
File csvFile = getDownloadedFiles().get(0);
|
File csvFile = getDownloadedFiles().get(0);
|
||||||
assertThat(csvFile.getName()).matches("Sample Table Widget.*.csv");
|
|
||||||
String fileContents = FileUtils.readFileToString(csvFile, StandardCharsets.UTF_8);
|
String fileContents = FileUtils.readFileToString(csvFile, StandardCharsets.UTF_8);
|
||||||
assertEquals("""
|
assertEquals("""
|
||||||
"Id","Name"
|
"Id","Name"
|
||||||
@ -105,7 +104,7 @@ public class DashboardTableWidgetExportTest extends QBaseSeleniumTest
|
|||||||
"3","Bart J."
|
"3","Bart J."
|
||||||
""", fileContents);
|
""", fileContents);
|
||||||
|
|
||||||
// qSeleniumLib.waitForever();
|
qSeleniumLib.waitForever();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -110,7 +110,7 @@ public class SavedFiltersTest extends QBaseSeleniumTest
|
|||||||
//////////////////////
|
//////////////////////
|
||||||
qSeleniumLib.waitForSelectorContaining(".MuiDataGrid-toolbarContainer BUTTON", "Filter").click();
|
qSeleniumLib.waitForSelectorContaining(".MuiDataGrid-toolbarContainer BUTTON", "Filter").click();
|
||||||
addQueryFilterInput(qSeleniumLib, 1, "First Name", "contains", "Jam", "Or");
|
addQueryFilterInput(qSeleniumLib, 1, "First Name", "contains", "Jam", "Or");
|
||||||
qSeleniumLib.waitForSelectorContaining("H5", "Person").click();
|
qSeleniumLib.waitForSelectorContaining("H3", "Person").click();
|
||||||
qSeleniumLib.waitForSelectorContaining("DIV", "Current Filter: Some People")
|
qSeleniumLib.waitForSelectorContaining("DIV", "Current Filter: Some People")
|
||||||
.findElement(By.cssSelector("CIRCLE"));
|
.findElement(By.cssSelector("CIRCLE"));
|
||||||
qSeleniumLib.waitForSelectorContaining(".MuiBadge-badge", "2");
|
qSeleniumLib.waitForSelectorContaining(".MuiBadge-badge", "2");
|
||||||
|
Reference in New Issue
Block a user