aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/lib/common/HighlightManager.ts63
-rw-r--r--src/listeners/message/highlight.ts3
2 files changed, 59 insertions, 7 deletions
diff --git a/src/lib/common/HighlightManager.ts b/src/lib/common/HighlightManager.ts
index 4347c27..6781f50 100644
--- a/src/lib/common/HighlightManager.ts
+++ b/src/lib/common/HighlightManager.ts
@@ -3,6 +3,10 @@ import assert from 'assert';
import { Collection, type Snowflake } from 'discord.js';
import { Time } from '../utils/BushConstants.js';
+const NOTIFY_COOLDOWN = 5 * Time.Minute;
+const OWNER_NOTIFY_COOLDOWN = 1 * Time.Minute;
+const LAST_MESSAGE_COOLDOWN = 5 * Time.Minute;
+
type users = Set<Snowflake>;
type channels = Set<Snowflake>;
type word = HighlightWord;
@@ -154,7 +158,7 @@ export class HighlightManager {
highlight.words = util.addToArray(highlight.words, hl);
- return !!(await highlight.save().catch(() => false));
+ return Boolean(await highlight.save().catch(() => false));
}
/**
@@ -181,7 +185,7 @@ export class HighlightManager {
highlight.words = util.removeFromArray(highlight.words, toRemove);
- return !!(await highlight.save().catch(() => false));
+ return Boolean(await highlight.save().catch(() => false));
}
/**
@@ -205,20 +209,52 @@ export class HighlightManager {
highlight.words = [];
- return !!(await highlight.save().catch(() => false));
+ return Boolean(await highlight.save().catch(() => false));
}
+ /**
+ * Sends a user a direct message to alert them of their highlight being triggered.
+ * @param message The message that triggered the highlight.
+ * @param user The user who's highlights was triggered.
+ * @param hl The highlight that was matched.
+ * @returns Whether or a dm was sent.
+ */
public async notify(message: BushMessage, user: Snowflake, hl: HighlightWord): Promise<boolean> {
assert(message.inGuild());
- if (this.lastedDMedUserCooldown.has(user)) {
- const lastDM = this.lastedDMedUserCooldown.get(user)!;
- if (new Date().getTime() - lastDM.getTime() < 5 * Time.Minute) {
+ dmCooldown: {
+ const lastDM = this.lastedDMedUserCooldown.get(user);
+ if (!lastDM) break dmCooldown;
+
+ const cooldown = client.ownerID.includes(user) ? OWNER_NOTIFY_COOLDOWN : NOTIFY_COOLDOWN;
+
+ if (new Date().getTime() - lastDM.getTime() < cooldown) {
void client.console.verbose('Highlight', `User <<${user}>> has been dmed recently.`);
return false;
}
}
+ talkCooldown: {
+ const lastTalked = this.userLastTalkedCooldown.get(message.guildId)?.get(user);
+ if (!lastTalked) break talkCooldown;
+
+ const now = new Date().getTime();
+ const talked = lastTalked.getTime();
+
+ if (now - talked < LAST_MESSAGE_COOLDOWN) {
+ void client.console.verbose('Highlight', `User <<${user}>> has talked too recently.`);
+
+ setTimeout(() => {
+ const newTalked = this.userLastTalkedCooldown.get(message.guildId)?.get(user)?.getTime();
+ if (talked !== newTalked) return;
+
+ void this.notify(message, user, hl);
+ }, LAST_MESSAGE_COOLDOWN).unref();
+
+ return false;
+ }
+ }
+
const recentMessages = message.channel.messages.cache
.filter((m) => m.createdTimestamp <= message.createdTimestamp && m.id !== message.id)
.filter((m) => m.cleanContent?.trim().length > 0)
@@ -254,4 +290,19 @@ export class HighlightManager {
})
.catch(() => false);
}
+
+ /**
+ * Updates the time that a user last talked in a particular guild.
+ * @param message The message the user sent.
+ */
+ public updateLastTalked(message: BushMessage): void {
+ if (!message.inGuild()) return;
+ const lastTalked = (
+ this.userLastTalkedCooldown.has(message.guildId)
+ ? this.userLastTalkedCooldown
+ : this.userLastTalkedCooldown.set(message.guildId, new Collection())
+ ).get(message.guildId)!;
+
+ lastTalked.set(message.author.id, new Date());
+ }
}
diff --git a/src/listeners/message/highlight.ts b/src/listeners/message/highlight.ts
index c3a6ce0..d7965c5 100644
--- a/src/listeners/message/highlight.ts
+++ b/src/listeners/message/highlight.ts
@@ -12,8 +12,9 @@ export default class HighlightListener extends BushListener {
public override async exec(...[message]: BushClientEvents['messageCreate']) {
if (!message.inGuild()) return;
if (message.author.bot || message.system) return;
- if (!((await message.guild.hasFeature('highlight')) && !message.author.isOwner())) return; // allows highlighting to be disabled on a guild-by-guild basis
+ if (!(await message.guild.hasFeature('highlight'))) return; // allows highlighting to be disabled on a guild-by-guild basis
+ client.highlightManager.updateLastTalked(message);
const res = client.highlightManager.checkMessage(message);
for (const [user, hl] of res.entries()) {