import { MetadataSrc } from 'pages/Simulateurs/metadata-src' import { useTranslation } from 'react-i18next' import { LegalStatus } from '@/selectors/companyStatusSelectors' export const LANDING_LEGAL_STATUS_LIST: Array = [ 'EI', 'EIRL', 'EURL', 'SAS', 'SARL', 'SASU', 'auto-entrepreneur', 'auto-entrepreneur-EIRL', 'SA', ] const status = Object.fromEntries( LANDING_LEGAL_STATUS_LIST.map((statut) => [statut, statut]) ) as { [statut in LegalStatus]: statut } const rawSitePathsFr = { index: '', créer: { index: 'créer', ...status, après: 'après-la-création', guideStatut: { index: 'statut-juridique', liste: 'liste', soleProprietorship: 'responsabilité', directorStatus: 'dirigeant', autoEntrepreneur: 'auto-entrepreneur-ou-entreprise-individuelle', multipleAssociates: 'nombre-associés', minorityDirector: 'gérant-majoritaire-ou-minoritaire', }, }, gérer: { index: 'gérer', entreprise: ':entreprise', embaucher: 'embaucher', sécuritéSociale: 'sécurité-sociale', 'déclaration-charges-sociales-indépendant': 'declaration-charges-sociales-independant', déclarationIndépendant: { index: 'aide-declaration-independants', beta: { index: 'beta', entreprise: 'entreprise', imposition: 'imposition', déclaration: 'declaration', cotisations: 'cotisations', }, }, formulaireMobilité: 'demande-mobilité', }, simulateurs: { index: 'simulateurs', 'auto-entrepreneur': 'auto-entrepreneur', 'entreprise-individuelle': 'entreprise-individuelle', eirl: 'eirl', sasu: 'sasu', eurl: 'eurl', indépendant: 'indépendant', comparaison: 'comparaison-régimes-sociaux', pamc: 'pamc', salarié: 'salaire-brut-net', 'artiste-auteur': 'artiste-auteur', 'profession-libérale': { index: 'profession-liberale', médecin: 'medecin', pharmacien: 'pharmacien', auxiliaire: 'auxiliaire-medical', 'chirurgien-dentiste': 'chirurgien-dentiste', 'sage-femme': 'sage-femme', avocat: 'avocat', 'expert-comptable': 'expert-comptable', }, 'chômage-partiel': 'chômage-partiel', économieCollaborative: { index: 'économie-collaborative', votreSituation: 'votre-situation', }, is: 'impot-societe', dividendes: 'dividendes', 'exonération-covid': 'exonération-covid', }, nouveautés: 'nouveautés', stats: 'stats', accessibilité: 'accessibilité', budget: 'budget', développeur: { index: 'développeur', iframe: 'iframe', library: 'bibliothèque-de-calcul', api: 'api', spreadsheet: 'tableur', }, documentation: { index: 'documentation', }, plan: 'plan-de-site', } as const const rawSitePathsEn = { ...rawSitePathsFr, créer: { ...rawSitePathsFr.créer, index: 'create', après: 'after-registration', guideStatut: { index: 'legal-status', liste: 'list', soleProprietorship: 'liability', directorStatus: 'director', autoEntrepreneur: 'auto-entrepreneur', multipleAssociates: 'multiple-associates', minorityDirector: 'chairman-or-managing-director', }, }, gérer: { index: 'manage', entreprise: ':entreprise', embaucher: 'hiring', sécuritéSociale: 'social-security', 'déclaration-charges-sociales-indépendant': 'declaration-social-charges-independent', déclarationIndépendant: { index: 'declaration-aid-independent', beta: { index: 'beta', imposition: 'taxation', entreprise: 'company', déclaration: 'declaration', cotisations: 'contributions', }, }, formulaireMobilité: 'posting-demand', }, simulateurs: { index: 'calculators', indépendant: 'independant', 'entreprise-individuelle': 'sole-proprietorship', 'auto-entrepreneur': 'auto-entrepreneur', eirl: 'eirl', sasu: 'sasu', eurl: 'eurl', pamc: 'pamc', comparaison: 'social-scheme-comparaison', salarié: 'salary', 'artiste-auteur': 'artist-author', 'chômage-partiel': 'partial-unemployement', 'profession-libérale': { index: 'liberal-profession', médecin: 'doctor', pharmacien: 'pharmacist', auxiliaire: 'medical-auxiliary', 'chirurgien-dentiste': 'dental-surgeon', 'sage-femme': 'midwife', avocat: 'lawyer', 'expert-comptable': 'accountant', }, économieCollaborative: { index: 'sharing-economy', votreSituation: 'your-situation', }, is: 'corporate-tax', dividendes: 'dividends', 'exonération-covid': 'exoneration-covid', }, nouveautés: 'news', accessibilité: 'accessibility', développeur: { ...rawSitePathsFr.développeur, index: 'developer', library: 'library', api: 'api', spreadsheet: 'spreadsheet', }, plan: 'site-map', } as const /** * Le but des types suivants est d'obtenir un typage statique des chaînes de caractères * comme "simulateurs.auto-entrepreneur" utilisés comme identifiants des routes (via les pathId dans metadata-src.ts). * Cela permet de ne pas avoir de faute dans les clés comme 'aide-embauche' au lieu de 'aides-embauche' */ // Transfrom string type like PathToType<'simulateurs.auto-entrepreneur', number> // into { simulateurs : { auto-entrepreneur: number }} type PathToType = T extends `${infer U}.${infer V}` ? { [key in U]: V extends string ? PathToType : never } : { [key in T]: W } // Transform type A | B into A & B type UnionToIntersection = ( T extends unknown ? (x: T) => void : never ) extends (x: infer R) => void ? R : never // Union of pathId type PathIds = MetadataSrc[keyof MetadataSrc]['pathId'] type RequiredPath = Required>> // If there is a type error here, check rawSitePathsFr object matches the metadata-src.ts pathId const checkedSitePathsFr: RequiredPath & typeof rawSitePathsFr = rawSitePathsFr // If there is a type error here, check rawSitePathsEn object matches the metadata-src.ts pathId const checkedSitePathsEn: RequiredPath & typeof rawSitePathsEn = rawSitePathsEn type SitePathsFr = typeof checkedSitePathsFr type SitePathsEn = typeof checkedSitePathsEn type GenericSitePath = { [key: string]: string | GenericSitePath } const encodeRelativeSitePaths = (base: T): T => { const sitepaths = Object.entries(base).reduce( (obj, [key, val]) => ({ ...obj, [key]: // Remove encodeURI next release of react router // Wait next version after v6.4.3, cf https://github.com/remix-run/react-router/issues/9580 typeof val === 'string' ? encodeURI(val) : encodeRelativeSitePaths(val), }), {} as T ) return sitepaths } const encodedRelativeSitePaths = encodeRelativeSitePaths({ fr: rawSitePathsFr, en: rawSitePathsEn, }) const encodedAbsoluteSitePaths = { fr: constructAbsoluteSitePaths(rawSitePathsFr), en: constructAbsoluteSitePaths(rawSitePathsEn), } export const relativeSitePaths = encodedRelativeSitePaths export const absoluteSitePaths = encodedAbsoluteSitePaths export type RelativeSitePaths = typeof relativeSitePaths[keyof typeof relativeSitePaths] export type AbsoluteSitePaths = typeof absoluteSitePaths[keyof typeof absoluteSitePaths] export const useSitePaths = (lang?: T) => { const { language } = useTranslation().i18n lang ??= language as T return { relativeSitePaths: relativeSitePaths[lang], absoluteSitePaths: absoluteSitePaths[lang], } } type SitePath = { [key: string]: string | SitePath } & { index: string } type SitePathBuilt = { [K in keyof T]: T[K] extends string ? K extends 'index' ? `${Root}${T[K]}` extends '' ? '/' : `${Root}${T[K]}` : T extends { index: string } ? `${Root}${T['index']}/${T[K]}` : `${Root}${T[K]}` : SitePathBuilt< T[K] extends SitePath ? T[K] : never, T extends { index: string } ? `${Root}${T['index']}/` : `${Root}` > } function constructAbsoluteSitePaths( obj: T, root = '' ): SitePathBuilt { const { index } = obj const entries = Object.entries(obj) return Object.fromEntries( entries.map(([k, value]) => [ k, typeof value === 'string' ? encodeURI(root + (k === 'index' ? value : index + '/' + value)) || '/' : constructAbsoluteSitePaths(value, root + index + '/'), ]) ) as SitePathBuilt } type Obj = { [k: string]: string | Obj } const deepReduce = ( fn: (acc: string[], val: string, key: string) => string[], initialValue: string[], object: Obj ): string[] => Object.entries(object).reduce( (acc, [key, value]) => typeof value === 'object' ? deepReduce(fn, acc, value) : fn(acc, value, key), initialValue ) type SiteMap = Array export const generateSiteMap = (sitePaths: AbsoluteSitePaths): SiteMap => deepReduce( (paths: Array, path: string) => /\/:/.test(path) ? paths : [...paths, ...[path]], [], sitePaths ) export const alternateLinks = () => { const basePathFr = import.meta.env.DEV && typeof window !== 'undefined' ? `http://${window.location.host}/mon-entreprise` : import.meta.env.VITE_FR_BASE_URL ?? '' const basePathEn = import.meta.env.DEV && typeof window !== 'undefined' ? `http://${window.location.host}/infrance` : import.meta.env.VITE_EN_BASE_URL ?? '' const enSiteMap = generateSiteMap(absoluteSitePaths.en).map( (path) => basePathEn + path ) const frSiteMap = generateSiteMap(absoluteSitePaths.fr).map( (path) => basePathFr + path ) return { en: Object.fromEntries( enSiteMap.map((key, i) => [key, { href: frSiteMap[i], hrefLang: 'fr' }]) ), fr: Object.fromEntries( frSiteMap.map((key, i) => [key, { href: enSiteMap[i], hrefLang: 'en' }]) ), } }