tg/command.go

175 lines
3.5 KiB
Go
Raw Normal View History

2023-08-19 09:12:26 +03:00
package tg
2023-08-12 14:35:33 +03:00
import (
tgbotapi "github.com/go-telegram-bot-api/telegram-bot-api/v5"
2023-08-12 14:35:33 +03:00
)
2023-09-30 09:55:45 +03:00
type CommandType uint8
const (
PrivateCommandType CommandType = iota
GroupCommandType
ChannelCommandType
)
2023-08-12 14:35:33 +03:00
type CommandName string
type Command struct {
Name CommandName
2023-09-30 09:55:45 +03:00
Type CommandType
2023-08-12 14:35:33 +03:00
Description string
Action Action
Widget Widget
2023-08-12 14:35:33 +03:00
}
2023-08-13 15:37:36 +03:00
type CommandMap map[CommandName]*Command
2023-08-12 14:35:33 +03:00
func NewCommand(name CommandName, desc string) Command {
if name == "" || desc == "" {
panic("name and description cannot be an empty string")
}
2023-08-12 14:35:33 +03:00
return &Command{
Name: name,
Description: desc,
2023-08-12 14:35:33 +03:00
}
}
func (c *Command) WithAction(a Action) *Command {
c.Action = a
2023-08-12 14:35:33 +03:00
return c
}
func (c Command) WithWidget(w Widget) Command {
c.Widget = w
return c
}
func (c Command) WidgetFunc(fn Func) Command {
return c.WithWidget(fn)
}
func (c Command) ToApi() tgbotapi.BotCommand {
ret := tgbotapi.BotCommand{}
ret.Command = string(c.Name)
ret.Description = c.Description
return ret
}
func (c Command) Go(pth Path, args ...any) Command {
return c.WithAction(ScreenGo{
Path: pth,
Args: args,
})
}
// The type is used to recognize commands and execute
// its actions and widgets .
type CommandCompo struct {
PreStart Action
Commands CommandMap
Usage Action
}
// Returns new empty CommandCompo.
func NewCommandCompo(cmds ...*Command) *CommandCompo {
ret := CommandCompo{}.WithCommands(cmds...)
return ret
}
// Set the commands to handle.
func (w CommandCompo) WithCommands(cmds ...*Command) *CommandCompo {
if w.Commands == nil {
w.Commands = make(CommandMap)
}
for _, cmd := range cmds {
if cmd.Name == "" {
panic("empty command name")
}
_, ok := w.Commands[cmd.Name]
if ok {
panic("duplicate command definition")
}
w.Commands[cmd.Name] = cmd
}
return w
}
// Set the prestart action.
func (w *CommandCompo) WithPreStart(a Action) *CommandCompo {
w.PreStart = a
return w
}
// Set the usage action.
func (w CommandCompo) WithUsage(a Action) *CommandCompo {
w.Usage = a
return w
}
// Set the usage action with function.
func (w CommandCompo) WithUsageFunc(fn ActionFunc) *CommandCompo {
return w.WithUsage(fn)
}
func (widget CommandCompo) Filter(
u *Update,
) bool {
if u.Message == nil || !u.Message.IsCommand() {
return false
}
return false
}
// Implementing server.
func (compo CommandCompo) Serve(c Context) {
2023-09-30 09:55:45 +03:00
/*commanders := make(map[CommandName] BotCommander)
for k, v := range compo.Commands {
commanders[k] = v
2023-09-30 09:55:45 +03:00
}*/
c.bot.DeleteCommands()
err := c.bot.SetCommands(
tgbotapi.NewBotCommandScopeChat(c.Session.Id.ToApi()),
2023-09-30 09:55:45 +03:00
compo.Commands,
)
if err != nil {
c.Sendf("error: %q", err)
}
2023-09-16 15:40:30 +03:00
var cmdUpdates *UpdateChan
for u := range c.Input() {
if c.Path() == "" && u.Message != nil {
// Skipping and executing the preinit action
// while we have the empty screen.
// E. g. the session did not start.
if !(u.Message.IsCommand() && u.Message.Command() == "start") {
c.WithUpdate(u).Run(compo.PreStart)
continue
}
}
if u.Message != nil && u.Message.IsCommand() {
// Command handling.
cmdName := CommandName(u.Message.Command())
cmd, ok := compo.Commands[cmdName]
if !ok {
c.WithUpdate(u).Run(compo.Usage)
continue
}
c.WithUpdate(u).Run(cmd.Action)
if cmd.Widget != nil {
2023-09-16 15:40:30 +03:00
cmdUpdates.Close()
cmdUpdates, _ = c.WithUpdate(u).RunWidget(cmd.Widget)
}
continue
}
if !cmdUpdates.Closed() {
// Send to the commands channel if we are
// executing one.
2023-09-16 15:40:30 +03:00
cmdUpdates.Send(u)
} else {
c.Skip(u)
}
}
}