524 lines
16 KiB
TypeScript
524 lines
16 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 { Component, Input, Optional, ViewChild, OnInit, OnDestroy } from '@angular/core';
|
|
import { CoreError } from '@classes/errors/error';
|
|
import { CoreTabsComponent } from '@components/tabs/tabs';
|
|
import { CoreCourseModuleMainActivityComponent } from '@features/course/classes/main-activity-component';
|
|
import { CoreCourseContentsPage } from '@features/course/pages/contents/contents';
|
|
import { IonContent } from '@ionic/angular';
|
|
import { CoreGroupInfo, CoreGroups } from '@services/groups';
|
|
import { CoreNavigator } from '@services/navigator';
|
|
import { CoreTextUtils } from '@services/utils/text';
|
|
import { CoreTimeUtils } from '@services/utils/time';
|
|
import { CoreUtils } from '@services/utils/utils';
|
|
import { CoreEventObserver, CoreEvents } from '@singletons/events';
|
|
import {
|
|
AddonModFeedback,
|
|
AddonModFeedbackGetFeedbackAccessInformationWSResponse,
|
|
AddonModFeedbackProvider,
|
|
AddonModFeedbackWSFeedback,
|
|
AddonModFeedbackWSItem,
|
|
} from '../../services/feedback';
|
|
import { AddonModFeedbackOffline } from '../../services/feedback-offline';
|
|
import {
|
|
AddonModFeedbackAutoSyncData,
|
|
AddonModFeedbackSync,
|
|
AddonModFeedbackSyncProvider,
|
|
AddonModFeedbackSyncResult,
|
|
} from '../../services/feedback-sync';
|
|
import { AddonModFeedbackModuleHandlerService } from '../../services/handlers/module';
|
|
import { AddonModFeedbackPrefetchHandler } from '../../services/handlers/prefetch';
|
|
|
|
/**
|
|
* Component that displays a feedback index page.
|
|
*/
|
|
@Component({
|
|
selector: 'addon-mod-feedback-index',
|
|
templateUrl: 'addon-mod-feedback-index.html',
|
|
})
|
|
export class AddonModFeedbackIndexComponent extends CoreCourseModuleMainActivityComponent implements OnInit, OnDestroy {
|
|
|
|
@ViewChild(CoreTabsComponent) tabsComponent?: CoreTabsComponent;
|
|
|
|
@Input() tab = 'overview';
|
|
@Input() group = 0;
|
|
|
|
component = AddonModFeedbackProvider.COMPONENT;
|
|
pluginName = 'feedback';
|
|
feedback?: AddonModFeedbackWSFeedback;
|
|
goPage?: number;
|
|
items: AddonModFeedbackItem[] = [];
|
|
warning?: string;
|
|
showAnalysis = false;
|
|
tabsReady = false;
|
|
firstSelectedTab?: number;
|
|
access?: AddonModFeedbackGetFeedbackAccessInformationWSResponse;
|
|
completedCount = 0;
|
|
itemsCount = 0;
|
|
groupInfo?: CoreGroupInfo;
|
|
|
|
overview = {
|
|
timeopen: 0,
|
|
openTimeReadable: '',
|
|
timeclose: 0,
|
|
closeTimeReadable: '',
|
|
};
|
|
|
|
tabsLoaded = {
|
|
overview: false,
|
|
analysis: false,
|
|
};
|
|
|
|
protected submitObserver: CoreEventObserver;
|
|
protected syncEventName = AddonModFeedbackSyncProvider.AUTO_SYNCED;
|
|
protected checkCompletionAfterLog = false;
|
|
|
|
constructor(
|
|
protected content?: IonContent,
|
|
@Optional() courseContentsPage?: CoreCourseContentsPage,
|
|
) {
|
|
super('AddonModLessonIndexComponent', content, courseContentsPage);
|
|
|
|
// Listen for form submit events.
|
|
this.submitObserver = CoreEvents.on(AddonModFeedbackProvider.FORM_SUBMITTED, async (data) => {
|
|
if (!this.feedback || data.feedbackId != this.feedback.id) {
|
|
return;
|
|
}
|
|
|
|
this.tabsLoaded.analysis = false;
|
|
this.tabsLoaded.overview = false;
|
|
this.showLoading = true;
|
|
|
|
// Prefetch data if needed.
|
|
if (!data.offline && this.isPrefetched()) {
|
|
await CoreUtils.ignoreErrors(AddonModFeedbackSync.prefetchAfterUpdate(
|
|
AddonModFeedbackPrefetchHandler.instance,
|
|
this.module,
|
|
this.courseId,
|
|
));
|
|
}
|
|
|
|
// Load the right tab.
|
|
if (data.tab != this.tab) {
|
|
this.tabChanged(data.tab);
|
|
} else {
|
|
this.loadContent(true);
|
|
}
|
|
}, this.siteId);
|
|
}
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
async ngOnInit(): Promise<void> {
|
|
super.ngOnInit();
|
|
|
|
try {
|
|
await this.loadContent(false, true);
|
|
} finally {
|
|
this.tabsReady = true;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
protected async logActivity(): Promise<void> {
|
|
if (!this.feedback) {
|
|
return; // Shouldn't happen.
|
|
}
|
|
|
|
await AddonModFeedback.logView(this.feedback.id);
|
|
|
|
this.callAnalyticsLogEvent();
|
|
}
|
|
|
|
/**
|
|
* Call analytics.
|
|
*/
|
|
protected callAnalyticsLogEvent(): void {
|
|
this.analyticsLogEvent('mod_feedback_view_feedback', {
|
|
url: this.tab === 'analysis' ? `/mod/feedback/analysis.php?id=${this.module.id}` : undefined,
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
protected async invalidateContent(): Promise<void> {
|
|
const promises: Promise<void>[] = [];
|
|
|
|
promises.push(AddonModFeedback.invalidateFeedbackData(this.courseId));
|
|
if (this.feedback) {
|
|
promises.push(AddonModFeedback.invalidateFeedbackAccessInformationData(this.feedback.id));
|
|
promises.push(AddonModFeedback.invalidateAnalysisData(this.feedback.id));
|
|
promises.push(CoreGroups.invalidateActivityAllowedGroups(this.feedback.coursemodule));
|
|
promises.push(CoreGroups.invalidateActivityGroupMode(this.feedback.coursemodule));
|
|
promises.push(AddonModFeedback.invalidateResumePageData(this.feedback.id));
|
|
}
|
|
|
|
this.tabsLoaded.analysis = false;
|
|
this.tabsLoaded.overview = false;
|
|
|
|
await Promise.all(promises);
|
|
}
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
protected isRefreshSyncNeeded(syncEventData: AddonModFeedbackAutoSyncData): boolean {
|
|
if (this.feedback && syncEventData.feedbackId == this.feedback.id) {
|
|
// Refresh the data.
|
|
this.content?.scrollToTop();
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
protected async fetchContent(refresh?: boolean, sync = false, showErrors = false): Promise<void> {
|
|
try {
|
|
this.feedback = await AddonModFeedback.getFeedback(this.courseId, this.module.id);
|
|
|
|
this.description = this.feedback.intro;
|
|
this.dataRetrieved.emit(this.feedback);
|
|
|
|
if (sync) {
|
|
// Try to synchronize the feedback.
|
|
await this.syncActivity(showErrors);
|
|
}
|
|
|
|
// Check if there are answers stored in offline.
|
|
this.access = await AddonModFeedback.getFeedbackAccessInformation(this.feedback.id, { cmId: this.module.id });
|
|
|
|
this.showAnalysis = (this.access.canviewreports || this.access.canviewanalysis) && !this.access.isempty;
|
|
this.firstSelectedTab = 0;
|
|
if (!this.showAnalysis) {
|
|
this.tab = 'overview';
|
|
}
|
|
|
|
if (this.tab == 'analysis') {
|
|
this.firstSelectedTab = 1;
|
|
|
|
return await this.fetchFeedbackAnalysisData();
|
|
}
|
|
|
|
await this.fetchFeedbackOverviewData();
|
|
} finally {
|
|
if (this.feedback) {
|
|
// Check if there are responses stored in offline.
|
|
this.hasOffline = await AddonModFeedbackOffline.hasFeedbackOfflineData(this.feedback.id);
|
|
}
|
|
|
|
if (this.tabsReady) {
|
|
// Make sure the right tab is selected.
|
|
this.tabsComponent?.selectTab(this.tab || 'overview');
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Convenience function to get feedback overview data.
|
|
*
|
|
* @returns Resolved when done.
|
|
*/
|
|
protected async fetchFeedbackOverviewData(): Promise<void> {
|
|
const promises: Promise<void>[] = [];
|
|
|
|
if (this.access!.cancomplete && this.access!.cansubmit && this.access!.isopen) {
|
|
promises.push(AddonModFeedback.getResumePage(this.feedback!.id, { cmId: this.module.id }).then((goPage) => {
|
|
this.goPage = goPage > 0 ? goPage : undefined;
|
|
|
|
return;
|
|
}));
|
|
}
|
|
|
|
if (this.access!.canedititems) {
|
|
this.overview.timeopen = (this.feedback!.timeopen || 0) * 1000;
|
|
this.overview.openTimeReadable = this.overview.timeopen ? CoreTimeUtils.userDate(this.overview.timeopen) : '';
|
|
this.overview.timeclose = (this.feedback!.timeclose || 0) * 1000;
|
|
this.overview.closeTimeReadable = this.overview.timeclose ? CoreTimeUtils.userDate(this.overview.timeclose) : '';
|
|
}
|
|
if (this.access!.canviewanalysis) {
|
|
// Get groups (only for teachers).
|
|
promises.push(this.fetchGroupInfo(this.module.id));
|
|
}
|
|
|
|
try {
|
|
await Promise.all(promises);
|
|
} finally {
|
|
this.tabsLoaded.overview = true;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Convenience function to get feedback analysis data.
|
|
*
|
|
* @returns Resolved when done.
|
|
*/
|
|
protected async fetchFeedbackAnalysisData(): Promise<void> {
|
|
try {
|
|
if (this.access!.canviewanalysis) {
|
|
// Get groups (only for teachers).
|
|
await this.fetchGroupInfo(this.module.id);
|
|
} else {
|
|
this.tabChanged('overview');
|
|
}
|
|
|
|
} finally {
|
|
this.tabsLoaded.analysis = true;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Fetch Group info data.
|
|
*
|
|
* @param cmId Course module ID.
|
|
* @returns Resolved when done.
|
|
*/
|
|
protected async fetchGroupInfo(cmId: number): Promise<void> {
|
|
this.groupInfo = await CoreGroups.getActivityGroupInfo(cmId);
|
|
|
|
await this.setGroup(CoreGroups.validateGroupId(this.group, this.groupInfo));
|
|
}
|
|
|
|
/**
|
|
* Parse the analysis info to show the info correctly formatted.
|
|
*
|
|
* @param item Item to parse.
|
|
* @returns Parsed item.
|
|
*/
|
|
protected parseAnalysisInfo(item: AddonModFeedbackItem): AddonModFeedbackItem {
|
|
switch (item.typ) {
|
|
case 'numeric':
|
|
item.average = item.data.reduce((prev, current) => prev + Number(current), 0) / item.data.length;
|
|
item.templateName = 'numeric';
|
|
break;
|
|
|
|
case 'info':
|
|
item.data = <string[]> item.data.map((dataItem) => {
|
|
const parsed = <Record<string, string>> CoreTextUtils.parseJSON(dataItem);
|
|
|
|
return parsed.show !== undefined ? parsed.show : false;
|
|
}).filter((dataItem) => dataItem); // Filter false entries.
|
|
|
|
case 'textfield':
|
|
case 'textarea':
|
|
item.templateName = 'list';
|
|
break;
|
|
|
|
case 'multichoicerated':
|
|
case 'multichoice': {
|
|
const parsedData = <Record<string, string | number>[]> item.data.map((dataItem) => {
|
|
const parsed = <Record<string, string | number>> CoreTextUtils.parseJSON(dataItem);
|
|
|
|
return parsed.answertext !== undefined ? parsed : false;
|
|
}).filter((dataItem) => dataItem); // Filter false entries.
|
|
|
|
// Format labels.
|
|
item.labels = parsedData.map((dataItem) => {
|
|
dataItem.quotient = (<number> dataItem.quotient * 100).toFixed(2);
|
|
let label = '';
|
|
|
|
if (dataItem.value !== undefined) {
|
|
label = '(' + dataItem.value + ') ';
|
|
}
|
|
label += dataItem.answertext;
|
|
label += Number(dataItem.quotient) > 0 ? ' (' + dataItem.quotient + '%)' : '';
|
|
|
|
return label;
|
|
});
|
|
|
|
item.chartData = parsedData.map((dataItem) => Number(dataItem.answercount));
|
|
|
|
if (item.typ === 'multichoicerated') {
|
|
item.average = parsedData.reduce((prev, current) => prev + Number(current.avg), 0.0);
|
|
}
|
|
|
|
const subtype = item.presentation.charAt(0);
|
|
|
|
// Display bar chart if there are no answers to avoid division by 0 error.
|
|
const single = subtype !== 'c' && item.chartData.some((count) => count > 0);
|
|
item.chartType = single ? 'doughnut' : 'bar';
|
|
item.templateName = 'chart';
|
|
break;
|
|
}
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return item;
|
|
}
|
|
|
|
/**
|
|
* Function to go to the questions form.
|
|
*
|
|
* @param preview Preview or edit the form.
|
|
*/
|
|
gotoAnswerQuestions(preview: boolean = false): void {
|
|
CoreNavigator.navigateToSitePath(
|
|
AddonModFeedbackModuleHandlerService.PAGE_NAME + `/${this.courseId}/${this.module.id}/form`,
|
|
{
|
|
params: {
|
|
preview,
|
|
fromIndex: true,
|
|
},
|
|
},
|
|
);
|
|
}
|
|
|
|
/**
|
|
* User entered the page that contains the component.
|
|
*/
|
|
ionViewDidEnter(): void {
|
|
super.ionViewDidEnter();
|
|
|
|
this.tabsComponent?.ionViewDidEnter();
|
|
}
|
|
|
|
/**
|
|
* User left the page that contains the component.
|
|
*/
|
|
ionViewDidLeave(): void {
|
|
super.ionViewDidLeave();
|
|
|
|
this.tabsComponent?.ionViewDidLeave();
|
|
}
|
|
|
|
/**
|
|
* Open non respondents page.
|
|
*/
|
|
openNonRespondents(): void {
|
|
CoreNavigator.navigateToSitePath(
|
|
AddonModFeedbackModuleHandlerService.PAGE_NAME + `/${this.courseId}/${this.module.id}/nonrespondents`,
|
|
{
|
|
params: {
|
|
group: this.group,
|
|
},
|
|
},
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Open attempts page.
|
|
*/
|
|
openAttempts(): void {
|
|
if (!this.access!.canviewreports || this.completedCount <= 0) {
|
|
return;
|
|
}
|
|
|
|
CoreNavigator.navigateToSitePath(
|
|
AddonModFeedbackModuleHandlerService.PAGE_NAME + `/${this.courseId}/${this.module.id}/attempts`,
|
|
{
|
|
params: {
|
|
group: this.group,
|
|
},
|
|
},
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Tab changed, fetch content again.
|
|
*
|
|
* @param tabName New tab name.
|
|
*/
|
|
tabChanged(tabName: string): void {
|
|
const tabHasChanged = this.tab !== undefined && this.tab !== tabName;
|
|
this.tab = tabName;
|
|
|
|
if (!this.tabsLoaded[this.tab]) {
|
|
this.loadContent(false, false, true);
|
|
}
|
|
|
|
if (tabHasChanged) {
|
|
this.callAnalyticsLogEvent();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Set group to see the analysis.
|
|
*
|
|
* @param groupId Group ID.
|
|
* @returns Resolved when done.
|
|
*/
|
|
async setGroup(groupId: number): Promise<void> {
|
|
this.group = groupId;
|
|
|
|
const analysis = await AddonModFeedback.getAnalysis(this.feedback!.id, { groupId, cmId: this.module.id });
|
|
|
|
this.completedCount = analysis.completedcount;
|
|
this.itemsCount = analysis.itemscount;
|
|
|
|
if (this.tab == 'analysis') {
|
|
let num = 1;
|
|
|
|
this.items = <AddonModFeedbackItem[]> analysis.itemsdata.map((itemData) => {
|
|
const item: AddonModFeedbackItem = Object.assign(itemData.item, {
|
|
data: itemData.data,
|
|
num: num++,
|
|
});
|
|
|
|
// Move data inside item.
|
|
if (item.data && item.data.length) {
|
|
return this.parseAnalysisInfo(item);
|
|
}
|
|
|
|
return false;
|
|
}).filter((item) => item);
|
|
|
|
this.warning = '';
|
|
if (analysis.warnings?.length) {
|
|
const warning = analysis.warnings.find((warning) => warning.warningcode == 'insufficientresponsesforthisgroup');
|
|
this.warning = warning?.message;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
protected sync(): Promise<AddonModFeedbackSyncResult> {
|
|
if (!this.feedback) {
|
|
throw new CoreError('Cannot sync without a feedback.');
|
|
}
|
|
|
|
return AddonModFeedbackSync.syncFeedback(this.feedback.id);
|
|
}
|
|
|
|
/**
|
|
* @inheritdoc
|
|
*/
|
|
ngOnDestroy(): void {
|
|
super.ngOnDestroy();
|
|
this.submitObserver?.off();
|
|
}
|
|
|
|
}
|
|
|
|
type AddonModFeedbackItem = AddonModFeedbackWSItem & {
|
|
data: string[];
|
|
num: number;
|
|
templateName?: string;
|
|
average?: number;
|
|
labels?: string[];
|
|
chartData?: number[];
|
|
chartType?: string;
|
|
};
|