1
0
mirror of https://github.com/LaCasemate/fab-manager.git synced 2025-01-24 13:52:21 +01:00

139 lines
5.0 KiB
TypeScript
Raw Normal View History

2022-05-02 17:54:58 +02:00
import React, { useEffect, useState } from 'react';
import { useForm } from 'react-hook-form';
import { FormInput } from '../form/form-input';
import SettingAPI from '../../api/setting';
2022-05-03 18:05:30 +02:00
import { supportedNetworks } from '../../models/social-network';
2022-05-02 17:54:58 +02:00
import { IApplication } from '../../models/application';
import { Loader } from '../base/loader';
import { react2angular } from 'react2angular';
import { SettingName } from '../../models/setting';
import Icons from '../../../../images/social-icons.svg';
import { Trash } from 'phosphor-react';
import { useTranslation } from 'react-i18next';
import { FabButton } from '../base/fab-button';
declare const Application: IApplication;
2022-05-03 18:05:30 +02:00
interface FabSocialsProps {
2022-05-04 14:57:31 +02:00
show: boolean,
onError: (message: string) => void,
onSuccess: (message: string) => void
2022-05-03 18:05:30 +02:00
}
/**
* Allows the Fablab to edit its corporate social networks, or to display them read-only to the end users (show=true)
*/
2022-05-04 14:57:31 +02:00
export const FabSocials: React.FC<FabSocialsProps> = ({ show = false, onError, onSuccess }) => {
2022-05-02 17:54:58 +02:00
const { t } = useTranslation('shared');
2022-05-04 14:57:31 +02:00
// regular expression to validate the the input fields
const urlRegex = /^(https?:\/\/)([\da-z.-]+)\.([-a-z\d.]{2,30})([/\w .-]*)*\/?$/;
2022-05-02 17:54:58 +02:00
2022-05-04 14:57:31 +02:00
const { handleSubmit, register, setValue, formState } = useForm();
2022-05-02 17:54:58 +02:00
2022-05-04 10:06:00 +02:00
const settingsList = supportedNetworks.map(el => el as SettingName);
2022-05-02 17:54:58 +02:00
const [fabNetworks, setFabNetworks] = useState([]);
2022-05-03 18:05:30 +02:00
const [selectedNetworks, setSelectedNetworks] = useState([]);
2022-05-02 17:54:58 +02:00
useEffect(() => {
2022-05-03 18:05:30 +02:00
SettingAPI.query(settingsList).then(res => {
setFabNetworks(Array.from(res, ([name, url]) => ({ name, url })));
2022-05-02 17:54:58 +02:00
}).catch(error => console.error(error));
}, []);
2022-05-03 18:05:30 +02:00
useEffect(() => {
setSelectedNetworks(fabNetworks.filter(el => el.url !== ''));
}, [fabNetworks]);
/**
* Callback triggered when the social networks are saved
*/
2022-05-03 18:05:30 +02:00
const onSubmit = (data) => {
const updatedNetworks = new Map<SettingName, string>();
Object.keys(data).forEach(key => updatedNetworks.set(key as SettingName, data[key]));
SettingAPI.bulkUpdate(updatedNetworks).then(res => {
const errorResults = Array.from(res.values()).filter(item => !item.status);
if (errorResults.length > 0) {
2022-05-04 14:57:31 +02:00
onError(t('app.shared.fab_socials.networks_update_error'));
} else {
onSuccess(t('app.shared.fab_socials.networks_update_success'));
2022-05-03 18:05:30 +02:00
}
});
};
/**
* Callback triggered when the user adds a network, from the list of available networks, to the editable networks.
*/
2022-05-02 17:54:58 +02:00
const selectNetwork = (network) => {
2022-05-03 18:05:30 +02:00
setSelectedNetworks([...selectedNetworks, network]);
2022-05-02 17:54:58 +02:00
};
/**
* Callback triggered when the user removes a network, from the list of editables networks, add put it back to the
* list of avaiable networks.
*/
2022-05-02 17:54:58 +02:00
const remove = (network) => {
2022-05-03 18:05:30 +02:00
setSelectedNetworks(selectedNetworks.filter(el => el !== network));
setValue(network.name, '');
2022-05-02 17:54:58 +02:00
};
return (
<div className="fab-socials">{show
2022-05-12 10:47:28 +02:00
? (selectedNetworks.length > 0) && <>
<h2>{t('app.shared.fab_socials.follow_us')}</h2>
<div className='social-icons'>
{fabNetworks.map((network, index) =>
selectedNetworks.includes(network) &&
<a key={index} href={network.url} target='_blank' rel="noreferrer">
<img src={`${Icons}#${network.name}`}></img>
</a>
)}
</div>
</>
2022-05-03 18:05:30 +02:00
: <form onSubmit={handleSubmit(onSubmit)}>
2022-05-04 14:57:31 +02:00
<div className='social-icons'>
2022-05-03 18:05:30 +02:00
{fabNetworks.map((network, index) =>
!selectedNetworks.includes(network) &&
2022-05-04 10:06:00 +02:00
<img key={index} src={`${Icons}#${network.name}`} onClick={() => selectNetwork(network)}></img>
2022-05-03 18:05:30 +02:00
)}
</div>
2022-05-04 14:57:31 +02:00
{selectNetwork.length && <div className='social-inputs'>
{fabNetworks.map((network, index) =>
selectedNetworks.includes(network) &&
<FormInput id={network.name}
key={index}
register={register}
rules={{
pattern: {
value: urlRegex,
message: t('app.shared.user_profile_form.website_invalid')
}
}}
formState={formState}
defaultValue={network.url}
label={network.name}
placeholder={t('app.shared.fab_socials.url_placeholder')}
icon={<img src={`${Icons}#${network.name}`}></img>}
addOn={<Trash size={16} />}
addOnAction={() => remove(network)} />
)}
</div>}
2022-05-03 18:05:30 +02:00
<FabButton type='submit'
className='save-btn'>
2022-05-03 18:05:30 +02:00
{t('app.shared.buttons.save')}
</FabButton>
</form>
}</div>
2022-05-02 17:54:58 +02:00
);
};
2022-05-03 18:05:30 +02:00
const FabSocialsWrapper: React.FC<FabSocialsProps> = (props) => {
2022-05-02 17:54:58 +02:00
return (
<Loader>
<FabSocials {...props} />
</Loader>
);
};
2022-05-04 14:57:31 +02:00
Application.Components.component('fabSocials', react2angular(FabSocialsWrapper, ['show', 'onError', 'onSuccess']));