@uex/web-extensions
Version:
Uex extensions for Angular 6+ web projects
268 lines • 25.3 kB
JavaScript
/**
* @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"]}