UNPKG

@uex/web-extensions

Version:

Uex extensions for Angular 6+ web projects

268 lines 25.3 kB
/** * @fileoverview added by tsickle * Generated from: lib/modules/button/directives/ripple.directive.ts * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc */ import { Directive, ElementRef, Input, HostListener } from '@angular/core'; import * as _ from 'lodash'; /** * Adiciona o efeito "ripple" ao clicar em um elemento * @abstract */ export class RippleDirectiveBase { /** * @param {?} elementRef */ constructor(elementRef) { this.elementRef = elementRef; /** * @ignore */ this._no_ripple = false; this._default_ripple_opacity = '0.15'; } /** * @private * @param {?} no_ripple * @return {?} */ set set_no_ripple(no_ripple) { this.no_ripple = _.isString(no_ripple) && no_ripple == '' ? true : (_.isBoolean(no_ripple) ? ((/** @type {?} */ (no_ripple))) : true); } /** * @param {?} value * @return {?} */ set no_ripple(value) { this._no_ripple = value; } /** * @return {?} */ get no_ripple() { return this._no_ripple; } /** * @param {?} e * @return {?} */ _onClick(e) { if (this.no_ripple) { return; } /** @type {?} */ const targetEl = this.elementRef ? this.elementRef.nativeElement : null; if (!targetEl) { return; } if (this._animate_timeout) { clearTimeout(this._animate_timeout); } /** @type {?} */ let rippleEl = ((/** @type {?} */ (targetEl.querySelector('.ripple')))); if (!!rippleEl) { rippleEl.classList.remove('animate'); } else { rippleEl = document.createElement('span'); rippleEl.classList.add('ripple'); targetEl.appendChild(rippleEl); } /** @type {?} */ let offset_x = e.offsetX; /** @type {?} */ let offset_y = e.offsetY; if (e.target != targetEl) { /** @type {?} */ const relative_offset = this._relativeOffset((/** @type {?} */ (e.target)), targetEl); offset_x += relative_offset.left; offset_y += relative_offset.top; } rippleEl.style.width = rippleEl.style.height = Math.max(targetEl.offsetWidth, targetEl.offsetHeight) + 'px'; rippleEl.style.left = (offset_x - rippleEl.offsetWidth / 2) + 'px'; rippleEl.style.top = (offset_y - rippleEl.offsetHeight / 2) + 'px'; // Apply element color and background color to wrapper /** @type {?} */ const element_style = window.getComputedStyle(targetEl, null); /** @type {?} */ const color = element_style.color; /** @type {?} */ const background_contrast_color = this._contrast(this._contrast(color)); rippleEl.style.color = color; rippleEl.style.background = background_contrast_color; rippleEl.classList.add('animate'); this._animate_timeout = setTimeout((/** * @return {?} */ () => rippleEl.classList.remove('animate')), 650); } /** * @private * @param {?} color * @param {?=} bw * @return {?} */ _contrast(color, bw = true) { /** @type {?} */ let colorHex = this._toHexadecimal(color); if (!/(^#[0-9A-F]{6}$)|(^#[0-9A-F]{3}$)/i.test(colorHex)) { return '#FFFFFF'; } if (colorHex.length === 4) { colorHex = colorHex.replace(/^#(.)(.)(.)/i, '#$1$1$2$2$3$3'); } /** @type {?} */ const r = parseInt(colorHex.substr(1, 2), 16); /** @type {?} */ const g = parseInt(colorHex.substr(3, 2), 16); /** @type {?} */ const b = parseInt(colorHex.substr(5, 2), 16); if (bw) { // https://github.com/lffg/yiq/blob/master/index.js /** @type {?} */ const yiq = ((r * 299) + (g * 587) + (b * 114)) / 1000; return this._toRGBA(yiq >= 128 ? '#000000' : '#FFFFFF'); } return this._toRGBA(`#${this._padZero(r)}${this._padZero(g)}${this._padZero(b)}`); } /** * @private * @param {?} color * @param {?=} allow_alpha * @return {?} */ _toHexadecimal(color, allow_alpha = false) { if (!color || (color && color.indexOf('(') == -1)) { return color; } /** @type {?} */ const parts = color.replace(/(rgba|rgb)/g, '').replace(/[\(\)]/g, '').split(',').map((/** * @param {?} part * @return {?} */ (part) => _.trim(part))); /** @type {?} */ const r = this._padZero(parseInt(parts[0], 10).toString(16)); /** @type {?} */ const g = this._padZero(parseInt(parts[1], 10).toString(16)); /** @type {?} */ const b = this._padZero(parseInt(parts[2], 10).toString(16)); if (parts[3] && allow_alpha) { /** @type {?} */ const a = parseFloat(parseFloat(parts[3].substring(0, parts[3].length - 1)).toFixed(2)); return `#${r}${g}${b}${this._padZero((a * 255).toString(16).substring(0, 2))}`; } return `#${r}${g}${b}`; } /** * @private * @param {?} color * @param {?=} opacity * @return {?} */ _toRGBA(color, opacity = this._default_ripple_opacity) { /** @type {?} */ const result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(color); return result ? `rgba(${parseInt(result[1], 16)}, ${parseInt(result[2], 16)}, ${parseInt(result[3], 16)}, ${opacity})` : color; } /** * @private * @param {?} str * @param {?=} len * @return {?} */ _padZero(str, len = 2) { if (_.isNumber(str)) { str = str.toString(); } /** @type {?} */ const zeros = new Array(len).join('0'); return (zeros + str).slice(-len); } /** * @private * @param {?} child_element * @param {?} parent_element * @return {?} */ _relativeOffset(child_element, parent_element) { /** @type {?} */ const offset_parent_element = ((/** @type {?} */ (child_element.offsetParent))); /** @type {?} */ const parent_rect = offset_parent_element.getBoundingClientRect(); /** @type {?} */ const child_rect = child_element.getBoundingClientRect(); /** @type {?} */ const offset = { top: child_rect.top - parent_rect.top, left: child_rect.left - parent_rect.left } // Reached top of document ; // Reached top of document if (offset_parent_element.tagName === 'BODY') { return offset; } // Parent element contains the 'top' element we want the offset to be relative to if (child_element.parentElement === parent_element) { return offset; } // Reached the 'top' element we want the offset to be relative to if (offset_parent_element === parent_element) { return offset; } // Get parent's relative offset /** @type {?} */ const parent_offset = this._relativeOffset(offset_parent_element, parent_element); offset.top += parent_offset.top; offset.left += parent_offset.left; return offset; } } RippleDirectiveBase.propDecorators = { set_no_ripple: [{ type: Input, args: ['no-ripple',] }], _onClick: [{ type: HostListener, args: ['mousedown', ['$event'],] }] }; if (false) { /** * @ignore * @type {?} * @protected */ RippleDirectiveBase.prototype._no_ripple; /** * @type {?} * @private */ RippleDirectiveBase.prototype._default_ripple_opacity; /** * @type {?} * @private */ RippleDirectiveBase.prototype._animate_timeout; /** @type {?} */ RippleDirectiveBase.prototype.elementRef; } export class RippleDirective extends RippleDirectiveBase { /** * @param {?} elementRef */ constructor(elementRef) { super(elementRef); this.elementRef = elementRef; } } RippleDirective.decorators = [ { type: Directive, args: [{ selector: '[with-ripple], .with-ripple' },] } ]; /** @nocollapse */ RippleDirective.ctorParameters = () => [ { type: ElementRef } ]; if (false) { /** @type {?} */ RippleDirective.prototype.elementRef; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ripple.directive.js","sourceRoot":"ng://@uex/web-extensions/","sources":["lib/modules/button/directives/ripple.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, ElementRef, Input, HostListener } from '@angular/core';\r\n\r\nimport * as _ from 'lodash';\r\n\r\n/**\r\n * Adiciona o efeito \"ripple\" ao clicar em um elemento\r\n */\r\nexport abstract class RippleDirectiveBase {\r\n\r\n  @Input('no-ripple')\r\n  private set set_no_ripple(no_ripple: boolean | string) {\r\n    this.no_ripple = _.isString(no_ripple) && no_ripple == '' ? true : (_.isBoolean(no_ripple) ? (<boolean>no_ripple) : true);\r\n  }\r\n\r\n  constructor(\r\n    public elementRef: ElementRef\r\n  ) {}\r\n\r\n  public set no_ripple(value: boolean) {\r\n    this._no_ripple = value;\r\n  }\r\n  public get no_ripple(): boolean {\r\n    return this._no_ripple;\r\n  }\r\n\r\n  /**\r\n   * @ignore\r\n   */\r\n  protected _no_ripple = false;\r\n\r\n  private _default_ripple_opacity = '0.15';\r\n  private _animate_timeout: any;\r\n\r\n  @HostListener('mousedown', ['$event']) public _onClick(e: MouseEvent) {\r\n    if (this.no_ripple) { return; }\r\n\r\n    const targetEl = this.elementRef ? this.elementRef.nativeElement : null;\r\n    if (!targetEl) { return; }\r\n\r\n    if (this._animate_timeout) { clearTimeout(this._animate_timeout); }\r\n\r\n    let rippleEl = (<HTMLElement>targetEl.querySelector('.ripple'));\r\n\r\n    if (!!rippleEl) {\r\n      rippleEl.classList.remove('animate');\r\n    } else {\r\n      rippleEl = document.createElement('span');\r\n      rippleEl.classList.add('ripple');\r\n      targetEl.appendChild(rippleEl);\r\n    }\r\n\r\n    let offset_x = e.offsetX;\r\n    let offset_y = e.offsetY;\r\n\r\n    if (e.target != targetEl) {\r\n      const relative_offset = this._relativeOffset(<HTMLElement>e.target, targetEl);\r\n\r\n      offset_x += relative_offset.left;\r\n      offset_y += relative_offset.top;\r\n    }\r\n\r\n    rippleEl.style.width = rippleEl.style.height = Math.max(targetEl.offsetWidth, targetEl.offsetHeight) + 'px';\r\n    rippleEl.style.left = (offset_x - rippleEl.offsetWidth / 2) + 'px';\r\n    rippleEl.style.top = (offset_y - rippleEl.offsetHeight / 2) + 'px';\r\n\r\n    // Apply element color and background color to wrapper\r\n    const element_style    = window.getComputedStyle(targetEl, null);\r\n    const color           = element_style.color;\r\n\r\n    const background_contrast_color = this._contrast(this._contrast(color));\r\n\r\n    rippleEl.style.color = color;\r\n    rippleEl.style.background = background_contrast_color;\r\n\r\n    rippleEl.classList.add('animate');\r\n    this._animate_timeout = setTimeout(() => rippleEl.classList.remove('animate'), 650);\r\n  }\r\n\r\n  private _contrast(color: string, bw: boolean = true): string {\r\n\r\n    let colorHex = this._toHexadecimal(color);\r\n\r\n    if (! /(^#[0-9A-F]{6}$)|(^#[0-9A-F]{3}$)/i.test(colorHex)) {\r\n      return '#FFFFFF';\r\n    }\r\n\r\n    if (colorHex.length === 4) {\r\n      colorHex = colorHex.replace(/^#(.)(.)(.)/i, '#$1$1$2$2$3$3');\r\n    }\r\n\r\n    const r = parseInt(colorHex.substr(1, 2), 16);\r\n    const g = parseInt(colorHex.substr(3, 2), 16);\r\n    const b = parseInt(colorHex.substr(5, 2), 16);\r\n\r\n    if (bw) {\r\n      // https://github.com/lffg/yiq/blob/master/index.js\r\n      const yiq = ((r * 299) + (g * 587) + (b * 114)) / 1000;\r\n      return this._toRGBA(yiq >= 128 ? '#000000' : '#FFFFFF');\r\n    }\r\n\r\n    return this._toRGBA(`#${this._padZero(r)}${this._padZero(g)}${this._padZero(b)}`);\r\n  }\r\n\r\n  private _toHexadecimal(color: string, allow_alpha: boolean = false): string {\r\n    if (!color || (color && color.indexOf('(') == -1)) { return color; }\r\n\r\n    const parts = color.replace(/(rgba|rgb)/g, '').replace(/[\\(\\)]/g, '').split(',').map((part) => _.trim(part));\r\n\r\n    const r = this._padZero(parseInt(parts[0], 10).toString(16));\r\n    const g = this._padZero(parseInt(parts[1], 10).toString(16));\r\n    const b = this._padZero(parseInt(parts[2], 10).toString(16));\r\n\r\n    if (parts[3] && allow_alpha) {\r\n      const a = parseFloat(parseFloat(parts[3].substring(0, parts[3].length - 1)).toFixed(2));\r\n      return `#${r}${g}${b}${this._padZero((a * 255).toString(16).substring(0, 2))}`;\r\n    }\r\n\r\n    return `#${r}${g}${b}`;\r\n  }\r\n\r\n  private _toRGBA(color: string, opacity: string = this._default_ripple_opacity) {\r\n    const result = /^#?([a-f\\d]{2})([a-f\\d]{2})([a-f\\d]{2})$/i.exec(color);\r\n    return result ? `rgba(${parseInt(result[1], 16)}, ${parseInt(result[2], 16)}, ${parseInt(result[3], 16)}, ${opacity})` : color;\r\n  }\r\n\r\n  private _padZero(str: string | number, len: number = 2): string {\r\n    if (_.isNumber(str)) { str = str.toString(); }\r\n    const zeros = new Array(len).join('0');\r\n    return (zeros + str).slice(-len);\r\n  }\r\n\r\n  private _relativeOffset(child_element: HTMLElement, parent_element: HTMLElement) {\r\n    const offset_parent_element = (<HTMLElement>child_element.offsetParent);\r\n    const parent_rect = offset_parent_element.getBoundingClientRect();\r\n    const child_rect = child_element.getBoundingClientRect();\r\n\r\n    const offset = {\r\n      top: child_rect.top - parent_rect.top,\r\n      left: child_rect.left - parent_rect.left\r\n    }\r\n\r\n    // Reached top of document\r\n    if (offset_parent_element.tagName === 'BODY') { return offset; }\r\n\r\n    // Parent element contains the 'top' element we want the offset to be relative to\r\n    if (child_element.parentElement === parent_element) { return offset; }\r\n\r\n    // Reached the 'top' element we want the offset to be relative to\r\n    if (offset_parent_element === parent_element) { return offset; }\r\n\r\n    // Get parent's relative offset\r\n    const parent_offset = this._relativeOffset(offset_parent_element, parent_element);\r\n    offset.top += parent_offset.top;\r\n    offset.left += parent_offset.left;\r\n    return offset;\r\n  }\r\n}\r\n\r\n@Directive({\r\n  selector: '[with-ripple], .with-ripple'\r\n})\r\nexport class RippleDirective extends RippleDirectiveBase {\r\n  constructor(\r\n    public elementRef: ElementRef\r\n  ) {\r\n    super(elementRef)\r\n  }\r\n}\r\n"]}