Merge pull request #3 from Kitzanos/feature/bando-form-editor

Form editor
This commit is contained in:
Vitalii Kiiko
2024-09-02 17:16:44 +02:00
committed by GitHub
41 changed files with 1531 additions and 356 deletions

View File

@@ -4,11 +4,34 @@ import Routes from './routes';
import { createI18n, setLocaleData } from '@wordpress/i18n'; import { createI18n, setLocaleData } from '@wordpress/i18n';
import { I18nProvider } from '@wordpress/react-i18n'; import { I18nProvider } from '@wordpress/react-i18n';
import './assets/scss/theme.scss'; import './assets/scss/theme.scss';
import AuthenticationService from './service/authentication-service';
// store
import { useStore, storeSet } from './store';
const i18n = createI18n({}, 'gepafin'); const i18n = createI18n({}, 'gepafin');
function App() { function App() {
const role = useStore().main.getRole();
const callback = (data) => {
if (data.status === 'SUCCESS') {
storeSet.main.userData(data.data);
} else {
storeSet.main.doLogout();
}
storeSet.main.unsetAsyncRequest();
}
const errCallback = (data) => {
storeSet.main.doLogout();
storeSet.main.unsetAsyncRequest();
}
useEffect(() => { useEffect(() => {
storeSet.main.setAsyncRequest();
AuthenticationService.me(callback, errCallback);
fetch('/languages/en_US.json') fetch('/languages/en_US.json')
.then((res) => res.json()) .then((res) => res.json())
.then(res => { .then(res => {
@@ -19,7 +42,7 @@ function App() {
return ( return (
<I18nProvider i18n={i18n}> <I18nProvider i18n={i18n}>
<BrowserRouter> <BrowserRouter>
<Routes/> <Routes role={role}/>
</BrowserRouter> </BrowserRouter>
</I18nProvider> </I18nProvider>
); );

View File

@@ -100,6 +100,8 @@
padding: 17px; padding: 17px;
border-radius: 6px; border-radius: 6px;
border: 1px solid var(--card-borderColor-color); border: 1px solid var(--card-borderColor-color);
container-name: section_with_border;
container-type: inline-size;
&.disabled { &.disabled {
filter: grayscale(1); filter: grayscale(1);
@@ -140,6 +142,15 @@
} }
} }
@container section_with_border (max-width: 600px) {
.appPageSection__withBorder {
.row {
flex-direction: column;
align-items: flex-start;
}
}
}
.appPageSection__hero { .appPageSection__hero {
display: flex; display: flex;
height: 172px; height: 172px;
@@ -198,4 +209,5 @@
display: flex; display: flex;
gap: 24px; gap: 24px;
padding: 24px 0 48px; padding: 24px 0 48px;
flex-wrap: wrap;
} }

View File

@@ -12,9 +12,9 @@
h2 { h2 {
color: #404D5B; color: #404D5B;
font-size: 21px; font-size: 14px;
font-style: normal; font-style: normal;
font-weight: 600; font-weight: 700;
line-height: normal; line-height: normal;
} }
} }
@@ -58,7 +58,7 @@
.actions { .actions {
display: flex; display: flex;
gap: 1rem; gap: 0.5rem;
} }
} }
@@ -69,9 +69,9 @@
h2 { h2 {
color: #404D5B; color: #404D5B;
font-size: 21px; font-size: 14px;
font-style: normal; font-style: normal;
font-weight: 600; font-weight: 700;
line-height: normal; line-height: normal;
} }
} }
@@ -114,3 +114,23 @@
flex-direction: column; flex-direction: column;
gap: 0.5rem; gap: 0.5rem;
} }
.formElementSettings__tabs {
width: 100%;
.p-tabview-nav-link[aria-selected="true"] {
background-color: var(--message-info-background);
}
.p-tabview-panel {
display: flex;
flex-direction: column;
gap: 1rem;
}
}
.formElementSettings__repeater {
display: flex;
flex-direction: column;
gap: 0.5rem;
}

View File

@@ -86,6 +86,7 @@ body {
padding: 0 24px 20px; padding: 0 24px 20px;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
width: calc(100% - 320px);
> footer { > footer {
margin-top: auto; margin-top: auto;

View File

@@ -36,16 +36,39 @@
} }
.p-message.p-message-error { .p-message.p-message-error {
background: #ffdbdb; background: var(--message-error-background);
border-left: 5px solid #C2504D; border-left: 5px solid var(--message-error-color);
span.p-message-detail, span.p-message-summary {
color: var(--message-error-color);
}
svg { svg {
path { path {
fill: var(--global-textColor); fill: var(--message-error-color);
} }
} }
} }
.p-message.p-message-info {
background: var(--message-info-background);
border-left: 5px solid var(--message-info-color);
span.p-message-detail, span.p-message-summary {
color: var(--message-info-color);
}
svg {
path {
fill: var(--message-info-color);
}
}
}
.p-dropdown-panel .p-dropdown-items .p-dropdown-item.p-highlight {
background: #C9E3CC;
}
.p-accordion { .p-accordion {
width: 100%; width: 100%;
} }

View File

@@ -1,13 +1,18 @@
.statsBigBadges { .statsBigBadges {
} }
.statsBigBadges__grid { .statsBigBadges__grid {
display: grid; display: grid;
align-items: center; align-items: stretch;
grid-template-columns: repeat(auto-fit, minmax(460px, 1fr)); grid-template-columns: repeat(2, 1fr);
gap: 24px; gap: 1rem;
width: 100%; width: 100%;
} }
.statsBigBadges__grid .statsBigBadges__gridItem span {
color: #FFF;
}
.statsBigBadges__gridItem { .statsBigBadges__gridItem {
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
@@ -43,3 +48,9 @@
background: var(--card-full-background-color-1); background: var(--card-full-background-color-1);
} }
} }
@media (max-width: 1240px) {
.statsBigBadges__grid {
grid-template-columns: 1fr;
}
}

View File

@@ -15,6 +15,10 @@
--global-textColor: #4B5563; --global-textColor: #4B5563;
--theme-highlight-background: #BADEBE; --theme-highlight-background: #BADEBE;
--primary-text: #3B7C43; --primary-text: #3B7C43;
--message-error-background: #ffdbdb;
--message-error-color: #C2504D;
--message-info-background: rgba(219, 234, 254, 0.70);
--message-info-color: #3B82F6;
--card-full-background-color-2: #EEC137; --card-full-background-color-2: #EEC137;
--card-full-background-color-3: #FA8E42; --card-full-background-color-3: #FA8E42;

View File

@@ -34,7 +34,7 @@ const DatepickerRange = ({
<Calendar id={field.name} <Calendar id={field.name}
value={field.value} value={field.value}
onChange={field.onChange} onChange={field.onChange}
dateFormat="dd/mm/yy"
mask="99/99/9999" mask="99/99/9999"
showIcon showIcon
minDate={minDate} minDate={minDate}

View File

@@ -21,9 +21,11 @@ const Fileupload = ({
doctype = 'images', doctype = 'images',
emptyText = __('Trascina qui il tuo file', 'gepafin'), emptyText = __('Trascina qui il tuo file', 'gepafin'),
chooseLabel = __('Aggiungi immagine', 'gepafin'), chooseLabel = __('Aggiungi immagine', 'gepafin'),
multiple = false multiple = false,
callId = 0
}) => { }) => {
const [stateFieldData, setStateFieldData] = useState([]); const [stateFieldData, setStateFieldData] = useState([]);
const [acceptFormats, setAcceptFormats] = useState('');
const inputRef = useRef(); const inputRef = useRef();
const customBase64Uploader = (event) => { const customBase64Uploader = (event) => {
@@ -31,14 +33,10 @@ const Fileupload = ({
for (const file of event.files) { for (const file of event.files) {
formData.append('file', file) formData.append('file', file)
} }
/*for (const pair of formData.entries()) { FileUploadService.uploadFile(callId, formData, callback, errorCallback, [['documentType', doctype.toUpperCase()]]);
console.log(pair[0], pair[1]);
}*/
FileUploadService.uploadFile(formData, callback, errorCallback, [['documentType', doctype.toUpperCase()]]);
}; };
const callback = (data) => { const callback = (data) => {
console.log('data', data);
if (data.status === 'SUCCESS') { if (data.status === 'SUCCESS') {
setStateFieldData(data.data); setStateFieldData(data.data);
const files = inputRef.current.getFiles(); const files = inputRef.current.getFiles();
@@ -86,7 +84,6 @@ const Fileupload = ({
setStateFieldData(prevState => { setStateFieldData(prevState => {
const newFiles = prevState.filter(o => o.id !== id); const newFiles = prevState.filter(o => o.id !== id);
inputRef.current.setUploadedFiles(newFiles); inputRef.current.setUploadedFiles(newFiles);
console.log('newFiles', newFiles);
return newFiles; return newFiles;
}); });
} }
@@ -96,39 +93,61 @@ const Fileupload = ({
console.log('err', err); console.log('err', err);
} }
const onBeforeDrop = (e) => {
return validateFileInputType(e.dataTransfer.files);
}
const validateFileInputType = ( files ) => {
const MIMEtype = new RegExp( acceptFormats );
return Array.prototype.every.call( files, function passesAcceptedFormat( file ){
return MIMEtype.test( file.type );
} );
}
useEffect(() => { useEffect(() => {
setStateFieldData(defaultValue); setStateFieldData(defaultValue);
register(fieldName, config) register(fieldName, config)
}, []); }, []);
useEffect(() => {
// eslint-disable-next-line no-useless-escape
setAcceptFormats(accept.replace( /\*/g, '.\*' ).replace( /,/g, '|' ));
}, [accept]);
useEffect(() => { useEffect(() => {
inputRef.current.setUploadedFiles(stateFieldData); inputRef.current.setUploadedFiles(stateFieldData);
setDataFn(fieldName, [...stateFieldData], { shouldValidate: true }); setDataFn(fieldName, [...stateFieldData], { shouldValidate: true });
}, [stateFieldData]) }, [stateFieldData])
return ( return (
<> callId && callId !== 0
<label htmlFor={fieldName} className={classNames({ 'p-error': errors[fieldName] })}> ? <>
{label}{config.required ? '*' : null} <label htmlFor={fieldName} className={classNames({ 'p-error': errors[fieldName] })}>
</label> {label}{config.required ? '*' : null}
<FileUpload {acceptFormats ? ' (' + acceptFormats.split('|').join(', ') + ')' : null}
ref={inputRef} </label>
id={fieldName} <FileUpload
name={fieldName} ref={inputRef}
url={'/document/uploadFile'} id={fieldName}
multiple={multiple} name={fieldName}
accept={accept} url={'/document/uploadFile'}
maxFileSize={1000000} multiple={multiple}
emptyTemplate={<p>{emptyText}</p>} accept={accept}
chooseLabel={chooseLabel} maxFileSize={1000000}
cancelLabel={__('Cancella', 'gepafin')} emptyTemplate={<p>{emptyText}</p>}
uploadLabel={__('Carica', 'gepafin')} chooseLabel={chooseLabel}
className={classNames({ 'p-invalid': errors[fieldName] })} cancelLabel={__('Cancella', 'gepafin')}
itemTemplate={itemTemplate} uploadLabel={__('Carica', 'gepafin')}
customUpload className={classNames({ 'p-invalid': errors[fieldName] })}
uploadHandler={customBase64Uploader}/> itemTemplate={itemTemplate}
{infoText ? <small>{infoText}</small> : null} customUpload
</>) onBeforeDrop={onBeforeDrop}
uploadHandler={customBase64Uploader}/>
{infoText ? <small>{infoText}</small> : null}
</>
: null
)
} }
export default Fileupload; export default Fileupload;

View File

@@ -1,7 +1,7 @@
import React, { useRef, useEffect, useState } from 'react'; import React, { useRef, useEffect, useState, useCallback } from 'react';
import { classNames } from 'primereact/utils'; import { classNames } from 'primereact/utils';
import { __ } from '@wordpress/i18n'; import { __ } from '@wordpress/i18n';
import { isEmpty } from 'ramda'; import { head, isNil, pluck } from 'ramda';
// components // components
import { InputText } from 'primereact/inputtext'; import { InputText } from 'primereact/inputtext';
@@ -22,19 +22,20 @@ const FormFieldRepeater = ({
}) => { }) => {
const forMenu = useRef(null); const forMenu = useRef(null);
const [stateFieldData, setStateFieldData] = useState([]); const [stateFieldData, setStateFieldData] = useState([]);
const [stateOptionsData, setStateOptionsData] = useState([]);
const menuItems = [ const menuItems = [
{ {
type: 'existing', type: 'existing',
label: __('Esistente', 'gepafin'), label: __('Esistente', 'gepafin'),
command: (data) => { command: (data) => {
setStateFieldData([...stateFieldData, { id: null, value: '', status: data.item.type }]); setStateFieldData([...stateFieldData, { id: null, value: '', lookUpDataId: 0 }]);
} }
}, },
{ {
type: 'new', type: 'new',
label: __('Nuovo', 'gepafin'), label: __('Nuovo', 'gepafin'),
command: (data) => { command: (data) => {
setStateFieldData([...stateFieldData, { id: null, value: '', status: data.item.type }]); setStateFieldData([...stateFieldData, { id: null, value: '', lookUpDataId: null }]);
} }
} }
] ]
@@ -45,14 +46,14 @@ const FormFieldRepeater = ({
} }
const selectItem = (e, index) => { const selectItem = (e, index) => {
const newData = stateFieldData.map((o, i) => { const targetedOption = head(stateOptionsData.filter(o => o.value === e.value));
if (i === index) {
o.value = e.value; if (targetedOption) {
o.id = e.id; const newData = stateFieldData.map((o, i) => {
} return i === index ? targetedOption : o;
return o; })
}) setStateFieldData(newData);
setStateFieldData(newData); }
} }
const onInputChange = (e, index) => { const onInputChange = (e, index) => {
@@ -67,24 +68,30 @@ const FormFieldRepeater = ({
} }
const properField = (item, i) => { const properField = (item, i) => {
return item.status === 'new' return !isNil(item.lookUpDataId)
? <InputText value={item.value} onInput={(e) => onInputChange(e, i)}/> ? <Dropdown value={item.value}
: <Dropdown value={item.value}
onChange={(e) => selectItem(e, i)} onChange={(e) => selectItem(e, i)}
optionDisabled={(opt) => usedExistingValues.includes(opt.value)} optionDisabled={(opt) => usedExistingValues().includes(opt.value)}
options={options} optionLabel="value"/> options={stateOptionsData}
optionLabel="value"/>
: <InputText value={item.value} onInput={(e) => onInputChange(e, i)}/>
} }
const usedExistingValues = stateFieldData const usedExistingValues = useCallback(() => {
.filter(o => o.status === 'existing') return stateFieldData
.map(o => o.value); .filter(o => o.lookUpDataId > 0)
.map(o => o.value)
}, [stateFieldData]);
useEffect(() => { useEffect(() => {
const storeFieldData = data[fieldName] ?? []; const storeFieldData = data[fieldName] ?? [];
const newData = storeFieldData.map(o => ({ ...o, status: o.id ? 'existing' : 'new' })); setStateFieldData(storeFieldData);
setStateFieldData(newData);
register(fieldName, config); register(fieldName, config);
}, []) }, []);
useEffect(() => {
setStateOptionsData([...options]);
}, [options]);
useEffect(() => { useEffect(() => {
setDataFn(fieldName, [...stateFieldData], { shouldValidate: true }); setDataFn(fieldName, [...stateFieldData], { shouldValidate: true });
@@ -100,7 +107,7 @@ const FormFieldRepeater = ({
{properField(o, i)} {properField(o, i)}
<Button icon="pi pi-times" className="p-button-danger" onClick={() => removeItem(i)}/> <Button icon="pi pi-times" className="p-button-danger" onClick={() => removeItem(i)}/>
</div> </div>
{o.status === 'new' && infoText ? <small>{infoText}</small> : null} {isNil(o.lookUpDataId) && infoText ? <small>{infoText}</small> : null}
</div>)} </div>)}
<Menu model={menuItems} popup ref={forMenu} id="aimedForMenu"/> <Menu model={menuItems} popup ref={forMenu} id="aimedForMenu"/>
<Button type="button" iconPos="right" label={__('Aggiungi', 'gepafin')} <Button type="button" iconPos="right" label={__('Aggiungi', 'gepafin')}

View File

@@ -1,7 +1,7 @@
import React, { useRef, useEffect, useState } from 'react'; import React, { useRef, useEffect, useState, useCallback } from 'react';
import { classNames } from 'primereact/utils'; import { classNames } from 'primereact/utils';
import { __ } from '@wordpress/i18n'; import { __ } from '@wordpress/i18n';
import { head } from 'ramda'; import { head, isNil, pluck } from 'ramda';
// components // components
import { InputText } from 'primereact/inputtext'; import { InputText } from 'primereact/inputtext';
@@ -23,20 +23,21 @@ const FormFieldRepeaterCriteria = ({
}) => { }) => {
const forMenu = useRef(null); const forMenu = useRef(null);
const [stateFieldData, setStateFieldData] = useState([]); const [stateFieldData, setStateFieldData] = useState([]);
const [stateOptionsData, setStateOptionsData] = useState([]);
const [threshold, setThreshold] = useState(0); const [threshold, setThreshold] = useState(0);
const menuItems = [ const menuItems = [
{ {
type: 'existing', type: 'existing',
label: __('Esistente', 'gepafin'), label: __('Esistente', 'gepafin'),
command: (data) => { command: (data) => {
setStateFieldData([...stateFieldData, { id: null, value: '', status: data.item.type }]); setStateFieldData([...stateFieldData, { id: null, value: '', lookUpDataId: 0 }]);
} }
}, },
{ {
type: 'new', type: 'new',
label: __('Nuovo', 'gepafin'), label: __('Nuovo', 'gepafin'),
command: (data) => { command: (data) => {
setStateFieldData([...stateFieldData, { id: null, value: '', status: data.item.type }]); setStateFieldData([...stateFieldData, { id: null, value: '', lookUpDataId: null }]);
} }
} }
] ]
@@ -47,17 +48,12 @@ const FormFieldRepeaterCriteria = ({
} }
const selectItem = (e, index) => { const selectItem = (e, index) => {
const targetedOption = head(options.filter(o => o.value === e.value)); const targetedOption = head(stateOptionsData.filter(o => o.value === e.value));
if (targetedOption) { if (targetedOption) {
const newData = stateFieldData.map((o, i) => { const newData = stateFieldData.map((o, i) => {
if (i === index) { return i === index ? targetedOption : o;
o.value = targetedOption.value; })
o.score = targetedOption.score;
o.id = targetedOption.id;
}
return o;
});
console.log('newData', newData)
setStateFieldData(newData); setStateFieldData(newData);
} }
} }
@@ -78,32 +74,38 @@ const FormFieldRepeaterCriteria = ({
} }
const properField = (item, i) => { const properField = (item, i) => {
return item.status === 'new' return !isNil(item.lookUpDataId)
? <InputText value={item.value} onInput={(e) => onInputChange(e.target.value, i, 'value')}/> ? <Dropdown value={item.value}
: <Dropdown value={item.value}
onChange={(e) => selectItem(e, i)} onChange={(e) => selectItem(e, i)}
optionDisabled={(opt) => usedExistingValues.includes(opt.value)} optionDisabled={(opt) => usedExistingValues().includes(opt.value)}
options={options} optionLabel="value"/> options={stateOptionsData} optionLabel="value"/>
: <InputText value={item.value} onInput={(e) => onInputChange(e.target.value, i, 'value')}/>
} }
const usedExistingValues = stateFieldData const usedExistingValues = useCallback(() => {
.filter(o => o.status === 'existing') return stateFieldData
.map(o => o.value); .filter(o => o.lookUpDataId > 0)
.map(o => o.value)
}, [stateFieldData]);
useEffect(() => { useEffect(() => {
const storeFieldData = data[fieldName] ?? []; const storeFieldData = data[fieldName] ?? [];
const newData = storeFieldData.map(o => ({ ...o, status: o.id ? 'existing' : 'new' })); setStateFieldData(storeFieldData);
setStateFieldData(newData); setStateOptionsData([...options, ...storeFieldData]);
setThreshold(data['threshold']) setThreshold(data['threshold'])
register(fieldName, config) register(fieldName, config)
register('threshold', { register('threshold', {
required: __('È obbligatorio', 'gepafin') required: __('È obbligatorio', 'gepafin')
}) })
}, []) }, []);
useEffect(() => {
setStateOptionsData([...options]);
}, [options]);
useEffect(() => { useEffect(() => {
setDataFn(fieldName, [...stateFieldData], { shouldValidate: true }); setDataFn(fieldName, [...stateFieldData], { shouldValidate: true });
}, [stateFieldData]) }, [stateFieldData]);
return ( return (
<div className={classNames(['appForm__field', 'formfieldrepeater'])}> <div className={classNames(['appForm__field', 'formfieldrepeater'])}>
@@ -126,7 +128,7 @@ const FormFieldRepeaterCriteria = ({
{properField(o, i)} {properField(o, i)}
<Button icon="pi pi-times" className="p-button-danger" onClick={() => removeItem(i)}/> <Button icon="pi pi-times" className="p-button-danger" onClick={() => removeItem(i)}/>
</div> </div>
{o.status === 'new' && infoText ? <small>{infoText}</small> : null} {isNil(o.lookUpDataId) && infoText ? <small>{infoText}</small> : null}
</div> </div>
<div> <div>
<label htmlFor="criterionMin">{__('Punteggio', 'gepafin')}</label> <label htmlFor="criterionMin">{__('Punteggio', 'gepafin')}</label>

View File

@@ -1,7 +1,7 @@
import React, { useEffect, useState } from 'react'; import React, { useEffect, useState, useCallback } from 'react';
import { classNames } from 'primereact/utils'; import { classNames } from 'primereact/utils';
import { __ } from '@wordpress/i18n'; import { __ } from '@wordpress/i18n';
import { isEmpty } from 'ramda'; import { head } from 'ramda';
// components // components
import { Button } from 'primereact/button'; import { Button } from 'primereact/button';
@@ -20,11 +20,10 @@ const FormFieldRepeaterFaq = ({
errors, errors,
register, register,
label, label,
config = {}, config = {}
setError,
clearErrors
}) => { }) => {
const [stateFieldData, setStateFieldData] = useState([]); const [stateFieldData, setStateFieldData] = useState([]);
const [stateOptionsData, setStateOptionsData] = useState([]);
const [question, setQuestion] = useState(''); const [question, setQuestion] = useState('');
const [answer, setAnswer] = useState(''); const [answer, setAnswer] = useState('');
const [editDataIndex, setEditDataIndex] = useState(null); const [editDataIndex, setEditDataIndex] = useState(null);
@@ -36,12 +35,15 @@ const FormFieldRepeaterFaq = ({
} }
const selectItem = (e) => { const selectItem = (e) => {
const chosen = {...e.value}; const targetedOption = head(stateOptionsData.filter(o => o.question === e.value));
setStateFieldData([...stateFieldData, chosen]);
if (targetedOption) {
setStateFieldData([...stateFieldData, targetedOption]);
}
} }
const addNewItem = () => { const addNewItem = () => {
const newItem = { id: 0, status: 'new', question: '', answer: '', visible: true }; const newItem = { id: null, lookUpDataId: null, question: '', response: '', visible: true };
setStateFieldData([...stateFieldData, newItem]); setStateFieldData([...stateFieldData, newItem]);
} }
@@ -49,7 +51,7 @@ const FormFieldRepeaterFaq = ({
e.preventDefault(); e.preventDefault();
const newData = stateFieldData.map((o, i) => { const newData = stateFieldData.map((o, i) => {
if (i === index) { if (i === index) {
o.visible = e.value; o.isVisible = e.value;
} }
return o; return o;
}); });
@@ -59,7 +61,7 @@ const FormFieldRepeaterFaq = ({
const editItem = (e, index) => { const editItem = (e, index) => {
e.stopPropagation(); e.stopPropagation();
setQuestion(stateFieldData[index].question); setQuestion(stateFieldData[index].question);
setAnswer(stateFieldData[index].answer); setAnswer(stateFieldData[index].response);
setEditDataIndex(index); setEditDataIndex(index);
setIsVisibleEditDialog(true); setIsVisibleEditDialog(true);
} }
@@ -83,7 +85,7 @@ const FormFieldRepeaterFaq = ({
const newData = stateFieldData.map((o, i) => { const newData = stateFieldData.map((o, i) => {
if (i === editDataIndex) { if (i === editDataIndex) {
o.question = question; o.question = question;
o.answer = answer; o.response = answer;
return o return o
} else { } else {
return o; return o;
@@ -102,25 +104,31 @@ const FormFieldRepeaterFaq = ({
const footerEditDialog = () => { const footerEditDialog = () => {
return <div> return <div>
<Button type="button" label={__('Anulla', 'gepafin')} onClick={hideEditDialog} outlined /> <Button type="button" label={__('Anulla', 'gepafin')} onClick={hideEditDialog} outlined/>
<Button type="button" label={__('Salva', 'gepafin')} onClick={saveEditDialog} /> <Button type="button" label={__('Salva', 'gepafin')} onClick={saveEditDialog}/>
</div> </div>
} }
const usedExistingValues = stateFieldData const usedExistingValues = useCallback(() => {
.filter(o => o.status === 'existing') return stateFieldData
.map(o => o.question); .filter(o => o.lookUpDataId)
.map(o => o.question)
}, [stateFieldData]);
useEffect(() => { useEffect(() => {
const storeFieldData = data[fieldName] ?? []; const storeFieldData = data[fieldName] ?? [];
const newData = storeFieldData.map(o => ({ ...o, status: o.id ? 'existing' : 'new' })) setStateFieldData(storeFieldData);
setStateFieldData(newData); setStateOptionsData([...options, ...storeFieldData]);
register(fieldName, config) register(fieldName, config)
}, []) }, []);
useEffect(() => {
setStateOptionsData([...options]);
}, [options]);
useEffect(() => { useEffect(() => {
setDataFn(fieldName, [...stateFieldData], { shouldValidate: true }); setDataFn(fieldName, [...stateFieldData], { shouldValidate: true });
}, [stateFieldData]) }, [stateFieldData]);
return ( return (
<div className={classNames(['appForm__field', 'formfieldrepeater'])}> <div className={classNames(['appForm__field', 'formfieldrepeater'])}>
@@ -131,8 +139,9 @@ const FormFieldRepeaterFaq = ({
<Button type="button" iconPos="left" label={__('Aggiungi', 'gepafin')} <Button type="button" iconPos="left" label={__('Aggiungi', 'gepafin')}
icon="pi pi-plus" onClick={addNewItem}/> icon="pi pi-plus" onClick={addNewItem}/>
<Dropdown onChange={(e) => selectItem(e)} <Dropdown onChange={(e) => selectItem(e)}
optionDisabled={(opt) => usedExistingValues.includes(opt.value)} optionDisabled={(opt) => usedExistingValues().includes(opt.value)}
options={options} optionLabel="question"/> options={stateOptionsData}
optionLabel="question"/>
</div> </div>
<Accordion activeIndex={0}> <Accordion activeIndex={0}>
{stateFieldData.map((o, i) => <AccordionTab key={i} {stateFieldData.map((o, i) => <AccordionTab key={i}
@@ -144,7 +153,7 @@ const FormFieldRepeaterFaq = ({
offIcon="pi pi-eye-slash" offIcon="pi pi-eye-slash"
onLabel="" onLabel=""
offLabel="" offLabel=""
checked={o.visible} checked={o.isVisible}
onChange={(e) => setChecked(e, i)}/> onChange={(e) => setChecked(e, i)}/>
{o.question} {o.question}
</div> </div>
@@ -160,7 +169,7 @@ const FormFieldRepeaterFaq = ({
} }
> >
<p className="m-0"> <p className="m-0">
{o.answer} {o.response}
</p> </p>
</AccordionTab>)} </AccordionTab>)}
</Accordion> </Accordion>
@@ -170,14 +179,18 @@ const FormFieldRepeaterFaq = ({
footer={footerEditDialog} footer={footerEditDialog}
style={{ maxWidth: '50rem' }} style={{ maxWidth: '50rem' }}
onHide={hideEditDialog}> onHide={hideEditDialog}>
<div className="appPage__spacer"></div>
<div className="appForm__field"> <div className="appForm__field">
<label>{__('Titolo FAQ', 'gepafin')}</label> <label>{__('Titolo FAQ', 'gepafin')}</label>
<InputText value={question} onChange={(e) => onChangeEditItem(e.target.value, 'question')}/> <InputText value={question} onChange={(e) => onChangeEditItem(e.target.value, 'question')}/>
</div> </div>
<div className="appForm__field"> <div className="appForm__field">
<label>{__('Risposta', 'gepafin')}</label> <label>{__('Risposta', 'gepafin')}</label>
<InputTextarea value={answer} onChange={(e) => onChangeEditItem(e.target.value, 'answer')} rows={5} cols={30} /> <InputTextarea value={answer} onChange={(e) => onChangeEditItem(e.target.value, 'response')}
rows={5}
cols={30}/>
</div> </div>
<div className="appPage__spacer"></div>
</Dialog> </Dialog>
</div> </div>
) )

View File

@@ -2,13 +2,14 @@ import React, { useRef } from 'react';
import { __ } from '@wordpress/i18n'; import { __ } from '@wordpress/i18n';
// store // store
import { storeSet } from '../../store'; import { storeSet, useTrackedStore } from '../../store';
// components // components
import { Menu } from 'primereact/menu'; import { Menu } from 'primereact/menu';
import { Avatar } from 'primereact/avatar'; import { Avatar } from 'primereact/avatar';
const TopBarProfileMenu = ({ menuLeftRef }) => { const TopBarProfileMenu = ({ menuLeftRef }) => {
const userData = useTrackedStore().main.userData();
let items = [ let items = [
{ {
@@ -17,8 +18,8 @@ const TopBarProfileMenu = ({ menuLeftRef }) => {
<div className="topBar__menuProfileItem"> <div className="topBar__menuProfileItem">
<Avatar image="https://primefaces.org/cdn/primereact/images/avatar/amyelsner.png" shape="circle" /> <Avatar image="https://primefaces.org/cdn/primereact/images/avatar/amyelsner.png" shape="circle" />
<div className="userInfo"> <div className="userInfo">
<span className="userName">Mario Rossi</span> <span className="userName">{`${userData.firstName} ${userData.lastName}`}</span>
<span className="userEmail">mario.rossi@example.com</span> <span className="userEmail">{userData.email}</span>
</div> </div>
</div> </div>
); );

View File

@@ -5,7 +5,7 @@ import equal from 'fast-deep-equal';
// tools // tools
const UnsavedChangesDetector = ({ initialData, getValuesFn }) => { const UnsavedChangesDetector = ({ initialData, getValuesFn }) => {
const [initial] = useState(initialData); const [initial, setInitial] = useState(initialData);
const warnIfUnsavedChanges = useCallback((event) => { const warnIfUnsavedChanges = useCallback((event) => {
const updatedData = getValuesFn(); const updatedData = getValuesFn();
@@ -16,7 +16,11 @@ const UnsavedChangesDetector = ({ initialData, getValuesFn }) => {
} }
return event.returnValue; return event.returnValue;
}, [initial]) }, [initialData]);
useEffect(() => {
setInitial(initialData);
}, [initialData])
useEffect(() => { useEffect(() => {
window.addEventListener('beforeunload', (e) => { window.addEventListener('beforeunload', (e) => {

View File

@@ -2,14 +2,17 @@ import { __ } from '@wordpress/i18n';
const getBandoLabel = (status) => { const getBandoLabel = (status) => {
switch (status) { switch (status) {
case 'publish': case 'PUBLISH':
return __('Pubblicato', 'gepafin'); return __('Pubblicato', 'gepafin');
case 'draft': case 'READY_TO_PUBLISH':
return __('Pronto', 'gepafin');
case 'DRAFT':
return __('Bozza', 'gepafin'); return __('Bozza', 'gepafin');
case 'closed': case 'EXPIRED':
return __('Chiuso', 'gepafin'); return __('Scaduto', 'gepafin');
default: default:
return ''; return '';

View File

@@ -1,12 +1,15 @@
const getBandoSeverity = (status) => { const getBandoSeverity = (status) => {
switch (status) { switch (status) {
case 'publish': case 'PUBLISH':
return 'success'; return 'success';
case 'draft': case 'READY_TO_PUBLISH':
return 'info';
case 'DRAFT':
return 'warning'; return 'warning';
case 'closed': case 'EXPIRED':
return 'closed'; return 'closed';
default: default:

View File

@@ -1,6 +1,6 @@
const getDateFromISOstring = (value, options = {}) => { const getDateFromISOstring = (value, options = {}) => {
const optionsMerged = options || { day: '2-digit', month: '2-digit', year: 'numeric', dateStyle: 'short' } const optionsMerged = options || { day: '2-digit', month: '2-digit', year: 'numeric', hour12: false }
return Intl.DateTimeFormat('it-IT', optionsMerged).format(new Date(value)) return value ? Intl.DateTimeFormat('it-IT', optionsMerged).format(new Date(value)) : value;
} }
export default getDateFromISOstring; export default getDateFromISOstring;

View File

@@ -1,52 +1,66 @@
import React from 'react'; import React from 'react';
import { __ } from '@wordpress/i18n'; import { __ } from '@wordpress/i18n';
import { intersection } from 'ramda';
// store
import { useStore } from '../../../../store';
// components // components
import { NavLink } from 'react-router-dom'; import { NavLink } from 'react-router-dom';
const AppSidebar = () => { const AppSidebar = () => {
const permissions = useStore().main.getPermissions();
const items = [ const items = [
{ {
label: __('Riepilogo', 'gepafin'), label: __('Riepilogo', 'gepafin'),
icon: 'pi pi-objects-column', icon: 'pi pi-objects-column',
href: '/', href: '/',
id: 1 id: 1,
enable: true
}, },
{ {
label: __('Gestione Bandi', 'gepafin'), label: __('Gestione Bandi', 'gepafin'),
icon: 'pi pi-file', icon: 'pi pi-file',
href: '/bandi', href: '/bandi',
id: 2 id: 2,
enable: intersection(permissions, ['VIEW_CALLS', 'MANAGE_TENDERS']).length
}, },
{ {
label: __('Gestione Utenti', 'gepafin'), label: __('Gestione Utenti', 'gepafin'),
icon: 'pi pi-users', icon: 'pi pi-users',
href: '/utenti', //href: '/utenti',
id: 3 id: 3,
enable: intersection(permissions, ['VIEW_USERS', 'MANAGE_USERS']).length
}, },
{ {
label: __('Configurazione', 'gepafin'), label: __('Configurazione', 'gepafin'),
icon: 'pi pi-cog', icon: 'pi pi-cog',
href: '/configurazione', //href: '/configurazione',
id: 4 id: 4,
enable: false
}, },
{ {
label: __('Report e Analisi', 'gepafin'), label: __('Report e Analisi', 'gepafin'),
icon: 'pi pi-chart-bar', icon: 'pi pi-chart-bar',
href: '/stats', //href: '/stats',
id: 5 id: 5,
enable: false
}, },
{ {
label: __('Log di Sistema', 'gepafin'), label: __('Log di Sistema', 'gepafin'),
icon: 'pi pi-receipt', icon: 'pi pi-receipt',
clickFn: () => {}, clickFn: () => {},
id: 6 id: 6,
enable: false
} }
] ]
return <aside> return <aside>
<ul> <ul>
{items.map(o => <li key={o.id}> {items
.filter(o => o.enable)
.map(o => <li key={o.id}>
{o.href {o.href
? <NavLink to={o.href}> ? <NavLink to={o.href}>
<i className={o.icon}></i> <i className={o.icon}></i>

View File

@@ -1,10 +1,17 @@
import React, { useState, useEffect} from 'react'; import React, { useState, useEffect} from 'react';
import { __ } from '@wordpress/i18n'; import { __ } from '@wordpress/i18n';
import { uniq } from 'ramda'; import { is, uniq } from 'ramda';
// store
import { storeSet, storeGet } from '../../../../store';
// tools // tools
import getBandoSeverity from '../../../../helpers/getBandoSeverity'; import getBandoSeverity from '../../../../helpers/getBandoSeverity';
import getBandoLabel from '../../../../helpers/getBandoLabel'; import getBandoLabel from '../../../../helpers/getBandoLabel';
import getDateFromISOstring from '../../../../helpers/getDateFromISOstring';
// api
import BandoService from '../../../../service/bando-service';
// components // components
import { FilterMatchMode, FilterOperator } from 'primereact/api'; import { FilterMatchMode, FilterOperator } from 'primereact/api';
@@ -20,6 +27,7 @@ import { Tag } from 'primereact/tag';
import ProperBandoLabel from '../../../../components/ProperBandoLabel'; import ProperBandoLabel from '../../../../components/ProperBandoLabel';
import { Link } from 'react-router-dom'; import { Link } from 'react-router-dom';
const AllBandiTable = () => { const AllBandiTable = () => {
const [items, setItems] = useState(null); const [items, setItems] = useState(null);
const [filters, setFilters] = useState(null); const [filters, setFilters] = useState(null);
@@ -28,56 +36,36 @@ const AllBandiTable = () => {
const [statuses, setStatuses] = useState([]); const [statuses, setStatuses] = useState([]);
useEffect(() => { useEffect(() => {
// TODO const isAsyncRequest = storeGet.main.isAsyncRequest();
const items = [ if (isAsyncRequest !== 0) {
{ return;
name: 'Bando Innovazione 2024', }
start_date: '2024-08-08T00:00:00+00:00',
end_date: '2024-08-30T00:00:00+00:00', storeSet.main.setAsyncRequest();
submissions: 24, BandoService.getBandi(getCallback, errGetCallbacks);
status: 'publish',
id: 11
},
{
name: 'Bando Sostenibilità 2024',
start_date: '2024-07-28T00:00:00+00:00',
end_date: '2024-08-15T00:00:00+00:00',
submissions: 35,
status: 'draft',
id: 9
},
{
name: 'Bando A',
start_date: '2024-06-28T00:00:00+00:00',
end_date: '2024-06-15T00:00:00+00:00',
submissions: 2,
status: 'closed',
id: 2
}
]
setItems(getFormattedBandiData(items));
setStatuses(uniq(items.map(o => o.status)))
setLoading(false);
initFilters();
}, []); }, []);
const getCallback = (data) => {
if (data.status === 'SUCCESS') {
setItems(getFormattedBandiData(data.data));
setStatuses(uniq(data.data.map(o => o.status)))
initFilters();
}
storeSet.main.unsetAsyncRequest();
}
const errGetCallbacks = (data) => {
console.log('errGetCallbacks', data)
storeSet.main.unsetAsyncRequest();
}
const getFormattedBandiData = (data) => { const getFormattedBandiData = (data) => {
return [...(data || [])].map((d) => { return data.map((d) => {
d.start_date = new Date(d.start_date); d.dates = d.dates.map(v => is(String, v) ? new Date(v) : (v ? v : ''));
d.end_date = new Date(d.end_date);
return d; return d;
}); });
}; };
const formatDate = (value) => {
return value.toLocaleDateString('it-IT', {
day: '2-digit',
month: '2-digit',
year: 'numeric'
});
};
const clearFilter = () => { const clearFilter = () => {
initFilters(); initFilters();
}; };
@@ -116,11 +104,11 @@ const AllBandiTable = () => {
}; };
const dateStartBodyTemplate = (rowData) => { const dateStartBodyTemplate = (rowData) => {
return formatDate(rowData.start_date); return getDateFromISOstring(rowData.dates[0]);
}; };
const dateEndBodyTemplate = (rowData) => { const dateEndBodyTemplate = (rowData) => {
return formatDate(rowData.end_date); return getDateFromISOstring(rowData.dates[1]);
}; };
const dateFilterTemplate = (options) => { const dateFilterTemplate = (options) => {

View File

@@ -1,10 +1,11 @@
import React from 'react'; import React from 'react';
import { __ } from '@wordpress/i18n'; import { __ } from '@wordpress/i18n';
import { isNil } from 'ramda';
// components // components
import { Button } from 'primereact/button'; import { Button } from 'primereact/button';
const BandoEditFormActions = ({ openPreview, openPreviewEvaluation }) => { const BandoEditFormActions = ({ id, openPreview, openPreviewEvaluation }) => {
return ( return (
<div className="appPageSection"> <div className="appPageSection">
<div className="appPageSection__actions"> <div className="appPageSection__actions">
@@ -13,7 +14,7 @@ const BandoEditFormActions = ({ openPreview, openPreviewEvaluation }) => {
label={__('Salva bozza', 'gepafin')} icon="pi pi-save" iconPos="right"/> label={__('Salva bozza', 'gepafin')} icon="pi pi-save" iconPos="right"/>
<Button <Button
type="button" type="button"
disabled={false} disabled={isNil(id)}
outlined outlined
onClick={openPreview} onClick={openPreview}
label={__('Anteprima beneficiario', 'gepafin')} icon="pi pi-eye" iconPos="right"/> label={__('Anteprima beneficiario', 'gepafin')} icon="pi pi-eye" iconPos="right"/>

View File

@@ -1,4 +1,4 @@
import React, { forwardRef, useImperativeHandle, useEffect } from 'react'; import React, { forwardRef, useImperativeHandle, useEffect, useState, useMemo } from 'react';
import { __ } from '@wordpress/i18n'; import { __ } from '@wordpress/i18n';
import { useNavigate } from 'react-router-dom'; import { useNavigate } from 'react-router-dom';
import { useForm } from 'react-hook-form'; import { useForm } from 'react-hook-form';
@@ -10,9 +10,16 @@ import FormFieldRepeater from '../../../../components/FormFieldRepeater';
import FormFieldRepeaterFaq from '../../../../components/FormFieldRepeaterFaq'; import FormFieldRepeaterFaq from '../../../../components/FormFieldRepeaterFaq';
import BandoEditFormActions from '../BandoEditFormActions'; import BandoEditFormActions from '../BandoEditFormActions';
import UnsavedChangesDetector from '../../../../components/UnsavedChangesDetector'; import UnsavedChangesDetector from '../../../../components/UnsavedChangesDetector';
import BandoService from '../../../../service/bando-service';
import LookupdataService from '../../../../service/lookupdata-service';
import { storeSet, useStore } from '../../../../store';
const BandoEditFormStep1 = forwardRef(function ({ initialData, getFormErrors }, ref) { const BandoEditFormStep1 = forwardRef(function ({ initialData, getFormErrors }, ref) {
const navigate = useNavigate(); const navigate = useNavigate();
const isAsyncRequest = useStore().main.isAsyncRequest();
const [aimedToOptions, setAimedToOptions] = useState([]);
const [faqOptions, setFaqOptions] = useState([]);
const [formInitialData, setFormInitialData] = useState(initialData);
const { const {
control, control,
handleSubmit, handleSubmit,
@@ -21,8 +28,13 @@ const BandoEditFormStep1 = forwardRef(function ({ initialData, getFormErrors },
register, register,
trigger, trigger,
getValues, getValues,
clearErrors clearErrors,
} = useForm({defaultValues: initialData, mode: 'onChange'}); reset
} = useForm({
defaultValues: useMemo(() => {
return formInitialData;
}, [formInitialData]), mode: 'onChange'
});
const values = getValues(); const values = getValues();
let minDateStart = new Date(); let minDateStart = new Date();
@@ -30,22 +42,67 @@ const BandoEditFormStep1 = forwardRef(function ({ initialData, getFormErrors },
if (!isNil(formData.dates) && formData.dates.length) { if (!isNil(formData.dates) && formData.dates.length) {
formData.dates = formData.dates.map(v => is(String, v) ? v : v.toISOString()); formData.dates = formData.dates.map(v => is(String, v) ? v : v.toISOString());
} }
console.log('onSubmit', formData);
if (!formData.id) {
BandoService.createBando(formData, createCallback, errCreateCallback);
} else {
BandoService.updateBandoStep1(formData.id, formData, createCallback, errCreateCallback);
}
}; };
// TODO temp data const createCallback = (data) => {
const exampleOfAimedToOptions = [{ id: 11, value: 'PMI con sede in Umbria' }]; if (data.status === 'SUCCESS') {
const exampleOfFaqOptions = [ const values = getValues();
{ id: 2, question: 'Question 1?', answer: 'Lorem ipsum dolor', visible: true } if (!values.id && data.data.id) {
]; navigate(`/bandi/${data.data.id}`);
// end of temp data } else {
setFormInitialData(data.data);
reset();
}
}
}
const errCreateCallback = (data) => {
console.log('errCreateCallback', data);
}
const openPreview = () => { const openPreview = () => {
navigate('/bandi/11/preview'); navigate(`/bandi/${values.id}/preview`);
} }
const openPreviewEvaluation = () => { const openPreviewEvaluation = () => {
navigate('/bandi/11/preview-evaluation'); navigate(`/bandi/${values.id}/preview-evaluation`);
}
const lookupdataCallback = (data) => {
if (data.status === 'SUCCESS') {
const aimedTo = data.data
.filter(o => o.type === 'AIMED_TO')
.map(o => {
delete o.type;
return {
...o,
lookUpDataId: o.id
};
});
setAimedToOptions(aimedTo);
const faqItems = data.data
.filter(o => o.type === 'FAQ')
.map(o => {
delete o.type;
return {
...o,
lookUpDataId: o.id
};
});
setFaqOptions(faqItems);
}
storeSet.main.unsetAsyncRequest();
}
const errLookupdataCallback = (data) => {
console.log('errLookupdataCallback', data);
storeSet.main.unsetAsyncRequest();
} }
useImperativeHandle( useImperativeHandle(
@@ -65,12 +122,26 @@ const BandoEditFormStep1 = forwardRef(function ({ initialData, getFormErrors },
}, [errors, isValid]); }, [errors, isValid]);
useEffect(() => { useEffect(() => {
setFormInitialData(initialData);
}, [initialData]);
useEffect(() => {
reset(formInitialData);
}, [formInitialData]);
useEffect(() => {
if (isAsyncRequest !== 0) {
return;
}
trigger().then(() => clearErrors()); trigger().then(() => clearErrors());
//storeSet.main.setAsyncRequest();
LookupdataService.getItems(lookupdataCallback, errLookupdataCallback, [['type', ['AIMED_TO', 'FAQ']]])
}, []); }, []);
return ( return (
<form className="appForm" onSubmit={handleSubmit(onSubmit)}> <form className="appForm" onSubmit={handleSubmit(onSubmit)}>
<UnsavedChangesDetector initialData={values} getValuesFn={getValues}/> <UnsavedChangesDetector initialData={formInitialData} getValuesFn={getValues}/>
<FormField <FormField
type="switch" type="switch"
fieldName="confidi" fieldName="confidi"
@@ -123,7 +194,7 @@ const BandoEditFormStep1 = forwardRef(function ({ initialData, getFormErrors },
data={values} data={values}
setDataFn={setValue} setDataFn={setValue}
fieldName="aimedTo" fieldName="aimedTo"
options={exampleOfAimedToOptions} options={aimedToOptions}
errors={errors} errors={errors}
register={register} register={register}
trigger={trigger} trigger={trigger}
@@ -191,7 +262,7 @@ const BandoEditFormStep1 = forwardRef(function ({ initialData, getFormErrors },
data={values} data={values}
setDataFn={setValue} setDataFn={setValue}
fieldName="faq" fieldName="faq"
options={exampleOfFaqOptions} options={faqOptions}
errors={errors} errors={errors}
register={register} register={register}
trigger={trigger} trigger={trigger}
@@ -210,6 +281,7 @@ const BandoEditFormStep1 = forwardRef(function ({ initialData, getFormErrors },
</div> </div>
<BandoEditFormActions <BandoEditFormActions
id={values.id}
openPreview={openPreview} openPreview={openPreview}
openPreviewEvaluation={openPreviewEvaluation}/> openPreviewEvaluation={openPreviewEvaluation}/>
</form> </form>

View File

@@ -1,4 +1,4 @@
import React, { forwardRef, useEffect, useImperativeHandle } from 'react'; import React, { forwardRef, useEffect, useImperativeHandle, useMemo, useState } from 'react';
import { __ } from '@wordpress/i18n'; import { __ } from '@wordpress/i18n';
import { useNavigate } from 'react-router-dom'; import { useNavigate } from 'react-router-dom';
import { useForm } from 'react-hook-form'; import { useForm } from 'react-hook-form';
@@ -10,9 +10,14 @@ import FormFieldRepeater from '../../../../components/FormFieldRepeater';
import FormFieldRepeaterCriteria from '../../../../components/FormFieldRepeaterCriteria'; import FormFieldRepeaterCriteria from '../../../../components/FormFieldRepeaterCriteria';
import BandoEditFormActions from '../BandoEditFormActions'; import BandoEditFormActions from '../BandoEditFormActions';
import UnsavedChangesDetector from '../../../../components/UnsavedChangesDetector'; import UnsavedChangesDetector from '../../../../components/UnsavedChangesDetector';
import BandoService from '../../../../service/bando-service';
import LookupdataService from '../../../../service/lookupdata-service';
const BandoEditFormStep2 = forwardRef(function ({ initialData, getFormErrors }, ref) { const BandoEditFormStep2 = forwardRef(function ({ initialData, getFormErrors }, ref) {
const navigate = useNavigate(); const navigate = useNavigate();
const [criteriaOptions, setCriteriaOptions] = useState([]);
const [checklistOptions, setChecklistOptions] = useState([]);
const [formInitialData, setFormInitialData] = useState(initialData);
const { const {
control, control,
handleSubmit, handleSubmit,
@@ -21,27 +26,71 @@ const BandoEditFormStep2 = forwardRef(function ({ initialData, getFormErrors },
register, register,
trigger, trigger,
getValues, getValues,
clearErrors clearErrors,
} = useForm({defaultValues: initialData, mode: 'onChange'}); reset
} = useForm({
defaultValues: useMemo(() => {
return formInitialData;
}, [formInitialData]), mode: 'onChange'
});
const values = getValues(); const values = getValues();
const step2Props = ['threshold', 'criteria', 'checkList', 'docs', 'images'];
const onSubmit = (formData) => { const onSubmit = (formData) => {
if (!isNil(formData.dates) && formData.dates.length) { if (!isNil(formData.dates) && formData.dates.length) {
formData.dates = formData.dates.map(v => is(String, v) ? v : v.toISOString()); formData.dates = formData.dates.map(v => is(String, v) ? v : v.toISOString());
} }
console.log('onSubmit', formData);
const forSubmit = Object.keys(formData).reduce((acc, cur) => {
if (step2Props.includes(cur)) {
acc[cur] = formData[cur];
}
return acc;
}, {});
BandoService.updateBandoStep2(formData.id, forSubmit, createCallback, errCreateCallback);
}; };
// TODO temp data const createCallback = (data) => {
const exampleOfCriteriaOptions = [ if (data.status === 'SUCCESS') {
{ id: 15, value: 'Innovatività del progetto', score: 9 }, setFormInitialData(data.data);
{ id: 16, value: 'Impatto sulla competitività dell\'azienda', score: 3 }, reset();
{ id: 17, value: 'Sostenibilità economico-finanziaria', score: 5 } }
]; }
const exampleOfChecklistOptions = [
{ id: 15, value: 'Innovatività del progetto' } const errCreateCallback = (data) => {
]; console.log('errCreateCallback', data);
// end of temp data }
const lookupdataCallback = (data) => {
if (data.status === 'SUCCESS') {
const criteria = data.data
.filter(o => o.type === 'EVALUATION_CRITERIA')
.map(o => {
delete o.type;
return {
...o,
score: 0,
lookUpDataId: o.id
};
});
setCriteriaOptions(criteria);
const checklist = data.data
.filter(o => o.type === 'CHECKLIST')
.map(o => {
delete o.type;
return {
...o,
lookUpDataId: o.id
};
});
setChecklistOptions(checklist);
}
}
const errLookupdataCallback = (data) => {
console.log('errLookupdataCallback', data)
}
const openPreview = () => { const openPreview = () => {
navigate('/bandi/preview/11'); navigate('/bandi/preview/11');
@@ -67,8 +116,17 @@ const BandoEditFormStep2 = forwardRef(function ({ initialData, getFormErrors },
} }
}, [errors, isValid]); }, [errors, isValid]);
useEffect(() => {
setFormInitialData(initialData);
}, [initialData]);
useEffect(() => {
reset(formInitialData);
}, [formInitialData]);
useEffect(() => { useEffect(() => {
trigger().then(() => clearErrors()); trigger().then(() => clearErrors());
LookupdataService.getItems(lookupdataCallback, errLookupdataCallback, [['type', ['CHECKLIST', 'EVALUATION_CRITERIA']]])
}, []); }, []);
return ( return (
@@ -78,7 +136,7 @@ const BandoEditFormStep2 = forwardRef(function ({ initialData, getFormErrors },
data={values} data={values}
setDataFn={setValue} setDataFn={setValue}
fieldName="criteria" fieldName="criteria"
options={exampleOfCriteriaOptions} options={criteriaOptions}
errors={errors} errors={errors}
register={register} register={register}
label={<>{__('Criteri di valutazione', 'gepafin')}* label={<>{__('Criteri di valutazione', 'gepafin')}*
@@ -95,17 +153,18 @@ const BandoEditFormStep2 = forwardRef(function ({ initialData, getFormErrors },
<FormField <FormField
type="fileupload" type="fileupload"
setDataFn={setValue} setDataFn={setValue}
fieldName="documentation" fieldName="docs"
label={__('Documentazione', 'gepafin')} label={__('Documentazione', 'gepafin')}
control={control} control={control}
errors={errors} errors={errors}
defaultValue={values['documentation']} defaultValue={values['docs']}
config={{ required: __('È obbligatorio', 'gepafin') }} config={{ required: __('È obbligatorio', 'gepafin') }}
accept="application/pdf,application/vnd.ms-excel" accept="application/pdf,application/vnd.ms-excel"
chooseLabel={__('Aggiungi documento', 'gepafin')} chooseLabel={__('Aggiungi documento', 'gepafin')}
multiple={true} multiple={true}
doctype='document' doctype='document'
register={register} register={register}
callId={values.id}
/> />
<FormField <FormField
@@ -118,13 +177,14 @@ const BandoEditFormStep2 = forwardRef(function ({ initialData, getFormErrors },
defaultValue={values['images']} defaultValue={values['images']}
doctype='images' doctype='images'
register={register} register={register}
callId={values.id}
/> />
<FormFieldRepeater <FormFieldRepeater
data={values} data={values}
setDataFn={setValue} setDataFn={setValue}
fieldName="checklist" fieldName="checkList"
options={exampleOfChecklistOptions} options={checklistOptions}
errors={errors} errors={errors}
register={register} register={register}
label={<>{__('Checklist valutazione Pre-Istruttoria', 'gepafin')}* label={<>{__('Checklist valutazione Pre-Istruttoria', 'gepafin')}*

View File

@@ -1,31 +1,37 @@
import React, { useState, useEffect, useRef } from 'react'; import React, { useState, useEffect, useRef } from 'react';
import { __ } from '@wordpress/i18n'; import { __ } from '@wordpress/i18n';
import { useNavigate, useParams } from 'react-router-dom'; import { useNavigate, useParams } from 'react-router-dom';
//import equal from 'fast-deep-equal';
import { is, isNil } from 'ramda'; import { is, isNil } from 'ramda';
// components // store
import { storeSet, useStore } from '../../store';
// api
import BandoService from '../../service/bando-service';
// tools
import getBandoLabel from '../../helpers/getBandoLabel'; import getBandoLabel from '../../helpers/getBandoLabel';
// components
import { Button } from 'primereact/button'; import { Button } from 'primereact/button';
import { Skeleton } from 'primereact/skeleton'; import { Skeleton } from 'primereact/skeleton';
import { Steps } from 'primereact/steps'; import { Steps } from 'primereact/steps';
import BandoEditFormStep1 from './components/BandoEditFormStep1'; import BandoEditFormStep1 from './components/BandoEditFormStep1';
import BandoEditFormStep2 from './components/BandoEditFormStep2'; import BandoEditFormStep2 from './components/BandoEditFormStep2';
import { Messages } from 'primereact/messages'; import { Messages } from 'primereact/messages';
import FormsService from '../../service/forms-service';
// TODO temp
import { bandoTest } from '../../tempData';
const BandoEdit = () => { const BandoEdit = () => {
const isAsyncRequest = useStore().main.isAsyncRequest();
const navigate = useNavigate(); const navigate = useNavigate();
const { id } = useParams(); const { id } = useParams();
const [activeStep, setActiveStep] = useState(0) const [activeStep, setActiveStep] = useState(null)
const [data, setData] = useState({}); const [data, setData] = useState({});
const [isLoading, setIsLoading] = useState(true); const [forms, setForms] = useState([]);
//const [selectedTemplate, setSelectedTemplate] = useState(null); //const [selectedTemplate, setSelectedTemplate] = useState(null);
//const [templates, setTemplate] = useState(null); //const [templates, setTemplate] = useState(null);
const formRef = useRef(null); const formRef = useRef(null);
const stepErrorMsgs = useRef(null); const bandoMsgs = useRef(null);
const stepItems = [ const stepItems = [
{ {
@@ -34,7 +40,7 @@ const BandoEdit = () => {
if (activeStep === 0) { if (activeStep === 0) {
return false return false
} }
stepErrorMsgs.current.clear(); bandoMsgs.current.clear();
const isFormValid = formRef.current.isFormValid(); const isFormValid = formRef.current.isFormValid();
//const values = formRef.current.getValues(); //const values = formRef.current.getValues();
//const diffData = equal(values, data); //const diffData = equal(values, data);
@@ -42,10 +48,13 @@ const BandoEdit = () => {
if (isFormValid) { if (isFormValid) {
goToStep(0) goToStep(0)
} else { } else {
stepErrorMsgs.current.show([ bandoMsgs.current.show([
{ sticky: true, severity: 'error', summary: 'Error', {
id: '98',
sticky: true, severity: 'error', summary: 'Error',
detail: __('Potrai andare su altro step dopo risolvere errori della forma', 'gepafin'), detail: __('Potrai andare su altro step dopo risolvere errori della forma', 'gepafin'),
closable: true } closable: true
}
]); ]);
} }
} }
@@ -56,7 +65,7 @@ const BandoEdit = () => {
if (activeStep === 1) { if (activeStep === 1) {
return false return false
} }
stepErrorMsgs.current.clear(); bandoMsgs.current.clear();
const isFormValid = formRef.current.isFormValid(); const isFormValid = formRef.current.isFormValid();
//const values = formRef.current.getValues(); //const values = formRef.current.getValues();
//const diffData = equal(values, data); //const diffData = equal(values, data);
@@ -64,10 +73,13 @@ const BandoEdit = () => {
if (isFormValid) { if (isFormValid) {
goToStep(1); goToStep(1);
} else { } else {
stepErrorMsgs.current.show([ bandoMsgs.current.show([
{ sticky: true, severity: 'error', summary: 'Error', {
id: '98',
sticky: true, severity: 'error', summary: 'Error',
detail: __('Potrai andare su altro step dopo risolvere errori della forma', 'gepafin'), detail: __('Potrai andare su altro step dopo risolvere errori della forma', 'gepafin'),
closable: true } closable: true
}
]); ]);
} }
} }
@@ -79,40 +91,151 @@ const BandoEdit = () => {
} }
const openBandoFormManagement = () => { const openBandoFormManagement = () => {
navigate('/bandi/11/forms'); navigate(`/bandi/${id}/forms`);
}
const validateBando = () => {
storeSet.main.setAsyncRequest();
bandoMsgs.current.clear();
BandoService.validateBando(id, validateCallback, errValidateCallback);
}
const validateCallback = (data) => {
if (data.status === 'SUCCESS') {
if (bandoMsgs.current) {
bandoMsgs.current.show([
{
id: '99',
sticky: true, severity: 'success', summary: '',
detail: __('Potrai pubblicare il tuo Bando.', 'gepafin'),
closable: false
}
]);
}
}
storeSet.main.unsetAsyncRequest();
}
const errValidateCallback = (data) => {
standardErrCallback(data);
}
const publishBando = () => {
storeSet.main.setAsyncRequest();
bandoMsgs.current.clear();
BandoService.updateBandoStatus(id, publishCallback, errPublishCallback, [['status', 'PUBLISH']]);
}
const publishCallback = (data) => {
if (data.status === 'SUCCESS') {
if (bandoMsgs.current) {
bandoMsgs.current.show([
{
id: '99',
sticky: true, severity: 'success', summary: '',
detail: __('Pubblicato!', 'gepafin'),
closable: false
}
]);
}
}
storeSet.main.unsetAsyncRequest();
}
const errPublishCallback = (data) => {
standardErrCallback(data);
}
const getCallback = (data) => {
if (data.status === 'SUCCESS') {
if (!isNil(data.data.dates) && data.data.dates.length) {
data.data.dates = data.data.dates.map(v => is(String, v) ? new Date(v) : v);
}
if (data.data.status === 'READY_TO_PUBLISH') {
bandoMsgs.current.clear();
bandoMsgs.current.show([
{
id: '1',
sticky: true, severity: 'info', summary: 'Info',
detail: __('Potrai pubblicare il tuo Bando.', 'gepafin'),
closable: false
}
]);
} else if (data.data.status === 'DRAFT') {
if (bandoMsgs.current) {
bandoMsgs.current.clear();
bandoMsgs.current.show([
{
id: '1',
sticky: true, severity: 'info', summary: 'Info',
detail: __('Potrai pubblicare il tuo Bando solo dopo aver completato tutti i campi obbligatori contrassegnati dagli asterischi.', 'gepafin'),
closable: false
}
]);
}
}
setData(data.data);
}
storeSet.main.unsetAsyncRequest();
}
const errGetCallback = (data) => {
standardErrCallback(data);
}
const standardErrCallback = (data) => {
if (bandoMsgs.current && data.message) {
bandoMsgs.current.show([
{
sticky: true, severity: 'error', summary: 'Error',
detail: data.message,
closable: true
}
]);
}
storeSet.main.unsetAsyncRequest();
}
const getFormsCallback = (data) => {
if (data.status === 'SUCCESS') {
setForms(data.data);
}
storeSet.main.unsetAsyncRequest();
} }
useEffect(() => { useEffect(() => {
storeSet.main.setAsyncRequest();
setActiveStep(0);
const parsed = parseInt(id) const parsed = parseInt(id)
const bandoId = !isNaN(parsed) ? parsed : 0; const bandoId = !isNaN(parsed) ? parsed : 0;
const data = 0 === bandoId if (bandoId === 0) {
? { setData({
status: 'draft', status: 'draft',
name: '' name: ''
} });
: bandoTest; storeSet.main.unsetAsyncRequest();
if (bandoId === 0) { if (bandoMsgs.current) {
setData(data); bandoMsgs.current.clear();
setIsLoading(false); bandoMsgs.current.show([
} else { {
setTimeout(() => { id: '1',
if (!isNil(data.dates) && data.dates.length) { sticky: true, severity: 'info', summary: 'Info',
data.dates = data.dates.map(v => is(String, v) ? new Date(v) : v); detail: __('Potrai pubblicare il tuo Bando solo dopo aver completato tutti i campi obbligatori contrassegnati dagli asterischi.', 'gepafin'),
closable: false
} }
]);
setData(data);
/*const templates = [
{ name: 'Il mio template', value: 22 },
{ name: 'Template #11', value: 11 },
];
setTemplate(templates);*/
setIsLoading(false);
}, 2000);
} }
}, [id]); } else {
BandoService.getBando(id, getCallback, errGetCallback);
FormsService.getFormsForCall(id, getFormsCallback, () => {
});
}
}, []);
return ( return (
<div className="appPage"> <div className="appPage">
@@ -146,7 +269,7 @@ const BandoEdit = () => {
icon="pi pi-check" icon="pi pi-check"
iconPos="right"/> iconPos="right"/>
</div> : null*/} </div> : null*/}
{!isLoading {!isAsyncRequest
? <Steps ? <Steps
model={stepItems} model={stepItems}
activeIndex={activeStep} activeIndex={activeStep}
@@ -155,13 +278,14 @@ const BandoEdit = () => {
<div className="appPage__spacer"></div> <div className="appPage__spacer"></div>
<Messages ref={stepErrorMsgs} /> <Messages ref={bandoMsgs}/>
{!isLoading {!isAsyncRequest
? <> ? <>
{activeStep === 0 {activeStep === 0
? <BandoEditFormStep1 initialData={data} ref={formRef}/> ? <BandoEditFormStep1 initialData={data} ref={formRef}/> : null}
: <BandoEditFormStep2 initialData={data} ref={formRef}/>} {activeStep === 1
? <BandoEditFormStep2 initialData={data} ref={formRef}/> : null}
<div className="appPageSection"> <div className="appPageSection">
<h2>{__('Crea o modifica il Form compilabile dal Beneficiario', 'gepafin')}</h2> <h2>{__('Crea o modifica il Form compilabile dal Beneficiario', 'gepafin')}</h2>
@@ -169,6 +293,34 @@ const BandoEdit = () => {
type="button" type="button"
onClick={openBandoFormManagement} onClick={openBandoFormManagement}
label={__('Crea/modifica form', 'gepafin')}/> label={__('Crea/modifica form', 'gepafin')}/>
{forms.length
? <ul className="">
{forms.map(o => <li key={o.id}>
{o.label}
</li>)}
</ul>
: <p>No forms created yet.</p>}
</div>
<div className="appPage__spacer"></div>
<div className="appPageSection">
<h2>{__('Publicca il Form', 'gepafin')}</h2>
<div className="appPageSection__actions">
<Button
type="button"
outlined
disabled={!(data.status === 'DRAFT')}
onClick={validateBando}
label={__('Validate', 'gepafin')}/>
<Button
type="button"
disabled={!(data.status === 'READY_TO_PUBLISH')}
onClick={publishBando}
label={__('Publish', 'gepafin')}/>
</div>
</div> </div>
</> </>
: <> : <>

View File

@@ -1,18 +1,25 @@
import React, { useState, useEffect, useRef } from 'react'; import React, { useState, useEffect } from 'react';
import { __ } from '@wordpress/i18n'; import { __ } from '@wordpress/i18n';
import { useParams, useNavigate } from 'react-router-dom'; import { useParams, useNavigate } from 'react-router-dom';
import { isEmpty } from 'ramda';
// components // components
import { Button } from 'primereact/button'; import { Button } from 'primereact/button';
import { Dropdown } from 'primereact/dropdown'; import { Dropdown } from 'primereact/dropdown';
// service
import FormsService from '../../service/forms-service';
// store
import { storeSet } from '../../store';
const BandoForms = () => { const BandoForms = () => {
const { id } = useParams(); const { id } = useParams();
const navigate = useNavigate() const navigate = useNavigate()
const [templates, setTemplate] = useState(null); const [templates, setTemplates] = useState(null);
const [selectedTemplate, setSelectedTemplate] = useState(null); const [selectedTemplate, setSelectedTemplate] = useState(null);
//const [data, setData] = useState({}); const [selectedForm, setSelectedForm] = useState(null);
//const [isLoading, setIsLoading] = useState(true); const [forms, setForms] = useState([]);
const doCreateNewForm = () => { const doCreateNewForm = () => {
navigate(`/bandi/${id}/forms/new`); navigate(`/bandi/${id}/forms/new`);
@@ -22,17 +29,35 @@ const BandoForms = () => {
navigate(`/bandi/${id}`); navigate(`/bandi/${id}`);
} }
const goToEditForm = () => {
if (selectedForm && selectedForm !== 0) {
navigate(`/bandi/${id}/forms/${selectedForm}`);
}
}
const goToEditFormFromTemplate = () => {
console.log('goToEditFormFromTemplate', selectedTemplate)
//navigate(`/bandi/${id}`);
}
const getFormsCallback = (data) => {
if (data.status === 'SUCCESS') {
const forms = data.data.map(o => ({label: o.label, value: o.id}))
setForms(forms);
}
storeSet.main.unsetAsyncRequest();
}
useEffect(() => { useEffect(() => {
const parsed = parseInt(id) const parsed = parseInt(id)
const bandoId = !isNaN(parsed) ? parsed : 0; const bandoId = !isNaN(parsed) ? parsed : 0;
const templates = [ setTemplates([
{ name: 'Il mio template', value: 22 }, {label: "Form template", value: 11}
{ name: 'Template #11', value: 11 }, ])
];
setTemplate(templates);
// TODO storeSet.main.setAsyncRequest();
FormsService.getFormsForCall(bandoId, getFormsCallback, () => {});
}, [id]); }, [id]);
return ( return (
@@ -66,8 +91,15 @@ const BandoForms = () => {
value={selectedTemplate} value={selectedTemplate}
onChange={(e) => setSelectedTemplate(e.value)} onChange={(e) => setSelectedTemplate(e.value)}
options={templates} options={templates}
optionLabel="name" optionLabel="label"
placeholder={__('Seleziona template', 'gepafin')}/> placeholder={__('Seleziona template', 'gepafin')}/>
<Button
type="button"
outlined
disabled={true}
onClick={goToEditFormFromTemplate}
label={__('Crea', 'gepafin')}
icon="pi pi-plus" iconPos="right"/>
</div> </div>
</div> </div>
@@ -82,15 +114,25 @@ const BandoForms = () => {
</div> </div>
</div> </div>
<div className="appPageSection__withBorder disabled"> <div className="appPageSection__withBorder">
<h2>{__('Modifica Form esistente', 'gepafin')}</h2> <h2>{__('Modifica form esistente', 'gepafin')}</h2>
<div className="row"> <div className="row">
<p>{__('Continua a lavorare su un form precedentemente salvato', 'gepafin')}</p> <p>{__('Continua a lavorare su un form precedentemente salvato', 'gepafin')}</p>
<Dropdown
id="form"
disabled={isEmpty(forms)}
value={selectedForm}
onChange={(e) => setSelectedForm(e.value)}
options={forms}
optionLabel="label"
placeholder={__('Seleziona form', 'gepafin')}/>
<Button <Button
type="button" type="button"
disabled={true} outlined
onClick={doCreateNewForm} disabled={!selectedForm}
label={__('Modifica', 'gepafin')}/> onClick={goToEditForm}
label={__('Modifica', 'gepafin')}
icon="pi pi-cog" iconPos="right"/>
</div> </div>
</div> </div>
</div> </div>

View File

@@ -0,0 +1,64 @@
import React, { useEffect, useState } from 'react';
import { __ } from '@wordpress/i18n';
// components
import { InputText } from 'primereact/inputtext';
import { Button } from 'primereact/button';
// tools
import uniqid from '../../../../../../helpers/uniqid';
const ElementSettingRepeater = ({
value,
name,
setDataFn
}) => {
const [stateFieldData, setStateFieldData] = useState([]);
const removeItem = (index) => {
const newData = stateFieldData.toSpliced(index, 1);
setStateFieldData(newData);
}
const addNewItem = () => {
setStateFieldData([...stateFieldData, { name: uniqid('o'), label: '' }]);
}
const onInputChange = (e, index) => {
const { value } = e.target;
const newData = stateFieldData.map((o, i) => {
if (i === index) {
o.label = value;
}
return o;
})
setStateFieldData(newData);
}
const properField = (item, i) => {
return <InputText value={item.label} onInput={(e) => onInputChange(e, i)}/>
}
useEffect(() => {
const storeFieldData = value ?? [];
setStateFieldData(storeFieldData);
}, []);
useEffect(() => {
setDataFn(name, [...stateFieldData]);
}, [stateFieldData])
return (
<div className="formElementSettings__repeater">
{stateFieldData.map((o, i) => <div key={i} className="formElementSettings__repeaterItem">
<div className="p-inputgroup flex-1">
{properField(o, i)}
<Button icon="pi pi-times" className="p-button-danger" onClick={() => removeItem(i)}/>
</div>
</div>)}
<Button type="button" label={__('Aggiungi', 'gepafin')} onClick={addNewItem}/>
</div>
)
}
export default ElementSettingRepeater;

View File

@@ -1,6 +1,6 @@
import React, { useEffect, useState } from 'react'; import React, { useEffect, useState } from 'react';
import { head } from 'ramda'; import { head, isNil } from 'ramda';
import { __ } from '@wordpress/i18n'; import { __, sprintf } from '@wordpress/i18n';
import { klona } from 'klona'; import { klona } from 'klona';
// store // store
@@ -10,12 +10,17 @@ import { storeSet, useStore } from '../../../../store';
import { InputText } from 'primereact/inputtext'; import { InputText } from 'primereact/inputtext';
import { Button } from 'primereact/button'; import { Button } from 'primereact/button';
import { Tag } from 'primereact/tag'; import { Tag } from 'primereact/tag';
import { TabView, TabPanel } from 'primereact/tabview';
import { InputSwitch } from 'primereact/inputswitch';
import ElementSettingRepeater from './components/ElementSettingRepeater';
const BuilderElementSettings = () => { const BuilderElementSettings = ({ closeSettings }) => {
const elements = useStore().main.elements(); const elements = useStore().main.formElements();
const activeElement = useStore().main.activeElement(); const activeElement = useStore().main.activeElement();
const [activeElementData, setActiveElementData] = useState({}); const [activeElementData, setActiveElementData] = useState({});
const [settings, setSettings] = useState([]); const [settings, setSettings] = useState([]);
const [validators, setValidators] = useState({});
const textBasedValidatorFields = ['min', 'max', 'minLength', 'maxLength', 'pattern', 'custom'];
const onChange = (value, name) => { const onChange = (value, name) => {
const newSettings = settings const newSettings = settings
@@ -30,10 +35,47 @@ const BuilderElementSettings = () => {
setSettings(newSettings); setSettings(newSettings);
} }
const onUpdateOptions = (name, value) => {
const newSettings = settings
.map(o => {
if (o.name === name) {
o.value = value;
}
return o;
});
setSettings(newSettings);
}
const saveSettings = () => { const saveSettings = () => {
activeElementData.settings = settings; activeElementData.settings = settings;
activeElementData.validators = validators;
const newElements = elements.map(o => o.id === activeElementData.id ? activeElementData : o); const newElements = elements.map(o => o.id === activeElementData.id ? activeElementData : o);
storeSet.main.elements(newElements); storeSet.main.formElements(newElements);
closeSettings();
}
const showField = (value, key) => {
let newValidators = klona(validators);
if (value) {
newValidators[key] = '';
} else {
newValidators[key] = null;
}
setValidators(newValidators);
}
const toggleRequired = (value, key) => {
let newValidators = klona(validators);
newValidators[key] = value;
setValidators(newValidators);
}
const onChangeValidator = (value, name) => {
let newValidators = klona(validators);
newValidators[name] = value;
setValidators(newValidators);
} }
useEffect(() => { useEffect(() => {
@@ -41,22 +83,59 @@ const BuilderElementSettings = () => {
if (chosen) { if (chosen) {
setActiveElementData(klona(chosen)); setActiveElementData(klona(chosen));
setSettings(klona(chosen.settings)); setSettings(klona(chosen.settings));
setValidators(klona(chosen.validators))
} else { } else {
setActiveElementData({}); setActiveElementData({});
setSettings([]); setSettings([]);
setValidators({})
} }
}, [activeElement]) }, [activeElement])
return (activeElementData return (activeElementData
? <div className="formElementSettings"> ? <div className="formElementSettings">
<Tag value={activeElementData.name} severity="info"/> <Tag value={activeElementData.name} severity="info"/>
{settings <TabView className="formElementSettings__tabs">
? settings.map((o) => <div className="formElementSettings__field" key={o.name}> <TabPanel header={__('Presentation', 'gepafin')}>
<label htmlFor={o.name}>{o.name}</label> {settings
<InputText id={o.name} aria-describedby={`${o.name}-help`} ? settings.map((o) => <div className="formElementSettings__field" key={o.name}>
value={o.value} <label htmlFor={o.name}>{o.name}</label>
onChange={(e) => onChange(e.target.value, o.name)}/> {o.name === 'options'
</div>) : null} ? <ElementSettingRepeater value={o.value} name={o.name} setDataFn={onUpdateOptions} />
: <InputText id={o.name} aria-describedby={`${o.name}-help`}
value={o.value}
onChange={(e) => onChange(e.target.value, o.name)}/>}
</div>) : null}
</TabPanel>
<TabPanel header={__('Validation', 'gepafin')}>
{validators
? Object.keys(validators).map((k) => <div
className="formElementSettings__field" key={k}>
{k === 'isRequired'
? <div className="formElementSettings__field">
<label htmlFor={k}>{__('Required?', 'gepafin')}</label>
<InputSwitch
checked={validators[k]}
onChange={(e) => toggleRequired(e.value, k)}/>
</div>
: null}
{textBasedValidatorFields.includes(k)
? <div className="formElementSettings__field">
<label htmlFor={`enable_${k}`}>{sprintf(__('Set %s', 'gepafin'), k)}</label>
<InputSwitch
checked={!isNil(validators[k])}
onChange={(e) => showField(e.value, k)}/>
</div>
: null}
{textBasedValidatorFields.includes(k) && !isNil(validators[k])
? <div className="formElementSettings__field">
<label htmlFor={k}>{k}</label>
<InputText id={k} aria-describedby={`${k}-help`}
value={validators[k]}
onChange={(e) => onChangeValidator(e.target.value, k)}/>
</div> : null}
</div>) : null}
</TabPanel>
</TabView>
<Button label={__('Salva', 'gepafin')} onClick={saveSettings}/> <Button label={__('Salva', 'gepafin')} onClick={saveSettings}/>
</div> </div>

View File

@@ -48,7 +48,7 @@ const FormBuilder = () => {
<> <>
<Sidebar visible={!isEmpty(activeElement)} onHide={closeSettings} className="formBuilder__elementSettings"> <Sidebar visible={!isEmpty(activeElement)} onHide={closeSettings} className="formBuilder__elementSettings">
<h2>{__('Impostazioni del campo modulo', 'gepafin')}</h2> <h2>{__('Impostazioni del campo modulo', 'gepafin')}</h2>
{!isEmpty(activeElement) ? <BuilderElementSettings/> : null} {!isEmpty(activeElement) ? <BuilderElementSettings closeSettings={closeSettings}/> : null}
</Sidebar> </Sidebar>
<div className="formBuilder"> <div className="formBuilder">
<div className="formBuilder__main"> <div className="formBuilder__main">

View File

@@ -6,52 +6,121 @@ import { HTML5Backend } from 'react-dnd-html5-backend';
import { klona } from 'klona'; import { klona } from 'klona';
// store // store
import { storeSet, storeGet } from '../../store'; import { storeSet, storeGet, useStore } from '../../store';
// components // components
import FormBuilder from './components/FormBuilder'; import FormBuilder from './components/FormBuilder';
import { Button } from 'primereact/button'; import { Button } from 'primereact/button';
// TODO temp // components
import { formData, elementItems } from '../../tempData';
import { InputText } from 'primereact/inputtext'; import { InputText } from 'primereact/inputtext';
import FormsService from '../../service/forms-service';
const BandoFormsEdit = () => { const BandoFormsEdit = () => {
const { id, formId } = useParams(); const { id, formId } = useParams();
const navigate = useNavigate(); const navigate = useNavigate();
const [isLoading, setIsLoading] = useState(true);
const [formName, setFormName] = useState(''); const [formName, setFormName] = useState('');
const isAsyncRequest = useStore().main.isAsyncRequest();
const getBandoId = () => {
const parsed = parseInt(id)
return !isNaN(parsed) ? parsed : 0;
}
const goBack = () => { const goBack = () => {
navigate(`/bandi/${id}/forms`); const bandoId = getBandoId();
navigate(`/bandi/${bandoId}/forms`);
} }
const doSave = () => { const doSave = () => {
console.log('doSave', storeGet.main.formElements()); const content = storeGet.main.formElements();
const bandoId = getBandoId();
const parsedFormId = parseInt(formId)
const bandoFormId = !isNaN(parsedFormId) ? parsedFormId : 0;
const formData = {
label: formName,
content
}
storeSet.main.setAsyncRequest();
if (bandoFormId === 0) {
FormsService.createFormForCall(bandoId, formData, formCreateCallback, errFormCreateCallback);
} else {
FormsService.updateForm(bandoFormId, formData, formCreateCallback, errFormCreateCallback)
}
}
const formCreateCallback = (data) => {
if (data.status === 'SUCCESS') {
const parsed = parseInt(id)
const bandoId = !isNaN(parsed) ? parsed : 0;
if (data.data.id) {
navigate(`/bandi/${bandoId}/forms/${data.data.id}`);
}
}
storeSet.main.unsetAsyncRequest();
}
const errFormCreateCallback = (data) => {
console.log('errFormCreateCallback', data)
storeSet.main.unsetAsyncRequest();
} }
const openPreview = () => { const openPreview = () => {
navigate(`/bandi/${id}/forms/${formId}/preview`); const bandoId = getBandoId();
navigate(`/bandi/${bandoId}/forms/${formId}/preview`);
} }
const doPublish = () => { const doPublish = () => {
console.log('doPublish'); console.log('doPublish');
} }
const getElementItemsCallback = (data) => {
if (data.status === 'SUCCESS') {
storeSet.main.elementItems(data.data);
}
storeSet.main.unsetAsyncRequest();
}
const errGetElementItemsCallbacks = (data) => {
console.log('errGetElementItemsCallbacks', data)
storeSet.main.unsetAsyncRequest();
}
const getFormCallback = (data) => {
if (data.status === 'SUCCESS') {
storeSet.main.formId(data.data.id);
storeSet.main.formLabel(data.data.label);
setFormName(data.data.label);
const elements = klona(data.data.content);
storeSet.main.formElements(elements);
}
storeSet.main.unsetAsyncRequest();
}
const errGetFormCallbacks = (data) => {
console.log('errGetElementItemsCallbacks', data)
storeSet.main.unsetAsyncRequest();
}
useEffect(() => { useEffect(() => {
//const parsed = parseInt(id) const parsedFormId = parseInt(formId)
//const bandoId = !isNaN(parsed) ? parsed : 0; const bandoFormId = !isNaN(parsedFormId) ? parsedFormId : 0;
//const parsedFormId = parseInt(formId)
//const bandoFormId = !isNaN(parsedFormId) ? parsedFormId : 0;
// 1. TODO get builder content data from API storeSet.main.setAsyncRequest();
storeSet.main.formLabel(formData.label); FormsService.getElementItems(getElementItemsCallback, errGetElementItemsCallbacks);
const elements = klona(formData.content);
storeSet.main.formElements(elements);
storeSet.main.elementItems(elementItems);
setIsLoading(false);
}, [id]); if (bandoFormId) {
storeSet.main.setAsyncRequest();
FormsService.getFormById(bandoFormId, getFormCallback, errGetFormCallbacks);
}
return () => {
storeSet.main.formId(0);
storeSet.main.formLabel('');
storeSet.main.formElements([]);
}
}, [id, formId]);
return ( return (
<div className="appPage"> <div className="appPage">
@@ -65,11 +134,11 @@ const BandoFormsEdit = () => {
<div className="appPage__spacer"></div> <div className="appPage__spacer"></div>
<div className="appForm__field"> <div className="appForm__field">
<label htmlFor="label">{__('Form label', 'gepafin')}</label> <label htmlFor="label">{__('Assegna un nome a questo form', 'gepafin')}</label>
<InputText <InputText
id="label" id="label"
value={formName} value={formName}
placeholder={__('Nome della forma', 'gepafin')} placeholder=""
onChange={(e) => setFormName(e.target.value)} onChange={(e) => setFormName(e.target.value)}
aria-describedby="label-help"/> aria-describedby="label-help"/>
</div> </div>
@@ -78,7 +147,7 @@ const BandoFormsEdit = () => {
<div className="appPageSection"> <div className="appPageSection">
<DndProvider backend={HTML5Backend}> <DndProvider backend={HTML5Backend}>
{!isLoading ? <FormBuilder/> : null} {!isAsyncRequest ? <FormBuilder/> : null}
</DndProvider> </DndProvider>
</div> </div>

View File

@@ -1,7 +1,10 @@
import React, { useState, useEffect } from 'react'; import React, { useState, useEffect, useRef } from 'react';
import { __, sprintf } from '@wordpress/i18n'; import { __, sprintf } from '@wordpress/i18n';
import { useNavigate, useParams } from 'react-router-dom'; import { useNavigate, useParams } from 'react-router-dom';
import { bandoTest } from '../../tempData'; import { is, isEmpty } from 'ramda';
// store
import { storeSet, useStore } from '../../store';
// tools // tools
import getNumberWithCurrency from '../../helpers/getNumberWithCurrency'; import getNumberWithCurrency from '../../helpers/getNumberWithCurrency';
@@ -13,13 +16,16 @@ import { Accordion } from 'primereact/accordion';
import { AccordionTab } from 'primereact/accordion'; import { AccordionTab } from 'primereact/accordion';
import { InputTextarea } from 'primereact/inputtextarea'; import { InputTextarea } from 'primereact/inputtextarea';
import { Button } from 'primereact/button'; import { Button } from 'primereact/button';
import BandoService from '../../service/bando-service';
import { Messages } from 'primereact/messages';
const BandoView = () => { const BandoView = () => {
const isAsyncRequest = useStore().main.isAsyncRequest();
const { id } = useParams(); const { id } = useParams();
const navigate = useNavigate(); const navigate = useNavigate();
const [data, setData] = useState({}); const [data, setData] = useState({});
const [newQuestion, setNewQuestion] = useState(''); const [newQuestion, setNewQuestion] = useState('');
const [isLoading, setIsLoading] = useState(true); const bandoMsgs = useRef(null);
const closePreview = () => { const closePreview = () => {
navigate(`/bandi/${id}`); navigate(`/bandi/${id}`);
@@ -41,20 +47,41 @@ const BandoView = () => {
} }
useEffect(() => { const getCallback = (data) => {
//const parsed = parseInt(id) if (data.status === 'SUCCESS') {
//const bandoId = !isNaN(parsed) ? parsed : 0; setData(getFormattedBandiData(data.data));
}
storeSet.main.unsetAsyncRequest();
}
setTimeout(() => { const errGetCallback = (data) => {
const data = bandoTest; if (bandoMsgs.current && data.message) {
setData(data); bandoMsgs.current.show([
setIsLoading(false) {
}, 3000); sticky: true, severity: 'error', summary: 'Error',
detail: data.message,
closable: true
}
]);
}
storeSet.main.unsetAsyncRequest();
}
const getFormattedBandiData = (data) => {
data.dates = data.dates.map(v => is(String, v) ? new Date(v) : (v ? v : ''));
return data;
};
useEffect(() => {
const parsed = parseInt(id)
const bandoId = !isNaN(parsed) ? parsed : 0;
BandoService.getBando(bandoId, getCallback, errGetCallback);
}, [id]); }, [id]);
return ( return (
<div className="appPage"> <div className="appPage">
{!isLoading {!isAsyncRequest && !isEmpty(data)
? <div className="appPage__pageHeader"> ? <div className="appPage__pageHeader">
<h1>{data.name}</h1> <h1>{data.name}</h1>
<p> <p>
@@ -68,8 +95,9 @@ const BandoView = () => {
</>} </>}
<div className="appPage__spacer"></div> <div className="appPage__spacer"></div>
<Messages ref={bandoMsgs}/>
{!isLoading {!isAsyncRequest && !isEmpty(data)
? <div className="appPage__content"> ? <div className="appPage__content">
<div className="appPageSection__preview"> <div className="appPageSection__preview">
<Button <Button
@@ -107,11 +135,11 @@ const BandoView = () => {
<div className="appPageSection__withBorder"> <div className="appPageSection__withBorder">
<p className="appPageSection__pMeta"> <p className="appPageSection__pMeta">
<span>{__('Data apertura', 'gepafin')}</span> <span>{__('Data apertura', 'gepafin')}</span>
<span></span> <span>{getDateFromISOstring(data.dates[0])}</span>
</p> </p>
<p className="appPageSection__pMeta"> <p className="appPageSection__pMeta">
<span>{__('Data chiusura', 'gepafin')}</span> <span>{__('Data chiusura', 'gepafin')}</span>
<span></span> <span>{getDateFromISOstring(data.dates[1])}</span>
</p> </p>
</div> </div>
</div> </div>
@@ -156,7 +184,7 @@ const BandoView = () => {
<h2>{__('Allegati', 'gepafin')}</h2> <h2>{__('Allegati', 'gepafin')}</h2>
<div className="row rowContent"> <div className="row rowContent">
<ul> <ul>
{data.documentation.map((o, i) => <li key={i}> {data.docs.map((o, i) => <li key={i}>
<a href={o.filePath} target="_blank" rel="noreferrer">{o.name}</a> <a href={o.filePath} target="_blank" rel="noreferrer">{o.name}</a>
</li>)} </li>)}
</ul> </ul>
@@ -168,7 +196,7 @@ const BandoView = () => {
<Accordion> <Accordion>
{data.faq.map((o, i) => <AccordionTab key={i} header={o.question}> {data.faq.map((o, i) => <AccordionTab key={i} header={o.question}>
<p> <p>
{o.answer} {o.response}
</p> </p>
</AccordionTab>)} </AccordionTab>)}
</Accordion> </Accordion>
@@ -196,12 +224,14 @@ const BandoView = () => {
<div className="appPageSection__actions"> <div className="appPageSection__actions">
<Button <Button
type="button" type="button"
disabled={true}
outlined outlined
onClick={scaricaBando} onClick={scaricaBando}
label={__('Scarica Bando Completo', 'gepafin')} label={__('Scarica Bando Completo', 'gepafin')}
icon="pi pi-download" iconPos="right"/> icon="pi pi-download" iconPos="right"/>
<Button <Button
type="button" type="button"
disabled={true}
outlined outlined
onClick={scaricaModulistica} onClick={scaricaModulistica}
label={__('Scarica Modulistica', 'gepafin')} label={__('Scarica Modulistica', 'gepafin')}

View File

@@ -0,0 +1,176 @@
import React, { useState, useEffect} from 'react';
import { __ } from '@wordpress/i18n';
import { uniq } from 'ramda';
// tools
import getBandoLabel from '../../../../helpers/getBandoLabel';
import getBandoSeverity from '../../../../helpers/getBandoSeverity';
// components
import { FilterMatchMode, FilterOperator } from 'primereact/api';
import { DataTable } from 'primereact/datatable';
import { Column } from 'primereact/column';
import { InputText } from 'primereact/inputtext';
import { IconField } from 'primereact/iconfield';
import { InputIcon } from 'primereact/inputicon';
import { Dropdown } from 'primereact/dropdown';
import { InputNumber } from 'primereact/inputnumber';
import { Button } from 'primereact/button';
import { Calendar } from 'primereact/calendar';
import { Tag } from 'primereact/tag';
import ProperBandoLabel from '../../../../components/ProperBandoLabel';
const LatestBandiTable = () => {
const [items, setItems] = useState(null);
const [filters, setFilters] = useState(null);
const [loading, setLoading] = useState(false);
const [globalFilterValue, setGlobalFilterValue] = useState('');
const [statuses, setStatuses] = useState([]);
useEffect(() => {
// TODO
const items = [
{
name: 'Bando Innovazione 2024',
start_date: '2024-08-08T00:00:00+00:00',
end_date: '2024-08-30T00:00:00+00:00',
submissions: 24,
status: 'publish',
id: 11
},
{
name: 'Bando Sostenibilità 2024',
start_date: '2024-07-28T00:00:00+00:00',
end_date: '2024-08-15T00:00:00+00:00',
submissions: 35,
status: 'publish',
id: 9
},
{
name: 'Bando A',
start_date: '2024-06-28T00:00:00+00:00',
end_date: '2024-06-15T00:00:00+00:00',
submissions: 2,
status: 'closed',
id: 2
}
]
setItems(getFormattedBandiData(items));
setStatuses(uniq(items.map(o => o.status)))
setLoading(false);
initFilters();
}, []);
const getFormattedBandiData = (data) => {
return [...(data || [])].map((d) => {
d.start_date = new Date(d.start_date);
d.end_date = new Date(d.end_date);
return d;
});
};
const formatDate = (value) => {
return value.toLocaleDateString('it-IT', {
day: '2-digit',
month: '2-digit',
year: 'numeric'
});
};
const clearFilter = () => {
initFilters();
};
const onGlobalFilterChange = (e) => {
const value = e.target.value;
let _filters = { ...filters };
_filters['global'].value = value;
setFilters(_filters);
setGlobalFilterValue(value);
};
const initFilters = () => {
setFilters({
global: { value: null, matchMode: FilterMatchMode.CONTAINS },
name: { operator: FilterOperator.AND, constraints: [{ value: null, matchMode: FilterMatchMode.STARTS_WITH }] },
start_date: { operator: FilterOperator.AND, constraints: [{ value: null, matchMode: FilterMatchMode.DATE_IS }] },
end_date: { operator: FilterOperator.AND, constraints: [{ value: null, matchMode: FilterMatchMode.DATE_IS }] },
submissions: { operator: FilterOperator.AND, constraints: [{ value: null, matchMode: FilterMatchMode.EQUALS }] },
status: { operator: FilterOperator.OR, constraints: [{ value: null, matchMode: FilterMatchMode.EQUALS }] },
});
setGlobalFilterValue('');
};
const renderHeader = () => {
return (
<div className="appTableHeader">
<Button type="button" icon="pi pi-filter-slash" label={__('Pulisci', 'gepafin')} outlined onClick={clearFilter} />
<IconField iconPosition="left">
<InputIcon className="pi pi-search" />
<InputText value={globalFilterValue} onChange={onGlobalFilterChange} placeholder={__('Cerca', 'gepafin')} />
</IconField>
</div>
);
};
const dateStartBodyTemplate = (rowData) => {
return formatDate(rowData.start_date);
};
const dateEndBodyTemplate = (rowData) => {
return formatDate(rowData.end_date);
};
const dateFilterTemplate = (options) => {
return <Calendar value={options.value} onChange={(e) => options.filterCallback(e.value, options.index)} dateFormat="mm/dd/yy" placeholder="mm/dd/yyyy" mask="99/99/9999" />;
};
const balanceFilterTemplate = (options) => {
return <InputNumber value={options.value} onChange={(e) => options.filterCallback(e.value, options.index)} />;
};
const statusBodyTemplate = (rowData) => {
return <ProperBandoLabel status={rowData.status}/>;
};
const statusFilterTemplate = (options) => {
return <Dropdown value={options.value} options={statuses} onChange={(e) => options.filterCallback(e.value, options.index)} itemTemplate={statusItemTemplate} placeholder="Select One" className="p-column-filter" showClear />;
};
const statusItemTemplate = (option) => {
return <Tag value={getBandoLabel(option)} severity={getBandoSeverity(option)} />;
};
const header = renderHeader();
return(
<div className="appPageSection__table">
<DataTable value={items} paginator showGridlines rows={10} loading={loading} dataKey="id"
filters={filters}
globalFilterFields={['name', 'status']}
header={header}
emptyMessage="Nothing found." onFilter={(e) => setFilters(e.filters)}>
<Column field="name" header={__('Nome Bando', 'gepafin')} filter filterPlaceholder="Search by name"
style={{ minWidth: '12rem' }}/>
<Column header={__('Data Pubblicazione', 'gepafin')} filterField="start_date" dataType="date"
style={{ minWidth: '10rem' }}
body={dateStartBodyTemplate} filter filterElement={dateFilterTemplate}/>
<Column header={__('Data Scadenza', 'gepafin')} filterField="end_date" dataType="date"
style={{ minWidth: '10rem' }}
body={dateEndBodyTemplate} filter filterElement={dateFilterTemplate}/>
<Column header={__('Domande ricevute', 'gepafin')} filterField="submissions" dataType="numeric"
style={{ minWidth: '10rem' }} field="submissions"
filter filterElement={balanceFilterTemplate}/>
<Column field="status" header={__('Stato', 'gepafin')} filterMenuStyle={{ width: '14rem' }}
style={{ minWidth: '12rem' }} body={statusBodyTemplate} filter
filterElement={statusFilterTemplate}/>
</DataTable>
</div>
)
}
export default LatestBandiTable;

View File

@@ -0,0 +1,123 @@
import React, { useState, useEffect} from 'react';
import { __ } from '@wordpress/i18n';
// components
import { FilterMatchMode, FilterOperator } from 'primereact/api';
import { DataTable } from 'primereact/datatable';
import { Column } from 'primereact/column';
import { InputText } from 'primereact/inputtext';
import { IconField } from 'primereact/iconfield';
import { InputIcon } from 'primereact/inputicon';
import { Button } from 'primereact/button';
import { Calendar } from 'primereact/calendar';
const LatestUsersActivityTable = () => {
const [items, setItems] = useState(null);
const [filters, setFilters] = useState(null);
const [loading, setLoading] = useState(false);
const [globalFilterValue, setGlobalFilterValue] = useState('');
useEffect(() => {
// TODO
const bandi = [
{
date: '2024-08-02T00:00:00+14:32',
email: 'mario.rossi@gepafin.it',
action: 'Valutazione Domanda',
details: 'Bando Innovazione 2024 - Domanda #123',
id: 11
},
{
date: '2024-08-01T00:00:00+08:23',
email: 'laura.bianchi@gepafin.it',
action: 'Creazione Bando',
details: 'Nuovo bando "Formazione 2025" in bozza',
id: 9
}
]
setItems(getFormattedBandiData(bandi));
setLoading(false);
initFilters();
}, []);
const getFormattedBandiData = (data) => {
return [...(data || [])].map((d) => {
d.date = new Date(d.date);
return d;
});
};
const formatDate = (value) => {
return value.toLocaleDateString('it-IT', {
day: '2-digit',
month: '2-digit',
year: 'numeric'
});
};
const clearFilter = () => {
initFilters();
};
const onGlobalFilterChange = (e) => {
const value = e.target.value;
let _filters = { ...filters };
_filters['global'].value = value;
setFilters(_filters);
setGlobalFilterValue(value);
};
const initFilters = () => {
setFilters({
global: { value: null, matchMode: FilterMatchMode.CONTAINS },
email: { operator: FilterOperator.AND, constraints: [{ value: null, matchMode: FilterMatchMode.STARTS_WITH }] },
date: { operator: FilterOperator.AND, constraints: [{ value: null, matchMode: FilterMatchMode.DATE_IS }] },
});
setGlobalFilterValue('');
};
const renderHeader = () => {
return (
<div className="appTableHeader">
<Button type="button" icon="pi pi-filter-slash" label={__('Pulisci', 'gepafin')} outlined onClick={clearFilter} />
<IconField iconPosition="left">
<InputIcon className="pi pi-search" />
<InputText value={globalFilterValue} onChange={onGlobalFilterChange} placeholder={__('Cerca', 'gepafin')} />
</IconField>
</div>
);
};
const dateBodyTemplate = (rowData) => {
return formatDate(rowData.date);
};
const dateFilterTemplate = (options) => {
return <Calendar value={options.value} onChange={(e) => options.filterCallback(e.value, options.index)} dateFormat="mm/dd/yy" placeholder="mm/dd/yyyy" mask="99/99/9999" />;
};
const header = renderHeader();
return(
<div className="appPageSection__table">
<DataTable value={items} paginator showGridlines rows={10} loading={loading} dataKey="id"
filters={filters}
globalFilterFields={['name', 'status']}
header={header}
emptyMessage="Nothing found." onFilter={(e) => setFilters(e.filters)}>
<Column header={__('Timestamp', 'gepafin')} filterField="date" dataType="date"
style={{ minWidth: '10rem' }}
body={dateBodyTemplate} filter filterElement={dateFilterTemplate}/>
<Column field="email" header={__('Utente', 'gepafin')} filter filterPlaceholder="Search by email"
style={{ minWidth: '12rem' }}/>
<Column field="action" header={__('Azione', 'gepafin')}/>
<Column field="dettails" header={__('Dettagli', 'gepafin')}/>
</DataTable>
</div>
)
}
export default LatestUsersActivityTable;

View File

@@ -0,0 +1,58 @@
import React from 'react';
import { __ } from '@wordpress/i18n';
import { useNavigate } from 'react-router-dom';
// components
import LatestBandiTable from './components/LatestBandiTable';
import LatestUsersActivityTable from './components/LatestUsersActivityTable';
import { Button } from 'primereact/button';
const DashboardBenefeciario = () => {
const navigate = useNavigate();
const onGoToCreateNewBando = () => {
navigate('/bandi/new');
}
const onGoToUsers = () => {
console.log('onGoToUsers')
}
const onGoToStats = () => {
console.log('onGoToStats')
}
const onGoToSettings = () => {
console.log('onGoToSettings')
}
return(
<div className="appPage">
<div className="appPage__pageHeader">
<h1>{__('Dashboard', 'gepafin')}</h1>
</div>
<div className="appPage__spacer"></div>
<div className="appPageSection statsBigBadges">
<h2>{__('Panoramica di Sistema', 'gepafin')}</h2>
<div className="statsBigBadges__grid">
<div className="statsBigBadges__gridItem">
<span>{__('Domande attivi', 'gepafin')}</span>
<span>3</span>
</div>
<div className="statsBigBadges__gridItem">
<span>{__('Bandi osservati', 'gepafin')}</span>
<span>42</span>
</div>
<div className="statsBigBadges__gridItem">
<span>{__('Documenti da integrare', 'gepafin')}</span>
<span>2</span>
</div>
</div>
</div>
</div>
)
}
export default DashboardBenefeciario;

View File

@@ -5,6 +5,7 @@ import PageNotFound from './pages/PageNotFound';
import Login from './pages/Login'; import Login from './pages/Login';
import ProtectedRoute from './components/ProtectedRoute'; import ProtectedRoute from './components/ProtectedRoute';
import Dashboard from './pages/Dashboard'; import Dashboard from './pages/Dashboard';
import DashboardBenefeciario from './pages/DashboardBenefeciario';
import DefaultLayout from './layouts/DefaultLayout'; import DefaultLayout from './layouts/DefaultLayout';
import Bandi from './pages/Bandi'; import Bandi from './pages/Bandi';
import BandoEdit from './pages/BandoEdit'; import BandoEdit from './pages/BandoEdit';
@@ -13,21 +14,41 @@ import BandoFormsEdit from './pages/BandoFormsEdit';
import BandoForms from './pages/BandoForms'; import BandoForms from './pages/BandoForms';
import BandoFormsPreview from './pages/BandoFormsPreview'; import BandoFormsPreview from './pages/BandoFormsPreview';
const routes = () => ( const routes = ({ role }) => {
<Routes>
<Route element={<ProtectedRoute/>}> return (
<Route path="/" element={<DefaultLayout><Dashboard/></DefaultLayout>}/> <Routes>
<Route path="/bandi" element={<DefaultLayout><Bandi/></DefaultLayout>}/> <Route element={<ProtectedRoute/>}>
<Route path="/bandi/:id" element={<DefaultLayout><BandoEdit/></DefaultLayout>}/> <Route path="/" element={<DefaultLayout>
<Route path="/bandi/:id/preview" element={<DefaultLayout><BandoView/></DefaultLayout>}/> {'ROLE_SUPER_ADMIN' === role ? <Dashboard/> : null}
<Route path="/bandi/:id/preview-evaluation" element={<DefaultLayout><BandoView/></DefaultLayout>}/> {'ROLE_BENEFICIARY' === role ? <DashboardBenefeciario/> : null}
<Route path="/bandi/:id/forms" element={<DefaultLayout><BandoForms/></DefaultLayout>}/> </DefaultLayout>}/>
<Route path="/bandi/:id/forms/:formId" element={<DefaultLayout><BandoFormsEdit/></DefaultLayout>}/> <Route path="/bandi" element={<DefaultLayout>
<Route path="/bandi/:id/forms/:formId/preview" element={<DefaultLayout><BandoFormsPreview/></DefaultLayout>}/> {'ROLE_SUPER_ADMIN' === role ? <Bandi/> : null}
</Route> </DefaultLayout>}/>
<Route exact path="/login" element={<Login/>}/> <Route path="/bandi/:id" element={<DefaultLayout>
{/*<Route exact path="/forgot-password" element={<ForgotPassword/>}/>*/} {'ROLE_SUPER_ADMIN' === role ? <BandoEdit/> : null}
<Route path="*" element={<PageNotFound/>}/> </DefaultLayout>}/>
</Routes>); <Route path="/bandi/:id/preview" element={<DefaultLayout>
{'ROLE_SUPER_ADMIN' === role ? <BandoView/> : null}
</DefaultLayout>}/>
<Route path="/bandi/:id/preview-evaluation" element={<DefaultLayout>
{'ROLE_SUPER_ADMIN' === role ? <BandoView/> : null}
</DefaultLayout>}/>
<Route path="/bandi/:id/forms" element={<DefaultLayout>
{'ROLE_SUPER_ADMIN' === role ? <BandoForms/> : null}
</DefaultLayout>}/>
<Route path="/bandi/:id/forms/:formId" element={<DefaultLayout>
{'ROLE_SUPER_ADMIN' === role ? <BandoFormsEdit/> : null}
</DefaultLayout>}/>
<Route path="/bandi/:id/forms/:formId/preview" element={<DefaultLayout>
{'ROLE_SUPER_ADMIN' === role ? <BandoFormsPreview/> : null}
</DefaultLayout>}/>
</Route>
<Route exact path="/login" element={<Login/>}/>
{/*<Route exact path="/forgot-password" element={<ForgotPassword/>}/>*/}
<Route path="*" element={<PageNotFound/>}/>
</Routes>)
};
export default routes; export default routes;

View File

@@ -76,4 +76,8 @@ export default class AuthenticationService {
static changePassword = (request, callback, errCallback) => { static changePassword = (request, callback, errCallback) => {
NetworkService.unauthorizedPatch(`${API_BASE_URL}/user/reset_password`, request, callback, errCallback); NetworkService.unauthorizedPatch(`${API_BASE_URL}/user/reset_password`, request, callback, errCallback);
} }
static me = (callback, errCallback) => {
NetworkService.get(`${API_BASE_URL}/user/me`, callback, errCallback);
};
} }

View File

@@ -0,0 +1,34 @@
import { NetworkService } from './network-service';
const API_BASE_URL = process.env.REACT_APP_API_EXECUTION_ADDRESS;
export default class BandoService {
static getBandi = (callback, errCallback) => {
NetworkService.get(`${API_BASE_URL}/call`, callback, errCallback);
};
static getBando = (id, callback, errCallback) => {
NetworkService.get(`${API_BASE_URL}/call/${id}`, callback, errCallback);
};
static validateBando = (id, callback, errCallback) => {
NetworkService.post(`${API_BASE_URL}/call/validate/${id}`, {}, callback, errCallback);
};
static createBando = (body, callback, errCallback, queryParams) => {
NetworkService.post(`${API_BASE_URL}/call/step1`, body, callback, errCallback, queryParams);
};
static updateBandoStep1 = (id, body, callback, errCallback) => {
NetworkService.put(`${API_BASE_URL}/call/step1/${id}`, body, callback, errCallback);
};
static updateBandoStep2 = (id, body, callback, errCallback) => {
NetworkService.put(`${API_BASE_URL}/call/step2/${id}`, body, callback, errCallback);
};
static updateBandoStatus = (id, callback, errCallback, queryParams) => {
NetworkService.put(`${API_BASE_URL}/call/${id}/status`, {}, callback, errCallback, queryParams);
};
}

View File

@@ -4,8 +4,8 @@ const API_BASE_URL = process.env.REACT_APP_API_EXECUTION_ADDRESS;
export default class FileUploadService { export default class FileUploadService {
static uploadFile = (body, callback, errCallback, queryParams) => { static uploadFile = (id, body, callback, errCallback, queryParams) => {
NetworkService.postMultiPart(`${API_BASE_URL}/document/uploadFile`, body, callback, errCallback, queryParams); NetworkService.postMultiPart(`${API_BASE_URL}/document/uploadFile/call/${id}`, body, callback, errCallback, queryParams);
}; };
static deleteFile = (body, callback, errCallback, queryParams) => { static deleteFile = (body, callback, errCallback, queryParams) => {

View File

@@ -0,0 +1,26 @@
import { NetworkService } from './network-service';
const API_BASE_URL = process.env.REACT_APP_API_EXECUTION_ADDRESS;
export default class FormsService {
static getElementItems = (callback, errCallback) => {
NetworkService.get(`${API_BASE_URL}/formField`, callback, errCallback);
};
static getFormsForCall = (id, callback, errCallback) => {
NetworkService.get(`${API_BASE_URL}/form/call/${id}`, callback, errCallback);
};
static createFormForCall = (id, body, callback, errCallback) => {
NetworkService.post(`${API_BASE_URL}/form/call/${id}`, body, callback, errCallback);
};
static updateForm = (id, body, callback, errCallback) => {
NetworkService.put(`${API_BASE_URL}/form/${id}`, body, callback, errCallback);
};
static getFormById = (id, callback, errCallback) => {
NetworkService.get(`${API_BASE_URL}/form/${id}`, callback, errCallback);
};
}

View File

@@ -0,0 +1,10 @@
import { NetworkService } from './network-service';
const API_BASE_URL = process.env.REACT_APP_API_EXECUTION_ADDRESS;
export default class LookupdataService {
static getItems = (callback, errCallback, queryParams) => {
NetworkService.get(`${API_BASE_URL}/lookUpData/type`, callback, errCallback, queryParams);
};
}

View File

@@ -4,10 +4,8 @@ const initialStore = {
// user // user
userData: {}, userData: {},
token: '', token: '',
// bando form
bandoFormErrors: {},
// form builder // form builder
formId: '', formId: 0,
formLabel: '', formLabel: '',
formElements: [], formElements: [],
elementItems: [], elementItems: [],

View File

@@ -2,6 +2,14 @@ const selectors = (state, get, api) => ({
getToken: () => { getToken: () => {
return get.token(); return get.token();
}, },
getRole: () => {
const userData = get.userData();
return userData.role ? userData.role.roleType : '';
},
getPermissions: () => {
const userData = get.userData();
return userData.role ? userData.role.permissions : [];
},
}) })
export default selectors; export default selectors;

View File

@@ -5,7 +5,7 @@ const zustandXOpts = {
persist: { persist: {
enabled: true, enabled: true,
partialize: (state) => ({ partialize: (state) => ({
userData: state.userData, //userData: state.userData,
token: state.token token: state.token
}), }),
} }