250 lines
15 KiB
HTML
250 lines
15 KiB
HTML
<ion-header>
|
|
<ion-toolbar>
|
|
<ion-buttons slot="start">
|
|
<ion-back-button [attr.aria-label]="'core.back' | translate"></ion-back-button>
|
|
</ion-buttons>
|
|
<ion-title>{{ 'addon.mod_lesson.detailedstats' | translate }}</ion-title>
|
|
</ion-toolbar>
|
|
</ion-header>
|
|
<ion-content>
|
|
<ion-refresher slot="fixed" [disabled]="!loaded" (ionRefresh)="doRefresh($event.target)">
|
|
<ion-refresher-content pullingText="{{ 'core.pulltorefresh' | translate }}"></ion-refresher-content>
|
|
</ion-refresher>
|
|
|
|
<core-loading [hideUntil]="loaded">
|
|
<div *ngIf="student">
|
|
<!-- Student data. -->
|
|
<ion-item class="ion-text-wrap" core-user-link [userId]="student.id" [courseId]="courseId" [title]="student.fullname">
|
|
<core-user-avatar [user]="student" slot="start" [userId]="student.id" [courseId]="courseId">
|
|
</core-user-avatar>
|
|
<ion-label>
|
|
<h2>{{student.fullname}}</h2>
|
|
<core-progress-bar [progress]="student.bestgrade"></core-progress-bar>
|
|
</ion-label>
|
|
</ion-item>
|
|
|
|
<!-- Retake selector if there is more than one retake. -->
|
|
<ion-item class="ion-text-wrap" *ngIf="student.attempts && student.attempts.length > 1">
|
|
<ion-label id="addon-mod_lesson-retakeslabel">{{ 'addon.mod_lesson.attemptheader' | translate }}</ion-label>
|
|
<ion-select [(ngModel)]="selectedRetake" (ionChange)="changeRetake(selectedRetake!)"
|
|
aria-labelledby="addon-mod_lesson-retakeslabel" interface="action-sheet">
|
|
<ion-select-option *ngFor="let retake of student.attempts" [value]="retake.try">
|
|
{{retake.label}}
|
|
</ion-select-option>
|
|
</ion-select>
|
|
</ion-item>
|
|
|
|
<!-- Retake stats. -->
|
|
<ion-list *ngIf="retake && retake.userstats && retake.userstats.gradeinfo" class="addon-mod_lesson-userstats">
|
|
<ion-item>
|
|
<ion-label class="ion-text-wrap">
|
|
<ion-grid class="ion-no-padding">
|
|
<ion-row>
|
|
<ion-col>
|
|
<h3 class="item-heading">{{ 'addon.mod_lesson.grade' | translate }}</h3>
|
|
<p>{{ 'core.percentagenumber' | translate:{$a: retake.userstats.grade} }}</p>
|
|
</ion-col>
|
|
|
|
<ion-col>
|
|
<h3 class="item-heading">{{ 'addon.mod_lesson.rawgrade' | translate }}</h3>
|
|
<p>{{ retake.userstats.gradeinfo.earned }} / {{ retake.userstats.gradeinfo.total }}</p>
|
|
</ion-col>
|
|
</ion-row>
|
|
</ion-grid>
|
|
</ion-label>
|
|
</ion-item>
|
|
<ion-item class="ion-text-wrap">
|
|
<ion-label>
|
|
<h3 class="item-heading">{{ 'addon.mod_lesson.timetaken' | translate }}</h3>
|
|
<p>{{ timeTakenReadable }}</p>
|
|
</ion-label>
|
|
</ion-item>
|
|
<ion-item class="ion-text-wrap">
|
|
<ion-label>
|
|
<h3 class="item-heading">{{ 'addon.mod_lesson.completed' | translate }}</h3>
|
|
<p>{{ retake.userstats.completed * 1000 | coreFormatDate }}</p>
|
|
</ion-label>
|
|
</ion-item>
|
|
</ion-list>
|
|
|
|
<!-- Not completed, no stats. -->
|
|
<ion-item class="ion-text-wrap" *ngIf="retake && (!retake.userstats || !retake.userstats.gradeinfo)">
|
|
<ion-label>{{ 'addon.mod_lesson.notcompleted' | translate }}</ion-label>
|
|
</ion-item>
|
|
|
|
<!-- Pages. -->
|
|
<ng-container *ngIf="retake">
|
|
<!-- The "text-dimmed" class does nothing, but the same goes for the "dimmed" class in Moodle. -->
|
|
<ion-card *ngFor="let page of retake.answerpages" class="addon-mod_lesson-answerpage"
|
|
[ngClass]="{'text-dimmed': page.grayout}">
|
|
<ion-card-header class="ion-text-wrap">
|
|
<ion-card-title>{{page.qtype}}: {{page.title}}</ion-card-title>
|
|
</ion-card-header>
|
|
<ion-item class="ion-text-wrap" lines="none">
|
|
<ion-label>
|
|
<h3 class="item-heading">{{ 'addon.mod_lesson.question' | translate }}</h3>
|
|
<p>
|
|
<core-format-text [component]="component" [componentId]="lesson?.coursemodule" [maxHeight]="50"
|
|
[text]="page.contents" contextLevel="module" [contextInstanceId]="lesson?.coursemodule"
|
|
[courseId]="courseId">
|
|
</core-format-text>
|
|
</p>
|
|
</ion-label>
|
|
</ion-item>
|
|
<ion-item class="ion-text-wrap" lines="none">
|
|
<ion-label>
|
|
<h3 class="item-heading">{{ 'addon.mod_lesson.answer' | translate }}</h3>
|
|
</ion-label>
|
|
</ion-item>
|
|
<ion-item class="ion-text-wrap" lines="none"
|
|
*ngIf="!page.answerdata || !page.answerdata.answers || !page.answerdata.answers.length">
|
|
<ion-label>
|
|
<p>{{ 'addon.mod_lesson.didnotanswerquestion' | translate }}</p>
|
|
</ion-label>
|
|
</ion-item>
|
|
<div *ngIf="page.answerdata && page.answerdata.answers && page.answerdata.answers.length"
|
|
class="addon-mod_lesson-answer">
|
|
<ng-container *ngFor="let answer of page.answerdata.answers">
|
|
<ion-item lines="none" *ngIf="page.isContent">
|
|
<ion-label class="ion-text-wrap">
|
|
<ion-grid class="ion-no-padding">
|
|
<!-- Content page, display a button and the content. -->
|
|
<ion-row>
|
|
<ion-col>
|
|
<ion-button expand="block" class="ion-text-wrap" color="light" [disabled]="true">
|
|
{{ answer[0].buttonText }}
|
|
</ion-button>
|
|
</ion-col>
|
|
<ion-col>
|
|
<p [innerHTML]="answer[0].content"></p>
|
|
</ion-col>
|
|
</ion-row>
|
|
</ion-grid>
|
|
</ion-label>
|
|
</ion-item>
|
|
|
|
<ng-container *ngIf="page.isQuestion">
|
|
<!-- Question page, show the right input for the answer. -->
|
|
|
|
<!-- Truefalse or matching. -->
|
|
<ion-item class="ion-text-wrap" *ngIf="answer[0].isCheckbox"
|
|
[ngClass]="{'addon-mod_lesson-highlight': answer[0].highlight}">
|
|
<ion-label>
|
|
<p>
|
|
<core-format-text [component]="component" [componentId]="lesson?.coursemodule"
|
|
[text]="answer[0].content" contextLevel="module"
|
|
[contextInstanceId]="lesson?.coursemodule" [courseId]="courseId">
|
|
</core-format-text>
|
|
</p>
|
|
<ion-badge *ngIf="answer[1]" color="dark">
|
|
<core-format-text [component]="component" [componentId]="lesson?.coursemodule"
|
|
[text]="answer[1]" contextLevel="module" [contextInstanceId]="lesson?.coursemodule"
|
|
[courseId]="courseId">
|
|
</core-format-text>
|
|
</ion-badge>
|
|
</ion-label>
|
|
<ion-checkbox [attr.name]="answer[0].name" [ngModel]="answer[0].checked" [disabled]="true"
|
|
slot="end">
|
|
</ion-checkbox>
|
|
</ion-item>
|
|
|
|
<!-- Short answer or numeric. -->
|
|
<ion-item class="ion-text-wrap" *ngIf="answer[0].isText" lines="none">
|
|
<ion-label>
|
|
<p>{{ answer[0].value }}</p>
|
|
<ion-badge *ngIf="answer[1]" color="dark">
|
|
<core-format-text [component]="component" [componentId]="lesson?.coursemodule"
|
|
[text]="answer[1]" contextLevel="module" [contextInstanceId]="lesson?.coursemodule"
|
|
[courseId]="courseId">
|
|
</core-format-text>
|
|
</ion-badge>
|
|
</ion-label>
|
|
</ion-item>
|
|
|
|
<!-- Matching. -->
|
|
<ion-item lines="none" *ngIf="answer[0].isSelect">
|
|
<ion-label class="ion-text-wrap">
|
|
<ion-grid class="ion-no-padding">
|
|
<ion-row>
|
|
<ion-col>
|
|
<p>
|
|
<core-format-text [component]="component" [componentId]="lesson?.coursemodule"
|
|
[text]=" answer[0].content" contextLevel="module"
|
|
[contextInstanceId]="lesson?.coursemodule" [courseId]="courseId">
|
|
</core-format-text>
|
|
</p>
|
|
</ion-col>
|
|
<ion-col>
|
|
<p>{{answer[0].value}}</p>
|
|
<ion-badge *ngIf="answer[1]" color="dark">
|
|
<core-format-text [component]="component" [componentId]="lesson?.coursemodule"
|
|
[text]="answer[1]" contextLevel="module"
|
|
[contextInstanceId]="lesson?.coursemodule" [courseId]="courseId">
|
|
</core-format-text>
|
|
</ion-badge>
|
|
</ion-col>
|
|
</ion-row>
|
|
</ion-grid>
|
|
</ion-label>
|
|
</ion-item>
|
|
|
|
<!-- Essay or couldn't determine. -->
|
|
<ion-item class="ion-text-wrap" lines="none"
|
|
*ngIf="!answer[0].isCheckbox && !answer[0].isText && !answer[0].isSelect">
|
|
<ion-label>
|
|
<p>
|
|
<core-format-text [component]="component" [componentId]="lesson?.coursemodule"
|
|
[text]="answer[0]" contextLevel="module" [contextInstanceId]="lesson?.coursemodule"
|
|
[courseId]="courseId">
|
|
</core-format-text>
|
|
</p>
|
|
<ion-badge *ngIf="answer[1]" color="dark">
|
|
<core-format-text [component]="component" [componentId]="lesson?.coursemodule"
|
|
[text]="answer[1]" contextLevel="module" [contextInstanceId]="lesson?.coursemodule"
|
|
[courseId]="courseId">
|
|
</core-format-text>
|
|
</ion-badge>
|
|
</ion-label>
|
|
</ion-item>
|
|
</ng-container>
|
|
|
|
<ion-item class="ion-text-wrap" *ngIf="!page.isContent && !page.isQuestion" lines="none">
|
|
<!-- Another page (end of branch, ...). -->
|
|
<ion-label>
|
|
<p>
|
|
<core-format-text [component]="component" [componentId]="lesson?.coursemodule"
|
|
[text]="answer[0]" contextLevel="module" [contextInstanceId]="lesson?.coursemodule"
|
|
[courseId]="courseId">
|
|
</core-format-text>
|
|
</p>
|
|
<ion-badge *ngIf="answer[1]" color="dark">
|
|
<core-format-text [component]="component" [componentId]="lesson?.coursemodule"
|
|
[text]="answer[1]" contextLevel="module" [contextInstanceId]="lesson?.coursemodule"
|
|
[courseId]="courseId">
|
|
</core-format-text>
|
|
</ion-badge>
|
|
</ion-label>
|
|
</ion-item>
|
|
</ng-container>
|
|
|
|
<ion-item class="ion-text-wrap" *ngIf="page.answerdata.response" lines="none">
|
|
<ion-label>
|
|
<h3 class="item-heading">{{ 'addon.mod_lesson.response' | translate }}</h3>
|
|
<p>
|
|
<core-format-text [component]="component" [componentId]="lesson?.coursemodule"
|
|
[text]="page.answerdata.response" contextLevel="module"
|
|
[contextInstanceId]="lesson?.coursemodule" [courseId]="courseId">
|
|
</core-format-text>
|
|
</p>
|
|
</ion-label>
|
|
</ion-item>
|
|
<ion-item class="ion-text-wrap" *ngIf="page.answerdata.score">
|
|
<ion-label><p>{{page.answerdata.score}}</p></ion-label>
|
|
</ion-item>
|
|
</div>
|
|
</ion-card>
|
|
</ng-container>
|
|
</div>
|
|
</core-loading>
|
|
</ion-content>
|