import * as tslib_1 from "tslib"; import { bindLifecycleEvents } from '../../providers/angular-delegate'; import { computeStackId, destroyView, getUrl, insertView, isTabSwitch, toSegments } from './stack-utils'; var StackController = /** @class */ (function () { function StackController(tabsPrefix, containerEl, router, navCtrl, zone, location) { this.containerEl = containerEl; this.router = router; this.navCtrl = navCtrl; this.zone = zone; this.location = location; this.views = []; this.skipTransition = false; this.nextId = 0; this.tabsPrefix = tabsPrefix !== undefined ? toSegments(tabsPrefix) : undefined; } StackController.prototype.createView = function (ref, activatedRoute) { var url = getUrl(this.router, activatedRoute); var element = (ref && ref.location && ref.location.nativeElement); var unlistenEvents = bindLifecycleEvents(this.zone, ref.instance, element); return { id: this.nextId++, stackId: computeStackId(this.tabsPrefix, url), unlistenEvents: unlistenEvents, element: element, ref: ref, url: url, }; }; StackController.prototype.getExistingView = function (activatedRoute) { var activatedUrlKey = getUrl(this.router, activatedRoute); var view = this.views.find(function (vw) { return vw.url === activatedUrlKey; }); if (view) { view.ref.changeDetectorRef.reattach(); } return view; }; StackController.prototype.setActive = function (enteringView) { var _this = this; var consumeResult = this.navCtrl.consumeTransition(); var direction = consumeResult.direction, animation = consumeResult.animation, animationBuilder = consumeResult.animationBuilder; var leavingView = this.activeView; var tabSwitch = isTabSwitch(enteringView, leavingView); if (tabSwitch) { direction = 'back'; animation = undefined; } var viewsSnapshot = this.views.slice(); var currentNavigation; var router = this.router; // Angular >= 7.2.0 if (router.getCurrentNavigation) { currentNavigation = router.getCurrentNavigation(); // Angular < 7.2.0 } else if (router.navigations && router.navigations.value) { currentNavigation = router.navigations.value; } /** * If the navigation action * sets `replaceUrl: true` * then we need to make sure * we remove the last item * from our views stack */ if (currentNavigation && currentNavigation.extras && currentNavigation.extras.replaceUrl) { if (this.views.length > 0) { this.views.splice(-1, 1); } } var reused = this.views.includes(enteringView); var views = this.insertView(enteringView, direction); // Trigger change detection before transition starts // This will call ngOnInit() the first time too, just after the view // was attached to the dom, but BEFORE the transition starts if (!reused) { enteringView.ref.changeDetectorRef.detectChanges(); } /** * If we are going back from a page that * was presented using a custom animation * we should default to using that * unless the developer explicitly * provided another animation. */ var customAnimation = enteringView.animationBuilder; if (animationBuilder === undefined && direction === 'back' && !tabSwitch && customAnimation !== undefined) { animationBuilder = customAnimation; } /** * Save any custom animation so that navigating * back will use this custom animation by default. */ if (leavingView) { leavingView.animationBuilder = animationBuilder; } // Wait until previous transitions finish return this.zone.runOutsideAngular(function () { return _this.wait(function () { // disconnect leaving page from change detection to // reduce jank during the page transition if (leavingView) { leavingView.ref.changeDetectorRef.detach(); } // In case the enteringView is the same as the leavingPage we need to reattach() enteringView.ref.changeDetectorRef.reattach(); return _this.transition(enteringView, leavingView, animation, _this.canGoBack(1), false, animationBuilder) .then(function () { return cleanupAsync(enteringView, views, viewsSnapshot, _this.location); }) .then(function () { return ({ enteringView: enteringView, direction: direction, animation: animation, tabSwitch: tabSwitch }); }); }); }); }; StackController.prototype.canGoBack = function (deep, stackId) { if (stackId === void 0) { stackId = this.getActiveStackId(); } return this.getStack(stackId).length > deep; }; StackController.prototype.pop = function (deep, stackId) { var _this = this; if (stackId === void 0) { stackId = this.getActiveStackId(); } return this.zone.run(function () { var views = _this.getStack(stackId); if (views.length <= deep) { return Promise.resolve(false); } var view = views[views.length - deep - 1]; var url = view.url; var viewSavedData = view.savedData; if (viewSavedData) { var primaryOutlet = viewSavedData.get('primary'); if (primaryOutlet && primaryOutlet.route && primaryOutlet.route._routerState && primaryOutlet.route._routerState.snapshot && primaryOutlet.route._routerState.snapshot.url) { url = primaryOutlet.route._routerState.snapshot.url; } } var animationBuilder = _this.navCtrl.consumeTransition().animationBuilder; return _this.navCtrl.navigateBack(url, tslib_1.__assign({}, view.savedExtras, { animation: animationBuilder })).then(function () { return true; }); }); }; StackController.prototype.startBackTransition = function () { var _this = this; var leavingView = this.activeView; if (leavingView) { var views = this.getStack(leavingView.stackId); var enteringView_1 = views[views.length - 2]; var customAnimation_1 = enteringView_1.animationBuilder; return this.wait(function () { return _this.transition(enteringView_1, // entering view leavingView, // leaving view 'back', _this.canGoBack(2), true, customAnimation_1); }); } return Promise.resolve(); }; StackController.prototype.endBackTransition = function (shouldComplete) { if (shouldComplete) { this.skipTransition = true; this.pop(1); } else if (this.activeView) { cleanup(this.activeView, this.views, this.views, this.location); } }; StackController.prototype.getLastUrl = function (stackId) { var views = this.getStack(stackId); return views.length > 0 ? views[views.length - 1] : undefined; }; /** * @internal */ StackController.prototype.getRootUrl = function (stackId) { var views = this.getStack(stackId); return views.length > 0 ? views[0] : undefined; }; StackController.prototype.getActiveStackId = function () { return this.activeView ? this.activeView.stackId : undefined; }; StackController.prototype.hasRunningTask = function () { return this.runningTask !== undefined; }; StackController.prototype.destroy = function () { this.containerEl = undefined; this.views.forEach(destroyView); this.activeView = undefined; this.views = []; }; StackController.prototype.getStack = function (stackId) { return this.views.filter(function (v) { return v.stackId === stackId; }); }; StackController.prototype.insertView = function (enteringView, direction) { this.activeView = enteringView; this.views = insertView(this.views, enteringView, direction); return this.views.slice(); }; StackController.prototype.transition = function (enteringView, leavingView, direction, showGoBack, progressAnimation, animationBuilder) { if (this.skipTransition) { this.skipTransition = false; return Promise.resolve(false); } if (leavingView === enteringView) { return Promise.resolve(false); } var enteringEl = enteringView ? enteringView.element : undefined; var leavingEl = leavingView ? leavingView.element : undefined; var containerEl = this.containerEl; if (enteringEl && enteringEl !== leavingEl) { enteringEl.classList.add('ion-page'); enteringEl.classList.add('ion-page-invisible'); if (enteringEl.parentElement !== containerEl) { containerEl.appendChild(enteringEl); } if (containerEl.commit) { return containerEl.commit(enteringEl, leavingEl, { deepWait: true, duration: direction === undefined ? 0 : undefined, direction: direction, showGoBack: showGoBack, progressAnimation: progressAnimation, animationBuilder: animationBuilder }); } } return Promise.resolve(false); }; StackController.prototype.wait = function (task) { return tslib_1.__awaiter(this, void 0, void 0, function () { var promise; var _this = this; return tslib_1.__generator(this, function (_a) { switch (_a.label) { case 0: if (!(this.runningTask !== undefined)) return [3 /*break*/, 2]; return [4 /*yield*/, this.runningTask]; case 1: _a.sent(); this.runningTask = undefined; _a.label = 2; case 2: promise = this.runningTask = task(); promise.finally(function () { return _this.runningTask = undefined; }); return [2 /*return*/, promise]; } }); }); }; return StackController; }()); export { StackController }; var cleanupAsync = function (activeRoute, views, viewsSnapshot, location) { if (typeof requestAnimationFrame === 'function') { return new Promise(function (resolve) { requestAnimationFrame(function () { cleanup(activeRoute, views, viewsSnapshot, location); resolve(); }); }); } return Promise.resolve(); }; var ɵ0 = cleanupAsync; var cleanup = function (activeRoute, views, viewsSnapshot, location) { viewsSnapshot .filter(function (view) { return !views.includes(view); }) .forEach(destroyView); views.forEach(function (view) { /** * In the event that a user navigated multiple * times in rapid succession, we want to make sure * we don't pre-emptively detach a view while * it is in mid-transition. * * In this instance we also do not care about query * params or fragments as it will be the same view regardless */ var locationWithoutParams = location.path().split('?')[0]; var locationWithoutFragment = locationWithoutParams.split('#')[0]; if (view !== activeRoute && view.url !== locationWithoutFragment) { var element = view.element; element.setAttribute('aria-hidden', 'true'); element.classList.add('ion-page-hidden'); view.ref.changeDetectorRef.detach(); } }); }; var ɵ1 = cleanup; export { ɵ0, ɵ1 }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stack-controller.js","sourceRoot":"ng://@ionic/angular/","sources":["directives/navigation/stack-controller.ts"],"names":[],"mappings":"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","sourcesContent":["import { Location } from '@angular/common';\nimport { ComponentRef, NgZone } from '@angular/core';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { AnimationBuilder, RouterDirection } from '@ionic/core';\n\nimport { bindLifecycleEvents } from '../../providers/angular-delegate';\nimport { NavController } from '../../providers/nav-controller';\n\nimport { RouteView, StackEvent, computeStackId, destroyView, getUrl, insertView, isTabSwitch, toSegments } from './stack-utils';\n\nexport class StackController {\n\n  private views: RouteView[] = [];\n  private runningTask?: Promise<any>;\n  private skipTransition = false;\n  private tabsPrefix: string[] | undefined;\n  private activeView: RouteView | undefined;\n  private nextId = 0;\n\n  constructor(\n    tabsPrefix: string | undefined,\n    private containerEl: HTMLIonRouterOutletElement,\n    private router: Router,\n    private navCtrl: NavController,\n    private zone: NgZone,\n    private location: Location\n  ) {\n    this.tabsPrefix = tabsPrefix !== undefined ? toSegments(tabsPrefix) : undefined;\n  }\n\n  createView(ref: ComponentRef<any>, activatedRoute: ActivatedRoute): RouteView {\n    const url = getUrl(this.router, activatedRoute);\n    const element = (ref && ref.location && ref.location.nativeElement) as HTMLElement;\n    const unlistenEvents = bindLifecycleEvents(this.zone, ref.instance, element);\n    return {\n      id: this.nextId++,\n      stackId: computeStackId(this.tabsPrefix, url),\n      unlistenEvents,\n      element,\n      ref,\n      url,\n    };\n  }\n\n  getExistingView(activatedRoute: ActivatedRoute): RouteView | undefined {\n    const activatedUrlKey = getUrl(this.router, activatedRoute);\n    const view = this.views.find(vw => vw.url === activatedUrlKey);\n    if (view) {\n      view.ref.changeDetectorRef.reattach();\n    }\n    return view;\n  }\n\n  setActive(enteringView: RouteView): Promise<StackEvent> {\n    const consumeResult = this.navCtrl.consumeTransition();\n    let { direction, animation, animationBuilder } = consumeResult;\n    const leavingView = this.activeView;\n    const tabSwitch = isTabSwitch(enteringView, leavingView);\n    if (tabSwitch) {\n      direction = 'back';\n      animation = undefined;\n    }\n\n    const viewsSnapshot = this.views.slice();\n\n    let currentNavigation;\n\n    const router = (this.router as any);\n\n    // Angular >= 7.2.0\n    if (router.getCurrentNavigation) {\n      currentNavigation = router.getCurrentNavigation();\n\n      // Angular < 7.2.0\n    } else if (\n      router.navigations &&\n      router.navigations.value\n    ) {\n      currentNavigation = router.navigations.value;\n    }\n\n    /**\n     * If the navigation action\n     * sets `replaceUrl: true`\n     * then we need to make sure\n     * we remove the last item\n     * from our views stack\n     */\n    if (\n      currentNavigation &&\n      currentNavigation.extras &&\n      currentNavigation.extras.replaceUrl\n    ) {\n      if (this.views.length > 0) {\n        this.views.splice(-1, 1);\n      }\n    }\n\n    const reused = this.views.includes(enteringView);\n    const views = this.insertView(enteringView, direction);\n\n    // Trigger change detection before transition starts\n    // This will call ngOnInit() the first time too, just after the view\n    // was attached to the dom, but BEFORE the transition starts\n    if (!reused) {\n      enteringView.ref.changeDetectorRef.detectChanges();\n    }\n\n    /**\n     * If we are going back from a page that\n     * was presented using a custom animation\n     * we should default to using that\n     * unless the developer explicitly\n     * provided another animation.\n     */\n    const customAnimation = enteringView.animationBuilder;\n    if (\n      animationBuilder === undefined &&\n      direction === 'back' &&\n      !tabSwitch &&\n      customAnimation !== undefined\n    ) {\n      animationBuilder = customAnimation;\n    }\n\n    /**\n     * Save any custom animation so that navigating\n     * back will use this custom animation by default.\n     */\n    if (leavingView) {\n      leavingView.animationBuilder = animationBuilder;\n    }\n\n    // Wait until previous transitions finish\n    return this.zone.runOutsideAngular(() => {\n      return this.wait(() => {\n        // disconnect leaving page from change detection to\n        // reduce jank during the page transition\n        if (leavingView) {\n          leavingView.ref.changeDetectorRef.detach();\n        }\n        // In case the enteringView is the same as the leavingPage we need to reattach()\n        enteringView.ref.changeDetectorRef.reattach();\n\n        return this.transition(enteringView, leavingView, animation, this.canGoBack(1), false, animationBuilder)\n          .then(() => cleanupAsync(enteringView, views, viewsSnapshot, this.location))\n          .then(() => ({\n            enteringView,\n            direction,\n            animation,\n            tabSwitch\n          }));\n      });\n    });\n  }\n\n  canGoBack(deep: number, stackId = this.getActiveStackId()): boolean {\n    return this.getStack(stackId).length > deep;\n  }\n\n  pop(deep: number, stackId = this.getActiveStackId()) {\n    return this.zone.run(() => {\n      const views = this.getStack(stackId);\n      if (views.length <= deep) {\n        return Promise.resolve(false);\n      }\n      const view = views[views.length - deep - 1];\n      let url = view.url;\n\n      const viewSavedData = view.savedData;\n      if (viewSavedData) {\n        const primaryOutlet = viewSavedData.get('primary');\n        if (\n          primaryOutlet &&\n          primaryOutlet.route &&\n          primaryOutlet.route._routerState &&\n          primaryOutlet.route._routerState.snapshot &&\n          primaryOutlet.route._routerState.snapshot.url\n        ) {\n          url = primaryOutlet.route._routerState.snapshot.url;\n        }\n      }\n      const { animationBuilder } = this.navCtrl.consumeTransition();\n      return this.navCtrl.navigateBack(url, { ...view.savedExtras, animation: animationBuilder }).then(() => true);\n    });\n  }\n\n  startBackTransition() {\n    const leavingView = this.activeView;\n    if (leavingView) {\n      const views = this.getStack(leavingView.stackId);\n      const enteringView = views[views.length - 2];\n      const customAnimation = enteringView.animationBuilder;\n\n      return this.wait(() => {\n        return this.transition(\n          enteringView, // entering view\n          leavingView, // leaving view\n          'back',\n          this.canGoBack(2),\n          true,\n          customAnimation\n        );\n      });\n    }\n    return Promise.resolve();\n  }\n\n  endBackTransition(shouldComplete: boolean) {\n    if (shouldComplete) {\n      this.skipTransition = true;\n      this.pop(1);\n    } else if (this.activeView) {\n      cleanup(this.activeView, this.views, this.views, this.location);\n    }\n  }\n\n  getLastUrl(stackId?: string) {\n    const views = this.getStack(stackId);\n    return views.length > 0 ? views[views.length - 1] : undefined;\n  }\n\n  /**\n   * @internal\n   */\n  getRootUrl(stackId?: string) {\n    const views = this.getStack(stackId);\n    return views.length > 0 ? views[0] : undefined;\n  }\n\n  getActiveStackId(): string | undefined {\n    return this.activeView ? this.activeView.stackId : undefined;\n  }\n\n  hasRunningTask(): boolean {\n    return this.runningTask !== undefined;\n  }\n\n  destroy() {\n    this.containerEl = undefined!;\n    this.views.forEach(destroyView);\n    this.activeView = undefined;\n    this.views = [];\n  }\n\n  private getStack(stackId: string | undefined) {\n    return this.views.filter(v => v.stackId === stackId);\n  }\n\n  private insertView(enteringView: RouteView, direction: RouterDirection) {\n    this.activeView = enteringView;\n    this.views = insertView(this.views, enteringView, direction);\n    return this.views.slice();\n  }\n\n  private transition(\n    enteringView: RouteView | undefined,\n    leavingView: RouteView | undefined,\n    direction: 'forward' | 'back' | undefined,\n    showGoBack: boolean,\n    progressAnimation: boolean,\n    animationBuilder?: AnimationBuilder\n  ) {\n    if (this.skipTransition) {\n      this.skipTransition = false;\n      return Promise.resolve(false);\n    }\n    if (leavingView === enteringView) {\n      return Promise.resolve(false);\n    }\n    const enteringEl = enteringView ? enteringView.element : undefined;\n    const leavingEl = leavingView ? leavingView.element : undefined;\n    const containerEl = this.containerEl;\n    if (enteringEl && enteringEl !== leavingEl) {\n      enteringEl.classList.add('ion-page');\n      enteringEl.classList.add('ion-page-invisible');\n      if (enteringEl.parentElement !== containerEl) {\n        containerEl.appendChild(enteringEl);\n      }\n\n      if ((containerEl as any).commit) {\n        return containerEl.commit(enteringEl, leavingEl, {\n          deepWait: true,\n          duration: direction === undefined ? 0 : undefined,\n          direction,\n          showGoBack,\n          progressAnimation,\n          animationBuilder\n        });\n      }\n    }\n    return Promise.resolve(false);\n  }\n\n  private async wait<T>(task: () => Promise<T>): Promise<T> {\n    if (this.runningTask !== undefined) {\n      await this.runningTask;\n      this.runningTask = undefined;\n    }\n    const promise = this.runningTask = task();\n    promise.finally(() => this.runningTask = undefined);\n    return promise;\n  }\n}\n\nconst cleanupAsync = (activeRoute: RouteView, views: RouteView[], viewsSnapshot: RouteView[], location: Location) => {\n  if (typeof (requestAnimationFrame as any) === 'function') {\n    return new Promise<any>(resolve => {\n      requestAnimationFrame(() => {\n        cleanup(activeRoute, views, viewsSnapshot, location);\n        resolve();\n      });\n    });\n  }\n  return Promise.resolve();\n};\n\nconst cleanup = (activeRoute: RouteView, views: RouteView[], viewsSnapshot: RouteView[], location: Location) => {\n  viewsSnapshot\n    .filter(view => !views.includes(view))\n    .forEach(destroyView);\n\n  views.forEach(view => {\n    /**\n     * In the event that a user navigated multiple\n     * times in rapid succession, we want to make sure\n     * we don't pre-emptively detach a view while\n     * it is in mid-transition.\n     *\n     * In this instance we also do not care about query\n     * params or fragments as it will be the same view regardless\n     */\n    const locationWithoutParams = location.path().split('?')[0];\n    const locationWithoutFragment = locationWithoutParams.split('#')[0];\n\n    if (view !== activeRoute && view.url !== locationWithoutFragment) {\n      const element = view.element;\n      element.setAttribute('aria-hidden', 'true');\n      element.classList.add('ion-page-hidden');\n      view.ref.changeDetectorRef.detach();\n    }\n  });\n};\n"]}