1
0
mirror of https://github.com/LaCasemate/fab-manager.git synced 2025-01-08 23:46:14 +01:00
fab-manager/app/frontend/src/javascript/components/socials/edit-socials.tsx

92 lines
3.6 KiB
TypeScript
Raw Normal View History

2022-04-28 17:31:31 +02:00
import React, { useState, useReducer } from 'react';
2022-05-04 14:57:31 +02:00
import { FormState, UseFormRegister, UseFormSetValue } from 'react-hook-form';
2022-04-28 17:31:31 +02:00
import { FieldValues } from 'react-hook-form/dist/types/fields';
import { User } from '../../models/user';
import { SocialNetwork } from '../../models/social-network';
import Icons from '../../../../images/social-icons.svg';
import { FormInput } from '../form/form-input';
import { Trash } from 'phosphor-react';
import { useTranslation } from 'react-i18next';
interface EditSocialsProps<TFieldValues> {
register: UseFormRegister<TFieldValues>,
2022-05-04 08:55:51 +02:00
setValue: UseFormSetValue<User>,
2022-04-28 17:31:31 +02:00
networks: SocialNetwork[],
formState: FormState<TFieldValues>,
disabled: boolean|((id: string) => boolean),
2022-04-28 17:31:31 +02:00
}
/**
* Allow a user to edit its personnal social networks
*/
export const EditSocials = <TFieldValues extends FieldValues>({ register, setValue, networks, formState, disabled }: EditSocialsProps<TFieldValues>) => {
2022-04-28 17:31:31 +02:00
const { t } = useTranslation('shared');
2022-05-04 14:57:31 +02:00
// regular expression to validate the the input fields
2022-05-30 16:52:51 +02:00
const urlRegex = /^(https?:\/\/)([^.]+)\.(.{2,30})(\/.*)*\/?$/;
2022-04-28 17:31:31 +02:00
const initSelectedNetworks = networks.filter(el => !['', null, undefined].includes(el.url));
2022-04-28 17:31:31 +02:00
const [selectedNetworks, setSelectedNetworks] = useState(initSelectedNetworks);
/**
* Callback triggered when the user adds a network, from the list of available networks, to the editable networks.
*/
2022-04-28 17:31:31 +02:00
const selectNetwork = (network) => {
setSelectedNetworks([...selectedNetworks, network]);
};
/**
* Return a derivated state of the selected networks list, depending on the given action.
*/
2022-04-28 17:31:31 +02:00
const reducer = (state, action) => {
switch (action.type) {
case 'delete':
setSelectedNetworks(selectedNetworks.filter(el => el !== action.payload.network));
2022-05-04 08:55:51 +02:00
setValue(action.payload.field, '');
2022-04-28 17:31:31 +02:00
return state.map(el => el === action.payload.network
? { ...el, url: '' }
: el);
case 'update':
return state.map(el => el === action.payload
? { ...el, url: action.payload.url }
: el);
default:
return state;
}
};
const [userNetworks, dispatch] = useReducer(reducer, networks);
return (
<>
<div className='social-icons'>
{userNetworks.map((network, index) =>
!selectedNetworks.includes(network) && <svg key={index} onClick={() => selectNetwork(network)} viewBox="0 0 24 24" >
<use href={`${Icons}#${network.name}`} />
</svg>
2022-04-28 17:31:31 +02:00
)}
</div>
2022-05-04 14:57:31 +02:00
{selectNetwork.length && <div className='social-inputs'>
2022-04-28 17:31:31 +02:00
{userNetworks.map((network, index) =>
selectedNetworks.includes(network) &&
<FormInput key={index}
2022-05-04 15:22:52 +02:00
id={`profile_attributes.${network.name}`}
2022-04-28 17:31:31 +02:00
register={register}
2022-05-04 14:57:31 +02:00
rules= {{
pattern: {
value: urlRegex,
2022-06-20 18:06:15 +02:00
message: t('app.shared.edit_socials.website_invalid')
2022-05-04 14:57:31 +02:00
}
}}
formState={formState}
2022-04-28 17:31:31 +02:00
defaultValue={network.url}
label={network.name}
disabled={disabled}
2022-06-20 18:06:15 +02:00
placeholder={t('app.shared.edit_socials.url_placeholder')}
icon={<svg viewBox="0 0 24 24"><use href={`${Icons}#${network.name}`}/></svg>}
2022-04-28 17:31:31 +02:00
addOn={<Trash size={16} />}
2022-05-04 15:22:52 +02:00
addOnAction={() => dispatch({ type: 'delete', payload: { network, field: `profile_attributes.${network.name}` } })} />
2022-04-28 17:31:31 +02:00
)}
2022-05-04 14:57:31 +02:00
</div>}
2022-04-28 17:31:31 +02:00
</>
);
};