mirror of
https://github.com/python-telegram-bot/python-telegram-bot.git
synced 2025-01-07 02:59:57 +01:00
dea24bcb7c
* pollbot.py example was sending the poll to the effective_user and not effective_chat... well, it's a poll to be answered by multiple uses. * Use User.mention_html() shortcut Co-authored-by: Leonardo <leonardo.rezende@trt19.jus.br> Co-authored-by: Hinrich Mahler <hinrich.mahler@freenet.de>
146 lines
6.1 KiB
Python
146 lines
6.1 KiB
Python
#!/usr/bin/env python
|
|
# -*- coding: utf-8 -*-
|
|
# This program is dedicated to the public domain under the CC0 license.
|
|
|
|
"""
|
|
Basic example for a bot that works with polls. Only 3 people are allowed to interact with each
|
|
poll/quiz the bot generates. The preview command generates a closed poll/quiz, excatly like the
|
|
one the user sends the bot
|
|
"""
|
|
import logging
|
|
|
|
from telegram import (Poll, ParseMode, KeyboardButton, KeyboardButtonPollType,
|
|
ReplyKeyboardMarkup, ReplyKeyboardRemove)
|
|
from telegram.ext import (Updater, CommandHandler, PollAnswerHandler, PollHandler, MessageHandler,
|
|
Filters)
|
|
|
|
logging.basicConfig(format='%(asctime)s - %(name)s - %(levelname)s - %(message)s',
|
|
level=logging.INFO)
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
def start(update, context):
|
|
"""Inform user about what this bot can do"""
|
|
update.message.reply_text('Please select /poll to get a Poll, /quiz to get a Quiz or /preview'
|
|
' to generate a preview for your poll')
|
|
|
|
|
|
def poll(update, context):
|
|
"""Sends a predefined poll"""
|
|
questions = ["Good", "Really good", "Fantastic", "Great"]
|
|
message = context.bot.send_poll(update.effective_chat.id, "How are you?", questions,
|
|
is_anonymous=False, allows_multiple_answers=True)
|
|
# Save some info about the poll the bot_data for later use in receive_poll_answer
|
|
payload = {message.poll.id: {"questions": questions, "message_id": message.message_id,
|
|
"chat_id": update.effective_chat.id, "answers": 0}}
|
|
context.bot_data.update(payload)
|
|
|
|
|
|
def receive_poll_answer(update, context):
|
|
"""Summarize a users poll vote"""
|
|
answer = update.poll_answer
|
|
poll_id = answer.poll_id
|
|
try:
|
|
questions = context.bot_data[poll_id]["questions"]
|
|
# this means this poll answer update is from an old poll, we can't do our answering then
|
|
except KeyError:
|
|
return
|
|
selected_options = answer.option_ids
|
|
answer_string = ""
|
|
for question_id in selected_options:
|
|
if question_id != selected_options[-1]:
|
|
answer_string += questions[question_id] + " and "
|
|
else:
|
|
answer_string += questions[question_id]
|
|
context.bot.send_message(context.bot_data[poll_id]["chat_id"],
|
|
"{} feels {}!".format(update.effective_user.mention_html(),
|
|
answer_string),
|
|
parse_mode=ParseMode.HTML)
|
|
context.bot_data[poll_id]["answers"] += 1
|
|
# Close poll after three participants voted
|
|
if context.bot_data[poll_id]["answers"] == 3:
|
|
context.bot.stop_poll(context.bot_data[poll_id]["chat_id"],
|
|
context.bot_data[poll_id]["message_id"])
|
|
|
|
|
|
def quiz(update, context):
|
|
"""Send a predefined poll"""
|
|
questions = ["1", "2", "4", "20"]
|
|
message = update.effective_message.reply_poll("How many eggs do you need for a cake?",
|
|
questions, type=Poll.QUIZ, correct_option_id=2)
|
|
# Save some info about the poll the bot_data for later use in receive_quiz_answer
|
|
payload = {message.poll.id: {"chat_id": update.effective_chat.id,
|
|
"message_id": message.message_id}}
|
|
context.bot_data.update(payload)
|
|
|
|
|
|
def receive_quiz_answer(update, context):
|
|
"""Close quiz after three participants took it"""
|
|
# the bot can receive closed poll updates we don't care about
|
|
if update.poll.is_closed:
|
|
return
|
|
if update.poll.total_voter_count == 3:
|
|
try:
|
|
quiz_data = context.bot_data[update.poll.id]
|
|
# this means this poll answer update is from an old poll, we can't stop it then
|
|
except KeyError:
|
|
return
|
|
context.bot.stop_poll(quiz_data["chat_id"], quiz_data["message_id"])
|
|
|
|
|
|
def preview(update, context):
|
|
"""Ask user to create a poll and display a preview of it"""
|
|
# using this without a type lets the user chooses what he wants (quiz or poll)
|
|
button = [[KeyboardButton("Press me!", request_poll=KeyboardButtonPollType())]]
|
|
message = "Press the button to let the bot generate a preview for your poll"
|
|
# using one_time_keyboard to hide the keyboard
|
|
update.effective_message.reply_text(message,
|
|
reply_markup=ReplyKeyboardMarkup(button,
|
|
one_time_keyboard=True))
|
|
|
|
|
|
def receive_poll(update, context):
|
|
"""On receiving polls, reply to it by a closed poll copying the received poll"""
|
|
actual_poll = update.effective_message.poll
|
|
# Only need to set the question and options, since all other parameters don't matter for
|
|
# a closed poll
|
|
update.effective_message.reply_poll(
|
|
question=actual_poll.question,
|
|
options=[o.text for o in actual_poll.options],
|
|
# with is_closed true, the poll/quiz is immediately closed
|
|
is_closed=True,
|
|
reply_markup=ReplyKeyboardRemove()
|
|
)
|
|
|
|
|
|
def help_handler(update, context):
|
|
"""Display a help message"""
|
|
update.message.reply_text("Use /quiz, /poll or /preview to test this "
|
|
"bot.")
|
|
|
|
|
|
def main():
|
|
# Create the Updater and pass it your bot's token.
|
|
# Make sure to set use_context=True to use the new context based callbacks
|
|
# Post version 12 this will no longer be necessary
|
|
updater = Updater("TOKEN", use_context=True)
|
|
dp = updater.dispatcher
|
|
dp.add_handler(CommandHandler('start', start))
|
|
dp.add_handler(CommandHandler('poll', poll))
|
|
dp.add_handler(PollAnswerHandler(receive_poll_answer))
|
|
dp.add_handler(CommandHandler('quiz', quiz))
|
|
dp.add_handler(PollHandler(receive_quiz_answer))
|
|
dp.add_handler(CommandHandler('preview', preview))
|
|
dp.add_handler(MessageHandler(Filters.poll, receive_poll))
|
|
dp.add_handler(CommandHandler('help', help_handler))
|
|
|
|
# Start the Bot
|
|
updater.start_polling()
|
|
|
|
# Run the bot until the user presses Ctrl-C or the process receives SIGINT,
|
|
# SIGTERM or SIGABRT
|
|
updater.idle()
|
|
|
|
|
|
if __name__ == '__main__':
|
|
main()
|