2022-04-22 19:50:19 +02:00
|
|
|
// This example demonstrates how to deal with messages and callback queries
|
|
|
|
// within a single dialogue.
|
2022-04-22 19:42:50 +02:00
|
|
|
//
|
|
|
|
// # Example
|
|
|
|
// ```
|
|
|
|
// - /start
|
|
|
|
// - Let's start! What's your full name?
|
|
|
|
// - John Doe
|
|
|
|
// - Select a product:
|
|
|
|
// [Apple, Banana, Orange, Potato]
|
|
|
|
// - <A user selects "Banana">
|
|
|
|
// - John Doe, product 'Banana' has been purchased successfully!
|
|
|
|
// ```
|
|
|
|
|
|
|
|
use teloxide::{
|
2022-04-22 20:11:27 +02:00
|
|
|
dispatching::dialogue::{self, GetChatId, InMemStorage},
|
2022-04-22 19:42:50 +02:00
|
|
|
prelude::*,
|
|
|
|
types::{InlineKeyboardButton, InlineKeyboardMarkup},
|
|
|
|
utils::command::BotCommands,
|
|
|
|
};
|
|
|
|
|
|
|
|
type MyDialogue = Dialogue<State, InMemStorage<State>>;
|
|
|
|
type HandlerResult = Result<(), Box<dyn std::error::Error + Send + Sync>>;
|
|
|
|
|
|
|
|
#[derive(Clone)]
|
|
|
|
pub enum State {
|
|
|
|
Start,
|
|
|
|
ReceiveFullName,
|
2022-04-23 18:45:43 +02:00
|
|
|
ReceiveProductChoice { full_name: String },
|
2022-04-22 19:42:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for State {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self::Start
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(BotCommands, Clone)]
|
|
|
|
#[command(rename = "lowercase", description = "These commands are supported:")]
|
|
|
|
enum Command {
|
|
|
|
#[command(description = "display this text.")]
|
|
|
|
Help,
|
|
|
|
#[command(description = "start the purchase procedure.")]
|
|
|
|
Start,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::main]
|
|
|
|
async fn main() {
|
|
|
|
pretty_env_logger::init();
|
|
|
|
log::info!("Starting dialogue_bot...");
|
|
|
|
|
|
|
|
let bot = Bot::from_env().auto_send();
|
|
|
|
|
|
|
|
Dispatcher::builder(
|
|
|
|
bot,
|
2022-04-22 20:11:27 +02:00
|
|
|
dialogue::enter::<Update, InMemStorage<State>, State, _>()
|
2022-04-22 19:42:50 +02:00
|
|
|
.branch(
|
|
|
|
Update::filter_message()
|
2022-04-23 18:56:45 +02:00
|
|
|
.branch(teloxide::handler![State::ReceiveFullName].endpoint(receive_full_name))
|
|
|
|
.branch(dptree::entry().filter_command::<Command>().endpoint(handle_command))
|
|
|
|
.branch(dptree::endpoint(invalid_state)),
|
2022-04-22 19:42:50 +02:00
|
|
|
)
|
|
|
|
.branch(
|
|
|
|
Update::filter_callback_query().chain(
|
2022-04-23 18:45:43 +02:00
|
|
|
teloxide::handler![State::ReceiveProductChoice { full_name }]
|
2022-04-22 19:42:50 +02:00
|
|
|
.endpoint(receive_product_selection),
|
|
|
|
),
|
2022-04-23 18:56:45 +02:00
|
|
|
),
|
2022-04-22 19:42:50 +02:00
|
|
|
)
|
|
|
|
.dependencies(dptree::deps![InMemStorage::<State>::new()])
|
|
|
|
.build()
|
|
|
|
.setup_ctrlc_handler()
|
|
|
|
.dispatch()
|
|
|
|
.await;
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn handle_command(
|
|
|
|
bot: AutoSend<Bot>,
|
|
|
|
msg: Message,
|
|
|
|
cmd: Command,
|
|
|
|
dialogue: MyDialogue,
|
|
|
|
) -> HandlerResult {
|
|
|
|
match cmd {
|
|
|
|
Command::Help => {
|
|
|
|
bot.send_message(msg.chat.id, Command::descriptions().to_string()).await?;
|
|
|
|
}
|
|
|
|
Command::Start => {
|
|
|
|
bot.send_message(msg.chat.id, "Let's start! What's your full name?").await?;
|
|
|
|
dialogue.update(State::ReceiveFullName).await?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn receive_full_name(
|
|
|
|
bot: AutoSend<Bot>,
|
|
|
|
msg: Message,
|
|
|
|
dialogue: MyDialogue,
|
|
|
|
) -> HandlerResult {
|
|
|
|
match msg.text().map(ToOwned::to_owned) {
|
|
|
|
Some(full_name) => {
|
|
|
|
let products = InlineKeyboardMarkup::default().append_row(
|
|
|
|
vec!["Apple", "Banana", "Orange", "Potato"].into_iter().map(|product| {
|
|
|
|
InlineKeyboardButton::callback(product.to_owned(), product.to_owned())
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
|
|
|
|
bot.send_message(msg.chat.id, "Select a product:").reply_markup(products).await?;
|
2022-04-23 18:45:43 +02:00
|
|
|
dialogue.update(State::ReceiveProductChoice { full_name }).await?;
|
2022-04-22 19:42:50 +02:00
|
|
|
}
|
|
|
|
None => {
|
|
|
|
bot.send_message(msg.chat.id, "Please, send me your full name.").await?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn receive_product_selection(
|
|
|
|
bot: AutoSend<Bot>,
|
|
|
|
q: CallbackQuery,
|
|
|
|
dialogue: MyDialogue,
|
|
|
|
full_name: String,
|
|
|
|
) -> HandlerResult {
|
|
|
|
if let Some(product) = &q.data {
|
|
|
|
if let Some(chat_id) = q.chat_id() {
|
|
|
|
bot.send_message(
|
|
|
|
chat_id,
|
|
|
|
format!("{full_name}, product '{product}' has been purchased successfully!"),
|
|
|
|
)
|
|
|
|
.await?;
|
2022-04-23 19:52:43 +02:00
|
|
|
dialogue.exit().await?;
|
2022-04-22 19:42:50 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
async fn invalid_state(bot: AutoSend<Bot>, msg: Message) -> HandlerResult {
|
|
|
|
bot.send_message(msg.chat.id, "Unable to handle the message. Type /help to see the usage.")
|
|
|
|
.await?;
|
|
|
|
Ok(())
|
|
|
|
}
|