ngx-nested-ellipsis
Advanced tools
Comparing version
@@ -423,3 +423,5 @@ (function (global, factory) { | ||
width: this.elem.clientWidth, | ||
height: this.elem.clientHeight | ||
height: this.elem.clientHeight, | ||
scrollWidth: this.elem.scrollWidth, | ||
scrollHeight: this.elem.scrollHeight | ||
}; | ||
@@ -551,3 +553,6 @@ this.applyEllipsis(); | ||
window.requestAnimationFrame(function () { | ||
if (_this.previousDimensions.width !== _this.elem.clientWidth || _this.previousDimensions.height !== _this.elem.clientHeight) { | ||
if (_this.previousDimensions.width !== _this.elem.clientWidth | ||
|| _this.previousDimensions.height !== _this.elem.clientHeight | ||
|| _this.previousDimensions.scrollWidth !== _this.elem.scrollWidth | ||
|| _this.previousDimensions.scrollHeight !== _this.elem.scrollHeight) { | ||
_this.ngZone.run(function () { | ||
@@ -558,2 +563,4 @@ _this.applyEllipsis(); | ||
_this.previousDimensions.height = _this.elem.clientHeight; | ||
_this.previousDimensions.scrollWidth = _this.elem.scrollWidth; | ||
_this.previousDimensions.scrollHeight = _this.elem.scrollHeight; | ||
} | ||
@@ -560,0 +567,0 @@ }); |
@@ -15,3 +15,3 @@ !function(e,t){"object"==typeof exports&&"undefined"!=typeof module?t(exports,require("@angular/core"),require("resize-observer-polyfill"),require("@angular/common"),require("rxjs"),require("rxjs/operators")):"function"==typeof define&&define.amd?define("ngx-nested-ellipsis",["exports","@angular/core","resize-observer-polyfill","@angular/common","rxjs","rxjs/operators"],t):t((e="undefined"!=typeof globalThis?globalThis:e||self)["ngx-nested-ellipsis"]={},e.ng.core,e.ResizeObserver,e.ng.common,e.rxjs,e.rxjs.operators)}(this,(function(e,t,i,n,r,s){"use strict";function o(e){return e&&"object"==typeof e&&"default"in e?e:{default:e}}var a=o(i); | ||
PERFORMANCE OF THIS SOFTWARE. | ||
***************************************************************************** */Object.create;function l(e){var t="function"==typeof Symbol&&Symbol.iterator,i=t&&e[t],n=0;if(i)return i.call(e);if(e&&"number"==typeof e.length)return{next:function(){return e&&n>=e.length&&(e=void 0),{value:e&&e[n++],done:!e}}};throw new TypeError(t?"Object is not iterable.":"Symbol.iterator is not defined.")}function c(e,t){var i="function"==typeof Symbol&&e[Symbol.iterator];if(!i)return e;var n,r,s=i.call(e),o=[];try{for(;(void 0===t||t-- >0)&&!(n=s.next()).done;)o.push(n.value)}catch(e){r={error:e}}finally{try{n&&!n.done&&(i=s.return)&&i.call(s)}finally{if(r)throw r.error}}return o}function h(){for(var e=[],t=0;t<arguments.length;t++)e=e.concat(c(arguments[t]));return e}Object.create;var d=function(e){this.elementRef=e};d.decorators=[{type:t.Component,args:[{selector:"nested-ellipsis-content",template:"\n <ng-content></ng-content>\n ",styles:["\n :host {\n display: block;\n width: 100%;\n height: 100%;\n overflow: hidden;\n }\n "]}]}],d.ctorParameters=function(){return[{type:t.ElementRef}]};var p=function(){function e(e,i,n,s,o,a){this.templateRef=e,this.viewContainer=i,this.resolver=n,this.renderer=s,this.ngZone=o,this.platformId=a,this.removeResizeListeners$=new r.Subject,this.change=new t.EventEmitter}return e.numericBinarySearch=function(e,t){for(var i,n=0,r=e,s=-1;n<=r;){t(i=Math.floor((n+r)/2))?(s=i,n=i+1):r=i-1}return s},e.prototype.flattenTextAndElementNodes=function(e){for(var t=[],i=0;i<e.childNodes.length;i++){var n=e.childNodes.item(i);(n instanceof HTMLElement||n instanceof CharacterData)&&(t.push(n),n instanceof HTMLElement&&t.push.apply(t,h(this.flattenTextAndElementNodes(n))))}return t},e.prototype.ngOnInit=function(){n.isPlatformBrowser(this.platformId)&&("boolean"!=typeof this.active&&(this.active=!0),void 0===this.indicator&&(this.indicator="..."),void 0===this.resizeDetection&&(this.resizeDetection="resize-observer"),this.wordBoundaries||(this.wordBoundaries=""),this.wordBoundaries="["+this.wordBoundaries.replace(/[-\/\\^$*+?.()|[\]{}]/g,"\\$&")+"]",this.compFactory=this.resolver.resolveComponentFactory(d),this.restoreView(),this.previousDimensions={width:this.elem.clientWidth,height:this.elem.clientHeight},this.applyEllipsis())},e.prototype.ngOnDestroy=function(){this.removeResizeListeners$.next(),this.removeResizeListeners$.complete()},e.prototype.ngAfterViewChecked=function(){"manual"!==this.resizeDetection&&this.templatesHaveChanged&&this.applyEllipsis()},e.prototype.nodesToHtml=function(e){var t=this.renderer.createElement("div");return t.append.apply(t,h(e.map((function(e){return e.cloneNode(!0)})))),t.innerHTML},e.prototype.templatesToHtml=function(e,t){var i=this.nodesToHtml(e.rootNodes);return i+=t?this.nodesToHtml(t.rootNodes):this.indicator},Object.defineProperty(e.prototype,"templatesHaveChanged",{get:function(){if(!this.templateView||!this.previousTemplateHtml)return!1;var e=this.templateRef.createEmbeddedView({});e.detectChanges();var t="string"!=typeof this.indicator?this.indicator.createEmbeddedView({}):null;t&&t.detectChanges();var i=this.templatesToHtml(e,t);return this.previousTemplateHtml!==i},enumerable:!1,configurable:!0}),e.prototype.restoreView=function(){this.viewContainer.clear(),this.templateView=this.templateRef.createEmbeddedView({}),this.templateView.detectChanges();var e=this.viewContainer.createComponent(this.compFactory,null,this.viewContainer.injector,[this.templateView.rootNodes]);this.elem=e.instance.elementRef.nativeElement,this.initialTextLength=this.currentLength,this.indicatorView="string"!=typeof this.indicator?this.indicator.createEmbeddedView({}):null,this.indicatorView&&this.indicatorView.detectChanges()},e.prototype.addResizeListener=function(){switch(this.resizeDetection){case"manual":break;case"window":this.addWindowResizeListener();break;default:"undefined"!=typeof console&&console.warn("\n No such ellipsisResizeDetection strategy: '"+this.resizeDetection+"'.\n Using 'resize-observer' instead.\n "),this.resizeDetection="resize-observer";case"resize-observer":this.addResizeObserver()}},e.prototype.addWindowResizeListener=function(){var e=this,t=this.renderer.listen("window","resize",(function(){e.ngZone.run((function(){e.applyEllipsis()}))}));this.removeResizeListeners$.pipe(s.take(1)).subscribe((function(){return t()}))},e.prototype.addResizeObserver=function(){var e=this,t=new a.default((function(){window.requestAnimationFrame((function(){e.previousDimensions.width===e.elem.clientWidth&&e.previousDimensions.height===e.elem.clientHeight||(e.ngZone.run((function(){e.applyEllipsis()})),e.previousDimensions.width=e.elem.clientWidth,e.previousDimensions.height=e.elem.clientHeight)}))}));t.observe(this.elem),this.removeResizeListeners$.pipe(s.take(1)).subscribe((function(){return t.disconnect()}))},e.prototype.truncateContents=function(e){this.restoreView();for(var t,i=this.flattenTextAndElementNodes(this.elem).filter((function(e){return[Node.TEXT_NODE,Node.ELEMENT_NODE].includes(e.nodeType)})),n=-1,r=this.initialTextLength,s=i.length-1;s>=0;s--){if((a=i[s])instanceof CharacterData?r-=a.data.length:r--,r<=e){if(a instanceof CharacterData)if("[]"!==this.wordBoundaries||this.mayTruncateAtFn){if(e-r!==a.data.length){for(var o=e-r-1;o>0&&("[]"!==this.wordBoundaries&&!a.data.charAt(o).match(this.wordBoundaries)||this.mayTruncateAtFn&&!this.mayTruncateAtFn(a,o));)o--;if(r>0&&0===o)continue;a.data=a.data.substr(0,o)}}else a.data=a.data.substr(0,e-r);n=s,t=a;break}}for(s=n+1;s<i.length;s++){var a;""!==(a=i[s]).textContent&&a.parentElement!==this.elem&&1===a.parentElement.childNodes.length?a.parentElement.remove():a.remove()}return this.currentLength!==this.initialTextLength?t:null},Object.defineProperty(e.prototype,"currentLength",{get:function(){return this.flattenTextAndElementNodes(this.elem).filter((function(e){return[Node.TEXT_NODE,Node.ELEMENT_NODE].includes(e.nodeType)})).map((function(e){return e instanceof CharacterData?e.data.length:1})).reduce((function(e,t){return e+t}),0)},enumerable:!1,configurable:!0}),e.prototype.truncateText=function(e){var t,i,n=this.truncateContents(e);if(n)if(this.indicatorView)try{for(var r=l(this.indicatorView.rootNodes),s=r.next();!s.done;s=r.next()){var o=s.value;this.renderer.appendChild(this.elem,o)}}catch(e){t={error:e}}finally{try{s&&!s.done&&(i=r.return)&&i.call(r)}finally{if(t)throw t.error}}else n instanceof CharacterData?n.data+=this.indicator:this.renderer.appendChild(this.elem,this.renderer.createText(this.indicator))},e.prototype.applyEllipsis=function(){var t=this;if(this.removeResizeListeners$.next(),this.restoreView(),this.previousTemplateHtml=this.templatesToHtml(this.templateView,this.indicatorView),this.active){var i=e.numericBinarySearch(this.initialTextLength,(function(e){return t.truncateText(e),!t.isOverflowing}));if(this.truncateText(i),this.addResizeListener(),this.change.observers.length>0){var n=this.currentLength;this.change.emit(n===this.initialTextLength?null:n)}}},Object.defineProperty(e.prototype,"isOverflowing",{get:function(){var e=this.elem.style.overflow;e&&"visible"!==e||(this.elem.style.overflow="hidden");var t=this.elem.clientWidth<this.elem.scrollWidth-1||this.elem.clientHeight<this.elem.scrollHeight-1;return this.elem.style.overflow=e,t},enumerable:!1,configurable:!0}),e}();p.decorators=[{type:t.Directive,args:[{selector:"[nestedEllipsis]",exportAs:"ngxNestedEllipsis"}]}],p.ctorParameters=function(){return[{type:t.TemplateRef},{type:t.ViewContainerRef},{type:t.ComponentFactoryResolver},{type:t.Renderer2},{type:t.NgZone},{type:Object,decorators:[{type:t.Inject,args:[t.PLATFORM_ID]}]}]},p.propDecorators={active:[{type:t.Input,args:["nestedEllipsis"]}],indicator:[{type:t.Input,args:["nestedEllipsisIndicator"]}],wordBoundaries:[{type:t.Input,args:["nestedEllipsisWordBoundaries"]}],mayTruncateAtFn:[{type:t.Input,args:["nestedEllipsisMayTruncateAtFn"]}],resizeDetection:[{type:t.Input,args:["nestedEllipsisResizeDetection"]}],change:[{type:t.Output,args:["nestedEllipsisChange"]}]};var u=function(){};u.decorators=[{type:t.NgModule,args:[{imports:[],declarations:[p,d],entryComponents:[d],exports:[p]}]}],e.NestedEllipsisContentComponent=d,e.NestedEllipsisDirective=p,e.NestedEllipsisModule=u,Object.defineProperty(e,"__esModule",{value:!0})})); | ||
***************************************************************************** */Object.create;function l(e){var t="function"==typeof Symbol&&Symbol.iterator,i=t&&e[t],n=0;if(i)return i.call(e);if(e&&"number"==typeof e.length)return{next:function(){return e&&n>=e.length&&(e=void 0),{value:e&&e[n++],done:!e}}};throw new TypeError(t?"Object is not iterable.":"Symbol.iterator is not defined.")}function c(e,t){var i="function"==typeof Symbol&&e[Symbol.iterator];if(!i)return e;var n,r,s=i.call(e),o=[];try{for(;(void 0===t||t-- >0)&&!(n=s.next()).done;)o.push(n.value)}catch(e){r={error:e}}finally{try{n&&!n.done&&(i=s.return)&&i.call(s)}finally{if(r)throw r.error}}return o}function h(){for(var e=[],t=0;t<arguments.length;t++)e=e.concat(c(arguments[t]));return e}Object.create;var d=function(e){this.elementRef=e};d.decorators=[{type:t.Component,args:[{selector:"nested-ellipsis-content",template:"\n <ng-content></ng-content>\n ",styles:["\n :host {\n display: block;\n width: 100%;\n height: 100%;\n overflow: hidden;\n }\n "]}]}],d.ctorParameters=function(){return[{type:t.ElementRef}]};var p=function(){function e(e,i,n,s,o,a){this.templateRef=e,this.viewContainer=i,this.resolver=n,this.renderer=s,this.ngZone=o,this.platformId=a,this.removeResizeListeners$=new r.Subject,this.change=new t.EventEmitter}return e.numericBinarySearch=function(e,t){for(var i,n=0,r=e,s=-1;n<=r;){t(i=Math.floor((n+r)/2))?(s=i,n=i+1):r=i-1}return s},e.prototype.flattenTextAndElementNodes=function(e){for(var t=[],i=0;i<e.childNodes.length;i++){var n=e.childNodes.item(i);(n instanceof HTMLElement||n instanceof CharacterData)&&(t.push(n),n instanceof HTMLElement&&t.push.apply(t,h(this.flattenTextAndElementNodes(n))))}return t},e.prototype.ngOnInit=function(){n.isPlatformBrowser(this.platformId)&&("boolean"!=typeof this.active&&(this.active=!0),void 0===this.indicator&&(this.indicator="..."),void 0===this.resizeDetection&&(this.resizeDetection="resize-observer"),this.wordBoundaries||(this.wordBoundaries=""),this.wordBoundaries="["+this.wordBoundaries.replace(/[-\/\\^$*+?.()|[\]{}]/g,"\\$&")+"]",this.compFactory=this.resolver.resolveComponentFactory(d),this.restoreView(),this.previousDimensions={width:this.elem.clientWidth,height:this.elem.clientHeight,scrollWidth:this.elem.scrollWidth,scrollHeight:this.elem.scrollHeight},this.applyEllipsis())},e.prototype.ngOnDestroy=function(){this.removeResizeListeners$.next(),this.removeResizeListeners$.complete()},e.prototype.ngAfterViewChecked=function(){"manual"!==this.resizeDetection&&this.templatesHaveChanged&&this.applyEllipsis()},e.prototype.nodesToHtml=function(e){var t=this.renderer.createElement("div");return t.append.apply(t,h(e.map((function(e){return e.cloneNode(!0)})))),t.innerHTML},e.prototype.templatesToHtml=function(e,t){var i=this.nodesToHtml(e.rootNodes);return i+=t?this.nodesToHtml(t.rootNodes):this.indicator},Object.defineProperty(e.prototype,"templatesHaveChanged",{get:function(){if(!this.templateView||!this.previousTemplateHtml)return!1;var e=this.templateRef.createEmbeddedView({});e.detectChanges();var t="string"!=typeof this.indicator?this.indicator.createEmbeddedView({}):null;t&&t.detectChanges();var i=this.templatesToHtml(e,t);return this.previousTemplateHtml!==i},enumerable:!1,configurable:!0}),e.prototype.restoreView=function(){this.viewContainer.clear(),this.templateView=this.templateRef.createEmbeddedView({}),this.templateView.detectChanges();var e=this.viewContainer.createComponent(this.compFactory,null,this.viewContainer.injector,[this.templateView.rootNodes]);this.elem=e.instance.elementRef.nativeElement,this.initialTextLength=this.currentLength,this.indicatorView="string"!=typeof this.indicator?this.indicator.createEmbeddedView({}):null,this.indicatorView&&this.indicatorView.detectChanges()},e.prototype.addResizeListener=function(){switch(this.resizeDetection){case"manual":break;case"window":this.addWindowResizeListener();break;default:"undefined"!=typeof console&&console.warn("\n No such ellipsisResizeDetection strategy: '"+this.resizeDetection+"'.\n Using 'resize-observer' instead.\n "),this.resizeDetection="resize-observer";case"resize-observer":this.addResizeObserver()}},e.prototype.addWindowResizeListener=function(){var e=this,t=this.renderer.listen("window","resize",(function(){e.ngZone.run((function(){e.applyEllipsis()}))}));this.removeResizeListeners$.pipe(s.take(1)).subscribe((function(){return t()}))},e.prototype.addResizeObserver=function(){var e=this,t=new a.default((function(){window.requestAnimationFrame((function(){e.previousDimensions.width===e.elem.clientWidth&&e.previousDimensions.height===e.elem.clientHeight&&e.previousDimensions.scrollWidth===e.elem.scrollWidth&&e.previousDimensions.scrollHeight===e.elem.scrollHeight||(e.ngZone.run((function(){e.applyEllipsis()})),e.previousDimensions.width=e.elem.clientWidth,e.previousDimensions.height=e.elem.clientHeight,e.previousDimensions.scrollWidth=e.elem.scrollWidth,e.previousDimensions.scrollHeight=e.elem.scrollHeight)}))}));t.observe(this.elem),this.removeResizeListeners$.pipe(s.take(1)).subscribe((function(){return t.disconnect()}))},e.prototype.truncateContents=function(e){this.restoreView();for(var t,i=this.flattenTextAndElementNodes(this.elem).filter((function(e){return[Node.TEXT_NODE,Node.ELEMENT_NODE].includes(e.nodeType)})),n=-1,r=this.initialTextLength,s=i.length-1;s>=0;s--){if((a=i[s])instanceof CharacterData?r-=a.data.length:r--,r<=e){if(a instanceof CharacterData)if("[]"!==this.wordBoundaries||this.mayTruncateAtFn){if(e-r!==a.data.length){for(var o=e-r-1;o>0&&("[]"!==this.wordBoundaries&&!a.data.charAt(o).match(this.wordBoundaries)||this.mayTruncateAtFn&&!this.mayTruncateAtFn(a,o));)o--;if(r>0&&0===o)continue;a.data=a.data.substr(0,o)}}else a.data=a.data.substr(0,e-r);n=s,t=a;break}}for(s=n+1;s<i.length;s++){var a;""!==(a=i[s]).textContent&&a.parentElement!==this.elem&&1===a.parentElement.childNodes.length?a.parentElement.remove():a.remove()}return this.currentLength!==this.initialTextLength?t:null},Object.defineProperty(e.prototype,"currentLength",{get:function(){return this.flattenTextAndElementNodes(this.elem).filter((function(e){return[Node.TEXT_NODE,Node.ELEMENT_NODE].includes(e.nodeType)})).map((function(e){return e instanceof CharacterData?e.data.length:1})).reduce((function(e,t){return e+t}),0)},enumerable:!1,configurable:!0}),e.prototype.truncateText=function(e){var t,i,n=this.truncateContents(e);if(n)if(this.indicatorView)try{for(var r=l(this.indicatorView.rootNodes),s=r.next();!s.done;s=r.next()){var o=s.value;this.renderer.appendChild(this.elem,o)}}catch(e){t={error:e}}finally{try{s&&!s.done&&(i=r.return)&&i.call(r)}finally{if(t)throw t.error}}else n instanceof CharacterData?n.data+=this.indicator:this.renderer.appendChild(this.elem,this.renderer.createText(this.indicator))},e.prototype.applyEllipsis=function(){var t=this;if(this.removeResizeListeners$.next(),this.restoreView(),this.previousTemplateHtml=this.templatesToHtml(this.templateView,this.indicatorView),this.active){var i=e.numericBinarySearch(this.initialTextLength,(function(e){return t.truncateText(e),!t.isOverflowing}));if(this.truncateText(i),this.addResizeListener(),this.change.observers.length>0){var n=this.currentLength;this.change.emit(n===this.initialTextLength?null:n)}}},Object.defineProperty(e.prototype,"isOverflowing",{get:function(){var e=this.elem.style.overflow;e&&"visible"!==e||(this.elem.style.overflow="hidden");var t=this.elem.clientWidth<this.elem.scrollWidth-1||this.elem.clientHeight<this.elem.scrollHeight-1;return this.elem.style.overflow=e,t},enumerable:!1,configurable:!0}),e}();p.decorators=[{type:t.Directive,args:[{selector:"[nestedEllipsis]",exportAs:"ngxNestedEllipsis"}]}],p.ctorParameters=function(){return[{type:t.TemplateRef},{type:t.ViewContainerRef},{type:t.ComponentFactoryResolver},{type:t.Renderer2},{type:t.NgZone},{type:Object,decorators:[{type:t.Inject,args:[t.PLATFORM_ID]}]}]},p.propDecorators={active:[{type:t.Input,args:["nestedEllipsis"]}],indicator:[{type:t.Input,args:["nestedEllipsisIndicator"]}],wordBoundaries:[{type:t.Input,args:["nestedEllipsisWordBoundaries"]}],mayTruncateAtFn:[{type:t.Input,args:["nestedEllipsisMayTruncateAtFn"]}],resizeDetection:[{type:t.Input,args:["nestedEllipsisResizeDetection"]}],change:[{type:t.Output,args:["nestedEllipsisChange"]}]};var u=function(){};u.decorators=[{type:t.NgModule,args:[{imports:[],declarations:[p,d],entryComponents:[d],exports:[p]}]}],e.NestedEllipsisContentComponent=d,e.NestedEllipsisDirective=p,e.NestedEllipsisModule=u,Object.defineProperty(e,"__esModule",{value:!0})})); | ||
//# sourceMappingURL=ngx-nested-ellipsis.umd.min.js.map |
@@ -101,3 +101,5 @@ import { Directive, Renderer2, Input, Output, EventEmitter, NgZone, Inject, PLATFORM_ID, TemplateRef, ViewContainerRef, ComponentFactoryResolver } from '@angular/core'; | ||
width: this.elem.clientWidth, | ||
height: this.elem.clientHeight | ||
height: this.elem.clientHeight, | ||
scrollWidth: this.elem.scrollWidth, | ||
scrollHeight: this.elem.scrollHeight | ||
}; | ||
@@ -226,3 +228,6 @@ this.applyEllipsis(); | ||
window.requestAnimationFrame(() => { | ||
if (this.previousDimensions.width !== this.elem.clientWidth || this.previousDimensions.height !== this.elem.clientHeight) { | ||
if (this.previousDimensions.width !== this.elem.clientWidth | ||
|| this.previousDimensions.height !== this.elem.clientHeight | ||
|| this.previousDimensions.scrollWidth !== this.elem.scrollWidth | ||
|| this.previousDimensions.scrollHeight !== this.elem.scrollHeight) { | ||
this.ngZone.run(() => { | ||
@@ -233,2 +238,4 @@ this.applyEllipsis(); | ||
this.previousDimensions.height = this.elem.clientHeight; | ||
this.previousDimensions.scrollWidth = this.elem.scrollWidth; | ||
this.previousDimensions.scrollHeight = this.elem.scrollHeight; | ||
} | ||
@@ -389,2 +396,2 @@ }); | ||
}; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nested-ellipsis.directive.js","sourceRoot":"../../../projects/ngx-nested-ellipsis/src/","sources":["lib/directives/nested-ellipsis.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Directive,\n  Renderer2,\n  Input,\n  Output,\n  EventEmitter,\n  NgZone,\n  OnDestroy,\n  Inject,\n  PLATFORM_ID,\n  TemplateRef,\n  ViewContainerRef,\n  ComponentFactoryResolver,\n  ComponentFactory,\n  EmbeddedViewRef,\n  AfterViewChecked,\n  OnInit\n} from '@angular/core';\nimport ResizeObserver from 'resize-observer-polyfill';\nimport { isPlatformBrowser } from '@angular/common';\nimport { NestedEllipsisContentComponent } from '../components/nested-ellipsis-content.component';\nimport { EllipsisResizeDetectionEnum } from '../enums/ellipsis-resize-detection.enum';\nimport { Subject } from 'rxjs';\nimport { take } from 'rxjs/operators';\n\n/**\n * Directive to truncate the contained text, if it exceeds the element's boundaries\n * and append characters (configurable, default '...') if so.\n */\n@Directive({\n  selector: '[nestedEllipsis]',\n  exportAs: 'ngxNestedEllipsis'\n})\nexport class NestedEllipsisDirective implements OnInit, OnDestroy, AfterViewChecked {\n  /**\n   * The referenced element\n   */\n  private elem: HTMLElement;\n\n  /**\n   * Component factory required for rendering EllipsisContent component\n   */\n  private compFactory: ComponentFactory<NestedEllipsisContentComponent>;\n\n  /**\n   * ViewRef of the main template (the one to be truncated)\n   */\n  private templateView: EmbeddedViewRef<unknown>;\n\n  /**\n   * ViewRef of the indicator template\n   */\n  private indicatorView: EmbeddedViewRef<unknown>;\n\n  /**\n   * Concatenated template html at the time of the last time the ellipsis has been applied\n   */\n  private previousTemplateHtml: string;\n\n  /**\n   * Text length before truncating\n   */\n  private initialTextLength: number;\n\n  /**\n   * Subject triggered when resize listeners should be removed\n   */\n  private removeResizeListeners$ = new Subject<void>();\n\n  private previousDimensions: {\n    width: number,\n    height: number\n  };\n\n  /**\n   * The ngxNestedEllipsis html attribute\n   * Passing true (default) will perform the directive's task,\n   * otherwise the template will be rendered without truncating its contents.\n   */\n  @Input('nestedEllipsis') active: boolean;\n\n  /**\n   * The ellipsisIndicator html attribute\n   * Passing a string (default: '...') will append it when the passed template has been truncated\n   * Passing a template will append that template instead\n   */\n  @Input('nestedEllipsisIndicator') indicator: string | TemplateRef<unknown>;\n\n  /**\n   * The ellipsisWordBoundaries html attribute\n   * Each character passed to this input will be interpreted\n   * as a word boundary at which the text may be truncated.\n   * Else the text may be truncated at any character.\n   */\n  @Input('nestedEllipsisWordBoundaries') wordBoundaries: string;\n\n  /**\n   * The ellipsisMayTruncateAtFn html attribute\n   * Function that lets you specify whether the contents may be truncated at a specific point or not:\n   * `(node: CharacterData, position: number) => boolean`\n   * `node` Text node that is being truncated\n   * `position` String position the text would be truncated at\n   * Should return true, if the text may be truncated here, else false\n   */\n  @Input('nestedEllipsisMayTruncateAtFn') mayTruncateAtFn: (node: CharacterData, position: number) => boolean;\n\n  /**\n   * The ellipsisResizeDetection html attribute\n   * Algorithm to use to detect element/window resize - any value of `EllipsisResizeDetectionEnum`\n   */\n  @Input('nestedEllipsisResizeDetection') resizeDetection: EllipsisResizeDetectionEnum;\n\n\n  /**\n   * The ellipsisChange html attribute\n   * This emits after which index the text has been truncated.\n   * If it hasn't been truncated, null is emitted.\n   */\n  @Output('nestedEllipsisChange') readonly change: EventEmitter<number> = new EventEmitter();\n\n  /**\n   * Utility method to quickly find the largest number for\n   * which `callback(number)` still returns true.\n   * @param  max      Highest possible number\n   * @param  callback Should return true as long as the passed number is valid\n   * @returns         Largest possible number\n   */\n  private static numericBinarySearch(max: number, callback: (n: number) => boolean): number {\n    let low = 0;\n    let high = max;\n    let best = -1;\n    let mid: number;\n\n    while (low <= high) {\n      mid = Math.floor((low + high) / 2);\n      const result = callback(mid);\n      if (!result) {\n        high = mid - 1;\n      } else {\n        best = mid;\n        low = mid + 1;\n      }\n    }\n\n    return best;\n  }\n\n  private flattenTextAndElementNodes(element: HTMLElement): (CharacterData | HTMLElement)[] {\n    const nodes: (CharacterData | HTMLElement)[] = [];\n    for (let i = 0; i < element.childNodes.length; i++) {\n      const child = element.childNodes.item(i);\n      if (child instanceof HTMLElement || child instanceof CharacterData) {\n        nodes.push(child);\n\n        if (child instanceof HTMLElement) {\n          nodes.push(...this.flattenTextAndElementNodes(child));\n        }\n      }\n    }\n\n    return nodes;\n  }\n\n\n  /**\n   * The directive's constructor\n   */\n  public constructor(\n    private readonly templateRef: TemplateRef<unknown>,\n    private readonly viewContainer: ViewContainerRef,\n    private readonly resolver: ComponentFactoryResolver,\n    private readonly renderer: Renderer2,\n    private readonly ngZone: NgZone,\n    @Inject(PLATFORM_ID) private platformId: Object\n  ) { }\n\n  /**\n   * Angular's onInit life cycle hook.\n   * Initializes the element for displaying the ellipsis.\n   */\n  ngOnInit() {\n    if (!isPlatformBrowser(this.platformId)) {\n      // in angular universal we don't have access to the ugly\n      // DOM manipulation properties we sadly need to access here,\n      // so wait until we're in the browser:\n      return;\n    }\n\n    if (typeof(this.active) !== 'boolean') {\n      this.active = true;\n    }\n\n    if (typeof(this.indicator) === 'undefined') {\n      this.indicator = '...';\n    }\n\n    if (typeof (this.resizeDetection) === 'undefined') {\n      this.resizeDetection = EllipsisResizeDetectionEnum.ResizeObserver;\n    }\n\n    // perform regex replace on word boundaries:\n    if (!this.wordBoundaries) {\n      this.wordBoundaries = '';\n    }\n    this.wordBoundaries = '[' + this.wordBoundaries.replace(/[-\\/\\\\^$*+?.()|[\\]{}]/g, '\\\\$&') + ']';\n\n    // initialize view:\n    this.compFactory = this.resolver.resolveComponentFactory(NestedEllipsisContentComponent);\n    this.restoreView();\n    this.previousDimensions = {\n      width: this.elem.clientWidth,\n      height: this.elem.clientHeight\n    };\n\n    this.applyEllipsis();\n  }\n\n\n  /**\n   * Angular's destroy life cycle hook.\n   * Remove event listeners\n   */\n  ngOnDestroy() {\n    this.removeResizeListeners$.next();\n    this.removeResizeListeners$.complete();\n  }\n\n  /**\n   * Angular's afterViewChecked life cycle hook.\n   * Reapply ellipsis, if any of the templates have changed\n   */\n  ngAfterViewChecked() {\n    if (this.resizeDetection !== 'manual') {\n      if (this.templatesHaveChanged) {\n        this.applyEllipsis();\n      }\n    }\n  }\n\n  /**\n   * Convert a list of Nodes to html\n   * @param nodes Nodes to convert\n   * @returns html code\n   */\n  private nodesToHtml(nodes: Node[]): string {\n    const div = <HTMLElement> this.renderer.createElement('div');\n    div.append(...nodes.map(node => node.cloneNode(true)));\n    return div.innerHTML;\n  }\n\n  /**\n   * Convert the passed templates to html\n   * @param templateView the main template view ref\n   * @param indicatorView the indicator template view ref\n   * @returns concatenated template html\n   */\n  private templatesToHtml(templateView: EmbeddedViewRef<unknown>, indicatorView?: EmbeddedViewRef<unknown>): string {\n    let html = this.nodesToHtml(templateView.rootNodes);\n    if (indicatorView) {\n      html += this.nodesToHtml(indicatorView.rootNodes);\n    } else {\n      html += <string> this.indicator;\n    }\n\n    return html;\n  }\n\n  /**\n   * Whether any of the passed templates have changed since the last time\n   * the ellipsis has been applied\n   */\n  private get templatesHaveChanged(): boolean {\n    if (!this.templateView || !this.previousTemplateHtml) {\n      return false;\n    }\n\n    const templateView = this.templateRef.createEmbeddedView({});\n    templateView.detectChanges();\n\n    const indicatorView = (typeof this.indicator !== 'string') ? this.indicator.createEmbeddedView({}) : null;\n    if (indicatorView) {\n      indicatorView.detectChanges();\n    }\n\n    const templateHtml = this.templatesToHtml(templateView, indicatorView);\n\n    return this.previousTemplateHtml !== templateHtml;\n  }\n\n  /**\n   * Restore the view from the templates (non-truncated)\n   */\n  private restoreView() {\n    this.viewContainer.clear();\n    this.templateView = this.templateRef.createEmbeddedView({});\n    this.templateView.detectChanges();\n    const componentRef = this.viewContainer.createComponent(\n      this.compFactory, null, this.viewContainer.injector, [this.templateView.rootNodes]\n    );\n    this.elem = componentRef.instance.elementRef.nativeElement;\n    this.initialTextLength = this.currentLength;\n\n    this.indicatorView = (typeof this.indicator !== 'string') ? this.indicator.createEmbeddedView({}) : null;\n    if (this.indicatorView) {\n      this.indicatorView.detectChanges();\n    }\n  }\n\n\n  /**\n   * Set up an event listener to call applyEllipsis() whenever a resize has been registered.\n   * The type of the listener (window/element) depends on the `ellipsisResizeDetection`.\n   */\n  private addResizeListener() {\n    switch (this.resizeDetection) {\n      case EllipsisResizeDetectionEnum.Manual:\n        // Users will trigger applyEllipsis via the public API\n        break;\n      case EllipsisResizeDetectionEnum.Window:\n        this.addWindowResizeListener();\n        break;\n      default:\n        if (typeof (console) !== 'undefined') {\n          console.warn(`\n            No such ellipsisResizeDetection strategy: '${this.resizeDetection}'.\n            Using '${EllipsisResizeDetectionEnum.ResizeObserver}' instead.\n          `);\n        }\n        this.resizeDetection = EllipsisResizeDetectionEnum.ResizeObserver;\n      // eslint-disable-next-line no-fallthrough\n      case EllipsisResizeDetectionEnum.ResizeObserver:\n        this.addResizeObserver();\n        break;\n    }\n  }\n\n  /**\n   * Set up an event listener to call applyEllipsis() whenever the window gets resized.\n   */\n  private addWindowResizeListener() {\n    const removeWindowResizeListener = this.renderer.listen('window', 'resize', () => {\n      this.ngZone.run(() => {\n        this.applyEllipsis();\n      });\n    });\n\n    this.removeResizeListeners$.pipe(take(1)).subscribe(() => removeWindowResizeListener());\n  }\n\n  /**\n   * Set up an event listener to call applyEllipsis() whenever ResizeObserver is triggered for the element.\n   */\n  private addResizeObserver() {\n    const resizeObserver = new ResizeObserver(() => {\n      window.requestAnimationFrame(() => {\n        if (this.previousDimensions.width !== this.elem.clientWidth || this.previousDimensions.height !== this.elem.clientHeight) {\n          this.ngZone.run(() => {\n            this.applyEllipsis();\n          });\n\n          this.previousDimensions.width = this.elem.clientWidth;\n          this.previousDimensions.height = this.elem.clientHeight;\n        }\n      });\n    });\n    resizeObserver.observe(this.elem);\n    this.removeResizeListeners$.pipe(take(1)).subscribe(() => resizeObserver.disconnect());\n  }\n\n\n  /**\n   * Get the original text's truncated version. If the text really needed to\n   * be truncated, this.ellipsisCharacters will be appended.\n   * @param max the maximum length the text may have\n   * @returns the text node that has been truncated or null if truncating wasn't required\n   */\n  private truncateContents(max: number): Node {\n    this.restoreView();\n    const nodes = <(HTMLElement | CharacterData)[]>this.flattenTextAndElementNodes(this.elem)\n      .filter(node => [Node.TEXT_NODE, Node.ELEMENT_NODE].includes(node.nodeType));\n\n    let foundIndex = -1;\n    let foundNode: Node;\n    let offset = this.initialTextLength;\n    for (let i = nodes.length - 1; i >= 0; i--) {\n      const node = nodes[i];\n\n      if (node instanceof CharacterData) {\n        offset -= node.data.length;\n      } else {\n        offset--;\n      }\n\n      if (offset <= max) {\n        if (node instanceof CharacterData) {\n          if (this.wordBoundaries === '[]' && !this.mayTruncateAtFn) {\n            node.data = node.data.substr(0, max - offset);\n          } else if (max - offset !== node.data.length) {\n            let j = max - offset - 1;\n            while (\n              j > 0 && (\n                (this.wordBoundaries !== '[]' && !node.data.charAt(j).match(this.wordBoundaries)) ||\n                (this.mayTruncateAtFn && !this.mayTruncateAtFn(node, j))\n              )\n            ) {\n              j--;\n            }\n            if (offset > 0 && j === 0) {\n              continue;\n            }\n            node.data = node.data.substr(0, j);\n          }\n        }\n        foundIndex = i;\n        foundNode = node;\n        break;\n      }\n    }\n\n    for (let i = foundIndex + 1; i < nodes.length; i++) {\n      const node = nodes[i];\n      if (node.textContent !== '' && node.parentElement !== this.elem && node.parentElement.childNodes.length === 1) {\n        node.parentElement.remove();\n      } else {\n        node.remove();\n      }\n    }\n\n    return (this.currentLength !== this.initialTextLength) ? foundNode : null;\n  }\n\n  private get currentLength(): number {\n    return this.flattenTextAndElementNodes(this.elem)\n      .filter(node => [Node.TEXT_NODE, Node.ELEMENT_NODE].includes(node.nodeType))\n      .map(node => (node instanceof CharacterData) ? node.data.length : 1)\n      .reduce((sum, length) => sum + length, 0);\n  }\n\n  /**\n   * Set the truncated text to be displayed in the inner div\n   * @param max the maximum length the text may have\n   * @param addMoreListener=false listen for click on the ellipsisCharacters anchor tag if the text has been truncated\n   */\n  private truncateText(max: number) {\n    const truncatedNode = this.truncateContents(max);\n\n    if (truncatedNode) {\n      if (!this.indicatorView) {\n        if (truncatedNode instanceof CharacterData) {\n          truncatedNode.data += <string> this.indicator;\n        } else {\n          this.renderer.appendChild(this.elem, this.renderer.createText(<string> this.indicator));\n        }\n      } else {\n        for (const node of this.indicatorView.rootNodes) {\n          this.renderer.appendChild(this.elem, node);\n        }\n      }\n    }\n  }\n\n\n  /**\n   * Display ellipsis in the EllipsisContentComponent if the text would exceed the boundaries\n   */\n  public applyEllipsis() {\n    // Remove the resize listener as changing the contained text would trigger events:\n    this.removeResizeListeners$.next();\n\n    // update from templates:\n    this.restoreView();\n\n    // remember template state:\n    this.previousTemplateHtml = this.templatesToHtml(this.templateView, this.indicatorView);\n\n    // abort if [nestedEllipsis]=\"false\" has been set\n    if (!this.active) {\n      return;\n    }\n\n    // Find the best length by trial and error:\n    const maxLength = NestedEllipsisDirective.numericBinarySearch(this.initialTextLength, curLength => {\n      this.truncateText(curLength);\n      return !this.isOverflowing;\n    });\n\n    // Apply the best length:\n    this.truncateText(maxLength);\n\n    // Re-attach the resize listener:\n    this.addResizeListener();\n\n    // Emit change event:\n    if (this.change.observers.length > 0) {\n      const currentLength = this.currentLength;\n      this.change.emit(\n        (currentLength === this.initialTextLength) ? null : currentLength\n      );\n    }\n  }\n\n\n  /**\n   * Whether the text is exceeding the element's boundaries or not\n   */\n  private get isOverflowing(): boolean {\n    // Enforce hidden overflow (required to compare client width/height with scroll width/height)\n    const currentOverflow = this.elem.style.overflow;\n    if (!currentOverflow || currentOverflow === 'visible') {\n      this.elem.style.overflow = 'hidden';\n    }\n\n    const isOverflowing = this.elem.clientWidth < this.elem.scrollWidth - 1 || this.elem.clientHeight < this.elem.scrollHeight - 1;\n\n    // Reset overflow to the original configuration:\n    this.elem.style.overflow = currentOverflow;\n\n    return isOverflowing;\n  }\n\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nested-ellipsis.directive.js","sourceRoot":"../../../projects/ngx-nested-ellipsis/src/","sources":["lib/directives/nested-ellipsis.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Directive,\n  Renderer2,\n  Input,\n  Output,\n  EventEmitter,\n  NgZone,\n  OnDestroy,\n  Inject,\n  PLATFORM_ID,\n  TemplateRef,\n  ViewContainerRef,\n  ComponentFactoryResolver,\n  ComponentFactory,\n  EmbeddedViewRef,\n  AfterViewChecked,\n  OnInit\n} from '@angular/core';\nimport ResizeObserver from 'resize-observer-polyfill';\nimport { isPlatformBrowser } from '@angular/common';\nimport { NestedEllipsisContentComponent } from '../components/nested-ellipsis-content.component';\nimport { EllipsisResizeDetectionEnum } from '../enums/ellipsis-resize-detection.enum';\nimport { Subject } from 'rxjs';\nimport { take } from 'rxjs/operators';\n\n/**\n * Directive to truncate the contained text, if it exceeds the element's boundaries\n * and append characters (configurable, default '...') if so.\n */\n@Directive({\n  selector: '[nestedEllipsis]',\n  exportAs: 'ngxNestedEllipsis'\n})\nexport class NestedEllipsisDirective implements OnInit, OnDestroy, AfterViewChecked {\n  /**\n   * The referenced element\n   */\n  private elem: HTMLElement;\n\n  /**\n   * Component factory required for rendering EllipsisContent component\n   */\n  private compFactory: ComponentFactory<NestedEllipsisContentComponent>;\n\n  /**\n   * ViewRef of the main template (the one to be truncated)\n   */\n  private templateView: EmbeddedViewRef<unknown>;\n\n  /**\n   * ViewRef of the indicator template\n   */\n  private indicatorView: EmbeddedViewRef<unknown>;\n\n  /**\n   * Concatenated template html at the time of the last time the ellipsis has been applied\n   */\n  private previousTemplateHtml: string;\n\n  /**\n   * Text length before truncating\n   */\n  private initialTextLength: number;\n\n  /**\n   * Subject triggered when resize listeners should be removed\n   */\n  private removeResizeListeners$ = new Subject<void>();\n\n  private previousDimensions: {\n    width: number,\n    height: number,\n    scrollWidth: number,\n    scrollHeight: number\n  };\n\n  /**\n   * The ngxNestedEllipsis html attribute\n   * Passing true (default) will perform the directive's task,\n   * otherwise the template will be rendered without truncating its contents.\n   */\n  @Input('nestedEllipsis') active: boolean;\n\n  /**\n   * The ellipsisIndicator html attribute\n   * Passing a string (default: '...') will append it when the passed template has been truncated\n   * Passing a template will append that template instead\n   */\n  @Input('nestedEllipsisIndicator') indicator: string | TemplateRef<unknown>;\n\n  /**\n   * The ellipsisWordBoundaries html attribute\n   * Each character passed to this input will be interpreted\n   * as a word boundary at which the text may be truncated.\n   * Else the text may be truncated at any character.\n   */\n  @Input('nestedEllipsisWordBoundaries') wordBoundaries: string;\n\n  /**\n   * The ellipsisMayTruncateAtFn html attribute\n   * Function that lets you specify whether the contents may be truncated at a specific point or not:\n   * `(node: CharacterData, position: number) => boolean`\n   * `node` Text node that is being truncated\n   * `position` String position the text would be truncated at\n   * Should return true, if the text may be truncated here, else false\n   */\n  @Input('nestedEllipsisMayTruncateAtFn') mayTruncateAtFn: (node: CharacterData, position: number) => boolean;\n\n  /**\n   * The ellipsisResizeDetection html attribute\n   * Algorithm to use to detect element/window resize - any value of `EllipsisResizeDetectionEnum`\n   */\n  @Input('nestedEllipsisResizeDetection') resizeDetection: EllipsisResizeDetectionEnum;\n\n\n  /**\n   * The ellipsisChange html attribute\n   * This emits after which index the text has been truncated.\n   * If it hasn't been truncated, null is emitted.\n   */\n  @Output('nestedEllipsisChange') readonly change: EventEmitter<number> = new EventEmitter();\n\n  /**\n   * Utility method to quickly find the largest number for\n   * which `callback(number)` still returns true.\n   * @param  max      Highest possible number\n   * @param  callback Should return true as long as the passed number is valid\n   * @returns         Largest possible number\n   */\n  private static numericBinarySearch(max: number, callback: (n: number) => boolean): number {\n    let low = 0;\n    let high = max;\n    let best = -1;\n    let mid: number;\n\n    while (low <= high) {\n      mid = Math.floor((low + high) / 2);\n      const result = callback(mid);\n      if (!result) {\n        high = mid - 1;\n      } else {\n        best = mid;\n        low = mid + 1;\n      }\n    }\n\n    return best;\n  }\n\n  private flattenTextAndElementNodes(element: HTMLElement): (CharacterData | HTMLElement)[] {\n    const nodes: (CharacterData | HTMLElement)[] = [];\n    for (let i = 0; i < element.childNodes.length; i++) {\n      const child = element.childNodes.item(i);\n      if (child instanceof HTMLElement || child instanceof CharacterData) {\n        nodes.push(child);\n\n        if (child instanceof HTMLElement) {\n          nodes.push(...this.flattenTextAndElementNodes(child));\n        }\n      }\n    }\n\n    return nodes;\n  }\n\n\n  /**\n   * The directive's constructor\n   */\n  public constructor(\n    private readonly templateRef: TemplateRef<unknown>,\n    private readonly viewContainer: ViewContainerRef,\n    private readonly resolver: ComponentFactoryResolver,\n    private readonly renderer: Renderer2,\n    private readonly ngZone: NgZone,\n    @Inject(PLATFORM_ID) private platformId: Object\n  ) { }\n\n  /**\n   * Angular's onInit life cycle hook.\n   * Initializes the element for displaying the ellipsis.\n   */\n  ngOnInit() {\n    if (!isPlatformBrowser(this.platformId)) {\n      // in angular universal we don't have access to the ugly\n      // DOM manipulation properties we sadly need to access here,\n      // so wait until we're in the browser:\n      return;\n    }\n\n    if (typeof(this.active) !== 'boolean') {\n      this.active = true;\n    }\n\n    if (typeof(this.indicator) === 'undefined') {\n      this.indicator = '...';\n    }\n\n    if (typeof (this.resizeDetection) === 'undefined') {\n      this.resizeDetection = EllipsisResizeDetectionEnum.ResizeObserver;\n    }\n\n    // perform regex replace on word boundaries:\n    if (!this.wordBoundaries) {\n      this.wordBoundaries = '';\n    }\n    this.wordBoundaries = '[' + this.wordBoundaries.replace(/[-\\/\\\\^$*+?.()|[\\]{}]/g, '\\\\$&') + ']';\n\n    // initialize view:\n    this.compFactory = this.resolver.resolveComponentFactory(NestedEllipsisContentComponent);\n    this.restoreView();\n    this.previousDimensions = {\n      width: this.elem.clientWidth,\n      height: this.elem.clientHeight,\n      scrollWidth: this.elem.scrollWidth,\n      scrollHeight: this.elem.scrollHeight\n    };\n\n    this.applyEllipsis();\n  }\n\n\n  /**\n   * Angular's destroy life cycle hook.\n   * Remove event listeners\n   */\n  ngOnDestroy() {\n    this.removeResizeListeners$.next();\n    this.removeResizeListeners$.complete();\n  }\n\n  /**\n   * Angular's afterViewChecked life cycle hook.\n   * Reapply ellipsis, if any of the templates have changed\n   */\n  ngAfterViewChecked() {\n    if (this.resizeDetection !== 'manual') {\n      if (this.templatesHaveChanged) {\n        this.applyEllipsis();\n      }\n    }\n  }\n\n  /**\n   * Convert a list of Nodes to html\n   * @param nodes Nodes to convert\n   * @returns html code\n   */\n  private nodesToHtml(nodes: Node[]): string {\n    const div = <HTMLElement> this.renderer.createElement('div');\n    div.append(...nodes.map(node => node.cloneNode(true)));\n    return div.innerHTML;\n  }\n\n  /**\n   * Convert the passed templates to html\n   * @param templateView the main template view ref\n   * @param indicatorView the indicator template view ref\n   * @returns concatenated template html\n   */\n  private templatesToHtml(templateView: EmbeddedViewRef<unknown>, indicatorView?: EmbeddedViewRef<unknown>): string {\n    let html = this.nodesToHtml(templateView.rootNodes);\n    if (indicatorView) {\n      html += this.nodesToHtml(indicatorView.rootNodes);\n    } else {\n      html += <string> this.indicator;\n    }\n\n    return html;\n  }\n\n  /**\n   * Whether any of the passed templates have changed since the last time\n   * the ellipsis has been applied\n   */\n  private get templatesHaveChanged(): boolean {\n    if (!this.templateView || !this.previousTemplateHtml) {\n      return false;\n    }\n\n    const templateView = this.templateRef.createEmbeddedView({});\n    templateView.detectChanges();\n\n    const indicatorView = (typeof this.indicator !== 'string') ? this.indicator.createEmbeddedView({}) : null;\n    if (indicatorView) {\n      indicatorView.detectChanges();\n    }\n\n    const templateHtml = this.templatesToHtml(templateView, indicatorView);\n\n    return this.previousTemplateHtml !== templateHtml;\n  }\n\n  /**\n   * Restore the view from the templates (non-truncated)\n   */\n  private restoreView() {\n    this.viewContainer.clear();\n    this.templateView = this.templateRef.createEmbeddedView({});\n    this.templateView.detectChanges();\n    const componentRef = this.viewContainer.createComponent(\n      this.compFactory, null, this.viewContainer.injector, [this.templateView.rootNodes]\n    );\n    this.elem = componentRef.instance.elementRef.nativeElement;\n    this.initialTextLength = this.currentLength;\n\n    this.indicatorView = (typeof this.indicator !== 'string') ? this.indicator.createEmbeddedView({}) : null;\n    if (this.indicatorView) {\n      this.indicatorView.detectChanges();\n    }\n  }\n\n\n  /**\n   * Set up an event listener to call applyEllipsis() whenever a resize has been registered.\n   * The type of the listener (window/element) depends on the `ellipsisResizeDetection`.\n   */\n  private addResizeListener() {\n    switch (this.resizeDetection) {\n      case EllipsisResizeDetectionEnum.Manual:\n        // Users will trigger applyEllipsis via the public API\n        break;\n      case EllipsisResizeDetectionEnum.Window:\n        this.addWindowResizeListener();\n        break;\n      default:\n        if (typeof (console) !== 'undefined') {\n          console.warn(`\n            No such ellipsisResizeDetection strategy: '${this.resizeDetection}'.\n            Using '${EllipsisResizeDetectionEnum.ResizeObserver}' instead.\n          `);\n        }\n        this.resizeDetection = EllipsisResizeDetectionEnum.ResizeObserver;\n      // eslint-disable-next-line no-fallthrough\n      case EllipsisResizeDetectionEnum.ResizeObserver:\n        this.addResizeObserver();\n        break;\n    }\n  }\n\n  /**\n   * Set up an event listener to call applyEllipsis() whenever the window gets resized.\n   */\n  private addWindowResizeListener() {\n    const removeWindowResizeListener = this.renderer.listen('window', 'resize', () => {\n      this.ngZone.run(() => {\n        this.applyEllipsis();\n      });\n    });\n\n    this.removeResizeListeners$.pipe(take(1)).subscribe(() => removeWindowResizeListener());\n  }\n\n  /**\n   * Set up an event listener to call applyEllipsis() whenever ResizeObserver is triggered for the element.\n   */\n  private addResizeObserver() {\n    const resizeObserver = new ResizeObserver(() => {\n      window.requestAnimationFrame(() => {\n        if (\n          this.previousDimensions.width !== this.elem.clientWidth\n          || this.previousDimensions.height !== this.elem.clientHeight\n          || this.previousDimensions.scrollWidth !== this.elem.scrollWidth\n          || this.previousDimensions.scrollHeight !== this.elem.scrollHeight\n        ) {\n          this.ngZone.run(() => {\n            this.applyEllipsis();\n          });\n\n          this.previousDimensions.width = this.elem.clientWidth;\n          this.previousDimensions.height = this.elem.clientHeight;\n          this.previousDimensions.scrollWidth = this.elem.scrollWidth;\n          this.previousDimensions.scrollHeight = this.elem.scrollHeight;\n        }\n      });\n    });\n    resizeObserver.observe(this.elem);\n    this.removeResizeListeners$.pipe(take(1)).subscribe(() => resizeObserver.disconnect());\n  }\n\n\n  /**\n   * Get the original text's truncated version. If the text really needed to\n   * be truncated, this.ellipsisCharacters will be appended.\n   * @param max the maximum length the text may have\n   * @returns the text node that has been truncated or null if truncating wasn't required\n   */\n  private truncateContents(max: number): Node {\n    this.restoreView();\n    const nodes = <(HTMLElement | CharacterData)[]>this.flattenTextAndElementNodes(this.elem)\n      .filter(node => [Node.TEXT_NODE, Node.ELEMENT_NODE].includes(node.nodeType));\n\n    let foundIndex = -1;\n    let foundNode: Node;\n    let offset = this.initialTextLength;\n    for (let i = nodes.length - 1; i >= 0; i--) {\n      const node = nodes[i];\n\n      if (node instanceof CharacterData) {\n        offset -= node.data.length;\n      } else {\n        offset--;\n      }\n\n      if (offset <= max) {\n        if (node instanceof CharacterData) {\n          if (this.wordBoundaries === '[]' && !this.mayTruncateAtFn) {\n            node.data = node.data.substr(0, max - offset);\n          } else if (max - offset !== node.data.length) {\n            let j = max - offset - 1;\n            while (\n              j > 0 && (\n                (this.wordBoundaries !== '[]' && !node.data.charAt(j).match(this.wordBoundaries)) ||\n                (this.mayTruncateAtFn && !this.mayTruncateAtFn(node, j))\n              )\n            ) {\n              j--;\n            }\n            if (offset > 0 && j === 0) {\n              continue;\n            }\n            node.data = node.data.substr(0, j);\n          }\n        }\n        foundIndex = i;\n        foundNode = node;\n        break;\n      }\n    }\n\n    for (let i = foundIndex + 1; i < nodes.length; i++) {\n      const node = nodes[i];\n      if (node.textContent !== '' && node.parentElement !== this.elem && node.parentElement.childNodes.length === 1) {\n        node.parentElement.remove();\n      } else {\n        node.remove();\n      }\n    }\n\n    return (this.currentLength !== this.initialTextLength) ? foundNode : null;\n  }\n\n  private get currentLength(): number {\n    return this.flattenTextAndElementNodes(this.elem)\n      .filter(node => [Node.TEXT_NODE, Node.ELEMENT_NODE].includes(node.nodeType))\n      .map(node => (node instanceof CharacterData) ? node.data.length : 1)\n      .reduce((sum, length) => sum + length, 0);\n  }\n\n  /**\n   * Set the truncated text to be displayed in the inner div\n   * @param max the maximum length the text may have\n   * @param addMoreListener=false listen for click on the ellipsisCharacters anchor tag if the text has been truncated\n   */\n  private truncateText(max: number) {\n    const truncatedNode = this.truncateContents(max);\n\n    if (truncatedNode) {\n      if (!this.indicatorView) {\n        if (truncatedNode instanceof CharacterData) {\n          truncatedNode.data += <string> this.indicator;\n        } else {\n          this.renderer.appendChild(this.elem, this.renderer.createText(<string> this.indicator));\n        }\n      } else {\n        for (const node of this.indicatorView.rootNodes) {\n          this.renderer.appendChild(this.elem, node);\n        }\n      }\n    }\n  }\n\n\n  /**\n   * Display ellipsis in the EllipsisContentComponent if the text would exceed the boundaries\n   */\n  public applyEllipsis() {\n    // Remove the resize listener as changing the contained text would trigger events:\n    this.removeResizeListeners$.next();\n\n    // update from templates:\n    this.restoreView();\n\n    // remember template state:\n    this.previousTemplateHtml = this.templatesToHtml(this.templateView, this.indicatorView);\n\n    // abort if [nestedEllipsis]=\"false\" has been set\n    if (!this.active) {\n      return;\n    }\n\n    // Find the best length by trial and error:\n    const maxLength = NestedEllipsisDirective.numericBinarySearch(this.initialTextLength, curLength => {\n      this.truncateText(curLength);\n      return !this.isOverflowing;\n    });\n\n    // Apply the best length:\n    this.truncateText(maxLength);\n\n    // Re-attach the resize listener:\n    this.addResizeListener();\n\n    // Emit change event:\n    if (this.change.observers.length > 0) {\n      const currentLength = this.currentLength;\n      this.change.emit(\n        (currentLength === this.initialTextLength) ? null : currentLength\n      );\n    }\n  }\n\n\n  /**\n   * Whether the text is exceeding the element's boundaries or not\n   */\n  private get isOverflowing(): boolean {\n    // Enforce hidden overflow (required to compare client width/height with scroll width/height)\n    const currentOverflow = this.elem.style.overflow;\n    if (!currentOverflow || currentOverflow === 'visible') {\n      this.elem.style.overflow = 'hidden';\n    }\n\n    const isOverflowing = this.elem.clientWidth < this.elem.scrollWidth - 1 || this.elem.clientHeight < this.elem.scrollHeight - 1;\n\n    // Reset overflow to the original configuration:\n    this.elem.style.overflow = currentOverflow;\n\n    return isOverflowing;\n  }\n\n}\n"]} |
@@ -126,3 +126,5 @@ import { Component, ElementRef, EventEmitter, Directive, TemplateRef, ViewContainerRef, ComponentFactoryResolver, Renderer2, NgZone, Inject, PLATFORM_ID, Input, Output, NgModule } from '@angular/core'; | ||
width: this.elem.clientWidth, | ||
height: this.elem.clientHeight | ||
height: this.elem.clientHeight, | ||
scrollWidth: this.elem.scrollWidth, | ||
scrollHeight: this.elem.scrollHeight | ||
}; | ||
@@ -251,3 +253,6 @@ this.applyEllipsis(); | ||
window.requestAnimationFrame(() => { | ||
if (this.previousDimensions.width !== this.elem.clientWidth || this.previousDimensions.height !== this.elem.clientHeight) { | ||
if (this.previousDimensions.width !== this.elem.clientWidth | ||
|| this.previousDimensions.height !== this.elem.clientHeight | ||
|| this.previousDimensions.scrollWidth !== this.elem.scrollWidth | ||
|| this.previousDimensions.scrollHeight !== this.elem.scrollHeight) { | ||
this.ngZone.run(() => { | ||
@@ -258,2 +263,4 @@ this.applyEllipsis(); | ||
this.previousDimensions.height = this.elem.clientHeight; | ||
this.previousDimensions.scrollWidth = this.elem.scrollWidth; | ||
this.previousDimensions.scrollHeight = this.elem.scrollHeight; | ||
} | ||
@@ -260,0 +267,0 @@ }); |
@@ -1,1 +0,1 @@ | ||
{"__symbolic":"module","version":4,"metadata":{"NestedEllipsisDirective":{"__symbolic":"class","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Directive","line":29,"character":1},"arguments":[{"selector":"[nestedEllipsis]","exportAs":"ngxNestedEllipsis"}]}],"members":{"active":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":79,"character":3},"arguments":["nestedEllipsis"]}]}],"indicator":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":86,"character":3},"arguments":["nestedEllipsisIndicator"]}]}],"wordBoundaries":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":94,"character":3},"arguments":["nestedEllipsisWordBoundaries"]}]}],"mayTruncateAtFn":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":104,"character":3},"arguments":["nestedEllipsisMayTruncateAtFn"]}]}],"resizeDetection":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":110,"character":3},"arguments":["nestedEllipsisResizeDetection"]}]}],"change":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Output","line":118,"character":3},"arguments":["nestedEllipsisChange"]}]}],"flattenTextAndElementNodes":[{"__symbolic":"method"}],"__ctor__":[{"__symbolic":"constructor","parameterDecorators":[null,null,null,null,null,[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Inject","line":173,"character":5},"arguments":[{"__symbolic":"reference","module":"@angular/core","name":"PLATFORM_ID","line":173,"character":12}]}]],"parameters":[{"__symbolic":"reference","name":"TemplateRef","module":"@angular/core","arguments":[{"__symbolic":"error","message":"Expression form not supported","line":168,"character":46,"module":"./lib/directives/nested-ellipsis.directive"}]},{"__symbolic":"reference","module":"@angular/core","name":"ViewContainerRef","line":169,"character":36},{"__symbolic":"reference","module":"@angular/core","name":"ComponentFactoryResolver","line":170,"character":31},{"__symbolic":"reference","module":"@angular/core","name":"Renderer2","line":171,"character":31},{"__symbolic":"reference","module":"@angular/core","name":"NgZone","line":172,"character":29},{"__symbolic":"reference","name":"Object"}]}],"ngOnInit":[{"__symbolic":"method"}],"ngOnDestroy":[{"__symbolic":"method"}],"ngAfterViewChecked":[{"__symbolic":"method"}],"nodesToHtml":[{"__symbolic":"method"}],"templatesToHtml":[{"__symbolic":"method"}],"restoreView":[{"__symbolic":"method"}],"addResizeListener":[{"__symbolic":"method"}],"addWindowResizeListener":[{"__symbolic":"method"}],"addResizeObserver":[{"__symbolic":"method"}],"truncateContents":[{"__symbolic":"method"}],"truncateText":[{"__symbolic":"method"}],"applyEllipsis":[{"__symbolic":"method"}]}},"NestedEllipsisContentComponent":{"__symbolic":"class","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Component","line":2,"character":1},"arguments":[{"selector":"nested-ellipsis-content","template":"\n <ng-content></ng-content>\n ","styles":["\n :host {\n display: block;\n width: 100%;\n height: 100%;\n overflow: hidden;\n }\n "]}]}],"members":{"__ctor__":[{"__symbolic":"constructor","parameters":[{"__symbolic":"reference","module":"@angular/core","name":"ElementRef","line":17,"character":33}]}]}},"EllipsisResizeDetectionEnum":{"ResizeObserver":"resize-observer","Window":"window","Manual":"manual"},"NestedEllipsisModule":{"__symbolic":"class","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"NgModule","line":4,"character":1},"arguments":[{"imports":[],"declarations":[{"__symbolic":"reference","name":"NestedEllipsisDirective"},{"__symbolic":"reference","name":"NestedEllipsisContentComponent"}],"entryComponents":[{"__symbolic":"reference","name":"NestedEllipsisContentComponent"}],"exports":[{"__symbolic":"reference","name":"NestedEllipsisDirective"}]}]}],"members":{}}},"origins":{"NestedEllipsisDirective":"./lib/directives/nested-ellipsis.directive","NestedEllipsisContentComponent":"./lib/components/nested-ellipsis-content.component","EllipsisResizeDetectionEnum":"./lib/enums/ellipsis-resize-detection.enum","NestedEllipsisModule":"./lib/nested-ellipsis.module"},"importAs":"ngx-nested-ellipsis"} | ||
{"__symbolic":"module","version":4,"metadata":{"NestedEllipsisDirective":{"__symbolic":"class","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Directive","line":29,"character":1},"arguments":[{"selector":"[nestedEllipsis]","exportAs":"ngxNestedEllipsis"}]}],"members":{"active":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":81,"character":3},"arguments":["nestedEllipsis"]}]}],"indicator":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":88,"character":3},"arguments":["nestedEllipsisIndicator"]}]}],"wordBoundaries":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":96,"character":3},"arguments":["nestedEllipsisWordBoundaries"]}]}],"mayTruncateAtFn":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":106,"character":3},"arguments":["nestedEllipsisMayTruncateAtFn"]}]}],"resizeDetection":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Input","line":112,"character":3},"arguments":["nestedEllipsisResizeDetection"]}]}],"change":[{"__symbolic":"property","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Output","line":120,"character":3},"arguments":["nestedEllipsisChange"]}]}],"flattenTextAndElementNodes":[{"__symbolic":"method"}],"__ctor__":[{"__symbolic":"constructor","parameterDecorators":[null,null,null,null,null,[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Inject","line":175,"character":5},"arguments":[{"__symbolic":"reference","module":"@angular/core","name":"PLATFORM_ID","line":175,"character":12}]}]],"parameters":[{"__symbolic":"reference","name":"TemplateRef","module":"@angular/core","arguments":[{"__symbolic":"error","message":"Expression form not supported","line":170,"character":46,"module":"./lib/directives/nested-ellipsis.directive"}]},{"__symbolic":"reference","module":"@angular/core","name":"ViewContainerRef","line":171,"character":36},{"__symbolic":"reference","module":"@angular/core","name":"ComponentFactoryResolver","line":172,"character":31},{"__symbolic":"reference","module":"@angular/core","name":"Renderer2","line":173,"character":31},{"__symbolic":"reference","module":"@angular/core","name":"NgZone","line":174,"character":29},{"__symbolic":"reference","name":"Object"}]}],"ngOnInit":[{"__symbolic":"method"}],"ngOnDestroy":[{"__symbolic":"method"}],"ngAfterViewChecked":[{"__symbolic":"method"}],"nodesToHtml":[{"__symbolic":"method"}],"templatesToHtml":[{"__symbolic":"method"}],"restoreView":[{"__symbolic":"method"}],"addResizeListener":[{"__symbolic":"method"}],"addWindowResizeListener":[{"__symbolic":"method"}],"addResizeObserver":[{"__symbolic":"method"}],"truncateContents":[{"__symbolic":"method"}],"truncateText":[{"__symbolic":"method"}],"applyEllipsis":[{"__symbolic":"method"}]}},"NestedEllipsisContentComponent":{"__symbolic":"class","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"Component","line":2,"character":1},"arguments":[{"selector":"nested-ellipsis-content","template":"\n <ng-content></ng-content>\n ","styles":["\n :host {\n display: block;\n width: 100%;\n height: 100%;\n overflow: hidden;\n }\n "]}]}],"members":{"__ctor__":[{"__symbolic":"constructor","parameters":[{"__symbolic":"reference","module":"@angular/core","name":"ElementRef","line":17,"character":33}]}]}},"EllipsisResizeDetectionEnum":{"ResizeObserver":"resize-observer","Window":"window","Manual":"manual"},"NestedEllipsisModule":{"__symbolic":"class","decorators":[{"__symbolic":"call","expression":{"__symbolic":"reference","module":"@angular/core","name":"NgModule","line":4,"character":1},"arguments":[{"imports":[],"declarations":[{"__symbolic":"reference","name":"NestedEllipsisDirective"},{"__symbolic":"reference","name":"NestedEllipsisContentComponent"}],"entryComponents":[{"__symbolic":"reference","name":"NestedEllipsisContentComponent"}],"exports":[{"__symbolic":"reference","name":"NestedEllipsisDirective"}]}]}],"members":{}}},"origins":{"NestedEllipsisDirective":"./lib/directives/nested-ellipsis.directive","NestedEllipsisContentComponent":"./lib/components/nested-ellipsis-content.component","EllipsisResizeDetectionEnum":"./lib/enums/ellipsis-resize-detection.enum","NestedEllipsisModule":"./lib/nested-ellipsis.module"},"importAs":"ngx-nested-ellipsis"} |
{ | ||
"name": "ngx-nested-ellipsis", | ||
"version": "1.1.0", | ||
"version": "1.1.1", | ||
"description": "Nested multiline html with ellipsis for angular 9+", | ||
@@ -5,0 +5,0 @@ "author": "Florian Lentsch", |
@@ -1,91 +0,24 @@ | ||
# ngx-nested-ellipsis | ||
# NgxNestedEllipsis | ||
Library for angular (>= 9.0.0) providing a directive to display an ellipsis if the containing text would overflow. | ||
This library was generated with [Angular CLI](https://github.com/angular/angular-cli) version 11.0.5. | ||
Supports dynamic html contents. (If you require text contents only, you might want to take a look at [ngx-ellipsis](https://github.com/lentschi/ngx-ellipsis), which offers better performance, but escapes any html contents to text.) | ||
## Code scaffolding | ||
## Demo | ||
Run `ng generate component component-name --project ngx-nested-ellipsis` to generate a new component. You can also use `ng generate directive|pipe|service|class|guard|interface|enum|module --project ngx-nested-ellipsis`. | ||
> Note: Don't forget to add `--project ngx-nested-ellipsis` or else it will be added to the default project in your `angular.json` file. | ||
For a demo either just checkout this project and run `npm install && npm run start` or visit [the StackBlitz demo page](https://stackblitz.com/github/lentschi/ngx-nested-ellipsis?file=src%2Fapp%2Fapp.component.html). | ||
## Build | ||
## Installation | ||
Run `ng build ngx-nested-ellipsis` to build the project. The build artifacts will be stored in the `dist/` directory. | ||
For use in an existing angular project run `npm install ngx-nested-ellipsis --save`. | ||
## Publishing | ||
Then add the installed module to your `app.module.ts`: | ||
After building your library with `ng build ngx-nested-ellipsis`, go to the dist folder `cd dist/ngx-nested-ellipsis` and run `npm publish`. | ||
```typescript | ||
import { NestedEllipsisModule } from 'ngx-nested-ellipsis'; | ||
// ... | ||
@NgModule({ | ||
// ... | ||
imports: [ | ||
// ... | ||
NestedEllipsisModule | ||
] | ||
// ... | ||
}) | ||
export class AppModule {} | ||
``` | ||
## Usage | ||
Anywhere in your template: | ||
```html | ||
<div style="width: 130px; height: 18px;"> | ||
<ng-template [nestedEllipsis]>Your very long <em>rich</em> text</ng-template> | ||
</div> | ||
``` | ||
As you can see, you need to define the dimensions of your element yourself. (ngx-nested-ellipsis doesn't automatically add any element styles.) But of course you don't need to use fixed widths/heights like in these examples. Flex layout shold work just fine for example. | ||
### Extra options | ||
You may add the following attributes to change the directive's behavior: | ||
| attribute | meaning | | ||
| ---- | ---- | | ||
| __nestedEllipsis__ | _required_ Passing true (default) will perform the directive's task otherwise the template will be rendered without truncating its contents.| | ||
| __nestedEllipsisIndicator__ | Passing a string (default: '...') will append it when the passed template has been truncated. Passing a template will append that template instead. | | ||
| __nestedEllipsisWordBoundaries__ | If you pass this attribute, the text won't be truncated at just any character but only at those in the attribute's value. For example `nestedEllipsisWordBoundaries=" "` will allow the text to break at spaces only | | ||
| __nestedEllipsisMayTruncateAtFn__ | Function that lets you specify whether the contents may be truncated at a certain point or not. (see [callback API](#nestedellipsismaytruncateatfn-api)) | | ||
| __nestedEllipsisResizeDetection__ | How resize events should be detected - these are the possible values: <ul><li>__resize-observer__: _default_ Use native ResizeObserver or a ponyfill if not implemented by the browser (See [Web/API/ResizeObserver](https://developer.mozilla.org/en-US/docs/Web/API/ResizeObserver) and [que-etc/resize-observer-polyfill](https://github.com/que-etc/resize-observer-polyfill))</li><li>__window__: Only listen if the whole window has been resized/changed orientation (Possibly better performance, but obviously won't trigger on resize caused directly or indirectly by javascript.)</li><li>__manual__: Ellipsis is never applied automatically. Instead the consuming app may use `#ell="ngxNestedEllipsis"` in the template and `this.ell.applyEllipsis()` in the component code.</li></ul> | | ||
| __nestedEllipsisChange__ | Event emitter - Will be emitted whenever the ellipsis has been recalculated (depending on `nestedEllipsisResizeDetection`). If the text had to be truncated the position of the last visible character will be emitted, else `null`.| | ||
### nestedEllipsisMayTruncateAtFn API | ||
Callback function parameters: | ||
| name | type | description | | ||
| ---- | ---- | ---- | | ||
| __node__ | CharacterData | Text node that is checked for truncation | | ||
| __position__ | number | Position within that text node where the text might be truncated | | ||
Return `true` if truncating at this point in this node should be allowed. | ||
## Build & publish on npm | ||
In case you want to contribute/fork: | ||
1. Run `npm install` | ||
1. Adept version and author in `./projects/ngx-nested-ellipsis/package.json` and commit the changes to your fork. | ||
1. Run `npm run build-lib` which outputs the build to `./dist/ngx-nested-ellipsis`. | ||
1. Copy README.md to `./dist/ngx-nested-ellipsis` and modify it accordingly. | ||
1. Run `cd ./dist/ngx-nested-ellipsis && npm publish`. | ||
## Running unit tests | ||
Run `npm run test ngx-nested-ellipsis` to execute the unit tests via [Karma](https://karma-runner.github.io). | ||
Run `ng test ngx-nested-ellipsis` to execute the unit tests via [Karma](https://karma-runner.github.io). | ||
## Thank you... | ||
## Further help | ||
- ... __Denis Rul__ for writing the [resize-observer-polyfill](https://github.com/que-etc/resize-observer-polyfill) package which is internally used by this module. | ||
## License | ||
MIT | ||
To get more help on the Angular CLI use `ng help` or go check out the [Angular CLI Overview and Command Reference](https://angular.io/cli) page. |
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Sorry, the diff of this file is not supported yet
Major refactor
Supply chain riskPackage has recently undergone a major refactor. It may be unstable or indicate significant internal changes. Use caution when updating to versions that include significant changes.
Found 1 instance in 1 package
260369
0.66%1881
1.24%0
-100%25
-72.83%