Сетевой игровой движок для игр в жанре action
, rpg
, rts
, tbs
, tower defense
с видом сверху/изометрией.
Движок не является коробочным решением готовым к использованию и не является конкурентом нормальным движкам. :)
Движок является серверной частью игры решающую вопросы:
Движок не решает вопросы графики и звука, это лежит на плечах клиента. В данном случае реализован клиент на игровом
движке Phaser3, в роли клиента может выступать все что умеет в веб сокеты. (или вообще
любые
сокеты с небольшой переделкой api). Например, хорошим клиентом может стать Ebitengine.
Игра "Veliri" (Session/MMO/Action) разработана с помощью этого движка:
(ссылка на видео)
Как работать с движком:
- форкаете
- работаете
Немного об архитектуре:
Сервер состоит из 3х частей:
Роутер
- является посредником между клиентом и нодой. Хранит в себе все игровые сущности и библиотеку с математикой,
поэтому импортируется как библиотека в ноду.
Нода
- это сервис где происходит сама "игра", но 1 ноде может быть много игор.
Клиент
- та часть, что видит игрок, является "терминалом" который просто выводит происходящие на экран с помощью
графического движка.
Для работы сервера необходимо поднять 1 роутер и хотя бы 1 ноду. К одному роутеру может быть подключено много нод что
позволяет горизонтально масштабировать игру.
- Роутер и нода общаются между собой по rpc.
- Роутер и клиент общаются через websocket.
Как подключается нода:
- когда нода запускается
она, стучится по rpc
на
veliriURL
указанный в main.ini
- дополнительным аргументом передает параметр
nodeUrl
указанный в main.ini (этот аргумент передает ip:
port машины на которой запущена нода)
- если соединение успешно,
то роутер ловит запрос, и
регистрирует ноду
в сторедже нод,
дает ему uuid и отсылает как ответ.
- при добавлении ноды в сторедж, роутер
поднимает обратный rpc канал
на ноду для полнодуплексной связи.
- когда оба канала открыты, на ноде будут запускаться новые сессии. В случае если на ноде происходит ошибка, или ошибка
при передаче, или отвал по таймауту то нода удаляется из стореджа.
API ноды
Для каждой ноды на роутере есть свое апи выраженные как методы ноды. Посмотреть
можно тут
, нода ловит эти запросы тут.
CreateBattle
- создает новую сессию
FindBattle
- ищет бой по uuid
InitBattle
- когда бой создан, игрок запрашивает его состояние при загрузке на клиенте
StartLoad
- состояние было успешно получено и игрок запрашивает все обьекты в игре.
Input
- ввод игрока (клава/мышь)
CreateUnit
- создает юнита игроку инициатору
CreateBot
- создает бота в конкретную команду
CreateObj
- создает объект (турель), в команду
Что посылает нода
Каждая нода отправляет на роутер
данные обновление мира для каждого активного боя. Роутер
их перенаправляет клиентам по
веб сокетам.
Маршрутизация сообщений для игроков на сокетах происходит
вот тут
Как запустить:
go mod tidy;
go run ./router/main.go;
go run ./node/main.go;
cd .\static\;
npm run dev;
#заходим http://localhost:8083/
## ИЛИ
cd .\static\;
npm run build;
#заходим http://localhost:8086/
Настройка сети на стороне клиента находится
вот тут
роутер ловит сообщения сокета вот тут
----
Сессия, игровое поле, GameLoop
Сессию определяет
объектBattle
,
но все игровые объекты прикрепляются к
объекту карты (Map
)
который встроен в Battle
.
Все игровые объекты находящиеся на карте (юниты, пули, строения) привязываются к ней
полем MapID
(например)
.
Юниты
и пули хранятся в
отдельных стореджах, при добавлении в сторедж,
он смотрит на
поле MapID
и кладет в соответствующий массив.
Строения же находятся в
карте карте (Map
)
.
type Map struct {
// Размер карты в пикселях (все дальности в игре измеряются в пикселях), в идеале квадрат, прямоугольник не тестировался)
XSize int `json:"x_size"`
YSize int `json:"y_size"`
// текстуры земли, не на что не влияют, нужны только для отрисовки на клиенте.
Flore map[int]map[int]*dynamic_map_object.Flore `json:"flore"`
// Не изменяемые объекты (например горы и овраги), игрок видит эти объекты всегда независимо от радара/обзора и они никогда не изменяются
StaticObjects map[int]*dynamic_map_object.Object `json:"-"`
// Тут находятся игровые объекты с которыми можно взаимодействовать (убить, построить, передвинуть и тд.), та же эти обьекты могут иметь поведение (например турели).
// Эти объекты игрок видит только когда открыл их в тумане войны. Когда объект ушел обратно в туман игрок запоминает его расположение и состояние.
// Игрок не видит изменения если с объектом вне поле его зрения.
DynamicObjects []*dynamic_map_object.Object `json:"-"`
// Карта не плоская и у каждой клетке 16x16px есть своя высота (в текущей реализации это влияет только на пули), если она указана то хранится тут если нет то используется DefaultLevel
LevelMap [][]*LvlMap `json:"level_map"`
// высота карты по умолчанию
DefaultLevel float64 `json:"default_level"`
// Кеширование непроходимых участков из-за объектов на карте для ускорения расчета коллизий. Подробнее смотри раздел коллизий.
GeoZones [][]*Zone `json:"-"`
}
Создание сессии и GameLoop
Создание сессии происходит на ноде при соответствующем запросе
по rpc.
Когда сессия успешно создана, она попадает
в сторедж
всех сессий на ноде.
Что бы сессия попала в GameLoop
,
специальный метод смотрит все
сессии и если она не инициализирована то запускает её.
GameLoop
- это игровой цикл который отслеживает и запускает все игровые механизмы (движения объектов, расчеты физики,
обзора,
нанесения урона, пользовательский ввод и все такое). Одна итерация GameLoop
это 1 кадр на стороне сервера, время этого
кадра ровняется _const.ServerTick в мс.
Если итерация отработала
быстрее _const.ServerTick то она
вычисляет дельту и спит это время, если дольше то все лагают.
Во время работы итерации собираются сообщения об изменениях или событиях в специальный
объект web_socket.MessagesStore{}
.
В конце работы итерации всем
игрокам отсылаются сообщения
предварительно попуская через фильтр видимости.
Подробнее о протоколе и способе обмена данными в
разделе Бинарный протокол связи между серверов и клиентом
Реализация игровых объектов: "юниты", "строения", "пули"
"Юниты", "строения", "пули" - это объекты которые находятся на карте, имеют физическую модель и обрабатываются
в GameLoop
.
Юниты
Юниты - это объекты, которые перемещаются по карте своим ходом. В игровое представление может быть как танк, так и
человек.
Характеристики юнита определяет
его "тело",
представляет собой куб (длинна, ширина, высота). Параметр Radius
для быстрого расчетов где не нужна
точность. Тело юнита имеет дальность обзора, слоты под оружие и тип шасси который определяет механику движения.
красный квадрат это длинна, ширина. Синий круг Radius
. Высота влияет на пули.
type Body struct {
Texture string `json:"texture"` // текстура для отрисовки на клиенте.
MaxHP int `json:"max_hp"`
Scale int `json:"scale"` // сервер заточен на текстуры размером 128х128px, это параметр говорит % от этого размера. Этот параметр влияет на все параметры габаритов и якорей для оружия (подробнее в разделе оружия).
Length int `json:"length"` // длинна полигона (Length * (Scale/100))
Width int `json:"width"` // ширина полигона (Width * (Scale/100))
Height int `json:"height"` // высота полигона (Height * (Scale/100))
Radius int `json:"radius"` // радиус корпуса для простого расчета колизи в поиске пути
RangeView int `json:"range_view"` // дальность видимости
RangeRadar int `json:"range_radar"` // дальность радара
Weapons map[int]*WeaponSlot `json:"weapons"` // слоты и якоря для оружия (подробнее в разделе оружия)
ChassisType string `json:"chassis_type"` // тип шасси, в текущий реализации есть только гусеницы. Можно реализовать свою механику перемещания. (например антиграв, колеса, полет)
Speed float64 `json:"speed"` // максимальная скорость движения (W)
ReverseSpeed float64 `json:"-"` // максимальная скорость движения назад (S)
PowerFactor float64 `json:"power_factor"`// сила ускорения
ReverseFactor float64 `json:"-"` // сила ускорения назад (тормоз)
TurnSpeed float64 `json:"turn_speed"` // скорость поворота
MoveDrag float64 `json:"move_drag"` // сила трения при движение
AngularDrag float64 `json:"-"` // сила трения при повороте
Weight float64 `json:"-"` // вес (влияет на столкновения)
}
Параметры связанные с
физикой перетекают
в
структуру физической модели
при первом обращение к ней.
Строения
Строения-это объекты, которые не могут самостоятельно перемещаться или не могут перемещаться вообще. У строений нет "
тела"
как у юнитов. Их физическую модель
представляет массив
из
непроходимых точек
.
При инициализации или движение обьекта геодата собирается в
методе SetGeoData
и после регестрируется
в Map.GeoZones
Некоторые строения, как и юниты могут иметь дальность обзора, оружие
и поведение
.
Общее (Юниты/Строения)
В строение и юнитов встроены общие структуры и интерфейсы:
{
physicalModel *physical_model.PhysicalModel // физическое представление объекта (именно с этой структурой происходит движение, столкновения, пападания пуль)
gunner *gunner.Gunner // интерфейс для использования оружия
BurstOfShots *burst_of_shots.BurstOfShots // очередь пуль которые вытеют из ствола, у оружия может быть несколько стволов и стрельба очередью
weaponTarget *target.Target // цель для атаки
visibleObjects *visible_objects.VisibleObjectsStore // обьекты которые "видит", в текущей реализации встраиваается экземляр "комманды"
}
Движение юнитов и строений обрабатывается
в GameLoop
одним методом
, стрельба
и поворот орудий тоже.
Пули
Пуля в игре
представляет собой шарик
радиусом _const.AmmoRadius, физика пули
работает на
формулах "Полета снаряда" (реализация)
. В зависимости от типа
снаряда (пуля
, лазер,
ракеты) может немного
менять поведение.
Пули в GameLoop
обрабатываются тут
Бинарный протокол связи между серверов и клиентом
Сборка сообщений в GameLoop
Во время итерации GameLoop
собираются сообщения которые после будут обработаны и отправлены в роутер.
Сообщения собираются в
объект MessagesStore, методом AddMsg(typeMsg, typeCheck string, msg web_socket_response.Response, attributes map[string]string)
.
Функция принимает в себя аргументами:
typeMsg
- тип сообщения, сообщения одного типа объединяются в 1 длинное сообщение.
typeCheck
- определяет функцию, которая будет обрабатывать сообщение.
msg
само сообщение, состоит из 2‑х уровней:
- Метаданные для обработки (например
Х
, Y
для того что бы проверить видит это сообщение игрок или нет).
- Сами данные которые будут отправлены игроку, могут быть как бинарные (поле:
BinaryMsg
), так и json (
поле: Data
).
Если BinaryMsg != nil, то считается что это бинарное сообщение и отправится только BinaryMsg
, иначе
отправится Data
.
attributes
- какие-то доп атрибуты, которые не уместились в других объектах.
В GameLoop
большинство сообщений являются бинарными и добавляются примерно одинаково:
MessagesStore.AddMsg("fireMsgs", "bin", web_socket_response.Response{
BinaryMsg: binary_msg.CreateFireGunBinaryMsg(m.TypeID, m.X, m.Y, m.Z, m.Rotate, m.AccumulationPercent),
X: m.X,
Y: m.Y,
}, nil)
Бинарное сообщение выглядит как набор байт, где 1 байт определяет команду. Остальные байты это информация самого
сообщения. Порядок и значений байт строго определены в сообщение. Все сообщения собираются
вот тут.
Структура сообщения выглядит примерно так:
[1[eventID], 4[unitID], 4[speed], 4[x], 4[y], 4[z]
- логическое представление
[0||0,0,0,0||0,0,0,0||0,0,0,0||0,0,0,0||0,0,0,0]
- массив (это 1 массив) байт который видит машина
- 1[eventID] - 1 байт, eventID - ид команды.
- 4[unitID] - 4 байта, unitID - ид юнита.
- и тд.
Обработка сообщений в GameLoop
Когда все сообщения
собранны, GameLoop
формирует пачки сообщений для каждого игрока
. Это нужно для того что бы убирать сообщения,
которые не видит
игрок из-за тумана войны или другой причине.
Когда для игрока формируется пачка сообщений они объединяются
в одно сообщение.
Важно что сообщения одного типа идут друг за другом, а не в разнобой, это позволяет указать длину сообщения одного типа,
и размазать его на клиента на много мелких сообщений.
Результатом формирования будет 1 большой сообщение со всеми изменениями мира для 1‑го игрока.
Примерно такого вида:
// [1[eventID],
// 4[data_size], data[data],
// ...
// 4[data_size], data[data],
- 1[eventID] - 1 байт, в данном случае это 100, говорит что это большой пакет данных который надо разобрать.
- 4[data_size] - размер всех сообщений 1‑го типа, то есть это например 100 сообщений типа
fireMsg
, которе в свою
очередь тоже будится биться на мелки и уже дальше обрабатываться
- ... - и так для каждого типа сообщений.
На клиенте эти сообщений можно
обрабатывать так:
function parseMegaPackData(data, store) {
let unitMoveSize = intFromBytes(data.slice(1, 5))
let stopByte = 5 + unitMoveSize;
BinaryReader(data.subarray(5, 5 + unitMoveSize), store)
for (; stopByte < data.length;) {
let subData = intFromBytes(data.slice(stopByte, stopByte + 4))
stopByte = 4 + stopByte;
BinaryReader(data.subarray(stopByte, stopByte + subData), store)
stopByte = subData + stopByte;
}
}
Ввод пользователя в основном ложится на плечи клиента. Клиент этот ввод должен проксировать по веб сокетам.
Например, так:
ws.send(JSON.stringify({
event: "i",
service: "battle",
select_units: gameStore.selectUnits,
w: scene.wasd.up.isDown,
a: scene.wasd.left.isDown,
s: scene.wasd.down.isDown,
d: scene.wasd.right.isDown,
x: x, // mouse x
y: y, // mouse y
fire: fire, // press left mouse
}));
Сервер принимает это сообщение и
проксирует его на ноду, которая кладет
все параметры в
объект WASD
.
Дальше в GameLoop
по этому объекту проверяется какие действия выполнять.
Баллистика, стрельба, слоты для оружия
Оружие в игре состоим из 3‑х объектов:
- Слот под оружие - является якорем куда будет прикреплен спрайт и от которого будет рассчитываться стартовая позиция
пули.
- Оружие - определяет тип оружия (баллистика, лазер, ракеты) и дополнительные параметры например углы атаки,
максимальная дальность, точность и тд.
- Снаряд - определяет параметры скорости полета, зоны поражения, урона и тд.
Примеры оружия
и снарядов.
Расчеты позиции крепления оружия
происходят тут, они
заточены под размер спрайта 128х128 и скейлятся от размера корпуса (body.Scale
). Если визуализировать, то выглядит
как, то так:
Стрельба
Движение и стрельба обрабатывается в GameLoop
вот тут.
- Метод
RotateGun
поворачивает оружие к WeaponTarget
.
- Метод
Fire
определяет надо ли
открыть огонь оружия и если да то создается пули которые начнут свой путь на карте.
Создание пуль
При создании пуль мы
даем пулям копию объекта цели, позицию огня (из каково ствола она будет летать, если оружие
многоствольное) и инициируем поля пули. Результатом будем массив пуль (оружие может стрелять очередью например).
Созданные пули не попадают сразу в мир, а попадают в сторедж "стрелка" (gunner
)
gunner.GetBurstOfShots().AddBullets(weaponSlot.Number, newBullets)
Это связано с тем что пули из ствола могут вылетать не мгновенно и не одновременно из всех стволов. За это отвечает
параметры оружия:
CountFireBullet
- кол-во путь выпускаемых за выстрел.
DelayFollowingFire
- задержка створа перед каждым выстрелом.
Например:
CountFireBullet = 8, DelayFollowingFire = 0
, вылетает 8 пуль без задержки, это дробовик.
CountFireBullet = 2, DelayFollowingFire = 32
, вылетают 2 пули с задержкой в 32 мс, это ракетница с очередью 2
выстрела.
Пули попадают в мир через
метод startAttack.
Перед запуском пули
мы вычисляем
стартовую позицию и углы запуска (включая угол по оси Z) относительно оружия. После
пуля попадает в мир
За очередь пуль которые еще надо выпустить в мир отвечать
объект BurstOfShots
, после запуска всех пуль начинается перезарядка оружия. Перезарядка делится на 2 уровня:
- задержка после выстрела (скорострельность)
Weapon.ReloadTime
- перезарядка оружия
Weapon.ReloadAmmoTime
, если снарядов в магазине больше не осталось.
Баллистика
В игре реализована баллистика в 3д пространстве, формулы используемые в расчетах
полета снаряда (реализация)
. Карта в игре тоже является не плоской, а с высотами (Map.LevelMap
). У физической модели объектов/юнитов высоту
определяет параметр PhysicalModel.Height
.
Точка входа для расчета пути всех снарядов в GameLoop
находится тут
.
Расчеты движения, высоты и коллизии пули
происходит тут
.
Обнаружение коллизий, реакция на коллизию
За обнаружение коллизий отвечает
пакет collisions.
Обнаружение коллизий нужно в основном для:
- обнаружение столкновений при движении
- поиск пути
- обнаружение попадания пуль
Обнаружение столкновений при движении и реакции на него.
Обнаружение коллизии при движении происходит в
методе checkMoveCollision
, проверка происходит для будущей полизиции которую стремится принять объект. Т.к. в игре есть разные модели то
можно выделить несколько типов проверки на коллизию (это проблема, но таков путь).
Юнит - Юнит
- (полигон - полигон);
Юнит - Объект
- (полигон - окружность);
Объект - Объект
- (окружность - окружность);
Юнит
- Граница карты, хард-код по позиции юнита;
Объект
- Граница карты, хард-код по позиции юнита;
Во время столкновения происходит:
Реализация "Обзора" игроков (создание/обновление игровых объектов на стороне клиента)
Обзор поделен на 4 уровня.
- объекты в "прямой видимости".
- объекты на радаре: радар показывает метки в тумане войны, но не сами объекты.
- объекты в памяти: объекты которые игрок видел в прямой видимости, но сейчас они в тумане войны (возможно уже
уничтожены, игрок этого не знает).
- статичные объекты: (земля, горы, дороги и т.д.) игроки их видят всегда.
Обзор насчитывается в GameLoop
методом View,
проверка обзора является самой дорогостоящей операцией в движке т.к. приходится проверять каждый объект с обзоров с
каждым объектом который может скрываться за туманом войны.
В текущей реализации объектом который "видит"
является команда,
которая встраивает
объекты которые, видит в другие объекты (юниты, турели). Но если игра предполагает индивидуальное зрение или какие-то
эффекты например "слепота" то надо будет расширить методы обзора на юниты и обрабатывать каждый отдельно (или делать
гибридную модель), что скажется на производительности (еще можно оптимизировать, но этот путь не для слабых духом).
Как происходит расчет обзора
Базовый алгоритм примерно такой:
- проверяем объект на "видимость"
методом
CheckViewCoordinate
.
- проверяем результат
с тем что смотрящий видел до этого (видел, видел, но объект изменился, не видел, видел только на радаре и тд).
- если произошло изменение состояния, то фиксируем его и отправляем изменение на
клиент. Всего есть 5 событий:
- Создание метки радара
- Удаление метки радара
- Создание объекта
- Обновление объекта
- Удаление объекта
- дополнительно удаляем
все объекты которые видит игрок, но они не попали в цикл обновления (например объект умер).
Алгоритм
для "памяти"
отличается тем что удаление объекта происходит только в том случае если его убили в зоне "прямой
видимости". Или
при отрытии зоны
места, где стоял объект.
ИИ
ИИ - основан на "дереве поведения", представлен как бинарный граф.
За правила поведения отвечает
объект BehaviorRules
type BehaviorRules struct {
Rules []*BehaviorRule `json:"rules"` // правла поведения, в данной реализации в массиве всегда 1 стартовое правило.
Meta *Meta `json:"meta"` // различные мето данные которые могут влияет на поведение и заполнятся/читаться правилами
}
type BehaviorRule struct { // само правило
Action string `json:"action"` // действие которое надо выполнить/проверить.
Meta *Meta `json:"meta"` // у каждого правила тоже есть мета
PassRule *BehaviorRule `json:"access_rule"` // это правло срабатывает в случае успеха
StopRule *BehaviorRule `json:"stop_rule"` // это правило срабатывает в случает провала
}
Правила обрабатываются в контролере ИИ на
ноде. Метод AI
будет рекурсивно вызываться до тех пор, пока аргумент rule не будет nil.
В текущей реализации есть 3 готовых правила: FindHostileInRangeView
, FollowAttackTarget
, Scouting
. Они лежат в
отдельном пакете, и все новые правила
рекумендуется класть именно туда.
Проверки могут быть самые разнообразные начиная анализом боевой ситуации до проверки полный у бота трюм или можно еще
копать. Например, конфигурация простейшего разведчика:
Правила не имеют ограничений и могут выполнять сразу и проверку и действие. Например метод FindHostileInRangeView
найдет врага и бот начнет в него стрелять, но что бы он начал преследовать цель надо выполнить
правило FollowAttackTarget
. То есть бот нашел врага, вернул true
и выполнилось следующие правило rule.PassRule
.
Поиск пути, управление движением ИИ
Алгоритм
В основе алгоритма
лежит А* с последующим
поиском прямых отрезков пути
.
зел. - результат А*, желтая линия - путь который пройдет юнит (всего 2 точки).
Как используется
Юнит запрашивает поиск пути 1 раз в "ИИ-тик" (1 сек)
от сюда, у юнита есть
специальный объект
для кеширования и запросов поиска пути.
Unit.movePath *MovePath
type MovePath struct {
path *[]*coordinate.Coordinate // текущий путь разбитый на части (масив точек которые надо преодолеть).
currentPoint int // текучая часть пути по которой идет юнит path[currentPoint]
followTarget *target.Target // цель куда стремится юнит
needFindPath bool // запрос на перерасчет пути, если bool то в методе констролера ИИ будет вызван `FindPath`
}
Боты создают запросы на поиск пути из методов ИИ например, что бы преследовать врага или идти на
базу(подробно: ИИ). Для игроков этот метод доступен в методах рпс если игра предполагает управление как в rts/tbs
играх.
При коллизии
с другим юнитом или объектом будет запрошен перерасчет пути до текущей цели.
Когда у Юнита есть заполненный обьект пути через GameLoop
в методе движения
вызывается метод
который, заставляет юнита двигаться в нужном направлении.
Этот метод берет текущий участок пути по которому идет юнит point := (*path)[currentPoint]
, проверяет достигнуть ли мы
цели или нет, если достигли то удаляем объект пути, если не достигли вызываем метод который смотрит в какую сторону
необходимо следовать юниту и эмулирует ввод клавиш отвечающий за движение (WASD).