Реализация Gacha/Lootbox механики мобильной игры
Gacha — один из самых прибыльных механизмов в мобильных играх и одновременно один из самых регулируемых. Apple с 2022 года требует раскрытия вероятностей для всех lootbox-механик. В ряде стран (Бельгия, Нидерланды) gacha с реальными деньгами приравнивается к азартным играм. Правильная реализация — это баланс привлекательной механики, математически честных вероятностей и юридической прозрачности.
Математическая модель вероятностей
Базовая gacha — взвешенный случайный выбор из пула наград. Простая реализация:
data class GachaItem(val id: String, val rarity: Rarity, val weight: Int)
fun rollGacha(pool: List<GachaItem>): GachaItem {
val totalWeight = pool.sumOf { it.weight }
var random = Random.nextInt(totalWeight)
for (item in pool) {
random -= item.weight
if (random < 0) return item
}
return pool.last()
}
Пример распределения весов:
- Common: weight 1000 → ~58.8% шанс
- Rare: weight 500 → ~29.4%
- Epic: weight 150 → ~8.8%
- Legendary: weight 50 → ~2.9%
Суммарный вес: 1700. Шанс legendary: 50/1700 ≈ 2.94%.
Pity-система: обязательна для честной механики
Pity (гарантированная награда) — механика, которая гарантирует legendary после N неудачных попыток. Без pity у игрока может быть streak из 200 роллов без legendary — математически возможно, практически разрушает опыт.
Soft pity — начиная с N-го ролла вероятность legendary постепенно растёт. Genshin Impact использует этот подход: с 74-го ролла шанс 5* начинает расти на 6% каждый ролл.
Hard pity — ровно на N-м ролле legendary гарантирован. Проще реализовать и честнее коммуницировать игроку.
data class PlayerGachaPity(
val rollsSinceLastLegendary: Int,
val softPityStart: Int = 74,
val hardPityAt: Int = 90
)
fun calculateEffectiveProbability(baseProbability: Float, pity: PlayerGachaPity): Float {
if (pity.rollsSinceLastLegendary >= pity.hardPityAt) return 1.0f
if (pity.rollsSinceLastLegendary >= pity.softPityStart) {
val excess = pity.rollsSinceLastLegendary - pity.softPityStart
return minOf(1.0f, baseProbability + excess * 0.06f)
}
return baseProbability
}
Pity-счётчик хранится на сервере, не на клиенте. Клиентский счётчик обнуляется при переустановке — игрок теряет накопленный pity, что вызывает справедливое возмущение и чарджбеки.
Серверная генерация результатов
Результат гача-ролла нельзя генерировать на клиенте. Схема:
- Клиент отправляет запрос
POST /gacha/rollсuserId,gachaPoolId, количеством роллов, токеном оплаты - Сервер проверяет баланс/транзакцию, достаёт pity-счётчик, генерирует результат с seeded PRNG
- Сервер записывает результат, обновляет pity, возвращает
rollId+ результаты - Клиент отображает анимацию и показывает результаты
Seed для PRNG генерируется на сервере, клиент не может предсказать или повлиять на результат.
Анимация открытия
Анимация gacha-ролла — не просто украшение, это психологически значимый момент. Длительность 2–4 секунды, нарастающий саспенс перед reveal. Для мобильной реализации используем:
- Unity Animator или Spine для 2D-анимаций карт/яиц
- Particle System для эффектов при reveal редких наград
- Haptic feedback (UIImpactFeedbackGenerator на iOS, VibrationEffect на Android) в момент reveal
Multi-pull анимации (x10 ролл) требуют отдельной логики: показываем все результаты последовательно с возрастающим темпом, common быстро, legendary — с полноэкранным эффектом.
Регуляторные требования
App Store (с 2022): приложения с lootbox должны отображать вероятности «перед покупкой». Это не опциональная фича — это обязательное условие, нарушение которого ведёт к отклонению при ревью.
Google Play: аналогичное требование с 2024 года. Вероятности должны быть доступны в интерфейсе до подтверждения покупки.
Реализация: экран «Детали гача-пула» с таблицей вероятностей по типам наград. Этот экран открывается кнопкой «i» (информация) рядом с кнопкой ролла.
| Редкость | Вероятность | Гарантия (hard pity) |
|---|---|---|
| Common | 58.8% | — |
| Rare | 29.4% | — |
| Epic | 8.8% | 40 роллов |
| Legendary | 2.9% | 90 роллов |
Типы gacha-пулов
Rate-up banner — временный пул с повышенным шансом конкретного персонажа/предмета. Ограничен по времени (7–14 дней). Создаёт urgency и пики дохода.
Standard banner — постоянный пул без rate-up. Принимает «потерянные» pity-победы в играх с системой «гарантированный featured 50/50».
Featured banner с guarantee — если rate-up не выпал на 1-м legendary, следующий legendary guaranteed будет featured. Это «50/50 система» — хорошо коммуницируется игрокам и создаёт справедливое ожидание.
Сроки реализации: базовый gacha с pity и серверной генерацией — 3–4 дня. Полная система с несколькими типами баннеров, UI вероятностей, анимациями и аналитикой — 7–10 дней.







