File

src/app/curriculum-courses/chapter-details/chapter-details.page.ts

Implements

OnInit OnDestroy ConsentPopoverActionsDelegate

Metadata

Index

Properties
Methods

Constructor

constructor(profileService: ProfileService, preferences: SharedPreferences, authService: AuthService, courseService: CourseService, downloadService: DownloadService, eventsBusService: EventsBusService, contentService: ContentService, headerService: AppHeaderService, translate: TranslateService, commonUtilService: CommonUtilService, router: Router, loginHandlerService: LoginHandlerService, appGlobalService: AppGlobalService, popoverCtrl: PopoverController, localCourseService: LocalCourseService, events: Events, zone: NgZone, datePipe: DatePipe, fileSizePipe: FileSizePipe, sbProgressLoader: SbProgressLoader, telemetryGeneratorService: TelemetryGeneratorService, location: Location, platform: Platform, contentPlayerHandler: ContentPlayerHandler, categoryKeyTranslator: CategoryKeyTranslator, tncUpdateHandlerService: TncUpdateHandlerService)
Parameters :
Name Type Optional
profileService ProfileService No
preferences SharedPreferences No
authService AuthService No
courseService CourseService No
downloadService DownloadService No
eventsBusService EventsBusService No
contentService ContentService No
headerService AppHeaderService No
translate TranslateService No
commonUtilService CommonUtilService No
router Router No
loginHandlerService LoginHandlerService No
appGlobalService AppGlobalService No
popoverCtrl PopoverController No
localCourseService LocalCourseService No
events Events No
zone NgZone No
datePipe DatePipe No
fileSizePipe FileSizePipe No
sbProgressLoader SbProgressLoader No
telemetryGeneratorService TelemetryGeneratorService No
location Location No
platform Platform No
contentPlayerHandler ContentPlayerHandler No
categoryKeyTranslator CategoryKeyTranslator No
tncUpdateHandlerService TncUpdateHandlerService No

Methods

checkChapterCompletion
checkChapterCompletion()
Returns : void
Async checkLoggedInOrGuestUser
checkLoggedInOrGuestUser()
Returns : any
Async continueLearning
continueLearning()
Returns : any
Async enrollIntoBatch
enrollIntoBatch(item: Batch)
Parameters :
Name Type Optional
item Batch No
Returns : any
Private generateContentNavExtras
generateContentNavExtras(content: Content, depth)
Parameters :
Name Type Optional
content Content No
depth No
Returns : NavigationExtras
generateImpressionEvent
generateImpressionEvent(objectId, objectType, objectVersion)
Parameters :
Name Optional
objectId No
objectType No
objectVersion No
Returns : void
Async getAllBatches
getAllBatches()
Returns : any
getAllContents
getAllContents(collection)
Parameters :
Name Optional
collection No
Returns : void
Async getBatchDetails
getBatchDetails()
Returns : any
getContentsSize
getContentsSize(data?)
Parameters :
Name Optional
data Yes
Returns : void
Async getContentState
getContentState(returnRefresh: boolean)
Parameters :
Name Type Optional
returnRefresh boolean No
Returns : any
getImportContentRequestBody
getImportContentRequestBody(identifiers, isChild: boolean)
Parameters :
Name Type Optional
identifiers No
isChild boolean No
Returns : Array<ContentImport>
Private getNextContent
getNextContent(courseHeirarchy, contentStateList: ContentState[])
Parameters :
Name Type Optional
courseHeirarchy No
contentStateList ContentState[] No
Returns : Content
Private getSubContentIds
getSubContentIds(content)
Parameters :
Name Optional
content No
Returns : Array<string>
handleHeaderEvents
handleHeaderEvents($event)
Parameters :
Name Optional
$event No
Returns : void
importContent
importContent(identifiers, isChild: boolean, isDownloadAllClicked?)
Parameters :
Name Type Optional
identifiers No
isChild boolean No
isDownloadAllClicked Yes
Returns : void
ionViewDidEnter
ionViewDidEnter()
Returns : void
Async ionViewWillEnter
ionViewWillEnter()
Returns : any
ionViewWillLeave
ionViewWillLeave()
Returns : void
Async joinTraining
joinTraining()
Returns : any
loadFirstChildren
loadFirstChildren(data)

get first child of unit

Parameters :
Name Optional
data No
Returns : any
Async navigateToBatchListPage
navigateToBatchListPage()

checks whether batches are available or not and then Navigate user to batch list page

Returns : any
navigateToChildrenDetailsPage
navigateToChildrenDetailsPage(content: Content, depth)

Redirect to child content details page

Parameters :
Name Type Optional
content Content No
depth No
Returns : void
ngOnDestroy
ngOnDestroy()
Returns : void
ngOnInit
ngOnInit()
Returns : void
onConsentPopoverDismiss
onConsentPopoverDismiss()
Returns : void
onConsentPopoverShow
onConsentPopoverShow()
Returns : void
openContentDetails
openContentDetails(event)
Parameters :
Name Optional
event No
Returns : void
Async promptToLogin
promptToLogin(batchdetail)
Parameters :
Name Optional
batchdetail No
Returns : any
saveContentContext
saveContentContext(userId, courseId, batchId, batchStatus)
Parameters :
Name Optional
userId No
courseId No
batchId No
batchStatus No
Returns : void
Async share
share()
Returns : any
Async showDownloadConfirmationAlert
showDownloadConfirmationAlert()
Returns : any
Async showOverflowMenu
showOverflowMenu(event)
Parameters :
Name Optional
event No
Returns : any
Private Async showProfileNameConfirmationPopup
showProfileNameConfirmationPopup()
Returns : any
Private Async startContent
startContent()
Returns : any
Async startLearning
startLearning()
Returns : any
subscribeSdkEvent
subscribeSdkEvent()

Subscribe Sunbird-SDK event to get content download progress

Returns : void
Async subscribeUtilityEvents
subscribeUtilityEvents()
Returns : any

Properties

Public authService
Type : AuthService
Decorators :
@Inject('AUTH_SERVICE')
backButtonFunc
Type : Subscription
batchDetails
Type : Batch
batches
Type : []
Default value : []
batchExp
Default value : false
cardType
Type : TocCardType
Default value : TocCardType.COURSE
chapter
Type : any
chapterProgress
Type : number
Default value : 0
childContents
Type : []
Default value : []
contentStatusData
Type : ContentStateResponse
Private corRelationList
Type : any
courseBatchesRequest
Type : CourseBatchesRequest
courseCardData
Type : any
courseContent
Type : any
courseContentData
Type : any
courseHeirarchy
Type : any
courseStartDate
currentCount
Type : number
Default value : 0
downloadIdentifiers
Default value : new Set()
downloadProgress
Type : number
Default value : 0
downloadSize
Type : number
Default value : 0
Private eventSubscription
Type : Subscription
Private extrasData
Type : any
faultyIdentifiers
Type : []
Default value : []
guestUser
Default value : true
Private hasInit
Default value : false
headerObservable
Type : Subscription
identifier
Type : string
isAlreadyEnrolled
Default value : false
isBatchNotStarted
Default value : false
isCertifiedCourse
Type : boolean
isChapterCompleted
Default value : false
isChapterStarted
Default value : false
isDownloadStarted
Default value : false
isFirstContent
Default value : false
isFromDeeplink
Default value : false
isNextContentFound
Default value : false
Optional loader
Type : HTMLIonLoadingElement
maxAssessmentLimit
Default value : AssessmentConstant.MAX_ATTEMPTS
nextContent
Type : Content
Public objRollup
Type : Rollup
queuedIdentifiers
Type : []
Default value : []
Public rollUpMap
Type : literal type
Default value : {}
showCollapsedPopup
Default value : true
showDownload
Type : boolean
subContentIds
Type : Array<string>
Default value : []
telemetryObject
Type : TelemetryObject
Public todayDate
Type : any
trackDownloads$
Type : Observable<DownloadTracking>
updatedCourseCardData
Type : Course
userId
viewedContents
Type : []
Default value : []
import { share } from 'rxjs/operators';
import { SbSharePopupComponent } from '@app/app/components/popups/sb-share-popup/sb-share-popup.component';
import { Component, OnInit, Inject, NgZone, OnDestroy } from '@angular/core';
import {
  AppHeaderService, CommonUtilService, LoginHandlerService, AppGlobalService, LocalCourseService,
  TelemetryGeneratorService
} from '@app/services';
import { TranslateService } from '@ngx-translate/core';
import { Router, NavigationExtras } from '@angular/router';
import { TocCardType } from '@project-sunbird/common-consumption';
import { SbPopoverComponent } from '@app/app/components/popups/sb-popover/sb-popover.component';
import { PopoverController, Platform } from '@ionic/angular';
import { Events } from '@app/util/events';
import {
  RouterLinks, PreferenceKey, EventTopics, AssessmentConstant,
  MimeType, ShareItemType, BatchConstants, ProfileConstants, MaxAttempt
} from '@app/app/app.constant';
import {
  SharedPreferences, AuthService, Batch, TelemetryObject, ContentState, Content, Course,
  CourseService, GetContentStateRequest, ContentStateResponse, CourseBatchStatus,
  CourseEnrollmentType, SortOrder, DownloadService, DownloadTracking, DownloadProgress,
  EventsBusEvent, DownloadEventType, EventsBusService, ContentImportRequest, ContentService,
  ContentImportResponse, ContentImportStatus, ContentEventType, ContentImportCompleted,
  ContentUpdate, ContentImport, Rollup, AuditState, ProfileService, CourseBatchesRequest
} from 'sunbird-sdk';
import { EnrollCourse } from '@app/app/enrolled-course-details-page/course.interface';
import { DatePipe, Location } from '@angular/common';
import { ContentActionsComponent } from './../../components/content-actions/content-actions.component';
import {
  PageId, Environment, InteractType,
  InteractSubtype, ImpressionType, AuditType
} from './../../../services/telemetry-constants';
import { Observable, Subscription } from 'rxjs';
import { ConfirmAlertComponent } from '@app/app/components';
import { FileSizePipe } from '@app/pipes/file-size/file-size';
import { ContentUtil } from '@app/util/content-util';
import { SbProgressLoader } from '@app/services/sb-progress-loader.service';
import { ContentPlayerHandler } from '@app/services/content/player/content-player-handler';
import { ConsentPopoverActionsDelegate } from '@app/services/local-course.service';
import { CategoryKeyTranslator } from '@app/pipes/category-key-translator/category-key-translator-pipe';
import {
  ProfileNameConfirmationPopoverComponent
} from '@app/app/components/popups/sb-profile-name-confirmation-popup/sb-profile-name-confirmation-popup.component';
import { TncUpdateHandlerService } from '@app/services/handlers/tnc-update-handler.service';

@Component({
  selector: 'app-chapter-details',
  templateUrl: './chapter-details.page.html',
  styleUrls: ['./chapter-details.page.scss', '../../enrolled-course-details-page/enrolled-course-details-page.scss'],
})
export class ChapterDetailsPage implements OnInit, OnDestroy, ConsentPopoverActionsDelegate {

  chapter: any;
  cardType: TocCardType = TocCardType.COURSE;
  isAlreadyEnrolled = false;
  batches = [];
  courseContentData: any;
  courseContent: any;
  batchExp = false;
  guestUser = true;
  isChapterCompleted = false;
  isChapterStarted = false;
  isBatchNotStarted = false;
  isFromDeeplink = false;
  userId;
  telemetryObject: TelemetryObject;
  updatedCourseCardData: Course;
  contentStatusData: ContentStateResponse;
  batchDetails: Batch;
  childContents = [];
  viewedContents = [];
  chapterProgress = 0;
  courseStartDate;
  // Contains identifier(s) of locally not available content(s)
  downloadIdentifiers = new Set();
  // Contains total size of locally not available content(s)
  downloadSize = 0;
  // Contains child content import / download progress
  downloadProgress = 0;
  public rollUpMap: { [key: string]: Rollup } = {};
  identifier: string;
  isDownloadStarted = false;
  showCollapsedPopup = true;
  showDownload: boolean;
  private eventSubscription: Subscription;
  queuedIdentifiers = [];
  faultyIdentifiers = [];
  currentCount = 0;
  subContentIds: Array<string> = [];

  trackDownloads$: Observable<DownloadTracking>;
  public todayDate: any;

  private extrasData: any;

  isNextContentFound = false;
  isFirstContent = false;
  nextContent: Content;
  headerObservable: Subscription;
  backButtonFunc: Subscription;
  public objRollup: Rollup;
  private corRelationList: any;
  loader?: HTMLIonLoadingElement;
  maxAssessmentLimit = AssessmentConstant.MAX_ATTEMPTS;
  isCertifiedCourse: boolean;
  courseHeirarchy: any;
  private hasInit = false;
  courseBatchesRequest: CourseBatchesRequest;
  courseCardData: any;

  constructor(
    @Inject('PROFILE_SERVICE') private profileService: ProfileService,
    @Inject('SHARED_PREFERENCES') private preferences: SharedPreferences,
    @Inject('AUTH_SERVICE') public authService: AuthService,
    @Inject('COURSE_SERVICE') private courseService: CourseService,
    @Inject('DOWNLOAD_SERVICE') private downloadService: DownloadService,
    @Inject('EVENTS_BUS_SERVICE') private eventsBusService: EventsBusService,
    @Inject('CONTENT_SERVICE') private contentService: ContentService,
    private headerService: AppHeaderService,
    private translate: TranslateService,
    private commonUtilService: CommonUtilService,
    private router: Router,
    private loginHandlerService: LoginHandlerService,
    private appGlobalService: AppGlobalService,
    private popoverCtrl: PopoverController,
    private localCourseService: LocalCourseService,
    private events: Events,
    private zone: NgZone,
    private datePipe: DatePipe,
    private fileSizePipe: FileSizePipe,
    private sbProgressLoader: SbProgressLoader,
    private telemetryGeneratorService: TelemetryGeneratorService,
    private location: Location,
    private platform: Platform,
    private contentPlayerHandler: ContentPlayerHandler,
    private categoryKeyTranslator: CategoryKeyTranslator,
    private tncUpdateHandlerService: TncUpdateHandlerService,
  ) {
    this.extrasData = this.router.getCurrentNavigation().extras.state;
    this.appGlobalService.preSignInData = null;
    this.courseContent = this.extrasData.courseContent;
    this.chapter = this.extrasData.chapterData;
    this.batches = this.extrasData.batches;
    this.isAlreadyEnrolled = this.extrasData.isAlreadyEnrolled;
    this.batchExp = this.extrasData.batchExp;
    this.isChapterCompleted = this.extrasData.isChapterCompleted;
    this.contentStatusData = this.extrasData.contentStatusData;
    this.isFromDeeplink = this.extrasData.isFromDeeplink;
    this.courseHeirarchy = this.extrasData.courseHeirarchy;
    this.courseContentData = this.courseContent;
    this.identifier = this.chapter.identifier;
    this.telemetryObject = ContentUtil.getTelemetryObject(this.chapter);
    this.corRelationList = this.extrasData.corRelation;
    this.courseCardData = this.extrasData.courseCardData;
  }

  ngOnInit() {
    this.subContentIds = [];
    this.getSubContentIds(this.chapter);
    if (this.chapter.children && this.chapter.children.length) {
      this.chapter.children.map((content) => {
        const displayScore = content && content.contentData && content.contentData.displayScore;
        content['displayScore'] = displayScore === undefined ? true : displayScore;
      });
    }
    if (this.chapter.hierarchyInfo) {
      this.objRollup = ContentUtil.generateRollUp(this.chapter.hierarchyInfo, this.identifier);
    }
    this.generateImpressionEvent(this.chapter.identifier, this.chapter.contentType, this.chapter.contentData.pkgVersion);
    this.trackDownloads$ = this.downloadService.trackDownloads(
      { groupBy: { fieldPath: 'rollUp.l1', value: this.courseContentData.identifier } }).pipe(share());
  }

  async ionViewWillEnter() {
    this.downloadIdentifiers = new Set();
    this.headerService.showHeaderWithBackButton();
    this.todayDate = window.dayjs().format('YYYY-MM-DD');
    this.subscribeUtilityEvents();
    this.subscribeSdkEvent();
    await this.checkLoggedInOrGuestUser();
    this.childContents = [];
    if (this.isFromDeeplink) {
      this.getAllBatches();
    }
    this.getAllContents(this.chapter);
    this.checkChapterCompletion();
    this.headerObservable = this.headerService.headerEventEmitted$.subscribe(event => {
      this.handleHeaderEvents(event);
    });
    this.backButtonFunc = this.platform.backButton.subscribeWithPriority(10, () => {
      this.telemetryGeneratorService.generateBackClickedTelemetry(PageId.CHAPTER_DETAILS, Environment.HOME, false);
      this.appGlobalService.generateCourseUnitCompleteTelemetry = false;
      this.location.back();
    });
    if (this.hasInit) {
      this.getContentState(false);
    } else {
      this.hasInit = !this.hasInit;
      this.getContentState(true);
    }

    if (!this.guestUser) {
      this.updatedCourseCardData = await this.courseService.getEnrolledCourses({ userId: this.userId, returnFreshCourses: false })
        .toPromise()
        .then((data) => {
          if (data.length > 0) {
            console.log('getEnrolledCourses', data);
            const courseList: Array<Course> = [];
            for (const course of data) {
              courseList.push(course);
            }
            this.appGlobalService.setEnrolledCourseList(courseList);
          }
          return data.find((element) =>
            (this.courseCardData.batchId && element.batchId === this.courseCardData.batchId)
            || (!this.courseCardData.batchId && element.courseId === this.courseCardData.identifier));
        })
        .catch(e => {
          console.log(e);
          return null;
        });
      if (this.updatedCourseCardData && !this.courseContent.batch) {
        this.courseContent = this.updatedCourseCardData;
        this.isAlreadyEnrolled = true;
      }
      if (this.isFromDeeplink) {
        this.getContentState(true);
      }
      this.getBatchDetails();
      console.log('this.courseCardData', this.courseContent);
      this.getContentsSize(this.chapter.children);
    }
  }
  handleHeaderEvents($event) {
    if ($event.name === 'back') {
      this.telemetryGeneratorService.generateBackClickedTelemetry(PageId.CHAPTER_DETAILS, Environment.HOME, true);
      this.location.back();
      this.appGlobalService.generateCourseUnitCompleteTelemetry = false;
    }
  }

  ionViewWillLeave(): void {
    this.events.publish('header:setzIndexToNormal');
    if (this.eventSubscription) {
      this.eventSubscription.unsubscribe();
    }
    if (this.headerObservable) {
      this.headerObservable.unsubscribe();
    }
    if (this.backButtonFunc) {
      this.backButtonFunc.unsubscribe();
    }
  }

  ionViewDidEnter(): void {
    this.sbProgressLoader.hide({ id: 'login' });
    this.sbProgressLoader.hide({ id: this.courseContent.identifier });
  }

  ngOnDestroy() {
    this.events.unsubscribe(EventTopics.ENROL_COURSE_SUCCESS);
  }

  async getAllBatches() {
    this.courseBatchesRequest = {
      filters: {
        courseId: this.courseContentData.identifier,
        status: [CourseBatchStatus.NOT_STARTED, CourseBatchStatus.IN_PROGRESS],
        enrollmentType: CourseEnrollmentType.OPEN
      },
      sort_by: { createdDate: SortOrder.DESC },
      fields: BatchConstants.REQUIRED_FIELDS
    };
    this.courseService.getCourseBatches(this.courseBatchesRequest).toPromise()
      .then(async (data: Batch[]) => {
        this.batches = data || [];
      })
      .catch(async (error: any) => {
        console.log('Error while fetching all Batches', error);
      });
  }

  async checkLoggedInOrGuestUser() {
    const session = await this.authService.getSession().toPromise();
    this.guestUser = !session;
    if (session) {
      this.userId = session.userToken;
    }
  }

  async getContentState(returnRefresh: boolean) {
    if (this.courseContent.batchId) {
      const request: GetContentStateRequest = {
        userId: this.appGlobalService.getUserId(),
        courseId: this.courseContentData.identifier,
        contentIds: this.courseContent && this.courseContent.contentData ?
          this.courseContent.contentData.leafNodes : this.courseContentData.contentData.leafNodes,
        returnRefreshedContentStates: returnRefresh,
        batchId: this.courseContent.batchId,
        fields: ['progress', 'score']
      };
      this.courseService.getContentState(request).toPromise()
        .then(async (res: ContentStateResponse) => {
          this.zone.run(() => {
            this.contentStatusData = res;
            this.checkChapterCompletion();
          }); }).catch(async () => {});
    }
  }

  async getBatchDetails() {
    if (this.courseContent && this.courseContent.batchId) {
      this.courseService.getBatchDetails({ batchId: this.courseContent.batchId }).toPromise()
        .then((data: Batch) => {
          this.zone.run(() => {
            if (!data) {
              return;
            }
            this.batchDetails = data;
            if (data.cert_templates && Object.keys(data.cert_templates).length) {
              this.isCertifiedCourse = true;
            } else {
              this.isCertifiedCourse = false;
            }
            if (this.batchDetails.status === 2) {
              this.batchExp = true;
            } else if (this.batchDetails.status === 0) {
              this.isBatchNotStarted = true;
              this.courseStartDate = this.batchDetails.startDate;
            }
            this.saveContentContext(this.appGlobalService.getUserId(),
              this.batchDetails.courseId, this.courseContent.batchId, this.batchDetails.status);
          });
        }).catch((err) => {
          this.saveContentContext(this.appGlobalService.getUserId(),
            this.courseContent.courseId, this.courseContent.batchId, this.courseContent.batch.status);
        });
    }
  }

  saveContentContext(userId, courseId, batchId, batchStatus) {
    const contentContextMap = new Map();
    // store content context in the below map
    contentContextMap['userId'] = userId;
    contentContextMap['courseId'] = courseId;
    contentContextMap['batchId'] = batchId;
    contentContextMap['isCertified'] = this.isCertifiedCourse;
    if (batchStatus) {
      contentContextMap['batchStatus'] = batchStatus;
    }
    if (this.courseContentData && this.courseContentData.contentData && this.courseContentData.contentData.leafNodes) {
      const leafNodeIds = this.courseContentData.contentData.leafNodes;
      contentContextMap['leafNodeIds'] = leafNodeIds;
    }

    // store the contentContextMap in shared preference and access it from SDK
    this.preferences.putString(PreferenceKey.CONTENT_CONTEXT, JSON.stringify(contentContextMap)).toPromise().then();
  }

  getAllContents(collection) {
    if (collection.children) {
      collection.children.forEach(element => {
        this.getAllContents(element);
      });
    } else {
      if (collection.mimeType !== MimeType.COLLECTION) {
        this.childContents.push(collection);
      }
    }
  }

  checkChapterCompletion() {
    if (this.contentStatusData && this.contentStatusData.contentList.length) {
      this.viewedContents = [];
      for (const content of this.childContents) {
        if (this.contentStatusData.contentList.find((c) => c.contentId === content.identifier && c.status === 2)) {
          this.viewedContents.push(content);
        }
      }
      if (this.viewedContents.length && this.viewedContents.length === this.childContents.length) {
        this.isChapterCompleted = true;
      } else if (this.viewedContents.length) {
        this.isChapterStarted = true;
      } else {
        this.isChapterStarted = false;
      }
      if (this.viewedContents.length) {
        this.chapterProgress = Math.round((this.viewedContents.length / this.childContents.length) * 100);
        console.log('chapterProgress', this.chapterProgress);
      }
      if (!this.chapterProgress || this.chapterProgress !== 100) {
        this.appGlobalService.generateCourseUnitCompleteTelemetry = true;
      }
      if (this.appGlobalService.generateCourseUnitCompleteTelemetry && this.chapterProgress === 100) {
        this.appGlobalService.generateCourseUnitCompleteTelemetry = false;
        const cdata = [
          {
            type: 'CourseId',
            id: this.courseContentData.identifier || ''
          },
          {
            type: 'BatchId',
            id: this.courseContent.batchId || ''
          },
          {
            type: 'UserId',
            id: this.userId || ''
          },
          {
            type: 'UnitId',
            id: this.identifier || ''
          },
        ];
        this.telemetryGeneratorService.generateAuditTelemetry(
          Environment.COURSE,
          AuditState.AUDIT_UPDATED,
          ['progress'],
          AuditType.UNIT_PROGRESS,
          this.telemetryObject.id,
          this.telemetryObject.type,
          this.telemetryObject.version,
          cdata,
          this.telemetryObject.rollup
        );
      }
    }
  }

  async subscribeUtilityEvents() {
    this.events.subscribe(EventTopics.ENROL_COURSE_SUCCESS, async (res) => {
      console.log('enrol succ event');
      this.isAlreadyEnrolled = true;
      this.updatedCourseCardData = await this.courseService
        .getEnrolledCourses({ userId: this.appGlobalService.getUserId(), returnFreshCourses: true })
        .toPromise()
        .then((cData) => {
          return cData.find((element) => element.courseId === this.courseContentData.identifier);
        });
      this.courseContent.batchId = res.batchId;
      console.log('enrol succ event -->', this.courseContent);
      await this.getBatchDetails();
      this.getContentState(true);
    });

  }

  async startLearning() {
    this.telemetryGeneratorService.generateInteractTelemetry(InteractType.TOUCH,
      InteractSubtype.START_CLICKED,
      Environment.HOME,
      PageId.CHAPTER_DETAILS,
      this.telemetryObject,
      undefined,
      this.objRollup,
      this.corRelationList
    );

    const key = PreferenceKey.DO_NOT_SHOW_PROFILE_NAME_CONFIRMATION_POPUP + '-' + this.userId;
    const doNotShow = await this.preferences.getBoolean(key).toPromise();
    const profile = await this.profileService.getActiveSessionProfile({
      requiredFields: ProfileConstants.REQUIRED_FIELDS
    }).toPromise();

    if (doNotShow || await this.tncUpdateHandlerService.isSSOUser(profile) || !this.isCertifiedCourse) {
      this.startContent();
    } else {
      this.showProfileNameConfirmationPopup();
    }
  }

  private async startContent() {
    if (this.childContents && this.childContents.length && !this.isBatchNotStarted) {
      const firstChild = this.loadFirstChildren(this.chapter);
      const telemetryDetails = {
        pageId: PageId.CHAPTER_DETAILS,
        corRelationList: this.corRelationList
      };
      const assessmentStatus = this.localCourseService.fetchAssessmentStatus(this.contentStatusData, firstChild);

      const maxAttempt: MaxAttempt =  await this.commonUtilService.handleAssessmentStatus(assessmentStatus);
      if (maxAttempt.isCloseButtonClicked || maxAttempt.limitExceeded) {
        return;
      }

      this.contentPlayerHandler.playContent(firstChild, this.generateContentNavExtras(firstChild, 1), telemetryDetails, true);
    } else if (!this.childContents || !this.childContents.length) {
      this.commonUtilService.showToast('NO_CONTENT_AVAILABLE_IN_MODULE');
    } else {
      this.commonUtilService.showToast(this.commonUtilService.translateMessage('COURSE_WILL_BE_AVAILABLE',
        this.datePipe.transform(this.courseStartDate, 'mediumDate')));
    }
  }

  async continueLearning() {
    this.isNextContentFound = false;
    this.isFirstContent = false;
    this.nextContent = undefined;
    this.getNextContent(this.chapter, this.contentStatusData.contentList);

    if (this.nextContent) {
      this.telemetryGeneratorService.generateInteractTelemetry(InteractType.TOUCH,
        InteractSubtype.RESUME_CLICKED,
        Environment.HOME,
        PageId.CHAPTER_DETAILS,
        this.telemetryObject,
        undefined,
        this.objRollup,
      );

      const assessmentStatus = this.localCourseService.fetchAssessmentStatus(this.contentStatusData, this.nextContent);

      const maxAttempt: MaxAttempt =  await this.commonUtilService.handleAssessmentStatus(assessmentStatus);
      if (maxAttempt.isCloseButtonClicked || maxAttempt.limitExceeded) {
        return;
      }

      const telemetryDetails = {
        pageId: PageId.CHAPTER_DETAILS,
        corRelationList: this.corRelationList
      };
      this.contentPlayerHandler.playContent(this.nextContent, this.generateContentNavExtras(this.nextContent, 1), telemetryDetails, true);
    } else {
      this.startLearning();
    }
  }

  async showOverflowMenu(event) {
    const actionPopover = await this.popoverCtrl.create({
      component: ContentActionsComponent,
      event,
      cssClass: 'cd-leave-training-popup',
      showBackdrop: false,
      componentProps: {
        content: this.courseContentData,
        batchDetails: this.batchDetails,
        pageName: PageId.CHAPTER_DETAILS,
        chapter: this.chapter,
        downloadIdentifiers: this.downloadIdentifiers
      },
    });
    await actionPopover.present();
    const { data } = await actionPopover.onDidDismiss();
    if (data && data.download) {
      this.showDownloadConfirmationAlert();
    } else if (data.share) {
      this.share();
    }
  }

  async share() {
    this.telemetryGeneratorService.generateInteractTelemetry(
      InteractType.TOUCH,
      InteractSubtype.SHARE_CLICKED,
      Environment.HOME,
      PageId.CHAPTER_DETAILS
    );
    const popover = await this.popoverCtrl.create({
      component: SbSharePopupComponent,
      componentProps: {
        content: this.courseContentData,
        moduleId: this.chapter.identifier,
        subContentIds: this.subContentIds,
        objRollup: this.objRollup,
        pageId: PageId.CHAPTER_DETAILS,
        shareItemType: ShareItemType.ROOT_COLECTION
      },
      cssClass: 'sb-popover',
    });
    await popover.present();
  }

  openContentDetails(event) {
    if (!event) {
      return;
    }
    if (event.event && event.event.isDisabled) {
      this.commonUtilService.showToast('FRMELMNTS_IMSG_LASTATTMPTEXCD');
      return;
    }
    if (event.event && event.event.isLastAttempt) {
      this.commonUtilService.showToast('ASSESSMENT_LAST_ATTEMPT_MESSAGE');
    }
    if ((event.event && Object.keys(event.event).length !== 0) || event.isFromDeeplink) {
      if (this.courseContentData.contentData.createdBy !== this.userId) {
        if (!this.isAlreadyEnrolled) {
          if (!this.isBatchNotStarted) {
            this.joinTraining();
          }
        } else {
          if (this.isBatchNotStarted) {
            this.commonUtilService.showToast(this.commonUtilService.translateMessage('COURSE_WILL_BE_AVAILABLE',
              this.datePipe.transform(this.courseStartDate, 'mediumDate')));
          } else {
            this.navigateToChildrenDetailsPage(event.data, 1);
          }
        }
      }
    }
  }

  /**
   * checks whether batches are available or not and then Navigate user to batch list page
   */
  async navigateToBatchListPage() {
    const loader = await this.commonUtilService.getLoader();
    const reqvalues = new Map();
    reqvalues['enrollReq'] = this.courseBatchesRequest;
    this.telemetryGeneratorService.generateInteractTelemetry(InteractType.TOUCH,
      InteractSubtype.ENROLL_CLICKED, Environment.HOME,
      PageId.CHAPTER_DETAILS, this.telemetryObject, reqvalues, this.objRollup);

      if (!this.commonUtilService.networkInfo.isNetworkAvailable) {
      this.commonUtilService.showToast('ERROR_NO_INTERNET_MESSAGE');
      return;
    }


    if (!this.localCourseService.isEnrollable(this.batches, this.courseContentData)) {
      return;
    }

    const ongoingBatches = [];
    if (this.batches.length === 1) {
      this.enrollIntoBatch(this.batches[0]);
    } else {
      this.batches.forEach(batch => {
        if (batch.status === 1) {
          ongoingBatches.push(batch);
        }
      });
      this.router.navigate([RouterLinks.COURSE_BATCHES], {
        state: {
          ongoingBatches,
          upcommingBatches: [],
          course: this.courseContentData,
          objRollup: this.objRollup,
          telemetryObject: this.telemetryObject,
          corRelationList: this.corRelationList
        }
      });
    }
  }

  async enrollIntoBatch(item: Batch) {
    this.loader = await this.commonUtilService.getLoader();
    if (this.guestUser) {
      this.promptToLogin(item);
    } else {
      await this.loader.present();
      const enrollCourseRequest = this.localCourseService.prepareEnrollCourseRequest(this.userId, item);
      this.telemetryGeneratorService.generateInteractTelemetry(InteractType.TOUCH,
        InteractSubtype.ENROLL_CLICKED,
        Environment.HOME,
        PageId.CHAPTER_DETAILS, undefined,
        this.localCourseService.prepareRequestValue(enrollCourseRequest));
      const enrollCourse: EnrollCourse = {
        userId: this.userId,
        batch: item,
        pageId: PageId.COURSE_BATCHES,
        courseId: this.courseContent.identifier,
        channel: this.courseContent.contentData.channel,
        userConsent: this.courseContent.contentData.userConsent
      };

      this.localCourseService.enrollIntoBatch(enrollCourse, this, this.courseContent).toPromise()
        .then(async (data: boolean) => {
          this.courseContent.batchId = item.id;
          this.commonUtilService.showToast(this.categoryKeyTranslator.transform('FRMELEMNTS_MSG_COURSE_ENROLLED', this.courseContent));
          this.events.publish(EventTopics.ENROL_COURSE_SUCCESS, {
            batchId: item.id,
            courseId: item.courseId
          });
          this.isAlreadyEnrolled = true;
        }, async (error) => {
          await this.loader.dismiss();
        });
    }
  }

  async promptToLogin(batchdetail) {
    const confirm = await this.popoverCtrl.create({
      component: SbPopoverComponent,
      componentProps: {
        sbPopoverMainTitle:
          this.categoryKeyTranslator.transform('FRMELEMNTS_MSG_YOU_MUST_JOIN_TO_ACCESS_TRAINING_DETAIL', this.courseContent),
        metaInfo: this.categoryKeyTranslator.transform('FRMELEMNTS_MSG_TRAININGS_ONLY_REGISTERED_USERS', this.courseContent),
        sbPopoverHeading: this.commonUtilService.translateMessage('OVERLAY_SIGN_IN'),
        isNotShowCloseIcon: true,
        actionsButtons: [
          {
            btntext: this.commonUtilService.translateMessage('OVERLAY_SIGN_IN'),
            btnClass: 'popover-color label-uppercase label-bold-font'
          },
        ]
      },
      cssClass: 'sb-popover info',
    });
    await confirm.present();
    const { data } = await confirm.onDidDismiss();
    if (data && data.canDelete) {
      this.preferences.putString(PreferenceKey.BATCH_DETAIL_KEY, JSON.stringify(batchdetail)).toPromise();
      this.preferences.putString(PreferenceKey.COURSE_DATA_KEY, JSON.stringify(this.courseContentData)).toPromise();
      this.appGlobalService.resetSavedQuizContent();
      this.router.navigate([RouterLinks.SIGN_IN], {state: {navigateToCourse: true}});
    }
  }

  /**
   * get first child of unit
   */
  loadFirstChildren(data) {
    if (data && (data.children === undefined)) {
      return data;
    } else {
      for (const child of data.children) {
        return this.loadFirstChildren(child);
      }
    }
  }

  async joinTraining() {
    if (!this.batches || !this.batches.length) {
      this.commonUtilService.showToast('NO_BATCHES_AVAILABLE');
      return;
    } else if (
      this.batches.length === 1 &&
      this.batches[0].enrollmentEndDate &&
      (new Date().setHours(0, 0, 0, 0) > new Date(this.batches[0].enrollmentEndDate).setHours(0, 0, 0, 0))
    ) {
      this.commonUtilService.showToast(
        'ENROLLMENT_ENDED_ON',
        null,
        null,
        null,
        null,
        this.datePipe.transform(this.batches[0].enrollmentEndDate)
      );
      return;
    }

    const confirm = await this.popoverCtrl.create({
      component: SbPopoverComponent,
      componentProps: {
        sbPopoverMainTitle: this.categoryKeyTranslator.transform('FRMELEMNTS_MSG_YOU_MUST_JOIN_AN_ACTIVE_BATCH', this.courseContent),
        metaInfo: this.commonUtilService.translateMessage('REGISTER_TO_COMPLETE_ACCESS'),
        sbPopoverHeading: this.categoryKeyTranslator.transform('FRMELEMNTS_LBL_JOIN_TRAINING', this.courseContent) + '?',
        isNotShowCloseIcon: true,
        actionsButtons: [
          {
            btntext: this.categoryKeyTranslator.transform('FRMELEMNTS_LBL_JOIN_TRAINING', this.courseContent),
            btnClass: 'popover-color label-uppercase label-bold-font'
          },
        ],
        // handler : this.handleEnrollCoursePopup.bind(this)
      },
      cssClass: 'sb-popover info',
    });
    await confirm.present();
    confirm.onDidDismiss().then(({ data }) => {
      if (data && data.canDelete) {
        this.navigateToBatchListPage();
      }
    });
  }

  /**
   * Redirect to child content details page
   */
  navigateToChildrenDetailsPage(content: Content, depth): void {
    const values = {
      contentClicked: content.identifier
    };
    this.telemetryGeneratorService.generateInteractTelemetry(
      InteractType.TOUCH,
      InteractSubtype.CONTENT_CLICKED,
      Environment.HOME,
      PageId.CHAPTER_DETAILS, this.telemetryObject,
      values,
      this.objRollup,
      this.corRelationList
    );

    this.router.navigate([RouterLinks.CONTENT_DETAILS], this.generateContentNavExtras(content, depth));
  }

  private generateContentNavExtras(content: Content, depth) {
    const contentState: ContentState = {
      batchId: this.courseContent.batchId ? this.courseContent.batchId : '',
      courseId: this.courseContentData.identifier
    };

    const params: NavigationExtras = {
      state: {
        content,
        depth,
        contentState,
        isChildContent: true,
        corRelation: this.corRelationList,
        isCourse: true,
        course: this.updatedCourseCardData
      }
    };
    return params;
  }

  getContentsSize(data?) {
    if (data) {
      data.forEach((value) => {
        if (value.contentData.size) {
          this.downloadSize += Number(value.contentData.size);
        }
        if (value.children) {
          this.getContentsSize(value.children);
        }
        if (value.isAvailableLocally === false) {
          this.downloadIdentifiers.add(value.contentData.identifier);
          this.rollUpMap[value.contentData.identifier] = ContentUtil.generateRollUp(value.hierarchyInfo, undefined);
        }
      });
    }
  }

  async showDownloadConfirmationAlert() {
    console.log('this.downloadIdentifiers', this.downloadIdentifiers);
    if (this.commonUtilService.networkInfo.isNetworkAvailable) {
      let contentTypeCount;
      if (this.downloadIdentifiers.size) {
        contentTypeCount = this.downloadIdentifiers.size;
      } else {
        contentTypeCount = '';
      }
      if (!this.isBatchNotStarted) {
        this.downloadProgress = 0;
      } else {
        this.commonUtilService.showToast(this.commonUtilService.translateMessage('COURSE_WILL_BE_AVAILABLE',
          this.datePipe.transform(this.courseStartDate, 'mediumDate')));
      }
      this.telemetryGeneratorService.generateInteractTelemetry(
        InteractType.TOUCH,
        InteractSubtype.DOWNLOAD_CLICKED,
        Environment.HOME,
        PageId.CHAPTER_DETAILS
      );
      const popover = await this.popoverCtrl.create({
        component: ConfirmAlertComponent,
        componentProps: {
          sbPopoverHeading: this.commonUtilService.translateMessage('DOWNLOAD'),
          sbPopoverMainTitle: this.chapter.name,
          isNotShowCloseIcon: true,
          actionsButtons: [
            {
              btntext: this.commonUtilService.translateMessage('DOWNLOAD'),
              btnClass: 'popover-color'
            },
          ],
          icon: null,
          metaInfo: this.commonUtilService.translateMessage('ITEMS', contentTypeCount)
            + ' (' + this.fileSizePipe.transform(this.downloadSize, 2) + ')',
        },
        cssClass: 'sb-popover info',
      });
      await popover.present();
      const response = await popover.onDidDismiss();
      if (response && response.data) {
        this.isDownloadStarted = true;
        this.showCollapsedPopup = false;
        this.events.publish('header:decreasezIndex');
        this.importContent(this.downloadIdentifiers, true, true);
        this.showDownload = true;
      } else {
      }
    } else {
      this.commonUtilService.showToast('ERROR_NO_INTERNET_MESSAGE');
    }
  }

  getImportContentRequestBody(identifiers, isChild: boolean): Array<ContentImport> {
    const requestParams = [];
    const folderPath = this.platform.is('ios') ? cordova.file.documentsDirectory : cordova.file.externalDataDirectory;
    identifiers.forEach((value) => {
      requestParams.push({
        isChildContent: isChild,
        destinationFolder: folderPath,
        contentId: value,
        // correlationData: this.corRelationList !== undefined ? this.corRelationList : [],
        correlationData: [],
        rollUp: this.rollUpMap[value]
      });
    });

    return requestParams;
  }

  importContent(identifiers, isChild: boolean, isDownloadAllClicked?) {
    const option: ContentImportRequest = {
      contentImportArray: this.getImportContentRequestBody(identifiers, isChild),
      contentStatusArray: ['Live'],
      fields: ['appIcon', 'name', 'subject', 'size', 'gradeLevel']
    };
    console.log('ContentImportRequest', option);
    this.contentService.importContent(option).toPromise()
      .then((data: ContentImportResponse[]) => {
        this.zone.run(() => {
          if (data && data[0].status === ContentImportStatus.NOT_FOUND) {
            this.headerService.showHeaderWithBackButton();
          }
          if (data && data.length && this.isDownloadStarted) {
            data.forEach((value) => {
              if (value.status === ContentImportStatus.ENQUEUED_FOR_DOWNLOAD) {
                this.queuedIdentifiers.push(value.identifier);
              } else if (value.status === ContentImportStatus.NOT_FOUND) {
                this.faultyIdentifiers.push(value.identifier);
              }
            });

            if (this.queuedIdentifiers.length === 0) {
            }
            if (this.faultyIdentifiers.length > 0) {
              const stackTrace: any = {};
              stackTrace.parentIdentifier = this.courseContentData.identifier;
              stackTrace.faultyIdentifiers = this.faultyIdentifiers;
              this.commonUtilService.showToast('UNABLE_TO_FETCH_CONTENT');
            }
          }
        });
      })
      .catch((error) => {
        this.zone.run(() => {
          if (error && error.error === 'NETWORK_ERROR') {
            this.commonUtilService.showToast('NEED_INTERNET_TO_CHANGE');
          } else {
            this.commonUtilService.showToast('UNABLE_TO_FETCH_CONTENT');
          }
        });
      });
  }

  /**
   * Subscribe Sunbird-SDK event to get content download progress
   */
  subscribeSdkEvent() {
    this.eventSubscription = this.eventsBusService.events()
      .subscribe((event: EventsBusEvent) => {
        this.zone.run(() => {
          // Show download percentage
          if (event.type === DownloadEventType.PROGRESS) {
            const downloadEvent = event as DownloadProgress;
            if (downloadEvent.payload.identifier === this.identifier) {
              this.downloadProgress = downloadEvent.payload.progress === -1 ? 0 : downloadEvent.payload.progress;

              if (this.downloadProgress === 100) {
                this.getBatchDetails();
                this.headerService.showHeaderWithBackButton();
              }
            }
          }

          // Get child content
          if (event.payload && event.type === ContentEventType.IMPORT_COMPLETED) {
            this.headerService.showHeaderWithBackButton();
            const contentImportCompleted = event as ContentImportCompleted;
            if (this.queuedIdentifiers.length && this.isDownloadStarted) {
              if (this.queuedIdentifiers.includes(contentImportCompleted.payload.contentId)) {
                this.currentCount++;
              }

              if (this.queuedIdentifiers.length === this.currentCount) {
                this.isDownloadStarted = false;
                this.currentCount = 0;
                this.showDownload = false;
                this.downloadIdentifiers = new Set();
                this.queuedIdentifiers.length = 0;
              }
            } else {
              this.courseContentData.isAvailableLocally = true;
            }
          }

          if (event.payload && event.type === ContentEventType.SERVER_CONTENT_DATA) {
            if (event.payload.size) {
              this.courseContent.contentData.size = event.payload.size;
            }
          }

          if (event.type === ContentEventType.IMPORT_PROGRESS) {
          }

          // For content update available
          const hierarchyInfo = this.courseContent.hierarchyInfo ? this.courseContent.hierarchyInfo : null;
          const contentUpdateEvent = event as ContentUpdate;
          if (contentUpdateEvent.payload && contentUpdateEvent.payload.contentId === this.courseContentData.identifier &&
            contentUpdateEvent.type === ContentEventType.UPDATE
            && hierarchyInfo === null) {
            this.zone.run(() => {
              this.headerService.hideHeader();
            });
          }

        });
      }) as any;
  }

  private getSubContentIds(content) {
    if (content && content.children) {
      content.children.forEach((ele) => {
        this.getSubContentIds(ele);
      });
    } else {
      if (content.mimeType !== MimeType.COLLECTION) {
        this.subContentIds.push(content.identifier);
      }
    }
    return this.subContentIds;
  }

  private getNextContent(courseHeirarchy, contentStateList: ContentState[]) {
    const result = contentStateList.find(({ contentId }) => contentId === courseHeirarchy.identifier);
    if (!this.isFirstContent && courseHeirarchy.mimeType !== MimeType.COLLECTION) {
      this.nextContent = courseHeirarchy;
      this.isFirstContent = true;
    }
    if ((result && (result.status === 0 || result.status === 1))
      || (!result && courseHeirarchy.mimeType !== MimeType.COLLECTION)) {
      this.nextContent = courseHeirarchy;
      this.isNextContentFound = true;
      this.isFirstContent = true;
    } else if (!this.isNextContentFound && courseHeirarchy && courseHeirarchy.children) {
      courseHeirarchy.children.forEach((ele) => {
        if (!this.isNextContentFound) {
          this.getNextContent(ele, contentStateList);
        }
      });
    }
    return this.nextContent;
  }
  generateImpressionEvent(objectId, objectType, objectVersion) {
    this.telemetryGeneratorService.generateImpressionTelemetry(ImpressionType.DETAIL,
      '', PageId.CHAPTER_DETAILS,
      Environment.HOME,
      objectId,
      objectType,
      objectVersion,
      this.objRollup,
      this.corRelationList
    );
  }

  onConsentPopoverShow() {
    if (this.loader) {
      this.loader.dismiss();
      this.loader = undefined;
    }
  }

  onConsentPopoverDismiss() { }

  private async showProfileNameConfirmationPopup() {
    const popUp = await this.popoverCtrl.create({
      component: ProfileNameConfirmationPopoverComponent,
      componentProps: {
      },
      cssClass: 'sb-popover sb-profile-name-confirmation-popover',
    });
    await popUp.present();
    const { data } = await popUp.onDidDismiss();
    if (data !== undefined) {
      if (data.buttonClicked) {
        this.startContent();
      }
    } else {
      this.telemetryGeneratorService.generateInteractTelemetry(
        InteractType.TOUCH,
        InteractSubtype.CLOSE_CLICKED,
        PageId.PROFILE_NAME_CONFIRMATION_POPUP,
        Environment.HOME
      );
    }
  }

}
<ion-content>
  <div class="cd-header-container">
    <div class="cd-title-container">
      <div class="cd-label">{{ courseContentData?.name }}</div>
      <div class="cd-header-title">{{ chapter.name}}</div>
      <div class="cd-of-menu">
        <img class="cd-menu-icon" (click)="showOverflowMenu($event)" src="assets/imgs/menu.svg" alt="menu-icon">
      </div>
    </div>
    <div class="new-div">
      <div *ngIf="!isAlreadyEnrolled">
        <div class="enrollment-info"
          *ngIf="batches && courseContentData?.contentData?.createdBy !== userId && (batches.length > 1 || (batches.length === 1 && !(todayDate > batches[0].enrollmentEndDate)))">
          <button class="enrolled-course-card-button label-uppercase label-bold-font" (click)="navigateToBatchListPage();">
            {{'FRMELEMNTS_LBL_JOIN_TRAINING' | categoryKeyTranslate : courseContentData }}
          </button>
        </div>
      </div>
      <div class="sb-course-end-container"
        *ngIf="batches && !isAlreadyEnrolled && !batches.length">
        <p class="main-heading">
          <strong>{{'NO_BATCHES_AVAILABLE'  | translate}}</strong>
        </p>
        <p class="sub-heading">{{'FRMELEMNTS_MSG_THERE_ARE_NO_BATCHES_AVAILABLE_FOR_THIS_TRAINING' | categoryKeyTranslate : courseContentData }}</p>
      </div>
      <div class="sb-course-end-container"
         *ngIf="batches && !isAlreadyEnrolled && batches.length === 1 && (todayDate > batches[0].enrollmentEndDate)">
        <p class="main-heading">
          <strong>{{'ENROLLMENT_ENDED_ON'  | translate : {'%s': enrollmentEndDate ? (enrollmentEndDate | date: 'dd/MM/yyyy') : ''} }}</strong>
        </p>
        <p class="sub-heading">{{'FRMELEMNTS_MSG_THERE_ARE_NO_BATCHES_AVAILABLE_FOR_THIS_TRAINING' | categoryKeyTranslate : courseContentData }}</p>
      </div>
    </div>
    <ion-card-content class="ion-no-padding">
      <div *ngIf="isAlreadyEnrolled">
        <div *ngIf="courseContent?.batchId">
          <div class="sb-course-progress-container" *ngIf="!isChapterCompleted && !batchExp">
            <p class="progress-label"><strong>{{'YOUR_PROGRESS_LABEL' | translate}}</strong></p>
            <p>{{viewedContents.length}}/{{childContents.length}} {{'COMPLETED' | translate}}</p>
            <app-pb-horizontal [progress]="chapterProgress"
              class="course-progress" [ngClass]="{'blurbackground': batchExp}"
              isOnBoardCard="flase" isCourseProgress="true"></app-pb-horizontal>
          </div>
          <div *ngIf="isChapterCompleted"
            class="sb-course-complete-container ion-padding">
            <div class="img-container">
              <img class="success-badge" src='assets/imgs/Badge green.svg'
                alt="completed">
            </div>
            <div class="text-container">
              <p><strong>{{'CHAPTER_TRAINING_COMPLETED_SUCCESSFULLY' | categoryKeyTranslate : courseContentData }}</strong></p>
            </div>
          </div>
        </div>
      </div>
    </ion-card-content>
    <div class="cd-flex-container" *ngIf="isAlreadyEnrolled">
      <div style="width: 100%;">
        <button class="cd-chapter-card-button" *ngIf="!isChapterCompleted && !isChapterStarted" (click)="startLearning();">
          <img class="play-icon" src="assets/imgs/Play@1.5x.svg" alt="play">
          {{'START_LEARNING' | translate}}
        </button>
        <button class="cd-chapter-card-button" *ngIf="isChapterCompleted" (click)="continueLearning();">
          <img class="play-icon" src="assets/imgs/Play@1.5x.svg" alt="play">
          {{'START_AGAIN' | translate}}
        </button>
        <button class="cd-chapter-card-button" *ngIf="!isChapterCompleted && isChapterStarted" (click)="continueLearning();">
          <img class="play-icon" src="assets/imgs/Play@1.5x.svg" alt="play">
          {{'CONTINUE_LEARNING' | translate}}
        </button>
      </div>
    </div>
  </div>
  <div class="cd-container">
    <div class="cd-flex-container">
      <div attr.aria-label="{{'CONTENTS' | translate}}, heading" class="cd-container-title">{{"CONTENTS" | translate}}</div>
      <div class="cd-container-progress" *ngIf="isAlreadyEnrolled">{{viewedContents.length}}/{{childContents.length}} {{"COMPLETED" | translate}}</div>
    </div>
    <div>
      <sb-toc-curriculum *ngIf="chapter" [type]="cardType"  [tocData]="chapter" [contentStatus]="contentStatusData?.contentList || []" 
      [maxAttempts]="maxAssessmentLimit" [disabled]="'disabled-toc-card'" [scoreLabel]="'BEST_SCORE' | translate"
      (tocCardClick)="openContentDetails($event)"></sb-toc-curriculum>
    </div>
    
  </div>
</ion-content>
<ng-container *ngIf="trackDownloads$ | async; let downloads">
  <app-sb-download-popup *ngIf="downloads.queued.length"
    (cancelDownloadEmit)="cancelDownload($event)" 
    [currentCount]="downloads.completed.length"
    [queuedIdentifiers]="(downloadIdentifiers.size)" 
    [downloadSize]="downloadSize"
    [collectionName]="chapter?.name" 
    [showDownload]="true"
    [showPopover]="showCollapsedPopup">
  </app-sb-download-popup>
</ng-container>

./chapter-details.page.scss

@import "src/assets/styles/_variables.scss";
.cd-header-container {
    padding: 16px;
    margin: 8px;
    border-radius: 2px;
    box-shadow: 0 3px 5px 4px rgba(0,0,0,0.05);
    position: relative;
    .cd-title-container{
        margin-bottom: 8px;
    }
    .cd-header-title {
        color: map-get($colors, gray_3A);
        font-size: 1rem;
        font-weight: bold;
    }

    .cd-of-menu{
        position: absolute;
        top: 1rem;
        right: 1rem;
        .cd-menu-icon{
            height: 1rem;
            width: 1.25rem;
            margin: 8px 8px 8px 16px;
        }
    }

    .cd-progress-container {
        background-color: map-get($colors, pale_blue);
        padding: 8px;
        margin-top: 8px;
        width: auto;

        .progress-label {
            font-size: 0.625rem !important;
            color: map-get($colors, primary_black);
            font-weight: bold;
        }
    }

    .cd-unblock-text {
        color: map-get($colors, gray_3A);
        font-size: 0.625rem;
        padding: 8px 0px;
    }

    .cd-chapter-card-button {
        background-color: $green;
        border: 1px solid $green;
        color: white;
        display: inline-block;
        font-size: 1rem;
        cursor: pointer;
        outline: none;
        width: 100%;
        padding: 16px 0px 16px 0px;
        box-shadow: 0 4px 7px 0 rgba(0,0,0,0.25);

        .play-icon {
            margin-right: px;
        }
    }
}

.cd-container {
    padding: 0px 16px 16px 16px;

    .cd-container-title {
        color: map-get($colors, gray_3A);
        font-size: 1.125rem;
        font-weight: bold;
    }

    .cd-container-progress {
        color: map-get($colors, granite_gray);
        font-size: 0.75rem;
    }

    .cd-container-sub-title {
        color: map-get($colors, gray_3A);
        font-size: $font-size-base;
    }
}

.cd-flex-container {
    display: flex;
    align-items: center;
    justify-content: space-between;
    margin-top: 8px;
    .cd-align-s {
        align-items: start;
        margin-bottom: 8px;
    }
}

.cd-label {
    color: map-get($colors, granite_gray);
    font-size: 0.625rem;
}

../../enrolled-course-details-page/enrolled-course-details-page.scss

@import 'src/assets/styles/_custom-mixins';
@import "src/assets/styles/variables";
@import "src/assets/styles/_variables.scss";


:host {
    ion-card{
      margin: 0 0 10px 0 !important;
      --ion-item-background: white;
    }
    .z-index-0 {
      z-index: 0 !important;
    }
    .retry-btn{
      border-color: $orange;
      color: $orange !important;
      padding: 8px 40px;
      font-size: $font-size-base;
      border-radius: 0 !important;
    }
    .sb-url {
      color: map-get($colors, bright_blue_a0);
      font-size: $font-size-base;
      text-decoration: underline;
    }
    .fontBold {
        font-weight: bold;
    }
    .ion-ios-radio-button-on:before {
        color: darkgrey;
    }
    .ion-ios-checkmark-circle:before {
        color: map-get($colors, office_green);
        margin-right: 4px;
        margin-left: 0px;
    }
    .structure-text{
        padding: 7px 0px 0px 4px;
    }
    .certified-section-color{
      color: map-get($colors, bright_blue_9f);
    }
    .certified-training-section{
        background-color: map-get($colors, pale_blue);
        padding: 8px;
    }
    .font-10{
        font-size: 0.625rem;
    }
    .font-12 {
        font-size: 0.75rem!important;
    }
    .font-13{
      font-size: 0.813rem !important;
    }
    .font-16{
        font-size: 1rem!important;
    }
    .font-14{
        font-size: $font-size-base !important;
    }
    .font-18{
        font-size: 1.125rem;
    }
    .padding-left-8{
        @include padding(0px 0px 0px 8px);
    }
    .child-item{
        --padding-start: 0 !important;
        @include padding(null, null, null, 0 !important)
    }
    .text-transform-none{
        text-transform: none !important;
    }
    .card-content-bottom-0{
        padding-bottom: 0px !important;
    }
    .card-content-top-0{
      padding-top: 0px !important;
  }
    .subtitle-color{
        color: var(--app-gray)
    }
    .enrolled-course-container {
      &.sticky {
        position: sticky;
        top: 0;
        left: 0;
        z-index: 10;
      }
    }
    .traning-derived-label-background {
      background-color: map-get($colors, white_f2);
      padding: 8px 0px 8px 15px;
    }

    ion-card ion-card-content p {
        font-size: $font-size-base;
    }
    .enrolled-course-card{
        background-color: map-get($colors, pale_green);
        margin-bottom: 0 !important;
    }
    .play-icon{
       display: inline;
       width: 1.313rem;
       vertical-align: middle;
       margin-right: 6.43px;
    }
    ion-content{
        --ion-background-color: white;
    }
    .hr-border-bottom {
        box-sizing: border-box;
        height: 1px;
        width: 100%;
        border: 0.5px solid map-get($colors, medium_gray);
        margin-top: 9px;
    }
    .hr-border-bottom-share {
      box-sizing: border-box;
      height: 1px;
      border: 1px solid #E9DED3;
      margin-top: 8px;
  }
    hr {
        border:0.5px solid $gray-300;
    }
    .blurbackground {
        .progress-inner-primary {
            background-color: gray !important;
        }
        .progress-inner-secondary {
            background-color: gray !important;
        }
    }
    .loading-backdrop {
        background-color: map-get($colors, white);
        height: 100%;
        z-index: 1000;
        opacity: 1 !important;
        top: 0;
        bottom: 0;
        .backdrop-container {
            width: 100%;
            padding: 16px;
            text-align: center;
            position: absolute;
            top: 50%;
            margin-top: -50px;
        }
        .backdrop-footer {
            position: absolute;
            width: 100%;
            bottom: 0;
            padding-bottom: 16px;
        }
    }
    .scroll-content {
        margin-bottom: 0 !important;
    }

    .toolbar-p-16{
        padding: ($base-block-space * 2) ($base-block-space * 2) ($base-block-space / 2) ($base-block-space * 2) !important;
    }

    .size_txt {
        font-size: 0.75rem;
    }
    .padding-right-3 {
        padding-right: 3px;
    }


    .list-mb-48{
        margin-bottom: ($base-block-space * 6) !important;
    }

    .sc-ion-buttons-md-s ion-button {
        height: 1.25rem;
        font-size: 0.625rem;
    }
    .label-margin-bottom{
      margin-bottom: 3px !important;
    }
    .sb-accordian-header{
      padding: 12px 14px;
      .dd-icon {
        position: relative;
        bottom: 0.188rem;
      }
      .sb-checkmark-icon{
        color: map-get($colors, office_green);
      }
    }
    .custom-card-header{
        padding-bottom: 0px !important;
    }
    .sb-accordian-arrow{
        float: right;
        font-size: $font-size-base;
        padding-top: 10px;
    }
    .enrollment-info{
      ion-icon{
        position: relative;
        top: 0.125rem;
      }
    }
    .enrolled-course-card-button{
        background-color: $green;
        border: none;
        color: white;
        text-align: center;
        text-decoration: none;
        display: inline-block;
        font-size: 1rem;
        cursor: pointer;
        outline: none;
        margin: 4px 0 8px;
        width: 100%;
        padding: 16px 0 16px 0;
    }
    .disable-css {
      opacity : 0.5;
     }
    .segment-button-checked{
      ion-label.sc-ion-label-md-h.sc-ion-label-md-s.md.hydrated{
        font-weight: bold;
      }
    }
    .border-right{
      border-right: 1.5px solid map-get($colors, white_e6);
      margin: 0px 0px 10px 11px;
    }
    .hr-div {
      padding: 0px 10px 0px 10px;
    }
    hr.hr-height {
      height: 2rem;
    }
    .segment-button-indicator{
      width: 84% !important;
      margin-right: 8% !important;
    }
    .info-icon-pr-8 {
      padding-right: 8px !important;
    }
  }
  .sb-topics-containe{
    padding-right: 5%;
  }
  .sb-hidden-content{
    background-color: map-get($colors, pale_blue) !important;
    // height: 34.15% !important;
    width: 100% !important;
    border-radius: 0 0 2px 2px !important;
    padding-top: 2% !important;
    padding-bottom: 5% !important;
  }
  .sb-accordian-card{
    // height: 6.1% !important;
    width: 100% !important;
    border-radius: 2px !important;
    background-color: map-get($colors, white) !important;
    box-shadow: 0px 5px 5px 3px map-get($colors, light_gray_dd);
  }
  .sb-topic-text {
      // padding-top: 5px;
      padding: 0;
      ion-icon{
        vertical-align: middle;
      }
    // height: 2.44%;
    // width: 80%;
    color: $blue;
    font-size: $font-size-base;
    // line-height: 20px;
  }

  .certificate-container {
    display: inline-flex;
    flex-wrap: nowrap;
    align-items:center;
    background-color: map-get($colors, pale_blue_f6);
    padding: 4px;
    .certificate-icon {
      padding-right: 5px;
    }
  }
  .sc-ion-segment-md-h{
    --color-checked : #{$blue};
    font-weight: bold;
    --color: #{map-get($colors, cornflower_blue)};
  }
  .wrapper {
    grid-template-columns: 100px 100px 100px auto;
    margin-bottom: -8px;
    border-radius: 5px;
    & > div {
      border-radius: 5px;
      padding: 1em 0;
    }
  }

  .pull-right{
      float: right;
  }
  .menu-icon{
    display: block;
    width: 100%;
    height: 1rem;
    margin-top: 18px;
    float: right;
  }
  .card-icons {
    font-size: 1.875rem;
    color: $blue;
  }
  .card-icons-downloadStarted {
    font-size: 1.875rem;
    color: map-get($colors, text_gray);
  }
  .card-icons-red{
    font-size: 1.875rem !important;
    color:map-get($colors , danger);
  }
  .sb-course-complete-container {
    white-space: nowrap;
    width: 100%;
    background-color: map-get($colors, light_grayish_green);
    @include padding(8px);
    .img-container{
      width: 20%;
      vertical-align: middle;
      display: inline-block;
    }
    .text-container{
      color: map-get($colors, office_green);
      font-family: "Noto Sans", sans-serif;
      font-size: $font-size-base !important;
      font-weight: bold;
      width: 80%;
      vertical-align: middle;
      display: inline-block;
      padding-left: 10px;
      @include padding(0px 0px 0px 11px);
      white-space: normal;
    }
  }
  .card-text{
    height: 1.125rem;
    width: auto;
    color: map-get($colors, granite_gray);
    font-family: "Noto Sans", sans-serif;
    font-size: 0.813rem !important;
    line-height: 1.25rem;
    text-align: center;
  }
  .card-button{
      background-color: transparent !important;
      outline: none;
      padding: 0px 16px;
  }
  .padding-bottom-6{
    padding-bottom: 6px;
  }
  .padding-bottom-8{
    padding-bottom: 8px;
  }
  .margin-bottom-2{
    margin-bottom: 2px;
  }
  .margin-bottom-16{
    margin-bottom: 16px;
  }

  .sb-course-progress-container{
      background-color: map-get($colors, pale_blue);
      padding: 8px;
      height: 4.38rem;
      width: auto;
  }

  .last-read-content{
    font-size: 1rem!important;
    font-weight:bold;
    @include rtl() {
      padding-bottom: 4px;
     }
     @include ltr() {
     padding-left: 4px;
     }
  }

  .sb-batch-end-container{
    background-color: map-get($colors, pale_red);
    padding: 8px 16px;
    height: auto;
    width: auto;
    .batch-end-main-heading{
      font-size: 0.75rem !important;
      color: $orange;
      font-weight: bold;
    }
    .batch-end-sub-heading{
      color: map-get($colors, primary_black);
      font-family: "Noto Sans", sans-serif;
      font-size: 0.75rem !important;
    }
  }
  .enrollment-end-date{
    font-size: 0.75rem;
    color: map-get($colors, primary_black);
    font-family: "Noto Sans", sans-serif;
    ion-icon{
      position: relative;
      top: 0.125rem;
    }
  }
  .padding-left-5em{
    padding-left: 5em !important;
  }
  .sb-course-end-container{
    background-color: map-get($colors, pale_red);
    padding: 10px;
    height: 4.38rem;
    width: auto;
    .main-heading{
      font-size: $font-size-base !important;
      color: $orange;
    }
    .sub-heading{
      color: map-get($colors, primary_black);
      font-family: "Noto Sans", sans-serif;
      font-size: 0.75rem !important;
    }
}

.information-icon {
  color: map-get($colors, dark);
  font-size: $font-size-base;
}

.progress-label {
  font-size: 0.625rem!important;
  color: map-get($colors, primary_black);
}

ion-segment-button{
  --ripple-color: transparent;
  --background-hover: transparent;
  --background-activated: transparent;
}

.flex-5 {
  flex: 5;
}

#card-mb-0{
  margin-bottom: 0 !important;
}

.card-box-shadow {
  box-shadow: 0px 5px 5px 3px map-get($colors, light_gray_dd);
  margin: 0 0 1rem 0 !important
}

.animation-card{
  padding:8px 16px;
  .animation-card-content{
    display:flex;
    justify-content:space-between;
    align-items: center;
    .skeleton-search-card{
      width:45%;
    }
  }
}

.animation-bottom-card {
  .skeleton-animation-item-1{
    width:40%; margin-bottom:20px
  }
  .skeleton-animation-item-2, .skeleton-animation-item-4, .skeleton-animation-item-6{
    width:48%; margin-bottom:8px
  }
  .skeleton-animation-item-3{
   width:25%;margin-bottom:8px
  }
  .skeleton-animation-item-5{
   width:25%;margin-bottom:20px
  }
  .skeleton-animation-item-7{
    width:25%;margin-bottom:16px;
  }
  .skeleton-animation-item-8, .skeleton-animation-item-9{
    width:100%;margin-bottom:8px;
  }
  .skeleton-animation-item-10 {
    width:48%;
  }
  .skeleton-animation-item-11 {
    width:25%;
  }
}

.bg-grey {
  background: map-get($colors, white_f0) !important;
}

.card-button[disabled]{
  opacity: 0.7 !important;
  pointer-events: none;
  cursor: default;
}

.course-modified {
  padding: 16px;
  margin-bottom: 16px;
  box-sizing: border-box;
  border: 1px solid map-get($colors, dark_orange);
  border-radius: 2px;
  background-color: map-get($colors, ligth_yellow);
  color: $gray-800;
  font-size: 0.75rem;

  .alert {
    width: 1.5rem;
    height: 1.5rem;
    display: inline-block;
    margin-right: 8px;
  }
}

.edit-setting{
  float: right;
  padding: 12px;
  color: $blue;
  font-weight: bold;
}
.sb-content-info{
  padding: 4px 14px;
}

.save-changes{
  color: map-get($colors, white);
  background-color: $blue;
  padding: 12px 45px;
  outline: none;
  border: none;
  text-transform: uppercase;
  font-weight: bold;
  font-size: 0.75rem;
}
.ds-m0 {
  color: map-get($colors, primary_black);
  font-family: "Noto Sans", sans-serif;
  font-size: 0.75rem;
  letter-spacing: 0;
  line-height: 1.25rem;
}

.ds-item{
  --padding-start: 0px;
}
ion-radio{
  margin-right: 10px !important;
}
.font-14-new{
  font-size: 1rem !important;
  margin-top: 8px;
}
.data-share-info{
  padding: 5px;
  margin-left: -6px;
  font-size: 0.938rem;
  text-transform: uppercase;
}

.data-share-header{
  margin-left: 2px;
}
.timer-info{
  text-align: center;
  width: 100%;
  font-size: 1.2rem;
  font-weight: 600;
  color: indianred;
  padding: 0 0 10px;
}
.data-list{
  background: unset !important;
}
.list-info{
margin-top: -13px; opacity: 1;
}
.consent-arrow{
  float: right
}

.consent-arrow {
  ion-icon{
    font-size: 1.5rem;
    color: #024f9d;
  }
}
.sb--card__meta1 {
  margin-top:0px;
  width: 100%;
  .data_1, .data_2 {
    color: var(--cc-sbcard-data1-2-text);
    border-radius: 0.5rem;
    font-size: $font-size-base;
    width:auto;flex: initial;
    white-space: nowrap;
  }
  .data_2 {
    
    &:last-child {
      margin-right:0px;
    }
  }

}


.course-details{
  ion-card-title{
    strong{
      color: var(--app-medium-gray);
    }
  }
  p{
    color: var(--app-gray);
  }
}
Legend
Html element
Component
Html element with directive

results matching ""

    No results matching ""