2016-03-25 01:42:44 +08:00
|
|
|
import React from 'react';
|
|
|
|
import {Actions} from 'nylas-exports';
|
2016-03-31 06:52:51 +08:00
|
|
|
import {RetinaImg, ScrollRegion} from 'nylas-component-kit';
|
2016-03-25 01:42:44 +08:00
|
|
|
|
|
|
|
import EmojiStore from './emoji-store';
|
|
|
|
import EmojiActions from './emoji-actions';
|
|
|
|
import categorizedEmojiList from './categorized-emoji';
|
|
|
|
|
|
|
|
class EmojiButtonPopover extends React.Component {
|
|
|
|
static displayName = 'EmojiButtonPopover';
|
|
|
|
|
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
const {categoryNames,
|
|
|
|
categorizedEmoji,
|
|
|
|
categoryPositions} = this.getStateFromStore();
|
|
|
|
this.state = {
|
|
|
|
emojiName: "Emoji Picker",
|
|
|
|
categoryNames: categoryNames,
|
|
|
|
categorizedEmoji: categorizedEmoji,
|
|
|
|
categoryPositions: categoryPositions,
|
|
|
|
searchValue: "",
|
|
|
|
activeTab: Object.keys(categorizedEmoji)[0],
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
componentDidMount() {
|
|
|
|
this._mounted = true;
|
2016-04-13 04:57:42 +08:00
|
|
|
this._emojiPreloadImage = new Image();
|
2016-03-25 01:42:44 +08:00
|
|
|
this.renderCanvas();
|
|
|
|
}
|
|
|
|
|
|
|
|
componentWillUnmount() {
|
2016-04-13 04:57:42 +08:00
|
|
|
this._emojiPreloadImage.onload = null;
|
|
|
|
this._emojiPreloadImage = null;
|
2016-03-25 01:42:44 +08:00
|
|
|
this._mounted = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
onMouseDown = (event) => {
|
|
|
|
const emojiName = this.calcEmojiByPosition(this.calcPosition(event));
|
|
|
|
if (!emojiName) return null;
|
2016-04-01 09:14:05 +08:00
|
|
|
EmojiActions.selectEmoji({emojiName: emojiName, replaceSelection: false});
|
2016-03-25 01:42:44 +08:00
|
|
|
Actions.closePopover();
|
2016-05-07 05:10:56 +08:00
|
|
|
return null
|
2016-03-25 01:42:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
onScroll = () => {
|
2016-04-07 03:00:59 +08:00
|
|
|
const emojiContainer = document.querySelector(".emoji-finder-container .scroll-region-content");
|
2016-03-25 01:42:44 +08:00
|
|
|
const tabContainer = document.querySelector(".emoji-tabs");
|
|
|
|
tabContainer.className = emojiContainer.scrollTop ? "emoji-tabs shadow" : "emoji-tabs";
|
|
|
|
if (emojiContainer.scrollTop === 0) {
|
|
|
|
this.setState({activeTab: Object.keys(this.state.categorizedEmoji)[0]});
|
|
|
|
} else {
|
2016-05-07 05:10:56 +08:00
|
|
|
for (const category of Object.keys(this.state.categoryPositions)) {
|
|
|
|
if (emojiContainer.scrollTop >= this.state.categoryPositions[category].top &&
|
|
|
|
emojiContainer.scrollTop <= this.state.categoryPositions[category].bottom) {
|
|
|
|
this.setState({activeTab: category});
|
2016-03-25 01:42:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
onHover = (event) => {
|
|
|
|
const emojiName = this.calcEmojiByPosition(this.calcPosition(event));
|
|
|
|
if (emojiName) {
|
|
|
|
this.setState({emojiName: emojiName});
|
|
|
|
} else {
|
|
|
|
this.setState({emojiName: "Emoji Picker"});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
onMouseOut = () => {
|
|
|
|
this.setState({emojiName: "Emoji Picker"});
|
|
|
|
}
|
|
|
|
|
|
|
|
onChange = (event) => {
|
|
|
|
const searchValue = event.target.value;
|
|
|
|
if (searchValue.length > 0) {
|
|
|
|
const searchMatches = this.findSearchMatches(searchValue);
|
|
|
|
this.setState({
|
|
|
|
categorizedEmoji: {
|
|
|
|
'Search Results': searchMatches,
|
|
|
|
},
|
|
|
|
categoryPositions: {
|
|
|
|
'Search Results': {
|
|
|
|
top: 25,
|
|
|
|
bottom: 25 + Math.ceil(searchMatches.length / 8) * 24,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
searchValue: searchValue,
|
|
|
|
activeTab: null,
|
|
|
|
}, this.renderCanvas);
|
|
|
|
} else {
|
|
|
|
this.setState(this.getStateFromStore, () => {
|
|
|
|
this.setState({
|
|
|
|
searchValue: searchValue,
|
|
|
|
activeTab: Object.keys(this.state.categorizedEmoji)[0],
|
|
|
|
}, this.renderCanvas);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
getStateFromStore = () => {
|
|
|
|
let categorizedEmoji = categorizedEmojiList;
|
|
|
|
const categoryPositions = {};
|
|
|
|
let categoryNames = [
|
|
|
|
'People',
|
|
|
|
'Nature',
|
|
|
|
'Food and Drink',
|
|
|
|
'Activity',
|
|
|
|
'Travel and Places',
|
|
|
|
'Objects',
|
|
|
|
'Symbols',
|
|
|
|
'Flags',
|
|
|
|
];
|
|
|
|
const frequentlyUsedEmoji = EmojiStore.frequentlyUsedEmoji();
|
|
|
|
if (frequentlyUsedEmoji.length > 0) {
|
|
|
|
categorizedEmoji = {'Frequently Used': frequentlyUsedEmoji};
|
2016-05-07 05:10:56 +08:00
|
|
|
for (const category of Object.keys(categorizedEmojiList)) {
|
|
|
|
categorizedEmoji[category] = categorizedEmojiList[category];
|
2016-03-25 01:42:44 +08:00
|
|
|
}
|
|
|
|
categoryNames = ["Frequently Used"].concat(categoryNames);
|
|
|
|
}
|
2016-03-25 02:57:49 +08:00
|
|
|
// Calculates where each category should be (variable because Frequently
|
|
|
|
// Used may or may not be present)
|
2016-03-25 01:42:44 +08:00
|
|
|
for (const name of categoryNames) {
|
|
|
|
categoryPositions[name] = {top: 0, bottom: 0};
|
|
|
|
}
|
|
|
|
let verticalPos = 25;
|
2016-05-07 05:10:56 +08:00
|
|
|
for (const category of Object.keys(categoryPositions)) {
|
|
|
|
const height = Math.ceil(categorizedEmoji[category].length / 8) * 24;
|
|
|
|
categoryPositions[category].top = verticalPos;
|
|
|
|
verticalPos += height;
|
|
|
|
categoryPositions[category].bottom = verticalPos;
|
|
|
|
verticalPos += 24;
|
2016-03-25 01:42:44 +08:00
|
|
|
}
|
|
|
|
return {
|
|
|
|
categoryNames: categoryNames,
|
|
|
|
categorizedEmoji: categorizedEmoji,
|
|
|
|
categoryPositions: categoryPositions,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
scrollToCategory(category) {
|
2016-04-07 03:00:59 +08:00
|
|
|
const container = document.querySelector(".emoji-finder-container .scroll-region-content");
|
2016-03-25 01:42:44 +08:00
|
|
|
if (this.state.searchValue.length > 0) {
|
|
|
|
this.setState({searchValue: ""});
|
|
|
|
this.setState(this.getStateFromStore, () => {
|
|
|
|
this.renderCanvas();
|
|
|
|
container.scrollTop = this.state.categoryPositions[category].top + 16;
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
container.scrollTop = this.state.categoryPositions[category].top + 16;
|
|
|
|
}
|
|
|
|
this.setState({activeTab: category})
|
|
|
|
}
|
|
|
|
|
|
|
|
findSearchMatches(searchValue) {
|
2016-03-25 02:57:49 +08:00
|
|
|
// TODO: Find matches for aliases, too.
|
2016-03-25 01:42:44 +08:00
|
|
|
const searchMatches = [];
|
|
|
|
for (const category of Object.keys(categorizedEmojiList)) {
|
|
|
|
categorizedEmojiList[category].forEach((emojiName) => {
|
|
|
|
if (emojiName.indexOf(searchValue) !== -1) {
|
|
|
|
searchMatches.push(emojiName);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return searchMatches;
|
|
|
|
}
|
|
|
|
|
2016-04-01 05:26:21 +08:00
|
|
|
calcPosition(event) {
|
2016-03-25 01:42:44 +08:00
|
|
|
const rect = event.target.getBoundingClientRect();
|
|
|
|
const position = {
|
|
|
|
x: event.pageX - rect.left / 2,
|
|
|
|
y: event.pageY - rect.top / 2,
|
|
|
|
};
|
|
|
|
return position;
|
|
|
|
}
|
|
|
|
|
|
|
|
calcEmojiByPosition = (position) => {
|
2016-05-07 05:10:56 +08:00
|
|
|
for (const category of Object.keys(this.state.categoryPositions)) {
|
|
|
|
const LEFT_BOUNDARY = 8;
|
|
|
|
const RIGHT_BOUNDARY = 204;
|
|
|
|
const EMOJI_WIDTH = 24.5;
|
|
|
|
const EMOJI_HEIGHT = 24;
|
|
|
|
const EMOJI_PER_ROW = 8;
|
|
|
|
if (position.x >= LEFT_BOUNDARY &&
|
|
|
|
position.x <= RIGHT_BOUNDARY &&
|
|
|
|
position.y >= this.state.categoryPositions[category].top &&
|
|
|
|
position.y <= this.state.categoryPositions[category].bottom) {
|
|
|
|
const x = Math.round((position.x + 5) / EMOJI_WIDTH);
|
|
|
|
const y = Math.round((position.y - this.state.categoryPositions[category].top + 10) / EMOJI_HEIGHT);
|
|
|
|
const index = x + (y - 1) * EMOJI_PER_ROW - 1;
|
|
|
|
return this.state.categorizedEmoji[category][index];
|
2016-03-25 01:42:44 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
renderTabs() {
|
|
|
|
const tabs = [];
|
|
|
|
this.state.categoryNames.forEach((category) => {
|
2016-04-13 04:57:42 +08:00
|
|
|
let className = `emoji-tab ${(category.replace(/ /g, '-')).toLowerCase()}`
|
|
|
|
if (category === this.state.activeTab) {
|
|
|
|
className += " active";
|
2016-03-25 01:42:44 +08:00
|
|
|
}
|
2016-04-13 04:57:42 +08:00
|
|
|
tabs.push(
|
|
|
|
<div key={`${category} container`} style={{flex: 1}}>
|
|
|
|
<RetinaImg
|
|
|
|
key={`${category} tab`}
|
|
|
|
className={className}
|
|
|
|
name={`icon-emojipicker-${(category.replace(/ /g, '-')).toLowerCase()}.png`}
|
|
|
|
mode={RetinaImg.Mode.ContentIsMask}
|
2016-05-07 05:10:56 +08:00
|
|
|
onMouseDown={() => this.scrollToCategory(category)}
|
|
|
|
/>
|
2016-04-13 04:57:42 +08:00
|
|
|
</div>
|
|
|
|
);
|
2016-03-25 01:42:44 +08:00
|
|
|
});
|
|
|
|
return tabs;
|
|
|
|
}
|
|
|
|
|
|
|
|
renderCanvas() {
|
|
|
|
const keys = Object.keys(this.state.categoryPositions);
|
2017-07-24 07:19:39 +08:00
|
|
|
this._canvasEl.height = this.state.categoryPositions[keys[keys.length - 1]].bottom * 2;
|
|
|
|
const ctx = this._canvasEl.getContext("2d");
|
2016-04-13 04:57:42 +08:00
|
|
|
ctx.font = "24px Nylas-Pro";
|
|
|
|
ctx.fillStyle = 'rgba(0, 0, 0, 0.5)';
|
2017-07-24 07:19:39 +08:00
|
|
|
ctx.clearRect(0, 0, this._canvasEl.width, this._canvasEl.height);
|
2016-03-25 01:42:44 +08:00
|
|
|
const position = {
|
|
|
|
x: 15,
|
|
|
|
y: 45,
|
|
|
|
}
|
2016-04-13 04:57:42 +08:00
|
|
|
|
|
|
|
let idx = 0;
|
|
|
|
const categoryNames = Object.keys(this.state.categorizedEmoji);
|
|
|
|
const renderNextCategory = () => {
|
|
|
|
if (!categoryNames[idx]) return;
|
|
|
|
if (!this._mounted) return;
|
|
|
|
this.renderCategory(categoryNames[idx], idx, ctx, position, renderNextCategory);
|
|
|
|
idx += 1;
|
|
|
|
}
|
|
|
|
renderNextCategory();
|
2016-03-25 01:42:44 +08:00
|
|
|
}
|
|
|
|
|
2016-05-07 05:10:56 +08:00
|
|
|
renderCategory(category, i, ctx, pos, callback) {
|
|
|
|
const position = pos
|
2016-04-13 04:57:42 +08:00
|
|
|
if (i > 0) {
|
|
|
|
position.x = 18;
|
|
|
|
position.y += 48;
|
2016-03-25 01:42:44 +08:00
|
|
|
}
|
2016-04-13 04:57:42 +08:00
|
|
|
ctx.fillText(category, position.x, position.y);
|
2016-03-25 01:42:44 +08:00
|
|
|
position.x = 18;
|
|
|
|
position.y += 48;
|
2016-04-13 04:57:42 +08:00
|
|
|
|
2016-04-13 05:46:03 +08:00
|
|
|
const emojiNames = this.state.categorizedEmoji[category];
|
|
|
|
if (!emojiNames || emojiNames.length === 0) return;
|
2016-04-13 04:57:42 +08:00
|
|
|
|
2016-04-13 05:46:03 +08:00
|
|
|
const emojiToDraw = emojiNames.map((emojiName, j) => {
|
2016-04-13 01:27:24 +08:00
|
|
|
const x = position.x;
|
|
|
|
const y = position.y;
|
2016-04-13 04:57:42 +08:00
|
|
|
const src = EmojiStore.getImagePath(emojiName);
|
|
|
|
|
2016-03-25 01:42:44 +08:00
|
|
|
if (position.x > 325 && j < this.state.categorizedEmoji[category].length - 1) {
|
|
|
|
position.x = 18;
|
|
|
|
position.y += 48;
|
|
|
|
} else {
|
|
|
|
position.x += 50;
|
|
|
|
}
|
2016-04-13 04:57:42 +08:00
|
|
|
|
|
|
|
return {src, x, y};
|
|
|
|
});
|
|
|
|
|
|
|
|
const drawEmojiAt = ({src, x, y} = {}) => {
|
|
|
|
if (!src) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this._emojiPreloadImage.onload = () => {
|
|
|
|
this._emojiPreloadImage.onload = null;
|
|
|
|
ctx.drawImage(this._emojiPreloadImage, x, y - 30, 32, 32);
|
2016-04-13 05:46:03 +08:00
|
|
|
if (emojiToDraw.length === 0) {
|
2016-04-13 04:57:42 +08:00
|
|
|
callback();
|
|
|
|
} else {
|
2016-04-13 05:46:03 +08:00
|
|
|
drawEmojiAt(emojiToDraw.shift());
|
2016-04-13 04:57:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
this._emojiPreloadImage.src = src;
|
|
|
|
}
|
|
|
|
|
2016-04-13 05:46:03 +08:00
|
|
|
drawEmojiAt(emojiToDraw.shift());
|
2016-03-25 01:42:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
return (
|
|
|
|
<div className="emoji-button-popover" tabIndex="-1">
|
|
|
|
<div className="emoji-tabs">
|
|
|
|
{this.renderTabs()}
|
|
|
|
</div>
|
2016-03-31 06:52:51 +08:00
|
|
|
<ScrollRegion
|
2016-03-25 01:42:44 +08:00
|
|
|
className="emoji-finder-container"
|
2016-05-07 05:10:56 +08:00
|
|
|
onScroll={this.onScroll}
|
|
|
|
>
|
2016-03-25 01:42:44 +08:00
|
|
|
<div className="emoji-search-container">
|
|
|
|
<input
|
|
|
|
type="text"
|
|
|
|
className="search"
|
|
|
|
value={this.state.searchValue}
|
|
|
|
onChange={this.onChange}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<canvas
|
2017-07-24 07:19:39 +08:00
|
|
|
ref={(el) => this._canvasEl = el}
|
2016-03-31 06:52:51 +08:00
|
|
|
width="400"
|
2016-03-25 01:42:44 +08:00
|
|
|
height="2000"
|
|
|
|
onMouseDown={this.onMouseDown}
|
|
|
|
onMouseOut={this.onMouseOut}
|
|
|
|
onMouseMove={this.onHover}
|
2016-05-07 05:10:56 +08:00
|
|
|
style={{zoom: "0.5"}}
|
2016-10-18 08:59:33 +08:00
|
|
|
/>
|
2016-03-31 06:52:51 +08:00
|
|
|
</ScrollRegion>
|
2016-03-25 01:42:44 +08:00
|
|
|
<div className="emoji-name">
|
|
|
|
{this.state.emojiName}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export default EmojiButtonPopover;
|