In my Ionic 5 app, I am going to show users an overlay with some text on it when the internet connection is not available.
I have following function to detect if the internet is available or not.
checkOnline$(): Observable<boolean> {
return merge(
of(navigator.onLine),
fromEvent(window, 'offline').pipe(mapTo(false)),
fromEvent(window, 'online').pipe(mapTo(true)),
);
}
Inside my constructor, I call above function.
this.checkOnline$().subscribe(isOnline => {
if (!isOnline) {
console.log('>>>>>> offLine');
this.online = false;
} else {
console.log('>>>>>> online');
this.online = true;
}
console.log('>>>>>> ##### this.online = ', this.online)
});
The property online
is used in the template to show or hide the overlay depending on the availability of internet.
<div id="offline-overlay" *ngIf="!online" class="center-vertically">
<div id="offline-overlay-text" class="center-content">
<span>ABCD EFGH IJKL</span>
</div>
</div>
Above setup is working fine. I get all console logs as expected and overlay is correctly shown or hidden according to the availability of wifi. BUT ON BROWSER ONLY.
When I push the app to a device, it does not work. To check if the property online
is assigned the correct values, I tried printing it in the template.
{{online}}
No, it does not get the correct value. but in the browser, it does get the correct value.
I debugged the app installed to the device to check the console logs. Those are logged as expected.
Why is this?
Why different behaviors on browsers and devices?
How can I fix this.
Thanks in advance.
EDIT:
I added logs to check if the property online
gets the correct value assigned. It does get the correct value when run on the device. Now it's just that the template not getting that value.
Found the solution.
It turned out that the code inside the subscription breaks out of Angular Zone.
I did following to fix the issue.
this.checkOnline$().subscribe(isOnline => {
console.log('************* isonline = ', isOnline);
if (isOnline) {
console.log('>>>>>> online');
this.zone.run(() => {
this.isOffline = false;
this.globalDataService.setAppOnline();
});
} else {
console.log('>>>>>> offline');
this.zone.run(() => {
this.isOffline = true;
this.globalDataService.setAppOffline();
});
}
});
Thanks to this answer: