2016-03-01 10:47:22 +08:00
|
|
|
import _ from 'underscore';
|
|
|
|
import NylasStore from 'nylas-store';
|
2017-09-27 02:33:08 +08:00
|
|
|
import { MessageStore } from 'nylas-exports';
|
2016-03-01 10:47:22 +08:00
|
|
|
|
|
|
|
class GithubStore extends NylasStore {
|
|
|
|
// It's very common practive for {NylasStore}s to listen to other parts of N1.
|
|
|
|
// Since Stores are singletons and constructed once on `require`, there is no
|
|
|
|
// teardown step to turn off listeners.
|
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
this.listenTo(MessageStore, this._onMessageStoreChanged);
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is the only public method on `GithubStore` and it's read only.
|
|
|
|
// All {NylasStore}s ONLY have reader methods. No setter methods. Use an
|
|
|
|
// `Action` instead!
|
|
|
|
//
|
|
|
|
// This is the computed & cached value that our `ViewOnGithubButton` will
|
|
|
|
// render.
|
|
|
|
link() {
|
2016-03-11 06:31:50 +08:00
|
|
|
return this._link;
|
2016-03-01 10:47:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Private methods
|
|
|
|
|
|
|
|
_onMessageStoreChanged() {
|
|
|
|
if (!MessageStore.threadId()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-09-27 02:33:08 +08:00
|
|
|
const itemIds = _.pluck(MessageStore.items(), 'id');
|
|
|
|
if (itemIds.length === 0 || _.isEqual(itemIds, this._lastItemIds)) {
|
2016-03-01 10:47:22 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this._lastItemIds = itemIds;
|
|
|
|
this._link = this._isRelevantThread() ? this._findGitHubLink() : null;
|
|
|
|
this.trigger();
|
|
|
|
}
|
|
|
|
|
|
|
|
_findGitHubLink() {
|
|
|
|
let msg = MessageStore.items()[0];
|
|
|
|
if (!msg.body) {
|
|
|
|
// The msg body may be null if it's collapsed. In that case, use the
|
|
|
|
// last message. This may be less relaiable since the last message
|
|
|
|
// might be a side-thread that doesn't contain the link in the quoted
|
|
|
|
// text.
|
|
|
|
msg = _.last(MessageStore.items());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Use a regex to parse the message body for GitHub URLs - this is a quick
|
|
|
|
// and dirty method to determine the GitHub object the email is about:
|
|
|
|
// https://regex101.com/r/aW8bI4/2
|
2017-09-27 02:33:08 +08:00
|
|
|
const re = /<a.*?href=['"](.*?)['"].*?view.*?it.*?on.*?github.*?\/a>/gim;
|
2016-03-01 10:47:22 +08:00
|
|
|
const firstMatch = re.exec(msg.body);
|
|
|
|
if (firstMatch) {
|
|
|
|
// [0] is the full match and [1] is the matching group
|
|
|
|
return firstMatch[1];
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
_isRelevantThread() {
|
|
|
|
const participants = MessageStore.thread().participants || [];
|
|
|
|
const githubDomainRegex = /@github\.com/gi;
|
2017-07-08 07:35:52 +08:00
|
|
|
return participants.some(contact => githubDomainRegex.test(contact.email));
|
2016-03-01 10:47:22 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
IMPORTANT NOTE:
|
|
|
|
|
|
|
|
All {NylasStore}s are constructed upon their first `require` by another
|
|
|
|
module. Since `require` is cached, they are only constructed once and
|
|
|
|
are therefore singletons.
|
|
|
|
*/
|
|
|
|
export default new GithubStore();
|