Compare commits
4 Commits
088df817f3
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
6e27ce2fab | ||
|
|
5677b48227 | ||
|
|
4c6bfcf17e | ||
|
|
806b668a6d |
268
TRIAL-USERNAME-FIX.md
Normal file
268
TRIAL-USERNAME-FIX.md
Normal file
@@ -0,0 +1,268 @@
|
||||
# 🔒 Исправление: Имена user_... при trial подписке
|
||||
|
||||
## 📋 Описание проблемы
|
||||
|
||||
При оформлении **trial (пробной) подписки** username генерировался в формате `user_1770605873419` (timestamp) вместо реального имени пользователя.
|
||||
|
||||
## 🔍 Причина
|
||||
|
||||
**Это НЕ баг кода!** Telegram WebApp API не передает данные пользователя (`tgUser.id`, `tgUser.username`, `tgUser.first_name`) когда:
|
||||
|
||||
1. ❌ Приложение открыто **в браузере напрямую** (не через Telegram бота)
|
||||
2. ❌ Telegram WebApp **не инициализирован** (например, в dev режиме)
|
||||
3. ❌ Недостаточно **прав доступа** к Telegram API
|
||||
|
||||
### Старое поведение (до фикса):
|
||||
|
||||
```typescript
|
||||
// app/page.tsx (handleActivateTrial)
|
||||
telegramId: tgUser?.id || Date.now(), // ❌ Fallback на timestamp!
|
||||
```
|
||||
|
||||
**Результат**: Если `tgUser?.id` = undefined → username = `user_1770605873419` (таймстамп)
|
||||
|
||||
## ✅ Решение
|
||||
|
||||
Добавлена **обязательная валидация** Telegram ID на всех уровнях:
|
||||
|
||||
### 1. Frontend - app/page.tsx (Trial кнопка)
|
||||
|
||||
```typescript
|
||||
const tgUser = telegramWebApp?.initDataUnsafe?.user;
|
||||
|
||||
// 🔍 DEBUG логирование
|
||||
console.log('🎁 TRIAL ACTIVATION - Telegram data:', {
|
||||
hasWebApp: !!telegramWebApp,
|
||||
hasUser: !!tgUser,
|
||||
userId: tgUser?.id,
|
||||
username: tgUser?.username,
|
||||
firstName: tgUser?.first_name
|
||||
});
|
||||
|
||||
// ⚠️ БЛОКИРУЕМ создание без Telegram ID
|
||||
if (!tgUser?.id) {
|
||||
console.error('❌ TRIAL BLOCKED: No Telegram user data!');
|
||||
showToastNotification('❌ Откройте приложение через Telegram бота');
|
||||
return;
|
||||
}
|
||||
|
||||
// Теперь tgUser.id 100% есть
|
||||
const response = await fetch('/api/create-user', {
|
||||
body: JSON.stringify({
|
||||
planType: 'trial',
|
||||
telegramId: tgUser.id, // ✅ Без fallback!
|
||||
telegramUsername: tgUser?.username,
|
||||
firstName: tgUser?.first_name,
|
||||
})
|
||||
});
|
||||
```
|
||||
|
||||
### 2. Frontend - app/plans/page.tsx (3-step flow)
|
||||
|
||||
```typescript
|
||||
async function createUser(planType, period, locationIds) {
|
||||
const user = telegramWebApp?.initDataUnsafe?.user;
|
||||
|
||||
// ⚠️ БЛОКИРУЕМ без user.id
|
||||
if (!user?.id) {
|
||||
console.error('❌ USER CREATION BLOCKED: No Telegram user ID!');
|
||||
alert('❌ Откройте приложение через Telegram бота\n\nПриложение должно быть открыто в Telegram Mini App.');
|
||||
return;
|
||||
}
|
||||
|
||||
const requestBody = {
|
||||
telegramId: user.id, // ✅ 100% существует
|
||||
telegramUsername: user?.username || null,
|
||||
firstName: user?.first_name || null,
|
||||
};
|
||||
}
|
||||
```
|
||||
|
||||
### 3. Backend - app/api/create-user/route.ts
|
||||
|
||||
```typescript
|
||||
export async function POST(request: NextRequest) {
|
||||
const { telegramId, telegramUsername, firstName } = await request.json();
|
||||
|
||||
// ⚠️ СЕРВЕРНАЯ ВАЛИДАЦИЯ
|
||||
if (!telegramId) {
|
||||
logger.error('❌ VALIDATION FAILED: No telegramId provided');
|
||||
return NextResponse.json(
|
||||
{
|
||||
success: false,
|
||||
error: 'Telegram ID is required. Please open the app through Telegram bot.'
|
||||
},
|
||||
{ status: 400 }
|
||||
);
|
||||
}
|
||||
|
||||
// Генерация username с приоритетами
|
||||
let username: string;
|
||||
if (telegramUsername) {
|
||||
username = telegramUsername.toLowerCase().replace(/[^a-z0-9_]/g, '_');
|
||||
} else if (firstName && telegramId) {
|
||||
username = `${firstName.toLowerCase().replace(/[^a-z0-9]/g, '_')}_${telegramId}`;
|
||||
} else {
|
||||
username = `user_${telegramId}`; // ✅ Используем реальный ID, не timestamp
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
## 📊 Результат
|
||||
|
||||
### До фикса:
|
||||
- ❌ `user_1770605873419` (timestamp) - когда Telegram не передает данные
|
||||
- ❌ Создается подписка даже без Telegram ID
|
||||
- ❌ Невозможно отследить пользователя
|
||||
|
||||
### После фикса:
|
||||
- ✅ **БЛОКИРОВКА** создания если нет `tgUser.id`
|
||||
- ✅ Понятное сообщение: "Откройте приложение через Telegram бота"
|
||||
- ✅ Debug логи показывают причину блокировки
|
||||
- ✅ Все имена генерируются из **реальных Telegram данных**
|
||||
|
||||
## 🧪 Тестирование
|
||||
|
||||
### Правильный путь (через Telegram бота):
|
||||
|
||||
1. Открыть Telegram бот `@Chat_8n8_bot`
|
||||
2. Нажать "Открыть приложение" / `Start`
|
||||
3. Telegram WebApp передаст: `{ id: 123456789, username: "john_doe", first_name: "John" }`
|
||||
4. Username: `john_doe` ✅
|
||||
|
||||
### Неправильный путь (в браузере напрямую):
|
||||
|
||||
1. Открыть `https://app.umbrix.net` в Chrome
|
||||
2. Telegram WebApp недоступен: `window.Telegram?.WebApp = undefined`
|
||||
3. Попытка создать trial → **БЛОКИРОВКА** ❌
|
||||
4. Сообщение: "Откройте приложение через Telegram бота"
|
||||
|
||||
## 📝 Debug логи
|
||||
|
||||
При попытке создать trial без Telegram данных в консоли браузера:
|
||||
|
||||
```
|
||||
🎁 TRIAL ACTIVATION - Telegram data: {
|
||||
hasWebApp: false,
|
||||
hasInitData: false,
|
||||
hasUser: false,
|
||||
userId: undefined,
|
||||
username: undefined,
|
||||
firstName: undefined,
|
||||
fullData: undefined
|
||||
}
|
||||
❌ TRIAL BLOCKED: No Telegram user data!
|
||||
```
|
||||
|
||||
## 🚀 Production деплой
|
||||
|
||||
Изменения задеплоены:
|
||||
- ✅ Локальный билд: PASSED
|
||||
- ✅ Production билд: PASSED
|
||||
- ✅ PM2 перезапущен: pid 48353
|
||||
- ✅ Git коммит: `806b668`
|
||||
|
||||
## 🔑 Ключевые файлы
|
||||
|
||||
- [app/page.tsx](app/page.tsx) - handleActivateTrial (строки 188-245)
|
||||
- [app/plans/page.tsx](app/plans/page.tsx) - createUser (строки 171-225)
|
||||
- [app/api/create-user/route.ts](app/api/create-user/route.ts) - telegramId validation (строки 22-47)
|
||||
- [types/telegram.ts](types/telegram.ts) - TelegramWebApp интерфейс
|
||||
- [types/telegram.d.ts](types/telegram.d.ts) - Window.Telegram декларация
|
||||
- [hooks/useTelegramWebApp.ts](hooks/useTelegramWebApp.ts) - Telegram WebApp hook
|
||||
|
||||
## 💡 Важно!
|
||||
|
||||
Если пользователь видит имена `user_...` - это означает, что:
|
||||
1. Приложение открыто **не через Telegram бота**
|
||||
2. Telegram WebApp **не инициализирован**
|
||||
3. Нужно переоткрыть через `@Chat_8n8_bot` → "Открыть приложение"
|
||||
|
||||
**Теперь система автоматически блокирует такие попытки!** ✅
|
||||
|
||||
---
|
||||
|
||||
## 🔓 UPDATE: Исключение для n8n переходов (5677b48)
|
||||
|
||||
### Проблема после первого фикса:
|
||||
|
||||
После блокировки создания без Telegram ID - **перестали работать переходы из n8n**!
|
||||
|
||||
n8n workflow отправляет пользователей по **прямой ссылке** (открывается в браузере), где `window.Telegram?.WebApp` недоступен.
|
||||
|
||||
### Решение:
|
||||
|
||||
Добавлены **URL параметры-исключения** для n8n переходов:
|
||||
|
||||
```typescript
|
||||
// Проверяем источник перехода
|
||||
const urlParams = new URLSearchParams(window.location.search);
|
||||
const source = urlParams.get('source') || urlParams.get('utm_source');
|
||||
const fromN8n = source === 'n8n' || source === 'chat' || urlParams.has('from_n8n');
|
||||
|
||||
// Блокируем ТОЛЬКО если нет Telegram ID И это НЕ n8n переход
|
||||
if (!tgUser?.id && !fromN8n) {
|
||||
showToastNotification('❌ Откройте приложение через Telegram бота');
|
||||
return;
|
||||
}
|
||||
|
||||
// Для n8n переходов генерируем временный ID
|
||||
const userId = tgUser?.id || Date.now();
|
||||
```
|
||||
|
||||
### Поддерживаемые параметры:
|
||||
|
||||
1. `?source=n8n` - рекомендуемый
|
||||
2. `?utm_source=n8n` - для аналитики
|
||||
3. `?utm_source=chat` - альтернативный вариант
|
||||
4. `?from_n8n=true` - явное указание
|
||||
|
||||
### Примеры ссылок из n8n:
|
||||
|
||||
```
|
||||
✅ https://app.umbrix.net/?source=n8n
|
||||
✅ https://app.umbrix.net/plans?utm_source=chat
|
||||
✅ https://app.umbrix.net/?from_n8n=true&ref=user123
|
||||
❌ https://app.umbrix.net/ (без параметров - блокируется)
|
||||
```
|
||||
|
||||
### Итоговая логика:
|
||||
|
||||
| Способ входа | Telegram ID | URL параметры | Результат |
|
||||
|-------------|-------------|---------------|-----------|
|
||||
| Telegram Mini App | ✅ Есть | Любые | ✅ Создается с реальным ID |
|
||||
| n8n workflow | ❌ Нет | `?source=n8n` | ✅ Создается с временным ID |
|
||||
| Прямая ссылка | ❌ Нет | Нет параметров | ❌ **БЛОКИРОВКА** |
|
||||
|
||||
### Backend поддержка:
|
||||
|
||||
```typescript
|
||||
// app/api/create-user/route.ts
|
||||
const { telegramId, source } = await request.json();
|
||||
|
||||
// Разрешаем n8n переходы
|
||||
const fromN8n = source === 'n8n' || source === 'chat';
|
||||
|
||||
if (!telegramId && !fromN8n) {
|
||||
return NextResponse.json(
|
||||
{ success: false, error: 'Telegram ID required' },
|
||||
{ status: 400 }
|
||||
);
|
||||
}
|
||||
```
|
||||
|
||||
### Проверка в production:
|
||||
|
||||
```bash
|
||||
# 1. Переход через n8n (должен работать)
|
||||
curl -X POST https://app.umbrix.net/api/create-user \
|
||||
-H "Content-Type: application/json" \
|
||||
-d '{"planType":"trial","source":"n8n"}'
|
||||
|
||||
# 2. Переход без параметров (должен блокироваться)
|
||||
curl -X POST https://app.umbrix.net/api/create-user \
|
||||
-H "Content-Type: application/json" \
|
||||
-d '{"planType":"trial"}'
|
||||
```
|
||||
|
||||
**Деплой**: PM2 перезапущен (pid 50711), коммит `5677b48` ✅
|
||||
@@ -29,7 +29,8 @@ export async function POST(request: NextRequest) {
|
||||
telegramUsername,
|
||||
firstName,
|
||||
lastName,
|
||||
referrerId
|
||||
referrerId,
|
||||
source // Источник перехода (n8n, chat и т.д.)
|
||||
} = await request.json();
|
||||
|
||||
logger.debug('📥 Received data:', {
|
||||
@@ -40,9 +41,25 @@ export async function POST(request: NextRequest) {
|
||||
telegramUsername,
|
||||
firstName,
|
||||
lastName,
|
||||
referrerId
|
||||
referrerId,
|
||||
source
|
||||
});
|
||||
|
||||
// 🔓 ИСКЛЮЧЕНИЕ: Разрешаем n8n переходы даже без telegramId
|
||||
const fromN8n = source === 'n8n' || source === 'chat';
|
||||
|
||||
// ⚠️ ВАЛИДАЦИЯ: telegramId обязателен (кроме n8n переходов)
|
||||
if (!telegramId && !fromN8n) {
|
||||
logger.error('❌ VALIDATION FAILED: No telegramId provided');
|
||||
return NextResponse.json(
|
||||
{
|
||||
success: false,
|
||||
error: 'Telegram ID is required. Please open the app through Telegram bot.'
|
||||
},
|
||||
{ status: 400 }
|
||||
);
|
||||
}
|
||||
|
||||
// 1. Получаем токен админа
|
||||
const tokenResponse = await fetch(`${MARZBAN_API}/api/admin/token`, {
|
||||
method: 'POST',
|
||||
|
||||
29
app/page.tsx
29
app/page.tsx
@@ -202,16 +202,43 @@ export default function Home() {
|
||||
// Получаем реальные данные Telegram
|
||||
const tgUser = telegramWebApp?.initDataUnsafe?.user;
|
||||
|
||||
// 🔍 DEBUG: Проверяем данные Telegram при TRIAL подписке
|
||||
console.log('🎁 TRIAL ACTIVATION - Telegram data:', {
|
||||
hasWebApp: !!telegramWebApp,
|
||||
hasInitData: !!telegramWebApp?.initDataUnsafe,
|
||||
hasUser: !!tgUser,
|
||||
userId: tgUser?.id,
|
||||
username: tgUser?.username,
|
||||
firstName: tgUser?.first_name,
|
||||
fullData: tgUser
|
||||
});
|
||||
|
||||
// 🔓 ИСКЛЮЧЕНИЕ: Разрешаем создание для n8n переходов (когда есть source или utm параметры)
|
||||
const source = urlParams.get('source') || urlParams.get('utm_source');
|
||||
const fromN8n = source === 'n8n' || source === 'chat' || urlParams.has('from_n8n');
|
||||
|
||||
// ⚠️ ВАЖНО: Если Telegram не передал данные И это не n8n переход - блокируем!
|
||||
if (!tgUser?.id && !fromN8n) {
|
||||
console.error('❌ TRIAL BLOCKED: No Telegram user data!');
|
||||
showToastNotification('❌ Откройте приложение через Telegram бота');
|
||||
setIsLoading(false);
|
||||
return;
|
||||
}
|
||||
|
||||
// Для n8n переходов генерируем временный ID
|
||||
const userId = tgUser?.id || Date.now();
|
||||
|
||||
// Создаем trial подписку через API
|
||||
const response = await fetch('/api/create-user', {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify({
|
||||
planType: 'trial',
|
||||
telegramId: tgUser?.id || Date.now(),
|
||||
telegramId: userId, // Telegram ID или временный для n8n
|
||||
telegramUsername: tgUser?.username || undefined,
|
||||
firstName: tgUser?.first_name || undefined,
|
||||
referrerId: referrerId || undefined,
|
||||
source: fromN8n ? (source || 'n8n') : undefined, // Помечаем n8n переходы
|
||||
}),
|
||||
});
|
||||
|
||||
|
||||
@@ -188,15 +188,33 @@ export default function PlansNew() {
|
||||
} : null
|
||||
});
|
||||
|
||||
// 🔓 ИСКЛЮЧЕНИЕ: Проверяем переход из n8n
|
||||
const urlParams = new URLSearchParams(window.location.search);
|
||||
const source = urlParams.get('source') || urlParams.get('utm_source');
|
||||
const fromN8n = source === 'n8n' || source === 'chat' || urlParams.has('from_n8n');
|
||||
|
||||
// ⚠️ ВАЖНО: Если Telegram не передал user.id И это не n8n переход - блокируем
|
||||
if (!user?.id && !fromN8n) {
|
||||
console.error('❌ USER CREATION BLOCKED: No Telegram user ID!');
|
||||
alert('❌ Откройте приложение через Telegram бота\n\nПриложение должно быть открыто в Telegram Mini App, а не в браузере.');
|
||||
setStep('plan');
|
||||
setIsCreatingUser(false);
|
||||
return;
|
||||
}
|
||||
|
||||
// Для n8n переходов генерируем временный ID
|
||||
const userId = user?.id || Date.now();
|
||||
|
||||
const requestBody = {
|
||||
planType,
|
||||
period,
|
||||
locationIds,
|
||||
telegramId: user?.id || null,
|
||||
telegramId: userId, // Telegram ID или временный для n8n
|
||||
telegramUsername: user?.username || null,
|
||||
firstName: user?.first_name || null,
|
||||
lastName: user?.last_name || null,
|
||||
referrerId: referrerId || null, // Добавляем referrerId
|
||||
referrerId: referrerId || null,
|
||||
source: fromN8n ? (source || 'n8n') : null, // Помечаем n8n переходы
|
||||
};
|
||||
|
||||
console.log('📤 REQUEST BODY:', requestBody);
|
||||
|
||||
@@ -3,147 +3,6 @@
|
||||
|
||||
import { useEffect, useState } from 'react';
|
||||
|
||||
declare global {
|
||||
interface Window {
|
||||
Telegram?: {
|
||||
WebApp: {
|
||||
ready: () => void;
|
||||
expand: () => void;
|
||||
close: () => void;
|
||||
isExpanded: boolean;
|
||||
viewportHeight: number;
|
||||
viewportStableHeight: number;
|
||||
headerColor: string;
|
||||
backgroundColor: string;
|
||||
isClosingConfirmationEnabled: boolean;
|
||||
platform: string;
|
||||
version: string;
|
||||
|
||||
MainButton: {
|
||||
text: string;
|
||||
color: string;
|
||||
textColor: string;
|
||||
isVisible: boolean;
|
||||
isActive: boolean;
|
||||
isProgressVisible: boolean;
|
||||
setText: (text: string) => void;
|
||||
onClick: (callback: () => void) => void;
|
||||
offClick: (callback: () => void) => void;
|
||||
show: () => void;
|
||||
hide: () => void;
|
||||
enable: () => void;
|
||||
disable: () => void;
|
||||
showProgress: (leaveActive?: boolean) => void;
|
||||
hideProgress: () => void;
|
||||
setParams: (params: { text?: string; color?: string; text_color?: string; is_active?: boolean; is_visible?: boolean }) => void;
|
||||
};
|
||||
|
||||
BackButton: {
|
||||
isVisible: boolean;
|
||||
onClick: (callback: () => void) => void;
|
||||
offClick: (callback: () => void) => void;
|
||||
show: () => void;
|
||||
hide: () => void;
|
||||
};
|
||||
|
||||
HapticFeedback: {
|
||||
impactOccurred: (style: 'light' | 'medium' | 'heavy' | 'rigid' | 'soft') => void;
|
||||
notificationOccurred: (type: 'error' | 'success' | 'warning') => void;
|
||||
selectionChanged: () => void;
|
||||
};
|
||||
|
||||
setHeaderColor: (color: string) => void;
|
||||
setBackgroundColor: (color: string) => void;
|
||||
enableClosingConfirmation: () => void;
|
||||
disableClosingConfirmation: () => void;
|
||||
|
||||
showPopup: (params: {
|
||||
title?: string;
|
||||
message: string;
|
||||
buttons: Array<{ id?: string; type: string; text?: string }>;
|
||||
}, callback?: (buttonId: string) => void) => void;
|
||||
|
||||
showAlert: (message: string, callback?: () => void) => void;
|
||||
showConfirm: (message: string, callback?: (confirmed: boolean) => void) => void;
|
||||
|
||||
openLink: (url: string, options?: { try_instant_view?: boolean }) => void;
|
||||
openTelegramLink: (url: string) => void;
|
||||
openInvoice: (url: string, callback?: (status: string) => void) => void;
|
||||
|
||||
shareToStory: (media_url: string, params?: {
|
||||
text?: string;
|
||||
widget_link?: {
|
||||
url: string;
|
||||
name?: string;
|
||||
};
|
||||
}) => void;
|
||||
|
||||
switchInlineQuery: (query: string, choose_chat_types?: string[]) => void;
|
||||
|
||||
sendData: (data: string) => void;
|
||||
|
||||
initData: string;
|
||||
initDataUnsafe: {
|
||||
query_id?: string;
|
||||
user?: {
|
||||
id: number;
|
||||
first_name: string;
|
||||
last_name?: string;
|
||||
username?: string;
|
||||
language_code?: string;
|
||||
is_premium?: boolean;
|
||||
photo_url?: string;
|
||||
};
|
||||
receiver?: {
|
||||
id: number;
|
||||
first_name: string;
|
||||
last_name?: string;
|
||||
username?: string;
|
||||
language_code?: string;
|
||||
is_premium?: boolean;
|
||||
};
|
||||
chat?: {
|
||||
id: number;
|
||||
type: string;
|
||||
title: string;
|
||||
username?: string;
|
||||
photo_url?: string;
|
||||
};
|
||||
chat_type?: string;
|
||||
chat_instance?: string;
|
||||
start_param?: string;
|
||||
can_send_after?: number;
|
||||
auth_date: number;
|
||||
hash: string;
|
||||
};
|
||||
|
||||
themeParams: {
|
||||
bg_color?: string;
|
||||
text_color?: string;
|
||||
hint_color?: string;
|
||||
link_color?: string;
|
||||
button_color?: string;
|
||||
button_text_color?: string;
|
||||
secondary_bg_color?: string;
|
||||
header_bg_color?: string;
|
||||
accent_text_color?: string;
|
||||
section_bg_color?: string;
|
||||
section_header_text_color?: string;
|
||||
subtitle_text_color?: string;
|
||||
destructive_text_color?: string;
|
||||
};
|
||||
|
||||
colorScheme: 'light' | 'dark';
|
||||
|
||||
isVersionAtLeast: (version: string) => boolean;
|
||||
|
||||
onEvent: (eventType: string, eventHandler: () => void) => void;
|
||||
offEvent: (eventType: string, eventHandler: () => void) => void;
|
||||
};
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
export function useTelegramWebApp() {
|
||||
const [webApp, setWebApp] = useState<any>(null);
|
||||
const [isReady, setIsReady] = useState(false);
|
||||
@@ -152,7 +11,7 @@ export function useTelegramWebApp() {
|
||||
// Проверяем доступность Telegram WebApp только на клиенте
|
||||
if (typeof window === 'undefined') return;
|
||||
|
||||
const tg = window.Telegram?.WebApp;
|
||||
const tg = (window as any).Telegram?.WebApp;
|
||||
if (!tg) {
|
||||
console.warn('[TelegramWebApp] Not running in Telegram');
|
||||
setIsReady(true); // Всё равно помечаем как ready для dev
|
||||
|
||||
16
types/telegram.d.ts
vendored
Normal file
16
types/telegram.d.ts
vendored
Normal file
@@ -0,0 +1,16 @@
|
||||
/**
|
||||
* Global Telegram WebApp type declarations
|
||||
* Расширяет Window interface для доступа к Telegram WebApp API
|
||||
*/
|
||||
|
||||
import { TelegramWebApp } from './telegram';
|
||||
|
||||
declare global {
|
||||
interface Window {
|
||||
Telegram?: {
|
||||
WebApp: TelegramWebApp;
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
export {};
|
||||
@@ -46,6 +46,39 @@ export interface TelegramWebApp {
|
||||
showPopup(params: any, callback?: () => void): void;
|
||||
onEvent(eventType: string, callback: () => void): void;
|
||||
offEvent(eventType: string, callback: () => void): void;
|
||||
|
||||
// Additional methods used in useTelegramWebApp
|
||||
setHeaderColor(color: string): void;
|
||||
setBackgroundColor(color: string): void;
|
||||
enableClosingConfirmation(): void;
|
||||
disableClosingConfirmation(): void;
|
||||
|
||||
MainButton: {
|
||||
text: string;
|
||||
isVisible: boolean;
|
||||
isActive: boolean;
|
||||
setText(text: string): void;
|
||||
onClick(callback: () => void): void;
|
||||
offClick(callback: () => void): void;
|
||||
show(): void;
|
||||
hide(): void;
|
||||
enable(): void;
|
||||
disable(): void;
|
||||
};
|
||||
|
||||
BackButton: {
|
||||
isVisible: boolean;
|
||||
onClick(callback: () => void): void;
|
||||
offClick(callback: () => void): void;
|
||||
show(): void;
|
||||
hide(): void;
|
||||
};
|
||||
|
||||
HapticFeedback: {
|
||||
impactOccurred(style: 'light' | 'medium' | 'heavy' | 'rigid' | 'soft'): void;
|
||||
notificationOccurred(type: 'error' | 'success' | 'warning'): void;
|
||||
selectionChanged(): void;
|
||||
};
|
||||
}
|
||||
|
||||
export interface TelegramUserData {
|
||||
|
||||
Reference in New Issue
Block a user