@ohayojp.com/components
Version:
Common business components of ohayojp.
411 lines • 33.8 kB
JavaScript
/**
* @fileoverview added by tsickle
* Generated from: ellipsis.component.ts
* @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingRequire,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
*/
import { __decorate, __metadata } from "tslib";
import { DOCUMENT } from '@angular/common';
import { ChangeDetectionStrategy, ChangeDetectorRef, Component, ElementRef, Inject, Input, NgZone, ViewChild, ViewEncapsulation, } from '@angular/core';
import { DomSanitizer } from '@angular/platform-browser';
import { InputBoolean, InputNumber } from '@ohayojp.com/util';
import { take } from 'rxjs/operators';
export class EllipsisComponent {
/**
* @param {?} el
* @param {?} ngZone
* @param {?} dom
* @param {?} doc
* @param {?} cdr
*/
constructor(el, ngZone, dom, doc, cdr) {
this.el = el;
this.ngZone = ngZone;
this.dom = dom;
this.doc = doc;
this.cdr = cdr;
// tslint:disable-next-line:no-string-literal
this.isSupportLineClamp = this.doc.body.style['webkitLineClamp'] !== undefined;
this.inited = false;
this.type = 'default';
this.cls = {};
this.text = '';
this.targetCount = 0;
// #region fields
this.tooltip = false;
this.fullWidthRecognition = false;
this.tail = '...';
}
// #endregion
/**
* @return {?}
*/
get linsWord() {
const { targetCount, text, tail } = this;
return (targetCount > 0 ? text.substring(0, targetCount) : '') + (targetCount > 0 && targetCount < text.length ? tail : '');
}
/**
* @private
* @param {?} str
* @return {?}
*/
getStrFullLength(str) {
return str.split('').reduce((/**
* @param {?} pre
* @param {?} cur
* @return {?}
*/
(pre, cur) => {
/** @type {?} */
const charCode = cur.charCodeAt(0);
if (charCode >= 0 && charCode <= 128) {
return pre + 1;
}
return pre + 2;
}), 0);
}
/**
* @private
* @param {?} str
* @param {?} maxLength
* @return {?}
*/
cutStrByFullLength(str, maxLength) {
/** @type {?} */
let showLength = 0;
return str.split('').reduce((/**
* @param {?} pre
* @param {?} cur
* @return {?}
*/
(pre, cur) => {
/** @type {?} */
const charCode = cur.charCodeAt(0);
if (charCode >= 0 && charCode <= 128) {
showLength += 1;
}
else {
showLength += 2;
}
if (showLength <= maxLength) {
return pre + cur;
}
return pre;
}), '');
}
/**
* @private
* @param {?} targetHeight
* @param {?} mid
* @param {?} begin
* @param {?} end
* @param {?} text
* @param {?} node
* @return {?}
*/
bisection(targetHeight, mid, begin, end, text, node) {
/** @type {?} */
const suffix = this.tail;
node.innerHTML = text.substring(0, mid) + suffix;
/** @type {?} */
let sh = node.offsetHeight;
if (sh <= targetHeight) {
node.innerHTML = text.substring(0, mid + 1) + suffix;
sh = node.offsetHeight;
if (sh > targetHeight || mid === begin) {
return mid;
}
begin = mid;
mid = end - begin === 1 ? begin + 1 : Math.floor((end - begin) / 2) + begin;
return this.bisection(targetHeight, mid, begin, end, text, node);
}
if (mid - 1 < 0) {
return mid;
}
node.innerHTML = text.substring(0, mid - 1) + suffix;
sh = node.offsetHeight;
if (sh <= targetHeight) {
return mid - 1;
}
end = mid;
mid = Math.floor((end - begin) / 2) + begin;
return this.bisection(targetHeight, mid, begin, end, text, node);
}
/**
* @private
* @return {?}
*/
genType() {
const { lines, length, isSupportLineClamp } = this;
this.cls = {
ellipsis: true,
ellipsis__lines: lines && !isSupportLineClamp,
'ellipsis__line-clamp': lines && isSupportLineClamp,
};
if (!lines && !length) {
this.type = 'default';
}
else if (!lines) {
this.type = 'length';
}
else if (isSupportLineClamp) {
this.type = 'line-clamp';
}
else {
this.type = 'line';
}
}
/**
* @private
* @return {?}
*/
gen() {
const { type, lines, length, fullWidthRecognition, tail, orgEl, cdr, ngZone } = this;
if (type === 'length') {
/** @type {?} */
const el = (/** @type {?} */ (orgEl.nativeElement));
if (el.children.length > 0) {
throw new Error('Ellipsis content must be string.');
}
/** @type {?} */
const lengthText = (/** @type {?} */ (el.textContent));
/** @type {?} */
const textLength = fullWidthRecognition ? this.getStrFullLength(lengthText) : lengthText.length;
if (textLength <= length || length < 0) {
this.text = lengthText;
}
else {
/** @type {?} */
let displayText;
if (length - tail.length <= 0) {
displayText = '';
}
else {
displayText = fullWidthRecognition ? this.cutStrByFullLength(lengthText, length) : lengthText.slice(0, length);
}
this.text = displayText + tail;
}
ngZone.run((/**
* @return {?}
*/
() => cdr.detectChanges()));
}
else if (type === 'line') {
const { shadowOrgEl, shadowTextEl } = this;
/** @type {?} */
const orgNode = (/** @type {?} */ (shadowOrgEl.nativeElement));
/** @type {?} */
const lineText = orgNode.innerText || (/** @type {?} */ (orgNode.textContent));
/** @type {?} */
const lineHeight = parseInt((/** @type {?} */ (getComputedStyle(this.getEl('.ellipsis')).lineHeight)), 10);
/** @type {?} */
const targetHeight = lines * lineHeight;
this.getEl('.ellipsis__handle').style.height = `${targetHeight}px`;
if (orgNode.offsetHeight <= targetHeight) {
this.text = lineText;
this.targetCount = lineText.length;
}
else {
// bisection
/** @type {?} */
const len = lineText.length;
/** @type {?} */
const mid = Math.ceil(len / 2);
/** @type {?} */
const count = this.bisection(targetHeight, mid, 0, len, lineText, shadowTextEl.nativeElement.firstChild);
this.text = lineText;
this.targetCount = count;
}
ngZone.run((/**
* @return {?}
*/
() => cdr.detectChanges()));
}
}
/**
* @private
* @param {?} cls
* @return {?}
*/
getEl(cls) {
return this.el.nativeElement.querySelector(cls);
}
/**
* @private
* @param {?} fn
* @return {?}
*/
executeOnStable(fn) {
if (this.ngZone.isStable) {
fn();
}
else {
this.ngZone.onStable.asObservable().pipe(take(1)).subscribe(fn);
}
}
/**
* @return {?}
*/
refresh() {
this.genType();
const { type, dom, orgEl, cdr } = this;
/** @type {?} */
const html = orgEl.nativeElement.innerHTML;
this.orgHtml = dom.bypassSecurityTrustHtml(html);
cdr.detectChanges();
this.executeOnStable((/**
* @return {?}
*/
() => {
this.gen();
if (type !== 'line') {
/** @type {?} */
const el = this.getEl('.ellipsis');
if (el) {
el.innerHTML = html;
}
}
}));
}
/**
* @return {?}
*/
ngAfterViewInit() {
this.inited = true;
this.refresh();
}
/**
* @return {?}
*/
ngOnChanges() {
if (this.inited) {
this.refresh();
}
}
}
EllipsisComponent.decorators = [
{
type: Component, args: [{
selector: 'ellipsis',
exportAs: 'ellipsis',
template: "<div (cdkObserveContent)=\"refresh()\" #orgEl style=\"display: none;\"><ng-content></ng-content></div>\n<ng-template #tooltipTpl let-con>\n <span *ngIf=\"tooltip; else con\" nz-tooltip [nzTooltipTitle]=\"titleTpl\" [nzTooltipOverlayStyle]=\"{ 'overflow-wrap': 'break-word', 'word-wrap': 'break-word' }\">\n <ng-container *ngTemplateOutlet=\"con\"></ng-container>\n <ng-template #titleTpl><div [innerHTML]=\"orgHtml\"></div></ng-template>\n </span>\n</ng-template>\n<ng-container [ngSwitch]=\"type\">\n <span *ngSwitchCase=\"'default'\" [ngClass]=\"cls\"></span>\n <ng-container *ngSwitchCase=\"'length'\">\n <ng-template [ngTemplateOutlet]=\"tooltipTpl\" [ngTemplateOutletContext]=\"{ $implicit: lengthTpl }\"></ng-template>\n <ng-template #lengthTpl>{{ text }}</ng-template>\n </ng-container>\n <ng-container *ngSwitchCase=\"'line-clamp'\">\n <ng-template [ngTemplateOutlet]=\"tooltipTpl\" [ngTemplateOutletContext]=\"{ $implicit: lineClampTpl }\"></ng-template>\n <ng-template #lineClampTpl>\n <div [ngClass]=\"cls\" [ngStyle]=\"{ '-webkit-line-clamp': lines, '-webkit-box-orient': 'vertical' }\"></div>\n </ng-template>\n </ng-container>\n <div *ngSwitchCase=\"'line'\" [ngClass]=\"cls\">\n <div class=\"ellipsis__handle\">\n <ng-template [ngTemplateOutlet]=\"tooltipTpl\" [ngTemplateOutletContext]=\"{ $implicit: lineTpl }\"></ng-template>\n <ng-template #lineTpl>{{ linsWord }}</ng-template>\n <div class=\"ellipsis__shadow\" #shadowOrgEl [innerHTML]=\"orgHtml\"></div>\n <div class=\"ellipsis__shadow\" #shadowTextEl>\n <span>{{ text }}</span>\n </div>\n </div>\n </div>\n</ng-container>\n",
preserveWhitespaces: false,
changeDetection: ChangeDetectionStrategy.OnPush,
encapsulation: ViewEncapsulation.None
}]
}
];
/** @nocollapse */
EllipsisComponent.ctorParameters = () => [
{ type: ElementRef },
{ type: NgZone },
{ type: DomSanitizer },
{ type: Document, decorators: [{ type: Inject, args: [DOCUMENT,] }] },
{ type: ChangeDetectorRef }
];
EllipsisComponent.propDecorators = {
orgEl: [{ type: ViewChild, args: ['orgEl', { static: false },] }],
shadowOrgEl: [{ type: ViewChild, args: ['shadowOrgEl', { static: false },] }],
shadowTextEl: [{ type: ViewChild, args: ['shadowTextEl', { static: false },] }],
tooltip: [{ type: Input }],
length: [{ type: Input }],
lines: [{ type: Input }],
fullWidthRecognition: [{ type: Input }],
tail: [{ type: Input }]
};
__decorate([
InputBoolean(),
__metadata("design:type", Object)
], EllipsisComponent.prototype, "tooltip", void 0);
__decorate([
InputNumber(null),
__metadata("design:type", Number)
], EllipsisComponent.prototype, "length", void 0);
__decorate([
InputNumber(null),
__metadata("design:type", Number)
], EllipsisComponent.prototype, "lines", void 0);
__decorate([
InputBoolean(),
__metadata("design:type", Object)
], EllipsisComponent.prototype, "fullWidthRecognition", void 0);
if (false) {
/** @type {?} */
EllipsisComponent.ngAcceptInputType_tooltip;
/** @type {?} */
EllipsisComponent.ngAcceptInputType_length;
/** @type {?} */
EllipsisComponent.ngAcceptInputType_lines;
/** @type {?} */
EllipsisComponent.ngAcceptInputType_fullWidthRecognition;
/**
* @type {?}
* @private
*/
EllipsisComponent.prototype.isSupportLineClamp;
/**
* @type {?}
* @private
*/
EllipsisComponent.prototype.orgEl;
/**
* @type {?}
* @private
*/
EllipsisComponent.prototype.shadowOrgEl;
/**
* @type {?}
* @private
*/
EllipsisComponent.prototype.shadowTextEl;
/**
* @type {?}
* @private
*/
EllipsisComponent.prototype.inited;
/** @type {?} */
EllipsisComponent.prototype.orgHtml;
/** @type {?} */
EllipsisComponent.prototype.type;
/** @type {?} */
EllipsisComponent.prototype.cls;
/** @type {?} */
EllipsisComponent.prototype.text;
/** @type {?} */
EllipsisComponent.prototype.targetCount;
/** @type {?} */
EllipsisComponent.prototype.tooltip;
/** @type {?} */
EllipsisComponent.prototype.length;
/** @type {?} */
EllipsisComponent.prototype.lines;
/** @type {?} */
EllipsisComponent.prototype.fullWidthRecognition;
/** @type {?} */
EllipsisComponent.prototype.tail;
/**
* @type {?}
* @private
*/
EllipsisComponent.prototype.el;
/**
* @type {?}
* @private
*/
EllipsisComponent.prototype.ngZone;
/**
* @type {?}
* @private
*/
EllipsisComponent.prototype.dom;
/**
* @type {?}
* @private
*/
EllipsisComponent.prototype.doc;
/**
* @type {?}
* @private
*/
EllipsisComponent.prototype.cdr;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ellipsis.component.js","sourceRoot":"C:/android/@ohayojp/packages/components/ellipsis/","sources":["ellipsis.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { DOCUMENT } from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  Inject,\n  Input,\n  NgZone,\n  OnChanges,\n  ViewChild,\n  ViewEncapsulation,\n} from '@angular/core';\nimport { DomSanitizer, SafeHtml } from '@angular/platform-browser';\nimport { BooleanInput, InputBoolean, InputNumber, NumberInput } from '@ohayo/util';\nimport { take } from 'rxjs/operators';\n\n@Component({\n  selector: 'ellipsis',\n  exportAs: 'ellipsis',\n  templateUrl: './ellipsis.component.html',\n  preserveWhitespaces: false,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n})\nexport class EllipsisComponent implements AfterViewInit, OnChanges {\n  static ngAcceptInputType_tooltip: BooleanInput;\n  static ngAcceptInputType_length: NumberInput;\n  static ngAcceptInputType_lines: NumberInput;\n  static ngAcceptInputType_fullWidthRecognition: BooleanInput;\n\n  // tslint:disable-next-line:no-string-literal\n  private isSupportLineClamp = this.doc.body.style['webkitLineClamp'] !== undefined;\n  @ViewChild('orgEl', { static: false }) private orgEl: ElementRef;\n  @ViewChild('shadowOrgEl', { static: false }) private shadowOrgEl: ElementRef;\n  @ViewChild('shadowTextEl', { static: false }) private shadowTextEl: ElementRef;\n  private inited = false;\n  orgHtml: SafeHtml;\n  type = 'default';\n  cls = {};\n  text = '';\n  targetCount = 0;\n\n  // #region fields\n\n  @Input() @InputBoolean() tooltip = false;\n  @Input() @InputNumber(null) length: number;\n  @Input() @InputNumber(null) lines: number;\n  @Input() @InputBoolean() fullWidthRecognition = false;\n  @Input() tail = '...';\n\n  // #endregion\n\n  get linsWord(): string {\n    const { targetCount, text, tail } = this;\n    return (targetCount > 0 ? text.substring(0, targetCount) : '') + (targetCount > 0 && targetCount < text.length ? tail : '');\n  }\n\n  constructor(\n    private el: ElementRef,\n    private ngZone: NgZone,\n    private dom: DomSanitizer,\n    @Inject(DOCUMENT) private doc: Document,\n    private cdr: ChangeDetectorRef,\n  ) { }\n\n  private getStrFullLength(str: string): number {\n    return str.split('').reduce((pre, cur) => {\n      const charCode = cur.charCodeAt(0);\n      if (charCode >= 0 && charCode <= 128) {\n        return pre + 1;\n      }\n      return pre + 2;\n    }, 0);\n  }\n\n  private cutStrByFullLength(str: string, maxLength: number): string {\n    let showLength = 0;\n    return str.split('').reduce((pre, cur) => {\n      const charCode = cur.charCodeAt(0);\n      if (charCode >= 0 && charCode <= 128) {\n        showLength += 1;\n      } else {\n        showLength += 2;\n      }\n      if (showLength <= maxLength) {\n        return pre + cur;\n      }\n      return pre;\n    }, '');\n  }\n\n  private bisection(targetHeight: number, mid: number, begin: number, end: number, text: string, node: HTMLElement): number {\n    const suffix = this.tail;\n    node.innerHTML = text.substring(0, mid) + suffix;\n    let sh = node.offsetHeight;\n\n    if (sh <= targetHeight) {\n      node.innerHTML = text.substring(0, mid + 1) + suffix;\n      sh = node.offsetHeight;\n      if (sh > targetHeight || mid === begin) {\n        return mid;\n      }\n      begin = mid;\n      mid = end - begin === 1 ? begin + 1 : Math.floor((end - begin) / 2) + begin;\n      return this.bisection(targetHeight, mid, begin, end, text, node);\n    }\n    if (mid - 1 < 0) {\n      return mid;\n    }\n    node.innerHTML = text.substring(0, mid - 1) + suffix;\n    sh = node.offsetHeight;\n    if (sh <= targetHeight) {\n      return mid - 1;\n    }\n    end = mid;\n    mid = Math.floor((end - begin) / 2) + begin;\n    return this.bisection(targetHeight, mid, begin, end, text, node);\n  }\n\n  private genType(): void {\n    const { lines, length, isSupportLineClamp } = this;\n    this.cls = {\n      ellipsis: true,\n      ellipsis__lines: lines && !isSupportLineClamp,\n      'ellipsis__line-clamp': lines && isSupportLineClamp,\n    };\n    if (!lines && !length) {\n      this.type = 'default';\n    } else if (!lines) {\n      this.type = 'length';\n    } else if (isSupportLineClamp) {\n      this.type = 'line-clamp';\n    } else {\n      this.type = 'line';\n    }\n  }\n\n  private gen(): void {\n    const { type, lines, length, fullWidthRecognition, tail, orgEl, cdr, ngZone } = this;\n    if (type === 'length') {\n      const el = orgEl.nativeElement as HTMLElement;\n      if (el.children.length > 0) {\n        throw new Error('Ellipsis content must be string.');\n      }\n      const lengthText = el.textContent!;\n      const textLength = fullWidthRecognition ? this.getStrFullLength(lengthText) : lengthText.length;\n      if (textLength <= length || length < 0) {\n        this.text = lengthText;\n      } else {\n        let displayText: string;\n        if (length - tail.length <= 0) {\n          displayText = '';\n        } else {\n          displayText = fullWidthRecognition ? this.cutStrByFullLength(lengthText, length) : lengthText.slice(0, length);\n        }\n        this.text = displayText + tail;\n      }\n      ngZone.run(() => cdr.detectChanges());\n    } else if (type === 'line') {\n      const { shadowOrgEl, shadowTextEl } = this;\n      const orgNode = shadowOrgEl.nativeElement as HTMLElement;\n      const lineText = orgNode.innerText || orgNode.textContent!;\n      const lineHeight = parseInt(getComputedStyle(this.getEl('.ellipsis')).lineHeight!, 10);\n      const targetHeight = lines * lineHeight;\n      this.getEl('.ellipsis__handle').style.height = `${targetHeight}px`;\n\n      if (orgNode.offsetHeight <= targetHeight) {\n        this.text = lineText;\n        this.targetCount = lineText.length;\n      } else {\n        // bisection\n        const len = lineText.length;\n        const mid = Math.ceil(len / 2);\n\n        const count = this.bisection(targetHeight, mid, 0, len, lineText, shadowTextEl.nativeElement.firstChild);\n        this.text = lineText;\n        this.targetCount = count;\n      }\n      ngZone.run(() => cdr.detectChanges());\n    }\n  }\n\n  private getEl(cls: string): HTMLElement {\n    return this.el.nativeElement.querySelector(cls);\n  }\n\n  private executeOnStable(fn: () => void): void {\n    if (this.ngZone.isStable) {\n      fn();\n    } else {\n      this.ngZone.onStable.asObservable().pipe(take(1)).subscribe(fn);\n    }\n  }\n\n  refresh(): void {\n    this.genType();\n    const { type, dom, orgEl, cdr } = this;\n    const html = orgEl.nativeElement.innerHTML;\n    this.orgHtml = dom.bypassSecurityTrustHtml(html);\n    cdr.detectChanges();\n    this.executeOnStable(() => {\n      this.gen();\n      if (type !== 'line') {\n        const el = this.getEl('.ellipsis');\n        if (el) {\n          el.innerHTML = html;\n        }\n      }\n    });\n  }\n\n  ngAfterViewInit(): void {\n    this.inited = true;\n    this.refresh();\n  }\n\n  ngOnChanges(): void {\n    if (this.inited) {\n      this.refresh();\n    }\n  }\n}\n"]}