Customizable Category Icons
Currently, the default emoji-mart category icons cannot be changed. We propose creating a categories object that uses the emoji-mart category icons on default but otherwise uses custom imgs/svgs passed in by the userrelease
parent
b1c499fc8d
commit
31cc0ab215
18
README.md
18
README.md
|
@ -50,6 +50,7 @@ import { Picker } from 'emoji-mart'
|
||||||
| **defaultSkin** | | `1` | Default skin color: `1, 2, 3, 4, 5, 6` |
|
| **defaultSkin** | | `1` | Default skin color: `1, 2, 3, 4, 5, 6` |
|
||||||
| **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 |
|
||||||
|
| **icons** | | `{}` | [Custom icons](#custom-icons) |
|
||||||
|
|
||||||
#### I18n
|
#### I18n
|
||||||
```js
|
```js
|
||||||
|
@ -234,6 +235,23 @@ const customEmojis = [
|
||||||
<Picker custom={customEmojis} />
|
<Picker custom={customEmojis} />
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Custom icons
|
||||||
|
You can provide custom icons which will override the default icons.
|
||||||
|
|
||||||
|
```js
|
||||||
|
import { Picker } from 'emoji-mart'
|
||||||
|
|
||||||
|
const customIcons = {
|
||||||
|
categories: {
|
||||||
|
recent: () => <img src='https://assets-cdn.github.com/images/icons/emoji/octocat.png?v7' />,
|
||||||
|
foods: () => <svg role="img" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><path d="M0 0l6.084 24H8L1.916 0zM21 5h-4l-1-4H4l3 12h3l1 4h13L21 5zM6.563 3h7.875l2 8H8.563l-2-8zm8.832 10l-2.856 1.904L12.063 13h3.332zM19 13l-1.5-6h1.938l2 8H16l3-2z"/></svg>,
|
||||||
|
people: () => <svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 16 16"><path d="M3 2l10 6-10 6z"></path></svg>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
<Picker icons={customIcons} />
|
||||||
|
```
|
||||||
|
|
||||||
## Headless search
|
## Headless search
|
||||||
The `Picker` doesn’t have to be mounted for you to take advantage of the advanced search results.
|
The `Picker` doesn’t have to be mounted for you to take advantage of the advanced search results.
|
||||||
|
|
||||||
|
|
|
@ -72,9 +72,11 @@
|
||||||
max-width: 22px;
|
max-width: 22px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.emoji-mart-anchors svg {
|
.emoji-mart-anchors svg,
|
||||||
|
.emoji-mart-anchors img {
|
||||||
fill: currentColor;
|
fill: currentColor;
|
||||||
max-height: 18px;
|
height: 18px;
|
||||||
|
width: 18px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.emoji-mart-scroll {
|
.emoji-mart-scroll {
|
||||||
|
|
|
@ -1,8 +1,6 @@
|
||||||
import React from 'react'
|
import React from 'react'
|
||||||
import PropTypes from 'prop-types'
|
import PropTypes from 'prop-types'
|
||||||
|
|
||||||
import SVGs from '../svgs'
|
|
||||||
|
|
||||||
export default class Anchors extends React.PureComponent {
|
export default class Anchors extends React.PureComponent {
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
super(props)
|
super(props)
|
||||||
|
@ -18,21 +16,6 @@ export default class Anchors extends React.PureComponent {
|
||||||
this.handleClick = this.handleClick.bind(this)
|
this.handleClick = this.handleClick.bind(this)
|
||||||
}
|
}
|
||||||
|
|
||||||
getSVG(id) {
|
|
||||||
this.SVGs || (this.SVGs = {})
|
|
||||||
|
|
||||||
if (this.SVGs[id]) {
|
|
||||||
return this.SVGs[id]
|
|
||||||
} else {
|
|
||||||
let svg = `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" width="24" height="24">
|
|
||||||
${SVGs[id]}
|
|
||||||
</svg>`
|
|
||||||
|
|
||||||
this.SVGs[id] = svg
|
|
||||||
return svg
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
handleClick(e) {
|
handleClick(e) {
|
||||||
var index = e.currentTarget.getAttribute('data-index')
|
var index = e.currentTarget.getAttribute('data-index')
|
||||||
var { categories, onAnchorClick } = this.props
|
var { categories, onAnchorClick } = this.props
|
||||||
|
@ -41,7 +24,7 @@ export default class Anchors extends React.PureComponent {
|
||||||
}
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
var { categories, onAnchorClick, color, i18n } = this.props,
|
var { categories, color, i18n, icons } = this.props,
|
||||||
{ selected } = this.state
|
{ selected } = this.state
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
@ -65,7 +48,9 @@ export default class Anchors extends React.PureComponent {
|
||||||
}`}
|
}`}
|
||||||
style={{ color: isSelected ? color : null }}
|
style={{ color: isSelected ? color : null }}
|
||||||
>
|
>
|
||||||
<div dangerouslySetInnerHTML={{ __html: this.getSVG(id) }} />
|
<div className="emoji-mart-anchor-icon">
|
||||||
|
{icons.categories[id]()}
|
||||||
|
</div>
|
||||||
<span
|
<span
|
||||||
className="emoji-mart-anchor-bar"
|
className="emoji-mart-anchor-bar"
|
||||||
style={{ backgroundColor: color }}
|
style={{ backgroundColor: color }}
|
||||||
|
@ -81,9 +66,11 @@ export default class Anchors extends React.PureComponent {
|
||||||
Anchors.propTypes = {
|
Anchors.propTypes = {
|
||||||
categories: PropTypes.array,
|
categories: PropTypes.array,
|
||||||
onAnchorClick: PropTypes.func,
|
onAnchorClick: PropTypes.func,
|
||||||
|
icons: PropTypes.object,
|
||||||
}
|
}
|
||||||
|
|
||||||
Anchors.defaultProps = {
|
Anchors.defaultProps = {
|
||||||
categories: [],
|
categories: [],
|
||||||
onAnchorClick: () => {},
|
onAnchorClick: () => {},
|
||||||
|
icons: {},
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,6 +3,7 @@ import '../../vendor/raf-polyfill'
|
||||||
import React from 'react'
|
import React from 'react'
|
||||||
import PropTypes from 'prop-types'
|
import PropTypes from 'prop-types'
|
||||||
|
|
||||||
|
import SVGs from '../../svgs'
|
||||||
import store from '../../utils/store'
|
import store from '../../utils/store'
|
||||||
import frequently from '../../utils/frequently'
|
import frequently from '../../utils/frequently'
|
||||||
import { deepMerge, measureScrollbar } from '../../utils'
|
import { deepMerge, measureScrollbar } from '../../utils'
|
||||||
|
@ -29,6 +30,21 @@ const I18N = {
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const toSVG = (val) => () => (
|
||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
dangerouslySetInnerHTML={{ __html: val }}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
|
||||||
|
const ICON_FNS = {
|
||||||
|
categories: Object.keys(SVGs).reduce((acc, cur) => {
|
||||||
|
acc[cur] = toSVG(SVGs[cur])
|
||||||
|
return acc
|
||||||
|
}, {}),
|
||||||
|
}
|
||||||
|
|
||||||
export default class NimblePicker extends React.PureComponent {
|
export default class NimblePicker extends React.PureComponent {
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
super(props)
|
super(props)
|
||||||
|
@ -48,6 +64,7 @@ export default class NimblePicker extends React.PureComponent {
|
||||||
|
|
||||||
this.data = props.data
|
this.data = props.data
|
||||||
this.i18n = deepMerge(I18N, props.i18n)
|
this.i18n = deepMerge(I18N, props.i18n)
|
||||||
|
this.icons = deepMerge(ICON_FNS, props.icons)
|
||||||
this.state = {
|
this.state = {
|
||||||
skin: props.skin || store.get('skin') || props.defaultSkin,
|
skin: props.skin || store.get('skin') || props.defaultSkin,
|
||||||
firstRender: true,
|
firstRender: true,
|
||||||
|
@ -485,6 +502,7 @@ export default class NimblePicker extends React.PureComponent {
|
||||||
color={color}
|
color={color}
|
||||||
categories={this.categories}
|
categories={this.categories}
|
||||||
onAnchorClick={this.handleAnchorClick}
|
onAnchorClick={this.handleAnchorClick}
|
||||||
|
icons={this.icons}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
|
@ -71,6 +71,7 @@ const PickerPropTypes = {
|
||||||
imageUrl: PropTypes.string.isRequired,
|
imageUrl: PropTypes.string.isRequired,
|
||||||
}),
|
}),
|
||||||
),
|
),
|
||||||
|
icons: PropTypes.object,
|
||||||
}
|
}
|
||||||
|
|
||||||
const PickerDefaultProps = {
|
const PickerDefaultProps = {
|
||||||
|
@ -96,6 +97,7 @@ const PickerDefaultProps = {
|
||||||
emojiTooltip: EmojiDefaultProps.tooltip,
|
emojiTooltip: EmojiDefaultProps.tooltip,
|
||||||
autoFocus: false,
|
autoFocus: false,
|
||||||
custom: [],
|
custom: [],
|
||||||
|
icons: {},
|
||||||
}
|
}
|
||||||
|
|
||||||
export {
|
export {
|
||||||
|
|
|
@ -41,6 +41,26 @@ storiesOf('Picker', module)
|
||||||
showPreview={boolean('Show preview', true)}
|
showPreview={boolean('Show preview', true)}
|
||||||
showSkinTones={boolean('Show skin tones', true)}
|
showSkinTones={boolean('Show skin tones', true)}
|
||||||
custom={CUSTOM_EMOJIS}
|
custom={CUSTOM_EMOJIS}
|
||||||
|
/>))
|
||||||
|
.add('with-custom-icons', () => (
|
||||||
|
<Picker
|
||||||
|
custom={CUSTOM_EMOJIS}
|
||||||
|
icons={
|
||||||
|
{
|
||||||
|
categories: {
|
||||||
|
recent: () => <img src='https://assets-cdn.github.com/images/icons/emoji/octocat.png?v7' />,
|
||||||
|
people: () => <svg aria-labelledby="simpleicons-reddit-icon" role="img" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><title id="simpleicons-reddit-icon">Reddit icon</title><path d="M2.204 14.049c-.06.276-.091.56-.091.847 0 3.443 4.402 6.249 9.814 6.249 5.41 0 9.812-2.804 9.812-6.249 0-.274-.029-.546-.082-.809l-.015-.032c-.021-.055-.029-.11-.029-.165-.302-1.175-1.117-2.241-2.296-3.103-.045-.016-.088-.039-.126-.07-.026-.02-.045-.042-.067-.064-1.792-1.234-4.356-2.008-7.196-2.008-2.815 0-5.354.759-7.146 1.971-.014.018-.029.033-.049.049-.039.033-.084.06-.13.075-1.206.862-2.042 1.937-2.354 3.123 0 .058-.014.114-.037.171l-.008.015zm9.773 5.441c-1.794 0-3.057-.389-3.863-1.197-.173-.174-.173-.457 0-.632.176-.165.46-.165.635 0 .63.629 1.685.943 3.228.943 1.542 0 2.591-.3 3.219-.929.165-.164.45-.164.629 0 .165.18.165.465 0 .645-.809.808-2.065 1.198-3.862 1.198l.014-.028zm-3.606-7.573c-.914 0-1.677.765-1.677 1.677 0 .91.763 1.65 1.677 1.65s1.651-.74 1.651-1.65c0-.912-.739-1.677-1.651-1.677zm7.233 0c-.914 0-1.678.765-1.678 1.677 0 .91.764 1.65 1.678 1.65s1.651-.74 1.651-1.65c0-.912-.739-1.677-1.651-1.677zm4.548-1.595c1.037.833 1.8 1.821 2.189 2.904.45-.336.719-.864.719-1.449 0-1.002-.815-1.816-1.818-1.816-.399 0-.778.129-1.09.363v-.002zM2.711 9.963c-1.003 0-1.817.816-1.817 1.818 0 .543.239 1.048.644 1.389.401-1.079 1.172-2.053 2.213-2.876-.302-.21-.663-.329-1.039-.329v-.002zm9.217 12.079c-5.906 0-10.709-3.205-10.709-7.142 0-.275.023-.544.068-.809C.494 13.598 0 12.729 0 11.777c0-1.496 1.227-2.713 2.725-2.713.674 0 1.303.246 1.797.682 1.856-1.191 4.357-1.941 7.112-1.992l1.812-5.524.404.095s.016 0 .016.002l4.223.993c.344-.798 1.138-1.36 2.065-1.36 1.229 0 2.231 1.004 2.231 2.234 0 1.232-1.003 2.234-2.231 2.234s-2.23-1.004-2.23-2.23l-3.851-.912-1.467 4.477c2.65.105 5.047.854 6.844 2.021.494-.464 1.144-.719 1.833-.719 1.498 0 2.718 1.213 2.718 2.711 0 .987-.54 1.886-1.378 2.365.029.255.059.494.059.749-.015 3.938-4.806 7.143-10.72 7.143l-.034.009zm8.179-19.187c-.74 0-1.34.599-1.34 1.338 0 .738.6 1.34 1.34 1.34.732 0 1.33-.6 1.33-1.334 0-.733-.598-1.332-1.347-1.332l.017-.012z"/></svg>,
|
||||||
|
nature: () => <svg version="1.1" xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 16 16"><path d="M8 16c4.418 0 8-3.582 8-8s-3.582-8-8-8-8 3.582-8 8 3.582 8 8 8zM8 1.5c3.59 0 6.5 2.91 6.5 6.5s-2.91 6.5-6.5 6.5-6.5-2.91-6.5-6.5 2.91-6.5 6.5-6.5zM4 5c0-0.552 0.448-1 1-1s1 0.448 1 1-0.448 1-1 1-1-0.448-1-1zM10 5c0-0.552 0.448-1 1-1s1 0.448 1 1-0.448 1-1 1-1-0.448-1-1z"></path><path d="M10.561 8.439c-0.586-0.586-1.536-0.586-2.121 0s-0.586 1.536 0 2.121c0.019 0.019 0.038 0.037 0.058 0.055 1.352 1.227 4.503-0.029 4.503-1.615-0.969 0.625-1.726 0.153-2.439-0.561z"></path><path d="M5.439 8.439c0.586-0.586 1.536-0.586 2.121 0s0.586 1.536 0 2.121c-0.019 0.019-0.038 0.037-0.058 0.055-1.352 1.227-4.503-0.029-4.503-1.615 0.969 0.625 1.726 0.153 2.439-0.561z"></path></svg>,
|
||||||
|
foods: () => <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>,
|
||||||
|
activity: () => <img src='https://assets-cdn.github.com/images/icons/emoji/shipit.png?v7' />,
|
||||||
|
places: () => <svg aria-labelledby="simpleicons-stackoverflow-icon" role="img" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><title id="simpleicons-stackoverflow-icon">Stack Overflow icon</title><path d="M18.986 21.865v-6.404h2.134V24H1.844v-8.539h2.13v6.404h15.012zM6.111 19.731H16.85v-2.137H6.111v2.137zm.259-4.852l10.48 2.189.451-2.07-10.478-2.187-.453 2.068zm1.359-5.056l9.705 4.53.903-1.95-9.706-4.53-.902 1.936v.014zm2.715-4.785l8.217 6.855 1.359-1.62-8.216-6.853-1.35 1.617-.01.001zM15.751 0l-1.746 1.294 6.405 8.604 1.746-1.294L15.749 0h.002z"/></svg>,
|
||||||
|
objects: () => <svg aria-labelledby="simpleicons-500px-icon" role="img" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" ><title id="simpleicons-500px-icon">Atlassian icon</title><path d="M19.31 23.957H15.71a1.26 1.26 0 0 1-1.312-.792c-1.332-2.665-2.78-5.288-3.987-8.046a15.25 15.25 0 0 1 .885-14.47c.166-.281.52-.625.791-.625s.593.375.74.666q5.444 10.89 10.898 21.788c.542 1.041.292 1.468-.905 1.479zm-14.573 0H1.04c-1.041 0-1.27-.417-.812-1.333q2.8-5.538 5.549-11.055c.5-1.041.895-1.041 1.592-.177a12.221 12.221 0 0 1 2.51 11.17c-.344 1.322-.532 1.405-1.864 1.405z"/></svg>,
|
||||||
|
symbols: () => <svg aria-labelledby="simpleicons-hipchat-icon" role="img" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><title id="simpleicons-hipchat-icon">HipChat icon</title><path d="M19.736 19.056s.103-.073.267-.198C22.46 16.958 24 14.203 24 11.139 24 5.424 18.627.787 12.003.787 5.377.787 0 5.424 0 11.139c0 5.717 5.371 10.356 11.998 10.356.847 0 1.694-.073 2.524-.228l.262-.045c1.683 1.092 4.139 1.99 6.288 1.99.665 0 .978-.546.552-1.104-.648-.795-1.541-2.068-1.888-3.052zm-1.462-4.526c-.716 1.069-2.934 2.889-6.254 2.889h-.046c-3.328 0-5.543-1.831-6.254-2.889a1.137 1.137 0 0 1-.273-.574.49.49 0 0 1 .447-.526c.008-.003.014-.003.021-.003.117.006.23.043.328.111a9.137 9.137 0 0 0 5.754 2.056 8.805 8.805 0 0 0 5.76-2.059.461.461 0 0 1 .313-.122c.267 0 .478.213.483.475a1.321 1.321 0 0 1-.268.643h-.011z"/></svg>,
|
||||||
|
flags: () => <svg version="1.1" xmlns="http://www.w3.org/2000/svg"width="16" height="16" viewBox="0 0 16 16"><path d="M4.75 9.5c0 1.795 1.455 3.25 3.25 3.25s3.25-1.455 3.25-3.25-1.455-3.25-3.25-3.25-3.25 1.455-3.25 3.25zM15 4h-3.5c-0.25-1-0.5-2-1.5-2h-4c-1 0-1.25 1-1.5 2h-3.5c-0.55 0-1 0.45-1 1v9c0 0.55 0.45 1 1 1h14c0.55 0 1-0.45 1-1v-9c0-0.55-0.45-1-1-1zM8 13.938c-2.451 0-4.438-1.987-4.438-4.438s1.987-4.438 4.438-4.438c2.451 0 4.438 1.987 4.438 4.438s-1.987 4.438-4.438 4.438zM15 7h-2v-1h2v1z"></path></svg>,
|
||||||
|
custom: () => <svg aria-labelledby="simpleicons-trello-icon" role="img" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"><title id="simpleicons-trello-icon">Trello icon</title><path d="M21 0H3C1.343 0 0 1.343 0 3v18c0 1.656 1.343 3 3 3h18c1.656 0 3-1.344 3-3V3c0-1.657-1.344-3-3-3zM10.44 18.18c0 .795-.645 1.44-1.44 1.44H4.56c-.795 0-1.44-.646-1.44-1.44V4.56c0-.795.645-1.44 1.44-1.44H9c.795 0 1.44.645 1.44 1.44v13.62zm10.44-6c0 .794-.645 1.44-1.44 1.44H15c-.795 0-1.44-.646-1.44-1.44V4.56c0-.795.646-1.44 1.44-1.44h4.44c.795 0 1.44.645 1.44 1.44v7.62z"/></svg>,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
/>
|
/>
|
||||||
));
|
));
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue