fix(education): query annuaire education in the browser

This commit is contained in:
Paul Chavard 2024-07-12 17:31:38 +02:00
parent 91b398d039
commit d6e0d83c5d
No known key found for this signature in database
4 changed files with 36 additions and 5 deletions

View file

@ -9,7 +9,8 @@ class EditableChamp::AnnuaireEducationComponent < EditableChamp::EditableChampBa
name: @form.field_name(:external_id), name: @form.field_name(:external_id),
selected_key: @champ.external_id, selected_key: @champ.external_id,
items: @champ.selected_items, items: @champ.selected_items,
loader: data_sources_data_source_education_path, loader: 'https://data.education.gouv.fr/api/records/1.0/search?dataset=fr-en-annuaire-education&rows=5',
coerce: 'AnnuaireEducation',
debounce: 500, debounce: 500,
minimum_input_length: 5) minimum_input_length: 5)
end end

View file

@ -232,6 +232,7 @@ export function RemoteComboBox({
minimumInputLength, minimumInputLength,
limit, limit,
debounce, debounce,
coerce,
formValue, formValue,
name, name,
form, form,
@ -245,9 +246,9 @@ export function RemoteComboBox({
const load = useMemo( const load = useMemo(
() => () =>
typeof loader == 'string' typeof loader == 'string'
? createLoader(loader, { minimumInputLength, limit }) ? createLoader(loader, { minimumInputLength, limit, coerce })
: loader, : loader,
[loader, minimumInputLength, limit] [loader, minimumInputLength, limit, coerce]
); );
const { selectedItem, onReset, ...comboBoxProps } = useRemoteList({ const { selectedItem, onReset, ...comboBoxProps } = useRemoteList({
allowsCustomValue, allowsCustomValue,

View file

@ -401,12 +401,39 @@ function getKey(item: Item) {
return item.value; return item.value;
} }
const AnnuaireEducationPayload = s.type({
records: s.array(
s.type({
fields: s.type({
identifiant_de_l_etablissement: s.string(),
nom_etablissement: s.string(),
nom_commune: s.string()
})
})
)
});
const Coerce = {
Default: s.array(Item),
AnnuaireEducation: s.coerce(
s.array(Item),
AnnuaireEducationPayload,
({ records }) =>
records.map((record) => ({
label: `${record.fields.nom_etablissement}, ${record.fields.nom_commune} (${record.fields.identifiant_de_l_etablissement})`,
value: record.fields.identifiant_de_l_etablissement,
data: record
}))
)
};
export const createLoader: ( export const createLoader: (
source: string, source: string,
options?: { options?: {
minimumInputLength?: number; minimumInputLength?: number;
limit?: number; limit?: number;
param?: string; param?: string;
coerce?: keyof typeof Coerce;
} }
) => Loader = ) => Loader =
(source, options) => (source, options) =>
@ -427,7 +454,8 @@ export const createLoader: (
}); });
if (response.ok) { if (response.ok) {
const json = await response.json(); const json = await response.json();
const [err, items] = s.validate(json, s.array(Item), { coerce: true }); const struct = Coerce[options?.coerce ?? 'Default'];
const [err, items] = s.validate(json, struct, { coerce: true });
if (!err) { if (!err) {
if (items.length > limit) { if (items.length > limit) {
const filteredItems = matchSorter(items, filterText, { const filteredItems = matchSorter(items, filterText, {

View file

@ -68,7 +68,8 @@ export const RemoteComboBoxProps = s.assign(
minimumInputLength: s.number(), minimumInputLength: s.number(),
limit: s.number(), limit: s.number(),
allowsCustomValue: s.boolean(), allowsCustomValue: s.boolean(),
debounce: s.number() debounce: s.number(),
coerce: s.enums(['Default', 'AnnuaireEducation'])
}) })
) )
); );