import navigation from '../../../client/state/navigation';
import AsyncSearch from '../../../util/AsyncSearch';
import { addRecentEmoji, getRecentEmojis } from './recent';
+import { TWEMOJI_BASE_URL } from '../../../util/twemojify';
import Text from '../../atoms/text/Text';
import RawIcon from '../../atoms/system-icons/RawIcon';
const emoji = groupEmojis[emojiIndex];
emojiRow.push(
<span key={emojiIndex}>
- {
- emoji.hexcode
- // This is a unicode emoji, and should be rendered with twemoji
- ? parse(twemoji.parse(
- emoji.unicode,
- {
- attributes: () => ({
- unicode: emoji.unicode,
- shortcodes: emoji.shortcodes?.toString(),
- hexcode: emoji.hexcode,
- loading: 'lazy',
- }),
- },
- ))
- // This is a custom emoji, and should be render as an mxc
- : (
- <img
- className="emoji"
- draggable="false"
- loading="lazy"
- alt={emoji.shortcode}
- unicode={`:${emoji.shortcode}:`}
- shortcodes={emoji.shortcode}
- src={initMatrix.matrixClient.mxcUrlToHttp(emoji.mxc)}
- data-mx-emoticon={emoji.mxc}
- />
- )
- }
- </span>,
+ {emoji.hexcode ? (
+ // This is a unicode emoji, and should be rendered with twemoji
+ parse(
+ twemoji.parse(emoji.unicode, {
+ attributes: () => ({
+ unicode: emoji.unicode,
+ shortcodes: emoji.shortcodes?.toString(),
+ hexcode: emoji.hexcode,
+ loading: 'lazy',
+ }),
+ base: TWEMOJI_BASE_URL,
+ })
+ )
+ ) : (
+ // This is a custom emoji, and should be render as an mxc
+ <img
+ className="emoji"
+ draggable="false"
+ loading="lazy"
+ alt={emoji.shortcode}
+ unicode={`:${emoji.shortcode}:`}
+ shortcodes={emoji.shortcode}
+ src={initMatrix.matrixClient.mxcUrlToHttp(emoji.mxc)}
+ data-mx-emoticon={emoji.mxc}
+ />
+ )}
+ </span>
);
}
- emojiBoard.push(<div key={r} className="emoji-row">{emojiRow}</div>);
+ emojiBoard.push(
+ <div key={r} className="emoji-row">
+ {emojiRow}
+ </div>
+ );
}
return emojiBoard;
}
return (
<div className="emoji-group">
- <Text className="emoji-group__header" variant="b2" weight="bold">{name}</Text>
+ <Text className="emoji-group__header" variant="b2" weight="bold">
+ {name}
+ </Text>
{groupEmojis.length !== 0 && <div className="emoji-set noselect">{getEmojiBoard()}</div>}
</div>
);
EmojiGroup.propTypes = {
name: PropTypes.string.isRequired,
- groupEmojis: PropTypes.arrayOf(PropTypes.shape({
- length: PropTypes.number,
- unicode: PropTypes.string,
- hexcode: PropTypes.string,
- mxc: PropTypes.string,
- shortcode: PropTypes.string,
- shortcodes: PropTypes.oneOfType([
- PropTypes.string,
- PropTypes.arrayOf(PropTypes.string),
- ]),
- })).isRequired,
+ groupEmojis: PropTypes.arrayOf(
+ PropTypes.shape({
+ length: PropTypes.number,
+ unicode: PropTypes.string,
+ hexcode: PropTypes.string,
+ mxc: PropTypes.string,
+ shortcode: PropTypes.string,
+ shortcodes: PropTypes.oneOfType([PropTypes.string, PropTypes.arrayOf(PropTypes.string)]),
+ })
+ ).isRequired,
};
const asyncSearch = new AsyncSearch();
if (searchedEmojis === null) return false;
- return <EmojiGroup key="-1" name={searchedEmojis.emojis.length === 0 ? 'No search result found' : 'Search results'} groupEmojis={searchedEmojis.emojis} />;
+ return (
+ <EmojiGroup
+ key="-1"
+ name={searchedEmojis.emojis.length === 0 ? 'No search result found' : 'Search results'}
+ groupEmojis={searchedEmojis.emojis}
+ />
+ );
}
function EmojiBoard({ onSelect, searchRef }) {
if (typeof shortcodes === 'undefined') shortcodes = undefined;
else shortcodes = shortcodes.split(',');
return {
- unicode, hexcode, shortcodes, mxc,
+ unicode,
+ hexcode,
+ shortcodes,
+ mxc,
};
}
const parentIds = initMatrix.roomList.getAllParentSpaces(room.roomId);
const parentRooms = [...parentIds].map((id) => mx.getRoom(id));
if (room) {
- const packs = getRelevantPacks(
- room.client,
- [room, ...parentRooms],
- ).filter((pack) => pack.getEmojis().length !== 0);
+ const packs = getRelevantPacks(room.client, [room, ...parentRooms]).filter(
+ (pack) => pack.getEmojis().length !== 0
+ );
// Set an index for each pack so that we know where to jump when the user uses the nav
for (let i = 0; i < packs.length; i += 1) {
/>
)}
<div className="emoji-board__nav-custom">
- {
- availableEmojis.map((pack) => {
- const src = initMatrix.matrixClient
- .mxcUrlToHttp(pack.avatarUrl ?? pack.getEmojis()[0].mxc);
- return (
- <IconButton
- onClick={() => openGroup(recentOffset + pack.packIndex)}
- src={src}
- key={pack.packIndex}
- tooltip={pack.displayName ?? 'Unknown'}
- tooltipPlacement="left"
- isImage
- />
- );
- })
- }
- </div>
- <div className="emoji-board__nav-twemoji">
- {
- [
- [0, EmojiIC, 'Smilies'],
- [1, DogIC, 'Animals'],
- [2, CupIC, 'Food'],
- [3, BallIC, 'Activities'],
- [4, PhotoIC, 'Travel'],
- [5, BulbIC, 'Objects'],
- [6, PeaceIC, 'Symbols'],
- [7, FlagIC, 'Flags'],
- ].map(([indx, ico, name]) => (
+ {availableEmojis.map((pack) => {
+ const src = initMatrix.matrixClient.mxcUrlToHttp(
+ pack.avatarUrl ?? pack.getEmojis()[0].mxc
+ );
+ return (
<IconButton
- onClick={() => openGroup(recentOffset + availableEmojis.length + indx)}
- key={indx}
- src={ico}
- tooltip={name}
+ onClick={() => openGroup(recentOffset + pack.packIndex)}
+ src={src}
+ key={pack.packIndex}
+ tooltip={pack.displayName ?? 'Unknown'}
tooltipPlacement="left"
+ isImage
/>
- ))
- }
+ );
+ })}
+ </div>
+ <div className="emoji-board__nav-twemoji">
+ {[
+ [0, EmojiIC, 'Smilies'],
+ [1, DogIC, 'Animals'],
+ [2, CupIC, 'Food'],
+ [3, BallIC, 'Activities'],
+ [4, PhotoIC, 'Travel'],
+ [5, BulbIC, 'Objects'],
+ [6, PeaceIC, 'Symbols'],
+ [7, FlagIC, 'Flags'],
+ ].map(([indx, ico, name]) => (
+ <IconButton
+ onClick={() => openGroup(recentOffset + availableEmojis.length + indx)}
+ key={indx}
+ src={ico}
+ tooltip={name}
+ tooltipPlacement="left"
+ />
+ ))}
</div>
</div>
</ScrollView>
<ScrollView ref={scrollEmojisRef} autoHide>
<div onMouseMove={hoverEmoji} onClick={selectEmoji}>
<SearchedEmoji />
- {recentEmojis.length > 0 && <EmojiGroup name="Recently used" groupEmojis={recentEmojis} />}
- {
- availableEmojis.map((pack) => (
- <EmojiGroup
- name={pack.displayName ?? 'Unknown'}
- key={pack.packIndex}
- groupEmojis={pack.getEmojis()}
- className="custom-emoji-group"
- />
- ))
- }
- {
- emojiGroups.map((group) => (
- <EmojiGroup key={group.name} name={group.name} groupEmojis={group.emojis} />
- ))
- }
+ {recentEmojis.length > 0 && (
+ <EmojiGroup name="Recently used" groupEmojis={recentEmojis} />
+ )}
+ {availableEmojis.map((pack) => (
+ <EmojiGroup
+ name={pack.displayName ?? 'Unknown'}
+ key={pack.packIndex}
+ groupEmojis={pack.getEmojis()}
+ className="custom-emoji-group"
+ />
+ ))}
+ {emojiGroups.map((group) => (
+ <EmojiGroup key={group.name} name={group.name} groupEmojis={group.emojis} />
+ ))}
</div>
</ScrollView>
</div>
<div ref={emojiInfo} className="emoji-board__content__info">
- <div>{ parse(twemoji.parse('🙂')) }</div>
+ <div>{parse(twemoji.parse('🙂', { base: TWEMOJI_BASE_URL }))}</div>
<Text>:slight_smile:</Text>
</div>
</div>
import parse from 'html-react-parser';
import twemoji from 'twemoji';
-import { twemojify } from '../../../util/twemojify';
+import { twemojify, TWEMOJI_BASE_URL } from '../../../util/twemojify';
import initMatrix from '../../../client/initMatrix';
import { getEmojiForCompletion } from '../emoji-board/custom-emoji';
function renderSuggestions({ prefix, option, suggestions }, fireCmd) {
function renderCmdSuggestions(cmdPrefix, cmds) {
- const cmdOptString = (typeof option === 'string') ? `/${option}` : '/?';
+ const cmdOptString = typeof option === 'string' ? `/${option}` : '/?';
return cmds.map((cmd) => (
<CmdItem
key={cmd}
// Renders a small Twemoji
function renderTwemoji(emoji) {
- return parse(twemoji.parse(
- emoji.unicode,
- {
+ return parse(
+ twemoji.parse(emoji.unicode, {
attributes: () => ({
unicode: emoji.unicode,
shortcodes: emoji.shortcodes?.toString(),
}),
- },
- ));
+ base: TWEMOJI_BASE_URL,
+ })
+ );
}
// Render a custom emoji
return emos.map((emoji) => (
<CmdItem
key={emoji.shortcode}
- onClick={() => fireCmd({
- prefix: emPrefix,
- result: emoji,
- })}
+ onClick={() =>
+ fireCmd({
+ prefix: emPrefix,
+ result: emoji,
+ })
+ }
>
<Text variant="b1">{renderEmoji(emoji)}</Text>
<Text variant="b2">{`:${emoji.shortcode}:`}</Text>
});
},
'@': () => {
- const members = mx.getRoom(roomId).getJoinedMembers().map((member) => ({
- name: member.name,
- userId: member.userId.slice(1),
- }));
+ const members = mx
+ .getRoom(roomId)
+ .getJoinedMembers()
+ .map((member) => ({
+ name: member.name,
+ userId: member.userId.slice(1),
+ }));
asyncSearch.setup(members, { keys: ['name', 'userId'], limit: 20 });
const endIndex = members.length > 20 ? 20 : members.length;
setCmd({ prefix, suggestions: members.slice(0, endIndex) });
</div>
<div className="cmd-bar__content">
<ScrollView horizontal vertical={false} invisible>
- <div className="cmd-bar__content-suggestions">
- { renderSuggestions(cmd, fireCmd) }
- </div>
+ <div className="cmd-bar__content-suggestions">{renderSuggestions(cmd, fireCmd)}</div>
</ScrollView>
</div>
</div>