UNPKG

ngx-joyride

Version:

[![npm version](https://badge.fury.io/js/ngx-joyride.svg)](https://badge.fury.io/js/ngx-joyride) [![Build Status](https://travis-ci.org/tnicola/ngx-joyride.svg?branch=master)](https://travis-ci.org/tnicola/ngx-joyride) [![codecov](https://codecov.io/gh/

225 lines 35.7 kB
import { Injectable } from '@angular/core'; import { JoyrideBackdropService } from './joyride-backdrop.service'; import { EventListenerService } from './event-listener.service'; import { JoyrideStepsContainerService, StepActionType } from './joyride-steps-container.service'; import { DocumentService } from './document.service'; import { StepDrawerService } from './step-drawer.service'; import { DomRefService } from './dom.service'; import { NO_POSITION } from '../directives/joyride.directive'; import { JoyrideOptionsService } from './joyride-options.service'; import { Router } from '@angular/router'; import { ReplaySubject } from 'rxjs'; import { JoyrideStepDoesNotExist, JoyrideStepOutOfRange } from '../models/joyride-error.class'; import { LoggerService } from './logger.service'; const SCROLLBAR_SIZE = 20; export const DISTANCE_FROM_TARGET = 15; export const ARROW_SIZE = 10; export class JoyrideStepService { constructor(backDropService, eventListener, stepsContainerService, documentService, DOMService, stepDrawerService, optionsService, router, logger) { this.backDropService = backDropService; this.eventListener = eventListener; this.stepsContainerService = stepsContainerService; this.documentService = documentService; this.DOMService = DOMService; this.stepDrawerService = stepDrawerService; this.optionsService = optionsService; this.router = router; this.logger = logger; this.winTopPosition = 0; this.winBottomPosition = 0; this.stepsObserver = new ReplaySubject(); this.initViewportPositions(); this.subscribeToScrollEvents(); this.subscribeToResizeEvents(); } initViewportPositions() { this.winTopPosition = 0; this.winBottomPosition = this.DOMService.getNativeWindow().innerHeight - SCROLLBAR_SIZE; } subscribeToScrollEvents() { this.eventListener.startListeningScrollEvents(); this.eventListener.scrollEvent.subscribe(scroll => { this.winTopPosition = scroll.scrollY; this.winBottomPosition = this.winTopPosition + this.DOMService.getNativeWindow().innerHeight - SCROLLBAR_SIZE; if (this.currentStep) this.backDropService.redraw(this.currentStep, scroll); }); } subscribeToResizeEvents() { this.eventListener.resizeEvent.subscribe(() => { if (this.currentStep) this.backDropService.redrawTarget(this.currentStep); }); } drawStep(step) { step.position = step.position === NO_POSITION ? this.optionsService.getStepDefaultPosition() : step.position; this.stepDrawerService.draw(step); } startTour() { this.stepsObserver = new ReplaySubject(); this.stepsContainerService.init(); this.documentService.setDocumentHeight(); this.tryShowStep(StepActionType.NEXT); this.eventListener.startListeningResizeEvents(); this.subscribeToStepsUpdates(); return this.stepsObserver.asObservable(); } close() { this.removeCurrentStep(); this.notifyTourIsFinished(); this.DOMService.getNativeWindow().scrollTo(0, 0); this.eventListener.stopListeningResizeEvents(); this.backDropService.remove(); } prev() { this.removeCurrentStep(); this.currentStep.prevCliked.emit(); this.tryShowStep(StepActionType.PREV); } next() { this.removeCurrentStep(); this.currentStep.nextClicked.emit(); this.tryShowStep(StepActionType.NEXT); } navigateToStepPage(action) { let stepRoute = this.stepsContainerService.getStepRoute(action); if (stepRoute) { this.router.navigate([stepRoute]); } } subscribeToStepsUpdates() { this.stepsContainerService.stepHasBeenModified.subscribe(updatedStep => { if (this.currentStep && this.currentStep.name === updatedStep.name) { this.currentStep = updatedStep; } }); } tryShowStep(actionType) { this.navigateToStepPage(actionType); const timeout = this.optionsService.getWaitingTime(); if (timeout > 100) this.backDropService.remove(); setTimeout(() => { try { this.showStep(actionType); } catch (error) { if (error instanceof JoyrideStepDoesNotExist) { this.tryShowStep(actionType); } else if (error instanceof JoyrideStepOutOfRange) { this.logger.error('Forcing the tour closure: First or Last step not found in the DOM.'); this.close(); } else { throw new Error(error); } } }, timeout); } showStep(actionType) { this.currentStep = this.stepsContainerService.get(actionType); if (this.currentStep == null) throw new JoyrideStepDoesNotExist(''); // Scroll the element to get it visible if it's in a scrollable element this.scrollIfElementBeyondOtherElements(); this.backDropService.draw(this.currentStep); this.drawStep(this.currentStep); this.scrollIfStepAndTargetAreNotVisible(); this.notifyStepClicked(actionType); } notifyStepClicked(actionType) { let stepInfo = { number: this.stepsContainerService.getStepNumber(this.currentStep.name), name: this.currentStep.name, route: this.currentStep.route, actionType }; this.stepsObserver.next(stepInfo); } notifyTourIsFinished() { if (this.currentStep) this.currentStep.tourDone.emit(); this.stepsObserver.complete(); } removeCurrentStep() { if (this.currentStep) this.stepDrawerService.remove(this.currentStep); } scrollIfStepAndTargetAreNotVisible() { this.scrollWhenTargetOrStepAreHiddenBottom(); this.scrollWhenTargetOrStepAreHiddenTop(); } scrollWhenTargetOrStepAreHiddenBottom() { let totalTargetBottom = this.getMaxTargetAndStepBottomPosition(); if (totalTargetBottom > this.winBottomPosition) { this.DOMService.getNativeWindow().scrollBy(0, totalTargetBottom - this.winBottomPosition); } } scrollWhenTargetOrStepAreHiddenTop() { let totalTargetTop = this.getMaxTargetAndStepTopPosition(); if (totalTargetTop < this.winTopPosition) { this.DOMService.getNativeWindow().scrollBy(0, totalTargetTop - this.winTopPosition); } } getMaxTargetAndStepBottomPosition() { let targetAbsoluteTop = this.documentService.getElementAbsoluteTop(this.currentStep.targetViewContainer.element); if (this.currentStep.position === 'top') { return targetAbsoluteTop + this.currentStep.stepInstance.targetHeight; } else if (this.currentStep.position === 'bottom') { return (targetAbsoluteTop + this.currentStep.stepInstance.targetHeight + this.currentStep.stepInstance.stepHeight + ARROW_SIZE + DISTANCE_FROM_TARGET); } else if (this.currentStep.position === 'right' || this.currentStep.position === 'left') { return Math.max(targetAbsoluteTop + this.currentStep.stepInstance.targetHeight, targetAbsoluteTop + this.currentStep.stepInstance.targetHeight / 2 + this.currentStep.stepInstance.stepHeight / 2); } } getMaxTargetAndStepTopPosition() { let targetAbsoluteTop = this.documentService.getElementAbsoluteTop(this.currentStep.targetViewContainer.element); if (this.currentStep.position === 'top') { return targetAbsoluteTop - (this.currentStep.stepInstance.stepHeight + ARROW_SIZE + DISTANCE_FROM_TARGET); } else if (this.currentStep.position === 'bottom') { return targetAbsoluteTop; } else if (this.currentStep.position === 'right' || this.currentStep.position === 'left') { return Math.min(targetAbsoluteTop, targetAbsoluteTop + this.currentStep.stepInstance.targetHeight / 2 - this.currentStep.stepInstance.stepHeight / 2); } } scrollIfElementBeyondOtherElements() { if (this.isElementBeyondOthers() === 2) { this.documentService.scrollToTheTop(this.currentStep.targetViewContainer.element); } if (this.isElementBeyondOthers() === 2) { this.documentService.scrollToTheBottom(this.currentStep.targetViewContainer.element); } if (this.isElementBeyondOthers() === 1 && this.documentService.isParentScrollable(this.currentStep.targetViewContainer.element)) { this.documentService.scrollIntoView(this.currentStep.targetViewContainer.element, this.currentStep.isElementOrAncestorFixed); } if (this.isElementBeyondOthers() === 1 && this.documentService.isParentScrollable(this.currentStep.targetViewContainer.element)) { this.currentStep.targetViewContainer.element.nativeElement.scrollIntoView(); } } isElementBeyondOthers() { return this.documentService.isElementBeyondOthers(this.currentStep.targetViewContainer.element, this.currentStep.isElementOrAncestorFixed, 'backdrop'); } } JoyrideStepService.decorators = [ { type: Injectable } ]; JoyrideStepService.ctorParameters = () => [ { type: JoyrideBackdropService }, { type: EventListenerService }, { type: JoyrideStepsContainerService }, { type: DocumentService }, { type: DomRefService }, { type: StepDrawerService }, { type: JoyrideOptionsService }, { type: Router }, { type: LoggerService } ]; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"joyride-step.service.js","sourceRoot":"","sources":["../../../../../projects/ngx-joyride/src/lib/services/joyride-step.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAE3C,OAAO,EAAE,sBAAsB,EAAE,MAAM,4BAA4B,CAAC;AACpE,OAAO,EAAE,oBAAoB,EAAE,MAAM,0BAA0B,CAAC;AAChE,OAAO,EAAE,4BAA4B,EAAE,cAAc,EAAE,MAAM,mCAAmC,CAAC;AACjG,OAAO,EAAE,eAAe,EAAE,MAAM,oBAAoB,CAAC;AACrD,OAAO,EAAE,iBAAiB,EAAE,MAAM,uBAAuB,CAAC;AAC1D,OAAO,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAC9C,OAAO,EAAE,WAAW,EAAE,MAAM,iCAAiC,CAAC;AAC9D,OAAO,EAAE,qBAAqB,EAAE,MAAM,2BAA2B,CAAC;AAClE,OAAO,EAAE,MAAM,EAAE,MAAM,iBAAiB,CAAC;AACzC,OAAO,EAAE,aAAa,EAAc,MAAM,MAAM,CAAC;AAEjD,OAAO,EAAE,uBAAuB,EAAE,qBAAqB,EAAE,MAAM,+BAA+B,CAAC;AAC/F,OAAO,EAAE,aAAa,EAAE,MAAM,kBAAkB,CAAC;AAEjD,MAAM,cAAc,GAAG,EAAE,CAAC;AAE1B,MAAM,CAAC,MAAM,oBAAoB,GAAG,EAAE,CAAC;AACvC,MAAM,CAAC,MAAM,UAAU,GAAG,EAAE,CAAC;AAU7B,MAAM,OAAO,kBAAkB;IAM3B,YACqB,eAAuC,EACvC,aAAmC,EACnC,qBAAmD,EACnD,eAAgC,EAChC,UAAyB,EACzB,iBAAoC,EACpC,cAAqC,EACrC,MAAc,EACd,MAAqB;QARrB,oBAAe,GAAf,eAAe,CAAwB;QACvC,kBAAa,GAAb,aAAa,CAAsB;QACnC,0BAAqB,GAArB,qBAAqB,CAA8B;QACnD,oBAAe,GAAf,eAAe,CAAiB;QAChC,eAAU,GAAV,UAAU,CAAe;QACzB,sBAAiB,GAAjB,iBAAiB,CAAmB;QACpC,mBAAc,GAAd,cAAc,CAAuB;QACrC,WAAM,GAAN,MAAM,CAAQ;QACd,WAAM,GAAN,MAAM,CAAe;QAblC,mBAAc,GAAW,CAAC,CAAC;QAC3B,sBAAiB,GAAW,CAAC,CAAC;QAC9B,kBAAa,GAAmC,IAAI,aAAa,EAAmB,CAAC;QAazF,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAC7B,IAAI,CAAC,uBAAuB,EAAE,CAAC;QAC/B,IAAI,CAAC,uBAAuB,EAAE,CAAC;IACnC,CAAC;IAEO,qBAAqB;QACzB,IAAI,CAAC,cAAc,GAAG,CAAC,CAAC;QACxB,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,UAAU,CAAC,eAAe,EAAE,CAAC,WAAW,GAAG,cAAc,CAAC;IAC5F,CAAC;IAEO,uBAAuB;QAC3B,IAAI,CAAC,aAAa,CAAC,0BAA0B,EAAE,CAAC;QAChD,IAAI,CAAC,aAAa,CAAC,WAAW,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE;YAC9C,IAAI,CAAC,cAAc,GAAG,MAAM,CAAC,OAAO,CAAC;YACrC,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,UAAU,CAAC,eAAe,EAAE,CAAC,WAAW,GAAG,cAAc,CAAC;YAC9G,IAAI,IAAI,CAAC,WAAW;gBAAE,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,IAAI,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;QAChF,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,uBAAuB;QAC3B,IAAI,CAAC,aAAa,CAAC,WAAW,CAAC,SAAS,CAAC,GAAG,EAAE;YAC1C,IAAI,IAAI,CAAC,WAAW;gBAAE,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAC9E,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,QAAQ,CAAC,IAAiB;QAC9B,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,KAAK,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,sBAAsB,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC;QAC7G,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACtC,CAAC;IAED,SAAS;QACL,IAAI,CAAC,aAAa,GAAG,IAAI,aAAa,EAAmB,CAAC;QAC1D,IAAI,CAAC,qBAAqB,CAAC,IAAI,EAAE,CAAC;QAClC,IAAI,CAAC,eAAe,CAAC,iBAAiB,EAAE,CAAC;QAEzC,IAAI,CAAC,WAAW,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;QACtC,IAAI,CAAC,aAAa,CAAC,0BAA0B,EAAE,CAAC;QAChD,IAAI,CAAC,uBAAuB,EAAE,CAAC;QAC/B,OAAO,IAAI,CAAC,aAAa,CAAC,YAAY,EAAE,CAAC;IAC7C,CAAC;IAED,KAAK;QACD,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAC5B,IAAI,CAAC,UAAU,CAAC,eAAe,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QACjD,IAAI,CAAC,aAAa,CAAC,yBAAyB,EAAE,CAAC;QAC/C,IAAI,CAAC,eAAe,CAAC,MAAM,EAAE,CAAC;IAClC,CAAC;IAED,IAAI;QACA,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,IAAI,CAAC,WAAW,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;QACnC,IAAI,CAAC,WAAW,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;IAC1C,CAAC;IAED,IAAI;QACA,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,IAAI,EAAE,CAAC;QACpC,IAAI,CAAC,WAAW,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;IAC1C,CAAC;IAEO,kBAAkB,CAAC,MAAsB;QAC7C,IAAI,SAAS,GAAG,IAAI,CAAC,qBAAqB,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;QAChE,IAAI,SAAS,EAAE;YACX,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC;SACrC;IACL,CAAC;IAEO,uBAAuB;QAC3B,IAAI,CAAC,qBAAqB,CAAC,mBAAmB,CAAC,SAAS,CAAC,WAAW,CAAC,EAAE;YACnE,IAAI,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,WAAW,CAAC,IAAI,KAAK,WAAW,CAAC,IAAI,EAAE;gBAChE,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC;aAClC;QACL,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,WAAW,CAAC,UAA0B;QAC1C,IAAI,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC;QACpC,MAAM,OAAO,GAAG,IAAI,CAAC,cAAc,CAAC,cAAc,EAAE,CAAC;QACrD,IAAI,OAAO,GAAG,GAAG;YAAE,IAAI,CAAC,eAAe,CAAC,MAAM,EAAE,CAAC;QACjD,UAAU,CAAC,GAAG,EAAE;YACZ,IAAI;gBACA,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;aAC7B;YAAC,OAAO,KAAK,EAAE;gBACZ,IAAI,KAAK,YAAY,uBAAuB,EAAE;oBAC1C,IAAI,CAAC,WAAW,CAAC,UAAU,CAAC,CAAC;iBAChC;qBAAM,IAAI,KAAK,YAAY,qBAAqB,EAAE;oBAC/C,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,oEAAoE,CAAC,CAAC;oBACxF,IAAI,CAAC,KAAK,EAAE,CAAC;iBAChB;qBAAM;oBACH,MAAM,IAAI,KAAK,CAAC,KAAK,CAAC,CAAC;iBAC1B;aACJ;QACL,CAAC,EAAE,OAAO,CAAC,CAAC;IAChB,CAAC;IAEO,QAAQ,CAAC,UAA0B;QACvC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;QAE9D,IAAI,IAAI,CAAC,WAAW,IAAI,IAAI;YAAE,MAAM,IAAI,uBAAuB,CAAC,EAAE,CAAC,CAAC;QACpE,uEAAuE;QACvE,IAAI,CAAC,kCAAkC,EAAE,CAAC;QAC1C,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAC5C,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAChC,IAAI,CAAC,kCAAkC,EAAE,CAAC;QAC1C,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC;IACvC,CAAC;IAEO,iBAAiB,CAAC,UAA0B;QAChD,IAAI,QAAQ,GAAoB;YAC5B,MAAM,EAAE,IAAI,CAAC,qBAAqB,CAAC,aAAa,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC;YACvE,IAAI,EAAE,IAAI,CAAC,WAAW,CAAC,IAAI;YAC3B,KAAK,EAAE,IAAI,CAAC,WAAW,CAAC,KAAK;YAC7B,UAAU;SACb,CAAC;QACF,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IACtC,CAAC;IAEO,oBAAoB;QACxB,IAAI,IAAI,CAAC,WAAW;YAAE,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACvD,IAAI,CAAC,aAAa,CAAC,QAAQ,EAAE,CAAC;IAClC,CAAC;IACO,iBAAiB;QACrB,IAAI,IAAI,CAAC,WAAW;YAAE,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;IAC1E,CAAC;IAEO,kCAAkC;QACtC,IAAI,CAAC,qCAAqC,EAAE,CAAC;QAC7C,IAAI,CAAC,kCAAkC,EAAE,CAAC;IAC9C,CAAC;IAEO,qCAAqC;QACzC,IAAI,iBAAiB,GAAG,IAAI,CAAC,iCAAiC,EAAE,CAAC;QACjE,IAAI,iBAAiB,GAAG,IAAI,CAAC,iBAAiB,EAAE;YAC5C,IAAI,CAAC,UAAU,CAAC,eAAe,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,iBAAiB,GAAG,IAAI,CAAC,iBAAiB,CAAC,CAAC;SAC7F;IACL,CAAC;IAEO,kCAAkC;QACtC,IAAI,cAAc,GAAG,IAAI,CAAC,8BAA8B,EAAE,CAAC;QAC3D,IAAI,cAAc,GAAG,IAAI,CAAC,cAAc,EAAE;YACtC,IAAI,CAAC,UAAU,CAAC,eAAe,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC;SACvF;IACL,CAAC;IAEO,iCAAiC;QACrC,IAAI,iBAAiB,GAAG,IAAI,CAAC,eAAe,CAAC,qBAAqB,CAAC,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;QACjH,IAAI,IAAI,CAAC,WAAW,CAAC,QAAQ,KAAK,KAAK,EAAE;YACrC,OAAO,iBAAiB,GAAG,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,YAAY,CAAC;SACzE;aAAM,IAAI,IAAI,CAAC,WAAW,CAAC,QAAQ,KAAK,QAAQ,EAAE;YAC/C,OAAO,CACH,iBAAiB;gBACjB,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,YAAY;gBAC1C,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,UAAU;gBACxC,UAAU;gBACV,oBAAoB,CACvB,CAAC;SACL;aAAM,IAAI,IAAI,CAAC,WAAW,CAAC,QAAQ,KAAK,OAAO,IAAI,IAAI,CAAC,WAAW,CAAC,QAAQ,KAAK,MAAM,EAAE;YACtF,OAAO,IAAI,CAAC,GAAG,CACX,iBAAiB,GAAG,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,YAAY,EAC9D,iBAAiB,GAAG,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,YAAY,GAAG,CAAC,GAAG,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,UAAU,GAAG,CAAC,CACpH,CAAC;SACL;IACL,CAAC;IAEO,8BAA8B;QAClC,IAAI,iBAAiB,GAAG,IAAI,CAAC,eAAe,CAAC,qBAAqB,CAAC,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;QACjH,IAAI,IAAI,CAAC,WAAW,CAAC,QAAQ,KAAK,KAAK,EAAE;YACrC,OAAO,iBAAiB,GAAG,CAAC,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,UAAU,GAAG,UAAU,GAAG,oBAAoB,CAAC,CAAC;SAC7G;aAAM,IAAI,IAAI,CAAC,WAAW,CAAC,QAAQ,KAAK,QAAQ,EAAE;YAC/C,OAAO,iBAAiB,CAAC;SAC5B;aAAM,IAAI,IAAI,CAAC,WAAW,CAAC,QAAQ,KAAK,OAAO,IAAI,IAAI,CAAC,WAAW,CAAC,QAAQ,KAAK,MAAM,EAAE;YACtF,OAAO,IAAI,CAAC,GAAG,CACX,iBAAiB,EACjB,iBAAiB,GAAG,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,YAAY,GAAG,CAAC,GAAG,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,UAAU,GAAG,CAAC,CACpH,CAAC;SACL;IACL,CAAC;IAEO,kCAAkC;QACtC,IAAI,IAAI,CAAC,qBAAqB,EAAE,KAAK,CAAC,EAAE;YACpC,IAAI,CAAC,eAAe,CAAC,cAAc,CAAC,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;SACrF;QACD,IAAI,IAAI,CAAC,qBAAqB,EAAE,KAAK,CAAC,EAAE;YACpC,IAAI,CAAC,eAAe,CAAC,iBAAiB,CAAC,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC;SACxF;QACD,IAAI,IAAI,CAAC,qBAAqB,EAAE,KAAK,CAAC,IAAI,IAAI,CAAC,eAAe,CAAC,kBAAkB,CAAC,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,OAAO,CAAC,EAAE;YAC7H,IAAI,CAAC,eAAe,CAAC,cAAc,CAAC,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,OAAO,EAAE,IAAI,CAAC,WAAW,CAAC,wBAAwB,CAAC,CAAC;SAChI;QACD,IAAI,IAAI,CAAC,qBAAqB,EAAE,KAAK,CAAC,IAAI,IAAI,CAAC,eAAe,CAAC,kBAAkB,CAAC,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,OAAO,CAAC,EAAE;YAC7H,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,OAAO,CAAC,aAAa,CAAC,cAAc,EAAE,CAAC;SAC/E;IACL,CAAC;IAEO,qBAAqB;QACzB,OAAO,IAAI,CAAC,eAAe,CAAC,qBAAqB,CAC7C,IAAI,CAAC,WAAW,CAAC,mBAAmB,CAAC,OAAO,EAC5C,IAAI,CAAC,WAAW,CAAC,wBAAwB,EACzC,UAAU,CACb,CAAC;IACN,CAAC;;;YA1NJ,UAAU;;;YA1BF,sBAAsB;YACtB,oBAAoB;YACpB,4BAA4B;YAC5B,eAAe;YAEf,aAAa;YADb,iBAAiB;YAGjB,qBAAqB;YACrB,MAAM;YAIN,aAAa","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { JoyrideStep } from '../models/joyride-step.class';\r\nimport { JoyrideBackdropService } from './joyride-backdrop.service';\r\nimport { EventListenerService } from './event-listener.service';\r\nimport { JoyrideStepsContainerService, StepActionType } from './joyride-steps-container.service';\r\nimport { DocumentService } from './document.service';\r\nimport { StepDrawerService } from './step-drawer.service';\r\nimport { DomRefService } from './dom.service';\r\nimport { NO_POSITION } from '../directives/joyride.directive';\r\nimport { JoyrideOptionsService } from './joyride-options.service';\r\nimport { Router } from '@angular/router';\r\nimport { ReplaySubject, Observable } from 'rxjs';\r\nimport { JoyrideStepInfo } from '../models/joyride-step-info.class';\r\nimport { JoyrideStepDoesNotExist, JoyrideStepOutOfRange } from '../models/joyride-error.class';\r\nimport { LoggerService } from './logger.service';\r\n\r\nconst SCROLLBAR_SIZE = 20;\r\n\r\nexport const DISTANCE_FROM_TARGET = 15;\r\nexport const ARROW_SIZE = 10;\r\n\r\nexport interface IJoyrideStepService {\r\n    startTour(): Observable<JoyrideStepInfo>;\r\n    close(): any;\r\n    prev(): any;\r\n    next(): any;\r\n}\r\n\r\n@Injectable()\r\nexport class JoyrideStepService implements IJoyrideStepService {\r\n    private currentStep: JoyrideStep;\r\n    private winTopPosition: number = 0;\r\n    private winBottomPosition: number = 0;\r\n    private stepsObserver: ReplaySubject<JoyrideStepInfo> = new ReplaySubject<JoyrideStepInfo>();\r\n\r\n    constructor(\r\n        private readonly backDropService: JoyrideBackdropService,\r\n        private readonly eventListener: EventListenerService,\r\n        private readonly stepsContainerService: JoyrideStepsContainerService,\r\n        private readonly documentService: DocumentService,\r\n        private readonly DOMService: DomRefService,\r\n        private readonly stepDrawerService: StepDrawerService,\r\n        private readonly optionsService: JoyrideOptionsService,\r\n        private readonly router: Router,\r\n        private readonly logger: LoggerService\r\n    ) {\r\n        this.initViewportPositions();\r\n        this.subscribeToScrollEvents();\r\n        this.subscribeToResizeEvents();\r\n    }\r\n\r\n    private initViewportPositions() {\r\n        this.winTopPosition = 0;\r\n        this.winBottomPosition = this.DOMService.getNativeWindow().innerHeight - SCROLLBAR_SIZE;\r\n    }\r\n\r\n    private subscribeToScrollEvents() {\r\n        this.eventListener.startListeningScrollEvents();\r\n        this.eventListener.scrollEvent.subscribe(scroll => {\r\n            this.winTopPosition = scroll.scrollY;\r\n            this.winBottomPosition = this.winTopPosition + this.DOMService.getNativeWindow().innerHeight - SCROLLBAR_SIZE;\r\n            if (this.currentStep) this.backDropService.redraw(this.currentStep, scroll);\r\n        });\r\n    }\r\n\r\n    private subscribeToResizeEvents() {\r\n        this.eventListener.resizeEvent.subscribe(() => {\r\n            if (this.currentStep) this.backDropService.redrawTarget(this.currentStep);\r\n        });\r\n    }\r\n\r\n    private drawStep(step: JoyrideStep) {\r\n        step.position = step.position === NO_POSITION ? this.optionsService.getStepDefaultPosition() : step.position;\r\n        this.stepDrawerService.draw(step);\r\n    }\r\n\r\n    startTour(): Observable<JoyrideStepInfo> {\r\n        this.stepsObserver = new ReplaySubject<JoyrideStepInfo>();\r\n        this.stepsContainerService.init();\r\n        this.documentService.setDocumentHeight();\r\n\r\n        this.tryShowStep(StepActionType.NEXT);\r\n        this.eventListener.startListeningResizeEvents();\r\n        this.subscribeToStepsUpdates();\r\n        return this.stepsObserver.asObservable();\r\n    }\r\n\r\n    close() {\r\n        this.removeCurrentStep();\r\n        this.notifyTourIsFinished();\r\n        this.DOMService.getNativeWindow().scrollTo(0, 0);\r\n        this.eventListener.stopListeningResizeEvents();\r\n        this.backDropService.remove();\r\n    }\r\n\r\n    prev() {\r\n        this.removeCurrentStep();\r\n        this.currentStep.prevCliked.emit();\r\n        this.tryShowStep(StepActionType.PREV);\r\n    }\r\n\r\n    next() {\r\n        this.removeCurrentStep();\r\n        this.currentStep.nextClicked.emit();\r\n        this.tryShowStep(StepActionType.NEXT);\r\n    }\r\n\r\n    private navigateToStepPage(action: StepActionType) {\r\n        let stepRoute = this.stepsContainerService.getStepRoute(action);\r\n        if (stepRoute) {\r\n            this.router.navigate([stepRoute]);\r\n        }\r\n    }\r\n\r\n    private subscribeToStepsUpdates() {\r\n        this.stepsContainerService.stepHasBeenModified.subscribe(updatedStep => {\r\n            if (this.currentStep && this.currentStep.name === updatedStep.name) {\r\n                this.currentStep = updatedStep;\r\n            }\r\n        });\r\n    }\r\n\r\n    private tryShowStep(actionType: StepActionType) {\r\n        this.navigateToStepPage(actionType);\r\n        const timeout = this.optionsService.getWaitingTime();\r\n        if (timeout > 100) this.backDropService.remove();\r\n        setTimeout(() => {\r\n            try {\r\n                this.showStep(actionType);\r\n            } catch (error) {\r\n                if (error instanceof JoyrideStepDoesNotExist) {\r\n                    this.tryShowStep(actionType);\r\n                } else if (error instanceof JoyrideStepOutOfRange) {\r\n                    this.logger.error('Forcing the tour closure: First or Last step not found in the DOM.');\r\n                    this.close();\r\n                } else {\r\n                    throw new Error(error);\r\n                }\r\n            }\r\n        }, timeout);\r\n    }\r\n\r\n    private showStep(actionType: StepActionType) {\r\n        this.currentStep = this.stepsContainerService.get(actionType);\r\n\r\n        if (this.currentStep == null) throw new JoyrideStepDoesNotExist('');\r\n        // Scroll the element to get it visible if it's in a scrollable element\r\n        this.scrollIfElementBeyondOtherElements();\r\n        this.backDropService.draw(this.currentStep);\r\n        this.drawStep(this.currentStep);\r\n        this.scrollIfStepAndTargetAreNotVisible();\r\n        this.notifyStepClicked(actionType);\r\n    }\r\n\r\n    private notifyStepClicked(actionType: StepActionType) {\r\n        let stepInfo: JoyrideStepInfo = {\r\n            number: this.stepsContainerService.getStepNumber(this.currentStep.name),\r\n            name: this.currentStep.name,\r\n            route: this.currentStep.route,\r\n            actionType\r\n        };\r\n        this.stepsObserver.next(stepInfo);\r\n    }\r\n\r\n    private notifyTourIsFinished() {\r\n        if (this.currentStep) this.currentStep.tourDone.emit();\r\n        this.stepsObserver.complete();\r\n    }\r\n    private removeCurrentStep() {\r\n        if (this.currentStep) this.stepDrawerService.remove(this.currentStep);\r\n    }\r\n\r\n    private scrollIfStepAndTargetAreNotVisible() {\r\n        this.scrollWhenTargetOrStepAreHiddenBottom();\r\n        this.scrollWhenTargetOrStepAreHiddenTop();\r\n    }\r\n\r\n    private scrollWhenTargetOrStepAreHiddenBottom() {\r\n        let totalTargetBottom = this.getMaxTargetAndStepBottomPosition();\r\n        if (totalTargetBottom > this.winBottomPosition) {\r\n            this.DOMService.getNativeWindow().scrollBy(0, totalTargetBottom - this.winBottomPosition);\r\n        }\r\n    }\r\n\r\n    private scrollWhenTargetOrStepAreHiddenTop() {\r\n        let totalTargetTop = this.getMaxTargetAndStepTopPosition();\r\n        if (totalTargetTop < this.winTopPosition) {\r\n            this.DOMService.getNativeWindow().scrollBy(0, totalTargetTop - this.winTopPosition);\r\n        }\r\n    }\r\n\r\n    private getMaxTargetAndStepBottomPosition(): number {\r\n        let targetAbsoluteTop = this.documentService.getElementAbsoluteTop(this.currentStep.targetViewContainer.element);\r\n        if (this.currentStep.position === 'top') {\r\n            return targetAbsoluteTop + this.currentStep.stepInstance.targetHeight;\r\n        } else if (this.currentStep.position === 'bottom') {\r\n            return (\r\n                targetAbsoluteTop +\r\n                this.currentStep.stepInstance.targetHeight +\r\n                this.currentStep.stepInstance.stepHeight +\r\n                ARROW_SIZE +\r\n                DISTANCE_FROM_TARGET\r\n            );\r\n        } else if (this.currentStep.position === 'right' || this.currentStep.position === 'left') {\r\n            return Math.max(\r\n                targetAbsoluteTop + this.currentStep.stepInstance.targetHeight,\r\n                targetAbsoluteTop + this.currentStep.stepInstance.targetHeight / 2 + this.currentStep.stepInstance.stepHeight / 2\r\n            );\r\n        }\r\n    }\r\n\r\n    private getMaxTargetAndStepTopPosition() {\r\n        let targetAbsoluteTop = this.documentService.getElementAbsoluteTop(this.currentStep.targetViewContainer.element);\r\n        if (this.currentStep.position === 'top') {\r\n            return targetAbsoluteTop - (this.currentStep.stepInstance.stepHeight + ARROW_SIZE + DISTANCE_FROM_TARGET);\r\n        } else if (this.currentStep.position === 'bottom') {\r\n            return targetAbsoluteTop;\r\n        } else if (this.currentStep.position === 'right' || this.currentStep.position === 'left') {\r\n            return Math.min(\r\n                targetAbsoluteTop,\r\n                targetAbsoluteTop + this.currentStep.stepInstance.targetHeight / 2 - this.currentStep.stepInstance.stepHeight / 2\r\n            );\r\n        }\r\n    }\r\n\r\n    private scrollIfElementBeyondOtherElements() {\r\n        if (this.isElementBeyondOthers() === 2) {\r\n            this.documentService.scrollToTheTop(this.currentStep.targetViewContainer.element);\r\n        }\r\n        if (this.isElementBeyondOthers() === 2) {\r\n            this.documentService.scrollToTheBottom(this.currentStep.targetViewContainer.element);\r\n        }\r\n        if (this.isElementBeyondOthers() === 1 && this.documentService.isParentScrollable(this.currentStep.targetViewContainer.element)) {\r\n            this.documentService.scrollIntoView(this.currentStep.targetViewContainer.element, this.currentStep.isElementOrAncestorFixed);\r\n        }\r\n        if (this.isElementBeyondOthers() === 1 && this.documentService.isParentScrollable(this.currentStep.targetViewContainer.element)) {\r\n            this.currentStep.targetViewContainer.element.nativeElement.scrollIntoView();\r\n        }\r\n    }\r\n\r\n    private isElementBeyondOthers() {\r\n        return this.documentService.isElementBeyondOthers(\r\n            this.currentStep.targetViewContainer.element,\r\n            this.currentStep.isElementOrAncestorFixed,\r\n            'backdrop'\r\n        );\r\n    }\r\n}\r\n"]}