Emoji provider, DDG working, style improvements
parent
769b3f0c2a
commit
b9d7743e5a
|
@ -27,6 +27,7 @@
|
||||||
"draft-js-export-html": "^0.2.2",
|
"draft-js-export-html": "^0.2.2",
|
||||||
"draft-js-export-markdown": "^0.2.0",
|
"draft-js-export-markdown": "^0.2.0",
|
||||||
"draft-js-import-markdown": "^0.1.6",
|
"draft-js-import-markdown": "^0.1.6",
|
||||||
|
"emojione": "^2.2.2",
|
||||||
"favico.js": "^0.3.10",
|
"favico.js": "^0.3.10",
|
||||||
"filesize": "^3.1.2",
|
"filesize": "^3.1.2",
|
||||||
"flux": "^2.0.3",
|
"flux": "^2.0.3",
|
||||||
|
@ -39,6 +40,7 @@
|
||||||
"optimist": "^0.6.1",
|
"optimist": "^0.6.1",
|
||||||
"q": "^1.4.1",
|
"q": "^1.4.1",
|
||||||
"react": "^15.0.1",
|
"react": "^15.0.1",
|
||||||
|
"react-addons-css-transition-group": "^15.1.0",
|
||||||
"react-dom": "^15.0.1",
|
"react-dom": "^15.0.1",
|
||||||
"react-gemini-scrollbar": "matrix-org/react-gemini-scrollbar#c3d942e",
|
"react-gemini-scrollbar": "matrix-org/react-gemini-scrollbar#c3d942e",
|
||||||
"sanitize-html": "^1.11.1",
|
"sanitize-html": "^1.11.1",
|
||||||
|
|
|
@ -2,12 +2,14 @@ import CommandProvider from './CommandProvider';
|
||||||
import DuckDuckGoProvider from './DuckDuckGoProvider';
|
import DuckDuckGoProvider from './DuckDuckGoProvider';
|
||||||
import RoomProvider from './RoomProvider';
|
import RoomProvider from './RoomProvider';
|
||||||
import UserProvider from './UserProvider';
|
import UserProvider from './UserProvider';
|
||||||
|
import EmojiProvider from './EmojiProvider';
|
||||||
|
|
||||||
const PROVIDERS = [
|
const PROVIDERS = [
|
||||||
CommandProvider,
|
CommandProvider,
|
||||||
DuckDuckGoProvider,
|
DuckDuckGoProvider,
|
||||||
RoomProvider,
|
RoomProvider,
|
||||||
UserProvider
|
UserProvider,
|
||||||
|
EmojiProvider
|
||||||
].map(completer => new completer());
|
].map(completer => new completer());
|
||||||
|
|
||||||
export function getCompletions(query: String) {
|
export function getCompletions(query: String) {
|
||||||
|
|
|
@ -0,0 +1,13 @@
|
||||||
|
export function TextualCompletion(props: {
|
||||||
|
title: ?string,
|
||||||
|
subtitle: ?string,
|
||||||
|
description: ?string
|
||||||
|
}) {
|
||||||
|
return (
|
||||||
|
<div className="mx_Autocomplete_Completion">
|
||||||
|
<span>{completion.title}</span>
|
||||||
|
<em>{completion.subtitle}</em>
|
||||||
|
<span style={{color: 'gray', float: 'right'}}>{completion.description}</span>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
|
@ -2,31 +2,50 @@ import AutocompleteProvider from './AutocompleteProvider';
|
||||||
import Q from 'q';
|
import Q from 'q';
|
||||||
import 'whatwg-fetch';
|
import 'whatwg-fetch';
|
||||||
|
|
||||||
const DDG_REGEX = /\/ddg\w+(.+)$/;
|
const DDG_REGEX = /\/ddg\s+(.+)$/;
|
||||||
const REFERER = 'vector';
|
const REFERER = 'vector';
|
||||||
|
|
||||||
export default class DuckDuckGoProvider extends AutocompleteProvider {
|
export default class DuckDuckGoProvider extends AutocompleteProvider {
|
||||||
static getQueryUri(query: String) {
|
static getQueryUri(query: String) {
|
||||||
return `http://api.duckduckgo.com/?q=${encodeURIComponent(query)}&format=json&t=${encodeURIComponent(REFERER)}`;
|
return `http://api.duckduckgo.com/?q=${encodeURIComponent(query)}`
|
||||||
|
+ `&format=json&no_redirect=1&no_html=1&t=${encodeURIComponent(REFERER)}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
getCompletions(query: String) {
|
getCompletions(query: String) {
|
||||||
if(!query)
|
let match = DDG_REGEX.exec(query);
|
||||||
|
if(!query || !match)
|
||||||
return Q.when([]);
|
return Q.when([]);
|
||||||
|
|
||||||
let promise = Q.defer();
|
return fetch(DuckDuckGoProvider.getQueryUri(match[1]), {
|
||||||
fetch(DuckDuckGoProvider.getQueryUri(query), {
|
|
||||||
method: 'GET'
|
method: 'GET'
|
||||||
}).then(response => {
|
})
|
||||||
let results = response.Results.map(result => {
|
.then(response => response.json())
|
||||||
|
.then(json => {
|
||||||
|
let results = json.Results.map(result => {
|
||||||
return {
|
return {
|
||||||
title: result.Text,
|
title: result.Text,
|
||||||
description: result.Result
|
description: result.Result
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
promise.resolve(results);
|
if(json.Answer) {
|
||||||
|
results.unshift({
|
||||||
|
title: json.Answer,
|
||||||
|
description: json.AnswerType
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if(json.RelatedTopics && json.RelatedTopics.length > 0) {
|
||||||
|
results.unshift({
|
||||||
|
title: json.RelatedTopics[0].Text
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if(json.AbstractText) {
|
||||||
|
results.unshift({
|
||||||
|
title: json.AbstractText
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// console.log(results);
|
||||||
|
return results;
|
||||||
});
|
});
|
||||||
return promise;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
getName() {
|
getName() {
|
||||||
|
|
|
@ -0,0 +1,41 @@
|
||||||
|
import AutocompleteProvider from './AutocompleteProvider';
|
||||||
|
import Q from 'q';
|
||||||
|
import {emojioneList, shortnameToImage} from 'emojione';
|
||||||
|
import Fuse from 'fuse.js';
|
||||||
|
|
||||||
|
const EMOJI_REGEX = /:\w*:?/g;
|
||||||
|
const EMOJI_SHORTNAMES = Object.keys(emojioneList);
|
||||||
|
|
||||||
|
export default class EmojiProvider extends AutocompleteProvider {
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
console.log(EMOJI_SHORTNAMES);
|
||||||
|
this.fuse = new Fuse(EMOJI_SHORTNAMES);
|
||||||
|
}
|
||||||
|
|
||||||
|
getCompletions(query: String) {
|
||||||
|
let completions = [];
|
||||||
|
const matches = query.match(EMOJI_REGEX);
|
||||||
|
console.log(matches);
|
||||||
|
if(!!matches) {
|
||||||
|
const command = matches[0];
|
||||||
|
completions = this.fuse.search(command).map(result => {
|
||||||
|
let shortname = EMOJI_SHORTNAMES[result];
|
||||||
|
let imageHTML = shortnameToImage(shortname);
|
||||||
|
return {
|
||||||
|
title: shortname,
|
||||||
|
component: (
|
||||||
|
<div className="mx_Autocomplete_Completion">
|
||||||
|
<span dangerouslySetInnerHTML={{__html: imageHTML}}></span> {shortname}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
};
|
||||||
|
}).slice(0, 4);
|
||||||
|
}
|
||||||
|
return Q.when(completions);
|
||||||
|
}
|
||||||
|
|
||||||
|
getName() {
|
||||||
|
return 'Emoji';
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,4 +1,5 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
|
import ReactCSSTransitionGroup from 'react-addons-css-transition-group';
|
||||||
|
|
||||||
import {getCompletions} from '../../../autocomplete/Autocompleter';
|
import {getCompletions} from '../../../autocomplete/Autocompleter';
|
||||||
|
|
||||||
|
@ -11,8 +12,11 @@ export default class Autocomplete extends React.Component {
|
||||||
}
|
}
|
||||||
|
|
||||||
componentWillReceiveProps(props, state) {
|
componentWillReceiveProps(props, state) {
|
||||||
|
if(props.query == this.props.query) return;
|
||||||
|
|
||||||
getCompletions(props.query).map(completionResult => {
|
getCompletions(props.query).map(completionResult => {
|
||||||
try {
|
try {
|
||||||
|
console.log(`${completionResult.provider.getName()}: ${JSON.stringify(completionResult.completions)}`);
|
||||||
completionResult.completions.then(completions => {
|
completionResult.completions.then(completions => {
|
||||||
let i = this.state.completions.findIndex(
|
let i = this.state.completions.findIndex(
|
||||||
completion => completion.provider === completionResult.provider
|
completion => completion.provider === completionResult.provider
|
||||||
|
@ -23,15 +27,16 @@ export default class Autocomplete extends React.Component {
|
||||||
let newCompletions = Object.assign([], this.state.completions);
|
let newCompletions = Object.assign([], this.state.completions);
|
||||||
completionResult.completions = completions;
|
completionResult.completions = completions;
|
||||||
newCompletions[i] = completionResult;
|
newCompletions[i] = completionResult;
|
||||||
console.log(newCompletions);
|
// console.log(newCompletions);
|
||||||
this.setState({
|
this.setState({
|
||||||
completions: newCompletions
|
completions: newCompletions
|
||||||
});
|
});
|
||||||
}, err => {
|
}, err => {
|
||||||
|
console.error(err);
|
||||||
});
|
});
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
// An error in one provider shouldn't mess up the rest.
|
// An error in one provider shouldn't mess up the rest.
|
||||||
|
console.error(e);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -42,23 +47,19 @@ export default class Autocomplete extends React.Component {
|
||||||
|
|
||||||
const position = pinElement.getBoundingClientRect();
|
const position = pinElement.getBoundingClientRect();
|
||||||
|
|
||||||
const style = {
|
|
||||||
position: 'fixed',
|
|
||||||
border: '1px solid gray',
|
|
||||||
background: 'white',
|
|
||||||
borderRadius: '4px'
|
|
||||||
};
|
|
||||||
|
|
||||||
this.props.pinTo.forEach(direction => {
|
|
||||||
style[direction] = position[direction];
|
|
||||||
});
|
|
||||||
|
|
||||||
const renderedCompletions = this.state.completions.map((completionResult, i) => {
|
const renderedCompletions = this.state.completions.map((completionResult, i) => {
|
||||||
console.log(completionResult);
|
// console.log(completionResult);
|
||||||
let completions = completionResult.completions.map((completion, i) => {
|
let completions = completionResult.completions.map((completion, i) => {
|
||||||
|
let Component = completion.component;
|
||||||
|
if(Component) {
|
||||||
|
return Component;
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div key={i} class="mx_Autocomplete_Completion">
|
<div key={i} className="mx_Autocomplete_Completion">
|
||||||
<strong>{completion.title}</strong>
|
<span>{completion.title}</span>
|
||||||
<em>{completion.subtitle}</em>
|
<em>{completion.subtitle}</em>
|
||||||
<span style={{color: 'gray', float: 'right'}}>{completion.description}</span>
|
<span style={{color: 'gray', float: 'right'}}>{completion.description}</span>
|
||||||
</div>
|
</div>
|
||||||
|
@ -67,16 +68,20 @@ export default class Autocomplete extends React.Component {
|
||||||
|
|
||||||
|
|
||||||
return completions.length > 0 ? (
|
return completions.length > 0 ? (
|
||||||
<div key={i} class="mx_Autocomplete_ProviderSection">
|
<div key={i} className="mx_Autocomplete_ProviderSection">
|
||||||
<strong>{completionResult.provider.getName()}</strong>
|
<span className="mx_Autocomplete_provider_name">{completionResult.provider.getName()}</span>
|
||||||
|
<ReactCSSTransitionGroup transitionName="autocomplete" transitionEnterTimeout={300} transitionLeaveTimeout={300}>
|
||||||
{completions}
|
{completions}
|
||||||
|
</ReactCSSTransitionGroup>
|
||||||
</div>
|
</div>
|
||||||
) : null;
|
) : null;
|
||||||
});
|
});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="mx_Autocomplete" style={style}>
|
<div className="mx_Autocomplete">
|
||||||
|
<ReactCSSTransitionGroup transitionName="autocomplete" transitionEnterTimeout={300} transitionLeaveTimeout={300}>
|
||||||
{renderedCompletions}
|
{renderedCompletions}
|
||||||
|
</ReactCSSTransitionGroup>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -212,13 +212,14 @@ module.exports = React.createClass({
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="mx_MessageComposer mx_fadable" style={{ opacity: this.props.opacity }}>
|
<div className="mx_MessageComposer mx_fadable" style={{ opacity: this.props.opacity }}>
|
||||||
|
<div className="mx_MessageComposer_autocomplete_wrapper">
|
||||||
|
<Autocomplete query={this.state.autocompleteQuery} pinSelector=".mx_RoomView_statusArea" pinTo={['top', 'left', 'width']} />
|
||||||
|
</div>
|
||||||
<div className="mx_MessageComposer_wrapper">
|
<div className="mx_MessageComposer_wrapper">
|
||||||
<div className="mx_MessageComposer_row">
|
<div className="mx_MessageComposer_row">
|
||||||
{controls}
|
{controls}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Autocomplete query={this.state.autocompleteQuery} pinSelector=".mx_RoomView_statusArea" pinTo={['top', 'left', 'width']} />
|
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -352,6 +352,10 @@ export default class MessageComposerInput extends React.Component {
|
||||||
} else {
|
} else {
|
||||||
this.onFinishedTyping();
|
this.onFinishedTyping();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(this.props.onContentChanged) {
|
||||||
|
this.props.onContentChanged(editorState.getCurrentContent().getPlainText());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
enableRichtext(enabled: boolean) {
|
enableRichtext(enabled: boolean) {
|
||||||
|
@ -521,5 +525,8 @@ MessageComposerInput.propTypes = {
|
||||||
onResize: React.PropTypes.func,
|
onResize: React.PropTypes.func,
|
||||||
|
|
||||||
// js-sdk Room object
|
// js-sdk Room object
|
||||||
room: React.PropTypes.object.isRequired
|
room: React.PropTypes.object.isRequired,
|
||||||
|
|
||||||
|
// called with current plaintext content (as a string) whenever it changes
|
||||||
|
onContentChanged: React.PropTypes.func
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in New Issue