Merge pull request #209 from Rena-Ryumae/rena/customskin

Custom emoji for skin tone selector
nolan/hinaloe-test
Etienne Lemay 2018-08-10 14:15:31 -04:00 committed by GitHub
commit f5a4a488f8
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
10 changed files with 244 additions and 67 deletions

View File

@ -48,6 +48,7 @@ import { Picker } from 'emoji-mart'
| **emojiTooltip** | | `false` | Show emojis short name when hovering (title) | | **emojiTooltip** | | `false` | Show emojis short name when hovering (title) |
| **skin** | | | Forces skin color: `1, 2, 3, 4, 5, 6` | | **skin** | | | Forces skin color: `1, 2, 3, 4, 5, 6` |
| **defaultSkin** | | `1` | Default skin color: `1, 2, 3, 4, 5, 6` | | **defaultSkin** | | `1` | Default skin color: `1, 2, 3, 4, 5, 6` |
| **skinEmoji** | | | The emoji used to pick a skin tone. Uses an emoji-less skin tone picker by default |
| **style** | | | Inline styles applied to the root element. Useful for positioning | | **style** | | | Inline styles applied to the root element. Useful for positioning |
| **title** | | `Emoji Mart™` | The title shown when no emojis are hovered | | **title** | | `Emoji Mart™` | The title shown when no emojis are hovered |
| **notFoundEmoji** | | `sleuth_or_spy` | The emoji shown when there are no search results | | **notFoundEmoji** | | `sleuth_or_spy` | The emoji shown when there are no search results |
@ -58,6 +59,7 @@ import { Picker } from 'emoji-mart'
```js ```js
search: 'Search', search: 'Search',
notfound: 'No Emoji Found', notfound: 'No Emoji Found',
skintext: 'Choose your default skin tone',
categories: { categories: {
search: 'Search Results', search: 'Search Results',
recent: 'Frequently Used', recent: 'Frequently Used',

View File

@ -205,6 +205,11 @@
text-align: right; text-align: right;
} }
.emoji-mart-preview-skins.custom {
right: 10px;
text-align: right;
}
.emoji-mart-preview-name { .emoji-mart-preview-name {
font-size: 14px; font-size: 14px;
} }
@ -247,12 +252,18 @@
background-color: #fff; background-color: #fff;
} }
.emoji-mart-skin-swatches-opened .emoji-mart-skin-swatch { .emoji-mart-skin-swatches.custom {
font-size: 0;
border: none;
background-color: #fff;
}
.emoji-mart-skin-swatches.opened .emoji-mart-skin-swatch {
width: 16px; width: 16px;
padding: 0 2px; padding: 0 2px;
} }
.emoji-mart-skin-swatches-opened .emoji-mart-skin-swatch-selected:after { .emoji-mart-skin-swatches.opened .emoji-mart-skin-swatch.selected:after {
opacity: .75; opacity: .75;
} }
@ -272,12 +283,13 @@
.emoji-mart-skin-swatch:nth-child(5) { transition-delay: .12s } .emoji-mart-skin-swatch:nth-child(5) { transition-delay: .12s }
.emoji-mart-skin-swatch:nth-child(6) { transition-delay: .15s } .emoji-mart-skin-swatch:nth-child(6) { transition-delay: .15s }
.emoji-mart-skin-swatch-selected { .emoji-mart-skin-swatch.selected {
position: relative; position: relative;
width: 16px; width: 16px;
padding: 0 2px; padding: 0 2px;
} }
.emoji-mart-skin-swatch-selected:after {
.emoji-mart-skin-swatch.selected:after {
content: ""; content: "";
position: absolute; position: absolute;
top: 50%; left: 50%; top: 50%; left: 50%;
@ -290,9 +302,63 @@
transition: opacity .2s ease-out; transition: opacity .2s ease-out;
} }
.emoji-mart-skin-swatch.custom {
display: inline-block;
width: 0;
height: 38px;
overflow: hidden;
vertical-align: middle;
transition-property: width, height;
transition-duration: .125s;
transition-timing-function: ease-out;
cursor: default;
}
.emoji-mart-skin-swatch.custom.selected {
position: relative;
width: 36px;
height: 38px;
padding: 0 2px 0 0;
}
.emoji-mart-skin-swatch.custom.selected:after {
content: "";
width: 0;
height: 0;
}
.emoji-mart-skin-swatches.custom .emoji-mart-skin-swatch.custom:hover {
background-color: #f4f4f4;
border-radius: 10%;
}
.emoji-mart-skin-swatches.custom.opened .emoji-mart-skin-swatch.custom {
width: 36px;
height: 38px;
padding: 0 2px 0 0;
}
.emoji-mart-skin-swatches.custom.opened .emoji-mart-skin-swatch.custom.selected:after {
opacity: .75;
}
.emoji-mart-skin-text.opened {
display: inline-block;
vertical-align: middle;
text-align: left;
color: #888;
font-size: 11px;
padding: 5px 2px;
width: 95px;
height: 40px;
border-radius: 10%;
background-color: #fff;
}
.emoji-mart-skin { .emoji-mart-skin {
display: inline-block; display: inline-block;
width: 100%; padding-top: 100%; width: 100%;
padding-top: 100%;
max-width: 12px; max-width: 12px;
border-radius: 100%; border-radius: 100%;
} }

View File

@ -4,6 +4,8 @@ export { default as Preview } from './preview'
export { default as NotFound } from './not-found' export { default as NotFound } from './not-found'
export { default as Search } from './search' export { default as Search } from './search'
export { default as Skins } from './skins' export { default as Skins } from './skins'
export { default as SkinsEmoji } from './skins-emoji'
export { default as SkinsDot } from './skins-dot'
export { default as Emoji } from './emoji/emoji' export { default as Emoji } from './emoji/emoji'
export { default as NimbleEmoji } from './emoji/nimble-emoji' export { default as NimbleEmoji } from './emoji/nimble-emoji'

View File

@ -15,6 +15,7 @@ import { Anchors, Category, Preview, Search } from '..'
const I18N = { const I18N = {
search: 'Search', search: 'Search',
notfound: 'No Emoji Found', notfound: 'No Emoji Found',
skintext: 'Choose your default skin tone',
categories: { categories: {
search: 'Search Results', search: 'Search Results',
recent: 'Frequently Used', recent: 'Frequently Used',
@ -471,6 +472,7 @@ export default class NimblePicker extends React.PureComponent {
exclude, exclude,
recent, recent,
autoFocus, autoFocus,
skinEmoji,
notFound, notFound,
notFoundEmoji, notFoundEmoji,
} = this.props, } = this.props,
@ -574,7 +576,9 @@ export default class NimblePicker extends React.PureComponent {
skinsProps={{ skinsProps={{
skin: skin, skin: skin,
onChange: this.handleSkinChange, onChange: this.handleSkinChange,
skinEmoji: skinEmoji,
}} }}
i18n={this.i18n}
/> />
</div> </div>
)} )}

View File

@ -2,7 +2,7 @@ import React from 'react'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import { getData } from '../utils' import { getData } from '../utils'
import { NimbleEmoji, Skins } from '.' import { NimbleEmoji, SkinsEmoji, SkinsDot } from '.'
export default class Preview extends React.PureComponent { export default class Preview extends React.PureComponent {
constructor(props) { constructor(props) {
@ -20,6 +20,7 @@ export default class Preview extends React.PureComponent {
showSkinTones, showSkinTones,
title, title,
emoji: idleEmoji, emoji: idleEmoji,
i18n,
} = this.props } = this.props
if (emoji) { if (emoji) {
@ -81,8 +82,27 @@ export default class Preview extends React.PureComponent {
</div> </div>
{showSkinTones && ( {showSkinTones && (
<div className="emoji-mart-preview-skins"> <div
<Skins {...skinsProps} /> className={`emoji-mart-preview-skins${
skinsProps.skinEmoji ? ' custom' : ''
}`}
>
{skinsProps.skinEmoji ? (
<SkinsEmoji
skin={skinsProps.skin}
emojiProps={emojiProps}
data={this.data}
skinEmoji={skinsProps.skinEmoji}
i18n={i18n}
onChange={skinsProps.onChange}
/>
) : (
<SkinsDot
skin={skinsProps.skin}
i18n={i18n}
onChange={skinsProps.onChange}
/>
)}
</div> </div>
)} )}
</div> </div>

View File

@ -0,0 +1,50 @@
import React from 'react'
import PropTypes from 'prop-types'
import { Skins } from '.'
export default class SkinsDot extends Skins {
constructor(props) {
super(props)
this.handleClick = this.handleClick.bind(this)
}
render() {
const { skin, i18n } = this.props
const { opened } = this.state
const skinToneNodes = []
for (let skinTone = 1; skinTone <= 6; skinTone++) {
const selected = skinTone === skin
skinToneNodes.push(
<span
key={`skin-tone-${skinTone}`}
className={`emoji-mart-skin-swatch${selected ? ' selected' : ''}`}
>
<span
onClick={this.handleClick}
data-skin={skinTone}
className={`emoji-mart-skin emoji-mart-skin-tone-${skinTone}`}
/>
</span>,
)
}
return (
<div className={`emoji-mart-skin-swatches${opened ? ' opened' : ''}`}>
{skinToneNodes}
</div>
)
}
}
SkinsDot.propTypes = {
onChange: PropTypes.func,
skin: PropTypes.number.isRequired,
i18n: PropTypes.object,
}
SkinsDot.defaultProps = {
onChange: () => {},
}

View File

@ -0,0 +1,72 @@
import React from 'react'
import PropTypes from 'prop-types'
import { NimbleEmoji, Skins } from '.'
export default class SkinsEmoji extends Skins {
constructor(props) {
super(props)
this.handleClick = this.handleClick.bind(this)
}
render() {
const { skin, emojiProps, data, skinEmoji, i18n } = this.props
const { opened } = this.state
const skinToneNodes = []
for (let skinTone = 1; skinTone <= 6; skinTone++) {
const selected = skinTone === skin
skinToneNodes.push(
<span
key={`skin-tone-${skinTone}`}
className={`emoji-mart-skin-swatch custom${
selected ? ' selected' : ''
}`}
>
<span
onClick={this.handleClick}
data-skin={skinTone}
className={`emoji-mart-skin-tone-${skinTone}`}
>
{NimbleEmoji({
emoji: skinEmoji,
data: data,
skin: skinTone,
backgroundImageFn: emojiProps.backgroundImageFn,
native: emojiProps.native,
set: emojiProps.set,
sheetSize: emojiProps.sheetSize,
size: 23,
})}
</span>
</span>,
)
}
return (
<div
className={`emoji-mart-skin-swatches custom${opened ? ' opened' : ''}`}
>
<div className={`emoji-mart-skin-text${opened ? ' opened' : ''}`}>
{i18n.skintext}
</div>
{skinToneNodes}
</div>
)
}
}
SkinsEmoji.propTypes = {
onChange: PropTypes.func,
skin: PropTypes.number.isRequired,
emojiProps: PropTypes.object.isRequired,
skinTone: PropTypes.number,
skinEmoji: PropTypes.string.isRequired,
i18n: PropTypes.object,
}
SkinsEmoji.defaultProps = {
onChange: () => {},
skinTone: null,
}

View File

@ -1,15 +1,14 @@
import React from 'react' import React from 'react'
import PropTypes from 'prop-types' import PropTypes from 'prop-types'
import { NimbleEmoji } from '.'
export default class Skins extends React.PureComponent { export default class Skins extends React.PureComponent {
constructor(props) { constructor(props) {
super(props) super(props)
this.state = { this.state = {
opened: false, opened: false,
} }
this.handleClick = this.handleClick.bind(this)
} }
handleClick(e) { handleClick(e) {
@ -27,42 +26,7 @@ export default class Skins extends React.PureComponent {
} }
render() { render() {
const { skin } = this.props return null
const { opened } = this.state
const skinToneNodes = []
for (let i = 0; i < 6; i++) {
const skinTone = i + 1
const selected = skinTone == skin
skinToneNodes.push(
<span
key={`skin-tone-${skinTone}`}
className={`emoji-mart-skin-swatch ${
selected ? 'emoji-mart-skin-swatch-selected' : ''
}`}
>
<span
onClick={this.handleClick}
data-skin={skinTone}
className={`emoji-mart-skin emoji-mart-skin-tone-${skinTone}`}
/>
</span>,
)
}
return (
<div>
<div
className={`emoji-mart-skin-swatches ${
opened ? 'emoji-mart-skin-swatches-opened' : ''
}`}
>
{skinToneNodes}
</div>
</div>
)
} }
} }

View File

@ -75,6 +75,7 @@ const PickerPropTypes = {
imageUrl: PropTypes.string.isRequired, imageUrl: PropTypes.string.isRequired,
}), }),
), ),
skinEmoji: PropTypes.string,
notFound: PropTypes.func, notFound: PropTypes.func,
notFoundEmoji: PropTypes.string, notFoundEmoji: PropTypes.string,
icons: PropTypes.object, icons: PropTypes.object,
@ -103,6 +104,7 @@ const PickerDefaultProps = {
emojiTooltip: EmojiDefaultProps.tooltip, emojiTooltip: EmojiDefaultProps.tooltip,
autoFocus: false, autoFocus: false,
custom: [], custom: [],
skinEmoji: '',
notFound: () => {}, notFound: () => {},
notFoundEmoji: 'sleuth_or_spy', notFoundEmoji: 'sleuth_or_spy',
icons: {}, icons: {},

View File

@ -32,7 +32,7 @@ const CUSTOM_EMOJIS = [
storiesOf('Picker', module) storiesOf('Picker', module)
.addDecorator(withKnobs) .addDecorator(withKnobs)
.add('default', () => ( .add('Default', () => (
<Picker <Picker
onClick={action('clicked')} onClick={action('clicked')}
onSelect={action('selected')} onSelect={action('selected')}
@ -51,29 +51,16 @@ storiesOf('Picker', module)
custom={CUSTOM_EMOJIS} custom={CUSTOM_EMOJIS}
/> />
)) ))
.add('with a custom not found image', () => (
.add('Custom “Not found” component', () => (
<Picker <Picker
notFound={() => ( notFound={() => (
<img src="https://assets-cdn.github.com/images/icons/emoji/octocat.png?v7" /> <img src="https://assets-cdn.github.com/images/icons/emoji/octocat.png?v7" />
)} )}
/> />
)) ))
.add('with a custom not found SVG', () => (
<Picker .add('Custom category icons', () => (
notFound={() => (
<svg
aria-labelledby="simpleicons-jira-icon"
role="img"
viewBox="0 0 24 24"
xmlns="http://www.w3.org/2000/svg"
>
<title id="simpleicons-jira-icon">Jira icon</title>
<path d="M23.323 11.33L13.001 1 12 0 4.225 7.775.67 11.33a.96.96 0 0 0 0 1.347l7.103 7.103L12 24l7.771-7.771.121-.121 3.431-3.431a.945.945 0 0 0 0-1.347zM12 15.551L8.449 12 12 8.453 15.548 12 12 15.551z" />
</svg>
)}
/>
))
.add('with-custom-icons', () => (
<Picker <Picker
custom={CUSTOM_EMOJIS} custom={CUSTOM_EMOJIS}
icons={{ icons={{
@ -181,9 +168,17 @@ storiesOf('Picker', module)
/> />
)) ))
.add('Custom skin emoji', () => (
<Picker
native={boolean('Unicode', true)}
emojiSize={24}
skinEmoji={text('Skin Preview Icon', 'v')}
/>
))
storiesOf('Emoji', module) storiesOf('Emoji', module)
.addDecorator(withKnobs) .addDecorator(withKnobs)
.add('default', () => ( .add('Default', () => (
<Emoji <Emoji
native={boolean('Unicode', true)} native={boolean('Unicode', true)}
set={select('Emoji pack', SETS, SETS[0])} set={select('Emoji pack', SETS, SETS[0])}
@ -199,7 +194,7 @@ storiesOf('Emoji', module)
storiesOf('Headless Search', module) storiesOf('Headless Search', module)
.addDecorator(withKnobs) .addDecorator(withKnobs)
.add('default', () => { .add('Default', () => {
let results = emojiIndex.search(text('Search', 'christmas'), { let results = emojiIndex.search(text('Search', 'christmas'), {
custom: CUSTOM_EMOJIS, custom: CUSTOM_EMOJIS,
}) })