@onehat/ui 0.3.208 → 0.3.211

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@onehat/ui",
3
- "version": "0.3.208",
3
+ "version": "0.3.211",
4
4
  "description": "Base UI for OneHat apps",
5
5
  "main": "src/index.js",
6
6
  "type": "module",
@@ -552,19 +552,21 @@ function Form(props) {
552
552
 
553
553
  let isRequired = false,
554
554
  requiredIndicator = null;
555
- if (getIsRequired) {
556
- isRequired = getIsRequired(formGetValues, formState);
557
- } else if (validatorToUse?.fields && validatorToUse.fields[name]?.exclusiveTests?.required) {
558
- // submitted validator
559
- isRequired = true;
560
- } else if ((propertyDef?.validator?.spec && !propertyDef.validator.spec.optional) ||
561
- (propertyDef?.requiredIfPhantom && isPhantom) ||
562
- (propertyDef?.requiredIfNotPhantom && !isPhantom)) {
563
- // property definition
564
- isRequired = true;
565
- }
566
- if (isRequired) {
567
- requiredIndicator = <Text color="#f00" fontSize="30px" pr={1}>*</Text>;
555
+ if (!isMultiple) { // Don't require fields if editing multiple records
556
+ if (getIsRequired) {
557
+ isRequired = getIsRequired(formGetValues, formState);
558
+ } else if (validatorToUse?.fields && validatorToUse.fields[name]?.exclusiveTests?.required) {
559
+ // submitted validator
560
+ isRequired = true;
561
+ } else if ((propertyDef?.validator?.spec && !propertyDef.validator.spec.optional) ||
562
+ (propertyDef?.requiredIfPhantom && isPhantom) ||
563
+ (propertyDef?.requiredIfNotPhantom && !isPhantom)) {
564
+ // property definition
565
+ isRequired = true;
566
+ }
567
+ if (isRequired) {
568
+ requiredIndicator = <Text color="#f00" fontSize="30px" pr={1}>*</Text>;
569
+ }
568
570
  }
569
571
  if (!disableLabels && label && editorType !== EDITOR_TYPE__INLINE) {
570
572
  const labelProps = {};
@@ -414,20 +414,26 @@ export default function withEditor(WrappedComponent, isTree = false) {
414
414
  setIsEditorShown(false);
415
415
  });
416
416
  },
417
- calculateEditorMode = () => {
418
- let mode = editorMode;
419
- if (!canEditorViewOnly && userCanEdit) {
420
- if (selection.length > 1) {
421
- if (!disableEdit) {
422
- // For multiple entities selected, change it to edit multiple mode
423
- mode = EDITOR_MODE__EDIT;
424
- }
425
- } else if (selection.length === 1 && !selection[0].isDestroyed && selection[0].isPhantom) {
426
- if (!disableAdd) {
427
- // When a phantom entity is selected, change it to add mode.
428
- mode = EDITOR_MODE__ADD;
417
+ calculateEditorMode = (isIgnoreNextSelectionChange = false) => {
418
+ // calculateEditorMode gets called only on selection changes
419
+ let mode;
420
+ if (isIgnoreNextSelectionChange) {
421
+ mode = editorMode;
422
+ if (!canEditorViewOnly && userCanEdit) {
423
+ if (selection.length > 1) {
424
+ if (!disableEdit) {
425
+ // For multiple entities selected, change it to edit multiple mode
426
+ mode = EDITOR_MODE__EDIT;
427
+ }
428
+ } else if (selection.length === 1 && !selection[0].isDestroyed && selection[0].isPhantom) {
429
+ if (!disableAdd) {
430
+ // When a phantom entity is selected, change it to add mode.
431
+ mode = EDITOR_MODE__ADD;
432
+ }
429
433
  }
430
434
  }
435
+ } else {
436
+ mode = selection.length > 1 ? EDITOR_MODE__EDIT : EDITOR_MODE__VIEW;
431
437
  }
432
438
  return mode;
433
439
  },
@@ -462,16 +468,7 @@ export default function withEditor(WrappedComponent, isTree = false) {
462
468
  }, []);
463
469
 
464
470
  useEffect(() => {
465
- // When selection changes, set the mode appropriately
466
- let mode;
467
- if (isIgnoreNextSelectionChange) {
468
- // on selection change from doAdd/doDuplicate/etc, calculate whether to put Editor in "add" or "edit" mode
469
- mode = calculateEditorMode();
470
- } else {
471
- // Most of the time, if selection changed, put the Editor in "view" mode
472
- mode = EDITOR_MODE__VIEW;
473
- }
474
- setEditorMode(mode);
471
+ setEditorMode(calculateEditorMode(isIgnoreNextSelectionChange));
475
472
 
476
473
  setIsIgnoreNextSelectionChange(false);
477
474
  setLastSelection(selection);
@@ -492,11 +489,7 @@ export default function withEditor(WrappedComponent, isTree = false) {
492
489
  // NOTE: If I don't calculate this on the fly for selection changes,
493
490
  // we see a flash of the previous state, since useEffect hasn't yet run.
494
491
  // (basically redo what's in the useEffect, above)
495
- if (isIgnoreNextSelectionChange) {
496
- editorMode = calculateEditorMode();
497
- } else {
498
- editorMode = EDITOR_MODE__VIEW;
499
- }
492
+ editorMode = calculateEditorMode(isIgnoreNextSelectionChange);
500
493
  }
501
494
 
502
495
  return <WrappedComponent
@@ -16,7 +16,8 @@ import ChartLine from '../Icons/ChartLine.js';
16
16
  import Pdf from '../Icons/Pdf.js';
17
17
  import Excel from '../Icons/Excel.js';
18
18
  import UiGlobals from '../../UiGlobals.js';
19
- import qs from 'qs';
19
+ import downloadInBackground from '../../Functions/downloadInBackground.js';
20
+ import downloadWithFetch from '../../Functions/downloadWithFetch.js';
20
21
  import _ from 'lodash';
21
22
 
22
23
  const
@@ -38,36 +39,8 @@ function Report(props) {
38
39
  showReportHeaders = true,
39
40
  h = '300px',
40
41
  } = props,
41
- styles = UiGlobals.styles,
42
42
  url = UiGlobals.baseURL + 'Reports/getReport',
43
43
  buttons = [],
44
- downloadInBackground = (data) => {
45
- const a = document.createElement('A');
46
- a.href = url + '?' + qs.stringify(data);
47
- a.download = true;
48
- document.body.appendChild(a);
49
- a.click();
50
- document.body.removeChild(a);
51
- },
52
- downloadWithFetch = (data) => {
53
- const options = {
54
- method: 'POST',
55
- headers: {
56
- 'Content-Type': 'application/json',
57
- },
58
- body: JSON.stringify(data),
59
- };
60
- fetch(url, options)
61
- .then( res => res.blob() )
62
- .then( blob => {
63
- const
64
- winName = 'ReportWindow',
65
- opts = 'resizable=yes,height=600,width=800,location=0,menubar=0,scrollbars=1',
66
- externalWindow = window.open('', winName, opts),
67
- file = externalWindow.URL.createObjectURL(blob);
68
- externalWindow.location.assign(file);
69
- });
70
- },
71
44
  getReport = (reportType, data) => {
72
45
  const params = {
73
46
  report_id: reportId,
@@ -78,10 +51,21 @@ function Report(props) {
78
51
  };
79
52
 
80
53
  if (reportType === EXCEL) {
81
- downloadInBackground(params);
54
+ downloadInBackground(url, params);
82
55
  } else {
83
56
  // opens a new window
84
- downloadWithFetch(params);
57
+ const win = window.open('');
58
+ win.document.write('<html><head><title>Downloading</title></head><body><img style="position:absolute;top:50%;left:50%;transform:translate(-50%, -50%);" src="' +
59
+ '' +
60
+ '"></body></html>');
61
+ const options = {
62
+ method: 'POST',
63
+ headers: {
64
+ 'Content-Type': 'application/json',
65
+ },
66
+ body: JSON.stringify(params),
67
+ };
68
+ downloadWithFetch(url, options, win);
85
69
  }
86
70
  };
87
71
 
@@ -34,6 +34,12 @@ function UploadsDownloadsWindow(props) {
34
34
  [importFile, setImportFile] = useState(null),
35
35
  [width, height] = useAdjustedWindowSize(400, 450),
36
36
  onDownload = (isTemplate = false) => {
37
+
38
+ const win = window.open('');
39
+ win.document.write('<html><head><title>Downloading</title></head><body><img style="position:absolute;top:50%;left:50%;transform:translate(-50%, -50%);" src="' +
40
+ '' +
41
+ '"></body></html>');
42
+
37
43
  const
38
44
  baseURL = Repository.api.baseURL,
39
45
  filters = Repository.filters.reduce((result, current) => {
@@ -61,7 +67,7 @@ function UploadsDownloadsWindow(props) {
61
67
  }),
62
68
  headers: _.merge({ 'Content-Type': 'application/json' }, Repository.headers),
63
69
  },
64
- fetchWindow = downloadWithFetch(url, options),
70
+ fetchWindow = downloadWithFetch(url, options, win),
65
71
  interval = setInterval(function() {
66
72
  const cookie = Cookies.get(download_token);
67
73
  if (fetchWindow.window && cookie) {
@@ -0,0 +1,11 @@
1
+ import qs from 'qs';
2
+
3
+ const downloadInBackground = (url, data) => {
4
+ const a = document.createElement('A');
5
+ a.href = url + '?' + qs.stringify(data);
6
+ a.download = true;
7
+ document.body.appendChild(a);
8
+ a.click();
9
+ document.body.removeChild(a);
10
+ };
11
+ export default downloadInBackground;
@@ -1,13 +1,16 @@
1
- const downloadWithFetch = (url, options = {}) => {
1
+ const downloadWithFetch = (url, options = {}, win = null) => {
2
2
  let obj = {};
3
3
  fetch(url, options)
4
4
  .then( res => res.blob() )
5
5
  .then( blob => {
6
- const
7
- winName = 'ReportWindow',
8
- opts = 'resizable=yes,height=600,width=800,location=0,menubar=0,scrollbars=1',
9
- win = window.open('', winName, opts),
10
- file = win.URL.createObjectURL(blob);
6
+ if (!win) {
7
+ const
8
+ winName = 'Download',
9
+ opts = 'resizable=yes,height=100,width=50,location=0,menubar=0,scrollbars=0';
10
+ win = window.open('', winName, opts);
11
+ }
12
+
13
+ const file = win.URL.createObjectURL(blob);
11
14
  obj.window = win;
12
15
  win.location.assign(file);
13
16
  });