forked from radiantly/live-countdown-bot
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathupdateManager.js
125 lines (110 loc) · 3.63 KB
/
updateManager.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
import { Message, DiscordAPIError, MessageManager } from "discord.js";
import {
getNextInQueue,
removeMessageWithReplyId,
updateCountObj,
updateRecomputedCountdown,
} from "./sqlite3.js";
import { computeTimeDiff } from "./computeTimeDiff.js";
import { assembleInlineMessage } from "./countdownHelper.js";
import { t } from "./lang.js";
// Automatically reject promise after 5000ms
export const timedPromise = (ms = 5000) =>
new Promise((_, reject) => setTimeout(reject, ms, new Error("Promise timed out")));
export const updateCountdowns = async (client, clientId) => {
setTimeout(updateCountdowns, 1000, client, clientId);
const countdownObj = getNextInQueue(clientId);
// Abort if no elements exist
if (!countdownObj) return;
const {
Channel: channel_id,
ReplyMessage: replyMsgId,
NextUpdate: thisUpdate,
CountObj: countObj,
} = countdownObj;
// Abort if thisUpdate is supposed to be too far away
if (Math.round(thisUpdate) > Date.now()) return;
const { parts, timers } = JSON.parse(countObj);
// Check if inline message
if (parts) {
const { assembledMessage, nextUpdate, priority, finishedTimers } = assembleInlineMessage(
timers,
parts
);
if (!nextUpdate) removeMessageWithReplyId(replyMsgId);
else
updateRecomputedCountdown({
replyMsgId,
nextUpdate,
priority,
});
// Check if a countdown in the inline countdown is finished
if (finishedTimers.length) {
// If it has a tag, tag users.
const tags = finishedTimers
.map(timerIndex => timers[timerIndex].tag)
.filter(Boolean)
.join(" ");
if (tags) {
const channel = await client.channels.fetch(channel_id);
channel.send(`${t("countdownDone", timers[finishedTimers[0]].lang)}! ${tags}`);
}
// Remove finished timers backwards
for (let i = finishedTimers.length - 1; i >= 0; i--) {
parts.splice(
finishedTimers[i],
2,
parts[finishedTimers[i]] +
t("inlineNoMinutes", timers[finishedTimers[i]].lang) +
parts[finishedTimers[i] + 1]
);
timers.splice(finishedTimers[i], 1);
}
// Update countObj
updateCountObj({ replyMsgId, countObj: JSON.stringify({ parts, timers }) });
}
await Promise.race([
timedPromise(),
new Message(client, {
id: replyMsgId,
channel_id,
}).edit({
content: assembledMessage,
}),
]);
} else {
// If not inline, it must be a normal message
const { lang } = timers[0];
const timeEnd = new Date(timers[0].timeEnd);
const timeLeft = timeEnd - Date.now();
const timeElapsed = timers[0].timeStart ? Date.now() - new Date(timers[0].timeStart) : Infinity;
let editText;
// If countdown is done
if (timeLeft < 10000) {
removeMessageWithReplyId(replyMsgId);
editText = t("countdownDone", lang);
// If tag exists, send a new message with the tag.
if (timers[0].tag) {
const channel = await client.channels.fetch(channel_id);
channel.send(`${t("countdownDone", lang)}! ${timers[0].tag}`);
}
} else {
const { humanDiff, timeLeftForNextUpdate } = computeTimeDiff(timeLeft, timeElapsed, lang);
editText = `${t("timeLeft", lang)}: ${humanDiff}`;
updateRecomputedCountdown({
replyMsgId,
nextUpdate: timeEnd - timeLeftForNextUpdate,
priority: timeLeftForNextUpdate ? 0 : 10,
});
}
await Promise.race([
timedPromise(),
new Message(client, {
id: replyMsgId,
channel_id,
}).edit({
content: editText,
}),
]);
}
};