forked from EVOgeek/Vmeda.Online
285 lines
9.3 KiB
TypeScript
285 lines
9.3 KiB
TypeScript
// (C) Copyright 2015 Moodle Pty Ltd.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
import { AfterViewInit, Component, OnDestroy, ViewChild } from '@angular/core';
|
|
import { Subscription } from 'rxjs';
|
|
|
|
import { CoreDomUtils } from '@services/utils/dom';
|
|
import { CoreUtils } from '@services/utils/utils';
|
|
import { CoreEventObserver, CoreEvents } from '@singletons/events';
|
|
import {
|
|
AddonNotifications, AddonNotificationsNotificationMessageFormatted, AddonNotificationsProvider,
|
|
} from '../../services/notifications';
|
|
import { CoreNavigator } from '@services/navigator';
|
|
import { CoreSplitViewComponent } from '@components/split-view/split-view';
|
|
import { CoreRoutedItemsManagerSourcesTracker } from '@classes/items-management/routed-items-manager-sources-tracker';
|
|
import { CorePushNotificationsDelegate } from '@features/pushnotifications/services/push-delegate';
|
|
import { CoreSites } from '@services/sites';
|
|
import { CoreMainMenuDeepLinkManager } from '@features/mainmenu/classes/deep-link-manager';
|
|
import { CoreTimeUtils } from '@services/utils/time';
|
|
import { AddonNotificationsNotificationsSource } from '@addons/notifications/classes/notifications-source';
|
|
import { CoreListItemsManager } from '@classes/items-management/list-items-manager';
|
|
import { AddonLegacyNotificationsNotificationsSource } from '@addons/notifications/classes/legacy-notifications-source';
|
|
import { CoreLocalNotifications } from '@services/local-notifications';
|
|
import { CoreConfig } from '@services/config';
|
|
import { CoreConstants } from '@/core/constants';
|
|
import { CorePlatform } from '@services/platform';
|
|
|
|
/**
|
|
* Page that displays the list of notifications.
|
|
*/
|
|
@Component({
|
|
selector: 'page-addon-notifications-list',
|
|
templateUrl: 'list.html',
|
|
styleUrls: ['list.scss', '../../notifications.scss'],
|
|
})
|
|
export class AddonNotificationsListPage implements AfterViewInit, OnDestroy {
|
|
|
|
@ViewChild(CoreSplitViewComponent) splitView!: CoreSplitViewComponent;
|
|
notifications!: CoreListItemsManager<AddonNotificationsNotificationMessageFormatted, AddonNotificationsNotificationsSource>;
|
|
fetchMoreNotificationsFailed = false;
|
|
canMarkAllNotificationsAsRead = false;
|
|
loadingMarkAllNotificationsAsRead = false;
|
|
hasNotificationsPermission = true;
|
|
permissionWarningHidden = false;
|
|
|
|
protected isCurrentView?: boolean;
|
|
protected cronObserver?: CoreEventObserver;
|
|
protected readObserver?: CoreEventObserver;
|
|
protected pushObserver?: Subscription;
|
|
protected pendingRefresh = false;
|
|
protected appResumeSubscription?: Subscription;
|
|
|
|
constructor() {
|
|
try {
|
|
const source = CoreRoutedItemsManagerSourcesTracker.getOrCreateSource(
|
|
CoreSites.getRequiredCurrentSite().isVersionGreaterEqualThan('4.0')
|
|
? AddonNotificationsNotificationsSource
|
|
: AddonLegacyNotificationsNotificationsSource,
|
|
[],
|
|
);
|
|
|
|
this.notifications = new CoreListItemsManager(source, AddonNotificationsListPage);
|
|
} catch(error) {
|
|
CoreDomUtils.showErrorModal(error);
|
|
CoreNavigator.back();
|
|
|
|
return;
|
|
}
|
|
|
|
this.checkPermission();
|
|
this.appResumeSubscription = CorePlatform.resume.subscribe(() => {
|
|
this.checkPermission();
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
async ngAfterViewInit(): Promise<void> {
|
|
await this.fetchInitialNotifications();
|
|
|
|
this.notifications.start(this.splitView);
|
|
|
|
this.cronObserver = CoreEvents.on(AddonNotificationsProvider.READ_CRON_EVENT, () => {
|
|
if (!this.isCurrentView) {
|
|
return;
|
|
}
|
|
|
|
this.refreshNotifications();
|
|
}, CoreSites.getCurrentSiteId());
|
|
|
|
this.pushObserver = CorePushNotificationsDelegate.on('receive').subscribe((notification) => {
|
|
// New notification received. If it's from current site, refresh the data.
|
|
if (!this.isCurrentView) {
|
|
this.pendingRefresh = true;
|
|
|
|
return;
|
|
}
|
|
|
|
if (!CoreUtils.isTrueOrOne(notification.notif) || !CoreSites.isCurrentSite(notification.site)) {
|
|
return;
|
|
}
|
|
|
|
this.refreshNotifications();
|
|
});
|
|
|
|
this.readObserver = CoreEvents.on(AddonNotificationsProvider.READ_CHANGED_EVENT, (data) => {
|
|
if (!data.id) {
|
|
return;
|
|
}
|
|
|
|
const notification = this.notifications.items.find((notification) => notification.id === data.id);
|
|
if (!notification) {
|
|
return;
|
|
}
|
|
|
|
notification.read = true;
|
|
notification.timeread = data.time;
|
|
this.loadMarkAllAsReadButton();
|
|
});
|
|
|
|
const deepLinkManager = new CoreMainMenuDeepLinkManager();
|
|
deepLinkManager.treatLink();
|
|
}
|
|
|
|
/**
|
|
* Check if the app has permission to display notifications.
|
|
*/
|
|
protected async checkPermission(): Promise<void> {
|
|
this.permissionWarningHidden = !!(await CoreConfig.get(CoreConstants.DONT_SHOW_NOTIFICATIONS_PERMISSION_WARNING, 0));
|
|
this.hasNotificationsPermission = await CoreLocalNotifications.hasNotificationsPermission();
|
|
}
|
|
|
|
/**
|
|
* Convenience function to get notifications. Gets unread notifications first.
|
|
*
|
|
* @param reload Whether to reload the list or load the next page.
|
|
*/
|
|
protected async fetchNotifications(reload: boolean): Promise<void> {
|
|
reload
|
|
? await this.notifications.reload()
|
|
: await this.notifications.load();
|
|
|
|
this.fetchMoreNotificationsFailed = false;
|
|
this.loadMarkAllAsReadButton();
|
|
}
|
|
|
|
/**
|
|
* Obtain the initial batch of notifications.
|
|
*/
|
|
private async fetchInitialNotifications(): Promise<void> {
|
|
try {
|
|
await this.fetchNotifications(true);
|
|
} catch (error) {
|
|
CoreDomUtils.showErrorModalDefault(error, 'Error loading notifications');
|
|
|
|
this.notifications.reset();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Load a new batch of Notifications.
|
|
*
|
|
* @param complete Completion callback.
|
|
*/
|
|
async fetchMoreNotifications(complete: () => void): Promise<void> {
|
|
try {
|
|
await this.fetchNotifications(false);
|
|
} catch (error) {
|
|
CoreDomUtils.showErrorModalDefault(error, 'Error loading more notifications');
|
|
|
|
this.fetchMoreNotificationsFailed = true;
|
|
}
|
|
|
|
complete();
|
|
}
|
|
|
|
/**
|
|
* Mark all notifications as read.
|
|
*
|
|
* @returns Promise resolved when done.
|
|
*/
|
|
async markAllNotificationsAsRead(): Promise<void> {
|
|
this.loadingMarkAllNotificationsAsRead = true;
|
|
|
|
await CoreUtils.ignoreErrors(AddonNotifications.markAllNotificationsAsRead());
|
|
|
|
CoreEvents.trigger(AddonNotificationsProvider.READ_CHANGED_EVENT, {
|
|
time: CoreTimeUtils.timestamp(),
|
|
}, CoreSites.getCurrentSiteId());
|
|
|
|
await this.refreshNotifications();
|
|
}
|
|
|
|
/**
|
|
* Load mark all notifications as read button.
|
|
*
|
|
* @returns Promise resolved when done.
|
|
*/
|
|
protected async loadMarkAllAsReadButton(): Promise<void> {
|
|
// Check if mark all as read should be displayed (there are unread notifications).
|
|
try {
|
|
this.loadingMarkAllNotificationsAsRead = true;
|
|
|
|
const unreadCountData = await AddonNotifications.getUnreadNotificationsCount();
|
|
|
|
this.canMarkAllNotificationsAsRead = unreadCountData.count > 0;
|
|
} finally {
|
|
this.loadingMarkAllNotificationsAsRead = false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Refresh notifications.
|
|
*
|
|
* @param refresher Refresher.
|
|
*/
|
|
async refreshNotifications(refresher?: HTMLIonRefresherElement): Promise<void> {
|
|
await CoreUtils.ignoreErrors(AddonNotifications.invalidateNotificationsList());
|
|
await CoreUtils.ignoreErrors(this.fetchNotifications(true));
|
|
|
|
refresher?.complete();
|
|
}
|
|
|
|
/**
|
|
* Open notification settings.
|
|
*/
|
|
openSettings(): void {
|
|
CoreLocalNotifications.openNotificationSettings();
|
|
}
|
|
|
|
/**
|
|
* Hide permission warning.
|
|
*/
|
|
hidePermissionWarning(): void {
|
|
CoreConfig.set(CoreConstants.DONT_SHOW_NOTIFICATIONS_PERMISSION_WARNING, 1);
|
|
this.permissionWarningHidden = true;
|
|
}
|
|
|
|
/**
|
|
* User entered the page.
|
|
*/
|
|
ionViewDidEnter(): void {
|
|
this.isCurrentView = true;
|
|
|
|
if (!this.pendingRefresh) {
|
|
return;
|
|
}
|
|
|
|
this.pendingRefresh = false;
|
|
|
|
this.refreshNotifications();
|
|
}
|
|
|
|
/**
|
|
* User left the page.
|
|
*/
|
|
ionViewDidLeave(): void {
|
|
this.isCurrentView = false;
|
|
}
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
ngOnDestroy(): void {
|
|
this.cronObserver?.off();
|
|
this.readObserver?.off();
|
|
this.pushObserver?.unsubscribe();
|
|
this.notifications?.destroy();
|
|
this.appResumeSubscription?.unsubscribe();
|
|
}
|
|
|
|
}
|