Dynamiske kodelister
Generert ved kjøring fra C#-kode
I en Altinn 3 app har man også mulighet til å ha dynamiske kodelister som blir fortløpende generert ved kjøring av appen. Dette gjør det mulig å lage dynamiske verdier, for eksempel ved å hente og filtrere verdier fra andre kilder. Dynamiske kodelister kan enten være åpne (tilgjengelig for alle, uten autentisering), eller sikret slik at brukeren må ha tilgang til instansen for å hente kodelisten.
For åpne kodelister implementerer man IAppOptionsProvider
interfacet, mens for sikrede kodelister implementerer man IInstanceAppOptionsProvider
. Fremgangsmåten er den samme for begge og modellen som returneres er lik. Implementeringen holdes adskilt for ikke å eksponere kodelister som skulle vært sikret.
Åpne kodelister
Under finner du et eksempel på hvordan dette kan settes opp for en åpen kodeliste.
using Altinn.App.Common.Models;
using Altinn.App.PlatformServices.Options;
using System.Collections.Generic;
using System.Threading.Tasks;
namespace Altinn.App.Core
{
public class CountryAppOptionsProvider : IAppOptionsProvider
{
public string Id { get; set; } = "countries";
public Task<AppOptions> GetAppOptionsAsync(string language, Dictionary<string, string> keyValuePairs)
{
var options = new AppOptions
{
Options = new List<AppOption>
{
new AppOption
{
Label = "Norway",
Value = "47"
},
new AppOption
{
Label = "Sweden",
Value = "46"
}
}
};
return Task.FromResult(options);
}
}
}
For at denne implementasjonen skal plukkes opp av applikasjonen må den registreres i Program.cs
:
services.AddTransient<IAppOptionsProvider, CountryAppOptionsProvider>();
Resultatet av denne implementasjonen vil bli tilgjengeleg på endepunktet {org}/{app}/api/options/countries
. Identifikatoren kan brukes i komponenter, så for å bruke kodelisten i en Dropdown
-komponent kan man sette optionsId
som i følgende eksempel:
{
"id": "dropdown-component",
"type": "Dropdown",
"textResourceBindings": {
"title": "Some title"
},
"dataModelBindings": {
"simpleBinding": "some.field"
},
"optionsId": "countries"
}
Sikrede kodelister
Om du ønsker å produsere kodelister som inneholder sensitive data som ikke skal være tilgjengelige i et åpent API kan man implementere IInstanceAppOptionsProvider
. Slike sikrede kodelister kontrollerer at brukeren har lesetilgang som definert i applikasjonens policy.xml
-fil før brukeren kan hente innholdet i kodelisten.
Under finner du et eksempel på hvordan man setter opp en sikret kodeliste.
using System.Collections.Generic;
using System.Threading.Tasks;
using Altinn.App.Common.Models;
using Altinn.App.PlatformServices.Models;
namespace Altinn.App.Core
{
public class ChildrenAppOptionsProvider : IInstanceAppOptionsProvider
{
public string Id { get; set; } = "children";
public Task<AppOptions> GetInstanceAppOptionsAsync(InstanceIdentifier instanceIdentifier, string language, Dictionary<string, string> keyValuePairs)
{
// ...
// Some custom code to get the list of children from the instance owner
// ...
var options = new AppOptions
{
Options = new List<AppOption>
{
new AppOption
{
Label = "Ole",
Value = "1"
},
new AppOption
{
Label = "Dole",
Value = "2"
},
new AppOption
{
Label = "Doffen",
Value = "3"
}
}
};
return Task.FromResult(options);
}
}
}
For at denne implementasjonen skal plukkes opp av applikasjonen må den registreres i Program.cs
:
services.AddTransient<IInstanceAppOptionsProvider, ChildrenAppOptionsProvider>();
Resultatet av denne implementasjonen vil bli tilgjengeleg på endepunktet {org}/{app}/instances/{instanceOwnerId}/{instanceGUID}/options/children
. Identifikatoren kan brukes i komponenter, så for å bruke kodelisten i en Dropdown
-komponent kan man sette optionsId
som i følgende eksempel. Det er også viktig å sette secure
-egenskapen til true
for å indikere at dette er en sikret kodeliste.
{
"id": "dropdown-component",
"type": "Dropdown",
"textResourceBindings": {
"title": "Some title"
},
"dataModelBindings": {
"simpleBinding": "some.field"
},
"optionsId": "children",
"secure": true
}
Spørringsparametre
Kodeliste-endepunktet du lager støtter spørringsparametre. Parameteren language
sendes med automatisk, og andre parametre kan sendes med fra komponentkonfigurasjonen. Disse kan du lese ut fra keyValuePairs
-parameteren i implementasjonen. Dette kan være nyttig for å filtrere kodelisten basert på data i datamodellen, eller på andre måter variere kodelisten basert på kontekst.
Som et eksempel kan vi se for oss et skjema med to Dropdown
-komponenter som er knyttet sammen. Den første lar brukeren velge et fylke, og den andre lar brukeren velge en kommune. Kommunene som vises i den andre komponenten skal være filtrert basert på fylket som er valgt i den første komponenten. Dette kan løses ved å sende med fylket som et spørringsparameter til kodelisten for kommuner.
Basert på uttrykk
Man kan legge til både statiske og dynamiske parametre ved å sette opp queryParameters
på den aktuelle komponenten:
{
"id": "dropdown-komponent",
"type": "Dropdown",
"textResourceBindings": {
"title": "NyGarantiLoyvetype"
},
"dataModelBindings": {
"simpleBinding": "soknad.nyGaranti.loyvetype"
},
"required": true,
"optionsId": "loyvetyper",
"queryParameters": {
"loyvetype": "garanti",
"orgnummer": ["dataModel", "soknad.transportorOrgnummer"],
"myndig": ["greaterThanEq", ["dataModel", "soknad.alder"], 18]
}
}
I eksempelet over vil parameteret loyvetype=garanti
alltid bli sendt med (dette er helt statisk og vil ikke endre seg). Parameteret orgnummer={nr}
vil bli sendt med, hvor {nr}
er verdien på feltet soknad.transportorOrgnummer
i datamodellen. Parameteret myndig={bool}
vil bli sendt med, hvor {bool}
blir enten true
eller false
basert på om verdien på feltet soknad.alder
er større enn eller lik 18.
Flere eksempler på uttrykk finner du i dokumentasjonen for dynamikk, og den fullstendige oversikten over tilgjengelige funksjoner finner du i referanseoversikten over uttrykk.
Basert på datamodellen
Denne tilnærmingen frarådes. Fra og med app-frontend versjon 4.9.0 er det mulig å bruke queryParameters
-egenskapen i stedet. Som beskrevet ovenfor, tillater denne egenskapen deg å legge til både statiske og dynamiske spørringsparametre ved hjelp av uttrykk - noe som gjør dem mer fleksible enn mapping
.
På et tidspunkt vil mapping
-egenskapen bli fjernet, men når det skjer vil verktøy bli gitt for å migrere eksisterende konfigurasjoner til å bruke queryParameters
i stedet.
Man kan legge til dynamiske parametre ved å sette opp mapping
på den aktuelle komponenten:
{
"id": "dropdown-komponent",
"type": "Dropdown",
"textResourceBindings": {
"title": "NyGarantiLoyvetype"
},
"dataModelBindings": {
"simpleBinding": "soknad.nyGaranti.loyvetype"
},
"required": true,
"optionsId": "loyvetyper",
"mapping": {
"soknad.transportorOrgnummer": "orgnummer"
}
}
I eksempelet over vil parameteren orgnummer={nr}
bli sendt med. {nr}
er verdien på feltet soknad.transportorOrgnummer
.
Om man setter opp en kobling til et datafelt og dette feltet endrer seg, så vil appen hente kodelisten på nytt. På denne måten kan man dynamisk styre hvilke valg som vises basert på informasjon gitt av sluttbruker.
Å sende med parametre fra repeterende grupper gjøres ved å legge ved en indeks-indikator for de relevante gruppene. Eksempel:
{
"id": "dropdown-group",
"type": "Dropdown",
"textResourceBindings": {
"title": "Select city"
},
"dataModelBindings": {
"simpleBinding": "Group.City"
},
"required": true,
"optionsId": "cities",
"mapping": {
"Group[{0}].Country": "country"
}
}
For nøstede repeterende grupper vil man følge det samme mønsteret, men med en ekstra indikator for den nøstede gruppa:
{
"id": "dropdown-nested-group",
"type": "Dropdown",
"textResourceBindings": {
"title": "Select city"
},
"dataModelBindings": {
"simpleBinding": "Group.SubGroup.City"
},
"required": true,
"optionsId": "cities",
"mapping": {
"Group[{0}].SubGroup[{1}].Country": "country"
}
}
For et komplett eksempel kan du se vår demo app.
Lurt å tenke på
- Metoden
GetAppOptionsAsync
får inn en språkkode i parameterenlanguage
. Språkkoder bør baseres på ISO 639-1 standarden eller W3C IANA Language Subtag Registry standarden. Sistnevnte bygger på ISO 639-1 standarden men garanterer at alle kodene er unike, noe ISO 639-1 ikke gjør. - En app kan ha mange implementasjoner av disse interfacene, en for hver kodeliste. Den rette implementasjonen finnes gjennom å se på hvilken kodeliste-identifikator det spørres etter, og sammenlignes med
Id
-egenskapen i implementasjonen. Dette er også identifikatoren som brukes ioptionsId
-egenskapen i komponentkonfigurasjonen. Dermed må ogsåId
-egenskapen i implementasjonen være unik per app. - Det kan være fristende å sette opp en dynamisk og sikret kodeliste hvor man henter ut data fra datamodellen og produserer kodelisten basert på dette. Dette er ikke anbefalt, da appens frontend bare henter kodelisten én gang for hvert unike sett med spørringsparametre. Det betyr at visningen av kodelisten ikke vil oppdatere seg i tråd med endringene i datamodellen.
- Et alternativ er å bruke funksjonaliteten for dynamiske kodelister basert på datamodell, i noen tilfeller sammen med tilsvarende kode i DataProcessor.
- Et annet alternativ kan være å bruke spørringsparametre, som beskrevet over.
- Dersom man bruker spørringsparametre kan det være lurt å tenke gjennom hvor mange unike kombinasjoner av parametre som typisk vil bli brukt i appen. Hvis det er mange, kan det være lurt å vurdere å bruke en annen tilnærming, som for eksempel å hente ut all data og filtrere gyldige alternativer i frontend ved hjelp av
optionFilter
. Mange ulike kombinasjoner av spørringsparametre kan føre til at appen må gjøre mye unødvendig arbeid for å hente kodelisten på nytt hver gang brukeren gjør en endring i skjemaet.