我有 2 个函数,这两个函数都编辑机器人(远程机器人)的消息,其中 1 个函数,当另一个函数随后调用时,应该阻止它编辑消息
用户可以编写“搜索”并随内联键盘一起接收消息,但是通过再次调用“搜索”,之前使用键盘的消息应更改为消息“键盘不再可用”并丢失键盘。使用内联键盘,用户还可以更改消息并为同一消息生成新键盘(因为他必须使用它输入多个参数)。但可能存在一种情况,用户在更改消息之前发送“搜索”一词后,单击内联按钮 - 在这种情况下,我希望消息仍然不执行以下代码中的这部分代码@bot.callback_query_handler():
global users_keyboard_data
users_keyboard_data = change_to_other_inline_keyboard(bot, call, users_keyboard_data)
也就是说,只有这部分应该在以下位置工作(因为它是首先调用的)@bot.message_handler():
if message.text == "Поиск":
global users_keyboard_data
users_keyboard_data = search(bot, message.chat.id, users_keyboard_data)
我没有复制所有代码,而是复制主要工作片段。我还想澄清一下,全局变量users_keyboard_data负责存储从内联键盘接收到的用户参数;如果有更简单的方法来存储它,请告诉我。
from telebot import TeleBot
from telebot.types import Message, CallbackQuery, InlineKeyboardButton, InlineKeyboardMarkup
from settings import config
users_keyboard_data = {}
bot = TeleBot(config.TOKEN)
def build_inline_keyboard(buttons: list[str]) -> InlineKeyboardMarkup:
return InlineKeyboardMarkup().add(*[InlineKeyboardButton(text=button, callback_data=button) for button in buttons])
def get_keys_if_chat_id_in_dict(chat_id: int, dictionary: dict) -> list[str]:
keys = []
for key in dictionary:
if key.split("_")[0] == str(chat_id):
keys.append(key)
return keys
def delete_previous_searches_besides(bot: TeleBot, id: int, users_keyboard_data: dict[dict[str: str]], besides: Message) -> dict[dict[str: str]]:
to_delete_lst = get_keys_if_chat_id_in_dict(id, users_keyboard_data)
to_delete_lst.remove(f"{id}_{besides.message_id}")
for key in to_delete_lst:
users_keyboard_data.pop(key, None)
bot.edit_message_text(chat_id=id, message_id=int(key.split("_")[1]), text="Клавиатура более недоступна")
return users_keyboard_data
def search(bot: TeleBot, id: int, users_keyboard_data: dict[dict[str: str]]) -> dict[dict[str: str]]:
message = bot.send_message(chat_id=id, text="Выберите округ", reply_markup=build_inline_keyboard(["Кнопка 1", "Кнопка 2"]))
users_keyboard_data[f"{id}_{message.message_id}"] = {}
users_keyboard_data = delete_previous_searches_besides(bot, id, users_keyboard_data, message)
return users_keyboard_data
def change_to_other_inline_keyboard(bot: TeleBot, call: CallbackQuery, users_keyboard_data: dict[dict[str: str]]) -> dict[dict[str: str]]:
users_keyboard_data[f"{call.from_user.id}_{call.message.message_id}"] = {"FIRST_DATA": f"{call.data}"}
bot.edit_message_text(chat_id=call.from_user.id, message_id=call.message.id, text="Новая клавиатура",
reply_markup=build_inline_keyboard(["Новая кнопка 1", "Новая кнопка 2"]))
return users_keyboard_data
@bot.message_handler()
def handle_message(message: Message) -> None:
if message.text == "Поиск":
global users_keyboard_data
users_keyboard_data = search(bot, message.chat.id, users_keyboard_data)
@bot.callback_query_handler(func=lambda call: True)
def handle_query(call: CallbackQuery) -> None:
global users_keyboard_data
users_keyboard_data = change_to_other_inline_keyboard(bot, call, users_keyboard_data)
if __name__ == "__main__":
bot.polling(none_stop=True)
我尝试这样做:
@bot.callback_query_handler(func=lambda call: True)
def handle_query(call: CallbackQuery) -> None:
if call.message.text != "Клавиатура более недоступна":
global users_keyboard_data
users_keyboard_data = change_to_other_inline_keyboard(bot, call, users_keyboard_data)
但即使远程机器人库中缺乏异步性,它仍然接受该条件为真。
那么,这对您意味着什么
Telebot?(嗯,基本上什么都没有,除了在你的情况下你有同步代码)。同步性似乎如此,它会影响什么。
正如您所看到的,这里的所有内容都一一运行,但是如果某些内容需要很长时间才能生成怎么办?假设按钮是在用户单击按钮之后生成的(这意味着无法访问类型的按钮的实现)。然后
Telegram它会发送一个更新,说明情况确实如此,并且该按钮包含除 之外的文本"Клавиатура более недоступна"。接下来,我们来筛选一下:一切看似简单、容易,但一座冰山突然出现,问题看似很小,但到底是什么?
@bot.callback_query_handler(func=lambda call: True)- 这看起来是一个简单、易于理解的过滤器,没有什么可担心的,但它会导致不应该发生的逻辑,即处理所有回调:使用func=lambda call: True,您可以处理所有回调请求,无论其内容如何。所以最好这样练习:
它将删除激活不可用按钮的能力。
或者您可以使其更加实用 - 编写一个处理程序,它会说:嘿,用户,该按钮不再可用,您为什么按下它?
如果您完全达到了正确逻辑的水平,那么您可以为不需要的按钮分配一个特定的按钮
data,并根据您的判断进行处理data。进一步关于您的消息,我想中断回调逻辑,但如何中断?
很简单:然后
"Поиск"做一个快速缓存,其本质就是快速保存用户和按钮的信息,或者更确切地说只是:然后你应用它来看看它是否需要改变,如果不需要,那么好吧,那么不需要,如果没有值或者它是True,那么它是必要的。
不过,为了更加简单和清晰,我建议您获取一个数据库,例如同步机器人的数据库(这实际上不是一个很好的数据库
SQLite,但它可以作为一个开始),然后您可以继续前进PostgreSQL,,和其他(好吧,或者为了存储缓存,您可以使用 Redis,但这当然是其他问题和答案的所有原因)。MySQLMongoDB我希望我能够帮助您解决您的问题,或者至少为您指出正确的思路。如果情况并非如此,请澄清评论中不清楚的内容。