zov-tech/bot/handlers/start.py
wasrusgen d6fbc3df13 bot: убрать reply-keyboard, оставить только inline buttons
Reply-keyboard web_app кнопки на мобиле тоже не передают initData
(или с issues). Оставляем только inline-кнопки в самом сообщении —
они работают на Telegram Desktop side-panel + на мобильных.

/start теперь шлёт ОДНО сообщение с inline-keyboard внутри.
/menu — пере-открыть выбор роли.
/hide — убрать reply-keyboard (если осталась от предыдущих версий).
2026-05-13 07:34:25 +03:00

105 lines
3.9 KiB
Python
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

import time
from aiogram import Router
from aiogram.filters import Command, CommandStart
from aiogram.types import (
InlineKeyboardButton,
InlineKeyboardMarkup,
Message,
ReplyKeyboardRemove,
WebAppInfo,
)
from config import Config
router = Router(name="start")
# ============================================================
# URL helpers
# ============================================================
def _bust_cache(url: str) -> str:
sep = "&" if "?" in url else "?"
return f"{url}{sep}t={int(time.time())}"
def _with_query(url: str, **params: str) -> str:
sep = "&" if "?" in url else "?"
pairs = "&".join(f"{k}={v}" for k, v in params.items() if v)
return f"{url}{sep}{pairs}" if pairs else url
def _wapp(miniapp_url: str, role: str) -> WebAppInfo:
return WebAppInfo(url=_bust_cache(_with_query(miniapp_url, role=role)))
# ============================================================
# Inline keyboard — единственный способ открыть MiniApp.
# Reply-кнопки с web_app не передают initData ни на Desktop side-panel,
# ни на мобильных. Inline-buttons открывают MiniApp в modal-режиме,
# где initData валидно передаётся на обеих платформах.
# ============================================================
def role_choice_inline(miniapp_url: str) -> InlineKeyboardMarkup:
return InlineKeyboardMarkup(
inline_keyboard=[
[
InlineKeyboardButton(text="👤 Я менеджер", web_app=_wapp(miniapp_url, "manager")),
InlineKeyboardButton(text="🏠 Я клиент", web_app=_wapp(miniapp_url, "client")),
],
[
InlineKeyboardButton(text="🔧 Я сотрудник", web_app=_wapp(miniapp_url, "staff")),
],
]
)
# ============================================================
# Commands
# ============================================================
@router.message(CommandStart())
async def cmd_start(message: Message, config: Config) -> None:
# Сразу убираем нижнюю reply-клавиатуру если она от предыдущих версий висит
await message.answer(
"👋 Здравствуйте, я бот-помощник от Руслана ВАСИЛЬЕВА.\n\n"
"Выберите, кто вы — кабинет откроется одним тапом.\n\n"
"<i>«Сотрудник» — для замерщиков и сборщиков ЗОВ.</i>",
reply_markup=role_choice_inline(config.miniapp_url),
)
@router.message(Command("menu"))
async def cmd_menu(message: Message, config: Config) -> None:
await message.answer(
"Выберите роль:",
reply_markup=role_choice_inline(config.miniapp_url),
)
@router.message(Command("hide"))
async def cmd_hide(message: Message) -> None:
await message.answer(
"Нижняя клавиатура убрана. Для выбора роли — /menu",
reply_markup=ReplyKeyboardRemove(),
)
# ============================================================
# /whoami — сотрудник присылает свой ID куратору, чтобы тот выдал роль
# ============================================================
@router.message(Command("whoami"))
async def cmd_whoami(message: Message) -> None:
user = message.from_user
if not user:
return
await message.answer(
f"<b>Ваш Telegram ID:</b> <code>{user.id}</code>\n"
f"Username: @{user.username or ''}\n"
f"Имя: {user.first_name or ''} {user.last_name or ''}".strip()
+ "\n\n"
"<i>Перешлите это сообщение куратору @wasrusgen чтобы вам выдали роль замерщика/сборщика.</i>"
)