MOBILE-3651 quiz: Implement entry page
This commit is contained in:
		
							parent
							
								
									7698fa673d
								
							
						
					
					
						commit
						e565df9ea4
					
				| @ -64,7 +64,7 @@ | |||||||
|                         </ion-item> |                         </ion-item> | ||||||
|                         <ion-button expand="block" type="submit"> |                         <ion-button expand="block" type="submit"> | ||||||
|                             {{ 'addon.mod_lesson.continue' | translate }} |                             {{ 'addon.mod_lesson.continue' | translate }} | ||||||
|                             <core-icon slot="end" name="fas-chevron-right"></core-icon> |                             <ion-icon slot="end" name="fas-chevron-right"></ion-icon> | ||||||
|                         </ion-button> |                         </ion-button> | ||||||
|                         <!-- Remove this once Ionic fixes this bug: https://github.com/ionic-team/ionic-framework/issues/19368 --> |                         <!-- Remove this once Ionic fixes this bug: https://github.com/ionic-team/ionic-framework/issues/19368 --> | ||||||
|                         <input type="submit" class="core-submit-hidden-enter" /> |                         <input type="submit" class="core-submit-hidden-enter" /> | ||||||
|  | |||||||
| @ -4,7 +4,7 @@ | |||||||
| 
 | 
 | ||||||
|         <ion-buttons slot="end"> |         <ion-buttons slot="end"> | ||||||
|             <ion-button (click)="closeModal()" [attr.aria-label]="'core.close' | translate"> |             <ion-button (click)="closeModal()" [attr.aria-label]="'core.close' | translate"> | ||||||
|                 <core-icon slot="icon-only" name="fas-times"></core-icon> |                 <ion-icon slot="icon-only" name="fas-times"></ion-icon> | ||||||
|             </ion-button> |             </ion-button> | ||||||
|         </ion-buttons> |         </ion-buttons> | ||||||
|     </ion-toolbar> |     </ion-toolbar> | ||||||
|  | |||||||
| @ -4,7 +4,7 @@ | |||||||
| 
 | 
 | ||||||
|         <ion-buttons slot="end"> |         <ion-buttons slot="end"> | ||||||
|             <ion-button (click)="closeModal()" [attr.aria-label]="'core.close' | translate"> |             <ion-button (click)="closeModal()" [attr.aria-label]="'core.close' | translate"> | ||||||
|                 <core-icon slot="icon-only" name="fas-times"></core-icon> |                 <ion-icon slot="icon-only" name="fas-times"></ion-icon> | ||||||
|             </ion-button> |             </ion-button> | ||||||
|         </ion-buttons> |         </ion-buttons> | ||||||
|     </ion-toolbar> |     </ion-toolbar> | ||||||
| @ -20,7 +20,7 @@ | |||||||
|         </ion-item> |         </ion-item> | ||||||
|         <ion-button expand="block" type="submit"> |         <ion-button expand="block" type="submit"> | ||||||
|             {{ 'addon.mod_lesson.continue' | translate }} |             {{ 'addon.mod_lesson.continue' | translate }} | ||||||
|             <core-icon slot="end" name="fas-chevron-right"></core-icon> |             <ion-icon slot="end" name="fas-chevron-right"></ion-icon> | ||||||
|         </ion-button> |         </ion-button> | ||||||
|         <!-- Remove this once Ionic fixes this bug: https://github.com/ionic-team/ionic-framework/issues/19368 --> |         <!-- Remove this once Ionic fixes this bug: https://github.com/ionic-team/ionic-framework/issues/19368 --> | ||||||
|         <input type="submit" class="core-submit-hidden-enter" /> |         <input type="submit" class="core-submit-hidden-enter" /> | ||||||
|  | |||||||
| @ -18,6 +18,7 @@ import { AddonModAssignModule } from './assign/assign.module'; | |||||||
| import { AddonModBookModule } from './book/book.module'; | import { AddonModBookModule } from './book/book.module'; | ||||||
| import { AddonModLessonModule } from './lesson/lesson.module'; | import { AddonModLessonModule } from './lesson/lesson.module'; | ||||||
| import { AddonModPageModule } from './page/page.module'; | import { AddonModPageModule } from './page/page.module'; | ||||||
|  | import { AddonModQuizModule } from './quiz/quiz.module'; | ||||||
| 
 | 
 | ||||||
| @NgModule({ | @NgModule({ | ||||||
|     declarations: [], |     declarations: [], | ||||||
| @ -26,6 +27,7 @@ import { AddonModPageModule } from './page/page.module'; | |||||||
|         AddonModBookModule, |         AddonModBookModule, | ||||||
|         AddonModLessonModule, |         AddonModLessonModule, | ||||||
|         AddonModPageModule, |         AddonModPageModule, | ||||||
|  |         AddonModQuizModule, | ||||||
|     ], |     ], | ||||||
|     providers: [], |     providers: [], | ||||||
|     exports: [], |     exports: [], | ||||||
|  | |||||||
							
								
								
									
										38
									
								
								src/addons/mod/quiz/components/components.module.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								src/addons/mod/quiz/components/components.module.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,38 @@ | |||||||
|  | // (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 { NgModule } from '@angular/core'; | ||||||
|  | 
 | ||||||
|  | import { CoreSharedModule } from '@/core/shared.module'; | ||||||
|  | import { CoreCourseComponentsModule } from '@features/course/components/components.module'; | ||||||
|  | import { AddonModQuizConnectionErrorComponent } from './connection-error/connection-error'; | ||||||
|  | import { AddonModQuizIndexComponent } from './index/index'; | ||||||
|  | 
 | ||||||
|  | @NgModule({ | ||||||
|  |     declarations: [ | ||||||
|  |         AddonModQuizIndexComponent, | ||||||
|  |         AddonModQuizConnectionErrorComponent, | ||||||
|  |     ], | ||||||
|  |     imports: [ | ||||||
|  |         CoreSharedModule, | ||||||
|  |         CoreCourseComponentsModule, | ||||||
|  |     ], | ||||||
|  |     providers: [ | ||||||
|  |     ], | ||||||
|  |     exports: [ | ||||||
|  |         AddonModQuizIndexComponent, | ||||||
|  |         AddonModQuizConnectionErrorComponent, | ||||||
|  |     ], | ||||||
|  | }) | ||||||
|  | export class AddonModQuizComponentsModule {} | ||||||
							
								
								
									
										199
									
								
								src/addons/mod/quiz/components/index/addon-mod-quiz-index.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										199
									
								
								src/addons/mod/quiz/components/index/addon-mod-quiz-index.html
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,199 @@ | |||||||
|  | <!-- Buttons to add to the header. --> | ||||||
|  | <core-navbar-buttons slot="end"> | ||||||
|  |     <core-context-menu> | ||||||
|  |         <core-context-menu-item *ngIf="externalUrl" [priority]="900" [content]="'core.openinbrowser' | translate" | ||||||
|  |             [href]="externalUrl" iconAction="fas-external-link-alt"> | ||||||
|  |         </core-context-menu-item> | ||||||
|  |         <core-context-menu-item *ngIf="description" [priority]="800" [content]="'core.moduleintro' | translate" | ||||||
|  |             (action)="expandDescription()" iconAction="fas-arrow-right"> | ||||||
|  |         </core-context-menu-item> | ||||||
|  |         <core-context-menu-item *ngIf="blog" [priority]="750" content="{{'addon.blog.blog' | translate}}" | ||||||
|  |             [iconAction]="'far-newspaper'" (action)="gotoBlog()"> | ||||||
|  |         </core-context-menu-item> | ||||||
|  |         <core-context-menu-item *ngIf="loaded && !hasOffline && isOnline" [priority]="700" [content]="'core.refresh' | translate" | ||||||
|  |             (action)="doRefresh(null, $event)" [iconAction]="refreshIcon" [closeOnClick]="false"> | ||||||
|  |         </core-context-menu-item> | ||||||
|  |         <core-context-menu-item *ngIf="loaded && hasOffline && isOnline"  [priority]="600" (action)="doRefresh(null, $event, true)" | ||||||
|  |             [content]="'core.settings.synchronizenow' | translate" [iconAction]="syncIcon" [closeOnClick]="false"> | ||||||
|  |         </core-context-menu-item> | ||||||
|  |         <core-context-menu-item *ngIf="prefetchStatusIcon" [priority]="500" [content]="prefetchText" (action)="prefetch($event)" | ||||||
|  |             [iconAction]="prefetchStatusIcon" [closeOnClick]="false"> | ||||||
|  |         </core-context-menu-item> | ||||||
|  |         <core-context-menu-item *ngIf="size" [priority]="400" [content]="'core.clearstoreddata' | translate:{$a: size}" | ||||||
|  |             iconDescription="fas-archive" (action)="removeFiles($event)" iconAction="fas-trash" [closeOnClick]="false"> | ||||||
|  |         </core-context-menu-item> | ||||||
|  |     </core-context-menu> | ||||||
|  | </core-navbar-buttons> | ||||||
|  | 
 | ||||||
|  | <!-- Content. --> | ||||||
|  | <core-loading [hideUntil]="loaded" class="core-loading-center"> | ||||||
|  |     <core-course-module-description [description]="description" [component]="component" [componentId]="componentId" | ||||||
|  |         contextLevel="module" [contextInstanceId]="module?.id" [courseId]="courseId"> | ||||||
|  |     </core-course-module-description> | ||||||
|  | 
 | ||||||
|  |     <!-- Access rules description messages. --> | ||||||
|  |     <ion-card *ngIf="gradeMethodReadable || accessRules.length || syncTime"> | ||||||
|  |         <ion-list> | ||||||
|  |             <ion-item class="ion-text-wrap" *ngFor="let rule of accessRules"> | ||||||
|  |                 <ion-label><p>{{ rule }}</p></ion-label> | ||||||
|  |             </ion-item> | ||||||
|  |             <ion-item class="ion-text-wrap" *ngIf="gradeMethodReadable"> | ||||||
|  |                 <ion-label> | ||||||
|  |                     <h3>{{ 'addon.mod_quiz.grademethod' | translate }}</h3> | ||||||
|  |                     <p>{{ gradeMethodReadable }}</p> | ||||||
|  |                 </ion-label> | ||||||
|  |             </ion-item> | ||||||
|  |             <ion-item class="ion-text-wrap" *ngIf="syncTime"> | ||||||
|  |                 <ion-label> | ||||||
|  |                     <h3>{{ 'core.lastsync' | translate }}</h3> | ||||||
|  |                     <p>{{ syncTime }}</p> | ||||||
|  |                 </ion-label> | ||||||
|  |             </ion-item> | ||||||
|  |         </ion-list> | ||||||
|  |     </ion-card> | ||||||
|  | 
 | ||||||
|  |     <!-- List of user attempts. --> | ||||||
|  |     <ion-card class="addon-mod_quiz-table" *ngIf="quiz && attempts.length"> | ||||||
|  |         <ion-card-header class="ion-text-wrap"> | ||||||
|  |             <ion-card-header> | ||||||
|  |                 <ion-card-title>{{ 'addon.mod_quiz.summaryofattempts' | translate }}</ion-card-title> | ||||||
|  |             </ion-card-header> | ||||||
|  |         </ion-card-header> | ||||||
|  |         <ion-card-content> | ||||||
|  |             <!-- "Header" of the table --> | ||||||
|  |             <ion-item class="ion-text-wrap addon-mod_quiz-table-header" detail="true"> | ||||||
|  |                 <ion-label> | ||||||
|  |                     <ion-row class="ion-align-items-center"> | ||||||
|  |                         <ion-col class="ion-text-center ion-hide-md-down" *ngIf="quiz.showAttemptColumn"> | ||||||
|  |                             <strong>{{ 'addon.mod_quiz.attemptnumber' | translate }}</strong> | ||||||
|  |                         </ion-col> | ||||||
|  |                         <ion-col class="ion-text-center ion-hide-md-up" *ngIf="quiz.showAttemptColumn"><strong>#</strong></ion-col> | ||||||
|  |                         <ion-col size="7"><strong>{{ 'addon.mod_quiz.attemptstate' | translate }}</strong></ion-col> | ||||||
|  |                         <ion-col class="ion-text-center ion-hide-md-down" *ngIf="quiz.showMarkColumn"> | ||||||
|  |                             <strong>{{ 'addon.mod_quiz.marks' | translate }} / {{ quiz.sumGradesFormatted }}</strong> | ||||||
|  |                         </ion-col> | ||||||
|  |                         <ion-col class="ion-text-center" *ngIf="quiz.showGradeColumn"> | ||||||
|  |                             <strong>{{ 'addon.mod_quiz.grade' | translate }} / {{ quiz.gradeFormatted }}</strong> | ||||||
|  |                         </ion-col> | ||||||
|  |                     </ion-row> | ||||||
|  |                 </ion-label> | ||||||
|  |             </ion-item> | ||||||
|  |             <!-- List of attempts. --> | ||||||
|  |             <ion-item class="ion-text-wrap" *ngFor="let attempt of attempts" button detail="true" | ||||||
|  |                 [ngClass]='{"addon-mod_quiz-highlighted": attempt.highlightGrade}' | ||||||
|  |                 [attr.aria-label]="'core.seemoredetail' | translate"> <!-- @todo navPush="AddonModQuizAttemptPage" [navParams]="{courseId: courseId, quizId: quiz.id, attemptId: attempt.id}" --> | ||||||
|  |                 <ion-label> | ||||||
|  |                     <ion-row class="ion-align-items-center"> | ||||||
|  |                         <ion-col class="ion-text-center" *ngIf="quiz.showAttemptColumn && attempt.preview"> | ||||||
|  |                             {{ 'addon.mod_quiz.preview' | translate }} | ||||||
|  |                         </ion-col> | ||||||
|  |                         <ion-col class="ion-text-center" *ngIf="quiz.showAttemptColumn && !attempt.preview"> | ||||||
|  |                             {{ attempt.attempt }} | ||||||
|  |                         </ion-col> | ||||||
|  |                         <ion-col size="7"> | ||||||
|  |                             <p *ngFor="let sentence of attempt.readableState">{{ sentence }}</p> | ||||||
|  |                         </ion-col> | ||||||
|  |                         <ion-col class="ion-text-center ion-hide-md-down" *ngIf="quiz.showMarkColumn"> | ||||||
|  |                             <p>{{ attempt.readableMark }}</p> | ||||||
|  |                         </ion-col> | ||||||
|  |                         <ion-col class="ion-text-center" *ngIf="quiz.showGradeColumn"><p>{{ attempt.readableGrade }}</p></ion-col> | ||||||
|  |                     </ion-row> | ||||||
|  |                 </ion-label> | ||||||
|  |             </ion-item> | ||||||
|  |         </ion-card-content> | ||||||
|  |     </ion-card> | ||||||
|  | 
 | ||||||
|  |     <!-- Result info. --> | ||||||
|  |     <ion-card *ngIf="quiz && showResults && | ||||||
|  |         (gradeResult || gradeOverridden || gradebookFeedback || (quiz.showFeedbackColumn && overallFeedback))"> | ||||||
|  |         <ion-list> | ||||||
|  |             <ion-item class="ion-text-wrap" *ngIf="gradeResult"> | ||||||
|  |                 <ion-label>{{ gradeResult }}</ion-label> | ||||||
|  |             </ion-item> | ||||||
|  |             <ion-item class="ion-text-wrap" *ngIf="gradeOverridden"> | ||||||
|  |                 <ion-label>{{ 'core.course.overriddennotice' | translate }}</ion-label> | ||||||
|  |             </ion-item> | ||||||
|  |             <ion-item class="ion-text-wrap" *ngIf="gradebookFeedback"> | ||||||
|  |                 <ion-label> | ||||||
|  |                     <h3 class="item-heading">{{ 'addon.mod_quiz.comment' | translate }}</h3> | ||||||
|  |                     <p><core-format-text [component]="component" [componentId]="componentId" [text]="gradebookFeedback" | ||||||
|  |                         contextLevel="module" [contextInstanceId]="module?.id" [courseId]="courseId"> | ||||||
|  |                     </core-format-text></p> | ||||||
|  |                 </ion-label> | ||||||
|  |             </ion-item> | ||||||
|  |             <ion-item class="ion-text-wrap" *ngIf="quiz.showFeedbackColumn && overallFeedback"> | ||||||
|  |                 <ion-label> | ||||||
|  |                     <h3 class="item-heading">{{ 'addon.mod_quiz.overallfeedback' | translate }}</h3> | ||||||
|  |                     <p><core-format-text [component]="component" [componentId]="componentId" [text]="overallFeedback" | ||||||
|  |                         contextLevel="module" [contextInstanceId]="module?.id" [courseId]="courseId"> | ||||||
|  |                     </core-format-text></p> | ||||||
|  |                 </ion-label> | ||||||
|  |             </ion-item> | ||||||
|  |         </ion-list> | ||||||
|  |     </ion-card> | ||||||
|  | 
 | ||||||
|  |     <!-- More data and button to start/continue. --> | ||||||
|  |     <ion-card *ngIf="quiz"> | ||||||
|  |         <ion-list> | ||||||
|  |             <!-- Error messages. --> | ||||||
|  |             <ion-item class="ion-text-wrap core-danger-item" *ngFor="let message of preventMessages"> | ||||||
|  |                 <ion-label><p>{{ message }}</p></ion-label> | ||||||
|  |             </ion-item> | ||||||
|  |             <ion-item class="ion-text-wrap core-danger-item" *ngIf="quiz.hasquestions === 0"> | ||||||
|  |                 <ion-label><p>{{ 'addon.mod_quiz.noquestions' | translate }}</p></ion-label> | ||||||
|  |             </ion-item> | ||||||
|  |             <ion-item class="ion-text-wrap core-danger-item" *ngIf="!hasSupportedQuestions && unsupportedQuestions.length"> | ||||||
|  |                 <ion-label> | ||||||
|  |                     <p>{{ 'addon.mod_quiz.errorquestionsnotsupported' | translate }}</p> | ||||||
|  |                     <p *ngFor="let type of unsupportedQuestions">{{ type }}</p> | ||||||
|  |                 </ion-label> | ||||||
|  |             </ion-item> | ||||||
|  |             <ion-item class="ion-text-wrap core-danger-item" *ngIf="unsupportedRules.length"> | ||||||
|  |                 <ion-label> | ||||||
|  |                     <p>{{ 'addon.mod_quiz.errorrulesnotsupported' | translate }}</p> | ||||||
|  |                     <p *ngFor="let name of unsupportedRules">{{ name }}</p> | ||||||
|  |                 </ion-label> | ||||||
|  |             </ion-item> | ||||||
|  |             <ion-item class="ion-text-wrap core-danger-item" *ngIf="behaviourSupported === false"> | ||||||
|  |                 <ion-label> | ||||||
|  |                     <p>{{ 'addon.mod_quiz.errorbehaviournotsupported' | translate }}</p> | ||||||
|  |                     <p>{{ quiz.preferredbehaviour }}</p> | ||||||
|  |                 </ion-label> | ||||||
|  |             </ion-item> | ||||||
|  | 
 | ||||||
|  |             <!-- Quiz has data to be synchronized --> | ||||||
|  |             <ion-card class="core-warning-card" *ngIf="buttonText && hasOffline && !showStatusSpinner"> | ||||||
|  |                 <ion-item class="ion-text-wrap"> | ||||||
|  |                     <ion-icon name="fas-exclamation-triangle" slot="start"></ion-icon> | ||||||
|  |                     <ion-label>{{ 'core.hasdatatosync' | translate: {$a: moduleName} }}</ion-label> | ||||||
|  |                 </ion-item> | ||||||
|  |             </ion-card> | ||||||
|  | 
 | ||||||
|  |             <!-- Other warnings. --> | ||||||
|  |             <ion-item class="core-warning-item ion-text-wrap" *ngIf="hasSupportedQuestions && unsupportedQuestions.length"> | ||||||
|  |                 <ion-label> | ||||||
|  |                     <p>{{ 'addon.mod_quiz.canattemptbutnotsubmit' | translate }}</p> | ||||||
|  |                     <p>{{ 'addon.mod_quiz.warningquestionsnotsupported' | translate }}</p> | ||||||
|  |                     <p *ngFor="let type of unsupportedQuestions">{{ type }}</p> | ||||||
|  |                 </ion-label> | ||||||
|  |             </ion-item> | ||||||
|  | 
 | ||||||
|  |             <!-- Button to start/continue. --> | ||||||
|  |             <ion-button *ngIf="buttonText && !showStatusSpinner" expand="block" (click)="attemptQuiz()" class="ion-margin"> | ||||||
|  |                 {{ buttonText | translate }} | ||||||
|  |             </ion-button> | ||||||
|  | 
 | ||||||
|  |             <!-- Button to open in browser if it cannot be attempted in the app. --> | ||||||
|  |             <ion-button class="ion-margin" *ngIf="!buttonText && ((!hasSupportedQuestions && unsupportedQuestions.length) || | ||||||
|  |                 unsupportedRules.length || behaviourSupported === false)" expand="block" [href]="externalUrl" core-link> | ||||||
|  |                 {{ 'core.openinbrowser' | translate }} | ||||||
|  |                 <ion-icon name="fas-external-link-alt" slot="end"></ion-icon> | ||||||
|  |             </ion-button> | ||||||
|  | 
 | ||||||
|  |             <!-- Spinner shown while downloading or calculating. --> | ||||||
|  |             <ion-item class="ion-text-center" *ngIf="showStatusSpinner"> | ||||||
|  |                 <ion-label><ion-spinner></ion-spinner></ion-label> | ||||||
|  |             </ion-item> | ||||||
|  |         </ion-list> | ||||||
|  |     </ion-card> | ||||||
|  | </core-loading> | ||||||
							
								
								
									
										44
									
								
								src/addons/mod/quiz/components/index/index.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								src/addons/mod/quiz/components/index/index.scss
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,44 @@ | |||||||
|  | :host { | ||||||
|  | 
 | ||||||
|  |     .addon-mod_quiz-table { | ||||||
|  |         .addon-mod_quiz-table-header { | ||||||
|  |             --detail-icon-opacity: 0; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         ion-card-content { | ||||||
|  |             padding-left: 0; | ||||||
|  |             padding-right: 0; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         .item:nth-child(even) { | ||||||
|  |             --background: var(--gray-lighter); | ||||||
|  |             // @include darkmode() { | ||||||
|  |             //     background-color: $core-dark-item-divider-bg-color; | ||||||
|  |             // } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         .addon-mod_quiz-highlighted, | ||||||
|  |         .item.addon-mod_quiz-highlighted, | ||||||
|  |         .addon-mod_quiz-highlighted p, | ||||||
|  |         .item.addon-mod_quiz-highlighted p { | ||||||
|  |             --background: var(--blue-light); | ||||||
|  |             color: var(--blue-dark); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     //     @include darkmode() { | ||||||
|  |     //         .addon-mod_quiz-highlighted, | ||||||
|  |     //         .item.addon-mod_quiz-highlighted, | ||||||
|  |     //         .addon-mod_quiz-highlighted p, | ||||||
|  |     //         .item.addon-mod_quiz-highlighted p { | ||||||
|  |     //             background-color: $blue-dark; | ||||||
|  |     //             color: $blue-light; | ||||||
|  |     //         } | ||||||
|  | 
 | ||||||
|  |     //         .item.addon-mod_quiz-highlighted.activated, | ||||||
|  |     //         .item.addon-mod_quiz-highlighted.activated p { | ||||||
|  |     //             background-color: $blue; | ||||||
|  |     //             color: $blue-light; | ||||||
|  |     //         } | ||||||
|  |     //     } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										658
									
								
								src/addons/mod/quiz/components/index/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										658
									
								
								src/addons/mod/quiz/components/index/index.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,658 @@ | |||||||
|  | // (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 { CoreConstants } from '@/core/constants'; | ||||||
|  | import { Component, OnDestroy, OnInit, Optional } from '@angular/core'; | ||||||
|  | 
 | ||||||
|  | import { CoreCourseModuleMainActivityComponent } from '@features/course/classes/main-activity-component'; | ||||||
|  | import { CoreCourseContentsPage } from '@features/course/pages/contents/contents'; | ||||||
|  | import { CoreCourse } from '@features/course/services/course'; | ||||||
|  | import { CoreCourseModulePrefetchDelegate } from '@features/course/services/module-prefetch-delegate'; | ||||||
|  | import { CoreQuestionBehaviourDelegate } from '@features/question/services/behaviour-delegate'; | ||||||
|  | import { IonContent } from '@ionic/angular'; | ||||||
|  | import { CoreDomUtils } from '@services/utils/dom'; | ||||||
|  | import { CoreTextUtils } from '@services/utils/text'; | ||||||
|  | import { CoreUtils } from '@services/utils/utils'; | ||||||
|  | import { Translate } from '@singletons'; | ||||||
|  | import { CoreEventObserver, CoreEvents } from '@singletons/events'; | ||||||
|  | import { AddonModQuizPrefetchHandler } from '../../services/handlers/prefetch'; | ||||||
|  | import { | ||||||
|  |     AddonModQuiz, | ||||||
|  |     AddonModQuizAttemptFinishedData, | ||||||
|  |     AddonModQuizAttemptWSData, | ||||||
|  |     AddonModQuizCombinedReviewOptions, | ||||||
|  |     AddonModQuizGetAttemptAccessInformationWSResponse, | ||||||
|  |     AddonModQuizGetQuizAccessInformationWSResponse, | ||||||
|  |     AddonModQuizGetUserBestGradeWSResponse, | ||||||
|  |     AddonModQuizProvider, | ||||||
|  | } from '../../services/quiz'; | ||||||
|  | import { AddonModQuizAttempt, AddonModQuizHelper, AddonModQuizQuizData } from '../../services/quiz-helper'; | ||||||
|  | import { | ||||||
|  |     AddonModQuizAutoSyncData, | ||||||
|  |     AddonModQuizSync, | ||||||
|  |     AddonModQuizSyncProvider, | ||||||
|  |     AddonModQuizSyncResult, | ||||||
|  | } from '../../services/quiz-sync'; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Component that displays a quiz entry page. | ||||||
|  |  */ | ||||||
|  | @Component({ | ||||||
|  |     selector: 'addon-mod-quiz-index', | ||||||
|  |     templateUrl: 'addon-mod-quiz-index.html', | ||||||
|  |     styleUrls: ['index.scss'], | ||||||
|  | }) | ||||||
|  | export class AddonModQuizIndexComponent extends CoreCourseModuleMainActivityComponent implements OnInit, OnDestroy { | ||||||
|  | 
 | ||||||
|  |     component = AddonModQuizProvider.COMPONENT; | ||||||
|  |     moduleName = 'quiz'; | ||||||
|  |     quiz?: AddonModQuizQuizData; // The quiz.
 | ||||||
|  |     now?: number; // Current time.
 | ||||||
|  |     syncTime?: string; // Last synchronization time.
 | ||||||
|  |     hasOffline = false; // Whether the quiz has offline data.
 | ||||||
|  |     hasSupportedQuestions = false; // Whether the quiz has at least 1 supported question.
 | ||||||
|  |     accessRules: string[] = []; // List of access rules of the quiz.
 | ||||||
|  |     unsupportedRules: string[] = []; // List of unsupported access rules of the quiz.
 | ||||||
|  |     unsupportedQuestions: string[] = []; // List of unsupported question types of the quiz.
 | ||||||
|  |     behaviourSupported = false; // Whether the quiz behaviour is supported.
 | ||||||
|  |     showResults = false; // Whether to show the result of the quiz (grade, etc.).
 | ||||||
|  |     gradeOverridden = false; // Whether grade has been overridden.
 | ||||||
|  |     gradebookFeedback?: string; // The feedback in the gradebook.
 | ||||||
|  |     gradeResult?: string; // Message with the grade.
 | ||||||
|  |     overallFeedback?: string; // The feedback for the grade.
 | ||||||
|  |     buttonText?: string; // Text to display in the start/continue button.
 | ||||||
|  |     preventMessages: string[] = []; // List of messages explaining why the quiz cannot be attempted.
 | ||||||
|  |     showStatusSpinner = true; // Whether to show a spinner due to quiz status.
 | ||||||
|  |     gradeMethodReadable?: string; // Grade method in a readable format.
 | ||||||
|  |     showReviewColumn = false; // Whether to show the review column.
 | ||||||
|  |     attempts: AddonModQuizAttempt[] = []; // List of attempts the user has made.
 | ||||||
|  | 
 | ||||||
|  |     protected fetchContentDefaultError = 'addon.mod_quiz.errorgetquiz'; // Default error to show when loading contents.
 | ||||||
|  |     protected syncEventName = AddonModQuizSyncProvider.AUTO_SYNCED; | ||||||
|  | 
 | ||||||
|  |     // protected quizData: any; // Quiz instance. This variable will store the quiz instance until it's ready to be shown
 | ||||||
|  |     protected autoReview?: AddonModQuizAttemptFinishedData; // Data to auto-review an attempt after finishing.
 | ||||||
|  |     protected quizAccessInfo?: AddonModQuizGetQuizAccessInformationWSResponse; // Quiz access info.
 | ||||||
|  |     protected attemptAccessInfo?: AddonModQuizGetAttemptAccessInformationWSResponse; // Last attempt access info.
 | ||||||
|  |     protected moreAttempts = false; // Whether user can create/continue attempts.
 | ||||||
|  |     protected options?: AddonModQuizCombinedReviewOptions; // Combined review options.
 | ||||||
|  |     protected bestGrade?: AddonModQuizGetUserBestGradeWSResponse; // Best grade data.
 | ||||||
|  |     protected gradebookData?: { grade?: number; feedback?: string }; // The gradebook grade and feedback.
 | ||||||
|  |     protected overallStats = false; // Equivalent to overallstats in mod_quiz_view_object in Moodle.
 | ||||||
|  |     protected finishedObserver?: CoreEventObserver; // It will observe attempt finished events.
 | ||||||
|  |     protected hasPlayed = false; // Whether the user has gone to the quiz player (attempted).
 | ||||||
|  |     protected candidateQuiz?: AddonModQuizQuizData; | ||||||
|  | 
 | ||||||
|  |     constructor( | ||||||
|  |         protected content?: IonContent, | ||||||
|  |         @Optional() courseContentsPage?: CoreCourseContentsPage, | ||||||
|  |     ) { | ||||||
|  |         super('AddonModQuizIndexComponent', content, courseContentsPage); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Component being initialized. | ||||||
|  |      */ | ||||||
|  |     async ngOnInit(): Promise<void> { | ||||||
|  |         super.ngOnInit(); | ||||||
|  | 
 | ||||||
|  |         // Listen for attempt finished events.
 | ||||||
|  |         this.finishedObserver = CoreEvents.on<AddonModQuizAttemptFinishedData>( | ||||||
|  |             AddonModQuizProvider.ATTEMPT_FINISHED_EVENT, | ||||||
|  |             (data) => { | ||||||
|  |                 // Go to review attempt if an attempt in this quiz was finished and synced.
 | ||||||
|  |                 if (this.quiz && data.quizId == this.quiz.id) { | ||||||
|  |                     this.autoReview = data; | ||||||
|  |                 } | ||||||
|  |             }, | ||||||
|  |             this.siteId, | ||||||
|  |         ); | ||||||
|  | 
 | ||||||
|  |         await this.loadContent(false, true); | ||||||
|  | 
 | ||||||
|  |         if (!this.quiz) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         try { | ||||||
|  |             await AddonModQuiz.instance.logViewQuiz(this.quiz.id, this.quiz.name); | ||||||
|  | 
 | ||||||
|  |             CoreCourse.instance.checkModuleCompletion(this.courseId!, this.module!.completiondata); | ||||||
|  |         } catch { | ||||||
|  |             // Ignore errors.
 | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Attempt the quiz. | ||||||
|  |      */ | ||||||
|  |     async attemptQuiz(): Promise<void> { | ||||||
|  |         if (this.showStatusSpinner || !this.quiz) { | ||||||
|  |             // Quiz is being downloaded or synchronized, abort.
 | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (!AddonModQuiz.instance.isQuizOffline(this.quiz)) { | ||||||
|  |             // Quiz isn't offline, just open it.
 | ||||||
|  |             return this.openQuiz(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // Quiz supports offline, check if it needs to be downloaded.
 | ||||||
|  |         // If the site doesn't support check updates, always prefetch it because we cannot tell if there's something new.
 | ||||||
|  |         const isDownloaded = this.currentStatus == CoreConstants.DOWNLOADED; | ||||||
|  | 
 | ||||||
|  |         if (isDownloaded && CoreCourseModulePrefetchDelegate.instance.canCheckUpdates()) { | ||||||
|  |             // Already downloaded, open it.
 | ||||||
|  |             return this.openQuiz(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // Prefetch the quiz.
 | ||||||
|  |         this.showStatusSpinner = true; | ||||||
|  | 
 | ||||||
|  |         try { | ||||||
|  |             await AddonModQuizPrefetchHandler.instance.prefetch(this.module!, this.courseId, true); | ||||||
|  | 
 | ||||||
|  |             // Success downloading, open quiz.
 | ||||||
|  |             this.openQuiz(); | ||||||
|  |         } catch (error) { | ||||||
|  |             if (this.hasOffline || (isDownloaded && !CoreCourseModulePrefetchDelegate.instance.canCheckUpdates())) { | ||||||
|  |                 // Error downloading but there is something offline, allow continuing it.
 | ||||||
|  |                 // If the site doesn't support check updates, continue too because we cannot tell if there's something new.
 | ||||||
|  |                 this.openQuiz(); | ||||||
|  |             } else { | ||||||
|  |                 CoreDomUtils.instance.showErrorModalDefault(error, 'core.errordownloading', true); | ||||||
|  |             } | ||||||
|  |         } finally { | ||||||
|  |             this.showStatusSpinner = false; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Get the quiz data. | ||||||
|  |      * | ||||||
|  |      * @param refresh If it's refreshing content. | ||||||
|  |      * @param sync If it should try to sync. | ||||||
|  |      * @param showErrors If show errors to the user of hide them. | ||||||
|  |      * @return Promise resolved when done. | ||||||
|  |      */ | ||||||
|  |     protected async fetchContent(refresh: boolean = false, sync: boolean = false, showErrors: boolean = false): Promise<void> { | ||||||
|  |         try { | ||||||
|  |             // First get the quiz instance.
 | ||||||
|  |             const quiz = await AddonModQuiz.instance.getQuiz(this.courseId!, this.module!.id); | ||||||
|  | 
 | ||||||
|  |             this.gradeMethodReadable = AddonModQuiz.instance.getQuizGradeMethod(quiz.grademethod); | ||||||
|  |             this.now = Date.now(); | ||||||
|  |             this.dataRetrieved.emit(quiz); | ||||||
|  |             this.description = quiz.intro || this.description; | ||||||
|  |             this.candidateQuiz = quiz; | ||||||
|  | 
 | ||||||
|  |             // Try to get warnings from automatic sync.
 | ||||||
|  |             const warnings = await AddonModQuizSync.instance.getSyncWarnings(quiz.id); | ||||||
|  | 
 | ||||||
|  |             if (warnings?.length) { | ||||||
|  |                 // Show warnings and delete them so they aren't shown again.
 | ||||||
|  |                 CoreDomUtils.instance.showErrorModal(CoreTextUtils.instance.buildMessage(warnings)); | ||||||
|  | 
 | ||||||
|  |                 await AddonModQuizSync.instance.setSyncWarnings(quiz.id, []); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (AddonModQuiz.instance.isQuizOffline(quiz) && sync) { | ||||||
|  |                 // Try to sync the quiz.
 | ||||||
|  |                 try { | ||||||
|  |                     await this.syncActivity(showErrors); | ||||||
|  |                 } catch { | ||||||
|  |                     // Ignore errors, keep getting data even if sync fails.
 | ||||||
|  |                     this.autoReview = undefined; | ||||||
|  |                 } | ||||||
|  |             } else { | ||||||
|  |                 this.autoReview = undefined; | ||||||
|  |                 this.showStatusSpinner = false; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (AddonModQuiz.instance.isQuizOffline(quiz)) { | ||||||
|  |                 // Handle status.
 | ||||||
|  |                 this.setStatusListener(); | ||||||
|  | 
 | ||||||
|  |                 // Get last synchronization time and check if sync button should be seen.
 | ||||||
|  |                 this.syncTime = await AddonModQuizSync.instance.getReadableSyncTime(quiz.id); | ||||||
|  |                 this.hasOffline = await AddonModQuizSync.instance.hasDataToSync(quiz.id); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             // Get quiz access info.
 | ||||||
|  |             this.quizAccessInfo = await AddonModQuiz.instance.getQuizAccessInformation(quiz.id, { cmId: this.module!.id }); | ||||||
|  | 
 | ||||||
|  |             this.showReviewColumn = this.quizAccessInfo.canreviewmyattempts; | ||||||
|  |             this.accessRules = this.quizAccessInfo.accessrules; | ||||||
|  |             this.unsupportedRules = AddonModQuiz.instance.getUnsupportedRules(this.quizAccessInfo.activerulenames); | ||||||
|  | 
 | ||||||
|  |             if (quiz.preferredbehaviour) { | ||||||
|  |                 this.behaviourSupported = CoreQuestionBehaviourDelegate.instance.isBehaviourSupported(quiz.preferredbehaviour); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             // Get question types in the quiz.
 | ||||||
|  |             const types = await AddonModQuiz.instance.getQuizRequiredQtypes(quiz.id, { cmId: this.module!.id }); | ||||||
|  | 
 | ||||||
|  |             this.unsupportedQuestions = AddonModQuiz.instance.getUnsupportedQuestions(types); | ||||||
|  |             this.hasSupportedQuestions = !!types.find((type) => type != 'random' && this.unsupportedQuestions.indexOf(type) == -1); | ||||||
|  | 
 | ||||||
|  |             await this.getAttempts(quiz); | ||||||
|  | 
 | ||||||
|  |             // Quiz is ready to be shown, move it to the variable that is displayed.
 | ||||||
|  |             this.quiz = quiz; | ||||||
|  |         } finally { | ||||||
|  |             this.fillContextMenu(refresh); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Get the user attempts in the quiz and the result info. | ||||||
|  |      * | ||||||
|  |      * @param quiz Quiz instance. | ||||||
|  |      * @return Promise resolved when done. | ||||||
|  |      */ | ||||||
|  |     protected async getAttempts(quiz: AddonModQuizQuizData): Promise<void> { | ||||||
|  | 
 | ||||||
|  |         // Get access information of last attempt (it also works if no attempts made).
 | ||||||
|  |         this.attemptAccessInfo = await AddonModQuiz.instance.getAttemptAccessInformation(quiz.id, 0, { cmId: this.module!.id }); | ||||||
|  | 
 | ||||||
|  |         // Get attempts.
 | ||||||
|  |         const attempts = await AddonModQuiz.instance.getUserAttempts(quiz.id, { cmId: this.module!.id }); | ||||||
|  | 
 | ||||||
|  |         this.attempts = await this.treatAttempts(quiz, attempts); | ||||||
|  | 
 | ||||||
|  |         // Check if user can create/continue attempts.
 | ||||||
|  |         if (this.attempts.length) { | ||||||
|  |             const last = this.attempts[this.attempts.length - 1]; | ||||||
|  |             this.moreAttempts = !AddonModQuiz.instance.isAttemptFinished(last.state) || !this.attemptAccessInfo.isfinished; | ||||||
|  |         } else { | ||||||
|  |             this.moreAttempts = !this.attemptAccessInfo.isfinished; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         this.getButtonText(quiz); | ||||||
|  | 
 | ||||||
|  |         await this.getResultInfo(quiz); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Get the text to show in the button. It also sets restriction messages if needed. | ||||||
|  |      * | ||||||
|  |      * @param quiz Quiz. | ||||||
|  |      */ | ||||||
|  |     protected getButtonText(quiz: AddonModQuizQuizData): void { | ||||||
|  |         this.buttonText = ''; | ||||||
|  | 
 | ||||||
|  |         if (quiz.hasquestions !== 0) { | ||||||
|  |             if (this.attempts.length && !AddonModQuiz.instance.isAttemptFinished(this.attempts[this.attempts.length - 1].state)) { | ||||||
|  |                 // Last attempt is unfinished.
 | ||||||
|  |                 if (this.quizAccessInfo?.canattempt) { | ||||||
|  |                     this.buttonText = 'addon.mod_quiz.continueattemptquiz'; | ||||||
|  |                 } else if (this.quizAccessInfo?.canpreview) { | ||||||
|  |                     this.buttonText = 'addon.mod_quiz.continuepreview'; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |             } else { | ||||||
|  |                 // Last attempt is finished or no attempts.
 | ||||||
|  |                 if (this.quizAccessInfo?.canattempt) { | ||||||
|  |                     this.preventMessages = this.attemptAccessInfo?.preventnewattemptreasons || []; | ||||||
|  |                     if (!this.preventMessages.length) { | ||||||
|  |                         if (!this.attempts.length) { | ||||||
|  |                             this.buttonText = 'addon.mod_quiz.attemptquiznow'; | ||||||
|  |                         } else { | ||||||
|  |                             this.buttonText = 'addon.mod_quiz.reattemptquiz'; | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } else if (this.quizAccessInfo?.canpreview) { | ||||||
|  |                     this.buttonText = 'addon.mod_quiz.previewquiznow'; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (!this.buttonText) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // So far we think a button should be printed, check if they will be allowed to access it.
 | ||||||
|  |         this.preventMessages = this.quizAccessInfo?.preventaccessreasons || []; | ||||||
|  | 
 | ||||||
|  |         if (!this.moreAttempts) { | ||||||
|  |             this.buttonText = ''; | ||||||
|  |         } else if (this.quizAccessInfo?.canattempt && this.preventMessages.length) { | ||||||
|  |             this.buttonText = ''; | ||||||
|  |         } else if (!this.hasSupportedQuestions || this.unsupportedRules.length || !this.behaviourSupported) { | ||||||
|  |             this.buttonText = ''; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Get result info to show. | ||||||
|  |      * | ||||||
|  |      * @param quiz Quiz. | ||||||
|  |      * @return Promise resolved when done. | ||||||
|  |      */ | ||||||
|  |     protected async getResultInfo(quiz: AddonModQuizQuizData): Promise<void> { | ||||||
|  | 
 | ||||||
|  |         if (!this.attempts.length || !quiz.showGradeColumn || !this.bestGrade?.hasgrade || | ||||||
|  |             this.gradebookData?.grade === undefined) { | ||||||
|  |             this.showResults = false; | ||||||
|  | 
 | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         const formattedGradebookGrade = AddonModQuiz.instance.formatGrade(this.gradebookData.grade, quiz.decimalpoints); | ||||||
|  |         const formattedBestGrade = AddonModQuiz.instance.formatGrade(this.bestGrade.grade, quiz.decimalpoints); | ||||||
|  |         let gradeToShow = formattedGradebookGrade; // By default we show the grade in the gradebook.
 | ||||||
|  | 
 | ||||||
|  |         this.showResults = true; | ||||||
|  |         this.gradeOverridden = formattedGradebookGrade != formattedBestGrade; | ||||||
|  |         this.gradebookFeedback = this.gradebookData.feedback; | ||||||
|  | 
 | ||||||
|  |         if (this.bestGrade.grade! > this.gradebookData.grade && this.gradebookData.grade == quiz.grade) { | ||||||
|  |             // The best grade is higher than the max grade for the quiz.
 | ||||||
|  |             // We'll do like Moodle web and show the best grade instead of the gradebook grade.
 | ||||||
|  |             this.gradeOverridden = false; | ||||||
|  |             gradeToShow = formattedBestGrade; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (this.overallStats) { | ||||||
|  |             // Show the quiz grade. The message shown is different if the quiz is finished.
 | ||||||
|  |             if (this.moreAttempts) { | ||||||
|  |                 this.gradeResult = Translate.instance.instant('addon.mod_quiz.gradesofar', { $a: { | ||||||
|  |                     method: this.gradeMethodReadable, | ||||||
|  |                     mygrade: gradeToShow, | ||||||
|  |                     quizgrade: quiz.gradeFormatted, | ||||||
|  |                 } }); | ||||||
|  |             } else { | ||||||
|  |                 const outOfShort = Translate.instance.instant('addon.mod_quiz.outofshort', { $a: { | ||||||
|  |                     grade: gradeToShow, | ||||||
|  |                     maxgrade: quiz.gradeFormatted, | ||||||
|  |                 } }); | ||||||
|  | 
 | ||||||
|  |                 this.gradeResult = Translate.instance.instant('addon.mod_quiz.yourfinalgradeis', { $a: outOfShort }); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (quiz.showFeedbackColumn) { | ||||||
|  |             // Get the quiz overall feedback.
 | ||||||
|  |             const response = await AddonModQuiz.instance.getFeedbackForGrade(quiz.id, this.gradebookData.grade, { | ||||||
|  |                 cmId: this.module!.id, | ||||||
|  |             }); | ||||||
|  | 
 | ||||||
|  |             this.overallFeedback = response.feedbacktext; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Go to review an attempt that has just been finished. | ||||||
|  |      * | ||||||
|  |      * @return Promise resolved when done. | ||||||
|  |      */ | ||||||
|  |     protected async goToAutoReview(): Promise<void> { | ||||||
|  |         if (!this.autoReview) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // If we go to auto review it means an attempt was finished. Check completion status.
 | ||||||
|  |         CoreCourse.instance.checkModuleCompletion(this.courseId!, this.module!.completiondata); | ||||||
|  | 
 | ||||||
|  |         // Verify that user can see the review.
 | ||||||
|  |         const attemptId = this.autoReview.attemptId; | ||||||
|  | 
 | ||||||
|  |         if (this.quizAccessInfo?.canreviewmyattempts) { | ||||||
|  |             try { | ||||||
|  |                 await AddonModQuiz.instance.getAttemptReview(attemptId, { page: -1, cmId: this.module!.id }); | ||||||
|  | 
 | ||||||
|  |                 // @todo this.navCtrl.push('AddonModQuizReviewPage', { courseId: this.courseId, quizId: quiz!.id, attemptId });
 | ||||||
|  |             } catch { | ||||||
|  |                 // Ignore errors.
 | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Checks if sync has succeed from result sync data. | ||||||
|  |      * | ||||||
|  |      * @param result Data returned on the sync function. | ||||||
|  |      * @return If suceed or not. | ||||||
|  |      */ | ||||||
|  |     protected hasSyncSucceed(result: AddonModQuizSyncResult): boolean { | ||||||
|  |         if (result.attemptFinished) { | ||||||
|  |             // An attempt was finished, check completion status.
 | ||||||
|  |             CoreCourse.instance.checkModuleCompletion(this.courseId!, this.module!.completiondata); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // If the sync call isn't rejected it means the sync was successful.
 | ||||||
|  |         return result.updated; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * User entered the page that contains the component. | ||||||
|  |      */ | ||||||
|  |     async ionViewDidEnter(): Promise<void> { | ||||||
|  |         super.ionViewDidEnter(); | ||||||
|  | 
 | ||||||
|  |         if (!this.hasPlayed) { | ||||||
|  |             this.autoReview = undefined; | ||||||
|  | 
 | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         this.hasPlayed = false; | ||||||
|  |         let promise = Promise.resolve(); | ||||||
|  | 
 | ||||||
|  |         // Update data when we come back from the player since the attempt status could have changed.
 | ||||||
|  |         // Check if we need to go to review an attempt automatically.
 | ||||||
|  |         if (this.autoReview && this.autoReview.synced) { | ||||||
|  |             promise = this.goToAutoReview(); | ||||||
|  |             this.autoReview = undefined; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // Refresh data.
 | ||||||
|  |         this.loaded = false; | ||||||
|  |         this.refreshIcon = CoreConstants.ICON_LOADING; | ||||||
|  |         this.syncIcon = CoreConstants.ICON_LOADING; | ||||||
|  |         this.content?.scrollToTop(); | ||||||
|  | 
 | ||||||
|  |         await promise; | ||||||
|  |         await CoreUtils.instance.ignoreErrors(this.refreshContent()); | ||||||
|  | 
 | ||||||
|  |         this.loaded = true; | ||||||
|  |         this.refreshIcon = CoreConstants.ICON_REFRESH; | ||||||
|  |         this.syncIcon = CoreConstants.ICON_SYNC; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * User left the page that contains the component. | ||||||
|  |      */ | ||||||
|  |     ionViewDidLeave(): void { | ||||||
|  |         super.ionViewDidLeave(); | ||||||
|  |         this.autoReview = undefined; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Perform the invalidate content function. | ||||||
|  |      * | ||||||
|  |      * @return Resolved when done. | ||||||
|  |      */ | ||||||
|  |     protected async invalidateContent(): Promise<void> { | ||||||
|  |         const promises: Promise<void>[] = []; | ||||||
|  | 
 | ||||||
|  |         promises.push(AddonModQuiz.instance.invalidateQuizData(this.courseId!)); | ||||||
|  | 
 | ||||||
|  |         if (this.quiz) { | ||||||
|  |             promises.push(AddonModQuiz.instance.invalidateUserAttemptsForUser(this.quiz.id)); | ||||||
|  |             promises.push(AddonModQuiz.instance.invalidateQuizAccessInformation(this.quiz.id)); | ||||||
|  |             promises.push(AddonModQuiz.instance.invalidateQuizRequiredQtypes(this.quiz.id)); | ||||||
|  |             promises.push(AddonModQuiz.instance.invalidateAttemptAccessInformation(this.quiz.id)); | ||||||
|  |             promises.push(AddonModQuiz.instance.invalidateCombinedReviewOptionsForUser(this.quiz.id)); | ||||||
|  |             promises.push(AddonModQuiz.instance.invalidateUserBestGradeForUser(this.quiz.id)); | ||||||
|  |             promises.push(AddonModQuiz.instance.invalidateGradeFromGradebook(this.courseId!)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         await Promise.all(promises); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Compares sync event data with current data to check if refresh content is needed. | ||||||
|  |      * | ||||||
|  |      * @param syncEventData Data receiven on sync observer. | ||||||
|  |      * @return True if refresh is needed, false otherwise. | ||||||
|  |      */ | ||||||
|  |     protected isRefreshSyncNeeded(syncEventData: AddonModQuizAutoSyncData): boolean { | ||||||
|  |         if (!this.courseId || !this.module) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (syncEventData.attemptFinished) { | ||||||
|  |             // An attempt was finished, check completion status.
 | ||||||
|  |             CoreCourse.instance.checkModuleCompletion(this.courseId, this.module.completiondata); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (this.quiz && syncEventData.quizId == this.quiz.id) { | ||||||
|  |             this.content?.scrollToTop(); | ||||||
|  | 
 | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Open a quiz to attempt it. | ||||||
|  |      */ | ||||||
|  |     protected openQuiz(): void { | ||||||
|  |         this.hasPlayed = true; | ||||||
|  | 
 | ||||||
|  |         // @todo this.navCtrl.push('player', {courseId: this.courseId, quizId: this.quiz.id, moduleUrl: this.module.url});
 | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Displays some data based on the current status. | ||||||
|  |      * | ||||||
|  |      * @param status The current status. | ||||||
|  |      * @param previousStatus The previous status. If not defined, there is no previous status. | ||||||
|  |      */ | ||||||
|  |     protected showStatus(status: string, previousStatus?: string): void { | ||||||
|  |         this.showStatusSpinner = status == CoreConstants.DOWNLOADING; | ||||||
|  | 
 | ||||||
|  |         if (status == CoreConstants.DOWNLOADED && previousStatus == CoreConstants.DOWNLOADING) { | ||||||
|  |             // Quiz downloaded now, maybe a new attempt was created. Load content again.
 | ||||||
|  |             this.loaded = false; | ||||||
|  |             this.loadContent(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Performs the sync of the activity. | ||||||
|  |      * | ||||||
|  |      * @return Promise resolved when done. | ||||||
|  |      */ | ||||||
|  |     protected sync(): Promise<AddonModQuizSyncResult> { | ||||||
|  |         return AddonModQuizSync.instance.syncQuiz(this.candidateQuiz!, true); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Treat user attempts. | ||||||
|  |      * | ||||||
|  |      * @param attempts The attempts to treat. | ||||||
|  |      * @return Promise resolved when done. | ||||||
|  |      */ | ||||||
|  |     protected async treatAttempts( | ||||||
|  |         quiz: AddonModQuizQuizData, | ||||||
|  |         attempts: AddonModQuizAttemptWSData[], | ||||||
|  |     ): Promise<AddonModQuizAttempt[]> { | ||||||
|  |         if (!attempts || !attempts.length) { | ||||||
|  |             // There are no attempts to treat.
 | ||||||
|  |             return []; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         const lastFinished = AddonModQuiz.instance.getLastFinishedAttemptFromList(attempts); | ||||||
|  |         const promises: Promise<unknown>[] = []; | ||||||
|  | 
 | ||||||
|  |         if (this.autoReview && lastFinished && lastFinished.id >= this.autoReview.attemptId) { | ||||||
|  |             // User just finished an attempt in offline and it seems it's been synced, since it's finished in online.
 | ||||||
|  |             // Go to the review of this attempt if the user hasn't left this view.
 | ||||||
|  |             if (!this.isDestroyed && this.isCurrentView) { | ||||||
|  |                 promises.push(this.goToAutoReview()); | ||||||
|  |             } | ||||||
|  |             this.autoReview = undefined; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // Get combined review options.
 | ||||||
|  |         promises.push(AddonModQuiz.instance.getCombinedReviewOptions(quiz.id, { cmId: this.module!.id }).then((options) => { | ||||||
|  |             this.options = options; | ||||||
|  | 
 | ||||||
|  |             return; | ||||||
|  |         })); | ||||||
|  | 
 | ||||||
|  |         // Get best grade.
 | ||||||
|  |         promises.push(this.getQuizGrade(quiz)); | ||||||
|  | 
 | ||||||
|  |         await Promise.all(promises); | ||||||
|  | 
 | ||||||
|  |         const grade = typeof this.gradebookData?.grade != 'undefined' ? this.gradebookData.grade : this.bestGrade?.grade; | ||||||
|  |         const quizGrade = AddonModQuiz.instance.formatGrade(grade, quiz.decimalpoints); | ||||||
|  | 
 | ||||||
|  |         // Calculate data to construct the header of the attempts table.
 | ||||||
|  |         AddonModQuizHelper.instance.setQuizCalculatedData(quiz, this.options!); | ||||||
|  | 
 | ||||||
|  |         this.overallStats = !!lastFinished && this.options!.alloptions.marks >= AddonModQuizProvider.QUESTION_OPTIONS_MARK_AND_MAX; | ||||||
|  | 
 | ||||||
|  |         // Calculate data to show for each attempt.
 | ||||||
|  |         const formattedAttempts = await Promise.all(attempts.map((attempt, index) => { | ||||||
|  |             // Highlight the highest grade if appropriate.
 | ||||||
|  |             const shouldHighlight = this.overallStats && quiz.grademethod == AddonModQuizProvider.GRADEHIGHEST && | ||||||
|  |                 attempts.length > 1; | ||||||
|  |             const isLast = index == attempts.length - 1; | ||||||
|  | 
 | ||||||
|  |             return AddonModQuizHelper.instance.setAttemptCalculatedData(quiz, attempt, shouldHighlight, quizGrade, isLast); | ||||||
|  |         })); | ||||||
|  | 
 | ||||||
|  |         return formattedAttempts; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Get quiz grade data. | ||||||
|  |      * | ||||||
|  |      * @param quiz Quiz. | ||||||
|  |      * @return Promise resolved when done. | ||||||
|  |      */ | ||||||
|  |     protected async getQuizGrade(quiz: AddonModQuizQuizData): Promise<void> { | ||||||
|  |         this.bestGrade = await AddonModQuiz.instance.getUserBestGrade(quiz.id, { cmId: this.module!.id }); | ||||||
|  | 
 | ||||||
|  |         try { | ||||||
|  |             // Get gradebook grade.
 | ||||||
|  |             const data = await AddonModQuiz.instance.getGradeFromGradebook(this.courseId!, this.module!.id); | ||||||
|  | 
 | ||||||
|  |             this.gradebookData = { | ||||||
|  |                 grade: data.graderaw, | ||||||
|  |                 feedback: data.feedback, | ||||||
|  |             }; | ||||||
|  |         } catch { | ||||||
|  |             // Fallback to quiz best grade if failure or not found.
 | ||||||
|  |             this.gradebookData = { | ||||||
|  |                 grade: this.bestGrade.grade, | ||||||
|  |             }; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Component being destroyed. | ||||||
|  |      */ | ||||||
|  |     ngOnDestroy(): void { | ||||||
|  |         super.ngOnDestroy(); | ||||||
|  | 
 | ||||||
|  |         this.finishedObserver?.off(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
							
								
								
									
										22
									
								
								src/addons/mod/quiz/pages/index/index.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/addons/mod/quiz/pages/index/index.html
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,22 @@ | |||||||
|  | <ion-header> | ||||||
|  |     <ion-toolbar> | ||||||
|  |         <ion-buttons slot="start"> | ||||||
|  |             <ion-back-button [attr.aria-label]="'core.back' | translate"></ion-back-button> | ||||||
|  |         </ion-buttons> | ||||||
|  |         <ion-title> | ||||||
|  |             <core-format-text [text]="title" contextLevel="module" [contextInstanceId]="module?.id" [courseId]="courseId"> | ||||||
|  |             </core-format-text> | ||||||
|  |         </ion-title> | ||||||
|  | 
 | ||||||
|  |         <ion-buttons slot="end"> | ||||||
|  |             <!-- The buttons defined by the component will be added in here. --> | ||||||
|  |         </ion-buttons> | ||||||
|  |     </ion-toolbar> | ||||||
|  | </ion-header> | ||||||
|  | <ion-content> | ||||||
|  |     <ion-refresher slot="fixed" [disabled]="!quizComponent?.loaded" (ionRefresh)="quizComponent?.doRefresh($event)"> | ||||||
|  |         <ion-refresher-content pullingText="{{ 'core.pulltorefresh' | translate }}"></ion-refresher-content> | ||||||
|  |     </ion-refresher> | ||||||
|  | 
 | ||||||
|  |     <addon-mod-quiz-index [module]="module" [courseId]="courseId" (dataRetrieved)="updateData($event)"></addon-mod-quiz-index> | ||||||
|  | </ion-content> | ||||||
							
								
								
									
										40
									
								
								src/addons/mod/quiz/pages/index/index.module.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								src/addons/mod/quiz/pages/index/index.module.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,40 @@ | |||||||
|  | // (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 { NgModule } from '@angular/core'; | ||||||
|  | import { RouterModule, Routes } from '@angular/router'; | ||||||
|  | 
 | ||||||
|  | import { CoreSharedModule } from '@/core/shared.module'; | ||||||
|  | import { AddonModQuizComponentsModule } from '../../components/components.module'; | ||||||
|  | import { AddonModQuizIndexPage } from './index'; | ||||||
|  | 
 | ||||||
|  | const routes: Routes = [ | ||||||
|  |     { | ||||||
|  |         path: '', | ||||||
|  |         component: AddonModQuizIndexPage, | ||||||
|  |     }, | ||||||
|  | ]; | ||||||
|  | 
 | ||||||
|  | @NgModule({ | ||||||
|  |     imports: [ | ||||||
|  |         RouterModule.forChild(routes), | ||||||
|  |         CoreSharedModule, | ||||||
|  |         AddonModQuizComponentsModule, | ||||||
|  |     ], | ||||||
|  |     declarations: [ | ||||||
|  |         AddonModQuizIndexPage, | ||||||
|  |     ], | ||||||
|  |     exports: [RouterModule], | ||||||
|  | }) | ||||||
|  | export class AddonModQuizIndexPageModule {} | ||||||
							
								
								
									
										69
									
								
								src/addons/mod/quiz/pages/index/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								src/addons/mod/quiz/pages/index/index.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,69 @@ | |||||||
|  | // (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, OnInit, ViewChild } from '@angular/core'; | ||||||
|  | 
 | ||||||
|  | import { CoreCourseWSModule } from '@features/course/services/course'; | ||||||
|  | import { CoreNavigator } from '@services/navigator'; | ||||||
|  | import { AddonModQuizIndexComponent } from '../../components/index'; | ||||||
|  | import { AddonModQuizQuizWSData } from '../../services/quiz'; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Page that displays the quiz entry page. | ||||||
|  |  */ | ||||||
|  | @Component({ | ||||||
|  |     selector: 'page-addon-mod-quiz-index', | ||||||
|  |     templateUrl: 'index.html', | ||||||
|  | }) | ||||||
|  | export class AddonModQuizIndexPage implements OnInit { | ||||||
|  | 
 | ||||||
|  |     @ViewChild(AddonModQuizIndexComponent) quizComponent?: AddonModQuizIndexComponent; | ||||||
|  | 
 | ||||||
|  |     title?: string; | ||||||
|  |     module?: CoreCourseWSModule; | ||||||
|  |     courseId?: number; | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Component being initialized. | ||||||
|  |      */ | ||||||
|  |     ngOnInit(): void { | ||||||
|  |         this.module = CoreNavigator.instance.getRouteParam('module'); | ||||||
|  |         this.courseId = CoreNavigator.instance.getRouteNumberParam('courseId'); | ||||||
|  |         this.title = this.module?.name; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Update some data based on the quiz instance. | ||||||
|  |      * | ||||||
|  |      * @param quiz Quiz instance. | ||||||
|  |      */ | ||||||
|  |     updateData(quiz: AddonModQuizQuizWSData): void { | ||||||
|  |         this.title = quiz.name || this.title; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * User entered the page. | ||||||
|  |      */ | ||||||
|  |     ionViewDidEnter(): void { | ||||||
|  |         this.quizComponent?.ionViewDidEnter(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * User left the page. | ||||||
|  |      */ | ||||||
|  |     ionViewDidLeave(): void { | ||||||
|  |         this.quizComponent?.ionViewDidLeave(); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
							
								
								
									
										28
									
								
								src/addons/mod/quiz/quiz-lazy.module.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								src/addons/mod/quiz/quiz-lazy.module.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,28 @@ | |||||||
|  | // (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 { NgModule } from '@angular/core'; | ||||||
|  | import { RouterModule, Routes } from '@angular/router'; | ||||||
|  | 
 | ||||||
|  | const routes: Routes = [ | ||||||
|  |     { | ||||||
|  |         path: ':courseId/:cmdId', | ||||||
|  |         loadChildren: () => import('./pages/index/index.module').then( m => m.AddonModQuizIndexPageModule), | ||||||
|  |     }, | ||||||
|  | ]; | ||||||
|  | 
 | ||||||
|  | @NgModule({ | ||||||
|  |     imports: [RouterModule.forChild(routes)], | ||||||
|  | }) | ||||||
|  | export class AddonModQuizLazyModule {} | ||||||
							
								
								
									
										56
									
								
								src/addons/mod/quiz/quiz.module.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								src/addons/mod/quiz/quiz.module.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,56 @@ | |||||||
|  | // (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 { APP_INITIALIZER, NgModule } from '@angular/core'; | ||||||
|  | import { Routes } from '@angular/router'; | ||||||
|  | 
 | ||||||
|  | import { CoreCourseModuleDelegate } from '@features/course/services/module-delegate'; | ||||||
|  | import { CoreCourseModulePrefetchDelegate } from '@features/course/services/module-prefetch-delegate'; | ||||||
|  | import { CoreMainMenuTabRoutingModule } from '@features/mainmenu/mainmenu-tab-routing.module'; | ||||||
|  | import { CORE_SITE_SCHEMAS } from '@services/sites'; | ||||||
|  | import { AddonModQuizComponentsModule } from './components/components.module'; | ||||||
|  | import { SITE_SCHEMA } from './services/database/quiz'; | ||||||
|  | import { AddonModQuizModuleHandler, AddonModQuizModuleHandlerService } from './services/handlers/module'; | ||||||
|  | import { AddonModQuizPrefetchHandler } from './services/handlers/prefetch'; | ||||||
|  | 
 | ||||||
|  | const routes: Routes = [ | ||||||
|  |     { | ||||||
|  |         path: AddonModQuizModuleHandlerService.PAGE_NAME, | ||||||
|  |         loadChildren: () => import('./quiz-lazy.module').then(m => m.AddonModQuizLazyModule), | ||||||
|  |     }, | ||||||
|  | ]; | ||||||
|  | 
 | ||||||
|  | @NgModule({ | ||||||
|  |     imports: [ | ||||||
|  |         CoreMainMenuTabRoutingModule.forChild(routes), | ||||||
|  |         AddonModQuizComponentsModule, | ||||||
|  |     ], | ||||||
|  |     providers: [ | ||||||
|  |         { | ||||||
|  |             provide: CORE_SITE_SCHEMAS, | ||||||
|  |             useValue: [SITE_SCHEMA], | ||||||
|  |             multi: true, | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             provide: APP_INITIALIZER, | ||||||
|  |             multi: true, | ||||||
|  |             deps: [], | ||||||
|  |             useFactory: () => () => { | ||||||
|  |                 CoreCourseModuleDelegate.instance.registerHandler(AddonModQuizModuleHandler.instance); | ||||||
|  |                 CoreCourseModulePrefetchDelegate.instance.registerHandler(AddonModQuizPrefetchHandler.instance); | ||||||
|  |             }, | ||||||
|  |         }, | ||||||
|  |     ], | ||||||
|  | }) | ||||||
|  | export class AddonModQuizModule {} | ||||||
							
								
								
									
										98
									
								
								src/addons/mod/quiz/services/handlers/module.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								src/addons/mod/quiz/services/handlers/module.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,98 @@ | |||||||
|  | // (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 { Injectable, Type } from '@angular/core'; | ||||||
|  | 
 | ||||||
|  | import { CoreConstants } from '@/core/constants'; | ||||||
|  | import { CoreCourseModuleHandler, CoreCourseModuleHandlerData } from '@features/course/services/module-delegate'; | ||||||
|  | import { CoreCourse, CoreCourseAnyModuleData } from '@features/course/services/course'; | ||||||
|  | import { CoreCourseModule } from '@features/course/services/course-helper'; | ||||||
|  | import { CoreNavigationOptions, CoreNavigator } from '@services/navigator'; | ||||||
|  | import { AddonModQuizIndexComponent } from '../../components/index'; | ||||||
|  | import { makeSingleton } from '@singletons'; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Handler to support quiz modules. | ||||||
|  |  */ | ||||||
|  | @Injectable({ providedIn: 'root' }) | ||||||
|  | export class AddonModQuizModuleHandlerService implements CoreCourseModuleHandler { | ||||||
|  | 
 | ||||||
|  |     static readonly PAGE_NAME = 'mod_quiz'; | ||||||
|  | 
 | ||||||
|  |     name = 'AddonModQuiz'; | ||||||
|  |     modName = 'quiz'; | ||||||
|  | 
 | ||||||
|  |     supportedFeatures = { | ||||||
|  |         [CoreConstants.FEATURE_GROUPS]: true, | ||||||
|  |         [CoreConstants.FEATURE_GROUPINGS]: true, | ||||||
|  |         [CoreConstants.FEATURE_MOD_INTRO]: true, | ||||||
|  |         [CoreConstants.FEATURE_COMPLETION_TRACKS_VIEWS]: true, | ||||||
|  |         [CoreConstants.FEATURE_COMPLETION_HAS_RULES]: true, | ||||||
|  |         [CoreConstants.FEATURE_GRADE_HAS_GRADE]: true, | ||||||
|  |         [CoreConstants.FEATURE_GRADE_OUTCOMES]: true, | ||||||
|  |         [CoreConstants.FEATURE_BACKUP_MOODLE2]: true, | ||||||
|  |         [CoreConstants.FEATURE_SHOW_DESCRIPTION]: true, | ||||||
|  |         [CoreConstants.FEATURE_CONTROLS_GRADE_VISIBILITY]: true, | ||||||
|  |         [CoreConstants.FEATURE_USES_QUESTIONS]: true, | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Check if the handler is enabled on a site level. | ||||||
|  |      * | ||||||
|  |      * @return Whether or not the handler is enabled on a site level. | ||||||
|  |      */ | ||||||
|  |     async isEnabled(): Promise<boolean> { | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Get the data required to display the module in the course contents view. | ||||||
|  |      * | ||||||
|  |      * @param module The module object. | ||||||
|  |      * @param courseId The course ID. | ||||||
|  |      * @param sectionId The section ID. | ||||||
|  |      * @return Data to render the module. | ||||||
|  |      */ | ||||||
|  |     getData(module: CoreCourseAnyModuleData): CoreCourseModuleHandlerData { | ||||||
|  |         return { | ||||||
|  |             icon: CoreCourse.instance.getModuleIconSrc(this.modName, 'modicon' in module ? module.modicon : undefined), | ||||||
|  |             title: module.name, | ||||||
|  |             class: 'addon-mod_quiz-handler', | ||||||
|  |             showDownloadButton: true, | ||||||
|  |             action: (event: Event, module: CoreCourseModule, courseId: number, options?: CoreNavigationOptions) => { | ||||||
|  |                 options = options || {}; | ||||||
|  |                 options.params = options.params || {}; | ||||||
|  |                 Object.assign(options.params, { module }); | ||||||
|  |                 const routeParams = '/' + courseId + '/' + module.id; | ||||||
|  | 
 | ||||||
|  |                 CoreNavigator.instance.navigateToSitePath(AddonModQuizModuleHandlerService.PAGE_NAME + routeParams, options); | ||||||
|  |             }, | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Get the component to render the module. This is needed to support singleactivity course format. | ||||||
|  |      * The component returned must implement CoreCourseModuleMainComponent. | ||||||
|  |      * | ||||||
|  |      * @param course The course object. | ||||||
|  |      * @param module The module object. | ||||||
|  |      * @return The component to use, undefined if not found. | ||||||
|  |      */ | ||||||
|  |     async getMainComponent(): Promise<Type<unknown>> { | ||||||
|  |         return AddonModQuizIndexComponent; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export class AddonModQuizModuleHandler extends makeSingleton(AddonModQuizModuleHandlerService) {} | ||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user