-
Notifications
You must be signed in to change notification settings - Fork 38
Data Mapping with TypeScript
The following code snippet may illustrate the mapping between .NET simple types and TypeScript types.
static readonly Dictionary<string, string> typeMap = new Dictionary<string, string>()
{
{typeof(int).FullName, "number"},
{typeof(uint).FullName, "number"},
{typeof(long).FullName, "number"},
{typeof(ulong).FullName, "number"},
{typeof(short).FullName, "number"},
{typeof(ushort).FullName, "number"},
{typeof(float).FullName, "number"},
{typeof(double).FullName, "number"},
{typeof(decimal).FullName, "number"},
{typeof(byte).FullName, "number"},
{typeof(sbyte).FullName, "number"},
{typeof(string).FullName, "string"},
{typeof(char).FullName, "string"},
{typeof(Guid).FullName, "string"},
{typeof(bool).FullName, "boolean"},
{typeof(void).FullName, "void"},
{typeof(object).FullName, "any"},
{typeof(DateTime).FullName, "Date"},
{typeof(DateTimeOffset).FullName, "Date"},
//TimeSpan is not so supported in Javascript
};
In addition to CLR array, the types illustrated in the following code snippet are translated into TypeScript array.
static readonly System.Collections.Generic.HashSet<string> arrayTypeNames = new System.Collections.Generic.HashSet<string>(
new string[]() {
typeof(IEnumerable<>).FullName,
typeof(IList<>).FullName,
typeof(ICollection<>).FullName,
typeof(IQueryable<>).FullName,
typeof(IReadOnlyList<>).FullName,
typeof(List<>).FullName,
typeof(System.Collections.ObjectModel.Collection<>).FullName,
typeof(IReadOnlyCollection<>).FullName
}
);
Hints:
Since Javascript supports only single dimensional array, this article "Tricky Array" may help you to deal with multidimensional array.
A dictionary is based on a hash table available only at run time, and it is generally impossible to transport a hash table over wire. However, common practices are to transport the key / value pairs in a list, and it is up to the client programs to decide whether to reassemble the list into a dictionary, or just use the list through indexer.
IDictionary<K,V> and Dictionary<K,V> are translated into Dictionary<ClientK, ClientV>.
Remarks:
Such transportation and transformation are done by Newtonsoft.JSON which can reassemble the key / value pairs into a dictionary.
IDictionary<K,V> and Dictionary<K,V> are translated into an indexer like
{[id: string]: DemoWebApi_DemoData_Client.Person }
KeyValuePair<K, V> is translated into a structure like
{Key: string, Value: DemoWebApi_DemoData_Client.Person }
Remarks:
In Javascript, the key of an indexer must be string.
By default, all members generated in a TypeScript interface are optional, unless the members in the .NET classes are decorated with one of the following:
[DataMember(IsRequired =true)]
...
[JsonProperty(Required = Required.Always)]
...
[Required]
...
Stream is supported only in HTTP GET. For jQuery, the return type is "any". For Angular2, the return type is "Response".
Byte array is supported only in HTTP GET. For jQuery and Angular 2, the return type is "Array".
TypeScript supports generics, and WebApiClientGen 2.5 supports custom generic classes.
Web API
[DataContract(Namespace = Constants.DataNamespace)]
public class MimsResult<T>
{
[DataMember]
public T Result { get; set; }
[DataMember]
public DateTime GeneratedAt { get; set; }
[DataMember]
public bool Success { get; set; } = true;
[DataMember]
public string Message { get; set; }
}
[DataContract(Namespace = Constants.DataNamespace)]
public class MimsPackage
{
[DataMember]
public MimsResult<Decimal> Result { get; set; }
[DataMember]
public string Tag { get; set; }
}
[DataContract(Namespace = Constants.DataNamespace)]
public class MyGeneric<T, K, U>
{
[DataMember]
public T MyT { get; set; }
[DataMember]
public K MyK { get; set; }
[DataMember]
public U MyU { get; set; }
[DataMember]
public string Status { get; set; }
}
[HttpPost]
[Route("Mims")]
public MimsResult<string> GetMims([FromBody] MimsPackage p)
...
[HttpPost]
[Route("MyGeneric")]
public MyGeneric<string, decimal, double> GetMyGeneric([FromBody] MyGeneric<string, decimal, double> s)
...
[HttpPost]
[Route("MyGenericPerson")]
public MyGeneric<string, decimal, Person> GetMyGenericPerson([FromBody] MyGeneric<string, decimal, Person> s)
...
Client API:
export interface MimsResult<T> {
result?: T;
generatedAt?: Date;
success?: boolean;
message?: string;
}
export interface MimsPackage {
result?: DemoWebApi_DemoData_Client.MimsResult<number>;
tag?: string;
}
export interface MyGeneric<T, K, U> {
myT?: T;
myK?: K;
myU?: U;
status?: string;
}
getMims(p: DemoWebApi_DemoData_Client.MimsPackage): Observable<DemoWebApi_DemoData_Client.MimsResult<string>> {
return this.http.post<DemoWebApi_DemoData_Client.MimsResult<string>>(this.baseUri + 'api/Entities/Mims', JSON.stringify(p), { headers: { 'Content-Type': 'application/json;charset=UTF-8' } });
}
getMyGeneric(s: DemoWebApi_DemoData_Client.MyGeneric<string, number, number>): Observable<DemoWebApi_DemoData_Client.MyGeneric<string, number, number>> {
return this.http.post<DemoWebApi_DemoData_Client.MyGeneric<string, number, number>>(this.baseUri + 'api/Entities/MyGeneric', JSON.stringify(s), { headers: { 'Content-Type': 'application/json;charset=UTF-8' } });
}
getMyGenericPerson(s: DemoWebApi_DemoData_Client.MyGeneric<string, number, DemoWebApi_DemoData_Client.Person>): Observable<DemoWebApi_DemoData_Client.MyGeneric<string, number, DemoWebApi_DemoData_Client.Person>> {
return this.http.post<DemoWebApi_DemoData_Client.MyGeneric<string, number, DemoWebApi_DemoData_Client.Person>>(this.baseUri + 'api/Entities/MyGenericPerson', JSON.stringify(s), { headers: { 'Content-Type': 'application/json;charset=UTF-8' } });
}
Remarks:
If you need to handle a lot Stream or Byte Array, it is better to put respective functions to a dedicated ApiController, and exclude the controller from the code generation process.