mirror of
https://github.com/diamondburned/arikawa.git
synced 2024-11-30 10:43:30 +00:00
diamondburned
273fcf1418
This commit adds subcommand aliases as well as additional code in HelpGenerate to cover for both subcommand and command aliases. A breaking change is that {,Must}RegisterSubcommandCustom methods are now replaced with normal {,Must}RegisterSubcommand methods. This is because they now use variadic strings, which could take 0, 1 or more arguments. This commit also allows AddMiddleware and similar methods to be given a method directly: sub.Plumb(cmds.PlumbedHandler) sub.AddMiddleware(cmds.PlumbedHandler, cmds.plumbMiddleware) This change closes issue #146.
147 lines
3.9 KiB
Go
147 lines
3.9 KiB
Go
package main
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/diamondburned/arikawa/v2/bot"
|
|
"github.com/diamondburned/arikawa/v2/bot/extras/arguments"
|
|
"github.com/diamondburned/arikawa/v2/bot/extras/middlewares"
|
|
"github.com/diamondburned/arikawa/v2/discord"
|
|
"github.com/diamondburned/arikawa/v2/gateway"
|
|
)
|
|
|
|
type Bot struct {
|
|
// Context must not be embedded.
|
|
Ctx *bot.Context
|
|
}
|
|
|
|
func (bot *Bot) Setup(sub *bot.Subcommand) {
|
|
// Only allow people in guilds to run guildInfo.
|
|
sub.AddMiddleware(bot.GuildInfo, middlewares.GuildOnly(bot.Ctx))
|
|
}
|
|
|
|
// Help prints the default help message.
|
|
func (bot *Bot) Help(*gateway.MessageCreateEvent) (string, error) {
|
|
return bot.Ctx.Help(), nil
|
|
}
|
|
|
|
// Add demonstrates the usage of typed arguments. Run it with "~add 1 2".
|
|
func (bot *Bot) Add(_ *gateway.MessageCreateEvent, a, b int) (string, error) {
|
|
return fmt.Sprintf("%d + %d = %d", a, b, a+b), nil
|
|
}
|
|
|
|
// Ping is a simple ping example, perhaps the most simple you could make it.
|
|
func (bot *Bot) Ping(*gateway.MessageCreateEvent) (string, error) {
|
|
return "Pong!", nil
|
|
}
|
|
|
|
// Say demonstrates how arguments.Flag could be used without the flag library.
|
|
func (bot *Bot) Say(_ *gateway.MessageCreateEvent, f bot.RawArguments) (string, error) {
|
|
if f != "" {
|
|
return string(f), nil
|
|
}
|
|
return "", errors.New("missing content")
|
|
}
|
|
|
|
// GuildInfo demonstrates the GuildOnly middleware done in (*Bot).Setup().
|
|
func (bot *Bot) GuildInfo(m *gateway.MessageCreateEvent) (string, error) {
|
|
g, err := bot.Ctx.GuildWithCount(m.GuildID)
|
|
if err != nil {
|
|
return "", fmt.Errorf("failed to get guild: %v", err)
|
|
}
|
|
|
|
return fmt.Sprintf(
|
|
"Your guild is %s, and its maximum members is %d",
|
|
g.Name, g.ApproximateMembers,
|
|
), nil
|
|
}
|
|
|
|
// Repeat tells the bot to wait for the user's response, then repeat what they
|
|
// said.
|
|
func (bot *Bot) Repeat(m *gateway.MessageCreateEvent) (string, error) {
|
|
_, err := bot.Ctx.SendMessage(m.ChannelID, "What do you want me to say?", nil)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), time.Minute)
|
|
defer cancel()
|
|
|
|
// This might miss events that are sent immediately after. To make sure all
|
|
// events are caught, ChanFor should be used.
|
|
v := bot.Ctx.WaitFor(ctx, func(v interface{}) bool {
|
|
// Incoming event is a message create event:
|
|
mg, ok := v.(*gateway.MessageCreateEvent)
|
|
if !ok {
|
|
return false
|
|
}
|
|
|
|
// Message is from the same author:
|
|
return mg.Author.ID == m.Author.ID
|
|
})
|
|
|
|
if v == nil {
|
|
return "", errors.New("timed out waiting for response")
|
|
}
|
|
|
|
ev := v.(*gateway.MessageCreateEvent)
|
|
return ev.Content, nil
|
|
}
|
|
|
|
// Embed is a simple embed creator. Its purpose is to demonstrate the usage of
|
|
// the ParseContent interface, as well as using the stdlib flag package.
|
|
func (bot *Bot) Embed(_ *gateway.MessageCreateEvent, f arguments.Flag) (*discord.Embed, error) {
|
|
fs := arguments.NewFlagSet()
|
|
|
|
var (
|
|
title = fs.String("title", "", "Title")
|
|
author = fs.String("author", "", "Author")
|
|
footer = fs.String("footer", "", "Footer")
|
|
color = fs.String("color", "#FFFFFF", "Color in hex format #hhhhhh")
|
|
)
|
|
|
|
if err := f.With(fs.FlagSet); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if len(fs.Args()) < 1 {
|
|
return nil, fmt.Errorf("usage: embed [flags] content...\n" + fs.Usage())
|
|
}
|
|
|
|
// Check if the color string is valid.
|
|
if !strings.HasPrefix(*color, "#") || len(*color) != 7 {
|
|
return nil, errors.New("invalid color, format must be #hhhhhh")
|
|
}
|
|
|
|
// Parse the color into decimal numbers.
|
|
colorHex, err := strconv.ParseInt((*color)[1:], 16, 64)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Make a new embed
|
|
embed := discord.Embed{
|
|
Title: *title,
|
|
Description: strings.Join(fs.Args(), " "),
|
|
Color: discord.Color(colorHex),
|
|
}
|
|
|
|
if *author != "" {
|
|
embed.Author = &discord.EmbedAuthor{
|
|
Name: *author,
|
|
}
|
|
}
|
|
if *footer != "" {
|
|
embed.Footer = &discord.EmbedFooter{
|
|
Text: *footer,
|
|
}
|
|
}
|
|
|
|
return &embed, err
|
|
}
|