feat: use emoji-picker-element, add emoji autocompletions/tooltips (#1804)
* feat: use emoji-picker-element, add emoji autocompletions/tooltips * fix: fix lint bug * test: fix emoji in chrome on linux in travis * test: try bionic in travis * chore: try to fix travis * chore: try to fix travis * fix: filter unsupported emoji * chore: try to fix travis * chore: try to fix travis * chore: try to fix travis * chore: try to fix travis * Revert "chore: try to fix travis" This reverts commit 3cd2d94469b2f1a20c847c2a69e088d7c8d1efdd. * fix: fix emoji autosuggest * test: fix test
This commit is contained in:
parent
85ce93177b
commit
1371175bce
|
@ -14,7 +14,7 @@ tests
|
|||
/static/icons.svg
|
||||
/static/robots.txt
|
||||
/static/inline-script.js.map
|
||||
/static/emoji-mart-all.json
|
||||
/static/emoji-all-en.json
|
||||
/src/inline-script/checksum.js
|
||||
yarn-error.log
|
||||
/now.json
|
||||
|
|
|
@ -8,7 +8,7 @@
|
|||
/static/icons.svg
|
||||
/static/robots.txt
|
||||
/static/inline-script.js.map
|
||||
/static/emoji-mart-all.json
|
||||
/static/emoji-all-en.json
|
||||
/src/inline-script/checksum.js
|
||||
yarn-error.log
|
||||
/now.json
|
||||
|
|
|
@ -7,6 +7,6 @@
|
|||
/static/*.css
|
||||
/static/icons.svg
|
||||
/static/inline-script.js.map
|
||||
/static/emoji-mart-all.json
|
||||
/static/emoji-all-en.json
|
||||
/src/inline-script/checksum.js
|
||||
yarn-error.log
|
||||
|
|
|
@ -1,7 +1,8 @@
|
|||
language: node_js
|
||||
node_js:
|
||||
- "10"
|
||||
dist: xenial
|
||||
dist: bionic
|
||||
group: dev
|
||||
sudo: false
|
||||
services:
|
||||
- redis-server
|
||||
|
@ -20,8 +21,7 @@ addons:
|
|||
- gcc
|
||||
- imagemagick
|
||||
- libffi-dev
|
||||
- libgdbm-dev
|
||||
- libgdbm-dev
|
||||
- libgdbm5
|
||||
- libicu-dev
|
||||
- libidn11-dev
|
||||
- libncurses5-dev
|
||||
|
@ -33,13 +33,13 @@ addons:
|
|||
- libxslt1-dev
|
||||
- libyaml-dev
|
||||
- pkg-config
|
||||
- postgresql-10
|
||||
- postgresql-client-10
|
||||
- postgresql-contrib-10
|
||||
- protobuf-compiler
|
||||
- redis-server
|
||||
- redis-tools
|
||||
- zlib1g-dev
|
||||
- fonts-noto-color-emoji # required for emoji-picker-element + Chrome on Linux
|
||||
before_install:
|
||||
- psql -d template1 -U postgres -c "CREATE USER pinafore WITH PASSWORD 'pinafore' CREATEDB;"
|
||||
- curl -o- -L https://yarnpkg.com/install.sh | bash -s
|
||||
|
|
|
@ -176,10 +176,10 @@ preprocessor.
|
|||
Highly modular, highly functional, lots of single-function files. Tends to help with tree-shaking and
|
||||
code-splitting, as well as avoiding circular dependencies.
|
||||
|
||||
### Preact is loaded dynamically
|
||||
### emoji-picker-element is loaded as a third-party bundle
|
||||
|
||||
This is a Svelte project, but `emoji-mart` is used for the emoji picker, and it's written in React. So we
|
||||
lazy-load the React-compatible Preact library when we load `emoji-mart`.
|
||||
`emoji-picker-element` uses Svelte 3, whereas we use Svelte 2. So it's just imported
|
||||
as a bundled custom element, not as a Svelte component.
|
||||
|
||||
### Some third-party code is bundled
|
||||
|
||||
|
|
|
@ -1,30 +1,22 @@
|
|||
import path from 'path'
|
||||
import fs from 'fs'
|
||||
import { promisify } from 'util'
|
||||
import CleanCSS from 'clean-css'
|
||||
import trimEmojiData from 'emoji-picker-element/trimEmojiData.cjs'
|
||||
|
||||
const writeFile = promisify(fs.writeFile)
|
||||
const readFile = promisify(fs.readFile)
|
||||
const copyFile = promisify(fs.copyFile)
|
||||
|
||||
async function compileThirdPartyCss () {
|
||||
let css = await readFile(path.resolve(__dirname, '../node_modules/emoji-mart/css/emoji-mart.css'), 'utf8')
|
||||
css = '/* compiled from emoji-mart.css */' + new CleanCSS().minify(css).styles
|
||||
await writeFile(path.resolve(__dirname, '../static/emoji-mart.css'), css, 'utf8')
|
||||
}
|
||||
|
||||
async function compileThirdPartyJson () {
|
||||
await copyFile(
|
||||
path.resolve(__dirname, '../node_modules/emoji-mart/data/all.json'),
|
||||
path.resolve(__dirname, '../static/emoji-mart-all.json')
|
||||
)
|
||||
}
|
||||
const writeFile = promisify(fs.writeFile)
|
||||
|
||||
async function main () {
|
||||
await Promise.all([
|
||||
compileThirdPartyCss(),
|
||||
compileThirdPartyJson()
|
||||
])
|
||||
let json = JSON.parse(await readFile(
|
||||
path.resolve(__dirname, '../node_modules/emojibase-data/en/data.json'),
|
||||
'utf8')
|
||||
)
|
||||
json = trimEmojiData(json)
|
||||
await writeFile(
|
||||
path.resolve(__dirname, '../static/emoji-all-en.json'),
|
||||
JSON.stringify(json),
|
||||
'utf8'
|
||||
)
|
||||
}
|
||||
|
||||
main().catch(err => {
|
||||
|
|
11
package.json
11
package.json
|
@ -49,21 +49,22 @@
|
|||
"@babel/runtime": "^7.8.4",
|
||||
"@rollup/plugin-replace": "^2.3.0",
|
||||
"@webcomponents/custom-elements": "^1.4.0",
|
||||
"arrow-key-navigation": "^1.1.0",
|
||||
"@webcomponents/shadydom": "^1.7.3",
|
||||
"array-flat-polyfill": "^1.0.1",
|
||||
"arrow-key-navigation": "^1.2.0",
|
||||
"babel-loader": "^8.0.6",
|
||||
"babel-plugin-transform-react-remove-prop-types": "^0.4.24",
|
||||
"blurhash": "^1.1.3",
|
||||
"cheerio": "^1.0.0-rc.3",
|
||||
"child-process-promise": "^2.2.1",
|
||||
"chokidar": "^3.3.1",
|
||||
"circular-dependency-plugin": "^5.2.0",
|
||||
"clean-css": "^4.2.3",
|
||||
"compression": "^1.7.4",
|
||||
"cross-env": "^7.0.0",
|
||||
"css-dedoupe": "^0.1.1",
|
||||
"css-loader": "^3.4.2",
|
||||
"emoji-mart": "nolanlawson/emoji-mart#8bb6fb6",
|
||||
"emoji-picker-element": "^1.0.0",
|
||||
"emoji-regex": "^9.0.0",
|
||||
"emojibase-data": "^5.0.1",
|
||||
"encoding": "^0.1.12",
|
||||
"escape-html": "^1.0.3",
|
||||
"esm": "^3.2.25",
|
||||
|
@ -88,7 +89,6 @@
|
|||
"page-lifecycle": "^0.1.2",
|
||||
"performance-now": "^2.1.0",
|
||||
"pinch-zoom-element": "^1.1.1",
|
||||
"preact": "^10.3.3",
|
||||
"promise-worker": "^2.0.1",
|
||||
"prop-types": "^15.7.2",
|
||||
"requestidlecallback": "^0.3.0",
|
||||
|
@ -134,6 +134,7 @@
|
|||
"Element",
|
||||
"Event",
|
||||
"FormData",
|
||||
"HTMLElement",
|
||||
"IDBKeyRange",
|
||||
"IDBObjectStore",
|
||||
"Image",
|
||||
|
|
|
@ -26,7 +26,7 @@
|
|||
*/
|
||||
img, svg, video,
|
||||
input[type="checkbox"], input[type="radio"],
|
||||
.inline-emoji, .theme-preview, .emoji-mart-emoji, .emoji-mart-skin {
|
||||
.inline-emoji, .theme-preview {
|
||||
filter: grayscale(100%);
|
||||
}
|
||||
</style>
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
import { store } from '../_store/store'
|
||||
|
||||
const emojiMapper = emoji => `:${emoji.shortcode}:`
|
||||
const emojiMapper = emoji => emoji.unicode ? emoji.unicode : `:${emoji.shortcodes[0]}:`
|
||||
const hashtagMapper = hashtag => `#${hashtag.name}`
|
||||
const accountMapper = account => `@${account.acct}`
|
||||
|
||||
|
@ -61,7 +61,7 @@ export function selectAutosuggestItem (item) {
|
|||
const endIndex = composeSelectionStart
|
||||
if (item.acct) {
|
||||
/* no await */ insertUsername(currentComposeRealm, item, startIndex, endIndex)
|
||||
} else if (item.shortcode) {
|
||||
} else if (item.shortcodes) {
|
||||
/* no await */ insertEmojiAtPosition(currentComposeRealm, item, startIndex, endIndex)
|
||||
} else { // hashtag
|
||||
/* no await */ insertHashtag(currentComposeRealm, item, startIndex, endIndex)
|
||||
|
|
|
@ -1,24 +1,45 @@
|
|||
import { store } from '../_store/store'
|
||||
import { SEARCH_RESULTS_LIMIT } from '../_static/autosuggest'
|
||||
import { scheduleIdleTask } from '../_utils/scheduleIdleTask'
|
||||
import * as emojiDatabase from '../_utils/emojiDatabase'
|
||||
import { SEARCH_RESULTS_LIMIT } from '../_static/autosuggest'
|
||||
import { testEmojiSupported } from '../_utils/testEmojiSupported'
|
||||
import { mark, stop } from '../_utils/marks'
|
||||
|
||||
function searchEmoji (searchText) {
|
||||
searchText = searchText.toLowerCase().substring(1)
|
||||
const { currentCustomEmoji } = store.get()
|
||||
const results = currentCustomEmoji.filter(emoji => emoji.shortcode.toLowerCase().startsWith(searchText))
|
||||
.sort((a, b) => a.shortcode.toLowerCase() < b.shortcode.toLowerCase() ? -1 : 1)
|
||||
.slice(0, SEARCH_RESULTS_LIMIT)
|
||||
async function searchEmoji (searchText) {
|
||||
let emojis = await emojiDatabase.findBySearchQuery(searchText)
|
||||
|
||||
const results = []
|
||||
|
||||
if (searchText.startsWith(':') && searchText.endsWith(':')) {
|
||||
// exact shortcode search
|
||||
const shortcode = searchText.substring(1, searchText.length - 1).toLowerCase()
|
||||
emojis = emojis.filter(_ => _.shortcodes.includes(shortcode))
|
||||
}
|
||||
|
||||
mark('testEmojiSupported')
|
||||
for (const emoji of emojis) {
|
||||
if (results.length === SEARCH_RESULTS_LIMIT) {
|
||||
break
|
||||
}
|
||||
if (emoji.url || testEmojiSupported(emoji.unicode)) { // emoji.url is a custom emoji
|
||||
results.push(emoji)
|
||||
}
|
||||
}
|
||||
stop('testEmojiSupported')
|
||||
return results
|
||||
}
|
||||
|
||||
export function doEmojiSearch (searchText) {
|
||||
let canceled = false
|
||||
|
||||
scheduleIdleTask(() => {
|
||||
scheduleIdleTask(async () => {
|
||||
if (canceled) {
|
||||
return
|
||||
}
|
||||
const results = await searchEmoji(searchText)
|
||||
if (canceled) {
|
||||
return
|
||||
}
|
||||
const results = searchEmoji(searchText)
|
||||
store.setForCurrentAutosuggest({
|
||||
autosuggestType: 'emoji',
|
||||
autosuggestSelected: 0,
|
||||
|
|
|
@ -31,7 +31,7 @@ export async function setupCustomEmojiForInstance (instanceName) {
|
|||
}
|
||||
|
||||
export function insertEmoji (realm, emoji) {
|
||||
const emojiText = emoji.custom ? emoji.colons : emoji.native
|
||||
const emojiText = emoji.unicode || `:${emoji.name}:`
|
||||
const { composeSelectionStart } = store.get()
|
||||
const idx = composeSelectionStart || 0
|
||||
const oldText = store.getComposeData(realm, 'text') || ''
|
||||
|
|
|
@ -3,7 +3,7 @@
|
|||
class="compose-autosuggest-list"
|
||||
role="listbox"
|
||||
>
|
||||
{#each items as item, i (item.shortcode || item.id || item.name)}
|
||||
{#each items as item, i (item.shortcodes ? `emoji-${item.unicode || item.name}` : item.id ? `account-${item.id}` : `hashtag-${item.name}`)}
|
||||
<li id="compose-autosuggest-active-item-{realm}-{i}"
|
||||
class="compose-autosuggest-list-item {i === selected ? 'selected' : ''}"
|
||||
role="option"
|
||||
|
@ -36,14 +36,22 @@
|
|||
<span class="compose-autosuggest-list-display-name compose-autosuggest-list-display-name-single">
|
||||
{item.name}
|
||||
</span>
|
||||
{:else}
|
||||
<img src={$autoplayGifs ? item.url : item.static_url}
|
||||
{:else} <!-- emoji -->
|
||||
{#if item.url}
|
||||
<!-- custom emoji -->
|
||||
<img src={item.url}
|
||||
class="compose-autosuggest-list-item-icon"
|
||||
alt=""
|
||||
aria-hidden="true"
|
||||
/>
|
||||
{:else}
|
||||
<!-- native emoji -->
|
||||
<span class="compose-autosuggest-list-item-icon compose-autosuggest-list-item-native-emoji">
|
||||
{item.unicode}
|
||||
</span>
|
||||
{/if}
|
||||
<span class="compose-autosuggest-list-display-name compose-autosuggest-list-display-name-single">
|
||||
{':' + item.shortcode + ':'}
|
||||
{item.shortcodes.map(_ => `:${_}:`).join(' ')}
|
||||
</span>
|
||||
{/if}
|
||||
</div>
|
||||
|
@ -89,6 +97,14 @@
|
|||
object-fit: contain;
|
||||
fill: var(--deemphasized-text-color);
|
||||
}
|
||||
.compose-autosuggest-list-item-native-emoji {
|
||||
font-family: PinaforeEmoji;
|
||||
font-size: 32px;
|
||||
line-height: 1;
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
}
|
||||
.compose-autosuggest-list-display-name {
|
||||
grid-area: display-name;
|
||||
font-size: 1.1em;
|
||||
|
@ -129,6 +145,9 @@
|
|||
width: 24px;
|
||||
height: 24px;
|
||||
}
|
||||
.compose-autosuggest-list-item-native-emoji {
|
||||
font-size: 18px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
<script>
|
||||
|
|
|
@ -4,234 +4,148 @@
|
|||
{title}
|
||||
shrinkWidthToFit={true}
|
||||
background="var(--main-bg)"
|
||||
on:show="onShow()"
|
||||
className="emoji-dialog"
|
||||
>
|
||||
<div class="emoji-container" {style} ref:container >
|
||||
{#if loaded}
|
||||
<div ref:emojiMartMountPoint></div>
|
||||
{:elseif error}
|
||||
<div>Failed to load emoji picker: {error}</div>
|
||||
{:else}
|
||||
<div class="emoji-container-loading" >
|
||||
<LoadingSpinner />
|
||||
</div>
|
||||
{/if}
|
||||
</div>
|
||||
<div class="emoji-container" ref:container ></div>
|
||||
</ModalDialog>
|
||||
<style>
|
||||
.emoji-container {
|
||||
max-width: calc(100vw - 20px);
|
||||
position: relative;
|
||||
}
|
||||
.emoji-container-loading {
|
||||
position: absolute;
|
||||
left: 0;
|
||||
right: 0;
|
||||
top: 0;
|
||||
bottom: 0;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
}
|
||||
:global(.emoji-mart-category .emoji-mart-emoji-custom span,
|
||||
.emoji-mart-preview-emoji .emoji-mart-emoji-custom span) {
|
||||
/* some custom emoji look repeated because they aren't tall enough */
|
||||
background-repeat: no-repeat;
|
||||
background-position: center center;
|
||||
}
|
||||
:global(.emoji-container .emoji-mart-category .emoji-mart-emoji span, .emoji-container .emoji-mart-anchor) {
|
||||
cursor: pointer;
|
||||
}
|
||||
:global(.emoji-container .emoji-mart-search-icon) {
|
||||
top: 6px; /* this looks a bit off-center with the native emoji */
|
||||
:global(emoji-picker) {
|
||||
--indicator-color: var(--main-theme-color);
|
||||
--outline-color: var(--focus-outline);
|
||||
}
|
||||
|
||||
:global(.emoji-container .emoji-mart-skin) {
|
||||
max-width: 24px;
|
||||
@media (max-width: 479px) {
|
||||
:global(emoji-picker) {
|
||||
--emoji-padding: 0.25rem;
|
||||
--input-padding: 0.125rem;
|
||||
}
|
||||
|
||||
:global(.emoji-container .emoji-mart-skin-swatch.selected) {
|
||||
width: 24px;
|
||||
.emoji-container, :global(emoji-picker) {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
:global(.emoji-container .emoji-mart-skin-swatches.opened .emoji-mart-skin-swatch) {
|
||||
width: 24px;
|
||||
}
|
||||
|
||||
:global(.emoji-container button:hover) {
|
||||
background: none;
|
||||
}
|
||||
|
||||
@media (max-width: 320px) {
|
||||
:global(.emoji-container .emoji-mart-preview) {
|
||||
height: 60px;
|
||||
:global(emoji-picker) {
|
||||
--num-columns: 6;
|
||||
}
|
||||
}
|
||||
|
||||
@media (max-width: 240px) {
|
||||
:global(.modal-dialog .modal-dialog-contents.emoji-dialog) {
|
||||
max-width: 100%;
|
||||
max-height: 100%;
|
||||
}
|
||||
.emoji-container {
|
||||
max-width: 100vw;
|
||||
width: 100%;
|
||||
}
|
||||
:global(.emoji-container .emoji-mart) {
|
||||
width: 100% !important;
|
||||
}
|
||||
:global(.emoji-container .emoji-mart-anchors img, .emoji-container .emoji-mart-anchors svg) {
|
||||
width: 14px;
|
||||
height: 14px;
|
||||
:global(emoji-picker) {
|
||||
--num-columns: 6;
|
||||
--emoji-size: 1.125rem;
|
||||
--emoji-padding: 0.125rem;
|
||||
height: 240px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
<script>
|
||||
/* global applyFocusVisiblePolyfill */
|
||||
import ModalDialog from './ModalDialog.html'
|
||||
import { store } from '../../../_store/store'
|
||||
import { insertEmoji } from '../../../_actions/emoji'
|
||||
import { show } from '../helpers/showDialog'
|
||||
import { close } from '../helpers/closeDialog'
|
||||
import { oncreate as onCreateDialog } from '../helpers/onCreateDialog'
|
||||
import LoadingSpinner from '../../../_components/LoadingSpinner.html'
|
||||
import { createEmojiMartPicker } from '../../../_react/createEmojiMartPicker'
|
||||
import { observe } from 'svelte-extras'
|
||||
import { render, unmountComponentAtNode } from 'react-dom'
|
||||
import { isDarkTheme } from '../../../_utils/isDarkTheme'
|
||||
import Picker from 'emoji-picker-element/picker'
|
||||
import 'focus-visible'
|
||||
import { registerShadowRoot, unregisterShadowRoot } from '../../../_thirdparty/a11y-dialog/a11y-dialog'
|
||||
import { doubleRAF } from '../../../_utils/doubleRAF'
|
||||
import { convertCustomEmojiToEmojiPickerFormat } from '../../../_utils/convertCustomEmojiToEmojiPickerFormat'
|
||||
|
||||
function applyShadowDomPolyfill (picker) {
|
||||
// polyfill for :host, plus other fixes mostly targeted at KaiOS.
|
||||
// We could use shadycss, but it doesn't really work for our use case (has to be injected
|
||||
// into the web component's connectedCallback directly).
|
||||
const style = picker.shadowRoot.querySelector('style')
|
||||
style.remove()
|
||||
if (!document.getElementById('emoji-picker-style')) {
|
||||
let css = style.textContent
|
||||
css = css.replace(/:host\(([.*?])\)/g, 'emoji-picker$1')
|
||||
css = css.replace(/:host/g, 'emoji-picker')
|
||||
css = css.replace(/\*/g, 'emoji-picker *')
|
||||
css = css.replace(/\b(button|input|input\[type=search\])\s*\{/, 'emoji-picker $1{')
|
||||
// fixes for KaiOS style bugs
|
||||
css += `
|
||||
emoji-picker .emoji-menu {
|
||||
grid-template-columns: repeat(8, 1fr);
|
||||
}
|
||||
|
||||
@media screen and (max-width: 320px) {
|
||||
emoji-picker .emoji-menu {
|
||||
grid-template-columns: repeat(6, 1fr);
|
||||
}
|
||||
.emoji-container {
|
||||
width: calc(100% - 20px);
|
||||
}
|
||||
}
|
||||
`
|
||||
const newStyle = document.createElement('style')
|
||||
newStyle.id = 'emoji-picker-style'
|
||||
newStyle.textContent = css
|
||||
document.head.appendChild(newStyle)
|
||||
}
|
||||
}
|
||||
|
||||
export default {
|
||||
async oncreate () {
|
||||
onCreateDialog.call(this)
|
||||
try {
|
||||
const Picker = await createEmojiMartPicker()
|
||||
this.set({ loaded: true })
|
||||
const { emojiMartMountPoint } = this.refs
|
||||
this.observe('emojiMartProps', emojiMartProps => {
|
||||
console.log('rendering react component')
|
||||
const fullProps = Object.assign({
|
||||
onSelect: this.onEmojiSelected.bind(this)
|
||||
}, emojiMartProps)
|
||||
const reactComponent = Picker(fullProps)
|
||||
render(reactComponent, emojiMartMountPoint)
|
||||
const { customEmoji, darkMode } = this.get()
|
||||
const { enableGrayscale, isUserTouching } = this.store.get()
|
||||
const picker = new Picker({
|
||||
dataSource: '/emoji-all-en.json',
|
||||
customEmoji
|
||||
})
|
||||
picker.classList.add(darkMode ? 'dark' : 'light')
|
||||
picker.addEventListener('emoji-click', this.onEmojiSelected.bind(this))
|
||||
// workaround for shortcuts -- see acceptShortcutEvent() in shortcuts.js
|
||||
picker.addEventListener('keydown', event => {
|
||||
if (event.key === 'Backspace' &&
|
||||
picker.shadowRoot.activeElement &&
|
||||
picker.shadowRoot.activeElement.tagName === 'INPUT') {
|
||||
event.stopPropagation() // prevent our hotkeys from activating when pressing backspace in the input
|
||||
}
|
||||
})
|
||||
// break into shadow DOM to fix grayscale in Welness grayscale mode
|
||||
if (enableGrayscale) {
|
||||
const style = document.createElement('style')
|
||||
style.textContent = '.emoji { filter: grayscale(100%); }'
|
||||
picker.shadowRoot.appendChild(style)
|
||||
}
|
||||
applyFocusVisiblePolyfill(picker.shadowRoot)
|
||||
registerShadowRoot(picker.shadowRoot)
|
||||
if (process.env.LEGACY && !HTMLElement.prototype.attachShadow.toString().includes('[native code]')) {
|
||||
applyShadowDomPolyfill(picker)
|
||||
}
|
||||
this.refs.container.appendChild(picker)
|
||||
this.on('destroy', () => unregisterShadowRoot(picker.shadowRoot))
|
||||
if (!isUserTouching) { // auto focus the input on desktop
|
||||
doubleRAF(() => { // triple rAF because a11y tries to focus as well
|
||||
requestAnimationFrame(() => {
|
||||
picker.shadowRoot.querySelector('input').focus()
|
||||
})
|
||||
this.on('destroy', () => {
|
||||
console.log('destroying react component')
|
||||
unmountComponentAtNode(emojiMartMountPoint)
|
||||
})
|
||||
} catch (error) {
|
||||
this.set({ error }) // should never happen, but you never know
|
||||
}
|
||||
},
|
||||
components: {
|
||||
ModalDialog,
|
||||
LoadingSpinner
|
||||
ModalDialog
|
||||
},
|
||||
store: () => store,
|
||||
data: () => ({
|
||||
loading: true,
|
||||
loaded: false,
|
||||
error: undefined
|
||||
}),
|
||||
computed: {
|
||||
// try to estimate size of emoji-mart based on mobile vs desktop
|
||||
style: ({ $isVeryTinyMobileSize, $isSmallMobileSize }) => (`
|
||||
min-width: ${$isVeryTinyMobileSize ? 200 : $isSmallMobileSize ? 250 : 300}px;
|
||||
min-height: ${$isVeryTinyMobileSize ? 0 : $isSmallMobileSize ? 300 : 400}px;
|
||||
${$isVeryTinyMobileSize ? 'overflow-y: auto; overflow-x: hidden;' : ''}
|
||||
`),
|
||||
emojiMartProps: ({ perLine, customEmoji, categoriesSorted, darkMode }) => ({
|
||||
color: 'var(--nav-bg)',
|
||||
emoji: 'sailboat',
|
||||
title: 'Emoji',
|
||||
showPreview: true,
|
||||
perLine,
|
||||
custom: customEmoji,
|
||||
include: categoriesSorted,
|
||||
darkMode
|
||||
}),
|
||||
darkMode: ({ $currentTheme }) => isDarkTheme($currentTheme),
|
||||
perLine: ({ $isSmallMobileSize, $isTinyMobileSize, $isMobileSize, $isVeryTinyMobileSize }) => (
|
||||
$isVeryTinyMobileSize
|
||||
? 5
|
||||
: $isTinyMobileSize
|
||||
? 7
|
||||
: $isSmallMobileSize
|
||||
? 8
|
||||
: $isMobileSize
|
||||
? 9
|
||||
: 10
|
||||
),
|
||||
categoriesSorted: ({ $currentCustomEmoji }) => {
|
||||
// Consistency with Mastodon FE, taken from
|
||||
// - app/javascript/mastodon/features/compose/components/emoji_picker_dropdown.js
|
||||
// - app/javascript/mastodon/features/emoji/emoji.js
|
||||
const customCategories = new Set(['custom'])
|
||||
for (const emoji of $currentCustomEmoji) {
|
||||
if (emoji.category) {
|
||||
customCategories.add(`custom-${emoji.category}`)
|
||||
}
|
||||
}
|
||||
|
||||
return [
|
||||
'recent',
|
||||
...Array.from(customCategories).sort(),
|
||||
'people',
|
||||
'nature',
|
||||
'foods',
|
||||
'activity',
|
||||
'places',
|
||||
'objects',
|
||||
'symbols',
|
||||
'flags'
|
||||
]
|
||||
},
|
||||
customEmoji: ({ $currentCustomEmoji, $autoplayGifs }) => {
|
||||
if (!$currentCustomEmoji) {
|
||||
return []
|
||||
}
|
||||
return $currentCustomEmoji.filter(emoji => emoji.visible_in_picker).map(emoji => ({
|
||||
name: emoji.shortcode,
|
||||
short_names: [emoji.shortcode],
|
||||
text: `:${emoji.shortcode}:`,
|
||||
emoticons: [],
|
||||
keywords: [emoji.shortcode],
|
||||
imageUrl: $autoplayGifs ? emoji.url : emoji.static_url,
|
||||
customCategory: emoji.category
|
||||
}))
|
||||
},
|
||||
// it's jarring on mobile if the emoji picker automatically pops open the keyboard
|
||||
autoFocus: ({ $isUserTouching }) => !$isUserTouching
|
||||
customEmoji: ({ $currentCustomEmoji, $autoplayGifs }) => (
|
||||
convertCustomEmojiToEmojiPickerFormat($currentCustomEmoji, $autoplayGifs)
|
||||
)
|
||||
},
|
||||
methods: {
|
||||
observe,
|
||||
show,
|
||||
close,
|
||||
onEmojiSelected (emoji) {
|
||||
onEmojiSelected (event) {
|
||||
const { realm } = this.get()
|
||||
insertEmoji(realm, emoji)
|
||||
insertEmoji(realm, event.detail.emoji)
|
||||
this.close()
|
||||
},
|
||||
onShow () {
|
||||
this.focusIfNecessary()
|
||||
},
|
||||
focusIfNecessary () {
|
||||
const { autoFocus } = this.get()
|
||||
if (!autoFocus) {
|
||||
return
|
||||
}
|
||||
// The setTimeout is to work around timing issues where
|
||||
// sometimes the search input isn't rendered yet.
|
||||
setTimeout(() => requestAnimationFrame(() => {
|
||||
const container = this.refs.container
|
||||
if (container) {
|
||||
const searchInput = container.querySelector('emoji-mart .emoji-mart-search input')
|
||||
if (searchInput) {
|
||||
// do this manually because emoji-mart's built in autofocus doesn't work consistently
|
||||
searchInput.focus()
|
||||
}
|
||||
}
|
||||
}), 50)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -126,7 +126,7 @@
|
|||
overflow-y: hidden;
|
||||
}
|
||||
|
||||
@media(min-width: 768px) {
|
||||
@media(min-width: 480px) {
|
||||
/* On desktop, some dialogs look bad if they expand to fit all the way. So we shrink
|
||||
them to fit if shrinkWidthToFit is true.*/
|
||||
.modal-dialog-contents.shrink-width-to-fit {
|
||||
|
|
|
@ -3,7 +3,8 @@
|
|||
<AccountProfileMovedBanner {account} />
|
||||
{/if}
|
||||
<div class={className}
|
||||
style="background-image: url({headerImage});">
|
||||
style="background-image: url({headerImage});"
|
||||
ref:accountProfile>
|
||||
<div class="account-profile-grid-wrapper">
|
||||
<div class="account-profile-grid">
|
||||
<AccountProfileHeader {account} {relationship} {verifyCredentials} />
|
||||
|
@ -115,8 +116,13 @@
|
|||
import AccountProfileFilters from './AccountProfileFilters.html'
|
||||
import { store } from '../../_store/store'
|
||||
import { classname } from '../../_utils/classname'
|
||||
import { scheduleIdleTask } from '../../_utils/scheduleIdleTask'
|
||||
import { addEmojiTooltips } from '../../_utils/addEmojiTooltips'
|
||||
|
||||
export default {
|
||||
oncreate () {
|
||||
scheduleIdleTask(() => addEmojiTooltips(this.refs.accountProfile))
|
||||
},
|
||||
store: () => store,
|
||||
computed: {
|
||||
headerImageIsMissing: ({ account }) => account.header.endsWith('missing.png'),
|
||||
|
|
|
@ -5,6 +5,7 @@
|
|||
aria-setsize={length}
|
||||
aria-label={ariaLabel}
|
||||
on:recalculateHeight
|
||||
ref:article
|
||||
>
|
||||
{#if showHeader}
|
||||
<StatusHeader {...params} />
|
||||
|
@ -129,6 +130,7 @@
|
|||
import { absoluteDateFormatter } from '../../_utils/formatters'
|
||||
import { composeNewStatusMentioning } from '../../_actions/mention'
|
||||
import { createStatusOrNotificationUuid } from '../../_utils/createStatusOrNotificationUuid'
|
||||
import { addEmojiTooltips } from '../../_utils/addEmojiTooltips'
|
||||
|
||||
const INPUT_TAGS = new Set(['a', 'button', 'input', 'textarea', 'label'])
|
||||
const isUserInputElement = node => INPUT_TAGS.has(node.localName)
|
||||
|
@ -150,6 +152,7 @@
|
|||
this.set({ preloadHiddenContent: true })
|
||||
})
|
||||
}
|
||||
scheduleIdleTask(() => addEmojiTooltips(this.refs.article))
|
||||
},
|
||||
components: {
|
||||
StatusSidebar,
|
||||
|
|
|
@ -1,20 +0,0 @@
|
|||
import { importEmojiMart } from '../_utils/asyncModules/importEmojiMart.js'
|
||||
import { loadCSS } from '../_utils/loadCSS'
|
||||
|
||||
async function fetchEmojiMartData () {
|
||||
return (await fetch('/emoji-mart-all.json')).json()
|
||||
}
|
||||
|
||||
let Picker // cache so we don't have to recreate every time
|
||||
|
||||
export async function createEmojiMartPicker () {
|
||||
if (!Picker) {
|
||||
loadCSS('/emoji-mart.css')
|
||||
const [data, createEmojiMartPickerFromData] = await Promise.all([
|
||||
fetchEmojiMartData(),
|
||||
importEmojiMart()
|
||||
])
|
||||
Picker = createEmojiMartPickerFromData(data)
|
||||
}
|
||||
return Picker
|
||||
}
|
|
@ -1,17 +0,0 @@
|
|||
// I wrap the emoji-mart React code itself here, so that we don't need to pass in a huge "data"
|
||||
// object via a JSON-stringified custom element attribute. Also, AFAICT there is no way when
|
||||
// using `remount` to pass in functions as attributes, since everything is stringified. So
|
||||
// I just fire a global event here when an emoji is clicked.
|
||||
|
||||
import NimblePicker from 'emoji-mart/dist-modern/components/picker/nimble-picker'
|
||||
import { createElement } from 'react'
|
||||
|
||||
export default function createEmojiMartPickerFromData (data) {
|
||||
return props => (
|
||||
createElement(NimblePicker, Object.assign({
|
||||
set: 'twitter', // same as Mastodon frontend
|
||||
data, // same as Mastodon frontend
|
||||
native: true
|
||||
}, props))
|
||||
)
|
||||
}
|
|
@ -0,0 +1,3 @@
|
|||
// same as the one used for PinaforeEmoji
|
||||
export const FONT_FAMILY = '"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol",' +
|
||||
'"Twemoji Mozilla","Noto Color Emoji","EmojiOne Color","Android Emoji",sans-serif'
|
|
@ -4,7 +4,8 @@ import { mark, stop } from '../../_utils/marks'
|
|||
const MIN_PREFIX_LENGTH = 2
|
||||
// Technically mastodon accounts allow dots, but it would be weird to do an autosuggest search if it ends with a dot.
|
||||
// Also this is rare. https://github.com/tootsuite/mastodon/pull/6844
|
||||
const VALID_CHARS = '\\w'
|
||||
// However for emoji search we allow some extra things (e.g. :+1:, :white_heart:)
|
||||
const VALID_CHARS = '[\\w\\+_\\-:]'
|
||||
const PREFIXES = '(?:@|:|#)'
|
||||
const REGEX = new RegExp(`(?:\\s|^)(${PREFIXES}${VALID_CHARS}{${MIN_PREFIX_LENGTH},})$`)
|
||||
|
||||
|
|
|
@ -0,0 +1,22 @@
|
|||
import { store } from '../store'
|
||||
import * as emojiDatabase from '../../_utils/emojiDatabase'
|
||||
import { convertCustomEmojiToEmojiPickerFormat } from '../../_utils/convertCustomEmojiToEmojiPickerFormat'
|
||||
|
||||
export function customEmojiObservers () {
|
||||
if (!process.browser) {
|
||||
return
|
||||
}
|
||||
|
||||
function setEmoji (currentEmoji, autoplayGifs) {
|
||||
const customEmojiInEmojiPickerFormat = convertCustomEmojiToEmojiPickerFormat(currentEmoji, autoplayGifs)
|
||||
emojiDatabase.setCustomEmoji(customEmojiInEmojiPickerFormat)
|
||||
}
|
||||
|
||||
store.observe('currentCustomEmoji', currentCustomEmoji => {
|
||||
setEmoji(currentCustomEmoji, store.get().autoplayGifs)
|
||||
}, { init: false })
|
||||
|
||||
store.observe('autoplayGifs', autoplayGifs => {
|
||||
setEmoji(store.get().currentCustomEmoji, autoplayGifs)
|
||||
}, { init: false })
|
||||
}
|
|
@ -4,6 +4,7 @@ import { notificationObservers } from './notificationObservers'
|
|||
import { autosuggestObservers } from './autosuggestObservers'
|
||||
import { notificationPermissionObservers } from './notificationPermissionObservers'
|
||||
import { customScrollbarObservers } from './customScrollbarObservers'
|
||||
import { customEmojiObservers } from './customEmojiObservers'
|
||||
import { cleanup } from './cleanup'
|
||||
|
||||
// These observers can be lazy-loaded when the user is actually logged in.
|
||||
|
@ -15,5 +16,6 @@ export function loggedInObservers () {
|
|||
autosuggestObservers()
|
||||
notificationPermissionObservers()
|
||||
customScrollbarObservers()
|
||||
customEmojiObservers()
|
||||
cleanup()
|
||||
}
|
||||
|
|
|
@ -2,10 +2,12 @@
|
|||
// around a Chrome bug with sticky positioning (https://github.com/nolanlawson/pinafore/issues/671)
|
||||
// Original: https://unpkg.com/a11y-dialog@4.0.1/a11y-dialog.js
|
||||
|
||||
var FOCUSABLE_ELEMENTS = ['a[href]', 'area[href]', 'input:not([disabled])', 'select:not([disabled])', 'textarea:not([disabled])', 'button:not([disabled])', 'iframe', 'object', 'embed', '[contenteditable]', '[tabindex]:not([tabindex^="-"])']
|
||||
var TAB_KEY = 9
|
||||
var ESCAPE_KEY = 27
|
||||
var focusedBeforeDialog
|
||||
const FOCUSABLE_ELEMENTS = ['a[href]', 'area[href]', 'input:not([disabled])', 'select:not([disabled])', 'textarea:not([disabled])', 'button:not([disabled])', 'iframe', 'object', 'embed', '[contenteditable]', '[tabindex]:not([tabindex^="-"])']
|
||||
const FOCUSABLE_ELEMENTS_QUERY = FOCUSABLE_ELEMENTS.join(',')
|
||||
const TAB_KEY = 9
|
||||
const ESCAPE_KEY = 27
|
||||
const shadowRoots = []
|
||||
let focusedBeforeDialog
|
||||
|
||||
/**
|
||||
* Define the constructor to instantiate a dialog
|
||||
|
@ -335,6 +337,17 @@ function setFocusToFirstItem (node) {
|
|||
}
|
||||
}
|
||||
|
||||
function isAncestor (node, ancestor) {
|
||||
let parent = node
|
||||
while (parent) {
|
||||
parent = parent.parentElement
|
||||
if (parent === ancestor) {
|
||||
return true
|
||||
}
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
/**
|
||||
* Get the focusable children of the given element
|
||||
*
|
||||
|
@ -342,8 +355,18 @@ function setFocusToFirstItem (node) {
|
|||
* @return {Array<Element>}
|
||||
*/
|
||||
function getFocusableChildren (node) {
|
||||
return $$(FOCUSABLE_ELEMENTS.join(','), node).filter(function (child) {
|
||||
return !!(child.offsetWidth || child.offsetHeight || child.getClientRects().length)
|
||||
const candidateFocusableChildren = $$(FOCUSABLE_ELEMENTS_QUERY, node)
|
||||
for (const shadowRoot of shadowRoots) {
|
||||
if (isAncestor(shadowRoot.getRootNode().host, node)) {
|
||||
// TODO: technically we should figure out the host's position in the DOM
|
||||
// and insert the children there, but this works for the emoji picker dialog well
|
||||
// enough, and that's our only shadow root, so it's fine for now.
|
||||
candidateFocusableChildren.push(...shadowRoot.querySelectorAll(FOCUSABLE_ELEMENTS_QUERY))
|
||||
}
|
||||
}
|
||||
return candidateFocusableChildren.filter(child => {
|
||||
return !!(child.offsetWidth || child.offsetHeight || child.getClientRects().length) &&
|
||||
child.tabIndex !== -1
|
||||
})
|
||||
}
|
||||
|
||||
|
@ -355,7 +378,14 @@ function getFocusableChildren (node) {
|
|||
*/
|
||||
function trapTabKey (node, event) {
|
||||
var focusableChildren = getFocusableChildren(node)
|
||||
var focusedItemIndex = focusableChildren.indexOf(document.activeElement)
|
||||
let activeElement = document.activeElement
|
||||
for (const shadowRoot of shadowRoots) {
|
||||
if (shadowRoot.getRootNode().host === activeElement) {
|
||||
activeElement = shadowRoot.activeElement
|
||||
break
|
||||
}
|
||||
}
|
||||
var focusedItemIndex = focusableChildren.indexOf(activeElement)
|
||||
|
||||
// If the SHIFT key is being pressed while tabbing (moving backwards) and
|
||||
// the currently focused item is the first one, move the focus to the last
|
||||
|
@ -389,4 +419,17 @@ function getSiblings (node) {
|
|||
return siblings
|
||||
}
|
||||
|
||||
export { A11yDialog }
|
||||
function registerShadowRoot (shadowRoot) {
|
||||
if (!shadowRoots.includes(shadowRoot)) {
|
||||
shadowRoots.push(shadowRoot)
|
||||
}
|
||||
}
|
||||
|
||||
function unregisterShadowRoot (shadowRoot) {
|
||||
const index = shadowRoots.indexOf(shadowRoot)
|
||||
if (index !== -1) {
|
||||
shadowRoots.splice(index, 1)
|
||||
}
|
||||
}
|
||||
|
||||
export { A11yDialog, registerShadowRoot, unregisterShadowRoot }
|
||||
|
|
|
@ -0,0 +1,19 @@
|
|||
import * as emojiDatabase from './emojiDatabase'
|
||||
|
||||
// Add a nice little tooltip to native emoji showing the shortcodes you can type to search for them
|
||||
// TODO: titles are not accessible to keyboard users or touch users, and also they don't show up
|
||||
// if they're part of a link... should we have another system?
|
||||
export async function addEmojiTooltips (domNode) {
|
||||
if (!domNode) {
|
||||
return
|
||||
}
|
||||
const emojis = domNode.querySelectorAll('.inline-emoji')
|
||||
if (emojis.length) {
|
||||
await Promise.all(Array.from(emojis).map(async emoji => {
|
||||
const emojiData = await emojiDatabase.findByUnicodeOrName(emoji.textContent)
|
||||
if (emojiData && emojiData.shortcodes) {
|
||||
emoji.title = emojiData.shortcodes.map(_ => `:${_}:`).join(', ')
|
||||
}
|
||||
}))
|
||||
}
|
||||
}
|
|
@ -21,3 +21,11 @@ export const importIntl = () => import(
|
|||
export const importFocusVisible = () => import(
|
||||
/* webpackChunkName: '$polyfill$-focus-visible' */ 'focus-visible'
|
||||
)
|
||||
|
||||
export const importShadowDomPolyfill = () => import(
|
||||
/* webpackChunkName: '$polyfill$-shadydom' */ '@webcomponents/shadydom'
|
||||
)
|
||||
|
||||
export const importArrayFlat = () => import(
|
||||
/* webpackChunkName: '$polyfill$-array-flat' */ 'array-flat-polyfill'
|
||||
)
|
||||
|
|
|
@ -0,0 +1,11 @@
|
|||
export function convertCustomEmojiToEmojiPickerFormat (customEmoji, autoplayGifs) {
|
||||
if (!customEmoji) {
|
||||
return []
|
||||
}
|
||||
return customEmoji.filter(emoji => emoji.visible_in_picker).map(emoji => ({
|
||||
name: emoji.shortcode,
|
||||
shortcodes: [emoji.shortcode],
|
||||
url: autoplayGifs ? emoji.url : emoji.static_url,
|
||||
category: emoji.category
|
||||
}))
|
||||
}
|
|
@ -6,7 +6,7 @@ export function createAutosuggestAccessibleLabel (
|
|||
const selected = searchResults[selectedIndex]
|
||||
let label
|
||||
if (autosuggestType === 'emoji') {
|
||||
label = `${selected.shortcode}`
|
||||
label = `${selected.unicode || selected.name}`
|
||||
} else if (autosuggestType === 'hashtag') {
|
||||
label = `#${selected.name}`
|
||||
} else { // account
|
||||
|
|
|
@ -0,0 +1,61 @@
|
|||
import Database from 'emoji-picker-element/database'
|
||||
import { lifecycle } from './lifecycle'
|
||||
|
||||
let database
|
||||
|
||||
function applySkinToneToEmoji (emoji, skinTone) {
|
||||
if (!emoji || emoji.url) { // nonexistent or custom emoji
|
||||
return emoji
|
||||
}
|
||||
const res = {
|
||||
unicode: emoji.unicode,
|
||||
shortcodes: emoji.shortcodes
|
||||
}
|
||||
if (skinTone > 0 && emoji.skins) { // non-default skin tone
|
||||
const tone = emoji.skins.find(_ => _.tone === skinTone)
|
||||
if (tone) {
|
||||
res.unicode = tone.unicode
|
||||
}
|
||||
}
|
||||
return res
|
||||
}
|
||||
|
||||
export function init () {
|
||||
if (!database) {
|
||||
database = new Database({
|
||||
dataSource: '/emoji-all-en.json'
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
export function setCustomEmoji (customEmoji) {
|
||||
init()
|
||||
database.customEmoji = customEmoji
|
||||
}
|
||||
|
||||
export async function findByUnicodeOrName (unicodeOrName) {
|
||||
init()
|
||||
const [emoji, skinTone] = await Promise.all([
|
||||
database.getEmojiByUnicodeOrName(unicodeOrName),
|
||||
database.getPreferredSkinTone()
|
||||
])
|
||||
return applySkinToneToEmoji(emoji, skinTone)
|
||||
}
|
||||
|
||||
export async function findBySearchQuery (query) {
|
||||
init()
|
||||
const [emojis, skinTone] = await Promise.all([
|
||||
database.getEmojiBySearchQuery(query),
|
||||
database.getPreferredSkinTone()
|
||||
])
|
||||
return emojis.map(emoji => applySkinToneToEmoji(emoji, skinTone))
|
||||
}
|
||||
|
||||
if (process.browser) {
|
||||
lifecycle.addEventListener('statechange', event => {
|
||||
if (event.newState === 'frozen' && database) { // page is frozen, close IDB connections
|
||||
console.log('closed emoji DB')
|
||||
database.close()
|
||||
}
|
||||
})
|
||||
}
|
|
@ -3,6 +3,7 @@ import { replaceEmoji } from './replaceEmoji'
|
|||
|
||||
export function emojifyText (text, emojis, autoplayGifs) {
|
||||
// replace native emoji with wrapped spans so we can give them the proper font-family
|
||||
// as well as show tooltips
|
||||
text = replaceEmoji(text, substring => `<span class="inline-emoji">${substring}</span>`)
|
||||
|
||||
// replace custom emoji
|
||||
|
|
|
@ -1,17 +1,22 @@
|
|||
import {
|
||||
importArrayFlat,
|
||||
importCustomElementsPolyfill,
|
||||
importIndexedDBGetAllShim,
|
||||
importIntersectionObserver,
|
||||
importIntl,
|
||||
importRequestIdleCallback
|
||||
importRequestIdleCallback,
|
||||
importShadowDomPolyfill
|
||||
} from './asyncPolyfills'
|
||||
|
||||
export function loadPolyfills () {
|
||||
return Promise.all([
|
||||
typeof IntersectionObserver === 'undefined' && importIntersectionObserver(),
|
||||
typeof requestIdleCallback === 'undefined' && importRequestIdleCallback(),
|
||||
!IDBObjectStore.prototype.getAll && importIndexedDBGetAllShim(),
|
||||
typeof customElements === 'undefined' && importCustomElementsPolyfill(),
|
||||
process.env.LEGACY && typeof Intl === 'undefined' && importIntl()
|
||||
// these legacy polyfills should be kept in sync with webpack/shared.config.js
|
||||
process.env.LEGACY && !Array.prototype.flat && importArrayFlat(),
|
||||
process.env.LEGACY && !IDBObjectStore.prototype.getAll && importIndexedDBGetAllShim(),
|
||||
process.env.LEGACY && typeof IntersectionObserver === 'undefined' && importIntersectionObserver(),
|
||||
process.env.LEGACY && typeof Intl === 'undefined' && importIntl(),
|
||||
process.env.LEGACY && typeof customElements === 'undefined' && importCustomElementsPolyfill(),
|
||||
process.env.LEGACY && !HTMLElement.prototype.attachShadow && importShadowDomPolyfill()
|
||||
])
|
||||
}
|
||||
|
|
|
@ -0,0 +1,39 @@
|
|||
// Copied from
|
||||
// https://github.com/nolanlawson/emoji-picker-element/blob/04f490a/src/picker/utils/testColorEmojiSupported.js
|
||||
|
||||
import { FONT_FAMILY } from '../_static/fonts'
|
||||
|
||||
const getTextFeature = (text, color) => {
|
||||
try {
|
||||
const canvas = document.createElement('canvas')
|
||||
canvas.width = canvas.height = 1
|
||||
|
||||
const ctx = canvas.getContext('2d')
|
||||
ctx.textBaseline = 'top'
|
||||
ctx.font = `100px ${FONT_FAMILY}`
|
||||
ctx.fillStyle = color
|
||||
ctx.scale(0.01, 0.01)
|
||||
ctx.fillText(text, 0, 0)
|
||||
|
||||
return ctx.getImageData(0, 0, 1, 1).data
|
||||
} catch (e) { /* ignore, return undefined */ }
|
||||
}
|
||||
|
||||
const compareFeatures = (feature1, feature2) => {
|
||||
const feature1Str = [...feature1].join(',')
|
||||
const feature2Str = [...feature2].join(',')
|
||||
return feature1Str === feature2Str && feature1Str !== '0,0,0,0'
|
||||
}
|
||||
|
||||
export function testColorEmojiSupported (text) {
|
||||
// Render white and black and then compare them to each other and ensure they're the same
|
||||
// color, and neither one is black. This shows that the emoji was rendered in color.
|
||||
const feature1 = getTextFeature(text, '#000')
|
||||
const feature2 = getTextFeature(text, '#fff')
|
||||
|
||||
const supported = feature1 && feature2 && compareFeatures(feature1, feature2)
|
||||
if (!supported) {
|
||||
console.log('Filtered unsupported emoji via color test', text)
|
||||
}
|
||||
return supported
|
||||
}
|
|
@ -0,0 +1,43 @@
|
|||
// Return true if the unicode is rendered as a double character, e.g.
|
||||
// "black cat" or "polar bar" or "person with red hair" or other emoji
|
||||
// that look like double or triple emoji if the unicode is not rendered properly
|
||||
|
||||
const BASELINE_EMOJI = '😀'
|
||||
|
||||
let baselineWidth
|
||||
|
||||
export function testEmojiRenderedAtCorrectSize (unicode) {
|
||||
if (!unicode.includes('\u200d')) { // ZWJ
|
||||
return true // benefit of the doubt
|
||||
}
|
||||
|
||||
let emojiTestDiv = document.getElementById('emoji-test')
|
||||
if (!emojiTestDiv) {
|
||||
emojiTestDiv = document.createElement('div')
|
||||
emojiTestDiv.id = 'emoji-test'
|
||||
emojiTestDiv.ariaHidden = true
|
||||
Object.assign(emojiTestDiv.style, {
|
||||
position: 'absolute',
|
||||
opacity: '0',
|
||||
'pointer-events': 'none',
|
||||
'font-family': 'PinaforeEmoji',
|
||||
'font-size': '14px',
|
||||
contain: 'content'
|
||||
})
|
||||
document.body.appendChild(emojiTestDiv)
|
||||
}
|
||||
emojiTestDiv.textContent = unicode
|
||||
const { width } = emojiTestDiv.getBoundingClientRect()
|
||||
if (typeof baselineWidth === 'undefined') {
|
||||
emojiTestDiv.textContent = BASELINE_EMOJI
|
||||
baselineWidth = emojiTestDiv.getBoundingClientRect().width
|
||||
}
|
||||
|
||||
// WebKit has some imprecision here, so round it
|
||||
const emojiSupported = width.toFixed(2) === baselineWidth.toFixed(2)
|
||||
if (!emojiSupported) {
|
||||
console.log('Filtered unsupported emoji via size test', unicode, 'width', width, 'baselineWidth', baselineWidth)
|
||||
}
|
||||
|
||||
return emojiSupported
|
||||
}
|
|
@ -0,0 +1,17 @@
|
|||
import { testColorEmojiSupported } from './testColorEmojiSupported'
|
||||
import { testEmojiRenderedAtCorrectSize } from './testEmojiRenderedAtCorrectSize'
|
||||
import { QuickLRU } from '../_thirdparty/quick-lru/quick-lru'
|
||||
|
||||
// avoid recomputing emoji support over and over again
|
||||
const emojiSupportCache = new QuickLRU({
|
||||
maxSize: 500
|
||||
})
|
||||
|
||||
export function testEmojiSupported (unicode) {
|
||||
let supported = emojiSupportCache.get(unicode)
|
||||
if (typeof supported !== 'boolean') {
|
||||
supported = !!(testColorEmojiSupported(unicode) && testEmojiRenderedAtCorrectSize(unicode))
|
||||
emojiSupportCache.set(unicode, supported)
|
||||
}
|
||||
return supported
|
||||
}
|
|
@ -1,3 +1,4 @@
|
|||
import './routes/_thirdparty/regenerator-runtime/runtime.js'
|
||||
import {
|
||||
assets as __assets__,
|
||||
shell as __shell__,
|
||||
|
@ -27,6 +28,7 @@ const assets = __assets__
|
|||
.filter(filename => filename !== '/robots.txt')
|
||||
.filter(filename => !filename.includes('traineddata.gz')) // cache on-demand
|
||||
.filter(filename => !filename.endsWith('.webapp')) // KaiOS manifest
|
||||
.filter(filename => !filename.includes('emoji-all-en.json')) // useless to cache; it already goes in IndexedDB
|
||||
|
||||
// `shell` is an array of all the files generated by webpack
|
||||
// also contains '/index.html' for some reason
|
||||
|
|
|
@ -1,8 +1,16 @@
|
|||
import {
|
||||
composeButton, composeInput, composeLengthIndicator, emojiButton, emojiSearchInput, getComposeSelectionStart,
|
||||
getNthStatusContent, getUrl,
|
||||
homeNavButton, modalDialog,
|
||||
notificationsNavButton, sleep,
|
||||
composeButton,
|
||||
composeInput,
|
||||
composeLengthIndicator,
|
||||
emojiButton,
|
||||
emojiSearchInput,
|
||||
firstEmojiInPicker,
|
||||
getComposeSelectionStart,
|
||||
getNthStatusContent,
|
||||
getUrl,
|
||||
homeNavButton,
|
||||
notificationsNavButton,
|
||||
sleep,
|
||||
times
|
||||
} from '../utils'
|
||||
import { loginAsFoobar } from '../roles'
|
||||
|
@ -63,7 +71,9 @@ test('shows compose limits for custom emoji', async t => {
|
|||
.typeText(composeInput, 'hello world ')
|
||||
.click(emojiButton)
|
||||
.typeText(emojiSearchInput, 'blobnom')
|
||||
.pressKey('enter')
|
||||
await sleep(1000)
|
||||
await t
|
||||
.click(firstEmojiInPicker)
|
||||
.expect(composeInput.value).eql('hello world :blobnom: ')
|
||||
.expect(composeLengthIndicator.innerText).eql('478')
|
||||
})
|
||||
|
@ -76,18 +86,24 @@ test('inserts custom emoji correctly', async t => {
|
|||
.expect(getComposeSelectionStart()).eql(6)
|
||||
.click(emojiButton)
|
||||
.typeText(emojiSearchInput, 'blobpats')
|
||||
.pressKey('enter')
|
||||
await sleep(1000)
|
||||
await t
|
||||
.click(firstEmojiInPicker)
|
||||
.expect(composeInput.value).eql('hello :blobpats: world')
|
||||
.selectText(composeInput, 0, 0)
|
||||
.expect(getComposeSelectionStart()).eql(0)
|
||||
.click(emojiButton)
|
||||
.typeText(emojiSearchInput, 'blobnom')
|
||||
.pressKey('enter')
|
||||
await sleep(1000)
|
||||
await t
|
||||
.click(firstEmojiInPicker)
|
||||
.expect(composeInput.value).eql(':blobnom: hello :blobpats: world')
|
||||
.typeText(composeInput, ' foobar ')
|
||||
.click(emojiButton)
|
||||
.typeText(emojiSearchInput, 'blobpeek')
|
||||
.pressKey('enter')
|
||||
await sleep(1000)
|
||||
await t
|
||||
.click(firstEmojiInPicker)
|
||||
.expect(composeInput.value).eql(':blobnom: hello :blobpats: world foobar :blobpeek: ')
|
||||
})
|
||||
|
||||
|
@ -96,13 +112,13 @@ test('inserts emoji without typing anything', async t => {
|
|||
await sleep(1000)
|
||||
await t
|
||||
.click(emojiButton)
|
||||
.click(modalDialog.find('button[aria-label="blobpats"]'))
|
||||
.expect(composeInput.value).eql(':blobpats: ')
|
||||
.click(firstEmojiInPicker)
|
||||
.expect(composeInput.value).eql(':blobnom: ')
|
||||
await sleep(1000)
|
||||
await t
|
||||
.click(emojiButton)
|
||||
.click(modalDialog.find('button[aria-label="blobpeek"]'))
|
||||
.expect(composeInput.value).eql(':blobpeek: :blobpats: ')
|
||||
.click(firstEmojiInPicker)
|
||||
.expect(composeInput.value).eql(':blobnom: :blobnom: ')
|
||||
})
|
||||
|
||||
test('inserts native emoji without typing anything', async t => {
|
||||
|
@ -115,7 +131,7 @@ test('inserts native emoji without typing anything', async t => {
|
|||
.typeText(emojiSearchInput, 'pineapple', { paste: true })
|
||||
await sleep(1000)
|
||||
await t
|
||||
.pressKey('enter')
|
||||
.click(firstEmojiInPicker)
|
||||
.expect(composeInput.value).eql('\ud83c\udf4d ')
|
||||
.click(emojiButton)
|
||||
await sleep(1000)
|
||||
|
@ -124,7 +140,7 @@ test('inserts native emoji without typing anything', async t => {
|
|||
.typeText(emojiSearchInput, 'elephant', { paste: true })
|
||||
await sleep(1000)
|
||||
await t
|
||||
.pressKey('enter')
|
||||
.click(firstEmojiInPicker)
|
||||
.expect(composeInput.value).eql('\ud83d\udc18 \ud83c\udf4d ')
|
||||
})
|
||||
|
||||
|
|
|
@ -9,7 +9,7 @@ import {
|
|||
getNthStatusSelector,
|
||||
composeModalEmojiButton,
|
||||
composeModalInput,
|
||||
composeModalComposeButton, emojiSearchInput
|
||||
composeModalComposeButton, emojiSearchInput, firstEmojiInPicker
|
||||
} from '../utils'
|
||||
import { loginAsFoobar } from '../roles'
|
||||
import { Selector as $ } from 'testcafe'
|
||||
|
@ -35,15 +35,23 @@ test('can compose using a dialog', async t => {
|
|||
.expect(getNthStatus(1).innerText).contains('hello from the modal', { timeout: 20000 })
|
||||
})
|
||||
|
||||
test('can use emoji dialog within compose dialog', async t => {
|
||||
// Skipped because TestCafé seems to believe the elements are not visible when they are.
|
||||
// Tested manually and it's fine; probably a TestCafé bug.
|
||||
test.skip('can use emoji dialog within compose dialog', async t => {
|
||||
await loginAsFoobar(t)
|
||||
await scrollToStatus(t, 16)
|
||||
await t.expect(composeButton.getAttribute('aria-label')).eql('Compose')
|
||||
await sleep(2000)
|
||||
await t.click(composeButton)
|
||||
await sleep(1000)
|
||||
await t
|
||||
.click(composeModalEmojiButton)
|
||||
await sleep(1000)
|
||||
await t
|
||||
.typeText(emojiSearchInput, 'blobpats')
|
||||
.pressKey('enter')
|
||||
await sleep(1000)
|
||||
await t
|
||||
.click(firstEmojiInPicker)
|
||||
.expect(composeModalInput.value).eql(':blobpats: ')
|
||||
.click(composeModalComposeButton)
|
||||
.expect(modalDialog.exists).notOk()
|
||||
|
|
|
@ -56,7 +56,6 @@ export const disableUnreadNotifications = $('#choice-disable-unread-notification
|
|||
export const leftRightChangesFocus = $('#choice-left-right-focus')
|
||||
export const disableHotkeys = $('#choice-disable-hotkeys')
|
||||
export const dialogOptionsOption = $('.modal-dialog button')
|
||||
export const emojiSearchInput = $('.emoji-mart-search input')
|
||||
export const confirmationDialogOKButton = $('.confirmation-dialog-form-flex button:nth-child(1)')
|
||||
export const confirmationDialogCancelButton = $('.confirmation-dialog-form-flex button:nth-child(2)')
|
||||
|
||||
|
@ -119,6 +118,16 @@ export const sleep = timeout => new Promise(resolve => setTimeout(resolve, timeo
|
|||
|
||||
export const getUrl = exec(() => window.location.href)
|
||||
|
||||
/* global emojiPickerSelector */
|
||||
const emojiPicker = $('emoji-picker')
|
||||
export const emojiSearchInput = $(() => {
|
||||
return emojiPickerSelector().shadowRoot.querySelector('input')
|
||||
}, { dependencies: { emojiPickerSelector: emojiPicker } })
|
||||
|
||||
export const firstEmojiInPicker = $(() => {
|
||||
return emojiPickerSelector().shadowRoot.querySelector('.emoji-menu button')
|
||||
}, { dependencies: { emojiPickerSelector: emojiPicker } })
|
||||
|
||||
export const getNumSyntheticListeners = exec(() => {
|
||||
return Object.keys(window.__eventBus.$e).map(key => window.__eventBus.listenerCount(key))
|
||||
.concat(window.__resizeListeners.size)
|
||||
|
|
|
@ -47,26 +47,6 @@ module.exports = {
|
|||
}
|
||||
}
|
||||
},
|
||||
{
|
||||
test: /\.m?js$/,
|
||||
include: /node_modules\/emoji-mart/,
|
||||
use: {
|
||||
loader: 'babel-loader',
|
||||
options: {
|
||||
plugins: [
|
||||
[
|
||||
'transform-react-remove-prop-types',
|
||||
{
|
||||
removeImport: true,
|
||||
additionalLibraries: [
|
||||
'../../utils/shared-props'
|
||||
]
|
||||
}
|
||||
]
|
||||
]
|
||||
}
|
||||
}
|
||||
},
|
||||
process.env.LEGACY && legacyBabel(),
|
||||
{
|
||||
test: /\.html$/,
|
||||
|
|
|
@ -36,7 +36,13 @@ const resolve = {
|
|||
'../../_components/SvgIcon.html': '../../_components/SvgIconLegacy.html',
|
||||
'../../../_components/SvgIcon.html': '../../../_components/SvgIconLegacy.html'
|
||||
} : {
|
||||
// these polyfills are only needed in legacy mode
|
||||
'array-flat-polyfill': 'lodash/noop',
|
||||
'indexeddb-getall-shim': 'lodash/noop',
|
||||
intl: 'lodash/noop',
|
||||
'intersection-observer': 'lodash/noop',
|
||||
'@webcomponents/custom-elements': 'lodash/noop',
|
||||
'@webcomponents/shadydom': 'lodash/noop',
|
||||
'./routes/_thirdparty/regenerator-runtime/runtime.js': 'lodash/noop',
|
||||
'../_thirdparty/regenerator-runtime/runtime.js': 'lodash/noop'
|
||||
})
|
||||
|
|
46
yarn.lock
46
yarn.lock
|
@ -1016,6 +1016,11 @@
|
|||
resolved "https://registry.yarnpkg.com/@webcomponents/custom-elements/-/custom-elements-1.4.1.tgz#9803aaa2286a13a4ba200a7a2ea767871598eb60"
|
||||
integrity sha512-vNCS1+3sxJOpoIsBjUQiXjGLngakEAGOD5Ale+6ikg6OZG5qI5O39frm3raPhud/IwnF4vec5ags05YBsgzcuA==
|
||||
|
||||
"@webcomponents/shadydom@^1.7.3":
|
||||
version "1.7.3"
|
||||
resolved "https://registry.yarnpkg.com/@webcomponents/shadydom/-/shadydom-1.7.3.tgz#cbb6b17be619d4cae2dd97f5d9bf1448523fabb9"
|
||||
integrity sha512-wAQCuHX6x3pJHRNGbgGhKgY1kZoXwXwWNLe0FdyUpCeyHqmGI8Q8wthyHroUQQIsmoKzM09iTlqO/1JmPWZERg==
|
||||
|
||||
"@xtuc/ieee754@^1.2.0":
|
||||
version "1.2.0"
|
||||
resolved "https://registry.yarnpkg.com/@xtuc/ieee754/-/ieee754-1.2.0.tgz#eef014a3145ae477a1cbc00cd1e552336dceb790"
|
||||
|
@ -1211,6 +1216,11 @@ array-find@^1.0.0:
|
|||
resolved "https://registry.yarnpkg.com/array-find/-/array-find-1.0.0.tgz#6c8e286d11ed768327f8e62ecee87353ca3e78b8"
|
||||
integrity sha1-bI4obRHtdoMn+OYuzuhzU8o+eLg=
|
||||
|
||||
array-flat-polyfill@^1.0.1:
|
||||
version "1.0.1"
|
||||
resolved "https://registry.yarnpkg.com/array-flat-polyfill/-/array-flat-polyfill-1.0.1.tgz#1e3a4255be619dfbffbfd1d635c1cf357cd034e7"
|
||||
integrity sha512-hfJmKupmQN0lwi0xG6FQ5U8Rd97RnIERplymOv/qpq8AoNKPPAnxJadjFA23FNWm88wykh9HmpLJUUwUtNU/iw==
|
||||
|
||||
array-flatten@1.1.1:
|
||||
version "1.1.1"
|
||||
resolved "https://registry.yarnpkg.com/array-flatten/-/array-flatten-1.1.1.tgz#9a5f699051b1e7073328f2a008968b64ea2955d2"
|
||||
|
@ -1247,10 +1257,10 @@ array-unique@^0.3.2:
|
|||
resolved "https://registry.yarnpkg.com/array-unique/-/array-unique-0.3.2.tgz#a894b75d4bc4f6cd679ef3244a9fd8f46ae2d428"
|
||||
integrity sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=
|
||||
|
||||
arrow-key-navigation@^1.1.0:
|
||||
version "1.1.0"
|
||||
resolved "https://registry.yarnpkg.com/arrow-key-navigation/-/arrow-key-navigation-1.1.0.tgz#c0f7021d006593e2e34e79aa1f032714877d3a76"
|
||||
integrity sha512-u73yfJRmKye5eZiMNrAUKaBIRt47/1NM8WEtVAPjjMDab/PVn0sKIuapvCxl7C+tI9nH0QOl1Tc2YL2aO9n9Zw==
|
||||
arrow-key-navigation@^1.2.0:
|
||||
version "1.2.0"
|
||||
resolved "https://registry.yarnpkg.com/arrow-key-navigation/-/arrow-key-navigation-1.2.0.tgz#edefc5f8b4fc4e384e7c20ddecf81db7ffc970a9"
|
||||
integrity sha512-ch4WOwtjXHFisaa7ey2duW1Qf2VJxoa+8llbsbWDP6wsCzm0DGAi8upv6GDhf5xGvbxhKW3Co9SDEhXq34xCtg==
|
||||
|
||||
asar@^2.0.1:
|
||||
version "2.1.0"
|
||||
|
@ -1855,11 +1865,6 @@ babel-plugin-transform-object-rest-spread@^6.22.0:
|
|||
babel-plugin-syntax-object-rest-spread "^6.8.0"
|
||||
babel-runtime "^6.26.0"
|
||||
|
||||
babel-plugin-transform-react-remove-prop-types@^0.4.24:
|
||||
version "0.4.24"
|
||||
resolved "https://registry.yarnpkg.com/babel-plugin-transform-react-remove-prop-types/-/babel-plugin-transform-react-remove-prop-types-0.4.24.tgz#f2edaf9b4c6a5fbe5c1d678bfb531078c1555f3a"
|
||||
integrity sha512-eqj0hVcJUR57/Ug2zE1Yswsw4LhuqqHhD+8v120T1cl3kjg76QwtyBrdIk4WVwK+lAhBJVYCd/v+4nc4y+8JsA==
|
||||
|
||||
babel-plugin-transform-regenerator@^6.22.0:
|
||||
version "6.26.0"
|
||||
resolved "https://registry.yarnpkg.com/babel-plugin-transform-regenerator/-/babel-plugin-transform-regenerator-6.26.0.tgz#e0703696fbde27f0a3efcacf8b4dca2f7b3a8f2f"
|
||||
|
@ -2618,7 +2623,7 @@ class-utils@^0.3.5:
|
|||
isobject "^3.0.0"
|
||||
static-extend "^0.1.1"
|
||||
|
||||
clean-css@4.2.x, clean-css@^4.2.3:
|
||||
clean-css@4.2.x:
|
||||
version "4.2.3"
|
||||
resolved "https://registry.yarnpkg.com/clean-css/-/clean-css-4.2.3.tgz#507b5de7d97b48ee53d84adb0160ff6216380f78"
|
||||
integrity sha512-VcMWDN54ZN/DS+g58HYL5/n4Zrqe8vHJpGA8KdgUXFU4fuP/aHNw8eld9SyEIyabIMJX/0RaY/fplOo5hYLSFA==
|
||||
|
@ -3398,11 +3403,10 @@ emittery@^0.4.1:
|
|||
resolved "https://registry.yarnpkg.com/emittery/-/emittery-0.4.1.tgz#abe9d3297389ba424ac87e53d1c701962ce7433d"
|
||||
integrity sha512-r4eRSeStEGf6M5SKdrQhhLK5bOwOBxQhIE3YSTnZE3GpKiLfnnhE+tPtrJE79+eDJgm39BM6LSoI8SCx4HbwlQ==
|
||||
|
||||
emoji-mart@nolanlawson/emoji-mart#8bb6fb6:
|
||||
version "2.11.2"
|
||||
resolved "https://codeload.github.com/nolanlawson/emoji-mart/tar.gz/8bb6fb6622355ca33b570041005e0297f95e8a30"
|
||||
dependencies:
|
||||
prop-types "^15.6.0"
|
||||
emoji-picker-element@^1.0.0:
|
||||
version "1.0.0"
|
||||
resolved "https://registry.yarnpkg.com/emoji-picker-element/-/emoji-picker-element-1.0.0.tgz#e2682a4cc66eaf5e9aa7105a3847644b25e4e9cb"
|
||||
integrity sha512-duHgKHfKR1sNQ+n+zwvzlpw1ErwwhucL2fa2OEl+0Ei12y6asXVKEH31DH4YNU8GNNmlpsxckNCwdoAjSkG85A==
|
||||
|
||||
emoji-regex@^7.0.1:
|
||||
version "7.0.3"
|
||||
|
@ -3419,6 +3423,11 @@ emoji-regex@^9.0.0:
|
|||
resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-9.0.0.tgz#48a2309cc8a1d2e9d23bc6a67c39b63032e76ea4"
|
||||
integrity sha512-6p1NII1Vm62wni/VR/cUMauVQoxmLVb9csqQlvLz+hO2gk8U2UYDfXHQSUYIBKmZwAKz867IDqG7B+u0mj+M6w==
|
||||
|
||||
emojibase-data@^5.0.1:
|
||||
version "5.0.1"
|
||||
resolved "https://registry.yarnpkg.com/emojibase-data/-/emojibase-data-5.0.1.tgz#ce6fe36b4affd3578e0be8779211018a2fdae960"
|
||||
integrity sha512-rYWlogJ2q5P78U8Xx1vhsXHcYKu1wFnr7+o6z9QHssZ1SsJLTCkJINZIPHRFWuDreAUK457TkqHpdOXElu0fzA==
|
||||
|
||||
emojis-list@^3.0.0:
|
||||
version "3.0.0"
|
||||
resolved "https://registry.yarnpkg.com/emojis-list/-/emojis-list-3.0.0.tgz#5570662046ad29e2e916e71aae260abdff4f6a78"
|
||||
|
@ -6499,11 +6508,6 @@ postcss@^7.0.14, postcss@^7.0.16, postcss@^7.0.23, postcss@^7.0.5, postcss@^7.0.
|
|||
source-map "^0.6.1"
|
||||
supports-color "^6.1.0"
|
||||
|
||||
preact@^10.3.3:
|
||||
version "10.3.4"
|
||||
resolved "https://registry.yarnpkg.com/preact/-/preact-10.3.4.tgz#e1542a4d3eba3e7a37f312a2c331231b97052024"
|
||||
integrity sha512-wMgzs/RGYf0I1PZf8ZFJdyU/3kCcwepJyVYe+N9FGajyQWarMoPrPfrQajcG0psPj6ySYv2cSuLYFCihvV/Qrw==
|
||||
|
||||
prelude-ls@~1.1.2:
|
||||
version "1.1.2"
|
||||
resolved "https://registry.yarnpkg.com/prelude-ls/-/prelude-ls-1.1.2.tgz#21932a549f5e52ffd9a827f570e04be62a97da54"
|
||||
|
@ -6551,7 +6555,7 @@ promisify-event@^1.0.0:
|
|||
dependencies:
|
||||
pinkie-promise "^2.0.0"
|
||||
|
||||
prop-types@^15.6.0, prop-types@^15.7.2:
|
||||
prop-types@^15.7.2:
|
||||
version "15.7.2"
|
||||
resolved "https://registry.yarnpkg.com/prop-types/-/prop-types-15.7.2.tgz#52c41e75b8c87e72b9d9360e0206b99dcbffa6c5"
|
||||
integrity sha512-8QQikdH7//R2vurIJSutZ1smHYTcLpRWEOlHnzcWHmBYrOGUysKwSsrC89BCiFj3CbrfJ/nXFdJepOVrY1GCHQ==
|
||||
|
|
Loading…
Reference in New Issue