use adapter to make discord.js optional
This commit is contained in:
17
notes.md
17
notes.md
@@ -1,17 +1,17 @@
|
|||||||
# core features
|
# core features
|
||||||
|
|
||||||
- [x] rendering core
|
- [ ] render to channel
|
||||||
- [ ] render to interaction
|
- [x] render to interaction
|
||||||
- [ ] ephemeral messages
|
- [ ] ephemeral messages
|
||||||
- [x] message content
|
- [x] message content
|
||||||
- embed
|
- embed
|
||||||
- [x] color
|
- [x] color
|
||||||
- [x] author
|
- [ ] author
|
||||||
- [x] description
|
- [x] description
|
||||||
- [x] title - text children, url
|
- [x] title - text children, url
|
||||||
- [x] footer - icon url, timestamp, text children
|
- [ ] footer - icon url, timestamp, text children
|
||||||
- [x] thumbnail - url
|
- [ ] thumbnail - url
|
||||||
- [x] image - url
|
- [ ] image - url
|
||||||
- [x] fields - name, value, inline
|
- [x] fields - name, value, inline
|
||||||
- message components
|
- message components
|
||||||
- [x] buttons
|
- [x] buttons
|
||||||
@@ -20,11 +20,16 @@
|
|||||||
- [ ] action row
|
- [ ] action row
|
||||||
- [x] button onClick
|
- [x] button onClick
|
||||||
- [ ] select onChange
|
- [ ] select onChange
|
||||||
|
- [x] deactivate
|
||||||
|
- [ ] destroy
|
||||||
|
|
||||||
# cool ideas / polish
|
# cool ideas / polish
|
||||||
|
|
||||||
|
- [ ] message property on reacord instance
|
||||||
- [ ] files
|
- [ ] files
|
||||||
- [ ] stickers
|
- [ ] stickers
|
||||||
- [ ] user mention component
|
- [ ] user mention component
|
||||||
- [ ] channel mention component
|
- [ ] channel mention component
|
||||||
- [ ] timestamp component
|
- [ ] timestamp component
|
||||||
|
- [ ] `useMessage`
|
||||||
|
- [ ] `useReactions`
|
||||||
|
|||||||
@@ -35,6 +35,11 @@
|
|||||||
"discord.js": "^13.3",
|
"discord.js": "^13.3",
|
||||||
"react": ">=17"
|
"react": ">=17"
|
||||||
},
|
},
|
||||||
|
"peerDependenciesMeta": {
|
||||||
|
"discord.js": {
|
||||||
|
"optional": true
|
||||||
|
}
|
||||||
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@itsmapleleaf/configs": "^1.1.2",
|
"@itsmapleleaf/configs": "^1.1.2",
|
||||||
"@typescript-eslint/eslint-plugin": "^5.8.0",
|
"@typescript-eslint/eslint-plugin": "^5.8.0",
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import { Client } from "discord.js"
|
import { Client } from "discord.js"
|
||||||
import "dotenv/config"
|
import "dotenv/config"
|
||||||
import React from "react"
|
import React from "react"
|
||||||
|
import { DiscordJsAdapter } from "../src/discord-js-adapter"
|
||||||
import { Reacord } from "../src/main.js"
|
import { Reacord } from "../src/main.js"
|
||||||
import { createCommandHandler } from "./command-handler.js"
|
import { createCommandHandler } from "./command-handler.js"
|
||||||
import { Counter } from "./counter.js"
|
import { Counter } from "./counter.js"
|
||||||
@@ -9,14 +10,17 @@ const client = new Client({
|
|||||||
intents: ["GUILDS"],
|
intents: ["GUILDS"],
|
||||||
})
|
})
|
||||||
|
|
||||||
const reacord = Reacord.create({ client, maxInstances: 2 })
|
const reacord = new Reacord({
|
||||||
|
adapter: new DiscordJsAdapter(client),
|
||||||
|
maxInstances: 2,
|
||||||
|
})
|
||||||
|
|
||||||
createCommandHandler(client, [
|
createCommandHandler(client, [
|
||||||
{
|
{
|
||||||
name: "counter",
|
name: "counter",
|
||||||
description: "shows a counter button",
|
description: "shows a counter button",
|
||||||
run: (interaction) => {
|
run: (interaction) => {
|
||||||
const reply = reacord.reply(interaction)
|
const reply = reacord.createCommandReply(interaction)
|
||||||
reply.render(<Counter onDeactivate={() => reply.deactivate()} />)
|
reply.render(<Counter onDeactivate={() => reply.deactivate()} />)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|||||||
9
src/adapter.ts
Normal file
9
src/adapter.ts
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
import type { CommandInteraction, ComponentInteraction } from "./interaction"
|
||||||
|
|
||||||
|
export type Adapter<InteractionInit> = {
|
||||||
|
addComponentInteractionListener(
|
||||||
|
listener: (interaction: ComponentInteraction) => void,
|
||||||
|
): void
|
||||||
|
|
||||||
|
createCommandInteraction(interactionInfo: InteractionInit): CommandInteraction
|
||||||
|
}
|
||||||
@@ -1,24 +1,16 @@
|
|||||||
import type {
|
|
||||||
ButtonInteraction,
|
|
||||||
CacheType,
|
|
||||||
EmojiResolvable,
|
|
||||||
MessageButtonStyle,
|
|
||||||
MessageComponentInteraction,
|
|
||||||
MessageOptions,
|
|
||||||
} from "discord.js"
|
|
||||||
import { MessageActionRow } from "discord.js"
|
|
||||||
import { nanoid } from "nanoid"
|
import { nanoid } from "nanoid"
|
||||||
import React from "react"
|
import React from "react"
|
||||||
import { ReacordElement } from "./element.js"
|
import { ReacordElement } from "./element.js"
|
||||||
import { last } from "./helpers/last.js"
|
import { last } from "./helpers/last.js"
|
||||||
import { toUpper } from "./helpers/to-upper.js"
|
import type { ButtonInteraction, ComponentInteraction } from "./interaction"
|
||||||
|
import type { MessageOptions } from "./message"
|
||||||
import { Node } from "./node.js"
|
import { Node } from "./node.js"
|
||||||
|
|
||||||
export type ButtonProps = {
|
export type ButtonProps = {
|
||||||
label?: string
|
label?: string
|
||||||
style?: Exclude<Lowercase<MessageButtonStyle>, "link">
|
style?: "primary" | "secondary" | "success" | "danger"
|
||||||
disabled?: boolean
|
disabled?: boolean
|
||||||
emoji?: EmojiResolvable
|
emoji?: string
|
||||||
onClick: (interaction: ButtonInteraction) => void
|
onClick: (interaction: ButtonInteraction) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -31,44 +23,38 @@ export function Button(props: ButtonProps) {
|
|||||||
class ButtonNode extends Node<ButtonProps> {
|
class ButtonNode extends Node<ButtonProps> {
|
||||||
private customId = nanoid()
|
private customId = nanoid()
|
||||||
|
|
||||||
private get buttonOptions() {
|
override modifyMessageOptions(options: MessageOptions): void {
|
||||||
return {
|
options.actionRows ??= []
|
||||||
type: "BUTTON",
|
|
||||||
|
let actionRow = last(options.actionRows)
|
||||||
|
|
||||||
|
if (
|
||||||
|
actionRow == undefined ||
|
||||||
|
actionRow.length >= 5 ||
|
||||||
|
actionRow[0]?.type === "select"
|
||||||
|
) {
|
||||||
|
actionRow = []
|
||||||
|
options.actionRows.push(actionRow)
|
||||||
|
}
|
||||||
|
|
||||||
|
actionRow.push({
|
||||||
|
type: "button",
|
||||||
customId: this.customId,
|
customId: this.customId,
|
||||||
style: toUpper(this.props.style ?? "secondary"),
|
style: this.props.style ?? "secondary",
|
||||||
disabled: this.props.disabled,
|
disabled: this.props.disabled,
|
||||||
emoji: this.props.emoji,
|
emoji: this.props.emoji,
|
||||||
label: this.props.label,
|
label: this.props.label,
|
||||||
} as const
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
override modifyMessageOptions(options: MessageOptions): void {
|
override handleComponentInteraction(interaction: ComponentInteraction) {
|
||||||
options.components ??= []
|
|
||||||
|
|
||||||
let actionRow = last(options.components)
|
|
||||||
|
|
||||||
if (
|
if (
|
||||||
!actionRow ||
|
interaction.type === "button" &&
|
||||||
actionRow.components.length >= 5 ||
|
interaction.customId === this.customId
|
||||||
actionRow.components[0]?.type === "SELECT_MENU"
|
|
||||||
) {
|
) {
|
||||||
actionRow = new MessageActionRow()
|
|
||||||
options.components.push(actionRow)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (actionRow instanceof MessageActionRow) {
|
|
||||||
actionRow.addComponents(this.buttonOptions)
|
|
||||||
} else {
|
|
||||||
actionRow.components.push(this.buttonOptions)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
override handleInteraction(
|
|
||||||
interaction: MessageComponentInteraction<CacheType>,
|
|
||||||
) {
|
|
||||||
if (interaction.isButton() && interaction.customId === this.customId) {
|
|
||||||
this.props.onClick(interaction)
|
this.props.onClick(interaction)
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
return false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
102
src/discord-js-adapter.ts
Normal file
102
src/discord-js-adapter.ts
Normal file
@@ -0,0 +1,102 @@
|
|||||||
|
import type * as Discord from "discord.js"
|
||||||
|
import type { Adapter } from "./adapter"
|
||||||
|
import { raise } from "./helpers/raise"
|
||||||
|
import { toUpper } from "./helpers/to-upper"
|
||||||
|
import type { CommandInteraction, ComponentInteraction } from "./interaction"
|
||||||
|
import type { Message, MessageOptions } from "./message"
|
||||||
|
|
||||||
|
export class DiscordJsAdapter implements Adapter<Discord.CommandInteraction> {
|
||||||
|
constructor(private client: Discord.Client) {}
|
||||||
|
|
||||||
|
addComponentInteractionListener(
|
||||||
|
listener: (interaction: ComponentInteraction) => void,
|
||||||
|
) {
|
||||||
|
this.client.on("interactionCreate", (interaction) => {
|
||||||
|
if (interaction.isButton()) {
|
||||||
|
listener(createReacordComponentInteraction(interaction))
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
createCommandInteraction(
|
||||||
|
interaction: Discord.CommandInteraction,
|
||||||
|
): CommandInteraction {
|
||||||
|
return {
|
||||||
|
type: "command",
|
||||||
|
id: interaction.id,
|
||||||
|
channelId: interaction.channelId,
|
||||||
|
reply: async (options) => {
|
||||||
|
const message = await interaction.reply({
|
||||||
|
...getDiscordMessageOptions(options),
|
||||||
|
fetchReply: true,
|
||||||
|
})
|
||||||
|
return createReacordMessage(message as Discord.Message)
|
||||||
|
},
|
||||||
|
followUp: async (options) => {
|
||||||
|
const message = await interaction.followUp({
|
||||||
|
...getDiscordMessageOptions(options),
|
||||||
|
fetchReply: true,
|
||||||
|
})
|
||||||
|
return createReacordMessage(message as Discord.Message)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function createReacordComponentInteraction(
|
||||||
|
interaction: Discord.MessageComponentInteraction,
|
||||||
|
): ComponentInteraction {
|
||||||
|
return {
|
||||||
|
type: "button",
|
||||||
|
id: interaction.id,
|
||||||
|
channelId: interaction.channelId,
|
||||||
|
customId: interaction.customId,
|
||||||
|
update: async (options) => {
|
||||||
|
await interaction.update(getDiscordMessageOptions(options))
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function createReacordMessage(message: Discord.Message): Message {
|
||||||
|
return {
|
||||||
|
edit: async (options) => {
|
||||||
|
await message.edit(getDiscordMessageOptions(options))
|
||||||
|
},
|
||||||
|
disableComponents: async () => {
|
||||||
|
for (const actionRow of message.components) {
|
||||||
|
for (const component of actionRow.components) {
|
||||||
|
component.setDisabled(true)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
await message.edit({
|
||||||
|
components: message.components,
|
||||||
|
})
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDiscordMessageOptions(
|
||||||
|
options: MessageOptions,
|
||||||
|
): Discord.MessageOptions {
|
||||||
|
return {
|
||||||
|
content: options.content,
|
||||||
|
embeds: options.embeds,
|
||||||
|
components: options.actionRows.map((row) => ({
|
||||||
|
type: "ACTION_ROW",
|
||||||
|
components: row.map((component) => {
|
||||||
|
if (component.type === "button") {
|
||||||
|
return {
|
||||||
|
type: "BUTTON",
|
||||||
|
customId: component.customId,
|
||||||
|
label: component.label ?? "",
|
||||||
|
style: toUpper(component.style ?? "secondary"),
|
||||||
|
disabled: component.disabled,
|
||||||
|
emoji: component.emoji,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
raise(`Unsupported component type: ${component.type}`)
|
||||||
|
}),
|
||||||
|
})),
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import type { MessageEmbedOptions } from "discord.js"
|
|
||||||
import { Node } from "../node.js"
|
import { Node } from "../node.js"
|
||||||
|
import type { EmbedOptions } from "./embed-options"
|
||||||
|
|
||||||
export abstract class EmbedChildNode<Props> extends Node<Props> {
|
export abstract class EmbedChildNode<Props> extends Node<Props> {
|
||||||
abstract modifyEmbedOptions(options: MessageEmbedOptions): void
|
abstract modifyEmbedOptions(options: EmbedOptions): void
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import type { MessageEmbedOptions } from "discord.js"
|
|
||||||
import React from "react"
|
import React from "react"
|
||||||
import { ReacordElement } from "../element.js"
|
import { ReacordElement } from "../element.js"
|
||||||
import { EmbedChildNode } from "./embed-child.js"
|
import { EmbedChildNode } from "./embed-child.js"
|
||||||
|
import type { EmbedOptions } from "./embed-options"
|
||||||
|
|
||||||
export type EmbedFieldProps = {
|
export type EmbedFieldProps = {
|
||||||
name: string
|
name: string
|
||||||
@@ -19,7 +19,7 @@ export function EmbedField(props: EmbedFieldProps) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class EmbedFieldNode extends EmbedChildNode<EmbedFieldProps> {
|
class EmbedFieldNode extends EmbedChildNode<EmbedFieldProps> {
|
||||||
override modifyEmbedOptions(options: MessageEmbedOptions): void {
|
override modifyEmbedOptions(options: EmbedOptions): void {
|
||||||
options.fields ??= []
|
options.fields ??= []
|
||||||
options.fields.push({
|
options.fields.push({
|
||||||
name: this.props.name,
|
name: this.props.name,
|
||||||
|
|||||||
19
src/embed/embed-options.ts
Normal file
19
src/embed/embed-options.ts
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
export type EmbedOptions = {
|
||||||
|
title?: string
|
||||||
|
description?: string
|
||||||
|
url?: string
|
||||||
|
timestamp?: string
|
||||||
|
color?: number
|
||||||
|
fields?: EmbedFieldOptions[]
|
||||||
|
author?: { name: string; url?: string; icon_url?: string }
|
||||||
|
thumbnail?: { url: string }
|
||||||
|
image?: { url: string }
|
||||||
|
video?: { url: string }
|
||||||
|
footer?: { text: string; icon_url?: string }
|
||||||
|
}
|
||||||
|
|
||||||
|
export type EmbedFieldOptions = {
|
||||||
|
name: string
|
||||||
|
value: string
|
||||||
|
inline?: boolean
|
||||||
|
}
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
import type { MessageEmbedOptions } from "discord.js"
|
|
||||||
import React from "react"
|
import React from "react"
|
||||||
import { ReacordElement } from "../element.js"
|
import { ReacordElement } from "../element.js"
|
||||||
import { EmbedChildNode } from "./embed-child.js"
|
import { EmbedChildNode } from "./embed-child.js"
|
||||||
|
import type { EmbedOptions } from "./embed-options"
|
||||||
|
|
||||||
export type EmbedTitleProps = {
|
export type EmbedTitleProps = {
|
||||||
children: string
|
children: string
|
||||||
@@ -18,7 +18,7 @@ export function EmbedTitle(props: EmbedTitleProps) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class EmbedTitleNode extends EmbedChildNode<EmbedTitleProps> {
|
class EmbedTitleNode extends EmbedChildNode<EmbedTitleProps> {
|
||||||
override modifyEmbedOptions(options: MessageEmbedOptions): void {
|
override modifyEmbedOptions(options: EmbedOptions): void {
|
||||||
options.title = this.props.children
|
options.title = this.props.children
|
||||||
options.url = this.props.url
|
options.url = this.props.url
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,13 +1,13 @@
|
|||||||
import type { MessageOptions } from "discord.js"
|
|
||||||
import React from "react"
|
import React from "react"
|
||||||
import { ReacordElement } from "../element.js"
|
import { ReacordElement } from "../element.js"
|
||||||
|
import type { MessageOptions } from "../message"
|
||||||
import { Node } from "../node.js"
|
import { Node } from "../node.js"
|
||||||
import { EmbedChildNode } from "./embed-child.js"
|
import { EmbedChildNode } from "./embed-child.js"
|
||||||
|
|
||||||
export type EmbedProps = {
|
export type EmbedProps = {
|
||||||
description?: string
|
description?: string
|
||||||
url?: string
|
url?: string
|
||||||
timestamp?: Date
|
timestamp?: string
|
||||||
color?: number
|
color?: number
|
||||||
footer?: {
|
footer?: {
|
||||||
text: string
|
text: string
|
||||||
|
|||||||
29
src/interaction.ts
Normal file
29
src/interaction.ts
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
import type { Message, MessageOptions } from "./message"
|
||||||
|
|
||||||
|
export type Interaction = CommandInteraction | ComponentInteraction
|
||||||
|
|
||||||
|
export type CommandInteraction = {
|
||||||
|
type: "command"
|
||||||
|
id: string
|
||||||
|
channelId: string
|
||||||
|
reply(messageOptions: MessageOptions): Promise<Message>
|
||||||
|
followUp(messageOptions: MessageOptions): Promise<Message>
|
||||||
|
}
|
||||||
|
|
||||||
|
export type ComponentInteraction = ButtonInteraction | SelectInteraction
|
||||||
|
|
||||||
|
export type ButtonInteraction = {
|
||||||
|
type: "button"
|
||||||
|
id: string
|
||||||
|
channelId: string
|
||||||
|
customId: string
|
||||||
|
update(options: MessageOptions): Promise<void>
|
||||||
|
}
|
||||||
|
|
||||||
|
export type SelectInteraction = {
|
||||||
|
type: "select"
|
||||||
|
id: string
|
||||||
|
channelId: string
|
||||||
|
customId: string
|
||||||
|
update(options: MessageOptions): Promise<void>
|
||||||
|
}
|
||||||
@@ -1,5 +1,9 @@
|
|||||||
|
export * from "./adapter"
|
||||||
export * from "./button"
|
export * from "./button"
|
||||||
|
export * from "./discord-js-adapter"
|
||||||
export * from "./embed/embed"
|
export * from "./embed/embed"
|
||||||
export * from "./embed/embed-field"
|
export * from "./embed/embed-field"
|
||||||
export * from "./embed/embed-title"
|
export * from "./embed/embed-title"
|
||||||
|
export * from "./interaction"
|
||||||
|
export * from "./message"
|
||||||
export * from "./reacord"
|
export * from "./reacord"
|
||||||
|
|||||||
28
src/message.ts
Normal file
28
src/message.ts
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
import type { EmbedOptions } from "./embed/embed-options"
|
||||||
|
|
||||||
|
export type MessageOptions = {
|
||||||
|
content: string
|
||||||
|
embeds: EmbedOptions[]
|
||||||
|
actionRows: Array<
|
||||||
|
Array<
|
||||||
|
| {
|
||||||
|
type: "button"
|
||||||
|
customId: string
|
||||||
|
label?: string
|
||||||
|
style?: "primary" | "secondary" | "success" | "danger"
|
||||||
|
disabled?: boolean
|
||||||
|
emoji?: string
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: "select"
|
||||||
|
customId: string
|
||||||
|
// todo
|
||||||
|
}
|
||||||
|
>
|
||||||
|
>
|
||||||
|
}
|
||||||
|
|
||||||
|
export type Message = {
|
||||||
|
edit(options: MessageOptions): Promise<void>
|
||||||
|
disableComponents(): Promise<void>
|
||||||
|
}
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
/* eslint-disable class-methods-use-this */
|
/* eslint-disable class-methods-use-this */
|
||||||
import type { MessageComponentInteraction, MessageOptions } from "discord.js"
|
|
||||||
import { Container } from "./container.js"
|
import { Container } from "./container.js"
|
||||||
|
import type { ComponentInteraction } from "./interaction"
|
||||||
|
import type { MessageOptions } from "./message"
|
||||||
|
|
||||||
export abstract class Node<Props> {
|
export abstract class Node<Props> {
|
||||||
readonly children = new Container<Node<unknown>>()
|
readonly children = new Container<Node<unknown>>()
|
||||||
@@ -16,9 +17,7 @@ export abstract class Node<Props> {
|
|||||||
|
|
||||||
modifyMessageOptions(options: MessageOptions) {}
|
modifyMessageOptions(options: MessageOptions) {}
|
||||||
|
|
||||||
handleInteraction(
|
handleComponentInteraction(interaction: ComponentInteraction): boolean {
|
||||||
interaction: MessageComponentInteraction,
|
return false
|
||||||
): true | undefined {
|
|
||||||
return undefined
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,13 +1,10 @@
|
|||||||
import type { Client, CommandInteraction } from "discord.js"
|
|
||||||
import type { ReactNode } from "react"
|
import type { ReactNode } from "react"
|
||||||
|
import type { Adapter } from "./adapter"
|
||||||
import { reconciler } from "./reconciler.js"
|
import { reconciler } from "./reconciler.js"
|
||||||
import { Renderer } from "./renderer.js"
|
import { Renderer } from "./renderer.js"
|
||||||
|
|
||||||
export type ReacordConfig = {
|
export type ReacordConfig<InteractionInit> = {
|
||||||
/**
|
adapter: Adapter<InteractionInit>
|
||||||
* A Discord.js client. Reacord will listen to interaction events
|
|
||||||
* and send them to active instances. */
|
|
||||||
client: Client
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The max number of active instances.
|
* The max number of active instances.
|
||||||
@@ -21,34 +18,30 @@ export type ReacordInstance = {
|
|||||||
deactivate: () => void
|
deactivate: () => void
|
||||||
}
|
}
|
||||||
|
|
||||||
export class Reacord {
|
export class Reacord<InteractionInit> {
|
||||||
private renderers: Renderer[] = []
|
private renderers: Renderer[] = []
|
||||||
|
|
||||||
private constructor(private readonly config: ReacordConfig) {}
|
constructor(private readonly config: ReacordConfig<InteractionInit>) {
|
||||||
|
config.adapter.addComponentInteractionListener((interaction) => {
|
||||||
|
for (const renderer of this.renderers) {
|
||||||
|
if (renderer.handleComponentInteraction(interaction)) return
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
private get maxInstances() {
|
private get maxInstances() {
|
||||||
return this.config.maxInstances ?? 50
|
return this.config.maxInstances ?? 50
|
||||||
}
|
}
|
||||||
|
|
||||||
static create(config: ReacordConfig) {
|
createCommandReply(target: InteractionInit): ReacordInstance {
|
||||||
const manager = new Reacord(config)
|
|
||||||
|
|
||||||
config.client.on("interactionCreate", (interaction) => {
|
|
||||||
if (!interaction.isMessageComponent()) return
|
|
||||||
for (const renderer of manager.renderers) {
|
|
||||||
if (renderer.handleInteraction(interaction)) return
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
return manager
|
|
||||||
}
|
|
||||||
|
|
||||||
reply(interaction: CommandInteraction): ReacordInstance {
|
|
||||||
if (this.renderers.length > this.maxInstances) {
|
if (this.renderers.length > this.maxInstances) {
|
||||||
this.deactivate(this.renderers[0]!)
|
this.deactivate(this.renderers[0]!)
|
||||||
}
|
}
|
||||||
|
|
||||||
const renderer = new Renderer(interaction)
|
const renderer = new Renderer(
|
||||||
|
this.config.adapter.createCommandInteraction(target),
|
||||||
|
)
|
||||||
|
|
||||||
this.renderers.push(renderer)
|
this.renderers.push(renderer)
|
||||||
|
|
||||||
const container = reconciler.createContainer(renderer, 0, false, {})
|
const container = reconciler.createContainer(renderer, 0, false, {})
|
||||||
|
|||||||
@@ -1,12 +1,9 @@
|
|||||||
import type {
|
|
||||||
CommandInteraction,
|
|
||||||
MessageComponentInteraction,
|
|
||||||
MessageOptions,
|
|
||||||
} from "discord.js"
|
|
||||||
import type { Subscription } from "rxjs"
|
import type { Subscription } from "rxjs"
|
||||||
import { Subject } from "rxjs"
|
import { Subject } from "rxjs"
|
||||||
import { concatMap } from "rxjs/operators"
|
import { concatMap } from "rxjs/operators"
|
||||||
import { Container } from "./container.js"
|
import { Container } from "./container.js"
|
||||||
|
import type { CommandInteraction, ComponentInteraction } from "./interaction"
|
||||||
|
import type { Message, MessageOptions } from "./message"
|
||||||
import type { Node } from "./node.js"
|
import type { Node } from "./node.js"
|
||||||
|
|
||||||
// keep track of interaction ids which have replies,
|
// keep track of interaction ids which have replies,
|
||||||
@@ -20,8 +17,8 @@ type UpdatePayload = {
|
|||||||
|
|
||||||
export class Renderer {
|
export class Renderer {
|
||||||
readonly nodes = new Container<Node<unknown>>()
|
readonly nodes = new Container<Node<unknown>>()
|
||||||
private componentInteraction?: MessageComponentInteraction
|
private componentInteraction?: ComponentInteraction
|
||||||
private messageId?: string
|
private message?: Message
|
||||||
private updates = new Subject<UpdatePayload>()
|
private updates = new Subject<UpdatePayload>()
|
||||||
private updateSubscription: Subscription
|
private updateSubscription: Subscription
|
||||||
private active = true
|
private active = true
|
||||||
@@ -52,10 +49,10 @@ export class Renderer {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
handleInteraction(interaction: MessageComponentInteraction) {
|
handleComponentInteraction(interaction: ComponentInteraction) {
|
||||||
for (const node of this.nodes) {
|
|
||||||
this.componentInteraction = interaction
|
this.componentInteraction = interaction
|
||||||
if (node.handleInteraction(interaction)) {
|
for (const node of this.nodes) {
|
||||||
|
if (node.handleComponentInteraction(interaction)) {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -65,7 +62,7 @@ export class Renderer {
|
|||||||
const options: MessageOptions = {
|
const options: MessageOptions = {
|
||||||
content: "",
|
content: "",
|
||||||
embeds: [],
|
embeds: [],
|
||||||
components: [],
|
actionRows: [],
|
||||||
}
|
}
|
||||||
for (const node of this.nodes) {
|
for (const node of this.nodes) {
|
||||||
node.modifyMessageOptions(options)
|
node.modifyMessageOptions(options)
|
||||||
@@ -74,24 +71,9 @@ export class Renderer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private async updateMessage({ options, action }: UpdatePayload) {
|
private async updateMessage({ options, action }: UpdatePayload) {
|
||||||
if (action === "deactivate" && this.messageId) {
|
if (action === "deactivate" && this.message) {
|
||||||
this.updateSubscription.unsubscribe()
|
this.updateSubscription.unsubscribe()
|
||||||
|
await this.message.disableComponents()
|
||||||
const message = await this.interaction.channel?.messages.fetch(
|
|
||||||
this.messageId,
|
|
||||||
)
|
|
||||||
if (!message) return
|
|
||||||
|
|
||||||
for (const actionRow of message.components) {
|
|
||||||
for (const component of actionRow.components) {
|
|
||||||
component.setDisabled(true)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
await this.interaction.channel?.messages.edit(message.id, {
|
|
||||||
components: message.components,
|
|
||||||
})
|
|
||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -102,25 +84,17 @@ export class Renderer {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.messageId) {
|
if (this.message) {
|
||||||
await this.interaction.channel?.messages.edit(this.messageId, options)
|
await this.message.edit(options)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if (repliedInteractionIds.has(this.interaction.id)) {
|
if (repliedInteractionIds.has(this.interaction.id)) {
|
||||||
const message = await this.interaction.followUp({
|
this.message = await this.interaction.followUp(options)
|
||||||
...options,
|
|
||||||
fetchReply: true,
|
|
||||||
})
|
|
||||||
this.messageId = message.id
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
repliedInteractionIds.add(this.interaction.id)
|
repliedInteractionIds.add(this.interaction.id)
|
||||||
const message = await this.interaction.reply({
|
this.message = await this.interaction.reply(options)
|
||||||
...options,
|
|
||||||
fetchReply: true,
|
|
||||||
})
|
|
||||||
this.messageId = message.id
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import type { MessageOptions } from "discord.js"
|
import type { MessageOptions } from "./message"
|
||||||
import { Node } from "./node.js"
|
import { Node } from "./node.js"
|
||||||
|
|
||||||
export class TextNode extends Node<string> {
|
export class TextNode extends Node<string> {
|
||||||
|
|||||||
Reference in New Issue
Block a user