2023-01-03 19:55:46 +00:00
// Copyright 2020 Signal Messenger, LLC
2020-10-30 20:34:04 +00:00
// SPDX-License-Identifier: AGPL-3.0-only
2020-10-21 16:53:32 +00:00
import Quill from 'quill' ;
import Delta from 'quill-delta' ;
import React from 'react' ;
2022-08-18 15:02:13 +00:00
import _ , { isNumber } from 'lodash' ;
2020-10-21 16:53:32 +00:00
import { Popper } from 'react-popper' ;
import classNames from 'classnames' ;
import { createPortal } from 'react-dom' ;
2022-08-18 15:02:13 +00:00
import type { VirtualElement } from '@popperjs/core' ;
2024-03-21 16:35:54 +00:00
import { convertShortName , isShortName } from '../../components/emoji/lib' ;
import type { SearchFnType } from '../../components/emoji/lib' ;
2020-10-21 16:53:32 +00:00
import { Emoji } from '../../components/emoji/Emoji' ;
2021-10-26 19:15:33 +00:00
import type { EmojiPickDataType } from '../../components/emoji/EmojiPicker' ;
2020-11-05 21:18:42 +00:00
import { getBlotTextPartitions , matchBlotTextPartitions } from '../util' ;
2022-09-27 20:24:21 +00:00
import { handleOutsideClick } from '../../util/handleOutsideClick' ;
2022-08-18 15:02:13 +00:00
import * as log from '../../logging/log' ;
2020-10-21 16:53:32 +00:00
2021-01-07 21:39:17 +00:00
const Keyboard = Quill . import ( 'modules/keyboard' ) ;
2021-01-14 18:07:05 +00:00
type EmojiPickerOptions = {
2020-10-21 16:53:32 +00:00
onPickEmoji : ( emoji : EmojiPickDataType ) = > void ;
setEmojiPickerElement : ( element : JSX.Element | null ) = > void ;
skinTone : number ;
2024-03-21 16:35:54 +00:00
search : SearchFnType ;
2021-01-14 18:07:05 +00:00
} ;
2020-10-21 16:53:32 +00:00
export class EmojiCompletion {
2024-03-21 16:35:54 +00:00
results : Array < string > ;
2020-10-21 16:53:32 +00:00
index : number ;
options : EmojiPickerOptions ;
root : HTMLDivElement ;
quill : Quill ;
2022-09-29 20:13:45 +00:00
outsideClickDestructor ? : ( ) = > void ;
2022-09-27 20:24:21 +00:00
2020-10-21 16:53:32 +00:00
constructor ( quill : Quill , options : EmojiPickerOptions ) {
this . results = [ ] ;
this . index = 0 ;
this . options = options ;
this . root = document . body . appendChild ( document . createElement ( 'div' ) ) ;
this . quill = quill ;
const clearResults = ( ) = > {
if ( this . results . length ) {
this . reset ( ) ;
}
return true ;
} ;
const changeIndex = ( by : number ) = > ( ) : boolean = > {
if ( this . results . length ) {
this . changeIndex ( by ) ;
return false ;
}
return true ;
} ;
2021-01-07 21:39:17 +00:00
this . quill . keyboard . addBinding ( { key : Keyboard.keys.UP } , changeIndex ( - 1 ) ) ;
this . quill . keyboard . addBinding ( { key : Keyboard.keys.RIGHT } , clearResults ) ;
this . quill . keyboard . addBinding ( { key : Keyboard.keys.DOWN } , changeIndex ( 1 ) ) ;
this . quill . keyboard . addBinding ( { key : Keyboard.keys.LEFT } , clearResults ) ;
2020-11-11 22:54:23 +00:00
this . quill . keyboard . addBinding (
{
// 186 + Shift = Colon
key : 186 ,
shiftKey : true ,
} ,
( ) = > this . onTextChange ( true )
) ;
this . quill . keyboard . addBinding (
{
// 58 = Also Colon
key : 58 ,
} ,
( ) = > this . onTextChange ( true )
) ;
2020-10-21 16:53:32 +00:00
2020-11-11 22:54:23 +00:00
this . quill . on (
'text-change' ,
_ . debounce ( ( ) = > this . onTextChange ( ) , 100 )
) ;
2020-11-03 01:19:52 +00:00
this . quill . on ( 'selection-change' , this . onSelectionChange . bind ( this ) ) ;
2020-10-21 16:53:32 +00:00
}
destroy ( ) : void {
2022-09-29 20:13:45 +00:00
this . outsideClickDestructor ? . ( ) ;
this . outsideClickDestructor = undefined ;
2020-10-21 16:53:32 +00:00
this . root . remove ( ) ;
}
changeIndex ( by : number ) : void {
this . index = ( this . index + by + this . results . length ) % this . results . length ;
this . render ( ) ;
}
getCurrentLeafTextPartitions ( ) : [ string , string ] {
const range = this . quill . getSelection ( ) ;
2020-11-05 21:18:42 +00:00
const [ blot , index ] = this . quill . getLeaf ( range ? range . index : - 1 ) ;
2020-10-21 16:53:32 +00:00
2021-09-17 16:21:33 +00:00
return getBlotTextPartitions ( blot . text , index ) ;
2020-10-21 16:53:32 +00:00
}
2020-11-03 01:19:52 +00:00
onSelectionChange ( ) : void {
// Selection should never change while we're editing an emoji
this . reset ( ) ;
}
2020-11-11 22:54:23 +00:00
onTextChange ( justPressedColon = false ) : boolean {
const PASS_THROUGH = true ;
const INTERCEPT = false ;
2020-10-21 16:53:32 +00:00
const range = this . quill . getSelection ( ) ;
2022-09-13 21:48:09 +00:00
if ( ! range ) {
return PASS_THROUGH ;
}
2020-10-21 16:53:32 +00:00
2020-11-05 21:18:42 +00:00
const [ blot , index ] = this . quill . getLeaf ( range . index ) ;
const [ leftTokenTextMatch , rightTokenTextMatch ] = matchBlotTextPartitions (
blot ,
index ,
2024-03-21 16:35:54 +00:00
/ ( ? < = ^ | \ s ) : ( [ - + 0 - 9 \ p { A l p h a } _ ] * ) ( : ? ) $ / i u ,
/ ^ ( [ - + 0 - 9 \ p { A l p h a } _ ] * ) : / i u
2020-10-27 00:13:49 +00:00
) ;
2020-10-21 16:53:32 +00:00
2020-11-05 21:18:42 +00:00
if ( leftTokenTextMatch ) {
const [ , leftTokenText , isSelfClosing ] = leftTokenTextMatch ;
2020-10-21 16:53:32 +00:00
2020-11-11 22:54:23 +00:00
if ( isSelfClosing || justPressedColon ) {
2020-11-05 21:18:42 +00:00
if ( isShortName ( leftTokenText ) ) {
2020-11-11 22:54:23 +00:00
const numberOfColons = isSelfClosing ? 2 : 1 ;
2024-03-21 16:35:54 +00:00
this . insertEmoji (
leftTokenText ,
range . index - leftTokenText . length - numberOfColons ,
leftTokenText . length + numberOfColons
) ;
return INTERCEPT ;
2020-10-21 16:53:32 +00:00
}
2024-03-21 16:35:54 +00:00
this . reset ( ) ;
return PASS_THROUGH ;
2020-10-21 16:53:32 +00:00
}
2020-11-05 21:18:42 +00:00
if ( rightTokenTextMatch ) {
const [ , rightTokenText ] = rightTokenTextMatch ;
const tokenText = leftTokenText + rightTokenText ;
if ( isShortName ( tokenText ) ) {
2024-03-21 16:35:54 +00:00
this . insertEmoji (
2020-11-05 21:18:42 +00:00
tokenText ,
2024-03-21 16:35:54 +00:00
range . index - leftTokenText . length - 1 ,
tokenText . length + 2
2020-10-21 16:53:32 +00:00
) ;
2024-03-21 16:35:54 +00:00
return INTERCEPT ;
2020-10-21 16:53:32 +00:00
}
}
2022-10-17 16:54:46 +00:00
if ( leftTokenText . length < 2 ) {
2020-11-05 21:18:42 +00:00
this . reset ( ) ;
2020-11-11 22:54:23 +00:00
return PASS_THROUGH ;
2020-11-05 21:18:42 +00:00
}
2020-10-21 16:53:32 +00:00
2024-03-21 16:35:54 +00:00
const showEmojiResults = this . options . search ( leftTokenText , 10 ) ;
2020-10-21 16:53:32 +00:00
2020-11-05 21:18:42 +00:00
if ( showEmojiResults . length > 0 ) {
this . results = showEmojiResults ;
2021-01-07 21:39:17 +00:00
this . index = Math . min ( this . results . length - 1 , this . index ) ;
2020-11-05 21:18:42 +00:00
this . render ( ) ;
} else if ( this . results . length !== 0 ) {
this . reset ( ) ;
}
} else if ( this . results . length !== 0 ) {
2020-10-21 16:53:32 +00:00
this . reset ( ) ;
}
2020-11-11 22:54:23 +00:00
return PASS_THROUGH ;
2020-10-21 16:53:32 +00:00
}
2023-05-10 00:40:19 +00:00
getAttributesForInsert ( index : number ) : Record < string , unknown > {
const character = index > 0 ? index - 1 : 0 ;
const contents = this . quill . getContents ( character , 1 ) ;
return contents . ops . reduce (
( acc , op ) = > ( { acc , . . . op . attributes } ) ,
{ } as Record < string , unknown >
) ;
}
2020-10-21 16:53:32 +00:00
completeEmoji ( ) : void {
const range = this . quill . getSelection ( ) ;
2022-09-14 21:40:44 +00:00
if ( range == null ) {
2022-09-13 21:48:09 +00:00
return ;
}
2020-10-21 16:53:32 +00:00
const emoji = this . results [ this . index ] ;
const [ leafText ] = this . getCurrentLeafTextPartitions ( ) ;
2024-03-21 16:35:54 +00:00
const tokenTextMatch = / : ( [ - + 0 - 9 \ p { A l p h a } _ ] * ) ( : ? ) $ / i u . e x e c ( l e a f T e x t ) ;
2020-10-21 16:53:32 +00:00
2022-09-14 21:40:44 +00:00
if ( tokenTextMatch == null ) {
2022-09-13 21:48:09 +00:00
return ;
}
2020-10-21 16:53:32 +00:00
const [ , tokenText ] = tokenTextMatch ;
this . insertEmoji (
emoji ,
range . index - tokenText . length - 1 ,
tokenText . length + 1 ,
true
) ;
}
insertEmoji (
2024-03-21 16:35:54 +00:00
shortName : string ,
2020-10-21 16:53:32 +00:00
index : number ,
range : number ,
withTrailingSpace = false
) : void {
2024-03-21 16:35:54 +00:00
const emoji = convertShortName ( shortName , this . options . skinTone ) ;
2020-10-21 16:53:32 +00:00
2023-12-18 23:22:46 +00:00
const delta = new Delta ( )
. retain ( index )
. delete ( range )
. insert ( {
emoji : { value : emoji } ,
} ) ;
2020-10-21 16:53:32 +00:00
if ( withTrailingSpace ) {
2023-05-10 00:40:19 +00:00
// The extra space we add won't be formatted unless we manually provide attributes
const attributes = this . getAttributesForInsert ( range - 1 ) ;
this . quill . updateContents ( delta . insert ( ' ' , attributes ) , 'user' ) ;
2020-10-21 16:53:32 +00:00
this . quill . setSelection ( index + 2 , 0 , 'user' ) ;
} else {
this . quill . updateContents ( delta , 'user' ) ;
this . quill . setSelection ( index + 1 , 0 , 'user' ) ;
}
this . options . onPickEmoji ( {
2024-03-21 16:35:54 +00:00
shortName ,
2020-10-21 16:53:32 +00:00
skinTone : this.options.skinTone ,
} ) ;
this . reset ( ) ;
}
reset ( ) : void {
if ( this . results . length ) {
this . results = [ ] ;
this . index = 0 ;
this . render ( ) ;
}
}
onUnmount ( ) : void {
2022-09-29 20:13:45 +00:00
this . outsideClickDestructor ? . ( ) ;
this . outsideClickDestructor = undefined ;
this . options . setEmojiPickerElement ( null ) ;
2020-10-21 16:53:32 +00:00
}
render ( ) : void {
const { results : emojiResults , index : emojiResultsIndex } = this ;
if ( emojiResults . length === 0 ) {
2022-09-29 20:13:45 +00:00
this . onUnmount ( ) ;
2020-10-21 16:53:32 +00:00
return ;
}
2022-08-18 15:02:13 +00:00
// a virtual reference to the text we are trying to auto-complete
const reference : VirtualElement = {
getBoundingClientRect() {
const selection = window . getSelection ( ) ;
// there's a selection and at least one range
2022-09-14 21:40:44 +00:00
if ( selection != null && selection . rangeCount !== 0 ) {
2022-08-18 15:02:13 +00:00
// grab the first range, the one the user is actually on right now
// clone it so we don't actually modify the user's selection/caret position
const range = selection . getRangeAt ( 0 ) . cloneRange ( ) ;
// if for any reason the range is a selection (not just a caret)
// collapse it to just a caret, so we can walk it back to the :word
range . collapse ( true ) ;
// if we can, position the popper at the beginning of the emoji text (:word)
2022-09-09 18:15:28 +00:00
const textBeforeCursor = range . endContainer . textContent ? . slice (
0 ,
range . startOffset
) ;
const startOfEmojiText = textBeforeCursor ? . lastIndexOf ( ':' ) ;
2022-08-18 15:02:13 +00:00
if (
2022-09-09 18:15:28 +00:00
textBeforeCursor &&
2022-08-18 15:02:13 +00:00
isNumber ( startOfEmojiText ) &&
startOfEmojiText !== - 1
) {
2022-09-09 18:15:28 +00:00
range . setStart ( range . endContainer , startOfEmojiText ) ;
2022-08-18 15:02:13 +00:00
} else {
log . warn (
2022-09-09 18:15:28 +00:00
` Could not find the beginning of the emoji word to be completed. startOfEmojiText= ${ startOfEmojiText } , textBeforeCursor.length= ${ textBeforeCursor ? . length } , range.offsets= ${ range . startOffset } - ${ range . endOffset } `
2022-08-18 15:02:13 +00:00
) ;
}
return range . getClientRects ( ) [ 0 ] ;
}
log . warn ( 'No selection range when auto-completing emoji' ) ;
return new DOMRect ( ) ; // don't crash just because we couldn't get a rectangle
} ,
} ;
2020-10-21 16:53:32 +00:00
const element = createPortal (
2022-08-18 15:02:13 +00:00
< Popper placement = "top-start" referenceElement = { reference } >
2020-10-21 16:53:32 +00:00
{ ( { ref , style } ) = > (
< div
ref = { ref }
2020-11-03 01:19:52 +00:00
className = "module-composition-input__suggestions"
2020-10-21 16:53:32 +00:00
style = { style }
role = "listbox"
aria - expanded
aria - activedescendant = { ` emoji-result-- ${
2024-03-21 16:35:54 +00:00
emojiResults . length ? emojiResults [ emojiResultsIndex ] : ''
2020-10-21 16:53:32 +00:00
} ` }
tabIndex = { 0 }
>
{ emojiResults . map ( ( emoji , index ) = > (
< button
type = "button"
2024-03-21 16:35:54 +00:00
key = { emoji }
id = { ` emoji-result-- ${ emoji } ` }
2020-10-21 16:53:32 +00:00
role = "option button"
aria - selected = { emojiResultsIndex === index }
onClick = { ( ) = > {
this . index = index ;
this . completeEmoji ( ) ;
} }
className = { classNames (
2020-11-03 01:19:52 +00:00
'module-composition-input__suggestions__row' ,
2020-10-21 16:53:32 +00:00
emojiResultsIndex === index
2020-11-03 01:19:52 +00:00
? 'module-composition-input__suggestions__row--selected'
2020-10-21 16:53:32 +00:00
: null
) }
>
< Emoji
2024-03-21 16:35:54 +00:00
shortName = { emoji }
2020-10-21 16:53:32 +00:00
size = { 16 }
skinTone = { this . options . skinTone }
/ >
2020-11-03 01:19:52 +00:00
< div className = "module-composition-input__suggestions__row__short-name" >
2024-03-21 16:35:54 +00:00
: { emoji } :
2020-10-21 16:53:32 +00:00
< / div >
< / button >
) ) }
< / div >
) }
< / Popper > ,
2022-09-29 20:13:45 +00:00
this . root
) ;
// Just to make sure that we don't propagate outside clicks until this
// is closed.
this . outsideClickDestructor ? . ( ) ;
this . outsideClickDestructor = handleOutsideClick (
( ) = > {
this . onUnmount ( ) ;
return true ;
} ,
{
name : 'quill.emoji.completion' ,
containerElements : [ this . root ] ,
}
2020-10-21 16:53:32 +00:00
) ;
this . options . setEmojiPickerElement ( element ) ;
}
}