.NET Core SignalR Hub+ Angular Observable
TL;DR: Deliver all the table content continuously paginated to the browser with .NET Core and Angular
Long Description:
I was thinking that delivering paginating data is kind of lame. Why should I , for seeing the whole data, should keep pressing next page / or scrolling down ? I understand that on the server it is necessary to do this – but on the client ?
And I was thinking – what about Observables in Angular , combined with SignalR ? Those observables could obtain data every time – and SignalR is pushing cotinously, until it arrives at the end of the data.
Done with talking – let’s show some code:
We have an Employee classs
1 2 3 4 5 6 | public class Employee { public string Name { get ; set ; } public int Id { get ; set ; } } |
code
and we are feeding with data in a DALEmployee
01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 17 18 19 20 21 22 23 | public class DalEmployee { public int FindCountEmployees( string name) { return ((name ?? "" ).Length + 1) * 10; } public async Task<Employee[]> FindEmployee( int pageNumber, int pageSize, string name) { await Task.Delay((pageNumber + 1) * 1000); var ret = new Employee[pageSize]; int startNumber = (pageNumber - 1) * pageSize; for ( int i = 0; i < pageSize; i++) { var e = new Employee(); e.Id = startNumber + i; e.Name = e.Id + "employee " ; ret[i] = e; } return ret; } } |
We have SignalR that delivers continously the data via a Hub that is connected to an URL
1 2 3 4 5 | //do not forget services.AddSignalR().AddJsonProtocol() ; app.UseSignalR(routes => { routes.MapHub<MyHub>( "/employees" ); }) |
and delivers all data continuously
01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 | public class MyHub: Hub { public override async Task OnConnectedAsync() { Console.WriteLine( "!!!!connected" ); await Task.Delay(1000); } public ChannelReader<Employee[]> GetEmps( string name) { var channel = Channel.CreateUnbounded<Employee[]>(); _ = WriteItems(channel.Writer, name); //it not awaits! return channel.Reader; } private async Task WriteItems(ChannelWriter<Employee[]> writer, string name) { var dal = new DalEmployee(); var nr = dal.FindCountEmployees(name); int nrPages = 10; for ( int i = 0; i < nr / nrPages; i++) { var data = await dal.FindEmployee(i+1, nrPages, name); await writer.WriteAsync(data); } writer.TryComplete(); } |
And this is all that is required on the .NET Core side
For the Angular Part
We have a hubconnectionbuilder that connect to the hub
01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 | export class MydataService { h:HubConnection; constructor( private http: HttpClient) { this .h= new HubConnectionBuilder() .withUrl(environment.urlAPI + "/employees" ) .withHubProtocol( new JsonHubProtocol()) //.withHubProtocol(new MessagePackHubProtocol()) .build(); window.alert( 'start' ); this .h.start() . catch ((t) => window.alert(1)); } } |
and a observable that returns the values
1 2 3 4 5 6 7 | public getAllEmpsStream(name:string): Observable<Employee[]>{ const subject = new Subject<Employee[]>(); this .h.stream<Employee[]>( "GetEmps" ,name).subscribe(subject); return subject.asObservable(); } |
The code on the component is aggregating the values into an array
01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 | findEmployees(){ this .load= true ; this .service .getAllEmpsStream( this .nameEmp) //.getEmps(this.nameEmp) .pipe(finalize(() => this .load= false )) .subscribe(it => { //window.alert(it.length); this .emp.push(...it); }); } |
And that is all. All the values are returned page after page without the need of human intervention…
Where is the Angular part?
Starts at
” For the Angular Part “