-
Notifications
You must be signed in to change notification settings - Fork 13
/
youpin.js
369 lines (311 loc) · 12.9 KB
/
youpin.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
const await = require('asyncawait/await');
const async = require('asyncawait/async');
const Promise = require('bluebird');
const _ = require('lodash');
var config = require("config");
var i18n = require('i18n');
i18n.configure( _.merge({}, config.get("i18n")) );
module.exports = (m, api, conversation, apiUserId) => {
'use strict';
const PAYLOAD_NEW_PIN = 'new_pin';
const PAYLOAD_CONTACT_US = 'contact_us';
const PAYLOAD_ENGLISH = 'english';
const PAYLOAD_THAI = 'thai';
const PIN_TYPE_BIKE = 'pin_type_bike';
const PIN_TYPE_BOAT = 'pin_type_boat';
const PIN_TYPE_TREE = 'pin_type_tree';
const PIN_TYPE_OTHERS = 'pin_type_others';
const STATE_WAIT_INTENT = 'wait_intent';
const STATE_DISABLED = 'disabled';
const STATE_WAIT_IMG = 'wait_image';
const STATE_WAIT_LOCATION = 'wait_pin';
const STATE_WAIT_DESC = 'wait_desc';
const STATE_WAIT_TAGS = 'wait_tags';
const categories = [
'footpath',
'pollution',
'roads',
'publictransport',
'garbage',
'drainage',
'trees',
'safety',
'violation'
];
function tagReplies(context){
let tags = [ m.createQuickReplyButton( context.__('#done'), 'isEnding' ) ];
let categoryTags = _.map( categories, cat => {
return m.createQuickReplyButton( `#${context.__(cat)}`, cat );
});
return tags.concat(categoryTags);
}
function greet(userid, context) {
let buttons = [
m.createPostbackButton(context.__('Report an issue'), PAYLOAD_NEW_PIN),
m.createPostbackButton(context.__('Contact us'), PAYLOAD_CONTACT_US)
];
if(context.language == 'en') {
buttons.push(m.createPostbackButton(context.__('Please say in Thai'), PAYLOAD_THAI));
}else{
buttons.push(m.createPostbackButton(context.__('Please say in Thai'), PAYLOAD_ENGLISH));
}
m.sendButton(
userid,
context.__('Hi {{name}}! What would you like to do today?', { name: context.profile.first_name }),
buttons
);
}
function addPhotos(attachments, context) {
attachments = _.filter( attachments, (item) => {
return _.includes( ['image','video'], item.type );
});
return Promise.map( attachments, (item) => {
return new Promise( (resolve, reject) => {
if (item.type === 'image') {
api.uploadPhotoFromURL( item.payload.url, (res) => {
item.payload.url = res.url;
resolve(item);
});
} else {
// TODO: upload video to firebase
resolve(item);
}
});
}).each( (item) => {
let type = "videos";
if (item.type === 'image') {
type = "photos";
}
context[type].push(item.payload.url);
});
}
function processText(messageText, context) {
// Sanitize string
messageText = messageText.trim().replace( /[\s\n\r]+/g, ' ');
let isEnding = false;
let endPos = -1;
endPos = messageText.indexOf(context.__('#done'));
if (endPos >= 0) {
isEnding = true;
messageText = messageText.substr(0, endPos);
}
if (messageText.length > 0) {
if (context.desc) {
context.desc.push(messageText);
context.descLength += messageText.length;
} else {
context.desc = [messageText];
context.descLength = messageText.length;
}
let hashtags = [];
// Hacky solution -- regex gets too complicated with unicode characters.
// https://github.com/twitter/twitter-text/blob/master/js/twitter-text.js
const tokens = messageText.split(' ');
tokens.forEach(str => {
if (str[0] == '#' || str[0] == '#') {
hashtags.push(str.substr(1));
}
});
if (hashtags.length > 0) {
context.hashtags.push.apply(context.hashtags, hashtags);
}
}
return isEnding;
}
return {
onMessaged: async (function(event) {
const userid = event.sender.id;
const timestamp = event.timestamp;
console.log(event.message);
const message = event.message;
let messageText = message ? message.text : undefined;
const isSticker = message ? !!message.sticker_id : false;
const attachments = message ? message.attachments : undefined;
console.log(event.postback);
const postback = event.postback ? event.postback.payload : undefined;
// eslint-disable-next-line
let context = await (conversation.getContext(userid));
console.log("---- Loaded previous context" ) ;
console.log(context);
// Override context
console.log('----- POSTBACK');
if (messageText === '#เริ่มใหม่' || postback === PAYLOAD_THAI) {
context = { url: "/?lang=th" };
} else if (postback === PAYLOAD_ENGLISH) {
context = { url: "/?lang=en" };
}
i18n.init(context);
context.lastReceived = timestamp;
if (context.state === STATE_DISABLED) {
return;
} else if (!context.state) {
// New session
context.firstReceived = timestamp;
let profile = await (new Promise( (resolve,reject) => {
m.getProfile(userid, resolve);
}));
context.profile = profile;
context.lastSent = (new Date()).getTime();
context.state = STATE_WAIT_INTENT;
greet( userid, context );
} else if (context.state === STATE_WAIT_INTENT) {
if (postback === PAYLOAD_NEW_PIN) {
context.lastSent = (new Date()).getTime();
m.sendText(userid, context.__("Awesome, let's get started!") );
await (new Promise( (resolve, reject) => {
setTimeout(() => {
context.lastSent = (new Date()).getTime();
context.state = STATE_WAIT_IMG;
context.photos = [];
context.videos = [];
m.sendText(userid, context.__("First, can you send me photos or videos of the issue you found?"));
resolve();
}, 1000);
}));
} else if (postback === PAYLOAD_CONTACT_US) {
context.lastSent = (new Date()).getTime();
context.state = STATE_DISABLED;
m.sendText(userid, context.__("You can leave us messages, and our staff will get back to you as soon as possible."));
} else {
m.sendText(userid, context.__("Slow down, could you please answer my question first?") );
}
} else if (context.state === STATE_WAIT_IMG) {
if (attachments) {
if (!isSticker && (attachments[0].type == 'image' || attachments[0].type == 'video')) {
context.lastSent = (new Date()).getTime();
m.sendText(userid, context.__('(Y) Sweet!') );
await (addPhotos(attachments, context));
await (new Promise( (resolve,reject) => {
setTimeout(() => {
context.lastSent = (new Date()).getTime();
context.state = STATE_WAIT_LOCATION;
m.sendText( userid, context.__("Next, can you help us locate the issue by sharing the location using Facebook Messenger App on your mobile phone?") );
resolve();
}, 1000);
}));
} else {
m.sendText( userid, context.__("Just photos or videos please. I'm getting confused! 😓") );
}
} else {
m.sendText(userid, context.__("Hurry up, I'm still waiting for photos or videos.") );
}
} else if (context.state === STATE_WAIT_LOCATION) {
if (attachments && attachments[0].type == 'location') {
context.lastSent = (new Date()).getTime();
m.sendText(userid, context.__("🚩 Ahh, got it.") );
const point = attachments[0].payload.coordinates;
context.location = [point.lat, point.long];
await (new Promise( (resolve,reject) => {
setTimeout(() => {
context.lastSent = (new Date()).getTime();
context.state = STATE_WAIT_DESC;
context.hashtags = [];
m.sendText(userid, context.__("Alright, can you explain the issue you'd like to report today? Please make it as detailed as possible."));
resolve();
}, 1000);
}));
} else if (!isSticker && attachments && attachments.length > 0 && (attachments[0].type == 'image' || attachments[0].type == 'video')) {
// Add photos/videos
context.lastSent = (new Date()).getTime();
m.sendText(userid, context.__("(Y) Cool! Don't forget to send me the location.") );
await (addPhotos(attachments, context));
} else {
m.sendText(userid, context.__("Let us know the location so that the responsible agency can take care of the problem quickly.") );
}
} else if (context.state === STATE_WAIT_DESC) {
if (messageText) {
const isEnding = processText(messageText, context);
if (isEnding) {
if (context.descLength < 10) {
context.lastSent = (new Date()).getTime();
m.sendText( userid, context.__("Provide us a little more detail please.") );
} else {
context.state = STATE_WAIT_TAGS;
context.categories = [];
m.sendTextWithReplies(userid, context.__("Could you please help me select appropriate categories for the issue? You can pick one from the list below or type #<category> for a custom category."),
tagReplies(context).slice(1)
);
}
} else {
if (context.desc.length == 1) {
// After 1st response
context.lastSent = (new Date()).getTime();
m.sendTextWithReplies(
userid,
context.__("You can keep on typing! Send '#done' when you finish so that we can proceed to the next step."),
_.take(tagReplies(context),1)
);
} else if (context.descLength > 140) {
context.lastSent = (new Date()).getTime();
m.sendTextWithReplies(
userid,
context.__("Done? If not, don't worry, I'm still listening."),
_.take(tagReplies(context),1)
);
}
}
} else if (!isSticker && attachments) {
if (attachments[0].type == 'image' || attachments[0].type == 'video') {
// Add photos/videos
context.lastSent = (new Date()).getTime();
m.sendText( userid, context.__("The photos/videos have been added.") );
addPhotos(attachments, context);
} else if (attachments[0].type == 'location') {
context.lastSent = (new Date()).getTime();
m.sendText( userid, context.__("🚩 The location has been updated.") );
const point = attachments[0].payload.coordinates;
context.location = [point.lat, point.long];
}
}
} else if (context.state === STATE_WAIT_TAGS) {
if (messageText) {
if (message.quick_reply) {
if (message.quick_reply.payload != 'isEnding') {
context.categories.push(message.quick_reply.payload);
}
}
const isEnding = processText(messageText, context);
if (isEnding) {
context.lastSent = (new Date()).getTime();
m.sendText( userid, context.__("Thank you very much, {{name}}. Please follow the link below to verify yourself and submit the report. We will notify the responsible agency as soon as possible.", { name : context.profile.first_name } ) );
const desc = context.desc.join(' ');
let res = await ( new Promise( (resolve,reject) => {
api.postPin(
{
categories: context.categories,
created_time: (new Date()).getTime(),
detail: desc,
location: {
coordinates: context.location
},
owner: apiUserId,
photos: context.photos,
provider: apiUserId,
status: 'unverified',
tags: context.hashtags
},
resolve
);
}));
const pinId = res._id
const elements = [{
title: 'ยุพิน | YouPin',
subtitle: desc,
item_url: `http://youpin.city/pins/${pinId}`,
image_url: context.photos[0]
}]
m.sendGeneric(userid, elements);
context = {};
await (conversation.updateContext(userid, context));
} else {
context.lastSent = (new Date()).getTime();
m.sendTextWithReplies(userid, context.__("Anything else? You can keep adding more tags.") , tagReplies(context) );
}
}
}
await (conversation.updateContext(userid, context));
console.log("-- Saved context --");
console.log(context);
})
};
};