router-outlet.mjs 65 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480
  1. import { ViewContainerRef, inject, Attribute, Directive, EventEmitter, Optional, Output, SkipSelf, EnvironmentInjector, Input, InjectionToken, Injectable, reflectComponentType, } from '@angular/core';
  2. import { Router, ActivatedRoute, ChildrenOutletContexts, PRIMARY_OUTLET } from '@angular/router';
  3. import { componentOnReady } from '@ionic/core/components';
  4. import { BehaviorSubject, combineLatest, of } from 'rxjs';
  5. import { distinctUntilChanged, filter, switchMap } from 'rxjs/operators';
  6. import { Config } from '../../providers/config';
  7. import { NavController } from '../../providers/nav-controller';
  8. import { StackController } from './stack-controller';
  9. import { getUrl, isTabSwitch } from './stack-utils';
  10. import * as i0 from "@angular/core";
  11. import * as i1 from "@angular/common";
  12. import * as i2 from "@angular/router";
  13. // TODO(FW-2827): types
  14. // eslint-disable-next-line @angular-eslint/directive-class-suffix
  15. export class IonRouterOutlet {
  16. parentOutlet;
  17. nativeEl;
  18. activatedView = null;
  19. tabsPrefix;
  20. _swipeGesture;
  21. stackCtrl;
  22. // Maintain map of activated route proxies for each component instance
  23. proxyMap = new WeakMap();
  24. // Keep the latest activated route in a subject for the proxy routes to switch map to
  25. currentActivatedRoute$ = new BehaviorSubject(null);
  26. activated = null;
  27. /** @internal */
  28. get activatedComponentRef() {
  29. return this.activated;
  30. }
  31. _activatedRoute = null;
  32. /**
  33. * The name of the outlet
  34. */
  35. name = PRIMARY_OUTLET;
  36. /** @internal */
  37. stackWillChange = new EventEmitter();
  38. /** @internal */
  39. stackDidChange = new EventEmitter();
  40. // eslint-disable-next-line @angular-eslint/no-output-rename
  41. activateEvents = new EventEmitter();
  42. // eslint-disable-next-line @angular-eslint/no-output-rename
  43. deactivateEvents = new EventEmitter();
  44. parentContexts = inject(ChildrenOutletContexts);
  45. location = inject(ViewContainerRef);
  46. environmentInjector = inject(EnvironmentInjector);
  47. inputBinder = inject(INPUT_BINDER, { optional: true });
  48. /** @nodoc */
  49. supportsBindingToComponentInputs = true;
  50. // Ionic providers
  51. config = inject(Config);
  52. navCtrl = inject(NavController);
  53. set animation(animation) {
  54. this.nativeEl.animation = animation;
  55. }
  56. set animated(animated) {
  57. this.nativeEl.animated = animated;
  58. }
  59. set swipeGesture(swipe) {
  60. this._swipeGesture = swipe;
  61. this.nativeEl.swipeHandler = swipe
  62. ? {
  63. canStart: () => this.stackCtrl.canGoBack(1) && !this.stackCtrl.hasRunningTask(),
  64. onStart: () => this.stackCtrl.startBackTransition(),
  65. onEnd: (shouldContinue) => this.stackCtrl.endBackTransition(shouldContinue),
  66. }
  67. : undefined;
  68. }
  69. constructor(name, tabs, commonLocation, elementRef, router, zone, activatedRoute, parentOutlet) {
  70. this.parentOutlet = parentOutlet;
  71. this.nativeEl = elementRef.nativeElement;
  72. this.name = name || PRIMARY_OUTLET;
  73. this.tabsPrefix = tabs === 'true' ? getUrl(router, activatedRoute) : undefined;
  74. this.stackCtrl = new StackController(this.tabsPrefix, this.nativeEl, router, this.navCtrl, zone, commonLocation);
  75. this.parentContexts.onChildOutletCreated(this.name, this);
  76. }
  77. ngOnDestroy() {
  78. this.stackCtrl.destroy();
  79. this.inputBinder?.unsubscribeFromRouteData(this);
  80. }
  81. getContext() {
  82. return this.parentContexts.getContext(this.name);
  83. }
  84. ngOnInit() {
  85. this.initializeOutletWithName();
  86. }
  87. // Note: Ionic deviates from the Angular Router implementation here
  88. initializeOutletWithName() {
  89. if (!this.activated) {
  90. // If the outlet was not instantiated at the time the route got activated we need to populate
  91. // the outlet when it is initialized (ie inside a NgIf)
  92. const context = this.getContext();
  93. if (context?.route) {
  94. this.activateWith(context.route, context.injector);
  95. }
  96. }
  97. new Promise((resolve) => componentOnReady(this.nativeEl, resolve)).then(() => {
  98. if (this._swipeGesture === undefined) {
  99. this.swipeGesture = this.config.getBoolean('swipeBackEnabled', this.nativeEl.mode === 'ios');
  100. }
  101. });
  102. }
  103. get isActivated() {
  104. return !!this.activated;
  105. }
  106. get component() {
  107. if (!this.activated) {
  108. throw new Error('Outlet is not activated');
  109. }
  110. return this.activated.instance;
  111. }
  112. get activatedRoute() {
  113. if (!this.activated) {
  114. throw new Error('Outlet is not activated');
  115. }
  116. return this._activatedRoute;
  117. }
  118. get activatedRouteData() {
  119. if (this._activatedRoute) {
  120. return this._activatedRoute.snapshot.data;
  121. }
  122. return {};
  123. }
  124. /**
  125. * Called when the `RouteReuseStrategy` instructs to detach the subtree
  126. */
  127. detach() {
  128. throw new Error('incompatible reuse strategy');
  129. }
  130. /**
  131. * Called when the `RouteReuseStrategy` instructs to re-attach a previously detached subtree
  132. */
  133. // eslint-disable-next-line @typescript-eslint/no-unused-vars
  134. attach(_ref, _activatedRoute) {
  135. throw new Error('incompatible reuse strategy');
  136. }
  137. deactivate() {
  138. if (this.activated) {
  139. if (this.activatedView) {
  140. // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
  141. const context = this.getContext();
  142. this.activatedView.savedData = new Map(context.children['contexts']);
  143. /**
  144. * Angular v11.2.10 introduced a change
  145. * where this route context is cleared out when
  146. * a router-outlet is deactivated, However,
  147. * we need this route information in order to
  148. * return a user back to the correct tab when
  149. * leaving and then going back to the tab context.
  150. */
  151. const primaryOutlet = this.activatedView.savedData.get('primary');
  152. if (primaryOutlet && context.route) {
  153. primaryOutlet.route = { ...context.route };
  154. }
  155. /**
  156. * Ensure we are saving the NavigationExtras
  157. * data otherwise it will be lost
  158. */
  159. this.activatedView.savedExtras = {};
  160. if (context.route) {
  161. const contextSnapshot = context.route.snapshot;
  162. this.activatedView.savedExtras.queryParams = contextSnapshot.queryParams;
  163. this.activatedView.savedExtras.fragment = contextSnapshot.fragment;
  164. }
  165. }
  166. const c = this.component;
  167. this.activatedView = null;
  168. this.activated = null;
  169. this._activatedRoute = null;
  170. this.deactivateEvents.emit(c);
  171. }
  172. }
  173. activateWith(activatedRoute, environmentInjector) {
  174. if (this.isActivated) {
  175. throw new Error('Cannot activate an already activated outlet');
  176. }
  177. this._activatedRoute = activatedRoute;
  178. let cmpRef;
  179. let enteringView = this.stackCtrl.getExistingView(activatedRoute);
  180. if (enteringView) {
  181. cmpRef = this.activated = enteringView.ref;
  182. const saved = enteringView.savedData;
  183. if (saved) {
  184. // self-restore
  185. // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
  186. const context = this.getContext();
  187. context.children['contexts'] = saved;
  188. }
  189. // Updated activated route proxy for this component
  190. this.updateActivatedRouteProxy(cmpRef.instance, activatedRoute);
  191. }
  192. else {
  193. const snapshot = activatedRoute._futureSnapshot;
  194. /**
  195. * Angular 14 introduces a new `loadComponent` property to the route config.
  196. * This function will assign a `component` property to the route snapshot.
  197. * We check for the presence of this property to determine if the route is
  198. * using standalone components.
  199. */
  200. const childContexts = this.parentContexts.getOrCreateContext(this.name).children;
  201. // We create an activated route proxy object that will maintain future updates for this component
  202. // over its lifecycle in the stack.
  203. const component$ = new BehaviorSubject(null);
  204. const activatedRouteProxy = this.createActivatedRouteProxy(component$, activatedRoute);
  205. const injector = new OutletInjector(activatedRouteProxy, childContexts, this.location.injector);
  206. // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
  207. const component = snapshot.routeConfig.component ?? snapshot.component;
  208. /**
  209. * View components need to be added as a child of ion-router-outlet
  210. * for page transitions and swipe to go back.
  211. * However, createComponent mounts components as siblings of the
  212. * ViewContainerRef. As a result, outletContent must reference
  213. * an ng-container inside of ion-router-outlet and not
  214. * ion-router-outlet itself.
  215. */
  216. cmpRef = this.activated = this.outletContent.createComponent(component, {
  217. index: this.outletContent.length,
  218. injector,
  219. environmentInjector: environmentInjector ?? this.environmentInjector,
  220. });
  221. // Once the component is created we can push it to our local subject supplied to the proxy
  222. component$.next(cmpRef.instance);
  223. // Calling `markForCheck` to make sure we will run the change detection when the
  224. // `RouterOutlet` is inside a `ChangeDetectionStrategy.OnPush` component.
  225. /**
  226. * At this point this.activated has been set earlier
  227. * in this function, so it is guaranteed to be non-null.
  228. */
  229. // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
  230. enteringView = this.stackCtrl.createView(this.activated, activatedRoute);
  231. // Store references to the proxy by component
  232. this.proxyMap.set(cmpRef.instance, activatedRouteProxy);
  233. this.currentActivatedRoute$.next({ component: cmpRef.instance, activatedRoute });
  234. }
  235. this.inputBinder?.bindActivatedRouteToOutletComponent(this);
  236. this.activatedView = enteringView;
  237. /**
  238. * The top outlet is set prior to the entering view's transition completing,
  239. * so that when we have nested outlets (e.g. ion-tabs inside an ion-router-outlet),
  240. * the tabs outlet will be assigned as the top outlet when a view inside tabs is
  241. * activated.
  242. *
  243. * In this scenario, activeWith is called for both the tabs and the root router outlet.
  244. * To avoid a race condition, we assign the top outlet synchronously.
  245. */
  246. this.navCtrl.setTopOutlet(this);
  247. const leavingView = this.stackCtrl.getActiveView();
  248. this.stackWillChange.emit({
  249. enteringView,
  250. tabSwitch: isTabSwitch(enteringView, leavingView),
  251. });
  252. this.stackCtrl.setActive(enteringView).then((data) => {
  253. this.activateEvents.emit(cmpRef.instance);
  254. this.stackDidChange.emit(data);
  255. });
  256. }
  257. /**
  258. * Returns `true` if there are pages in the stack to go back.
  259. */
  260. canGoBack(deep = 1, stackId) {
  261. return this.stackCtrl.canGoBack(deep, stackId);
  262. }
  263. /**
  264. * Resolves to `true` if it the outlet was able to sucessfully pop the last N pages.
  265. */
  266. pop(deep = 1, stackId) {
  267. return this.stackCtrl.pop(deep, stackId);
  268. }
  269. /**
  270. * Returns the URL of the active page of each stack.
  271. */
  272. getLastUrl(stackId) {
  273. const active = this.stackCtrl.getLastUrl(stackId);
  274. return active ? active.url : undefined;
  275. }
  276. /**
  277. * Returns the RouteView of the active page of each stack.
  278. * @internal
  279. */
  280. getLastRouteView(stackId) {
  281. return this.stackCtrl.getLastUrl(stackId);
  282. }
  283. /**
  284. * Returns the root view in the tab stack.
  285. * @internal
  286. */
  287. getRootView(stackId) {
  288. return this.stackCtrl.getRootUrl(stackId);
  289. }
  290. /**
  291. * Returns the active stack ID. In the context of ion-tabs, it means the active tab.
  292. */
  293. getActiveStackId() {
  294. return this.stackCtrl.getActiveStackId();
  295. }
  296. /**
  297. * Since the activated route can change over the life time of a component in an ion router outlet, we create
  298. * a proxy so that we can update the values over time as a user navigates back to components already in the stack.
  299. */
  300. createActivatedRouteProxy(component$, activatedRoute) {
  301. const proxy = new ActivatedRoute();
  302. proxy._futureSnapshot = activatedRoute._futureSnapshot;
  303. proxy._routerState = activatedRoute._routerState;
  304. proxy.snapshot = activatedRoute.snapshot;
  305. proxy.outlet = activatedRoute.outlet;
  306. proxy.component = activatedRoute.component;
  307. // Setup wrappers for the observables so consumers don't have to worry about switching to new observables as the state updates
  308. proxy._paramMap = this.proxyObservable(component$, 'paramMap');
  309. proxy._queryParamMap = this.proxyObservable(component$, 'queryParamMap');
  310. proxy.url = this.proxyObservable(component$, 'url');
  311. proxy.params = this.proxyObservable(component$, 'params');
  312. proxy.queryParams = this.proxyObservable(component$, 'queryParams');
  313. proxy.fragment = this.proxyObservable(component$, 'fragment');
  314. proxy.data = this.proxyObservable(component$, 'data');
  315. return proxy;
  316. }
  317. /**
  318. * Create a wrapped observable that will switch to the latest activated route matched by the given component
  319. */
  320. proxyObservable(component$, path) {
  321. return component$.pipe(
  322. // First wait until the component instance is pushed
  323. filter((component) => !!component), switchMap((component) => this.currentActivatedRoute$.pipe(filter((current) => current !== null && current.component === component), switchMap((current) => current && current.activatedRoute[path]), distinctUntilChanged())));
  324. }
  325. /**
  326. * Updates the activated route proxy for the given component to the new incoming router state
  327. */
  328. updateActivatedRouteProxy(component, activatedRoute) {
  329. const proxy = this.proxyMap.get(component);
  330. if (!proxy) {
  331. throw new Error(`Could not find activated route proxy for view`);
  332. }
  333. proxy._futureSnapshot = activatedRoute._futureSnapshot;
  334. proxy._routerState = activatedRoute._routerState;
  335. proxy.snapshot = activatedRoute.snapshot;
  336. proxy.outlet = activatedRoute.outlet;
  337. proxy.component = activatedRoute.component;
  338. this.currentActivatedRoute$.next({ component, activatedRoute });
  339. }
  340. /** @nocollapse */ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: IonRouterOutlet, deps: [{ token: 'name', attribute: true }, { token: 'tabs', attribute: true, optional: true }, { token: i1.Location }, { token: i0.ElementRef }, { token: i2.Router }, { token: i0.NgZone }, { token: i2.ActivatedRoute }, { token: IonRouterOutlet, optional: true, skipSelf: true }], target: i0.ɵɵFactoryTarget.Directive });
  341. /** @nocollapse */ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: IonRouterOutlet, selector: "ion-router-outlet", inputs: { animated: "animated", animation: "animation", mode: "mode", swipeGesture: "swipeGesture", name: "name" }, outputs: { stackWillChange: "stackWillChange", stackDidChange: "stackDidChange", activateEvents: "activate", deactivateEvents: "deactivate" }, exportAs: ["outlet"], ngImport: i0 });
  342. }
  343. i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: IonRouterOutlet, decorators: [{
  344. type: Directive,
  345. args: [{
  346. selector: 'ion-router-outlet',
  347. exportAs: 'outlet',
  348. // eslint-disable-next-line @angular-eslint/no-inputs-metadata-property
  349. inputs: ['animated', 'animation', 'mode', 'swipeGesture'],
  350. }]
  351. }], ctorParameters: function () { return [{ type: undefined, decorators: [{
  352. type: Attribute,
  353. args: ['name']
  354. }] }, { type: undefined, decorators: [{
  355. type: Optional
  356. }, {
  357. type: Attribute,
  358. args: ['tabs']
  359. }] }, { type: i1.Location }, { type: i0.ElementRef }, { type: i2.Router }, { type: i0.NgZone }, { type: i2.ActivatedRoute }, { type: IonRouterOutlet, decorators: [{
  360. type: SkipSelf
  361. }, {
  362. type: Optional
  363. }] }]; }, propDecorators: { name: [{
  364. type: Input
  365. }], stackWillChange: [{
  366. type: Output
  367. }], stackDidChange: [{
  368. type: Output
  369. }], activateEvents: [{
  370. type: Output,
  371. args: ['activate']
  372. }], deactivateEvents: [{
  373. type: Output,
  374. args: ['deactivate']
  375. }] } });
  376. class OutletInjector {
  377. route;
  378. childContexts;
  379. parent;
  380. constructor(route, childContexts, parent) {
  381. this.route = route;
  382. this.childContexts = childContexts;
  383. this.parent = parent;
  384. }
  385. get(token, notFoundValue) {
  386. if (token === ActivatedRoute) {
  387. return this.route;
  388. }
  389. if (token === ChildrenOutletContexts) {
  390. return this.childContexts;
  391. }
  392. return this.parent.get(token, notFoundValue);
  393. }
  394. }
  395. // TODO: FW-4785 - Remove this once Angular 15 support is dropped
  396. const INPUT_BINDER = new InjectionToken('');
  397. /**
  398. * Injectable used as a tree-shakable provider for opting in to binding router data to component
  399. * inputs.
  400. *
  401. * The RouterOutlet registers itself with this service when an `ActivatedRoute` is attached or
  402. * activated. When this happens, the service subscribes to the `ActivatedRoute` observables (params,
  403. * queryParams, data) and sets the inputs of the component using `ComponentRef.setInput`.
  404. * Importantly, when an input does not have an item in the route data with a matching key, this
  405. * input is set to `undefined`. If it were not done this way, the previous information would be
  406. * retained if the data got removed from the route (i.e. if a query parameter is removed).
  407. *
  408. * The `RouterOutlet` should unregister itself when destroyed via `unsubscribeFromRouteData` so that
  409. * the subscriptions are cleaned up.
  410. */
  411. class RoutedComponentInputBinder {
  412. outletDataSubscriptions = new Map();
  413. bindActivatedRouteToOutletComponent(outlet) {
  414. this.unsubscribeFromRouteData(outlet);
  415. this.subscribeToRouteData(outlet);
  416. }
  417. unsubscribeFromRouteData(outlet) {
  418. this.outletDataSubscriptions.get(outlet)?.unsubscribe();
  419. this.outletDataSubscriptions.delete(outlet);
  420. }
  421. subscribeToRouteData(outlet) {
  422. const { activatedRoute } = outlet;
  423. const dataSubscription = combineLatest([activatedRoute.queryParams, activatedRoute.params, activatedRoute.data])
  424. .pipe(switchMap(([queryParams, params, data], index) => {
  425. data = { ...queryParams, ...params, ...data };
  426. // Get the first result from the data subscription synchronously so it's available to
  427. // the component as soon as possible (and doesn't require a second change detection).
  428. if (index === 0) {
  429. return of(data);
  430. }
  431. // Promise.resolve is used to avoid synchronously writing the wrong data when
  432. // two of the Observables in the `combineLatest` stream emit one after
  433. // another.
  434. return Promise.resolve(data);
  435. }))
  436. .subscribe((data) => {
  437. // Outlet may have been deactivated or changed names to be associated with a different
  438. // route
  439. if (!outlet.isActivated ||
  440. !outlet.activatedComponentRef ||
  441. outlet.activatedRoute !== activatedRoute ||
  442. activatedRoute.component === null) {
  443. this.unsubscribeFromRouteData(outlet);
  444. return;
  445. }
  446. const mirror = reflectComponentType(activatedRoute.component);
  447. if (!mirror) {
  448. this.unsubscribeFromRouteData(outlet);
  449. return;
  450. }
  451. for (const { templateName } of mirror.inputs) {
  452. outlet.activatedComponentRef.setInput(templateName, data[templateName]);
  453. }
  454. });
  455. this.outletDataSubscriptions.set(outlet, dataSubscription);
  456. }
  457. /** @nocollapse */ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: RoutedComponentInputBinder, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
  458. /** @nocollapse */ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: RoutedComponentInputBinder });
  459. }
  460. i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: RoutedComponentInputBinder, decorators: [{
  461. type: Injectable
  462. }] });
  463. export const provideComponentInputBinding = () => {
  464. return {
  465. provide: INPUT_BINDER,
  466. useFactory: componentInputBindingFactory,
  467. deps: [Router],
  468. };
  469. };
  470. function componentInputBindingFactory(router) {
  471. /**
  472. * We cast the router to any here, since the componentInputBindingEnabled
  473. * property is not available until Angular v16.
  474. */
  475. if (router?.componentInputBindingEnabled) {
  476. return new RoutedComponentInputBinder();
  477. }
  478. return null;
  479. }
  480. //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"router-outlet.js","sourceRoot":"","sources":["../../../../../common/src/directives/navigation/router-outlet.ts"],"names":[],"mappings":"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","sourcesContent":["import { Location } from '@angular/common';\nimport {\n  ComponentRef,\n  ElementRef,\n  Injector,\n  NgZone,\n  OnDestroy,\n  OnInit,\n  ViewContainerRef,\n  inject,\n  Attribute,\n  Directive,\n  EventEmitter,\n  Optional,\n  Output,\n  SkipSelf,\n  EnvironmentInjector,\n  Input,\n  InjectionToken,\n  Injectable,\n  reflectComponentType,\n} from '@angular/core';\nimport type { Provider } from '@angular/core';\nimport { OutletContext, Router, ActivatedRoute, ChildrenOutletContexts, PRIMARY_OUTLET, Data } from '@angular/router';\nimport { componentOnReady } from '@ionic/core/components';\nimport type { AnimationBuilder } from '@ionic/core/components';\nimport { Observable, BehaviorSubject, Subscription, combineLatest, of } from 'rxjs';\nimport { distinctUntilChanged, filter, switchMap } from 'rxjs/operators';\n\nimport { Config } from '../../providers/config';\nimport { NavController } from '../../providers/nav-controller';\n\nimport { StackController } from './stack-controller';\nimport { RouteView, StackDidChangeEvent, StackWillChangeEvent, getUrl, isTabSwitch } from './stack-utils';\n\n// TODO(FW-2827): types\n\n@Directive({\n  selector: 'ion-router-outlet',\n  exportAs: 'outlet',\n  // eslint-disable-next-line @angular-eslint/no-inputs-metadata-property\n  inputs: ['animated', 'animation', 'mode', 'swipeGesture'],\n})\n// eslint-disable-next-line @angular-eslint/directive-class-suffix\nexport abstract class IonRouterOutlet implements OnDestroy, OnInit {\n  abstract outletContent: any;\n\n  nativeEl: HTMLIonRouterOutletElement;\n  activatedView: RouteView | null = null;\n  tabsPrefix: string | undefined;\n\n  private _swipeGesture?: boolean;\n  private stackCtrl: StackController;\n\n  // Maintain map of activated route proxies for each component instance\n  private proxyMap = new WeakMap<any, ActivatedRoute>();\n  // Keep the latest activated route in a subject for the proxy routes to switch map to\n  private currentActivatedRoute$ = new BehaviorSubject<{ component: any; activatedRoute: ActivatedRoute } | null>(null);\n\n  private activated: ComponentRef<any> | null = null;\n  /** @internal */\n  get activatedComponentRef(): ComponentRef<any> | null {\n    return this.activated;\n  }\n  private _activatedRoute: ActivatedRoute | null = null;\n\n  /**\n   * The name of the outlet\n   */\n  @Input() name = PRIMARY_OUTLET;\n\n  /** @internal */\n  @Output() stackWillChange = new EventEmitter<StackWillChangeEvent>();\n  /** @internal */\n  @Output() stackDidChange = new EventEmitter<StackDidChangeEvent>();\n\n  // eslint-disable-next-line @angular-eslint/no-output-rename\n  @Output('activate') activateEvents = new EventEmitter<any>();\n  // eslint-disable-next-line @angular-eslint/no-output-rename\n  @Output('deactivate') deactivateEvents = new EventEmitter<any>();\n\n  private parentContexts = inject(ChildrenOutletContexts);\n  private location = inject(ViewContainerRef);\n  private environmentInjector = inject(EnvironmentInjector);\n  private inputBinder = inject(INPUT_BINDER, { optional: true });\n  /** @nodoc */\n  readonly supportsBindingToComponentInputs = true;\n\n  // Ionic providers\n  private config = inject(Config);\n  private navCtrl = inject(NavController);\n\n  set animation(animation: AnimationBuilder) {\n    this.nativeEl.animation = animation;\n  }\n\n  set animated(animated: boolean) {\n    this.nativeEl.animated = animated;\n  }\n\n  set swipeGesture(swipe: boolean) {\n    this._swipeGesture = swipe;\n\n    this.nativeEl.swipeHandler = swipe\n      ? {\n          canStart: () => this.stackCtrl.canGoBack(1) && !this.stackCtrl.hasRunningTask(),\n          onStart: () => this.stackCtrl.startBackTransition(),\n          onEnd: (shouldContinue) => this.stackCtrl.endBackTransition(shouldContinue),\n        }\n      : undefined;\n  }\n\n  constructor(\n    @Attribute('name') name: string,\n    @Optional() @Attribute('tabs') tabs: string,\n    commonLocation: Location,\n    elementRef: ElementRef,\n    router: Router,\n    zone: NgZone,\n    activatedRoute: ActivatedRoute,\n    @SkipSelf() @Optional() readonly parentOutlet?: IonRouterOutlet\n  ) {\n    this.nativeEl = elementRef.nativeElement;\n    this.name = name || PRIMARY_OUTLET;\n    this.tabsPrefix = tabs === 'true' ? getUrl(router, activatedRoute) : undefined;\n    this.stackCtrl = new StackController(this.tabsPrefix, this.nativeEl, router, this.navCtrl, zone, commonLocation);\n    this.parentContexts.onChildOutletCreated(this.name, this as any);\n  }\n\n  ngOnDestroy(): void {\n    this.stackCtrl.destroy();\n    this.inputBinder?.unsubscribeFromRouteData(this);\n  }\n\n  getContext(): OutletContext | null {\n    return this.parentContexts.getContext(this.name);\n  }\n\n  ngOnInit(): void {\n    this.initializeOutletWithName();\n  }\n\n  // Note: Ionic deviates from the Angular Router implementation here\n  private initializeOutletWithName() {\n    if (!this.activated) {\n      // If the outlet was not instantiated at the time the route got activated we need to populate\n      // the outlet when it is initialized (ie inside a NgIf)\n      const context = this.getContext();\n      if (context?.route) {\n        this.activateWith(context.route, context.injector);\n      }\n    }\n\n    new Promise((resolve) => componentOnReady(this.nativeEl, resolve)).then(() => {\n      if (this._swipeGesture === undefined) {\n        this.swipeGesture = this.config.getBoolean('swipeBackEnabled', (this.nativeEl as any).mode === 'ios');\n      }\n    });\n  }\n\n  get isActivated(): boolean {\n    return !!this.activated;\n  }\n\n  get component(): Record<string, unknown> {\n    if (!this.activated) {\n      throw new Error('Outlet is not activated');\n    }\n    return this.activated.instance;\n  }\n\n  get activatedRoute(): ActivatedRoute {\n    if (!this.activated) {\n      throw new Error('Outlet is not activated');\n    }\n    return this._activatedRoute as ActivatedRoute;\n  }\n\n  get activatedRouteData(): Data {\n    if (this._activatedRoute) {\n      return this._activatedRoute.snapshot.data;\n    }\n    return {};\n  }\n\n  /**\n   * Called when the `RouteReuseStrategy` instructs to detach the subtree\n   */\n  detach(): ComponentRef<any> {\n    throw new Error('incompatible reuse strategy');\n  }\n\n  /**\n   * Called when the `RouteReuseStrategy` instructs to re-attach a previously detached subtree\n   */\n  // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  attach(_ref: ComponentRef<any>, _activatedRoute: ActivatedRoute): void {\n    throw new Error('incompatible reuse strategy');\n  }\n\n  deactivate(): void {\n    if (this.activated) {\n      if (this.activatedView) {\n        // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n        const context = this.getContext()!;\n        this.activatedView.savedData = new Map(context.children['contexts']);\n\n        /**\n         * Angular v11.2.10 introduced a change\n         * where this route context is cleared out when\n         * a router-outlet is deactivated, However,\n         * we need this route information in order to\n         * return a user back to the correct tab when\n         * leaving and then going back to the tab context.\n         */\n        const primaryOutlet = this.activatedView.savedData.get('primary');\n        if (primaryOutlet && context.route) {\n          primaryOutlet.route = { ...context.route };\n        }\n\n        /**\n         * Ensure we are saving the NavigationExtras\n         * data otherwise it will be lost\n         */\n        this.activatedView.savedExtras = {};\n        if (context.route) {\n          const contextSnapshot = context.route.snapshot;\n\n          this.activatedView.savedExtras.queryParams = contextSnapshot.queryParams;\n          (this.activatedView.savedExtras.fragment as string | null) = contextSnapshot.fragment;\n        }\n      }\n      const c = this.component;\n      this.activatedView = null;\n      this.activated = null;\n      this._activatedRoute = null;\n      this.deactivateEvents.emit(c);\n    }\n  }\n\n  activateWith(activatedRoute: ActivatedRoute, environmentInjector: EnvironmentInjector | null): void {\n    if (this.isActivated) {\n      throw new Error('Cannot activate an already activated outlet');\n    }\n    this._activatedRoute = activatedRoute;\n\n    let cmpRef: any;\n    let enteringView = this.stackCtrl.getExistingView(activatedRoute);\n    if (enteringView) {\n      cmpRef = this.activated = enteringView.ref;\n      const saved = enteringView.savedData;\n      if (saved) {\n        // self-restore\n        // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n        const context = this.getContext()!;\n        context.children['contexts'] = saved;\n      }\n      // Updated activated route proxy for this component\n      this.updateActivatedRouteProxy(cmpRef.instance, activatedRoute);\n    } else {\n      const snapshot = (activatedRoute as any)._futureSnapshot;\n\n      /**\n       * Angular 14 introduces a new `loadComponent` property to the route config.\n       * This function will assign a `component` property to the route snapshot.\n       * We check for the presence of this property to determine if the route is\n       * using standalone components.\n       */\n      const childContexts = this.parentContexts.getOrCreateContext(this.name).children;\n\n      // We create an activated route proxy object that will maintain future updates for this component\n      // over its lifecycle in the stack.\n      const component$ = new BehaviorSubject<any>(null);\n      const activatedRouteProxy = this.createActivatedRouteProxy(component$, activatedRoute);\n\n      const injector = new OutletInjector(activatedRouteProxy, childContexts, this.location.injector);\n\n      // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n      const component = snapshot.routeConfig!.component ?? snapshot.component;\n\n      /**\n       * View components need to be added as a child of ion-router-outlet\n       * for page transitions and swipe to go back.\n       * However, createComponent mounts components as siblings of the\n       * ViewContainerRef. As a result, outletContent must reference\n       * an ng-container inside of ion-router-outlet and not\n       * ion-router-outlet itself.\n       */\n      cmpRef = this.activated = this.outletContent.createComponent(component, {\n        index: this.outletContent.length,\n        injector,\n        environmentInjector: environmentInjector ?? this.environmentInjector,\n      });\n\n      // Once the component is created we can push it to our local subject supplied to the proxy\n      component$.next(cmpRef.instance);\n\n      // Calling `markForCheck` to make sure we will run the change detection when the\n      // `RouterOutlet` is inside a `ChangeDetectionStrategy.OnPush` component.\n\n      /**\n       * At this point this.activated has been set earlier\n       * in this function, so it is guaranteed to be non-null.\n       */\n      // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n      enteringView = this.stackCtrl.createView(this.activated!, activatedRoute);\n\n      // Store references to the proxy by component\n      this.proxyMap.set(cmpRef.instance, activatedRouteProxy);\n      this.currentActivatedRoute$.next({ component: cmpRef.instance, activatedRoute });\n    }\n\n    this.inputBinder?.bindActivatedRouteToOutletComponent(this);\n\n    this.activatedView = enteringView;\n\n    /**\n     * The top outlet is set prior to the entering view's transition completing,\n     * so that when we have nested outlets (e.g. ion-tabs inside an ion-router-outlet),\n     * the tabs outlet will be assigned as the top outlet when a view inside tabs is\n     * activated.\n     *\n     * In this scenario, activeWith is called for both the tabs and the root router outlet.\n     * To avoid a race condition, we assign the top outlet synchronously.\n     */\n    this.navCtrl.setTopOutlet(this);\n\n    const leavingView = this.stackCtrl.getActiveView();\n\n    this.stackWillChange.emit({\n      enteringView,\n      tabSwitch: isTabSwitch(enteringView, leavingView),\n    });\n\n    this.stackCtrl.setActive(enteringView).then((data) => {\n      this.activateEvents.emit(cmpRef.instance);\n      this.stackDidChange.emit(data);\n    });\n  }\n\n  /**\n   * Returns `true` if there are pages in the stack to go back.\n   */\n  canGoBack(deep = 1, stackId?: string): boolean {\n    return this.stackCtrl.canGoBack(deep, stackId);\n  }\n\n  /**\n   * Resolves to `true` if it the outlet was able to sucessfully pop the last N pages.\n   */\n  pop(deep = 1, stackId?: string): Promise<boolean> {\n    return this.stackCtrl.pop(deep, stackId);\n  }\n\n  /**\n   * Returns the URL of the active page of each stack.\n   */\n  getLastUrl(stackId?: string): string | undefined {\n    const active = this.stackCtrl.getLastUrl(stackId);\n    return active ? active.url : undefined;\n  }\n\n  /**\n   * Returns the RouteView of the active page of each stack.\n   * @internal\n   */\n  getLastRouteView(stackId?: string): RouteView | undefined {\n    return this.stackCtrl.getLastUrl(stackId);\n  }\n\n  /**\n   * Returns the root view in the tab stack.\n   * @internal\n   */\n  getRootView(stackId?: string): RouteView | undefined {\n    return this.stackCtrl.getRootUrl(stackId);\n  }\n\n  /**\n   * Returns the active stack ID. In the context of ion-tabs, it means the active tab.\n   */\n  getActiveStackId(): string | undefined {\n    return this.stackCtrl.getActiveStackId();\n  }\n\n  /**\n   * Since the activated route can change over the life time of a component in an ion router outlet, we create\n   * a proxy so that we can update the values over time as a user navigates back to components already in the stack.\n   */\n  private createActivatedRouteProxy(component$: Observable<any>, activatedRoute: ActivatedRoute): ActivatedRoute {\n    const proxy: any = new ActivatedRoute();\n\n    proxy._futureSnapshot = (activatedRoute as any)._futureSnapshot;\n    proxy._routerState = (activatedRoute as any)._routerState;\n    proxy.snapshot = activatedRoute.snapshot;\n    proxy.outlet = activatedRoute.outlet;\n    proxy.component = activatedRoute.component;\n\n    // Setup wrappers for the observables so consumers don't have to worry about switching to new observables as the state updates\n    (proxy as any)._paramMap = this.proxyObservable(component$, 'paramMap');\n    (proxy as any)._queryParamMap = this.proxyObservable(component$, 'queryParamMap');\n    proxy.url = this.proxyObservable(component$, 'url');\n    proxy.params = this.proxyObservable(component$, 'params');\n    proxy.queryParams = this.proxyObservable(component$, 'queryParams');\n    proxy.fragment = this.proxyObservable(component$, 'fragment');\n    proxy.data = this.proxyObservable(component$, 'data');\n\n    return proxy as ActivatedRoute;\n  }\n\n  /**\n   * Create a wrapped observable that will switch to the latest activated route matched by the given component\n   */\n  private proxyObservable(component$: Observable<any>, path: string): Observable<any> {\n    return component$.pipe(\n      // First wait until the component instance is pushed\n      filter((component) => !!component),\n      switchMap((component) =>\n        this.currentActivatedRoute$.pipe(\n          filter((current) => current !== null && current.component === component),\n          switchMap((current) => current && (current.activatedRoute as any)[path]),\n          distinctUntilChanged()\n        )\n      )\n    );\n  }\n\n  /**\n   * Updates the activated route proxy for the given component to the new incoming router state\n   */\n  private updateActivatedRouteProxy(component: any, activatedRoute: ActivatedRoute): void {\n    const proxy = this.proxyMap.get(component);\n    if (!proxy) {\n      throw new Error(`Could not find activated route proxy for view`);\n    }\n\n    (proxy as any)._futureSnapshot = (activatedRoute as any)._futureSnapshot;\n    (proxy as any)._routerState = (activatedRoute as any)._routerState;\n    proxy.snapshot = activatedRoute.snapshot;\n    proxy.outlet = activatedRoute.outlet;\n    proxy.component = activatedRoute.component;\n\n    this.currentActivatedRoute$.next({ component, activatedRoute });\n  }\n}\n\nclass OutletInjector implements Injector {\n  constructor(private route: ActivatedRoute, private childContexts: ChildrenOutletContexts, private parent: Injector) {}\n\n  get(token: any, notFoundValue?: any): any {\n    if (token === ActivatedRoute) {\n      return this.route;\n    }\n\n    if (token === ChildrenOutletContexts) {\n      return this.childContexts;\n    }\n\n    return this.parent.get(token, notFoundValue);\n  }\n}\n\n// TODO: FW-4785 - Remove this once Angular 15 support is dropped\nconst INPUT_BINDER = new InjectionToken<RoutedComponentInputBinder>('');\n\n/**\n * Injectable used as a tree-shakable provider for opting in to binding router data to component\n * inputs.\n *\n * The RouterOutlet registers itself with this service when an `ActivatedRoute` is attached or\n * activated. When this happens, the service subscribes to the `ActivatedRoute` observables (params,\n * queryParams, data) and sets the inputs of the component using `ComponentRef.setInput`.\n * Importantly, when an input does not have an item in the route data with a matching key, this\n * input is set to `undefined`. If it were not done this way, the previous information would be\n * retained if the data got removed from the route (i.e. if a query parameter is removed).\n *\n * The `RouterOutlet` should unregister itself when destroyed via `unsubscribeFromRouteData` so that\n * the subscriptions are cleaned up.\n */\n@Injectable()\nclass RoutedComponentInputBinder {\n  private outletDataSubscriptions = new Map<IonRouterOutlet, Subscription>();\n\n  bindActivatedRouteToOutletComponent(outlet: IonRouterOutlet): void {\n    this.unsubscribeFromRouteData(outlet);\n    this.subscribeToRouteData(outlet);\n  }\n\n  unsubscribeFromRouteData(outlet: IonRouterOutlet): void {\n    this.outletDataSubscriptions.get(outlet)?.unsubscribe();\n    this.outletDataSubscriptions.delete(outlet);\n  }\n\n  private subscribeToRouteData(outlet: IonRouterOutlet) {\n    const { activatedRoute } = outlet;\n    const dataSubscription = combineLatest([activatedRoute.queryParams, activatedRoute.params, activatedRoute.data])\n      .pipe(\n        switchMap(([queryParams, params, data], index) => {\n          data = { ...queryParams, ...params, ...data };\n          // Get the first result from the data subscription synchronously so it's available to\n          // the component as soon as possible (and doesn't require a second change detection).\n          if (index === 0) {\n            return of(data);\n          }\n          // Promise.resolve is used to avoid synchronously writing the wrong data when\n          // two of the Observables in the `combineLatest` stream emit one after\n          // another.\n          return Promise.resolve(data);\n        })\n      )\n      .subscribe((data) => {\n        // Outlet may have been deactivated or changed names to be associated with a different\n        // route\n        if (\n          !outlet.isActivated ||\n          !outlet.activatedComponentRef ||\n          outlet.activatedRoute !== activatedRoute ||\n          activatedRoute.component === null\n        ) {\n          this.unsubscribeFromRouteData(outlet);\n          return;\n        }\n\n        const mirror = reflectComponentType(activatedRoute.component);\n        if (!mirror) {\n          this.unsubscribeFromRouteData(outlet);\n          return;\n        }\n\n        for (const { templateName } of mirror.inputs) {\n          outlet.activatedComponentRef.setInput(templateName, data[templateName]);\n        }\n      });\n\n    this.outletDataSubscriptions.set(outlet, dataSubscription);\n  }\n}\n\nexport const provideComponentInputBinding = (): Provider => {\n  return {\n    provide: INPUT_BINDER,\n    useFactory: componentInputBindingFactory,\n    deps: [Router],\n  };\n};\n\nfunction componentInputBindingFactory(router?: Router) {\n  /**\n   * We cast the router to any here, since the componentInputBindingEnabled\n   * property is not available until Angular v16.\n   */\n  if ((router as any)?.componentInputBindingEnabled) {\n    return new RoutedComponentInputBinder();\n  }\n  return null;\n}\n"]}