From 2fbbfeac31ac9f3e1d36afad2681ee5a4593acbf Mon Sep 17 00:00:00 2001 From: andrew-giangrant Date: Thu, 25 Mar 2021 22:08:11 -0700 Subject: [PATCH] feat: Optional virtualize, imageUrlFn (#369) --- README.md | 36 ++-- package.json | 8 +- src/lib/picker/category.component.ts | 197 ++++++++++++++------ src/lib/picker/emoji-search.service.ts | 61 +++--- src/lib/picker/ngx-emoji/emoji.component.ts | 31 ++- src/lib/picker/ngx-emoji/emoji.service.ts | 33 ++-- src/lib/picker/picker.component.html | 12 +- src/lib/picker/picker.component.ts | 41 ++-- src/lib/picker/preview.component.ts | 109 ++++++----- 9 files changed, 306 insertions(+), 222 deletions(-) diff --git a/README.md b/README.md index 490ccd88..5132b32b 100644 --- a/README.md +++ b/README.md @@ -6,7 +6,7 @@ [![CircleCI](https://badgen.net/circleci/github/scttcper/ngx-emoji-mart)](https://circleci.com/gh/scttcper/ngx-emoji-mart) [![codecov](https://img.shields.io/codecov/c/github/scttcper/ngx-emoji-mart.svg)](https://codecov.io/github/scttcper/ngx-emoji-mart) -**DEMO**: https://ngx-emoji-mart.vercel.app +**DEMO**: https://ngx-emoji-mart.vercel.app This project is a port of [emoji-mart](https://github.com/missive/emoji-mart) by missive @@ -67,9 +67,7 @@ use component - + @@ -92,9 +90,10 @@ use component | **totalFrequentLines** | `4` | number of lines of frequently used emojis | | **i18n** | [`{…}`](#i18n) | [An object](#i18n) containing localized strings | | **isNative** | `false` | Renders the native unicode emoji | -| **set** | `apple` | The emoji set: `'apple', 'google', 'twitter', 'facebook'` | +| **set** | `apple` | The emoji set: `'apple', 'google', 'twitter', 'facebook'` | | **sheetSize** | `64` | The emoji [sheet size](#sheet-sizes): `16, 20, 32, 64` | | **backgroundImageFn** | `((set, sheetSize) => …)` | A Fn that returns that image sheet to use for emojis. Useful for avoiding a request if you have the sheet locally. | +| **imageUrlFn** | `((emoji) => string)` | A Fn that returns the url used for the given emoji. Useful for fetching your own assets. | | **emojisToShowFilter** | `((emoji) => true)` | A Fn to choose whether an emoji should be displayed or not | | **showPreview** | `true` | Display preview section | | **enableSearch** | `true` | Display search bar | @@ -109,6 +108,8 @@ use component | **showSingleCategory** | | show only one category at a time to increase rendering performance | | **useButton** | `false` | Uses button elements for emoji instead of spans | | **enableFrequentEmojiSort** | `false` | Enables re-sorting of emoji on click | +| **virtualize** | `false` | Enables experimental virtualized rendering to render only emoji categories in view | +| **virtualizeOffset** | `0` | use with virtualize option to add or subtract the amount of pixels used to determine whether or not render the category | #### I18n @@ -213,10 +214,10 @@ import { EmojiModule } from '@ctrl/ngx-emoji-mart/ngx-emoji'; ``` | Prop | Required | Default | Description | -| -------------------------------------------- | :------: | ------------------------- | ---------------------------------------------------------------------------------------------------------------- | +| -------------------------------------------- | :------: | ------------------------- | ---------------------------------------------------------------------------------------------------------------- | --- | | **emoji** | ✓ | | Either a string or an `emoji` object | | **size** | ✓ | | The emoji width and height. | -| **isNative** | | `false` | Renders the native unicode emoji | +| **isNative** | | `false` | Renders the native unicode emoji | | **(emojiClick)** | | | Params: `{ emoji, $event }` | | **(emojiLeave)** | | | Params: `{ emoji, $event }` | | **(emojiOver)** | | | Params: `{ emoji, $event }` | @@ -225,9 +226,9 @@ import { EmojiModule } from '@ctrl/ngx-emoji-mart/ngx-emoji'; | **sheetSize** | | `64` | The emoji [sheet size](#sheet-sizes): `16, 20, 32, 64` | | **backgroundImageFn** | | `((set, sheetSize) => …)` | Fn that returns that image sheet to use for emojis. Useful for avoiding a request if you have the sheet locally. | | **skin** | | `1` | Skin color: `1, 2, 3, 4, 5, 6` | -| **tooltip** | | `false` | Show emoji short name when hovering (title) | | -| **hideObsolete** | | `false` | Hides ex: "cop" emoji in favor of female and male emoji | | -| **useButton** | | `false` | Uses button element instead of span | | +| **tooltip** | | `false` | Show emoji short name when hovering (title) | | +| **hideObsolete** | | `false` | Hides ex: "cop" emoji in favor of female and male emoji | | +| **useButton** | | `false` | Uses button element instead of span | | #### Unsupported emojis fallback @@ -236,17 +237,11 @@ Certain sets don’t support all emojis (i.e. Facebook doesn't support `:shrug:` To have the component render `:shrug:` you would need to: ```ts -emojiFallback = (emoji: any, props: any) => - emoji ? `:${emoji.shortNames[0]}:` : props.emoji; +emojiFallback = (emoji: any, props: any) => (emoji ? `:${emoji.shortNames[0]}:` : props.emoji); ``` ```html - + ``` ## Custom emojis @@ -269,8 +264,7 @@ const customEmojis = [ text: '', emoticons: [], keywords: ['test', 'flag'], - spriteUrl: - 'https://unpkg.com/emoji-datasource-twitter@6.0.0/img/twitter/sheets-256/64.png', + spriteUrl: 'https://unpkg.com/emoji-datasource-twitter@6.0.0/img/twitter/sheets-256/64.png', sheet_x: 1, sheet_y: 1, size: 64, @@ -292,7 +286,7 @@ The `Picker` doesn’t have to be mounted for you to take advantage of the advan import { EmojiSearch } from '@ctrl/ngx-emoji-mart'; class ex { constructor(private emojiSearch: EmojiSearch) { - this.emojiSearch.search('christmas').map((o) => o.native); + this.emojiSearch.search('christmas').map(o => o.native); // => [🎄, 🎅🏼, 🔔, 🎁, ⛄️, ❄️] } } diff --git a/package.json b/package.json index bda3e491..c24f24c6 100644 --- a/package.json +++ b/package.json @@ -21,13 +21,13 @@ "@angular-devkit/build-angular": "0.1101.2", "@angular/cli": "11.1.2", "@angular/common": "11.1.1", - "@angular/compiler-cli": "11.1.1", "@angular/compiler": "11.1.1", + "@angular/compiler-cli": "11.1.1", "@angular/core": "11.1.1", "@angular/forms": "11.1.1", "@angular/language-service": "11.1.1", - "@angular/platform-browser-dynamic": "11.1.1", "@angular/platform-browser": "11.1.1", + "@angular/platform-browser-dynamic": "11.1.1", "@ctrl/ngx-github-buttons": "6.1.0", "@types/inflection": "1.5.28", "@types/jasmine": "3.6.3", @@ -41,12 +41,12 @@ "emojilib": "2.4.0", "inflection": "1.12.0", "jasmine-core": "3.6.0", + "karma": "6.0.3", "karma-chrome-launcher": "3.1.0", "karma-coverage-istanbul-reporter": "3.0.3", - "karma-jasmine-html-reporter": "1.5.4", "karma-jasmine": "4.0.1", + "karma-jasmine-html-reporter": "1.5.4", "karma-mocha-reporter": "2.2.5", - "karma": "6.0.3", "ng-packagr": "11.1.2", "rxjs": "6.6.3", "stringify-object": "3.3.0", diff --git a/src/lib/picker/category.component.ts b/src/lib/picker/category.component.ts index 83c909eb..26f7ecb6 100644 --- a/src/lib/picker/category.component.ts +++ b/src/lib/picker/category.component.ts @@ -1,58 +1,92 @@ +import { Emoji, EmojiService } from '@ctrl/ngx-emoji-mart/ngx-emoji'; import { + AfterViewInit, ChangeDetectionStrategy, ChangeDetectorRef, Component, ElementRef, EventEmitter, Input, + OnChanges, OnInit, Output, - ViewChild, + SimpleChanges, + ViewChild } from '@angular/core'; - -import { Emoji, EmojiService } from '@ctrl/ngx-emoji-mart/ngx-emoji'; +import { Observable, Subject } from 'rxjs'; import { EmojiFrequentlyService } from './emoji-frequently.service'; + @Component({ selector: 'emoji-category', template: ` -
-
- - -
- - - - +
+
+ + +
+ +
+
+ +
+
-
-
+
+
+ +
+ +
+ {{ i18n.notfound }} +
+
+
+ + +
- -
- {{ i18n.notfound }} -
- - -
+ `, changeDetection: ChangeDetectionStrategy.OnPush, preserveWhitespaces: false, }) -export class CategoryComponent implements OnInit { +export class CategoryComponent implements OnChanges, OnInit, AfterViewInit { @Input() emojis?: any[] | null; @Input() hasStickyPosition = true; @Input() name = ''; @@ -86,6 +118,8 @@ export class CategoryComponent implements OnInit { @Input() id: any; @Input() hideObsolete = true; @Input() notFoundEmoji?: string; + @Input() virtualize = false; + @Input() virtualizeOffset = 0; @Input() emojiIsNative?: Emoji['isNative']; @Input() emojiSkin!: Emoji['skin']; @Input() emojiSize!: Emoji['size']; @@ -94,6 +128,7 @@ export class CategoryComponent implements OnInit { @Input() emojiForceSize!: Emoji['forceSize']; @Input() emojiTooltip!: Emoji['tooltip']; @Input() emojiBackgroundImageFn?: Emoji['backgroundImageFn']; + @Input() emojiImageUrlFn?: Emoji['imageUrlFn']; @Input() emojiUseButton?: boolean; @Output() emojiOver: Emoji['emojiOver'] = new EventEmitter(); @Output() emojiLeave: Emoji['emojiLeave'] = new EventEmitter(); @@ -101,12 +136,15 @@ export class CategoryComponent implements OnInit { @ViewChild('container', { static: true }) container!: ElementRef; @ViewChild('label', { static: true }) label!: ElementRef; containerStyles: any = {}; + private filteredEmojisSubject = new Subject(); + filteredEmojis$: Observable = this.filteredEmojisSubject.asObservable(); labelStyles: any = {}; labelSpanStyles: any = {}; margin = 0; minMargin = 0; maxMargin = 0; top = 0; + rows = 0; constructor( public ref: ChangeDetectorRef, @@ -126,12 +164,38 @@ export class CategoryComponent implements OnInit { // this.labelSpanStyles = { position: 'absolute' }; } } + + ngOnChanges(changes: SimpleChanges) { + if (changes.emojis?.currentValue?.length !== changes.emojis?.previousValue?.length) { + this.ngAfterViewInit(); + } + } + + ngAfterViewInit() { + if (!this.virtualize || !this.emojis?.length) { + return; + } + + this.emojis = this.filterEmojis(); + + const { width } = this.container.nativeElement.getBoundingClientRect(); + + const perRow = Math.floor(width / (this.emojiSize + 12)); + this.rows = Math.ceil(this.emojis.length / perRow); + + this.containerStyles = { + ...this.containerStyles, + minHeight: `${this.rows * (this.emojiSize + 12) + 28}px`, + }; + + this.ref?.detectChanges(); + + this.handleScroll(this.container.nativeElement.parentNode.parentNode.scrollTop); + } + memoizeSize() { const parent = this.container.nativeElement.parentNode.parentNode; - const { - top, - height, - } = this.container.nativeElement.getBoundingClientRect(); + const { top, height } = this.container.nativeElement.getBoundingClientRect(); const parentTop = parent.getBoundingClientRect().top; const labelHeight = this.label.nativeElement.getBoundingClientRect().height; @@ -148,7 +212,19 @@ export class CategoryComponent implements OnInit { margin = margin < this.minMargin ? this.minMargin : margin; margin = margin > this.maxMargin ? this.maxMargin : margin; + if (this.virtualize) { + const { top, height } = this.container.nativeElement.getBoundingClientRect(); + const parentHeight = this.container.nativeElement.parentNode.parentNode.clientHeight; + + if (parentHeight + (parentHeight + this.virtualizeOffset) >= top && -height - (parentHeight + this.virtualizeOffset) <= top) { + this.filteredEmojisSubject.next(this.emojis); + } else { + this.filteredEmojisSubject.next([]); + } + } + if (margin === this.margin) { + this.ref.detectChanges(); return false; } @@ -157,12 +233,14 @@ export class CategoryComponent implements OnInit { } this.margin = margin; + this.ref.detectChanges(); return true; } getEmojis() { if (this.name === 'Recent') { - let frequentlyUsed = this.recent || this.frequently.get(this.perLine, this.totalFrequentLines); + let frequentlyUsed = + this.recent || this.frequently.get(this.perLine, this.totalFrequentLines); if (!frequentlyUsed || !frequentlyUsed.length) { frequentlyUsed = this.frequently.get(this.perLine, this.totalFrequentLines); } @@ -198,4 +276,19 @@ export class CategoryComponent implements OnInit { trackById(index: number, item: any) { return item; } + + private filterEmojis(): any[] { + const newEmojis = []; + for (const emoji of this.emojis || []) { + if (!emoji) { + continue; + } + const data = this.emojiService.getData(emoji); + if (!data || (data.obsoletedBy && this.hideObsolete) || (!data.unified && !data.custom)) { + continue; + } + newEmojis.push(emoji); + } + return newEmojis; + } } diff --git a/src/lib/picker/emoji-search.service.ts b/src/lib/picker/emoji-search.service.ts index be8b8aca..fa7a30cf 100644 --- a/src/lib/picker/emoji-search.service.ts +++ b/src/lib/picker/emoji-search.service.ts @@ -1,10 +1,6 @@ import { Injectable } from '@angular/core'; -import { - categories, - EmojiData, - EmojiService, -} from '@ctrl/ngx-emoji-mart/ngx-emoji'; +import { categories, EmojiData, EmojiService } from '@ctrl/ngx-emoji-mart/ngx-emoji'; import { intersect } from './utils'; @Injectable({ providedIn: 'root' }) @@ -24,13 +20,13 @@ export class EmojiSearch { const { shortNames, emoticons } = emojiData; const id = shortNames[0]; - emoticons.forEach(emoticon => { + for (const emoticon of emoticons) { if (this.emoticonsList[emoticon]) { - return; + continue; } this.emoticonsList[emoticon] = id; - }); + } this.emojisList[id] = this.emojiService.getSanitizedData(id); this.originalPool[id] = emojiData; @@ -38,14 +34,14 @@ export class EmojiSearch { } addCustomToPool(custom: any, pool: any) { - custom.forEach((emoji: any) => { + for (const emoji of custom) { const emojiId = emoji.id || emoji.shortNames[0]; if (emojiId && !pool[emojiId]) { pool[emojiId] = this.emojiService.getData(emoji); this.emojisList[emojiId] = this.emojiService.getSanitizedData(emoji); } - }); + } } search( @@ -79,28 +75,21 @@ export class EmojiSearch { if (include.length || exclude.length) { pool = {}; - categories.forEach(category => { - const isIncluded = - include && include.length - ? include.indexOf(category.id) > -1 - : true; - const isExcluded = - exclude && exclude.length - ? exclude.indexOf(category.id) > -1 - : false; + for (const category of categories || []) { + const isIncluded = include && include.length ? include.indexOf(category.id) > -1 : true; + const isExcluded = exclude && exclude.length ? exclude.indexOf(category.id) > -1 : false; + if (!isIncluded || isExcluded) { - return; + continue; } - category.emojis?.forEach( - emojiId => { - // Need to make sure that pool gets keyed - // with the correct id, which is why we call emojiService.getData below - const emoji = this.emojiService.getData(emojiId); - pool[emoji?.id ?? ''] = emoji; - } - ); - }); + for (const emojiId of category.emojis || []) { + // Need to make sure that pool gets keyed + // with the correct id, which is why we call emojiService.getData below + const emoji = this.emojiService.getData(emojiId); + pool[emoji?.id ?? ''] = emoji; + } + } if (custom.length) { const customIsIncluded = @@ -142,7 +131,7 @@ export class EmojiSearch { emoji.name, emoji.id, emoji.keywords, - emoji.emoticons + emoji.emoticons, ); } const query = this.emojiSearch[id]; @@ -217,15 +206,19 @@ export class EmojiSearch { return; } - (Array.isArray(strings) ? strings : [strings]).forEach(str => { - (split ? str.split(/[-|_|\s]+/) : [str]).forEach(s => { + const arr = Array.isArray(strings) ? strings : [strings]; + + for (const str of arr) { + const substrings = split ? str.split(/[-|_|\s]+/) : [str]; + + for (let s of substrings) { s = s.toLowerCase(); if (!search.includes(s)) { search.push(s); } - }); - }); + } + } }; addToSearch(shortNames, true); diff --git a/src/lib/picker/ngx-emoji/emoji.component.ts b/src/lib/picker/ngx-emoji/emoji.component.ts index 0903c672..c4336527 100644 --- a/src/lib/picker/ngx-emoji/emoji.component.ts +++ b/src/lib/picker/ngx-emoji/emoji.component.ts @@ -4,7 +4,7 @@ import { EventEmitter, Input, OnChanges, - Output + Output, } from '@angular/core'; import { EmojiData } from './data/data.interfaces'; @@ -16,7 +16,7 @@ export interface Emoji { forceSize: boolean; tooltip: boolean; skin: 1 | 2 | 3 | 4 | 5 | 6; - sheetSize: 16 | 20 | 32 | 64; + sheetSize: 16 | 20 | 32 | 64 | 72; sheetRows?: number; set: 'apple' | 'google' | 'twitter' | 'facebook' | ''; size: number; @@ -26,6 +26,7 @@ export interface Emoji { emojiOver: EventEmitter; emojiLeave: EventEmitter; emojiClick: EventEmitter; + imageUrlFn?: (emoji: EmojiData | null) => string; } export interface EmojiEvent { @@ -72,7 +73,7 @@ export interface EmojiEvent { `, changeDetection: ChangeDetectionStrategy.OnPush, - preserveWhitespaces: false + preserveWhitespaces: false, }) export class EmojiComponent implements OnChanges, Emoji { @Input() skin: Emoji['skin'] = 1; @@ -100,6 +101,7 @@ export class EmojiComponent implements OnChanges, Emoji { isVisible = true; // TODO: replace 4.0.3 w/ dynamic get verison from emoji-datasource in package.json @Input() backgroundImageFn: Emoji['backgroundImageFn'] = DEFAULT_BACKGROUNDFN; + @Input() imageUrlFn?: Emoji['imageUrlFn']; constructor(private emojiService: EmojiService) {} @@ -126,10 +128,7 @@ export class EmojiComponent implements OnChanges, Emoji { return (this.isVisible = false); } - this.label = [data.native] - .concat(data.shortNames) - .filter(Boolean) - .join(', '); + this.label = [data.native].concat(data.shortNames).filter(Boolean).join(', '); if (this.isNative && data.unified && data.native) { // hide older emoji before the split into gendered emoji @@ -145,23 +144,20 @@ export class EmojiComponent implements OnChanges, Emoji { this.style = { width: `${this.size}px`, height: `${this.size}px`, - display: 'inline-block' + display: 'inline-block', }; if (data.spriteUrl && this.sheetRows && this.sheetColumns) { this.style = { ...this.style, backgroundImage: `url(${data.spriteUrl})`, backgroundSize: `${100 * this.sheetColumns}% ${100 * this.sheetRows}%`, - backgroundPosition: this.emojiService.getSpritePosition( - data.sheet, - this.sheetColumns - ) + backgroundPosition: this.emojiService.getSpritePosition(data.sheet, this.sheetColumns), }; } else { this.style = { ...this.style, backgroundImage: `url(${data.imageUrl})`, - backgroundSize: 'contain' + backgroundSize: 'contain', }; } } else { @@ -180,7 +176,8 @@ export class EmojiComponent implements OnChanges, Emoji { this.sheetSize, this.sheetRows, this.backgroundImageFn, - this.sheetColumns + this.sheetColumns, + this.imageUrlFn?.(this.getData()), ); } } @@ -192,11 +189,7 @@ export class EmojiComponent implements OnChanges, Emoji { } getSanitizedData(): EmojiData { - return this.emojiService.getSanitizedData( - this.emoji, - this.skin, - this.set - ) as EmojiData; + return this.emojiService.getSanitizedData(this.emoji, this.skin, this.set) as EmojiData; } handleClick($event: Event) { diff --git a/src/lib/picker/ngx-emoji/emoji.service.ts b/src/lib/picker/ngx-emoji/emoji.service.ts index 988dc95e..830a4e7c 100644 --- a/src/lib/picker/ngx-emoji/emoji.service.ts +++ b/src/lib/picker/ngx-emoji/emoji.service.ts @@ -1,19 +1,13 @@ import { Injectable } from '@angular/core'; -import { - CompressedEmojiData, - EmojiData, - EmojiVariation, -} from './data/data.interfaces'; +import { CompressedEmojiData, EmojiData, EmojiVariation } from './data/data.interfaces'; import { emojis } from './data/emojis'; import { Emoji } from './emoji.component'; const COLONS_REGEX = /^(?:\:([^\:]+)\:)(?:\:skin-tone-(\d)\:)?$/; const SKINS = ['1F3FA', '1F3FB', '1F3FC', '1F3FD', '1F3FE', '1F3FF']; -export const DEFAULT_BACKGROUNDFN = ( - set: string, - sheetSize: number, -) => `https://unpkg.com/emoji-datasource-${set}@6.0.0/img/${set}/sheets-256/${sheetSize}.png`; +export const DEFAULT_BACKGROUNDFN = (set: string, sheetSize: number) => + `https://unpkg.com/emoji-datasource-${set}@6.0.0/img/${set}/sheets-256/${sheetSize}.png`; @Injectable({ providedIn: 'root' }) export class EmojiService { @@ -78,11 +72,7 @@ export class EmojiService { }); } - getData( - emoji: EmojiData | string, - skin?: Emoji['skin'], - set?: Emoji['set'], - ): EmojiData | null { + getData(emoji: EmojiData | string, skin?: Emoji['skin'], set?: Emoji['set']): EmojiData | null { let emojiData: any; if (typeof emoji === 'string') { @@ -144,14 +134,17 @@ export class EmojiService { sheetRows: Emoji['sheetRows'] = 57, backgroundImageFn: Emoji['backgroundImageFn'] = DEFAULT_BACKGROUNDFN, sheetColumns = 58, + url?: string, ) { + const hasImageUrl = !!url; + url = url || backgroundImageFn(set, sheetSize); return { width: `${size}px`, height: `${size}px`, display: 'inline-block', - 'background-image': `url(${backgroundImageFn(set, sheetSize)})`, - 'background-size': `${100 * sheetColumns}% ${100 * sheetRows}%`, - 'background-position': this.getSpritePosition(sheet, sheetColumns), + 'background-image': `url(${url})`, + 'background-size': hasImageUrl ? '100% 100%' : `${100 * sheetColumns}% ${100 * sheetRows}%`, + 'background-position': hasImageUrl ? undefined : this.getSpritePosition(sheet, sheetColumns), }; } @@ -174,11 +167,7 @@ export class EmojiService { return { ...emoji }; } - getSanitizedData( - emoji: string | EmojiData, - skin?: Emoji['skin'], - set?: Emoji['set'], - ) { + getSanitizedData(emoji: string | EmojiData, skin?: Emoji['skin'], set?: Emoji['set']) { return this.sanitize(this.getData(emoji, skin, set)); } } diff --git a/src/lib/picker/picker.component.html b/src/lib/picker/picker.component.html index cfb64ac4..85ee4652 100644 --- a/src/lib/picker/picker.component.html +++ b/src/lib/picker/picker.component.html @@ -1,6 +1,8 @@ -
+ [ngStyle]="style" +>
diff --git a/src/lib/picker/picker.component.ts b/src/lib/picker/picker.component.ts index cd37de2d..b30f86f4 100644 --- a/src/lib/picker/picker.component.ts +++ b/src/lib/picker/picker.component.ts @@ -74,8 +74,7 @@ export class PickerComponent implements OnInit, OnDestroy { @Input() title = 'Emoji Mart™'; @Input() emoji = 'department_store'; @Input() darkMode = !!( - typeof matchMedia === 'function' && - matchMedia('(prefers-color-scheme: dark)').matches + typeof matchMedia === 'function' && matchMedia('(prefers-color-scheme: dark)').matches ); @Input() color = '#ae65c5'; @Input() hideObsolete = true; @@ -95,6 +94,7 @@ export class PickerComponent implements OnInit, OnDestroy { @Input() autoFocus = false; @Input() custom: any[] = []; @Input() hideRecent = true; + @Input() imageUrlFn: Emoji['imageUrlFn']; @Input() include?: string[]; @Input() exclude?: string[]; @Input() notFoundEmoji = 'sleuth_or_spy'; @@ -104,6 +104,8 @@ export class PickerComponent implements OnInit, OnDestroy { @Input() enableFrequentEmojiSort = false; @Input() enableSearch = true; @Input() showSingleCategory = false; + @Input() virtualize = false; + @Input() virtualizeOffset = 0; @Output() emojiClick = new EventEmitter(); @Output() emojiSelect = new EventEmitter(); @Output() skinChange = new EventEmitter(); @@ -113,6 +115,7 @@ export class PickerComponent implements OnInit, OnDestroy { @ViewChildren(CategoryComponent) categoryRefs!: QueryList; scrollHeight = 0; clientHeight = 0; + clientWidth = 0; selected?: string; nextScroll?: string; scrollTop?: number; @@ -141,10 +144,7 @@ export class PickerComponent implements OnInit, OnDestroy { private scrollListener!: () => void; @Input() - backgroundImageFn: Emoji['backgroundImageFn'] = ( - set: string, - sheetSize: number, - ) => + backgroundImageFn: Emoji['backgroundImageFn'] = (set: string, sheetSize: number) => `https://unpkg.com/emoji-datasource-${this.set}@6.0.0/img/${this.set}/sheets-256/${this.sheetSize}.png` constructor( @@ -193,13 +193,9 @@ export class PickerComponent implements OnInit, OnDestroy { for (const category of allCategories) { const isIncluded = - this.include && this.include.length - ? this.include.indexOf(category.id) > -1 - : true; + this.include && this.include.length ? this.include.indexOf(category.id) > -1 : true; const isExcluded = - this.exclude && this.exclude.length - ? this.exclude.indexOf(category.id) > -1 - : false; + this.exclude && this.exclude.length ? this.exclude.indexOf(category.id) > -1 : false; if (!isIncluded || isExcluded) { continue; } @@ -255,7 +251,9 @@ export class PickerComponent implements OnInit, OnDestroy { // Need to be careful if small number of categories const categoriesToLoadFirst = Math.min(this.categories.length, 3); - this.setActiveCategories(this.activeCategories = this.categories.slice(0, categoriesToLoadFirst)); + this.setActiveCategories( + (this.activeCategories = this.categories.slice(0, categoriesToLoadFirst)), + ); // Trim last active category const lastActiveCategoryEmojis = this.categories[categoriesToLoadFirst - 1].emojis!.slice(); @@ -305,7 +303,7 @@ export class PickerComponent implements OnInit, OnDestroy { setActiveCategories(categoriesToMakeActive: Array) { if (this.showSingleCategory) { this.activeCategories = categoriesToMakeActive.filter( - x => (x.name === this.selected || x === this.SEARCH_CATEGORY) + x => x.name === this.selected || x === this.SEARCH_CATEGORY, ); } else { this.activeCategories = categoriesToMakeActive; @@ -318,6 +316,7 @@ export class PickerComponent implements OnInit, OnDestroy { const target = this.scrollRef.nativeElement; this.scrollHeight = target.scrollHeight; this.clientHeight = target.clientHeight; + this.clientWidth = target.clientWidth; } } handleAnchorClick($event: { category: EmojiCategory; index: number }) { @@ -343,13 +342,18 @@ export class PickerComponent implements OnInit, OnDestroy { } this.scrollRef.nativeElement.scrollTop = top; } - this.selected = $event.category.name; this.nextScroll = $event.category.name; + + // handle component scrolling to load emojis + for (const category of this.categories) { + const componentToScroll = this.categoryRefs.find(({ id }) => id === category.id); + componentToScroll?.handleScroll(this.scrollRef.nativeElement.scrollTop); + } } categoryTrack(index: number, item: any) { return item.id; } - handleScroll() { + handleScroll(noSelectionChange = false) { if (this.nextScroll) { this.selected = this.nextScroll; this.nextScroll = undefined; @@ -389,9 +393,11 @@ export class PickerComponent implements OnInit, OnDestroy { this.scrollTop = target.scrollTop; } // This will allow us to run the change detection only when the category changes. - if (activeCategory && activeCategory.name !== this.selected) { + if (!noSelectionChange && activeCategory && activeCategory.name !== this.selected) { this.selected = activeCategory.name; this.ref.detectChanges(); + } else if (noSelectionChange) { + this.ref.detectChanges(); } } handleSearch($emojis: any[] | null) { @@ -445,6 +451,7 @@ export class PickerComponent implements OnInit, OnDestroy { this.previewEmoji = $event.emoji; this.cancelAnimationFrame(); + this.ref?.detectChanges(); } handleEmojiLeave() { if (!this.showPreview || !this.previewRef) { diff --git a/src/lib/picker/preview.component.ts b/src/lib/picker/preview.component.ts index 0229dfff..eb139004 100644 --- a/src/lib/picker/preview.component.ts +++ b/src/lib/picker/preview.component.ts @@ -1,3 +1,4 @@ +import { Emoji, EmojiData, EmojiService } from '@ctrl/ngx-emoji-mart/ngx-emoji'; import { ChangeDetectionStrategy, ChangeDetectorRef, @@ -5,64 +6,69 @@ import { EventEmitter, Input, OnChanges, - Output, + Output } from '@angular/core'; -import { Emoji, EmojiData, EmojiService } from '@ctrl/ngx-emoji-mart/ngx-emoji'; @Component({ selector: 'emoji-preview', template: ` -
-
- -
- -
-
{{ emojiData.name }}
-
- - :{{ short_name }}: - +
+
+
-
- - {{ emoticon }} - + +
+
{{ emojiData.name }}
+
+ + :{{ short_name }}: + +
+
+ + {{ emoticon }} + +
-
-
-
- -
+
+
+ +
-
- {{ title }} -
+
+ {{ title }} +
-
- - +
+ + +
-
`, changeDetection: ChangeDetectionStrategy.OnPush, preserveWhitespaces: false, @@ -78,20 +84,22 @@ export class PreviewComponent implements OnChanges { @Input() emojiSet?: Emoji['set']; @Input() emojiSheetSize?: Emoji['sheetSize']; @Input() emojiBackgroundImageFn?: Emoji['backgroundImageFn']; + @Input() emojiImageUrlFn?: Emoji['imageUrlFn']; @Output() skinChange = new EventEmitter(); emojiData: Partial = {}; listedEmoticons?: string[]; - constructor( - public ref: ChangeDetectorRef, - private emojiService: EmojiService, - ) {} + constructor(public ref: ChangeDetectorRef, private emojiService: EmojiService) {} ngOnChanges() { if (!this.emoji) { return; } - this.emojiData = this.emojiService.getData(this.emoji, this.emojiSkin, this.emojiSet) as EmojiData; + this.emojiData = this.emojiService.getData( + this.emoji, + this.emojiSkin, + this.emojiSet, + ) as EmojiData; const knownEmoticons: string[] = []; const listedEmoticons: string[] = []; const emoitcons = this.emojiData.emoticons || []; @@ -103,5 +111,6 @@ export class PreviewComponent implements OnChanges { listedEmoticons.push(emoticon); }); this.listedEmoticons = listedEmoticons; + this.ref?.detectChanges(); } }