2023-08-10 15:49:25 +03:00
|
|
|
package tx
|
2023-07-09 01:28:59 +03:00
|
|
|
|
2023-07-12 14:20:52 +03:00
|
|
|
import (
|
2023-07-12 14:59:07 +03:00
|
|
|
"fmt"
|
2023-08-10 15:49:25 +03:00
|
|
|
|
|
|
|
apix "github.com/go-telegram-bot-api/telegram-bot-api/v5"
|
2023-07-12 14:20:52 +03:00
|
|
|
)
|
|
|
|
|
2023-07-09 01:28:59 +03:00
|
|
|
// The type represents way to interact with user in
|
|
|
|
// handling functions. Is provided to Act() function always.
|
|
|
|
type Context struct {
|
2023-07-12 14:06:05 +03:00
|
|
|
*Session
|
2023-08-12 14:35:33 +03:00
|
|
|
B *Bot
|
|
|
|
updates chan *Update
|
|
|
|
|
|
|
|
// Is true if currently reading the Update.
|
|
|
|
readingUpdate bool
|
|
|
|
}
|
|
|
|
|
|
|
|
// Context for interaction inside groups.
|
|
|
|
type GroupContext struct {
|
|
|
|
*GroupSession
|
|
|
|
B *Bot
|
2023-07-12 14:06:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Goroutie function to handle each user.
|
2023-08-11 10:41:17 +03:00
|
|
|
func (c *Context) handleUpdateChan(updates chan *Update) {
|
|
|
|
bot := c.B
|
|
|
|
session := c.Session
|
2023-08-12 15:54:05 +03:00
|
|
|
c.run(bot.Start, nil)
|
2023-07-12 14:06:05 +03:00
|
|
|
for u := range updates {
|
2023-07-12 14:20:52 +03:00
|
|
|
screen := bot.Screens[session.CurrentScreenId]
|
2023-08-11 10:41:17 +03:00
|
|
|
// 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
|
|
|
var act Action
|
|
|
|
if u.Message.IsCommand() && !c.readingUpdate {
|
|
|
|
cmdName := CommandName(u.Message.Command())
|
|
|
|
cmd, ok := bot.Behaviour.Commands[cmdName]
|
|
|
|
if ok {
|
|
|
|
act = cmd.Action
|
|
|
|
} else {
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
kbd := bot.Keyboards[screen.KeyboardId]
|
|
|
|
btns := kbd.buttonMap()
|
|
|
|
text := u.Message.Text
|
|
|
|
btn, ok := btns[text]
|
2023-08-12 15:54:05 +03:00
|
|
|
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
|
|
|
}
|
2023-08-10 15:49:25 +03:00
|
|
|
|
2023-08-12 14:35:33 +03:00
|
|
|
if act != nil {
|
2023-08-12 15:54:05 +03:00
|
|
|
c.run(act, u)
|
2023-08-11 10:41:17 +03:00
|
|
|
}
|
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
|
2023-08-10 15:49:25 +03:00
|
|
|
|
2023-07-12 14:06:05 +03:00
|
|
|
_, err := bot.Request(cb)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2023-07-12 14:30:05 +03:00
|
|
|
kbd := bot.Keyboards[screen.InlineKeyboardId]
|
|
|
|
btns := kbd.buttonMap()
|
2023-08-12 15:54:05 +03:00
|
|
|
btn, ok := btns[data]
|
|
|
|
if !ok && c.readingUpdate {
|
|
|
|
c.updates <- u
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
c.run(btn.Action, u)
|
2023-07-12 14:06:05 +03:00
|
|
|
}
|
|
|
|
}
|
2023-07-09 01:28:59 +03:00
|
|
|
}
|
|
|
|
|
2023-08-12 15:54:05 +03:00
|
|
|
func (c *Context) run(a Action, u *Update) {
|
|
|
|
go a.Act(&A{c, u})
|
2023-08-11 10:41:17 +03:00
|
|
|
}
|
|
|
|
|
2023-07-12 00:33:51 +03:00
|
|
|
// Changes screen of user to the Id one.
|
2023-08-12 15:54:05 +03:00
|
|
|
func (c *Arg) ChangeScreen(screenId ScreenId) error {
|
2023-07-12 14:06:05 +03:00
|
|
|
// Return if it will not change anything.
|
|
|
|
if c.CurrentScreenId == screenId {
|
|
|
|
return nil
|
|
|
|
}
|
2023-08-10 15:49:25 +03:00
|
|
|
|
|
|
|
if !c.B.ScreenExist(screenId) {
|
2023-07-09 01:28:59 +03:00
|
|
|
return ScreenNotExistErr
|
|
|
|
}
|
2023-08-10 15:49:25 +03:00
|
|
|
|
2023-08-12 14:35:33 +03:00
|
|
|
// Stop the reading by sending the nil.
|
|
|
|
if c.readingUpdate {
|
|
|
|
c.updates <- nil
|
|
|
|
}
|
|
|
|
|
2023-07-12 14:06:05 +03:00
|
|
|
screen := c.B.Screens[screenId]
|
2023-08-12 15:54:05 +03:00
|
|
|
screen.Render(c.Context)
|
2023-08-10 15:49:25 +03:00
|
|
|
|
2023-07-12 14:06:05 +03:00
|
|
|
c.Session.ChangeScreen(screenId)
|
|
|
|
c.KeyboardId = screen.KeyboardId
|
2023-08-10 15:49:25 +03:00
|
|
|
|
2023-08-11 10:41:17 +03:00
|
|
|
if screen.Action != nil {
|
2023-08-12 15:54:05 +03:00
|
|
|
c.run(screen.Action, c.U)
|
2023-08-11 10:41:17 +03:00
|
|
|
}
|
|
|
|
|
2023-07-09 01:28:59 +03:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-08-11 10:41:17 +03:00
|
|
|
// Returns the next update ignoring current screen.
|
|
|
|
func (c *Context) ReadUpdate() (*Update, error) {
|
2023-08-11 12:45:50 +03:00
|
|
|
c.readingUpdate = true
|
|
|
|
u := <-c.updates
|
|
|
|
c.readingUpdate = false
|
|
|
|
if u == nil {
|
|
|
|
return nil, NotAvailableErr
|
2023-08-11 10:41:17 +03:00
|
|
|
}
|
|
|
|
|
2023-08-11 12:45:50 +03:00
|
|
|
return u, nil
|
2023-08-11 10:41:17 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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
|
|
|
|
}
|
|
|
|
|
2023-07-12 00:33:51 +03:00
|
|
|
// Sends to the user specified text.
|
2023-08-11 10:41:17 +03:00
|
|
|
func (c *Context) Send(v ...any) error {
|
|
|
|
msg := apix.NewMessage(c.Id.ToTelegram(), fmt.Sprint(v...))
|
2023-07-12 14:59:07 +03:00
|
|
|
_, err := c.B.Send(msg)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-08-11 10:41:17 +03:00
|
|
|
// Sends the formatted with fmt.Sprintf message to the user.
|
2023-07-12 14:59:07 +03:00
|
|
|
func (c *Context) Sendf(format string, v ...any) error {
|
|
|
|
return c.Send(fmt.Sprintf(format, v...))
|
2023-07-09 01:28:59 +03:00
|
|
|
}
|