Skip to content
This repository was archived by the owner on Sep 11, 2024. It is now read-only.

Commit b9d7743

Browse files
committed
Emoji provider, DDG working, style improvements
1 parent 769b3f0 commit b9d7743

File tree

8 files changed

+127
-37
lines changed

8 files changed

+127
-37
lines changed

package.json

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@
2727
"draft-js-export-html": "^0.2.2",
2828
"draft-js-export-markdown": "^0.2.0",
2929
"draft-js-import-markdown": "^0.1.6",
30+
"emojione": "^2.2.2",
3031
"favico.js": "^0.3.10",
3132
"filesize": "^3.1.2",
3233
"flux": "^2.0.3",
@@ -39,6 +40,7 @@
3940
"optimist": "^0.6.1",
4041
"q": "^1.4.1",
4142
"react": "^15.0.1",
43+
"react-addons-css-transition-group": "^15.1.0",
4244
"react-dom": "^15.0.1",
4345
"react-gemini-scrollbar": "matrix-org/react-gemini-scrollbar#c3d942e",
4446
"sanitize-html": "^1.11.1",

src/autocomplete/Autocompleter.js

Lines changed: 3 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -2,12 +2,14 @@ import CommandProvider from './CommandProvider';
22
import DuckDuckGoProvider from './DuckDuckGoProvider';
33
import RoomProvider from './RoomProvider';
44
import UserProvider from './UserProvider';
5+
import EmojiProvider from './EmojiProvider';
56

67
const PROVIDERS = [
78
CommandProvider,
89
DuckDuckGoProvider,
910
RoomProvider,
10-
UserProvider
11+
UserProvider,
12+
EmojiProvider
1113
].map(completer => new completer());
1214

1315
export function getCompletions(query: String) {

src/autocomplete/Components.js

Lines changed: 13 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
export function TextualCompletion(props: {
2+
title: ?string,
3+
subtitle: ?string,
4+
description: ?string
5+
}) {
6+
return (
7+
<div className="mx_Autocomplete_Completion">
8+
<span>{completion.title}</span>
9+
<em>{completion.subtitle}</em>
10+
<span style={{color: 'gray', float: 'right'}}>{completion.description}</span>
11+
</div>
12+
);
13+
}

src/autocomplete/DuckDuckGoProvider.js

Lines changed: 33 additions & 14 deletions
Original file line numberDiff line numberDiff line change
@@ -2,31 +2,50 @@ import AutocompleteProvider from './AutocompleteProvider';
22
import Q from 'q';
33
import 'whatwg-fetch';
44

5-
const DDG_REGEX = /\/ddg\w+(.+)$/;
5+
const DDG_REGEX = /\/ddg\s+(.+)$/;
66
const REFERER = 'vector';
77

88
export default class DuckDuckGoProvider extends AutocompleteProvider {
99
static getQueryUri(query: String) {
10-
return `http://api.duckduckgo.com/?q=${encodeURIComponent(query)}&format=json&t=${encodeURIComponent(REFERER)}`;
10+
return `http://api.duckduckgo.com/?q=${encodeURIComponent(query)}`
11+
+ `&format=json&no_redirect=1&no_html=1&t=${encodeURIComponent(REFERER)}`;
1112
}
1213

1314
getCompletions(query: String) {
14-
if(!query)
15+
let match = DDG_REGEX.exec(query);
16+
if(!query || !match)
1517
return Q.when([]);
1618

17-
let promise = Q.defer();
18-
fetch(DuckDuckGoProvider.getQueryUri(query), {
19+
return fetch(DuckDuckGoProvider.getQueryUri(match[1]), {
1920
method: 'GET'
20-
}).then(response => {
21-
let results = response.Results.map(result => {
22-
return {
23-
title: result.Text,
24-
description: result.Result
25-
};
21+
})
22+
.then(response => response.json())
23+
.then(json => {
24+
let results = json.Results.map(result => {
25+
return {
26+
title: result.Text,
27+
description: result.Result
28+
};
29+
});
30+
if(json.Answer) {
31+
results.unshift({
32+
title: json.Answer,
33+
description: json.AnswerType
34+
});
35+
}
36+
if(json.RelatedTopics && json.RelatedTopics.length > 0) {
37+
results.unshift({
38+
title: json.RelatedTopics[0].Text
39+
});
40+
}
41+
if(json.AbstractText) {
42+
results.unshift({
43+
title: json.AbstractText
44+
});
45+
}
46+
// console.log(results);
47+
return results;
2648
});
27-
promise.resolve(results);
28-
});
29-
return promise;
3049
}
3150

3251
getName() {

src/autocomplete/EmojiProvider.js

Lines changed: 41 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
import AutocompleteProvider from './AutocompleteProvider';
2+
import Q from 'q';
3+
import {emojioneList, shortnameToImage} from 'emojione';
4+
import Fuse from 'fuse.js';
5+
6+
const EMOJI_REGEX = /:\w*:?/g;
7+
const EMOJI_SHORTNAMES = Object.keys(emojioneList);
8+
9+
export default class EmojiProvider extends AutocompleteProvider {
10+
constructor() {
11+
super();
12+
console.log(EMOJI_SHORTNAMES);
13+
this.fuse = new Fuse(EMOJI_SHORTNAMES);
14+
}
15+
16+
getCompletions(query: String) {
17+
let completions = [];
18+
const matches = query.match(EMOJI_REGEX);
19+
console.log(matches);
20+
if(!!matches) {
21+
const command = matches[0];
22+
completions = this.fuse.search(command).map(result => {
23+
let shortname = EMOJI_SHORTNAMES[result];
24+
let imageHTML = shortnameToImage(shortname);
25+
return {
26+
title: shortname,
27+
component: (
28+
<div className="mx_Autocomplete_Completion">
29+
<span dangerouslySetInnerHTML={{__html: imageHTML}}></span> {shortname}
30+
</div>
31+
)
32+
};
33+
}).slice(0, 4);
34+
}
35+
return Q.when(completions);
36+
}
37+
38+
getName() {
39+
return 'Emoji';
40+
}
41+
}

src/components/views/rooms/Autocomplete.js

Lines changed: 24 additions & 19 deletions
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
import React from 'react';
2+
import ReactCSSTransitionGroup from 'react-addons-css-transition-group';
23

34
import {getCompletions} from '../../../autocomplete/Autocompleter';
45

@@ -11,8 +12,11 @@ export default class Autocomplete extends React.Component {
1112
}
1213

1314
componentWillReceiveProps(props, state) {
15+
if(props.query == this.props.query) return;
16+
1417
getCompletions(props.query).map(completionResult => {
1518
try {
19+
console.log(`${completionResult.provider.getName()}: ${JSON.stringify(completionResult.completions)}`);
1620
completionResult.completions.then(completions => {
1721
let i = this.state.completions.findIndex(
1822
completion => completion.provider === completionResult.provider
@@ -23,15 +27,16 @@ export default class Autocomplete extends React.Component {
2327
let newCompletions = Object.assign([], this.state.completions);
2428
completionResult.completions = completions;
2529
newCompletions[i] = completionResult;
26-
console.log(newCompletions);
30+
// console.log(newCompletions);
2731
this.setState({
2832
completions: newCompletions
2933
});
3034
}, err => {
31-
35+
console.error(err);
3236
});
3337
} catch (e) {
3438
// An error in one provider shouldn't mess up the rest.
39+
console.error(e);
3540
}
3641
});
3742
}
@@ -42,23 +47,19 @@ export default class Autocomplete extends React.Component {
4247

4348
const position = pinElement.getBoundingClientRect();
4449

45-
const style = {
46-
position: 'fixed',
47-
border: '1px solid gray',
48-
background: 'white',
49-
borderRadius: '4px'
50-
};
5150

52-
this.props.pinTo.forEach(direction => {
53-
style[direction] = position[direction];
54-
});
5551

5652
const renderedCompletions = this.state.completions.map((completionResult, i) => {
57-
console.log(completionResult);
53+
// console.log(completionResult);
5854
let completions = completionResult.completions.map((completion, i) => {
55+
let Component = completion.component;
56+
if(Component) {
57+
return Component;
58+
}
59+
5960
return (
60-
<div key={i} class="mx_Autocomplete_Completion">
61-
<strong>{completion.title}</strong>
61+
<div key={i} className="mx_Autocomplete_Completion">
62+
<span>{completion.title}</span>
6263
<em>{completion.subtitle}</em>
6364
<span style={{color: 'gray', float: 'right'}}>{completion.description}</span>
6465
</div>
@@ -67,16 +68,20 @@ export default class Autocomplete extends React.Component {
6768

6869

6970
return completions.length > 0 ? (
70-
<div key={i} class="mx_Autocomplete_ProviderSection">
71-
<strong>{completionResult.provider.getName()}</strong>
72-
{completions}
71+
<div key={i} className="mx_Autocomplete_ProviderSection">
72+
<span className="mx_Autocomplete_provider_name">{completionResult.provider.getName()}</span>
73+
<ReactCSSTransitionGroup transitionName="autocomplete" transitionEnterTimeout={300} transitionLeaveTimeout={300}>
74+
{completions}
75+
</ReactCSSTransitionGroup>
7376
</div>
7477
) : null;
7578
});
7679

7780
return (
78-
<div className="mx_Autocomplete" style={style}>
79-
{renderedCompletions}
81+
<div className="mx_Autocomplete">
82+
<ReactCSSTransitionGroup transitionName="autocomplete" transitionEnterTimeout={300} transitionLeaveTimeout={300}>
83+
{renderedCompletions}
84+
</ReactCSSTransitionGroup>
8085
</div>
8186
);
8287
}

src/components/views/rooms/MessageComposer.js

Lines changed: 3 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -212,13 +212,14 @@ module.exports = React.createClass({
212212

213213
return (
214214
<div className="mx_MessageComposer mx_fadable" style={{ opacity: this.props.opacity }}>
215+
<div className="mx_MessageComposer_autocomplete_wrapper">
216+
<Autocomplete query={this.state.autocompleteQuery} pinSelector=".mx_RoomView_statusArea" pinTo={['top', 'left', 'width']} />
217+
</div>
215218
<div className="mx_MessageComposer_wrapper">
216219
<div className="mx_MessageComposer_row">
217220
{controls}
218221
</div>
219222
</div>
220-
221-
<Autocomplete query={this.state.autocompleteQuery} pinSelector=".mx_RoomView_statusArea" pinTo={['top', 'left', 'width']} />
222223
</div>
223224
);
224225
}

src/components/views/rooms/MessageComposerInput.js

Lines changed: 8 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -352,6 +352,10 @@ export default class MessageComposerInput extends React.Component {
352352
} else {
353353
this.onFinishedTyping();
354354
}
355+
356+
if(this.props.onContentChanged) {
357+
this.props.onContentChanged(editorState.getCurrentContent().getPlainText());
358+
}
355359
}
356360

357361
enableRichtext(enabled: boolean) {
@@ -521,5 +525,8 @@ MessageComposerInput.propTypes = {
521525
onResize: React.PropTypes.func,
522526

523527
// js-sdk Room object
524-
room: React.PropTypes.object.isRequired
528+
room: React.PropTypes.object.isRequired,
529+
530+
// called with current plaintext content (as a string) whenever it changes
531+
onContentChanged: React.PropTypes.func
525532
};

0 commit comments

Comments
 (0)