mirror of
https://github.com/Kingsrook/qqq-frontend-material-dashboard.git
synced 2025-07-17 12:50:43 +00:00
Merged feature/CE-1772-generate-labels-poc into dev
This commit is contained in:
20602
package-lock.json
generated
20602
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@ -40,16 +40,17 @@ import Snackbar from "@mui/material/Snackbar";
|
|||||||
import TextField from "@mui/material/TextField";
|
import TextField from "@mui/material/TextField";
|
||||||
import Tooltip from "@mui/material/Tooltip";
|
import Tooltip from "@mui/material/Tooltip";
|
||||||
import FormData from "form-data";
|
import FormData from "form-data";
|
||||||
import React, {useEffect, useReducer, useRef, useState} from "react";
|
|
||||||
import AceEditor from "react-ace";
|
|
||||||
import {QCancelButton, QSaveButton} from "qqq/components/buttons/DefaultButtons";
|
import {QCancelButton, QSaveButton} from "qqq/components/buttons/DefaultButtons";
|
||||||
import DynamicSelect from "qqq/components/forms/DynamicSelect";
|
import DynamicSelect from "qqq/components/forms/DynamicSelect";
|
||||||
import ScriptDocsForm from "qqq/components/scripts/ScriptDocsForm";
|
import ScriptDocsForm from "qqq/components/scripts/ScriptDocsForm";
|
||||||
import ScriptTestForm from "qqq/components/scripts/ScriptTestForm";
|
import ScriptTestForm from "qqq/components/scripts/ScriptTestForm";
|
||||||
import Client from "qqq/utils/qqq/Client";
|
import Client from "qqq/utils/qqq/Client";
|
||||||
|
|
||||||
|
import "ace-builds/src-noconflict/ace";
|
||||||
import "ace-builds/src-noconflict/mode-javascript";
|
import "ace-builds/src-noconflict/mode-javascript";
|
||||||
import "ace-builds/src-noconflict/theme-github";
|
import "ace-builds/src-noconflict/theme-github";
|
||||||
|
import React, {useEffect, useReducer, useRef, useState} from "react";
|
||||||
|
import AceEditor from "react-ace";
|
||||||
import "ace-builds/src-noconflict/ext-language_tools";
|
import "ace-builds/src-noconflict/ext-language_tools";
|
||||||
|
|
||||||
export interface ScriptEditorProps
|
export interface ScriptEditorProps
|
||||||
@ -69,15 +70,15 @@ const qController = Client.getInstance();
|
|||||||
|
|
||||||
function buildInitialFileContentsMap(scriptRevisionRecord: QRecord, scriptTypeFileSchemaList: QRecord[]): { [name: string]: string }
|
function buildInitialFileContentsMap(scriptRevisionRecord: QRecord, scriptTypeFileSchemaList: QRecord[]): { [name: string]: string }
|
||||||
{
|
{
|
||||||
const rs: {[name: string]: string} = {};
|
const rs: { [name: string]: string } = {};
|
||||||
|
|
||||||
if(!scriptTypeFileSchemaList)
|
if (!scriptTypeFileSchemaList)
|
||||||
{
|
{
|
||||||
console.log("Missing scriptTypeFileSchemaList");
|
console.log("Missing scriptTypeFileSchemaList");
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
let files = scriptRevisionRecord?.associatedRecords?.get("files")
|
let files = scriptRevisionRecord?.associatedRecords?.get("files");
|
||||||
|
|
||||||
for (let i = 0; i < scriptTypeFileSchemaList.length; i++)
|
for (let i = 0; i < scriptTypeFileSchemaList.length; i++)
|
||||||
{
|
{
|
||||||
@ -88,7 +89,7 @@ function buildInitialFileContentsMap(scriptRevisionRecord: QRecord, scriptTypeFi
|
|||||||
for (let j = 0; j < files?.length; j++)
|
for (let j = 0; j < files?.length; j++)
|
||||||
{
|
{
|
||||||
let file = files[j];
|
let file = files[j];
|
||||||
if(file.values.get("fileName") == name)
|
if (file.values.get("fileName") == name)
|
||||||
{
|
{
|
||||||
contents = file.values.get("contents");
|
contents = file.values.get("contents");
|
||||||
}
|
}
|
||||||
@ -103,9 +104,9 @@ function buildInitialFileContentsMap(scriptRevisionRecord: QRecord, scriptTypeFi
|
|||||||
|
|
||||||
function buildFileTypeMap(scriptTypeFileSchemaList: QRecord[]): { [name: string]: string }
|
function buildFileTypeMap(scriptTypeFileSchemaList: QRecord[]): { [name: string]: string }
|
||||||
{
|
{
|
||||||
const rs: {[name: string]: string} = {};
|
const rs: { [name: string]: string } = {};
|
||||||
|
|
||||||
if(!scriptTypeFileSchemaList)
|
if (!scriptTypeFileSchemaList)
|
||||||
{
|
{
|
||||||
console.log("Missing scriptTypeFileSchemaList");
|
console.log("Missing scriptTypeFileSchemaList");
|
||||||
}
|
}
|
||||||
@ -125,21 +126,21 @@ function ScriptEditor({title, scriptId, scriptRevisionRecord, closeCallback, tab
|
|||||||
{
|
{
|
||||||
const [closing, setClosing] = useState(false);
|
const [closing, setClosing] = useState(false);
|
||||||
|
|
||||||
const [apiName, setApiName] = useState(scriptRevisionRecord ? scriptRevisionRecord.values.get("apiName") : null)
|
const [apiName, setApiName] = useState(scriptRevisionRecord ? scriptRevisionRecord.values.get("apiName") : null);
|
||||||
const [apiNameLabel, setApiNameLabel] = useState(scriptRevisionRecord ? scriptRevisionRecord.displayValues.get("apiName") : null)
|
const [apiNameLabel, setApiNameLabel] = useState(scriptRevisionRecord ? scriptRevisionRecord.displayValues.get("apiName") : null);
|
||||||
const [apiVersion, setApiVersion] = useState(scriptRevisionRecord ? scriptRevisionRecord.values.get("apiVersion") : null)
|
const [apiVersion, setApiVersion] = useState(scriptRevisionRecord ? scriptRevisionRecord.values.get("apiVersion") : null);
|
||||||
const [apiVersionLabel, setApiVersionLabel] = useState(scriptRevisionRecord ? scriptRevisionRecord.displayValues.get("apiVersion") : null)
|
const [apiVersionLabel, setApiVersionLabel] = useState(scriptRevisionRecord ? scriptRevisionRecord.displayValues.get("apiVersion") : null);
|
||||||
|
|
||||||
const fileNamesFromSchema = scriptTypeFileSchemaList.map((schemaRecord) => schemaRecord.values.get("name"))
|
const fileNamesFromSchema = scriptTypeFileSchemaList.map((schemaRecord) => schemaRecord.values.get("name"));
|
||||||
const [availableFileNames, setAvailableFileNames] = useState(fileNamesFromSchema);
|
const [availableFileNames, setAvailableFileNames] = useState(fileNamesFromSchema);
|
||||||
const [openEditorFileNames, setOpenEditorFileNames] = useState([fileNamesFromSchema[0]])
|
const [openEditorFileNames, setOpenEditorFileNames] = useState([fileNamesFromSchema[0]]);
|
||||||
const [fileContents, setFileContents] = useState(buildInitialFileContentsMap(scriptRevisionRecord, scriptTypeFileSchemaList))
|
const [fileContents, setFileContents] = useState(buildInitialFileContentsMap(scriptRevisionRecord, scriptTypeFileSchemaList));
|
||||||
const [fileTypes, setFileTypes] = useState(buildFileTypeMap(scriptTypeFileSchemaList))
|
const [fileTypes, setFileTypes] = useState(buildFileTypeMap(scriptTypeFileSchemaList));
|
||||||
console.log(`file types: ${JSON.stringify(fileTypes)}`);
|
console.log(`file types: ${JSON.stringify(fileTypes)}`);
|
||||||
|
|
||||||
const [commitMessage, setCommitMessage] = useState("")
|
const [commitMessage, setCommitMessage] = useState("");
|
||||||
const [openTool, setOpenTool] = useState(null);
|
const [openTool, setOpenTool] = useState(null);
|
||||||
const [errorAlert, setErrorAlert] = useState("")
|
const [errorAlert, setErrorAlert] = useState("");
|
||||||
const [promptForCommitMessageOpen, setPromptForCommitMessageOpen] = useState(false);
|
const [promptForCommitMessageOpen, setPromptForCommitMessageOpen] = useState(false);
|
||||||
const [, forceUpdate] = useReducer((x) => x + 1, 0);
|
const [, forceUpdate] = useReducer((x) => x + 1, 0);
|
||||||
const ref = useRef();
|
const ref = useRef();
|
||||||
@ -241,19 +242,19 @@ function ScriptEditor({title, scriptId, scriptRevisionRecord, closeCallback, tab
|
|||||||
// need this to make Ace recognize new height.
|
// need this to make Ace recognize new height.
|
||||||
setTimeout(() =>
|
setTimeout(() =>
|
||||||
{
|
{
|
||||||
window.dispatchEvent(new Event("resize"))
|
window.dispatchEvent(new Event("resize"));
|
||||||
}, 100);
|
}, 100);
|
||||||
};
|
};
|
||||||
|
|
||||||
const saveClicked = (overrideCommitMessage?: string) =>
|
const saveClicked = (overrideCommitMessage?: string) =>
|
||||||
{
|
{
|
||||||
if(!apiName || !apiVersion)
|
if (!apiName || !apiVersion)
|
||||||
{
|
{
|
||||||
setErrorAlert("You must select a value for both API Name and API Version.")
|
setErrorAlert("You must select a value for both API Name and API Version.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!commitMessage && !overrideCommitMessage)
|
if (!commitMessage && !overrideCommitMessage)
|
||||||
{
|
{
|
||||||
setPromptForCommitMessageOpen(true);
|
setPromptForCommitMessageOpen(true);
|
||||||
return;
|
return;
|
||||||
@ -267,18 +268,18 @@ function ScriptEditor({title, scriptId, scriptRevisionRecord, closeCallback, tab
|
|||||||
formData.append("scriptId", scriptId);
|
formData.append("scriptId", scriptId);
|
||||||
formData.append("commitMessage", overrideCommitMessage ?? commitMessage);
|
formData.append("commitMessage", overrideCommitMessage ?? commitMessage);
|
||||||
|
|
||||||
if(apiName)
|
if (apiName)
|
||||||
{
|
{
|
||||||
formData.append("apiName", apiName);
|
formData.append("apiName", apiName);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(apiVersion)
|
if (apiVersion)
|
||||||
{
|
{
|
||||||
formData.append("apiVersion", apiVersion);
|
formData.append("apiVersion", apiVersion);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
const fileNamesFromSchema = scriptTypeFileSchemaList.map((schemaRecord) => schemaRecord.values.get("name"))
|
const fileNamesFromSchema = scriptTypeFileSchemaList.map((schemaRecord) => schemaRecord.values.get("name"));
|
||||||
formData.append("fileNames", fileNamesFromSchema.join(","));
|
formData.append("fileNames", fileNamesFromSchema.join(","));
|
||||||
|
|
||||||
for (let fileName in fileContents)
|
for (let fileName in fileContents)
|
||||||
@ -299,58 +300,58 @@ function ScriptEditor({title, scriptId, scriptRevisionRecord, closeCallback, tab
|
|||||||
|
|
||||||
if (processResult instanceof QJobError)
|
if (processResult instanceof QJobError)
|
||||||
{
|
{
|
||||||
const jobError = processResult as QJobError
|
const jobError = processResult as QJobError;
|
||||||
setErrorAlert(jobError.userFacingError ?? jobError.error)
|
setErrorAlert(jobError.userFacingError ?? jobError.error);
|
||||||
setClosing(false);
|
setClosing(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
closeCallback(null, "saved", "Saved New Script Version");
|
closeCallback(null, "saved", "Saved New Script Version");
|
||||||
}
|
}
|
||||||
catch(e)
|
catch (e)
|
||||||
{
|
{
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
setErrorAlert(e.message ?? "Unexpected error saving script")
|
setErrorAlert(e.message ?? "Unexpected error saving script");
|
||||||
setClosing(false);
|
setClosing(false);
|
||||||
}
|
}
|
||||||
})();
|
})();
|
||||||
}
|
};
|
||||||
|
|
||||||
const cancelClicked = () =>
|
const cancelClicked = () =>
|
||||||
{
|
{
|
||||||
setClosing(true);
|
setClosing(true);
|
||||||
closeCallback(null, "cancelled");
|
closeCallback(null, "cancelled");
|
||||||
}
|
};
|
||||||
|
|
||||||
const updateCode = (value: string, event: any, index: number) =>
|
const updateCode = (value: string, event: any, index: number) =>
|
||||||
{
|
{
|
||||||
fileContents[openEditorFileNames[index]] = value;
|
fileContents[openEditorFileNames[index]] = value;
|
||||||
forceUpdate();
|
forceUpdate();
|
||||||
}
|
};
|
||||||
|
|
||||||
const updateCommitMessage = (event: React.ChangeEvent<HTMLInputElement>) =>
|
const updateCommitMessage = (event: React.ChangeEvent<HTMLInputElement>) =>
|
||||||
{
|
{
|
||||||
setCommitMessage(event.target.value);
|
setCommitMessage(event.target.value);
|
||||||
}
|
};
|
||||||
|
|
||||||
const closePromptForCommitMessage = (wasSaveClicked: boolean, message?: string) =>
|
const closePromptForCommitMessage = (wasSaveClicked: boolean, message?: string) =>
|
||||||
{
|
{
|
||||||
setPromptForCommitMessageOpen(false);
|
setPromptForCommitMessageOpen(false);
|
||||||
|
|
||||||
if(wasSaveClicked)
|
if (wasSaveClicked)
|
||||||
{
|
{
|
||||||
setCommitMessage(message)
|
setCommitMessage(message);
|
||||||
saveClicked(message);
|
saveClicked(message);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
setClosing(false);
|
setClosing(false);
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
|
|
||||||
const changeApiName = (apiNamePossibleValue?: QPossibleValue) =>
|
const changeApiName = (apiNamePossibleValue?: QPossibleValue) =>
|
||||||
{
|
{
|
||||||
if(apiNamePossibleValue)
|
if (apiNamePossibleValue)
|
||||||
{
|
{
|
||||||
setApiName(apiNamePossibleValue.id);
|
setApiName(apiNamePossibleValue.id);
|
||||||
}
|
}
|
||||||
@ -358,11 +359,11 @@ function ScriptEditor({title, scriptId, scriptRevisionRecord, closeCallback, tab
|
|||||||
{
|
{
|
||||||
setApiName(null);
|
setApiName(null);
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
|
|
||||||
const changeApiVersion = (apiVersionPossibleValue?: QPossibleValue) =>
|
const changeApiVersion = (apiVersionPossibleValue?: QPossibleValue) =>
|
||||||
{
|
{
|
||||||
if(apiVersionPossibleValue)
|
if (apiVersionPossibleValue)
|
||||||
{
|
{
|
||||||
setApiVersion(apiVersionPossibleValue.id);
|
setApiVersion(apiVersionPossibleValue.id);
|
||||||
}
|
}
|
||||||
@ -370,33 +371,33 @@ function ScriptEditor({title, scriptId, scriptRevisionRecord, closeCallback, tab
|
|||||||
{
|
{
|
||||||
setApiVersion(null);
|
setApiVersion(null);
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
|
|
||||||
const handleSelectingFile = (event: SelectChangeEvent, index: number) =>
|
const handleSelectingFile = (event: SelectChangeEvent, index: number) =>
|
||||||
{
|
{
|
||||||
openEditorFileNames[index] = event.target.value
|
openEditorFileNames[index] = event.target.value;
|
||||||
setOpenEditorFileNames(openEditorFileNames);
|
setOpenEditorFileNames(openEditorFileNames);
|
||||||
forceUpdate();
|
forceUpdate();
|
||||||
}
|
};
|
||||||
|
|
||||||
const splitEditorClicked = () =>
|
const splitEditorClicked = () =>
|
||||||
{
|
{
|
||||||
openEditorFileNames.push(availableFileNames[0])
|
openEditorFileNames.push(availableFileNames[0]);
|
||||||
setOpenEditorFileNames(openEditorFileNames);
|
setOpenEditorFileNames(openEditorFileNames);
|
||||||
forceUpdate();
|
forceUpdate();
|
||||||
}
|
};
|
||||||
|
|
||||||
const closeEditorClicked = (index: number) =>
|
const closeEditorClicked = (index: number) =>
|
||||||
{
|
{
|
||||||
openEditorFileNames.splice(index, 1)
|
openEditorFileNames.splice(index, 1);
|
||||||
setOpenEditorFileNames(openEditorFileNames);
|
setOpenEditorFileNames(openEditorFileNames);
|
||||||
forceUpdate();
|
forceUpdate();
|
||||||
}
|
};
|
||||||
|
|
||||||
const computeEditorWidth = (): string =>
|
const computeEditorWidth = (): string =>
|
||||||
{
|
{
|
||||||
return (100 / openEditorFileNames.length) + "%"
|
return (100 / openEditorFileNames.length) + "%";
|
||||||
}
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box className="scriptEditor" sx={{position: "absolute", overflowY: "auto", height: "100%", width: "100%"}} p={6}>
|
<Box className="scriptEditor" sx={{position: "absolute", overflowY: "auto", height: "100%", width: "100%"}} p={6}>
|
||||||
@ -408,7 +409,7 @@ function ScriptEditor({title, scriptId, scriptRevisionRecord, closeCallback, tab
|
|||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
setErrorAlert("")
|
setErrorAlert("");
|
||||||
}} anchorOrigin={{vertical: "top", horizontal: "center"}}>
|
}} anchorOrigin={{vertical: "top", horizontal: "center"}}>
|
||||||
<Alert color="error" onClose={() => setErrorAlert("")}>
|
<Alert color="error" onClose={() => setErrorAlert("")}>
|
||||||
{errorAlert}
|
{errorAlert}
|
||||||
@ -464,19 +465,19 @@ function ScriptEditor({title, scriptId, scriptRevisionRecord, closeCallback, tab
|
|||||||
<Box>
|
<Box>
|
||||||
{
|
{
|
||||||
openEditorFileNames.length > 1 &&
|
openEditorFileNames.length > 1 &&
|
||||||
<Tooltip title="Close this editor split" enterDelay={500}>
|
<Tooltip title="Close this editor split" enterDelay={500}>
|
||||||
<IconButton size="small" onClick={() => closeEditorClicked(index)}>
|
<IconButton size="small" onClick={() => closeEditorClicked(index)}>
|
||||||
<Icon>close</Icon>
|
<Icon>close</Icon>
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
}
|
}
|
||||||
{
|
{
|
||||||
index == openEditorFileNames.length - 1 &&
|
index == openEditorFileNames.length - 1 &&
|
||||||
<Tooltip title="Open a new editor split" enterDelay={500}>
|
<Tooltip title="Open a new editor split" enterDelay={500}>
|
||||||
<IconButton size="small" onClick={splitEditorClicked}>
|
<IconButton size="small" onClick={splitEditorClicked}>
|
||||||
<Icon>vertical_split</Icon>
|
<Icon>vertical_split</Icon>
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
}
|
}
|
||||||
</Box>
|
</Box>
|
||||||
</Box>
|
</Box>
|
||||||
@ -526,29 +527,29 @@ function ScriptEditor({title, scriptId, scriptRevisionRecord, closeCallback, tab
|
|||||||
</Grid>
|
</Grid>
|
||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
<CommitMessagePrompt isOpen={promptForCommitMessageOpen} closeHandler={closePromptForCommitMessage}/>
|
<CommitMessagePrompt isOpen={promptForCommitMessageOpen} closeHandler={closePromptForCommitMessage} />
|
||||||
</Card>
|
</Card>
|
||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
function CommitMessagePrompt(props: {isOpen: boolean, closeHandler: (wasSaveClicked: boolean, message?: string) => void})
|
function CommitMessagePrompt(props: { isOpen: boolean, closeHandler: (wasSaveClicked: boolean, message?: string) => void })
|
||||||
{
|
{
|
||||||
const [commitMessage, setCommitMessage] = useState("No commit message given")
|
const [commitMessage, setCommitMessage] = useState("No commit message given");
|
||||||
|
|
||||||
const updateCommitMessage = (event: React.ChangeEvent<HTMLInputElement>) =>
|
const updateCommitMessage = (event: React.ChangeEvent<HTMLInputElement>) =>
|
||||||
{
|
{
|
||||||
setCommitMessage(event.target.value);
|
setCommitMessage(event.target.value);
|
||||||
}
|
};
|
||||||
|
|
||||||
const keyPressHandler = (e: React.KeyboardEvent<HTMLDivElement>) =>
|
const keyPressHandler = (e: React.KeyboardEvent<HTMLDivElement>) =>
|
||||||
{
|
{
|
||||||
if(e.key === "Enter")
|
if (e.key === "Enter")
|
||||||
{
|
{
|
||||||
props.closeHandler(true, commitMessage);
|
props.closeHandler(true, commitMessage);
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog
|
<Dialog
|
||||||
@ -579,10 +580,10 @@ function CommitMessagePrompt(props: {isOpen: boolean, closeHandler: (wasSaveClic
|
|||||||
</DialogContent>
|
</DialogContent>
|
||||||
<DialogActions>
|
<DialogActions>
|
||||||
<QCancelButton onClickHandler={() => props.closeHandler(false)} disabled={false} />
|
<QCancelButton onClickHandler={() => props.closeHandler(false)} disabled={false} />
|
||||||
<QSaveButton label="Save" onClickHandler={() => props.closeHandler(true, commitMessage)} disabled={false}/>
|
<QSaveButton label="Save" onClickHandler={() => props.closeHandler(true, commitMessage)} disabled={false} />
|
||||||
</DialogActions>
|
</DialogActions>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
)
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default ScriptEditor;
|
export default ScriptEditor;
|
||||||
|
@ -50,6 +50,7 @@ import DeveloperModeUtils from "qqq/utils/DeveloperModeUtils";
|
|||||||
import Client from "qqq/utils/qqq/Client";
|
import Client from "qqq/utils/qqq/Client";
|
||||||
import ValueUtils from "qqq/utils/qqq/ValueUtils";
|
import ValueUtils from "qqq/utils/qqq/ValueUtils";
|
||||||
|
|
||||||
|
import "ace-builds/src-noconflict/ace";
|
||||||
import "ace-builds/src-noconflict/mode-java";
|
import "ace-builds/src-noconflict/mode-java";
|
||||||
import "ace-builds/src-noconflict/mode-javascript";
|
import "ace-builds/src-noconflict/mode-javascript";
|
||||||
import "ace-builds/src-noconflict/mode-json";
|
import "ace-builds/src-noconflict/mode-json";
|
||||||
|
@ -34,6 +34,7 @@ import BaseLayout from "qqq/layouts/BaseLayout";
|
|||||||
import Client from "qqq/utils/qqq/Client";
|
import Client from "qqq/utils/qqq/Client";
|
||||||
import ValueUtils from "qqq/utils/qqq/ValueUtils";
|
import ValueUtils from "qqq/utils/qqq/ValueUtils";
|
||||||
|
|
||||||
|
import "ace-builds/src-noconflict/ace";
|
||||||
import "ace-builds/src-noconflict/mode-java";
|
import "ace-builds/src-noconflict/mode-java";
|
||||||
import "ace-builds/src-noconflict/mode-javascript";
|
import "ace-builds/src-noconflict/mode-javascript";
|
||||||
import "ace-builds/src-noconflict/mode-json";
|
import "ace-builds/src-noconflict/mode-json";
|
||||||
|
@ -113,7 +113,7 @@ export default class DataGridUtils
|
|||||||
{
|
{
|
||||||
console.log(`row-click mouse-up happened ${diff} x or y pixels away from the mouse-down - so not considering it a click.`);
|
console.log(`row-click mouse-up happened ${diff} x or y pixels away from the mouse-down - so not considering it a click.`);
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
**
|
**
|
||||||
@ -133,13 +133,13 @@ export default class DataGridUtils
|
|||||||
row[field.name] = ValueUtils.getDisplayValue(field, record, "query");
|
row[field.name] = ValueUtils.getDisplayValue(field, record, "query");
|
||||||
});
|
});
|
||||||
|
|
||||||
if(tableMetaData.exposedJoins)
|
if (tableMetaData.exposedJoins)
|
||||||
{
|
{
|
||||||
for (let i = 0; i < tableMetaData.exposedJoins.length; i++)
|
for (let i = 0; i < tableMetaData.exposedJoins.length; i++)
|
||||||
{
|
{
|
||||||
const join = tableMetaData.exposedJoins[i];
|
const join = tableMetaData.exposedJoins[i];
|
||||||
|
|
||||||
if(join?.joinTable?.fields?.values())
|
if (join?.joinTable?.fields?.values())
|
||||||
{
|
{
|
||||||
const fields = [...join.joinTable.fields.values()];
|
const fields = [...join.joinTable.fields.values()];
|
||||||
fields.forEach((field) =>
|
fields.forEach((field) =>
|
||||||
@ -151,15 +151,15 @@ export default class DataGridUtils
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!row["id"])
|
if (!row["id"])
|
||||||
{
|
{
|
||||||
row["id"] = record.values.get(tableMetaData.primaryKeyField) ?? row[tableMetaData.primaryKeyField];
|
row["id"] = record.values.get(tableMetaData.primaryKeyField) ?? row[tableMetaData.primaryKeyField];
|
||||||
if(row["id"] === null || row["id"] === undefined)
|
if (row["id"] === null || row["id"] === undefined)
|
||||||
{
|
{
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// DataGrid gets very upset about a null or undefined here, so, try to make it happier //
|
// DataGrid gets very upset about a null or undefined here, so, try to make it happier //
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////
|
||||||
if(!allowEmptyId)
|
if (!allowEmptyId)
|
||||||
{
|
{
|
||||||
row["id"] = "--";
|
row["id"] = "--";
|
||||||
}
|
}
|
||||||
@ -170,7 +170,7 @@ export default class DataGridUtils
|
|||||||
});
|
});
|
||||||
|
|
||||||
return (rows);
|
return (rows);
|
||||||
}
|
};
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
**
|
**
|
||||||
@ -180,24 +180,24 @@ export default class DataGridUtils
|
|||||||
const columns = [] as GridColDef[];
|
const columns = [] as GridColDef[];
|
||||||
this.addColumnsForTable(tableMetaData, linkBase, columns, columnSort, null, null);
|
this.addColumnsForTable(tableMetaData, linkBase, columns, columnSort, null, null);
|
||||||
|
|
||||||
if(metaData)
|
if (metaData)
|
||||||
{
|
{
|
||||||
if(tableMetaData.exposedJoins)
|
if (tableMetaData.exposedJoins)
|
||||||
{
|
{
|
||||||
for (let i = 0; i < tableMetaData.exposedJoins.length; i++)
|
for (let i = 0; i < tableMetaData.exposedJoins.length; i++)
|
||||||
{
|
{
|
||||||
const join = tableMetaData.exposedJoins[i];
|
const join = tableMetaData.exposedJoins[i];
|
||||||
let joinTableName = join.joinTable.name;
|
let joinTableName = join.joinTable.name;
|
||||||
if(metaData.tables.has(joinTableName) && metaData.tables.get(joinTableName).readPermission)
|
if (metaData.tables.has(joinTableName) && metaData.tables.get(joinTableName).readPermission)
|
||||||
{
|
{
|
||||||
let joinLinkBase = null;
|
let joinLinkBase = null;
|
||||||
joinLinkBase = metaData.getTablePath(join.joinTable);
|
joinLinkBase = metaData.getTablePath(join.joinTable);
|
||||||
if(joinLinkBase)
|
if (joinLinkBase)
|
||||||
{
|
{
|
||||||
joinLinkBase += joinLinkBase.endsWith("/") ? "" : "/";
|
joinLinkBase += joinLinkBase.endsWith("/") ? "" : "/";
|
||||||
}
|
}
|
||||||
|
|
||||||
if(join?.joinTable?.fields?.values())
|
if (join?.joinTable?.fields?.values())
|
||||||
{
|
{
|
||||||
this.addColumnsForTable(join.joinTable, joinLinkBase, columns, columnSort, joinTableName + ".", join.label + ": ");
|
this.addColumnsForTable(join.joinTable, joinLinkBase, columns, columnSort, joinTableName + ".", join.label + ": ");
|
||||||
}
|
}
|
||||||
@ -220,7 +220,7 @@ export default class DataGridUtils
|
|||||||
////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////
|
||||||
// this sorted by sections - e.g., manual sorting by the meta-data... //
|
// this sorted by sections - e.g., manual sorting by the meta-data... //
|
||||||
////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////
|
||||||
if(columnSort === "bySection")
|
if (columnSort === "bySection")
|
||||||
{
|
{
|
||||||
for (let i = 0; i < tableMetaData.sections.length; i++)
|
for (let i = 0; i < tableMetaData.sections.length; i++)
|
||||||
{
|
{
|
||||||
@ -241,19 +241,23 @@ export default class DataGridUtils
|
|||||||
///////////////////////////
|
///////////////////////////
|
||||||
// sort by labels... mmm //
|
// sort by labels... mmm //
|
||||||
///////////////////////////
|
///////////////////////////
|
||||||
sortedKeys.push(...tableMetaData.fields.keys())
|
sortedKeys.push(...tableMetaData.fields.keys());
|
||||||
sortedKeys.sort((a: string, b: string): number =>
|
sortedKeys.sort((a: string, b: string): number =>
|
||||||
{
|
{
|
||||||
return (tableMetaData.fields.get(a).label.localeCompare(tableMetaData.fields.get(b).label))
|
return (tableMetaData.fields.get(a).label.localeCompare(tableMetaData.fields.get(b).label));
|
||||||
})
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
sortedKeys.forEach((key) =>
|
sortedKeys.forEach((key) =>
|
||||||
{
|
{
|
||||||
const field = tableMetaData.fields.get(key);
|
const field = tableMetaData.fields.get(key);
|
||||||
if(field.isHeavy)
|
if (!field)
|
||||||
{
|
{
|
||||||
if(field.type == QFieldType.BLOB)
|
return;
|
||||||
|
}
|
||||||
|
if (field.isHeavy)
|
||||||
|
{
|
||||||
|
if (field.type == QFieldType.BLOB)
|
||||||
{
|
{
|
||||||
////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////
|
||||||
// assume we DO want heavy blobs - as download links. //
|
// assume we DO want heavy blobs - as download links. //
|
||||||
@ -270,7 +274,7 @@ export default class DataGridUtils
|
|||||||
|
|
||||||
const column = this.makeColumnFromField(field, tableMetaData, namePrefix, labelPrefix);
|
const column = this.makeColumnFromField(field, tableMetaData, namePrefix, labelPrefix);
|
||||||
|
|
||||||
if(key === tableMetaData.primaryKeyField && linkBase && namePrefix == null)
|
if (key === tableMetaData.primaryKeyField && linkBase && namePrefix == null)
|
||||||
{
|
{
|
||||||
columns.splice(0, 0, column);
|
columns.splice(0, 0, column);
|
||||||
}
|
}
|
||||||
@ -346,9 +350,9 @@ export default class DataGridUtils
|
|||||||
(cellValues.value)
|
(cellValues.value)
|
||||||
);
|
);
|
||||||
|
|
||||||
const helpRoles = ["QUERY_SCREEN", "READ_SCREENS", "ALL_SCREENS"]
|
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?
|
const showHelp = hasHelpContent(field.helpContents, helpRoles); // todo - maybe - take helpHelpActive from context all the way down to here?
|
||||||
if(showHelp)
|
if (showHelp)
|
||||||
{
|
{
|
||||||
const formattedHelpContent = <HelpContent helpContents={field.helpContents} roles={helpRoles} heading={headerName} helpContentKey={`table:${tableMetaData.name};field:${fieldName}`} />;
|
const formattedHelpContent = <HelpContent helpContents={field.helpContents} roles={helpRoles} heading={headerName} helpContentKey={`table:${tableMetaData.name};field:${fieldName}`} />;
|
||||||
column.renderHeader = (params: GridColumnHeaderParams) => (
|
column.renderHeader = (params: GridColumnHeaderParams) => (
|
||||||
@ -361,7 +365,7 @@ export default class DataGridUtils
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (column);
|
return (column);
|
||||||
}
|
};
|
||||||
|
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
@ -390,7 +394,7 @@ export default class DataGridUtils
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if(field.possibleValueSourceName)
|
if (field.possibleValueSourceName)
|
||||||
{
|
{
|
||||||
return (200);
|
return (200);
|
||||||
}
|
}
|
||||||
@ -415,6 +419,6 @@ export default class DataGridUtils
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (200);
|
return (200);
|
||||||
}
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -19,7 +19,8 @@
|
|||||||
* 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 Client from "qqq/utils/qqq/Client";
|
import {QFieldMetaData} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QFieldMetaData";
|
||||||
|
import {QFieldType} from "@kingsrook/qqq-frontend-core/lib/model/metaData/QFieldType";
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
** Utility functions for basic html/webpage/browser things.
|
** Utility functions for basic html/webpage/browser things.
|
||||||
@ -68,10 +69,15 @@ export default class HtmlUtils
|
|||||||
** it was originally built like this when we had to submit full access token to backend...
|
** it was originally built like this when we had to submit full access token to backend...
|
||||||
**
|
**
|
||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
static downloadUrlViaIFrame = (url: string, filename: string) =>
|
static downloadUrlViaIFrame = (field: QFieldMetaData, url: string, filename: string) =>
|
||||||
{
|
{
|
||||||
if(url.startsWith("data:"))
|
if (url.startsWith("data:") || url.startsWith("http"))
|
||||||
{
|
{
|
||||||
|
if (url.startsWith("http"))
|
||||||
|
{
|
||||||
|
url += encodeURIComponent(`?response-content-disposition=attachment; ${filename}`);
|
||||||
|
}
|
||||||
|
|
||||||
const link = document.createElement("a");
|
const link = document.createElement("a");
|
||||||
link.download = filename;
|
link.download = filename;
|
||||||
link.href = url;
|
link.href = url;
|
||||||
@ -93,8 +99,14 @@ export default class HtmlUtils
|
|||||||
// todo - onload event handler to let us know when done?
|
// todo - onload event handler to let us know when done?
|
||||||
document.body.appendChild(iframe);
|
document.body.appendChild(iframe);
|
||||||
|
|
||||||
|
var method = "get";
|
||||||
|
if (QFieldType.BLOB == field.type)
|
||||||
|
{
|
||||||
|
method = "post";
|
||||||
|
}
|
||||||
|
|
||||||
const form = document.createElement("form");
|
const form = document.createElement("form");
|
||||||
form.setAttribute("method", "post");
|
form.setAttribute("method", method);
|
||||||
form.setAttribute("action", url);
|
form.setAttribute("action", url);
|
||||||
form.setAttribute("target", "downloadIframe");
|
form.setAttribute("target", "downloadIframe");
|
||||||
iframe.appendChild(form);
|
iframe.appendChild(form);
|
||||||
@ -117,7 +129,7 @@ export default class HtmlUtils
|
|||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
static openInNewWindow = (url: string, filename: string) =>
|
static openInNewWindow = (url: string, filename: string) =>
|
||||||
{
|
{
|
||||||
if(url.startsWith("data:"))
|
if (url.startsWith("data:"))
|
||||||
{
|
{
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
/////////////////////////////////////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
@ -153,4 +165,4 @@ export default class HtmlUtils
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -28,18 +28,17 @@ import "datejs"; // https://github.com/datejs/Datejs
|
|||||||
import {Chip, ClickAwayListener, Icon} from "@mui/material";
|
import {Chip, ClickAwayListener, Icon} from "@mui/material";
|
||||||
import Box from "@mui/material/Box";
|
import Box from "@mui/material/Box";
|
||||||
import Button from "@mui/material/Button";
|
import Button from "@mui/material/Button";
|
||||||
import IconButton from "@mui/material/IconButton";
|
|
||||||
import Tooltip from "@mui/material/Tooltip";
|
import Tooltip from "@mui/material/Tooltip";
|
||||||
import {makeStyles} from "@mui/styles";
|
|
||||||
import parse from "html-react-parser";
|
import parse from "html-react-parser";
|
||||||
import React, {Fragment, useReducer, useState} from "react";
|
|
||||||
import AceEditor from "react-ace";
|
|
||||||
import {Link} from "react-router-dom";
|
|
||||||
import HtmlUtils from "qqq/utils/HtmlUtils";
|
import HtmlUtils from "qqq/utils/HtmlUtils";
|
||||||
import Client from "qqq/utils/qqq/Client";
|
import Client from "qqq/utils/qqq/Client";
|
||||||
|
|
||||||
|
import "ace-builds/src-noconflict/ace";
|
||||||
import "ace-builds/src-noconflict/mode-sql";
|
import "ace-builds/src-noconflict/mode-sql";
|
||||||
|
import React, {Fragment, useReducer, useState} from "react";
|
||||||
|
import AceEditor from "react-ace";
|
||||||
import "ace-builds/src-noconflict/mode-velocity";
|
import "ace-builds/src-noconflict/mode-velocity";
|
||||||
|
import {Link} from "react-router-dom";
|
||||||
|
|
||||||
/*******************************************************************************
|
/*******************************************************************************
|
||||||
** Utility class for working with QQQ Values
|
** Utility class for working with QQQ Values
|
||||||
@ -198,7 +197,7 @@ class ValueUtils
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (field.type == QFieldType.BLOB)
|
if (field.type == QFieldType.BLOB || field.hasAdornment(AdornmentType.FILE_DOWNLOAD))
|
||||||
{
|
{
|
||||||
return (<BlobComponent field={field} url={rawValue} filename={displayValue} usage={usage} />);
|
return (<BlobComponent field={field} url={rawValue} filename={displayValue} usage={usage} />);
|
||||||
}
|
}
|
||||||
@ -219,7 +218,7 @@ class ValueUtils
|
|||||||
|
|
||||||
if (field.type === QFieldType.DATE_TIME)
|
if (field.type === QFieldType.DATE_TIME)
|
||||||
{
|
{
|
||||||
if(displayValue && displayValue != rawValue)
|
if (displayValue && displayValue != rawValue)
|
||||||
{
|
{
|
||||||
//////////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
// if the date-time actually has a displayValue set, and it isn't just the //
|
// if the date-time actually has a displayValue set, and it isn't just the //
|
||||||
@ -276,7 +275,7 @@ class ValueUtils
|
|||||||
// to millis) back to it //
|
// to millis) back to it //
|
||||||
////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////
|
||||||
date = new Date(date);
|
date = new Date(date);
|
||||||
date.setTime(date.getTime() + date.getTimezoneOffset() * 60 * 1000)
|
date.setTime(date.getTime() + date.getTimezoneOffset() * 60 * 1000);
|
||||||
}
|
}
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
return (`${date.toString("yyyy-MM-dd")}`);
|
return (`${date.toString("yyyy-MM-dd")}`);
|
||||||
@ -474,7 +473,7 @@ class ValueUtils
|
|||||||
*******************************************************************************/
|
*******************************************************************************/
|
||||||
public static cleanForCsv(param: any): string
|
public static cleanForCsv(param: any): string
|
||||||
{
|
{
|
||||||
if(param === undefined || param === null)
|
if (param === undefined || param === null)
|
||||||
{
|
{
|
||||||
return ("");
|
return ("");
|
||||||
}
|
}
|
||||||
@ -499,7 +498,7 @@ class ValueUtils
|
|||||||
////////////////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// little private component here, for rendering an AceEditor with some buttons/controls/state //
|
// little private component here, for rendering an AceEditor with some buttons/controls/state //
|
||||||
////////////////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
function CodeViewer({name, mode, code}: {name: string; mode: string; code: string;}): JSX.Element
|
function CodeViewer({name, mode, code}: { name: string; mode: string; code: string; }): JSX.Element
|
||||||
{
|
{
|
||||||
const [activeCode, setActiveCode] = useState(code);
|
const [activeCode, setActiveCode] = useState(code);
|
||||||
const [isFormatted, setIsFormatted] = useState(false);
|
const [isFormatted, setIsFormatted] = useState(false);
|
||||||
@ -596,7 +595,7 @@ function CodeViewer({name, mode, code}: {name: string; mode: string; code: strin
|
|||||||
////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// little private component here, for rendering "secret-ish" values, that you can click to reveal or copy //
|
// little private component here, for rendering "secret-ish" values, that you can click to reveal or copy //
|
||||||
////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
function RevealComponent({fieldName, value, usage}: {fieldName: string, value: string, usage: string;}): JSX.Element
|
function RevealComponent({fieldName, value, usage}: { fieldName: string, value: string, usage: string; }): JSX.Element
|
||||||
{
|
{
|
||||||
const [adornmentFieldsMap, setAdornmentFieldsMap] = useState(new Map<string, boolean>);
|
const [adornmentFieldsMap, setAdornmentFieldsMap] = useState(new Map<string, boolean>);
|
||||||
const [, forceUpdate] = useReducer((x) => x + 1, 0);
|
const [, forceUpdate] = useReducer((x) => x + 1, 0);
|
||||||
@ -653,7 +652,7 @@ function RevealComponent({fieldName, value, usage}: {fieldName: string, value: s
|
|||||||
</Tooltip>
|
</Tooltip>
|
||||||
</ClickAwayListener>
|
</ClickAwayListener>
|
||||||
</Box>
|
</Box>
|
||||||
):(
|
) : (
|
||||||
<Box display="inline"><Icon onClick={(e) => handleRevealIconClick(e, fieldName)} sx={{cursor: "pointer", fontSize: "15px !important", position: "relative", top: "3px", marginRight: "5px"}}>visibility_off</Icon>{displayValue}</Box>
|
<Box display="inline"><Icon onClick={(e) => handleRevealIconClick(e, fieldName)} sx={{cursor: "pointer", fontSize: "15px !important", position: "relative", top: "3px", marginRight: "5px"}}>visibility_off</Icon>{displayValue}</Box>
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
@ -680,7 +679,7 @@ function BlobComponent({field, url, filename, usage}: BlobComponentProps): JSX.E
|
|||||||
const download = (event: React.MouseEvent<HTMLSpanElement>) =>
|
const download = (event: React.MouseEvent<HTMLSpanElement>) =>
|
||||||
{
|
{
|
||||||
event.stopPropagation();
|
event.stopPropagation();
|
||||||
HtmlUtils.downloadUrlViaIFrame(url, filename);
|
HtmlUtils.downloadUrlViaIFrame(field, url, filename);
|
||||||
};
|
};
|
||||||
|
|
||||||
const open = (event: React.MouseEvent<HTMLSpanElement>) =>
|
const open = (event: React.MouseEvent<HTMLSpanElement>) =>
|
||||||
@ -689,7 +688,7 @@ function BlobComponent({field, url, filename, usage}: BlobComponentProps): JSX.E
|
|||||||
HtmlUtils.openInNewWindow(url, filename);
|
HtmlUtils.openInNewWindow(url, filename);
|
||||||
};
|
};
|
||||||
|
|
||||||
if(!filename || !url)
|
if (!filename || !url)
|
||||||
{
|
{
|
||||||
return (<React.Fragment />);
|
return (<React.Fragment />);
|
||||||
}
|
}
|
||||||
@ -704,10 +703,22 @@ function BlobComponent({field, url, filename, usage}: BlobComponentProps): JSX.E
|
|||||||
usage == "view" && filename
|
usage == "view" && filename
|
||||||
}
|
}
|
||||||
<Tooltip placement={tooltipPlacement} title="Open file">
|
<Tooltip placement={tooltipPlacement} title="Open file">
|
||||||
<Icon className={"blobIcon"} fontSize="small" onClick={(e) => open(e)}>open_in_new</Icon>
|
{
|
||||||
|
field.type == QFieldType.BLOB ? (
|
||||||
|
<Icon className={"blobIcon"} fontSize="small" onClick={(e) => open(e)}>open_in_new</Icon>
|
||||||
|
) : (
|
||||||
|
<a style={{color: "inherit"}} rel="noopener noreferrer" href={url} target="_blank"><Icon className={"blobIcon"} fontSize="small">open_in_new</Icon></a>
|
||||||
|
)
|
||||||
|
}
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
<Tooltip placement={tooltipPlacement} title="Download file">
|
<Tooltip placement={tooltipPlacement} title="Download file">
|
||||||
<Icon className={"blobIcon"} fontSize="small" onClick={(e) => download(e)}>save_alt</Icon>
|
{
|
||||||
|
field.type == QFieldType.BLOB ? (
|
||||||
|
<Icon className={"blobIcon"} fontSize="small" onClick={(e) => download(e)}>save_alt</Icon>
|
||||||
|
) : (
|
||||||
|
<a style={{color: "inherit"}} href={url} download="test.pdf"><Icon className={"blobIcon"} fontSize="small">save_alt</Icon></a>
|
||||||
|
)
|
||||||
|
}
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
{
|
{
|
||||||
usage == "query" && filename
|
usage == "query" && filename
|
||||||
@ -717,5 +728,4 @@ function BlobComponent({field, url, filename, usage}: BlobComponentProps): JSX.E
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
export default ValueUtils;
|
export default ValueUtils;
|
||||||
|
Reference in New Issue
Block a user