tabs.mjs 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. import { Directive, ElementRef, EventEmitter, HostListener, Output, ViewChild, } from '@angular/core';
  2. import * as i0 from "@angular/core";
  3. import * as i1 from "../../providers/nav-controller";
  4. // eslint-disable-next-line @angular-eslint/directive-class-suffix
  5. export class IonTabs {
  6. navCtrl;
  7. tabsInner;
  8. /**
  9. * Emitted before the tab view is changed.
  10. */
  11. ionTabsWillChange = new EventEmitter();
  12. /**
  13. * Emitted after the tab view is changed.
  14. */
  15. ionTabsDidChange = new EventEmitter();
  16. tabBarSlot = 'bottom';
  17. hasTab = false;
  18. selectedTab;
  19. leavingTab;
  20. constructor(navCtrl) {
  21. this.navCtrl = navCtrl;
  22. }
  23. ngAfterViewInit() {
  24. /**
  25. * Developers must pass at least one ion-tab
  26. * inside of ion-tabs if they want to use a
  27. * basic tab-based navigation without the
  28. * history stack or URL updates associated
  29. * with the router.
  30. */
  31. const firstTab = this.tabs.length > 0 ? this.tabs.first : undefined;
  32. if (firstTab) {
  33. this.hasTab = true;
  34. this.setActiveTab(firstTab.tab);
  35. this.tabSwitch();
  36. }
  37. }
  38. ngAfterContentInit() {
  39. this.detectSlotChanges();
  40. }
  41. ngAfterContentChecked() {
  42. this.detectSlotChanges();
  43. }
  44. /**
  45. * @internal
  46. */
  47. onStackWillChange({ enteringView, tabSwitch }) {
  48. const stackId = enteringView.stackId;
  49. if (tabSwitch && stackId !== undefined) {
  50. this.ionTabsWillChange.emit({ tab: stackId });
  51. }
  52. }
  53. /**
  54. * @internal
  55. */
  56. onStackDidChange({ enteringView, tabSwitch }) {
  57. const stackId = enteringView.stackId;
  58. if (tabSwitch && stackId !== undefined) {
  59. if (this.tabBar) {
  60. this.tabBar.selectedTab = stackId;
  61. }
  62. this.ionTabsDidChange.emit({ tab: stackId });
  63. }
  64. }
  65. /**
  66. * When a tab button is clicked, there are several scenarios:
  67. * 1. If the selected tab is currently active (the tab button has been clicked
  68. * again), then it should go to the root view for that tab.
  69. *
  70. * a. Get the saved root view from the router outlet. If the saved root view
  71. * matches the tabRootUrl, set the route view to this view including the
  72. * navigation extras.
  73. * b. If the saved root view from the router outlet does
  74. * not match, navigate to the tabRootUrl. No navigation extras are
  75. * included.
  76. *
  77. * 2. If the current tab tab is not currently selected, get the last route
  78. * view from the router outlet.
  79. *
  80. * a. If the last route view exists, navigate to that view including any
  81. * navigation extras
  82. * b. If the last route view doesn't exist, then navigate
  83. * to the default tabRootUrl
  84. */
  85. select(tabOrEvent) {
  86. const isTabString = typeof tabOrEvent === 'string';
  87. const tab = isTabString ? tabOrEvent : tabOrEvent.detail.tab;
  88. /**
  89. * If the tabs are not using the router, then
  90. * the tab switch logic is handled by the tabs
  91. * component itself.
  92. */
  93. if (this.hasTab) {
  94. this.setActiveTab(tab);
  95. this.tabSwitch();
  96. return;
  97. }
  98. const alreadySelected = this.outlet.getActiveStackId() === tab;
  99. const tabRootUrl = `${this.outlet.tabsPrefix}/${tab}`;
  100. /**
  101. * If this is a nested tab, prevent the event
  102. * from bubbling otherwise the outer tabs
  103. * will respond to this event too, causing
  104. * the app to get directed to the wrong place.
  105. */
  106. if (!isTabString) {
  107. tabOrEvent.stopPropagation();
  108. }
  109. if (alreadySelected) {
  110. const activeStackId = this.outlet.getActiveStackId();
  111. const activeView = this.outlet.getLastRouteView(activeStackId);
  112. // If on root tab, do not navigate to root tab again
  113. if (activeView?.url === tabRootUrl) {
  114. return;
  115. }
  116. const rootView = this.outlet.getRootView(tab);
  117. const navigationExtras = rootView && tabRootUrl === rootView.url && rootView.savedExtras;
  118. return this.navCtrl.navigateRoot(tabRootUrl, {
  119. ...navigationExtras,
  120. animated: true,
  121. animationDirection: 'back',
  122. });
  123. }
  124. else {
  125. const lastRoute = this.outlet.getLastRouteView(tab);
  126. /**
  127. * If there is a lastRoute, goto that, otherwise goto the fallback url of the
  128. * selected tab
  129. */
  130. const url = lastRoute?.url || tabRootUrl;
  131. const navigationExtras = lastRoute?.savedExtras;
  132. return this.navCtrl.navigateRoot(url, {
  133. ...navigationExtras,
  134. animated: true,
  135. animationDirection: 'back',
  136. });
  137. }
  138. }
  139. setActiveTab(tab) {
  140. const tabs = this.tabs;
  141. const selectedTab = tabs.find((t) => t.tab === tab);
  142. if (!selectedTab) {
  143. console.error(`[Ionic Error]: Tab with id: "${tab}" does not exist`);
  144. return;
  145. }
  146. this.leavingTab = this.selectedTab;
  147. this.selectedTab = selectedTab;
  148. this.ionTabsWillChange.emit({ tab });
  149. selectedTab.el.active = true;
  150. }
  151. tabSwitch() {
  152. const { selectedTab, leavingTab } = this;
  153. if (this.tabBar && selectedTab) {
  154. this.tabBar.selectedTab = selectedTab.tab;
  155. }
  156. if (leavingTab?.tab !== selectedTab?.tab) {
  157. if (leavingTab?.el) {
  158. leavingTab.el.active = false;
  159. }
  160. }
  161. if (selectedTab) {
  162. this.ionTabsDidChange.emit({ tab: selectedTab.tab });
  163. }
  164. }
  165. getSelected() {
  166. if (this.hasTab) {
  167. return this.selectedTab?.tab;
  168. }
  169. return this.outlet.getActiveStackId();
  170. }
  171. /**
  172. * Detects changes to the slot attribute of the tab bar.
  173. *
  174. * If the slot attribute has changed, then the tab bar
  175. * should be relocated to the new slot position.
  176. */
  177. detectSlotChanges() {
  178. this.tabBars.forEach((tabBar) => {
  179. // el is a protected attribute from the generated component wrapper
  180. const currentSlot = tabBar.el.getAttribute('slot');
  181. if (currentSlot !== this.tabBarSlot) {
  182. this.tabBarSlot = currentSlot;
  183. this.relocateTabBar();
  184. }
  185. });
  186. }
  187. /**
  188. * Relocates the tab bar to the new slot position.
  189. */
  190. relocateTabBar() {
  191. /**
  192. * `el` is a protected attribute from the generated component wrapper.
  193. * To avoid having to manually create the wrapper for tab bar, we
  194. * cast the tab bar to any and access the protected attribute.
  195. */
  196. const tabBar = this.tabBar.el;
  197. if (this.tabBarSlot === 'top') {
  198. /**
  199. * A tab bar with a slot of "top" should be inserted
  200. * at the top of the container.
  201. */
  202. this.tabsInner.nativeElement.before(tabBar);
  203. }
  204. else {
  205. /**
  206. * A tab bar with a slot of "bottom" or without a slot
  207. * should be inserted at the end of the container.
  208. */
  209. this.tabsInner.nativeElement.after(tabBar);
  210. }
  211. }
  212. /** @nocollapse */ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: IonTabs, deps: [{ token: i1.NavController }], target: i0.ɵɵFactoryTarget.Directive });
  213. /** @nocollapse */ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.2.12", type: IonTabs, selector: "ion-tabs", outputs: { ionTabsWillChange: "ionTabsWillChange", ionTabsDidChange: "ionTabsDidChange" }, host: { listeners: { "ionTabButtonClick": "select($event)" } }, viewQueries: [{ propertyName: "tabsInner", first: true, predicate: ["tabsInner"], descendants: true, read: ElementRef, static: true }], ngImport: i0 });
  214. }
  215. i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: IonTabs, decorators: [{
  216. type: Directive,
  217. args: [{
  218. selector: 'ion-tabs',
  219. }]
  220. }], ctorParameters: function () { return [{ type: i1.NavController }]; }, propDecorators: { tabsInner: [{
  221. type: ViewChild,
  222. args: ['tabsInner', { read: ElementRef, static: true }]
  223. }], ionTabsWillChange: [{
  224. type: Output
  225. }], ionTabsDidChange: [{
  226. type: Output
  227. }], select: [{
  228. type: HostListener,
  229. args: ['ionTabButtonClick', ['$event']]
  230. }] } });
  231. //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tabs.js","sourceRoot":"","sources":["../../../../../common/src/directives/navigation/tabs.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterContentChecked,\n  AfterContentInit,\n  Directive,\n  ElementRef,\n  EventEmitter,\n  HostListener,\n  Output,\n  ViewChild,\n  AfterViewInit,\n  QueryList,\n} from '@angular/core';\n\nimport { NavController } from '../../providers/nav-controller';\n\nimport { StackDidChangeEvent, StackWillChangeEvent } from './stack-utils';\n\n@Directive({\n  selector: 'ion-tabs',\n})\n// eslint-disable-next-line @angular-eslint/directive-class-suffix\nexport abstract class IonTabs implements AfterViewInit, AfterContentInit, AfterContentChecked {\n  /**\n   * Note: These must be redeclared on each child class since it needs\n   * access to generated components such as IonRouterOutlet and IonTabBar.\n   */\n  abstract outlet: any;\n  abstract tabBar: any;\n  abstract tabBars: QueryList<any>;\n  abstract tabs: QueryList<any>;\n\n  @ViewChild('tabsInner', { read: ElementRef, static: true }) tabsInner: ElementRef<HTMLDivElement>;\n\n  /**\n   * Emitted before the tab view is changed.\n   */\n  @Output() ionTabsWillChange = new EventEmitter<{ tab: string }>();\n  /**\n   * Emitted after the tab view is changed.\n   */\n  @Output() ionTabsDidChange = new EventEmitter<{ tab: string }>();\n\n  private tabBarSlot = 'bottom';\n\n  private hasTab = false;\n  private selectedTab?: { tab: string };\n  private leavingTab?: any;\n\n  constructor(private navCtrl: NavController) {}\n\n  ngAfterViewInit(): void {\n    /**\n     * Developers must pass at least one ion-tab\n     * inside of ion-tabs if they want to use a\n     * basic tab-based navigation without the\n     * history stack or URL updates associated\n     * with the router.\n     */\n    const firstTab = this.tabs.length > 0 ? this.tabs.first : undefined;\n\n    if (firstTab) {\n      this.hasTab = true;\n      this.setActiveTab(firstTab.tab);\n      this.tabSwitch();\n    }\n  }\n\n  ngAfterContentInit(): void {\n    this.detectSlotChanges();\n  }\n\n  ngAfterContentChecked(): void {\n    this.detectSlotChanges();\n  }\n\n  /**\n   * @internal\n   */\n  onStackWillChange({ enteringView, tabSwitch }: StackWillChangeEvent): void {\n    const stackId = enteringView.stackId;\n    if (tabSwitch && stackId !== undefined) {\n      this.ionTabsWillChange.emit({ tab: stackId });\n    }\n  }\n\n  /**\n   * @internal\n   */\n  onStackDidChange({ enteringView, tabSwitch }: StackDidChangeEvent): void {\n    const stackId = enteringView.stackId;\n    if (tabSwitch && stackId !== undefined) {\n      if (this.tabBar) {\n        this.tabBar.selectedTab = stackId;\n      }\n      this.ionTabsDidChange.emit({ tab: stackId });\n    }\n  }\n\n  /**\n   * When a tab button is clicked, there are several scenarios:\n   * 1. If the selected tab is currently active (the tab button has been clicked\n   *    again), then it should go to the root view for that tab.\n   *\n   *   a. Get the saved root view from the router outlet. If the saved root view\n   *      matches the tabRootUrl, set the route view to this view including the\n   *      navigation extras.\n   *   b. If the saved root view from the router outlet does\n   *      not match, navigate to the tabRootUrl. No navigation extras are\n   *      included.\n   *\n   * 2. If the current tab tab is not currently selected, get the last route\n   *    view from the router outlet.\n   *\n   *   a. If the last route view exists, navigate to that view including any\n   *      navigation extras\n   *   b. If the last route view doesn't exist, then navigate\n   *      to the default tabRootUrl\n   */\n  @HostListener('ionTabButtonClick', ['$event'])\n  select(tabOrEvent: string | CustomEvent): Promise<boolean> | undefined {\n    const isTabString = typeof tabOrEvent === 'string';\n    const tab = isTabString ? tabOrEvent : (tabOrEvent as CustomEvent).detail.tab;\n\n    /**\n     * If the tabs are not using the router, then\n     * the tab switch logic is handled by the tabs\n     * component itself.\n     */\n    if (this.hasTab) {\n      this.setActiveTab(tab);\n      this.tabSwitch();\n\n      return;\n    }\n\n    const alreadySelected = this.outlet.getActiveStackId() === tab;\n    const tabRootUrl = `${this.outlet.tabsPrefix}/${tab}`;\n\n    /**\n     * If this is a nested tab, prevent the event\n     * from bubbling otherwise the outer tabs\n     * will respond to this event too, causing\n     * the app to get directed to the wrong place.\n     */\n    if (!isTabString) {\n      (tabOrEvent as CustomEvent).stopPropagation();\n    }\n\n    if (alreadySelected) {\n      const activeStackId = this.outlet.getActiveStackId();\n      const activeView = this.outlet.getLastRouteView(activeStackId);\n\n      // If on root tab, do not navigate to root tab again\n      if (activeView?.url === tabRootUrl) {\n        return;\n      }\n\n      const rootView = this.outlet.getRootView(tab);\n      const navigationExtras = rootView && tabRootUrl === rootView.url && rootView.savedExtras;\n      return this.navCtrl.navigateRoot(tabRootUrl, {\n        ...navigationExtras,\n        animated: true,\n        animationDirection: 'back',\n      });\n    } else {\n      const lastRoute = this.outlet.getLastRouteView(tab);\n      /**\n       * If there is a lastRoute, goto that, otherwise goto the fallback url of the\n       * selected tab\n       */\n      const url = lastRoute?.url || tabRootUrl;\n      const navigationExtras = lastRoute?.savedExtras;\n\n      return this.navCtrl.navigateRoot(url, {\n        ...navigationExtras,\n        animated: true,\n        animationDirection: 'back',\n      });\n    }\n  }\n\n  private setActiveTab(tab: string): void {\n    const tabs = this.tabs;\n    const selectedTab = tabs.find((t: any) => t.tab === tab);\n\n    if (!selectedTab) {\n      console.error(`[Ionic Error]: Tab with id: \"${tab}\" does not exist`);\n      return;\n    }\n\n    this.leavingTab = this.selectedTab;\n    this.selectedTab = selectedTab;\n\n    this.ionTabsWillChange.emit({ tab });\n\n    selectedTab.el.active = true;\n  }\n\n  private tabSwitch(): void {\n    const { selectedTab, leavingTab } = this;\n\n    if (this.tabBar && selectedTab) {\n      this.tabBar.selectedTab = selectedTab.tab;\n    }\n\n    if (leavingTab?.tab !== selectedTab?.tab) {\n      if (leavingTab?.el) {\n        leavingTab.el.active = false;\n      }\n    }\n\n    if (selectedTab) {\n      this.ionTabsDidChange.emit({ tab: selectedTab.tab });\n    }\n  }\n\n  getSelected(): string | undefined {\n    if (this.hasTab) {\n      return this.selectedTab?.tab;\n    }\n\n    return this.outlet.getActiveStackId();\n  }\n\n  /**\n   * Detects changes to the slot attribute of the tab bar.\n   *\n   * If the slot attribute has changed, then the tab bar\n   * should be relocated to the new slot position.\n   */\n  private detectSlotChanges(): void {\n    this.tabBars.forEach((tabBar: any) => {\n      // el is a protected attribute from the generated component wrapper\n      const currentSlot = tabBar.el.getAttribute('slot');\n\n      if (currentSlot !== this.tabBarSlot) {\n        this.tabBarSlot = currentSlot;\n        this.relocateTabBar();\n      }\n    });\n  }\n\n  /**\n   * Relocates the tab bar to the new slot position.\n   */\n  private relocateTabBar(): void {\n    /**\n     * `el` is a protected attribute from the generated component wrapper.\n     * To avoid having to manually create the wrapper for tab bar, we\n     * cast the tab bar to any and access the protected attribute.\n     */\n    const tabBar = (this.tabBar as any).el as HTMLElement;\n\n    if (this.tabBarSlot === 'top') {\n      /**\n       * A tab bar with a slot of \"top\" should be inserted\n       * at the top of the container.\n       */\n      this.tabsInner.nativeElement.before(tabBar);\n    } else {\n      /**\n       * A tab bar with a slot of \"bottom\" or without a slot\n       * should be inserted at the end of the container.\n       */\n      this.tabsInner.nativeElement.after(tabBar);\n    }\n  }\n}\n"]}