tg/src/tx/context.go

202 lines
4 KiB
Go
Raw Normal View History

package tx
2023-07-09 01:28:59 +03:00
2023-07-12 14:20:52 +03:00
import (
"fmt"
apix "github.com/go-telegram-bot-api/telegram-bot-api/v5"
2023-07-12 14:20:52 +03:00
)
type context struct {
2023-07-12 14:06:05 +03:00
*Session
*Bot
2023-08-12 14:35:33 +03:00
updates chan *Update
// Is true if currently reading the Update.
readingUpdate bool
2023-08-15 16:02:14 +03:00
curScreen, prevScreen *Screen
2023-08-12 14:35:33 +03:00
}
// The type represents way to interact with user in
// handling functions. Is provided to Act() function always.
2023-07-12 14:06:05 +03:00
// Goroutie function to handle each user.
func (c *context) handleUpdateChan(updates chan *Update) {
2023-08-13 15:37:36 +03:00
var act Action
beh := c.behaviour
if beh.Init != nil {
c.run(beh.Init, nil)
}
2023-07-12 14:06:05 +03:00
for u := range updates {
2023-08-15 16:02:14 +03:00
screen := c.curScreen
// The part is added to implement custom update handling.
2023-07-12 14:06:05 +03:00
if u.Message != nil {
2023-08-12 14:35:33 +03:00
if u.Message.IsCommand() && !c.readingUpdate {
cmdName := CommandName(u.Message.Command())
2023-08-13 15:37:36 +03:00
cmd, ok := beh.Commands[cmdName]
2023-08-12 14:35:33 +03:00
if ok {
act = cmd.Action
} else {
}
} else {
2023-08-15 16:02:14 +03:00
kbd := screen.Keyboard
if kbd == nil {
if c.readingUpdate {
c.updates <- u
}
continue
}
2023-08-12 14:35:33 +03:00
btns := kbd.buttonMap()
text := u.Message.Text
btn, ok := btns[text]
if !ok {
if u.Message.Location != nil {
for _, b := range btns {
if b.SendLocation {
btn = b
ok = true
}
}
} else if c.readingUpdate {
// Skipping the update sending it to
// the reading goroutine.
c.updates <- u
continue
}
2023-08-12 14:35:33 +03:00
}
if ok {
act = btn.Action
}
2023-07-12 14:06:05 +03:00
}
} else if u.CallbackQuery != nil {
cb := apix.NewCallback(u.CallbackQuery.ID, u.CallbackQuery.Data)
2023-07-12 14:30:05 +03:00
data := u.CallbackQuery.Data
_, err := c.Request(cb)
2023-07-12 14:06:05 +03:00
if err != nil {
panic(err)
}
2023-08-15 16:02:14 +03:00
kbd := screen.InlineKeyboard
if kbd == nil {
if c.readingUpdate {
c.updates <- u
}
continue
}
2023-07-12 14:30:05 +03:00
btns := kbd.buttonMap()
btn, ok := btns[data]
if !ok && c.readingUpdate {
c.updates <- u
continue
}
2023-08-15 16:02:14 +03:00
if !ok {
c.Sendf("%q", btns)
continue
}
2023-08-13 15:37:36 +03:00
act = btn.Action
}
if act != nil {
c.run(act, u)
2023-07-12 14:06:05 +03:00
}
}
2023-07-09 01:28:59 +03:00
}
func (c *context) run(a Action, u *Update) {
go a.Act(&Context{
context: c,
Update: u,
2023-08-13 15:37:36 +03:00
})
2023-07-09 01:28:59 +03:00
}
// Returns the next update ignoring current screen.
func (c *context) ReadUpdate() (*Update, error) {
c.readingUpdate = true
u := <-c.updates
c.readingUpdate = false
if u == nil {
return nil, NotAvailableErr
}
return u, nil
}
// Returns the next text message that the user sends.
func (c *context) ReadTextMessage() (string, error) {
u, err := c.ReadUpdate()
if err != nil {
return "", err
}
if u.Message == nil {
return "", WrongUpdateType{}
}
return u.Message.Text, nil
}
// Sends to the user specified text.
func (c *context) Send(v ...any) error {
msg := apix.NewMessage(c.Id.ToTelegram(), fmt.Sprint(v...))
_, err := c.Bot.Send(msg)
return err
}
// Sends the formatted with fmt.Sprintf message to the user.
func (c *context) Sendf(format string, v ...any) error {
return c.Send(fmt.Sprintf(format, v...))
2023-07-09 01:28:59 +03:00
}
2023-08-13 15:37:36 +03:00
// Context for interaction inside groups.
type groupContext struct {
2023-08-13 15:37:36 +03:00
*GroupSession
*Bot
2023-08-13 15:37:36 +03:00
updates chan *Update
}
func (c *groupContext) run(a GroupAction, u *Update) {
go a.Act(&GroupContext{
groupContext: c,
2023-08-13 15:37:36 +03:00
Update: u,
})
}
func (c *groupContext) handleUpdateChan(updates chan *Update) {
2023-08-13 15:37:36 +03:00
var act GroupAction
beh := c.groupBehaviour
2023-08-13 15:37:36 +03:00
for u := range updates {
if u.Message != nil {
msg := u.Message
if msg.IsCommand() {
cmdName := CommandName(msg.Command())
// Skipping the commands sent not to us.
atName := msg.CommandWithAt()[len(cmdName)+1:]
if c.Bot.Me.UserName != atName {
2023-08-13 15:37:36 +03:00
continue
}
cmd, ok := beh.Commands[cmdName]
if !ok {
// Some lack of command handling
continue
}
act = cmd.Action
}
}
if act != nil {
c.run(act, u)
}
}
}
func (c *groupContext) Sendf(format string, v ...any) error {
2023-08-13 15:37:36 +03:00
return c.Send(fmt.Sprintf(format, v...))
}
// Sends into the chat specified values converted to strings.
func (c *groupContext) Send(v ...any) error {
2023-08-13 15:37:36 +03:00
msg := apix.NewMessage(c.Id.ToTelegram(), fmt.Sprint(v...))
_, err := c.Bot.Send(msg)
2023-08-13 15:37:36 +03:00
return err
}