ngx-joyride
Version:
An Angular Tour (Joyride) library built entirely in Angular, without using any heavy external dependencies like Bootstrap or JQuery. From now on you can easily guide your users through your site showing them all the sections and features.
141 lines • 19.8 kB
JavaScript
import { Directive, Input, Output, EventEmitter, Inject, PLATFORM_ID } from '@angular/core';
import { JoyrideStep } from '../models/joyride-step.class';
import { JoyrideError } from '../models/joyride-error.class';
import { isPlatformBrowser } from '@angular/common';
import { Observable } from 'rxjs';
import * as i0 from "@angular/core";
import * as i1 from "../services/joyride-steps-container.service";
import * as i2 from "../services/dom.service";
import * as i3 from "@angular/router";
import * as i4 from "../services/templates.service";
export const NO_POSITION = 'NO_POSITION';
export class JoyrideDirective {
constructor(joyrideStepsContainer, viewContainerRef, domService, router, templateService, platformId) {
this.joyrideStepsContainer = joyrideStepsContainer;
this.viewContainerRef = viewContainerRef;
this.domService = domService;
this.router = router;
this.templateService = templateService;
this.platformId = platformId;
this.stepPosition = NO_POSITION;
this.prev = new EventEmitter();
this.next = new EventEmitter();
this.done = new EventEmitter();
this.subscriptions = [];
this.windowRef = this.domService.getNativeWindow();
this.step = new JoyrideStep();
}
ngAfterViewInit() {
if (!isPlatformBrowser(this.platformId))
return;
if (this.prevTemplate)
this.templateService.setPrevButton(this.prevTemplate);
if (this.nextTemplate)
this.templateService.setNextButton(this.nextTemplate);
if (this.doneTemplate)
this.templateService.setDoneButton(this.doneTemplate);
if (this.counterTemplate)
this.templateService.setCounter(this.counterTemplate);
this.step.position = this.stepPosition;
this.step.targetViewContainer = this.viewContainerRef;
this.setAsyncFields(this.step);
this.step.stepContent = this.stepContent;
this.step.stepContentParams = this.stepContentParams;
this.step.nextClicked = this.next;
this.step.prevCliked = this.prev;
this.step.tourDone = this.done;
if (!this.name)
throw new JoyrideError("All the steps should have the 'joyrideStep' property set with a custom name.");
this.step.name = this.name;
this.step.route = this.router.url.substr(0, 1) === '/' ? this.router.url.substr(1) : this.router.url;
this.step.transformCssStyle = this.windowRef.getComputedStyle(this.viewContainerRef.element.nativeElement).transform;
this.step.isElementOrAncestorFixed =
this.isElementFixed(this.viewContainerRef.element) ||
this.isAncestorsFixed(this.viewContainerRef.element.nativeElement.parentElement);
this.joyrideStepsContainer.addStep(this.step);
}
ngOnChanges(changes) {
if (changes['title'] || changes['text']) {
this.setAsyncFields(this.step);
}
}
isElementFixed(element) {
return this.windowRef.getComputedStyle(element.nativeElement).position === 'fixed';
}
setAsyncFields(step) {
if (this.title instanceof Observable) {
this.subscriptions.push(this.title.subscribe(title => {
step.title.next(title);
}));
}
else {
step.title.next(this.title);
}
if (this.text instanceof Observable) {
this.subscriptions.push(this.text.subscribe(text => {
step.text.next(text);
}));
}
else {
step.text.next(this.text);
}
}
isAncestorsFixed(nativeElement) {
if (!nativeElement || !nativeElement.parentElement)
return false;
let isElementFixed = this.windowRef.getComputedStyle(nativeElement.parentElement).position === 'fixed';
if (nativeElement.nodeName === 'BODY') {
return isElementFixed;
}
if (isElementFixed)
return true;
else
return this.isAncestorsFixed(nativeElement.parentElement);
}
ngOnDestroy() {
this.subscriptions.forEach(sub => {
sub.unsubscribe();
});
}
}
JoyrideDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.1.1", ngImport: i0, type: JoyrideDirective, deps: [{ token: i1.JoyrideStepsContainerService }, { token: i0.ViewContainerRef }, { token: i2.DomRefService }, { token: i3.Router }, { token: i4.TemplatesService }, { token: PLATFORM_ID }], target: i0.ɵɵFactoryTarget.Directive });
JoyrideDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.1.1", type: JoyrideDirective, selector: "joyrideStep, [joyrideStep]", inputs: { name: ["joyrideStep", "name"], nextStep: "nextStep", title: "title", text: "text", stepPosition: "stepPosition", stepContent: "stepContent", stepContentParams: "stepContentParams", prevTemplate: "prevTemplate", nextTemplate: "nextTemplate", doneTemplate: "doneTemplate", counterTemplate: "counterTemplate" }, outputs: { prev: "prev", next: "next", done: "done" }, usesOnChanges: true, ngImport: i0 });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.1.1", ngImport: i0, type: JoyrideDirective, decorators: [{
type: Directive,
args: [{
selector: 'joyrideStep, [joyrideStep]'
}]
}], ctorParameters: function () { return [{ type: i1.JoyrideStepsContainerService }, { type: i0.ViewContainerRef }, { type: i2.DomRefService }, { type: i3.Router }, { type: i4.TemplatesService }, { type: Object, decorators: [{
type: Inject,
args: [PLATFORM_ID]
}] }]; }, propDecorators: { name: [{
type: Input,
args: ['joyrideStep']
}], nextStep: [{
type: Input
}], title: [{
type: Input
}], text: [{
type: Input
}], stepPosition: [{
type: Input
}], stepContent: [{
type: Input
}], stepContentParams: [{
type: Input
}], prevTemplate: [{
type: Input
}], nextTemplate: [{
type: Input
}], doneTemplate: [{
type: Input
}], counterTemplate: [{
type: Input
}], prev: [{
type: Output
}], next: [{
type: Output
}], done: [{
type: Output
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"joyride.directive.js","sourceRoot":"","sources":["../../../../../projects/ngx-joyride/src/lib/directives/joyride.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n    Directive,\n    ElementRef,\n    AfterViewInit,\n    Input,\n    ViewContainerRef,\n    TemplateRef,\n    Output,\n    EventEmitter,\n    Inject,\n    PLATFORM_ID,\n    OnChanges,\n    SimpleChanges,\n    OnDestroy\n} from '@angular/core';\nimport { JoyrideStep } from '../models/joyride-step.class';\nimport { JoyrideStepsContainerService } from '../services/joyride-steps-container.service';\nimport { JoyrideError } from '../models/joyride-error.class';\nimport { Router } from '@angular/router';\nimport { DomRefService } from '../services/dom.service';\nimport { isPlatformBrowser } from '@angular/common';\nimport { TemplatesService } from '../services/templates.service';\nimport { Observable, Subscription } from 'rxjs';\n\nexport const NO_POSITION = 'NO_POSITION';\n\n@Directive({\n    selector: 'joyrideStep, [joyrideStep]'\n})\nexport class JoyrideDirective implements AfterViewInit, OnChanges, OnDestroy {\n    @Input('joyrideStep')\n    name: string;\n\n    @Input()\n    nextStep?: string;\n\n    @Input()\n    title?: string | Observable<string>;\n\n    @Input()\n    text?: string | Observable<string>;\n\n    @Input()\n    stepPosition?: string = NO_POSITION;\n\n    @Input()\n    stepContent?: TemplateRef<any>;\n\n    @Input()\n    stepContentParams?: Object;\n\n    @Input()\n    prevTemplate?: TemplateRef<any>;\n\n    @Input()\n    nextTemplate?: TemplateRef<any>;\n\n    @Input()\n    doneTemplate?: TemplateRef<any>;\n\n    @Input()\n    counterTemplate?: TemplateRef<any>;\n\n    @Output()\n    prev?: EventEmitter<any> = new EventEmitter<any>();\n\n    @Output()\n    next?: EventEmitter<any> = new EventEmitter<any>();\n\n    @Output()\n    done?: EventEmitter<any> = new EventEmitter<any>();\n\n    private windowRef: Window;\n    private step: JoyrideStep;\n    private subscriptions: Subscription[] = [];\n\n    constructor(\n        private readonly joyrideStepsContainer: JoyrideStepsContainerService,\n        private viewContainerRef: ViewContainerRef,\n        private readonly domService: DomRefService,\n        private readonly router: Router,\n        private readonly templateService: TemplatesService,\n        @Inject(PLATFORM_ID) private platformId: Object\n    ) {\n        this.windowRef = this.domService.getNativeWindow();\n        this.step = new JoyrideStep();\n    }\n\n    ngAfterViewInit() {\n        if (!isPlatformBrowser(this.platformId)) return;\n        if (this.prevTemplate) this.templateService.setPrevButton(this.prevTemplate);\n        if (this.nextTemplate) this.templateService.setNextButton(this.nextTemplate);\n        if (this.doneTemplate) this.templateService.setDoneButton(this.doneTemplate);\n        if (this.counterTemplate) this.templateService.setCounter(this.counterTemplate);\n        this.step.position = this.stepPosition;\n        this.step.targetViewContainer = this.viewContainerRef;\n        this.setAsyncFields(this.step);\n        this.step.stepContent = this.stepContent;\n        this.step.stepContentParams = this.stepContentParams;\n        this.step.nextClicked = this.next;\n        this.step.prevCliked = this.prev;\n        this.step.tourDone = this.done;\n        if (!this.name) throw new JoyrideError(\"All the steps should have the 'joyrideStep' property set with a custom name.\");\n        this.step.name = this.name;\n        this.step.route = this.router.url.substr(0, 1) === '/' ? this.router.url.substr(1) : this.router.url;\n        this.step.transformCssStyle = this.windowRef.getComputedStyle(this.viewContainerRef.element.nativeElement).transform;\n        this.step.isElementOrAncestorFixed =\n            this.isElementFixed(this.viewContainerRef.element) ||\n            this.isAncestorsFixed(this.viewContainerRef.element.nativeElement.parentElement);\n\n        this.joyrideStepsContainer.addStep(this.step);\n    }\n\n    ngOnChanges(changes: SimpleChanges) {\n        if (changes['title'] || changes['text']) {\n            this.setAsyncFields(this.step);\n        }\n    }\n\n    private isElementFixed(element: ElementRef) {\n        return this.windowRef.getComputedStyle(element.nativeElement).position === 'fixed';\n    }\n\n    private setAsyncFields(step: JoyrideStep) {\n        if (this.title instanceof Observable) {\n            this.subscriptions.push(\n                this.title.subscribe(title => {\n                    step.title.next(title);\n                })\n            );\n        } else {\n            step.title.next(this.title);\n        }\n        if (this.text instanceof Observable) {\n            this.subscriptions.push(\n                this.text.subscribe(text => {\n                    step.text.next(text);\n                })\n            );\n        } else {\n            step.text.next(this.text);\n        }\n    }\n\n    private isAncestorsFixed(nativeElement: any): boolean {\n        if (!nativeElement || !nativeElement.parentElement) return false;\n        let isElementFixed = this.windowRef.getComputedStyle(nativeElement.parentElement).position === 'fixed';\n        if (nativeElement.nodeName === 'BODY') {\n            return isElementFixed;\n        }\n        if (isElementFixed) return true;\n        else return this.isAncestorsFixed(nativeElement.parentElement);\n    }\n\n    ngOnDestroy(): void {\n        this.subscriptions.forEach(sub => {\n            sub.unsubscribe();\n        });\n    }\n}\n"]}