diff --git a/addon/components/paper-ink-bar.hbs b/addon/components/paper-ink-bar.hbs new file mode 100644 index 000000000..fc13e442e --- /dev/null +++ b/addon/components/paper-ink-bar.hbs @@ -0,0 +1,2 @@ + + \ No newline at end of file diff --git a/addon/components/paper-ink-bar.js b/addon/components/paper-ink-bar.js index 41819dadf..18bc6214e 100644 --- a/addon/components/paper-ink-bar.js +++ b/addon/components/paper-ink-bar.js @@ -1,15 +1,8 @@ -/* eslint-disable ember/no-classic-components, ember/require-tagless-components, prettier/prettier */ -import { computed } from '@ember/object'; -import Component from '@ember/component'; +import Component from '@glimmer/component'; import { htmlSafe } from '@ember/template'; -export default Component.extend({ - tagName: 'md-ink-bar', - - attributeBindings: ['style'], - classNameBindings: ['movingRight:md-right:md-left'], - - style: computed('left', 'right', function() { - return htmlSafe(`left: ${this.left}px; right: ${this.right}px;`); - }) -}); +export default class PaperInkBar extends Component { + get style() { + return htmlSafe(`left: ${this.args.left}px; right: ${this.args.right}px;`); + } +} diff --git a/addon/components/paper-tab.hbs b/addon/components/paper-tab.hbs index f5007a8b9..1a49402b7 100644 --- a/addon/components/paper-tab.hbs +++ b/addon/components/paper-tab.hbs @@ -1,7 +1,25 @@ -{{! template-lint-disable no-curly-component-invocation }} -{{#if (has-block)}} - {{yield}} -{{else}} - {{@name}} -{{/if}} - \ No newline at end of file +{{#let (element this.tag) as |Tag|}} + + {{#if (has-block)}} + {{yield}} + {{else}} + {{@name}} + {{/if}} + + +{{/let}} \ No newline at end of file diff --git a/addon/components/paper-tab.js b/addon/components/paper-tab.js index a31c0cb49..25ed0aa81 100644 --- a/addon/components/paper-tab.js +++ b/addon/components/paper-tab.js @@ -1,72 +1,194 @@ -/* eslint-disable ember/classic-decorator-hooks, ember/classic-decorator-no-classic-methods, ember/no-classic-components, ember/no-computed-properties-in-native-classes, ember/no-mixins */ -import { - classNames, - attributeBindings, - classNameBindings, - tagName, -} from '@ember-decorators/component'; - -import { computed } from '@ember/object'; -import Component from '@ember/component'; +/** + * @module ember-paper + */ +import Focusable from './-focusable'; +import { tracked } from '@glimmer/tracking'; +import { action } from '@ember/object'; import { htmlSafe } from '@ember/template'; -import { ChildMixin } from 'ember-composability-tools'; -import FocusableMixin from 'ember-paper/mixins/focusable-mixin'; -import { invokeAction } from 'ember-paper/utils/invoke-action'; - -@tagName('md-tab-item') -@classNames('md-tab') -@classNameBindings('isSelected:md-active') -@attributeBindings('isSelected:aria-selected', 'style', 'maybeHref:href') -export default class PaperTab extends Component.extend( - ChildMixin, - FocusableMixin -) { - // tags have browser styles or are usually styled by the user - // this makes sure that tab item still looks good with an anchor tag - @computed('href') +import { assert } from '@ember/debug'; + +/** + * @class PaperTab + * @extends Focusable + */ +export default class PaperTab extends Focusable { + /** + * Reference to the component's DOM element. + * + * @type {HTMLElement} + */ + element; + /** + * The parent this component is bound to. + * + * @type {PaperTabs} + */ + parent; + /** + * Marks whether the component should register itself to the supplied parent. + * + * @type {Boolean} + */ + shouldRegister; + /** + * The top level tag to render. One of {'a', 'md-tab'}. + * + * @type {string} + * @private + * @default 'md-tab-item' + */ + tag; + /** + * provides a proxy value if one is not supplied by the user. + * + * @type {number|*} + * @private + */ + @tracked _value; + /** + * the number of pixels that the upper left corner of the current element is + * offset to the left within the {@link HTMLElement.offsetParent} node. + * + * @type{number} + */ + @tracked left; + /** + * the layout width of the element as an integer. + * + * @type{number} + */ + @tracked width; + + /** + * @constructor + * @param owner + * @param args + */ + constructor(owner, args) { + super(owner, args); + + this.tag = 'md-tab-item'; + if (this.args.href) { + this.tag = 'a'; + } + + this.shouldRegister = this.args.shouldRegister || true; + if (this.shouldRegister) { + assert( + 'A parent component should be supplied to ', + this.args.parentComponent + ); + this.parent = this.args.parentComponent; + } + } + + /** + * Performs any required DOM setup. + * + * @param {HTMLElement} element - the node that has been added to the DOM. + */ + @action didInsertNode(element) { + this.element = element; + this.left = element.offsetLeft; + this.width = element.offsetWidth; + + this.registerListeners(element); + + if (this.shouldRegister) { + this.parent.registerChild(this); + } + } + + /** + * didUpdateNode is called when tracked component attributes change. + */ + @action didUpdateNode() { + if (this.args.value) { + this.value = this.args.value; + } + } + + /** + * Performs any required DOM teardown. + * + * @param {HTMLElement} element - the node to be removed from the DOM. + */ + @action willDestroyNode(element) { + this.unregisterListeners(element); + } + + /** + * lifecycle hook to perform non-DOM related teardown. + */ + willDestroy() { + super.willDestroy(); + + if (this.shouldRegister) { + this.parent.unregisterChild(this); + } + } + + /** + * tags have browser styles or are usually styled by the user + * this makes sure that tab item still looks good with an anchor tag. + * + * @returns {string|undefined} + */ get style() { - if (this.href) { + if (this.args.href) { return htmlSafe('text-decoration: none; border: none;'); } else { return undefined; } } - @computed('href', 'disabled') + /** + * maybeHref returns the user supplied href link url. + * + * @returns {string|undefined} + */ get maybeHref() { - if (this.href && !this.disabled) { - return this.href; + if (this.args.href && !this.disabled) { + return this.args.href; } else { return undefined; } } - @computed('selected', 'value') + /** + * computes whether this is the currently selected tab. + * + * @returns {boolean} + */ get isSelected() { - return this.selected === this.value; - } - - init() { - super.init(...arguments); - if (this.href) { - this.set('tagName', 'a'); - } + return this.args.selected === this.value; } // this method is called by the parent updateDimensions() { // this is the true current width // it is used to calculate the ink bar position & pagination offset - this.setProperties({ - left: this.element.offsetLeft, - width: this.element.offsetWidth, - }); + this.left = this.element.offsetLeft; + this.width = this.element.offsetWidth; } - click() { + get value() { + // enable support for user supplied value + return this.args.value || this._value; + } + set value(value) { + this._value = value; + } + + @action handleClick(e) { if (!this.disabled) { - invokeAction(this, 'onClick', ...arguments); - invokeAction(this, 'onSelect', this); + if (this.args.onClick) { + this.args.onClick(e); + } + + if (this.args.onSelect) { + this.args.onSelect(this); + } } } } diff --git a/addon/components/paper-tabs.hbs b/addon/components/paper-tabs.hbs index 5df13b113..f835f7c25 100644 --- a/addon/components/paper-tabs.hbs +++ b/addon/components/paper-tabs.hbs @@ -1,31 +1,76 @@ -{{! template-lint-disable no-action }} - + + + {{#if this.shouldPaginate}} + + + + + {{! + md-next-button has a css rule that transforms the icon 180 degrees + to point the right way + }} + + + {{/if}} - {{#if this.shouldPaginate}} - - {{paper-icon "keyboard-arrow-left"}} - - - {{paper-icon "keyboard-arrow-left"}} - - {{/if}} + + + {{yield + (hash + tab=(component + 'paper-tab' + noInk=this.noInk + parentComponent=this + selected=this.selected + onSelect=this.localOnChange + ) + ) + }} - - - - {{yield (hash - tab=(component "paper-tab" - noInk=this.noInk - selected=this.selected - onSelect=(action "localOnChange") - ) - )}} - - {{#if this.inkBar}} - - {{/if}} - - - - - \ No newline at end of file + {{#if this.inkBar}} + {{#let this.inkBar as |inkBar|}} + + {{/let}} + {{/if}} + + + + \ No newline at end of file diff --git a/addon/components/paper-tabs.js b/addon/components/paper-tabs.js index 5f46e1532..6aa617a81 100644 --- a/addon/components/paper-tabs.js +++ b/addon/components/paper-tabs.js @@ -1,93 +1,281 @@ -/* eslint-disable ember/classic-decorator-no-classic-methods, ember/no-classic-components, ember/no-computed-properties-in-native-classes, ember/no-get */ -import { - classNames, - classNameBindings, - attributeBindings, - tagName, -} from '@ember-decorators/component'; -import { action, computed } from '@ember/object'; +import Component from '@glimmer/component'; +import { tracked } from '@glimmer/tracking'; +import { A } from '@ember/array'; +import { action } from '@ember/object'; import { inject as service } from '@ember/service'; -import { gt } from '@ember/object/computed'; -import Component from '@ember/component'; import { htmlSafe } from '@ember/template'; -import { scheduleOnce, join } from '@ember/runloop'; -import { ParentMixin } from 'ember-composability-tools'; -import { invokeAction } from 'ember-paper/utils/invoke-action'; - -@tagName('md-tabs') -@classNames('md-no-tab-content', 'md-default-theme') -@classNameBindings('warn:md-warn', 'accent:md-accent', 'primary:md-primary') -@attributeBindings('borderBottom:md-border-bottom') -export default class PaperTabs extends Component.extend(ParentMixin) { - @service - constants; - - selected = 0; // select first tab by default - noInkBar = false; - noInk = false; - ariaLabel = null; - stretch = 'sm'; - movingRight = true; - - @computed('noInkBar', '_selectedTab.{width,left}', 'wrapperWidth') + +export default class PaperTabs extends Component { + @service constants; + @service fastboot; + + /** + * Reference to the component's DOM element. + * + * @type {HTMLElement} + * @private + */ + @tracked element; + /** + * Reference to the component's md-tabs-canvas DOM element. + * + * @type {HTMLElement} + * @private + */ + @tracked elementTabsCanvas; + /** + * Reference to the component's md-pagination-wrapper DOM element. + * + * @type {HTMLElement} + * @private + */ + @tracked elementPaginationWrapper; + /** + * Array of tab components. + * + * @type {A} + */ + @tracked children; + /** + * tracks the outer width of the element displaying the tabs. + * + * `>[[ tab ][ tab ] ]<` + * + * @type {number} + */ + @tracked canvasWidth; + /** + * tracks the width of the tabs within the tab canvas. See {@link canvasWidth}. + * + * `[>[ tab ][ tab ]< ]` + * + * @type {number} + */ + @tracked wrapperWidth; + /** + * used to offset tabs based on pagination. + * + * `>[ tab ][ tab ]< [ [<-] [ tab ] [->] ]` + * + * @type {number} + */ + @tracked currentOffset; + /** + * holds a reference to the currently selected tab. + * + * @type {PaperTab} + * @private + */ + @tracked selectedTab = null; + /** + * tracks the direction in which inkbar should be animated. + * + * @type {boolean} + */ + @tracked movingRight = true; + /** + * set to true if the tab bar should enable paginating through tabs via + * left/right navigation arrows. + * + * @type {boolean} + */ + @tracked shouldPaginate = false; + /** + * _selected provides auto-tracking of selected tabs, this is kept private, so + * that {@link selected} can output either the auto-tracked number, or a + * user supplied value. + * + * @type {number|any} + * @private + * @default 0 - the first tab when using auto-numbering. + */ + @tracked _selected; + + constructor(owner, args) { + super(owner, args); + + this.children = A([]); + this.canvasWidth = 0; + this.wrapperWidth = 0; + this.currentOffset = 0; + this._selected = 0; + } + + /** + * Performs any required DOM setup. + * @param element + */ + @action didInsertNode(element) { + this.element = element; + this.elementTabsCanvas = element.querySelector('md-tabs-canvas'); + this.elementPaginationWrapper = element.querySelector( + 'md-pagination-wrapper' + ); + + window.addEventListener('resize', this.updateCanvasWidth); + window.addEventListener('orientationchange', this.updateCanvasWidth); + + // Do an initial sizing computation. + this.didUpdateNode(element); + } + + @action didUpdateNode() { + // this makes sure that the tabs react to stretch and center changes + // this method is also called whenever one of the tab is re-rendered (content changes) + this.updateCanvasWidth(); + this.updateSelectedTab(); + this.fixOffsetIfNeeded(); + } + + /** + * Performs any required DOM teardown. + * @param {HTMLElement} element + */ + @action willDestroyNode() { + window.removeEventListener('resize', this.updateCanvasWidth); + window.removeEventListener('orientationchange', this.updateCanvasWidth); + } + + /** + * Registers a child form component + * @param {PaperTab} child - The paper tab component to register + */ + @action registerChild(child) { + // automatically set value if not manually set + if (child.value === undefined) { + child.value = this.children.length; + } + + this.children.pushObject(child); + } + /** + * Removes a registered child form component + * @param {Component} child - The form component to unregister + */ + @action unregisterChild(child) { + this.children.removeObject(child); + } + + /** + * whether we can allow paginating to a previous number of tabs to display. + * + * @returns {boolean} + */ + get canPageBack() { + return this.currentOffset > 0; + } + + /** + * whether we can allow paginating to the next number of tabs to display. + * @returns {boolean} + */ + get canPageForward() { + return this.wrapperWidth - this.currentOffset > this.canvasWidth; + } + + /** + * disables the animated bar that sits at the bottom of the currently + * selected tab. + * + * @type {boolean} + */ + get noInkBar() { + return this.args.noInkBar || false; + } + + /** + * passes down to the individual tabs. + * + * @type {boolean} + */ + get noInk() { + return this.args.noInk || false; + } + get inkBar() { if (this.noInkBar) { return null; } - let selectedTab = this._selectedTab; - if (!selectedTab || selectedTab.get('left') === undefined) { + let selectedTab = this.selectedTab; + if (!selectedTab || selectedTab.left === undefined) { return null; } return { - left: selectedTab.get('left'), - right: - this.wrapperWidth - selectedTab.get('left') - selectedTab.get('width'), + left: selectedTab.left, + right: this.wrapperWidth - selectedTab.left - selectedTab.width, }; } - @computed('currentOffset') + /** + * returns a 3d translate based on the number of tabs to offset by depending + * on the current pagination. + * + * @returns {string} + */ get paginationStyle() { return htmlSafe( `transform: translate3d(-${this.currentOffset}px, 0px, 0px);` ); } - shouldPaginate = true; + /** + * Returns a user supplied value, or the auto-tracked tab selection via + * {@link _selected}. + * + * @returns {any|number} + */ + get selected() { + return this.args.selected || this._selected; + } - @computed('shouldPaginate', 'center') + /** + * returns true if the tabs should be centered. + * + * Only applicable if the number of tabs don't overflow, causing pagination + * to be enabled. + * + * @returns {boolean} + */ get shouldCenter() { - return !this.shouldPaginate && this.center; + let center = this.args.center ?? false; + return !this.shouldPaginate && center; } - @computed('shouldPaginate', 'currentStretch') + /** + * returns true if the tabs should be stretched based on either matching a + * provided media query {e.g. sm, md} or being set to true. + * + * Only applicable if the number of tabs don't overflow, causing pagination + * to be enabled. + * + * @returns {boolean} + */ get shouldStretch() { - return !this.shouldPaginate && this.currentStretch; - } - - didInsertElement() { - super.didInsertElement(...arguments); + let stretch = this.args.stretch ?? 'sm'; + let shouldStretch; - this.updateCanvasWidth = () => { - join(() => { - this.updateDimensions(); - this.updateStretchTabs(); - }); - }; - - window.addEventListener('resize', this.updateCanvasWidth); - window.addEventListener('orientationchange', this.updateCanvasWidth); + // if `true` or `false` is specified, always/never "stretch tabs" + // otherwise proceed with normal matchMedia test + if (typeof stretch === 'boolean') { + shouldStretch = stretch; + } else { + let mediaQuery = this.constants.MEDIA[stretch] || stretch; + shouldStretch = !this.fastboot.isFastBoot + ? window.matchMedia(mediaQuery).matches + : false; + } - scheduleOnce('afterRender', this, this.fixOffsetIfNeeded); + return !this.shouldPaginate && shouldStretch; } - didRender() { - super.didRender(...arguments); - // this makes sure that the tabs react to stretch and center changes - // this method is also called whenever one of the tab is re-rendered (content changes) - this.updateSelectedTab(); - this.updateCanvasWidth(); + /** + * forces re-computation of element widths and tab offset. + */ + @action updateCanvasWidth() { + this.updateDimensions(); + this.fixOffsetIfNeeded(); } /** @@ -97,53 +285,39 @@ export default class PaperTabs extends Component.extend(ParentMixin) { * nested because we pass the 'selected' property to them that will * invalidate their 'isSelected' property. */ - updateSelectedTab() { - let selectedTab = this.childComponents.findBy('isSelected'); - let previousSelectedTab = this._selectedTab; - + @action updateSelectedTab() { + let selectedTab = this.children.findBy('isSelected'); + let previousSelectedTab = this.selectedTab; if (selectedTab === previousSelectedTab) { return; } - this.set( - 'movingRight', + this.movingRight = !selectedTab || - !previousSelectedTab || - previousSelectedTab.get('left') < selectedTab.get('left') - ); - this.set('_selectedTab', selectedTab); - - scheduleOnce('afterRender', this, this.fixOffsetIfNeeded); + !previousSelectedTab || + previousSelectedTab.left < selectedTab.left; + this.selectedTab = selectedTab; } - willDestroyElement() { - super.willDestroyElement(...arguments); - window.removeEventListener('resize', this.updateCanvasWidth); - window.removeEventListener('orientationchange', this.updateCanvasWidth); - } - - registerChild(childComponent) { - super.registerChild(...arguments); - // automatically set value if not manually set - if (childComponent.get('value') === undefined) { - let length = this.childComponents.get('length'); - childComponent.set('value', length - 1); - } - } - - fixOffsetIfNeeded() { - if (this.isDestroying || this.isDestroyed) { + /** + * updates the pagination tab offset if needed. + */ + @action fixOffsetIfNeeded() { + if (this.isDestroying || this.isDestroyed || !this.selectedTab) { + // Don't attempt to compute if elements have not been added or are being + // removed from the DOM. return; } let canvasWidth = this.canvasWidth; let currentOffset = this.currentOffset; - let tabLeftOffset = this.get('_selectedTab.left'); - let tabRightOffset = tabLeftOffset + this.get('_selectedTab.width'); + let { left, width } = this.selected; + let tabLeftOffset = left; + let tabRightOffset = tabLeftOffset + width; let newOffset; - if (canvasWidth < this.get('_selectedTab.width')) { + if (canvasWidth < width) { // align with selectedTab if canvas smaller than selected tab newOffset = tabLeftOffset; } else if (tabRightOffset - currentOffset > canvasWidth) { @@ -160,81 +334,71 @@ export default class PaperTabs extends Component.extend(ParentMixin) { return; } - this.set('currentOffset', newOffset); + this.currentOffset = newOffset; } - updateDimensions() { - let canvasWidth = this.element.querySelector('md-tabs-canvas').offsetWidth; - let wrapperWidth = this.element.querySelector( - 'md-pagination-wrapper' - ).offsetWidth; - this.childComponents.invoke('updateDimensions'); - this.set('canvasWidth', canvasWidth); - this.set('wrapperWidth', wrapperWidth); - this.set('shouldPaginate', wrapperWidth > canvasWidth); - } - - updateStretchTabs() { - let stretch = this.stretch; - let currentStretch; - - // if `true` or `false` is specified, always/never "stretch tabs" - // otherwise proceed with normal matchMedia test - if (typeof stretch === 'boolean') { - currentStretch = stretch; - } else { - let mediaQuery = this.constants.MEDIA[stretch] || stretch; - currentStretch = window.matchMedia(mediaQuery).matches; + /** + * sets widths based on the current tab canvas and pagination wrapper + * elements. + */ + @action updateDimensions() { + if (!this.element) { + // node not added to the DOM yet... + return; } - this.set('currentStretch', currentStretch); - } - - currentOffset = 0; - - @gt('currentOffset', 0) - canPageBack; - - @computed('wrapperWidth', 'currentOffset', 'canvasWidth') - get canPageForward() { - return this.wrapperWidth - this.currentOffset > this.canvasWidth; + let canvasWidth = this.elementTabsCanvas.offsetWidth; + let wrapperWidth = this.elementPaginationWrapper.offsetWidth; + this.children.forEach((c) => c.updateDimensions()); + this.canvasWidth = canvasWidth; + this.wrapperWidth = wrapperWidth; + this.shouldPaginate = wrapperWidth > canvasWidth; } - @action - previousPage() { - let tab = this.childComponents.find((t) => { - // ensure we are no stuck because of a tab with a width > canvasWidth - return t.get('left') + t.get('width') >= this.currentOffset; + /** + * computes if a tab offset is required when paginating backwards. + */ + @action previousPage() { + let tab = this.children.find((t) => { + // ensure we are not stuck because of a tab with a width > canvasWidth + return t.left + t.width >= this.currentOffset; }); if (tab) { - let left = Math.max(0, tab.get('left') - this.canvasWidth); - this.set('currentOffset', left); + this.currentOffset = Math.max(0, tab.left - this.canvasWidth); } } - @action - nextPage() { - let tab = this.childComponents.find((t) => { - // ensure tab's offset is greater than current - // otherwise if the tab's width is greater than canvas we cannot paginate through it + /** + * computes if a tab offset is required when paginating forwards. + */ + @action nextPage() { + let tab = this.children.find((t) => { + // ensure tab's offset is greater than current otherwise if the tab's + // width is greater than canvas we cannot paginate through it. return ( - t.get('left') > this.currentOffset && + t.left > this.currentOffset && // paginate until the first partially hidden tab - t.get('left') + t.get('width') - this.currentOffset > this.canvasWidth + t.left + t.width - this.currentOffset > this.canvasWidth ); }); if (tab) { - this.set('currentOffset', tab.get('left')); + this.currentOffset = tab.left; } } - @action - localOnChange(selected) { + /** + * sets the current selected tab value, or passes the value up to the user to + * pass back down. + * + * @param {PaperTab} selected + */ + @action localOnChange(selected) { // support non DDAU scenario - if (this.onChange) { - invokeAction(this, 'onChange', selected.get('value')); + if (this.args.onChange) { + this.args.onChange(selected.value); } else { - this.set('selected', selected.get('value')); + // update our private reference + this._selected = selected.value; } } }