From da3b0c9a28f5cdcfaa484dc2c492341aa3bad906 Mon Sep 17 00:00:00 2001 From: Matthew Hartstonge Date: Fri, 15 Nov 2024 17:10:06 +1300 Subject: [PATCH] feat(addon/components/paper-switch): converts to a glimmer component. --- addon/components/paper-switch.hbs | 51 ++++-- addon/components/paper-switch.js | 247 +++++++++++++++++++----------- 2 files changed, 191 insertions(+), 107 deletions(-) diff --git a/addon/components/paper-switch.hbs b/addon/components/paper-switch.hbs index 461fc73e0..9fbe85472 100644 --- a/addon/components/paper-switch.hbs +++ b/addon/components/paper-switch.hbs @@ -1,19 +1,36 @@ -{{! template-lint-disable no-curly-component-invocation }} -
-
-
-
-
- + +
+
+
+
+
+ +
-
-{{#if (has-block)}} -
- {{yield}} -
-{{else}} -
- {{@label}} -
-{{/if}} + {{#if (has-block)}} +
+ {{yield}} +
+ {{else}} +
+ {{@label}} +
+ {{/if}} + \ No newline at end of file diff --git a/addon/components/paper-switch.js b/addon/components/paper-switch.js index 4418177dc..90a2a12c7 100644 --- a/addon/components/paper-switch.js +++ b/addon/components/paper-switch.js @@ -1,79 +1,102 @@ -/* eslint-disable ember/no-classic-components, ember/no-component-lifecycle-hooks, ember/no-get, ember/no-mixins, ember/require-tagless-components, no-unused-vars */ +/* global Hammer */ /** * @module ember-paper */ -import { inject as service } from '@ember/service'; - -import Component from '@ember/component'; +import Focusable from './-focusable'; +import { tracked } from '@glimmer/tracking'; import { assert } from '@ember/debug'; -import { get, computed } from '@ember/object'; -import { bind } from '@ember/runloop'; +import { action } from '@ember/object'; +import { inject as service } from '@ember/service'; import { htmlSafe } from '@ember/template'; -import FocusableMixin from 'ember-paper/mixins/focusable-mixin'; -import ProxiableMixin from 'ember-paper/mixins/proxiable-mixin'; -import { invokeAction } from 'ember-paper/utils/invoke-action'; - -/* global Hammer */ /** * @class PaperSwitch - * @extends Ember.Component - * @uses FocusableMixin - * @uses ProxiableMixin + * @extends Component */ -export default Component.extend(FocusableMixin, ProxiableMixin, { - tagName: 'md-switch', - classNames: ['paper-switch', 'md-default-theme'], - classNameBindings: [ - 'value:md-checked', - 'dragging:md-dragging', - 'warn:md-warn', - 'accent:md-accent', - 'primary:md-primary', - ], - toggle: true, - constants: service(), - value: false, - disabled: false, - dragging: false, - - thumbContainerStyle: computed('dragging', 'dragAmount', function () { - if (!this.dragging) { - return htmlSafe(''); +export default class PaperSwitch extends Focusable { + @service constants; + + /** + * Reference to the component's DOM element + * @type {HTMLElement} + */ + element; + /** + * The parent this component is bound to. + * @type {PaperRadioGroup|PaperForm|PaperItem|PaperTabs} + */ + parent; + /** + * Marks whether the component should register itself to the supplied parent + * @type {Boolean} + */ + shouldRegister; + /** + * Marks whether the component should skip being proxied. + * @type {Boolean} + */ + skipProxy; + + /* Focusable Overrides */ + toggle = true; + + /** + * specifies the positive amount the switch has been dragged. + * @type {number|null} + */ + @tracked dragAmount = null; + /** + * specifies whether the switch is currently being dragged. + * @type {boolean} + */ + @tracked dragging = false; + /** + * specifies the width of the switch to calculate drag deltas. + * @type {number} + */ + @tracked switchWidth = 0; + + // Lifecycle hooks + constructor(owner, args) { + super(owner, args); + + this.shouldRegister = this.args.shouldRegister || false; + this.skipProxy = this.args.skipProxy || false; + this.toggle = this.args.toggle || false; + + if (this.shouldRegister) { + assert( + 'A parent component should be supplied to when shouldRegister=true', + this.args.parentComponent + ); + this.parent = this.args.parentComponent; } - let translate = Math.max(0, Math.min(100, this.dragAmount * 100)); - let transformProp = `translate3d(${translate}%, 0, 0)`; - return htmlSafe( - `transform: ${transformProp};-webkit-transform: ${transformProp}` + assert( + ' requires an `onChange` action or null for no action.', + this.args.onChange !== undefined ); - }), - - didInsertElement() { - this._super(...arguments); + } + + /** + * Performs any required DOM setup. + * @param element + */ + @action didInsertNode(element) { + this.element = element; + this.registerListeners(element); + + if (this.shouldRegister) { + this.parent.registerChild(this); + } // Only setup if the switch is not disabled if (!this.disabled) { this._setupSwitch(); } - }, - - init() { - this._super(...arguments); - assert( - '{{paper-switch}} requires an `onChange` action or null for no action.', - this.onChange !== undefined - ); - }, - - willDestroyElement() { - this._super(...arguments); - this._teardownSwitch(); - }, - - didUpdateAttrs() { - this._super(...arguments); + } + @action didUpdateNode() { if (!this.disabled && !this._switchContainerHammer) { this._setupSwitch(); } else if (!this.disabled && this._switchContainerHammer) { @@ -81,13 +104,53 @@ export default Component.extend(FocusableMixin, ProxiableMixin, { } else if (this.disabled && this._switchContainerHammer) { this._switchContainerHammer.set({ enable: false }); } - }, + } + + /** + * Performs any required DOM teardown. + * @param element + */ + @action willDestroyNode(element) { + this.unregisterListeners(element); + this._teardownSwitch(); + } - _setupSwitch() { - this.set( - 'switchWidth', - this.element.querySelector('.md-thumb-container').offsetWidth + willDestroy() { + super.willDestroy(...arguments); + + if (this.shouldRegister) { + this.parent.unregisterChild(this); + } + } + + /** + * specifies the current switch value. + * @type {boolean} + */ + get value() { + return this.args.value; + } + + /** + * Calculates and returns a css animation transform for the switch's thumb. + * @returns {string} + */ + get thumbContainerStyle() { + if (!this.dragging) { + return htmlSafe(''); + } + + let translate = Math.max(0, Math.min(100, this.dragAmount * 100)); + let transformProp = `translate3d(${translate}%, 0, 0)`; + return htmlSafe( + `transform: ${transformProp};-webkit-transform: ${transformProp}` ); + } + + _setupSwitch() { + this.switchWidth = this.element.querySelector( + '.md-thumb-container' + ).offsetWidth; let switchContainer = this.element.querySelector('.md-container'); let switchHammer = new Hammer(switchContainer); @@ -96,20 +159,20 @@ export default Component.extend(FocusableMixin, ProxiableMixin, { // Enable dragging the switch container switchHammer.get('pan').set({ threshold: 1 }); switchHammer - .on('panstart', bind(this, this._dragStart)) - .on('panmove', bind(this, this._drag)) - .on('panend', bind(this, this._dragEnd)); + .on('panstart', this._dragStart.bind(this)) + .on('panmove', this._drag.bind(this)) + .on('panend', this._dragEnd.bind(this)); // Enable tapping gesture on the switch this._switchHammer = new Hammer(this.element); - this._switchHammer.on('tap', bind(this, this._dragEnd)); + this._switchHammer.on('tap', this._dragEnd.bind(this)); - this._onClickHandleNativeClick = bind(this, this._handleNativeClick); + this._onClickHandleNativeClick = this._handleNativeClick.bind(this); this.element .querySelector('.md-container') .addEventListener('click', this._onClickHandleNativeClick); - }, + } _handleNativeClick(ev) { let bubbles = this.bubbles; @@ -119,7 +182,7 @@ export default Component.extend(FocusableMixin, ProxiableMixin, { } return bubbles; - }, + } _teardownSwitch() { if (this._switchContainerHammer) { @@ -130,19 +193,19 @@ export default Component.extend(FocusableMixin, ProxiableMixin, { .querySelector('.md-container') .removeEventListener('click', this._onClickHandleNativeClick); this._onClickHandleNativeClick = null; - }, + } _dragStart() { - this.set('dragAmount', +this.value); - this.set('dragging', true); - }, + this.dragAmount = +this.value; + this.dragging = true; + } _drag(event) { if (!this.disabled) { // Set the amount the switch has been dragged - this.set('dragAmount', +this.value + event.deltaX / this.switchWidth); + this.dragAmount = +this.value + event.deltaX / this.switchWidth; } - }, + } _dragEnd() { if (!this.disabled) { @@ -154,31 +217,35 @@ export default Component.extend(FocusableMixin, ProxiableMixin, { (value && dragAmount < 0.5) || (!value && dragAmount > 0.5) ) { - invokeAction(this, 'onChange', !value); + if (this.args.onChange) { + this.args.onChange(!value); + } } - this.set('dragging', false); - this.set('dragAmount', null); + this.dragging = false; + this.dragAmount = null; } - }, + } - focusIn() { + @action handleFocusIn() { // Focusing in w/o being pressed should use the default behavior if (!this.pressed) { - this._super(...arguments); + super.handleFocusIn(...arguments); } - }, + } - keyPress(ev) { + @action handleKeyPress(ev) { if ( - ev.which === this.get('constants.KEYCODE.SPACE') || - ev.which === this.get('constants.KEYCODE.ENTER') + ev.which === this.constants.KEYCODE.SPACE || + ev.which === this.constants.KEYCODE.ENTER ) { ev.preventDefault(); this._dragEnd(); } - }, + } processProxy() { - invokeAction(this, 'onChange', !this.value); - }, -}); + if (this.args.onChange) { + this.args.onChange(!this.value); + } + } +}