Emoji provider, DDG working, style improvements

pull/21833/head
Aviral Dasgupta 2016-06-17 04:58:09 +05:30
parent 769b3f0c2a
commit b9d7743e5a
8 changed files with 127 additions and 37 deletions

View File

@ -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",

View File

@ -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) {

View File

@ -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>
);
}

View File

@ -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() {

View File

@ -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';
}
}

View File

@ -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>
); );
} }

View File

@ -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>
); );
} }

View File

@ -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
}; };