Avoid emojifying on invisible text (#5558)
This commit is contained in:
		
							parent
							
								
									84cfee2488
								
							
						
					
					
						commit
						782224c991
					
				
					 3 changed files with 60 additions and 9 deletions
				
			
		|  | @ -57,5 +57,21 @@ describe('emoji', () => { | |||
|     it('does an emoji whose filename is irregular', () => { | ||||
|       expect(emojify('↙️')).toEqual('<img draggable="false" class="emojione" alt="↙️" title=":arrow_lower_left:" src="/emoji/2199.svg" />'); | ||||
|     }); | ||||
| 
 | ||||
|     it('avoid emojifying on invisible text', () => { | ||||
|       expect(emojify('<a href="http://example.com/test%F0%9F%98%84"><span class="invisible">http://</span><span class="ellipsis">example.com/te</span><span class="invisible">st😄</span></a>')) | ||||
|         .toEqual('<a href="http://example.com/test%F0%9F%98%84"><span class="invisible">http://</span><span class="ellipsis">example.com/te</span><span class="invisible">st😄</span></a>'); | ||||
|       expect(emojify('<span class="invisible">:luigi:</span>', { ':luigi:': { static_url: 'luigi.exe' } })) | ||||
|         .toEqual('<span class="invisible">:luigi:</span>'); | ||||
|     }); | ||||
| 
 | ||||
|     it('avoid emojifying on invisible text with nested tags', () => { | ||||
|       expect(emojify('<span class="invisible">😄<span class="foo">bar</span>😴</span>😇')) | ||||
|         .toEqual('<span class="invisible">😄<span class="foo">bar</span>😴</span><img draggable="false" class="emojione" alt="😇" title=":innocent:" src="/emoji/1f607.svg" />'); | ||||
|       expect(emojify('<span class="invisible">😄<span class="invisible">😕</span>😴</span>😇')) | ||||
|         .toEqual('<span class="invisible">😄<span class="invisible">😕</span>😴</span><img draggable="false" class="emojione" alt="😇" title=":innocent:" src="/emoji/1f607.svg" />'); | ||||
|       expect(emojify('<span class="invisible">😄<br/>😴</span>😇')) | ||||
|         .toEqual('<span class="invisible">😄<br/>😴</span><img draggable="false" class="emojione" alt="😇" title=":innocent:" src="/emoji/1f607.svg" />'); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
|  |  | |||
|  | @ -7,10 +7,12 @@ const trie = new Trie(Object.keys(unicodeMapping)); | |||
| const assetHost = process.env.CDN_HOST || ''; | ||||
| 
 | ||||
| const emojify = (str, customEmojis = {}) => { | ||||
|   let rtn = ''; | ||||
|   const tagCharsWithoutEmojis = '<&'; | ||||
|   const tagCharsWithEmojis = Object.keys(customEmojis).length ? '<&:' : '<&'; | ||||
|   let rtn = '', tagChars = tagCharsWithEmojis, invisible = 0; | ||||
|   for (;;) { | ||||
|     let match, i = 0, tag; | ||||
|     while (i < str.length && (tag = '<&:'.indexOf(str[i])) === -1 && !(match = trie.search(str.slice(i)))) { | ||||
|     while (i < str.length && (tag = tagChars.indexOf(str[i])) === -1 && (invisible || !(match = trie.search(str.slice(i))))) { | ||||
|       i += str.codePointAt(i) < 65536 ? 1 : 2; | ||||
|     } | ||||
|     let rend, replacement = ''; | ||||
|  | @ -34,7 +36,26 @@ const emojify = (str, customEmojis = {}) => { | |||
|       })()) rend = ++i; | ||||
|     } else if (tag >= 0) { // <, &
 | ||||
|       rend = str.indexOf('>;'[tag], i + 1) + 1; | ||||
|       if (!rend) break; | ||||
|       if (!rend) { | ||||
|         break; | ||||
|       } | ||||
|       if (tag === 0) { | ||||
|         if (invisible) { | ||||
|           if (str[i + 1] === '/') { // closing tag
 | ||||
|             if (!--invisible) { | ||||
|               tagChars = tagCharsWithEmojis; | ||||
|             } | ||||
|           } else if (str[rend - 2] !== '/') { // opening tag
 | ||||
|             invisible++; | ||||
|           } | ||||
|         } else { | ||||
|           if (str.startsWith('<span class="invisible">', i)) { | ||||
|             // avoid emojifying on invisible text
 | ||||
|             invisible = 1; | ||||
|             tagChars = tagCharsWithoutEmojis; | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|       i = rend; | ||||
|     } else { // matched to unicode emoji
 | ||||
|       const { filename, shortCode } = unicodeMapping[match]; | ||||
|  |  | |||
		Reference in a new issue