tabset.js 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317
  1. /**
  2. * @fileoverview added by tsickle
  3. * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
  4. */
  5. import { Component, ContentChildren, Directive, EventEmitter, Input, Output, QueryList, TemplateRef, ViewEncapsulation } from '@angular/core';
  6. import { NgbTabsetConfig } from './tabset-config';
  7. /** @type {?} */
  8. var nextId = 0;
  9. /**
  10. * A directive to wrap tab titles that need to contain HTML markup or other directives.
  11. *
  12. * Alternatively you could use the `NgbTab.title` input for string titles.
  13. */
  14. var NgbTabTitle = /** @class */ (function () {
  15. function NgbTabTitle(templateRef) {
  16. this.templateRef = templateRef;
  17. }
  18. NgbTabTitle.decorators = [
  19. { type: Directive, args: [{ selector: 'ng-template[ngbTabTitle]' },] }
  20. ];
  21. /** @nocollapse */
  22. NgbTabTitle.ctorParameters = function () { return [
  23. { type: TemplateRef }
  24. ]; };
  25. return NgbTabTitle;
  26. }());
  27. export { NgbTabTitle };
  28. if (false) {
  29. /** @type {?} */
  30. NgbTabTitle.prototype.templateRef;
  31. }
  32. /**
  33. * A directive to wrap content to be displayed in a tab.
  34. */
  35. var NgbTabContent = /** @class */ (function () {
  36. function NgbTabContent(templateRef) {
  37. this.templateRef = templateRef;
  38. }
  39. NgbTabContent.decorators = [
  40. { type: Directive, args: [{ selector: 'ng-template[ngbTabContent]' },] }
  41. ];
  42. /** @nocollapse */
  43. NgbTabContent.ctorParameters = function () { return [
  44. { type: TemplateRef }
  45. ]; };
  46. return NgbTabContent;
  47. }());
  48. export { NgbTabContent };
  49. if (false) {
  50. /** @type {?} */
  51. NgbTabContent.prototype.templateRef;
  52. }
  53. /**
  54. * A directive representing an individual tab.
  55. */
  56. var NgbTab = /** @class */ (function () {
  57. function NgbTab() {
  58. /**
  59. * The tab identifier.
  60. *
  61. * Must be unique for the entire document for proper accessibility support.
  62. */
  63. this.id = "ngb-tab-" + nextId++;
  64. /**
  65. * If `true`, the current tab is disabled and can't be toggled.
  66. */
  67. this.disabled = false;
  68. }
  69. /**
  70. * @return {?}
  71. */
  72. NgbTab.prototype.ngAfterContentChecked = /**
  73. * @return {?}
  74. */
  75. function () {
  76. // We are using @ContentChildren instead of @ContentChild as in the Angular version being used
  77. // only @ContentChildren allows us to specify the {descendants: false} option.
  78. // Without {descendants: false} we are hitting bugs described in:
  79. // https://github.com/ng-bootstrap/ng-bootstrap/issues/2240
  80. this.titleTpl = this.titleTpls.first;
  81. this.contentTpl = this.contentTpls.first;
  82. };
  83. NgbTab.decorators = [
  84. { type: Directive, args: [{ selector: 'ngb-tab' },] }
  85. ];
  86. NgbTab.propDecorators = {
  87. id: [{ type: Input }],
  88. title: [{ type: Input }],
  89. disabled: [{ type: Input }],
  90. titleTpls: [{ type: ContentChildren, args: [NgbTabTitle, { descendants: false },] }],
  91. contentTpls: [{ type: ContentChildren, args: [NgbTabContent, { descendants: false },] }]
  92. };
  93. return NgbTab;
  94. }());
  95. export { NgbTab };
  96. if (false) {
  97. /**
  98. * The tab identifier.
  99. *
  100. * Must be unique for the entire document for proper accessibility support.
  101. * @type {?}
  102. */
  103. NgbTab.prototype.id;
  104. /**
  105. * The tab title.
  106. *
  107. * Use the [`NgbTabTitle`](#/components/tabset/api#NgbTabTitle) directive for non-string titles.
  108. * @type {?}
  109. */
  110. NgbTab.prototype.title;
  111. /**
  112. * If `true`, the current tab is disabled and can't be toggled.
  113. * @type {?}
  114. */
  115. NgbTab.prototype.disabled;
  116. /** @type {?} */
  117. NgbTab.prototype.titleTpl;
  118. /** @type {?} */
  119. NgbTab.prototype.contentTpl;
  120. /** @type {?} */
  121. NgbTab.prototype.titleTpls;
  122. /** @type {?} */
  123. NgbTab.prototype.contentTpls;
  124. }
  125. /**
  126. * The payload of the change event fired right before the tab change.
  127. * @record
  128. */
  129. export function NgbTabChangeEvent() { }
  130. if (false) {
  131. /**
  132. * The id of the currently active tab.
  133. * @type {?}
  134. */
  135. NgbTabChangeEvent.prototype.activeId;
  136. /**
  137. * The id of the newly selected tab.
  138. * @type {?}
  139. */
  140. NgbTabChangeEvent.prototype.nextId;
  141. /**
  142. * Calling this function will prevent tab switching.
  143. * @type {?}
  144. */
  145. NgbTabChangeEvent.prototype.preventDefault;
  146. }
  147. /**
  148. * A component that makes it easy to create tabbed interface.
  149. */
  150. var NgbTabset = /** @class */ (function () {
  151. function NgbTabset(config) {
  152. /**
  153. * If `true`, non-visible tabs content will be removed from DOM. Otherwise it will just be hidden.
  154. */
  155. this.destroyOnHide = true;
  156. /**
  157. * A tab change event emitted right before the tab change happens.
  158. *
  159. * See [`NgbTabChangeEvent`](#/components/tabset/api#NgbTabChangeEvent) for payload details.
  160. */
  161. this.tabChange = new EventEmitter();
  162. this.type = config.type;
  163. this.justify = config.justify;
  164. this.orientation = config.orientation;
  165. }
  166. Object.defineProperty(NgbTabset.prototype, "justify", {
  167. /**
  168. * The horizontal alignment of the tabs with flexbox utilities.
  169. */
  170. set: /**
  171. * The horizontal alignment of the tabs with flexbox utilities.
  172. * @param {?} className
  173. * @return {?}
  174. */
  175. function (className) {
  176. if (className === 'fill' || className === 'justified') {
  177. this.justifyClass = "nav-" + className;
  178. }
  179. else {
  180. this.justifyClass = "justify-content-" + className;
  181. }
  182. },
  183. enumerable: true,
  184. configurable: true
  185. });
  186. /**
  187. * Selects the tab with the given id and shows its associated content panel.
  188. *
  189. * Any other tab that was previously selected becomes unselected and its associated pane is removed from DOM or
  190. * hidden depending on the `destroyOnHide` value.
  191. */
  192. /**
  193. * Selects the tab with the given id and shows its associated content panel.
  194. *
  195. * Any other tab that was previously selected becomes unselected and its associated pane is removed from DOM or
  196. * hidden depending on the `destroyOnHide` value.
  197. * @param {?} tabId
  198. * @return {?}
  199. */
  200. NgbTabset.prototype.select = /**
  201. * Selects the tab with the given id and shows its associated content panel.
  202. *
  203. * Any other tab that was previously selected becomes unselected and its associated pane is removed from DOM or
  204. * hidden depending on the `destroyOnHide` value.
  205. * @param {?} tabId
  206. * @return {?}
  207. */
  208. function (tabId) {
  209. /** @type {?} */
  210. var selectedTab = this._getTabById(tabId);
  211. if (selectedTab && !selectedTab.disabled && this.activeId !== selectedTab.id) {
  212. /** @type {?} */
  213. var defaultPrevented_1 = false;
  214. this.tabChange.emit({ activeId: this.activeId, nextId: selectedTab.id, preventDefault: (/**
  215. * @return {?}
  216. */
  217. function () { defaultPrevented_1 = true; }) });
  218. if (!defaultPrevented_1) {
  219. this.activeId = selectedTab.id;
  220. }
  221. }
  222. };
  223. /**
  224. * @return {?}
  225. */
  226. NgbTabset.prototype.ngAfterContentChecked = /**
  227. * @return {?}
  228. */
  229. function () {
  230. // auto-correct activeId that might have been set incorrectly as input
  231. /** @type {?} */
  232. var activeTab = this._getTabById(this.activeId);
  233. this.activeId = activeTab ? activeTab.id : (this.tabs.length ? this.tabs.first.id : null);
  234. };
  235. /**
  236. * @private
  237. * @param {?} id
  238. * @return {?}
  239. */
  240. NgbTabset.prototype._getTabById = /**
  241. * @private
  242. * @param {?} id
  243. * @return {?}
  244. */
  245. function (id) {
  246. /** @type {?} */
  247. var tabsWithId = this.tabs.filter((/**
  248. * @param {?} tab
  249. * @return {?}
  250. */
  251. function (tab) { return tab.id === id; }));
  252. return tabsWithId.length ? tabsWithId[0] : null;
  253. };
  254. NgbTabset.decorators = [
  255. { type: Component, args: [{
  256. selector: 'ngb-tabset',
  257. exportAs: 'ngbTabset',
  258. encapsulation: ViewEncapsulation.None,
  259. template: "\n <ul [class]=\"'nav nav-' + type + (orientation == 'horizontal'? ' ' + justifyClass : ' flex-column')\" role=\"tablist\">\n <li class=\"nav-item\" *ngFor=\"let tab of tabs\">\n <a [id]=\"tab.id\" class=\"nav-link\" [class.active]=\"tab.id === activeId\" [class.disabled]=\"tab.disabled\"\n href (click)=\"select(tab.id); $event.preventDefault()\" role=\"tab\" [attr.tabindex]=\"(tab.disabled ? '-1': undefined)\"\n [attr.aria-controls]=\"(!destroyOnHide || tab.id === activeId ? tab.id + '-panel' : null)\"\n [attr.aria-selected]=\"tab.id === activeId\" [attr.aria-disabled]=\"tab.disabled\">\n {{tab.title}}<ng-template [ngTemplateOutlet]=\"tab.titleTpl?.templateRef\"></ng-template>\n </a>\n </li>\n </ul>\n <div class=\"tab-content\">\n <ng-template ngFor let-tab [ngForOf]=\"tabs\">\n <div\n class=\"tab-pane {{tab.id === activeId ? 'active' : null}}\"\n *ngIf=\"!destroyOnHide || tab.id === activeId\"\n role=\"tabpanel\"\n [attr.aria-labelledby]=\"tab.id\" id=\"{{tab.id}}-panel\">\n <ng-template [ngTemplateOutlet]=\"tab.contentTpl?.templateRef\"></ng-template>\n </div>\n </ng-template>\n </div>\n "
  260. }] }
  261. ];
  262. /** @nocollapse */
  263. NgbTabset.ctorParameters = function () { return [
  264. { type: NgbTabsetConfig }
  265. ]; };
  266. NgbTabset.propDecorators = {
  267. tabs: [{ type: ContentChildren, args: [NgbTab,] }],
  268. activeId: [{ type: Input }],
  269. destroyOnHide: [{ type: Input }],
  270. justify: [{ type: Input }],
  271. orientation: [{ type: Input }],
  272. type: [{ type: Input }],
  273. tabChange: [{ type: Output }]
  274. };
  275. return NgbTabset;
  276. }());
  277. export { NgbTabset };
  278. if (false) {
  279. /** @type {?} */
  280. NgbTabset.prototype.justifyClass;
  281. /** @type {?} */
  282. NgbTabset.prototype.tabs;
  283. /**
  284. * The identifier of the tab that should be opened **initially**.
  285. *
  286. * For subsequent tab switches use the `.select()` method and the `(tabChange)` event.
  287. * @type {?}
  288. */
  289. NgbTabset.prototype.activeId;
  290. /**
  291. * If `true`, non-visible tabs content will be removed from DOM. Otherwise it will just be hidden.
  292. * @type {?}
  293. */
  294. NgbTabset.prototype.destroyOnHide;
  295. /**
  296. * The orientation of the tabset.
  297. * @type {?}
  298. */
  299. NgbTabset.prototype.orientation;
  300. /**
  301. * Type of navigation to be used for tabs.
  302. *
  303. * Currently Bootstrap supports only `"tabs"` and `"pills"`.
  304. *
  305. * Since `3.0.0` can also be an arbitrary string (ex. for custom themes).
  306. * @type {?}
  307. */
  308. NgbTabset.prototype.type;
  309. /**
  310. * A tab change event emitted right before the tab change happens.
  311. *
  312. * See [`NgbTabChangeEvent`](#/components/tabset/api#NgbTabChangeEvent) for payload details.
  313. * @type {?}
  314. */
  315. NgbTabset.prototype.tabChange;
  316. }
  317. //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tabset.js","sourceRoot":"ng://@ng-bootstrap/ng-bootstrap/","sources":["tabset/tabset.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterContentChecked,\n  Component,\n  ContentChildren,\n  Directive,\n  EventEmitter,\n  Input,\n  Output,\n  QueryList,\n  TemplateRef,\n  ViewEncapsulation\n} from '@angular/core';\nimport {NgbTabsetConfig} from './tabset-config';\n\nlet nextId = 0;\n\n/**\n * A directive to wrap tab titles that need to contain HTML markup or other directives.\n *\n * Alternatively you could use the `NgbTab.title` input for string titles.\n */\n@Directive({selector: 'ng-template[ngbTabTitle]'})\nexport class NgbTabTitle {\n  constructor(public templateRef: TemplateRef<any>) {}\n}\n\n/**\n * A directive to wrap content to be displayed in a tab.\n */\n@Directive({selector: 'ng-template[ngbTabContent]'})\nexport class NgbTabContent {\n  constructor(public templateRef: TemplateRef<any>) {}\n}\n\n/**\n * A directive representing an individual tab.\n */\n@Directive({selector: 'ngb-tab'})\nexport class NgbTab implements AfterContentChecked {\n  /**\n   * The tab identifier.\n   *\n   * Must be unique for the entire document for proper accessibility support.\n   */\n  @Input() id = `ngb-tab-${nextId++}`;\n\n  /**\n   * The tab title.\n   *\n   * Use the [`NgbTabTitle`](#/components/tabset/api#NgbTabTitle) directive for non-string titles.\n   */\n  @Input() title: string;\n\n  /**\n   * If `true`, the current tab is disabled and can't be toggled.\n   */\n  @Input() disabled = false;\n\n  titleTpl: NgbTabTitle | null;\n  contentTpl: NgbTabContent | null;\n\n  @ContentChildren(NgbTabTitle, {descendants: false}) titleTpls: QueryList<NgbTabTitle>;\n  @ContentChildren(NgbTabContent, {descendants: false}) contentTpls: QueryList<NgbTabContent>;\n\n  ngAfterContentChecked() {\n    // We are using @ContentChildren instead of @ContentChild as in the Angular version being used\n    // only @ContentChildren allows us to specify the {descendants: false} option.\n    // Without {descendants: false} we are hitting bugs described in:\n    // https://github.com/ng-bootstrap/ng-bootstrap/issues/2240\n    this.titleTpl = this.titleTpls.first;\n    this.contentTpl = this.contentTpls.first;\n  }\n}\n\n/**\n * The payload of the change event fired right before the tab change.\n */\nexport interface NgbTabChangeEvent {\n  /**\n   * The id of the currently active tab.\n   */\n  activeId: string;\n\n  /**\n   * The id of the newly selected tab.\n   */\n  nextId: string;\n\n  /**\n   * Calling this function will prevent tab switching.\n   */\n  preventDefault: () => void;\n}\n\n/**\n * A component that makes it easy to create tabbed interface.\n */\n@Component({\n  selector: 'ngb-tabset',\n  exportAs: 'ngbTabset',\n  encapsulation: ViewEncapsulation.None,\n  template: `\n    <ul [class]=\"'nav nav-' + type + (orientation == 'horizontal'?  ' ' + justifyClass : ' flex-column')\" role=\"tablist\">\n      <li class=\"nav-item\" *ngFor=\"let tab of tabs\">\n        <a [id]=\"tab.id\" class=\"nav-link\" [class.active]=\"tab.id === activeId\" [class.disabled]=\"tab.disabled\"\n          href (click)=\"select(tab.id); $event.preventDefault()\" role=\"tab\" [attr.tabindex]=\"(tab.disabled ? '-1': undefined)\"\n          [attr.aria-controls]=\"(!destroyOnHide || tab.id === activeId ? tab.id + '-panel' : null)\"\n          [attr.aria-selected]=\"tab.id === activeId\" [attr.aria-disabled]=\"tab.disabled\">\n          {{tab.title}}<ng-template [ngTemplateOutlet]=\"tab.titleTpl?.templateRef\"></ng-template>\n        </a>\n      </li>\n    </ul>\n    <div class=\"tab-content\">\n      <ng-template ngFor let-tab [ngForOf]=\"tabs\">\n        <div\n          class=\"tab-pane {{tab.id === activeId ? 'active' : null}}\"\n          *ngIf=\"!destroyOnHide || tab.id === activeId\"\n          role=\"tabpanel\"\n          [attr.aria-labelledby]=\"tab.id\" id=\"{{tab.id}}-panel\">\n          <ng-template [ngTemplateOutlet]=\"tab.contentTpl?.templateRef\"></ng-template>\n        </div>\n      </ng-template>\n    </div>\n  `\n})\nexport class NgbTabset implements AfterContentChecked {\n  justifyClass: string;\n\n  @ContentChildren(NgbTab) tabs: QueryList<NgbTab>;\n\n  /**\n   * The identifier of the tab that should be opened **initially**.\n   *\n   * For subsequent tab switches use the `.select()` method and the `(tabChange)` event.\n   */\n  @Input() activeId: string;\n\n  /**\n   * If `true`, non-visible tabs content will be removed from DOM. Otherwise it will just be hidden.\n   */\n  @Input() destroyOnHide = true;\n\n  /**\n   * The horizontal alignment of the tabs with flexbox utilities.\n   */\n  @Input()\n  set justify(className: 'start' | 'center' | 'end' | 'fill' | 'justified') {\n    if (className === 'fill' || className === 'justified') {\n      this.justifyClass = `nav-${className}`;\n    } else {\n      this.justifyClass = `justify-content-${className}`;\n    }\n  }\n\n  /**\n   * The orientation of the tabset.\n   */\n  @Input() orientation: 'horizontal' | 'vertical';\n\n  /**\n   * Type of navigation to be used for tabs.\n   *\n   * Currently Bootstrap supports only `\"tabs\"` and `\"pills\"`.\n   *\n   * Since `3.0.0` can also be an arbitrary string (ex. for custom themes).\n   */\n  @Input() type: 'tabs' | 'pills' | string;\n\n  /**\n   * A tab change event emitted right before the tab change happens.\n   *\n   * See [`NgbTabChangeEvent`](#/components/tabset/api#NgbTabChangeEvent) for payload details.\n   */\n  @Output() tabChange = new EventEmitter<NgbTabChangeEvent>();\n\n  constructor(config: NgbTabsetConfig) {\n    this.type = config.type;\n    this.justify = config.justify;\n    this.orientation = config.orientation;\n  }\n\n  /**\n   * Selects the tab with the given id and shows its associated content panel.\n   *\n   * Any other tab that was previously selected becomes unselected and its associated pane is removed from DOM or\n   * hidden depending on the `destroyOnHide` value.\n   */\n  select(tabId: string) {\n    let selectedTab = this._getTabById(tabId);\n    if (selectedTab && !selectedTab.disabled && this.activeId !== selectedTab.id) {\n      let defaultPrevented = false;\n\n      this.tabChange.emit(\n          {activeId: this.activeId, nextId: selectedTab.id, preventDefault: () => { defaultPrevented = true; }});\n\n      if (!defaultPrevented) {\n        this.activeId = selectedTab.id;\n      }\n    }\n  }\n\n  ngAfterContentChecked() {\n    // auto-correct activeId that might have been set incorrectly as input\n    let activeTab = this._getTabById(this.activeId);\n    this.activeId = activeTab ? activeTab.id : (this.tabs.length ? this.tabs.first.id : null);\n  }\n\n  private _getTabById(id: string): NgbTab {\n    let tabsWithId: NgbTab[] = this.tabs.filter(tab => tab.id === id);\n    return tabsWithId.length ? tabsWithId[0] : null;\n  }\n}\n"]}