Вам нравятся старые Legend of Zelda времён SNES и GBA? Может быть, вам пришлась по вкусу Dark Souls? А, возможно, вы ещё и фанат Quake?

Но что объединяет все эти игры? Для меня это в первую очередь дизайн уровней. Головоломки, удобные шорткаты и нелинейность исследования - вот то, что делает карту игры частью общего игрового процесса и вдыхает жизнь в процесс исследования мира.

В наше время расцвета жанра rogue-lite вопрос генерации игровых уровней актуален как никогда. Однако по-настоящему интересные уровни в жанре - большая редкость, я бы даже сказал, феноменальная. Чаще всего уровни представляют собой просто наборы заранее заготовленных комнат-коробок, случайным образом приставленных друг к другу, без какой-либо логичной высокоуровневой картины, с малой связностью или же, напротив, хаотичной повсеместной связностью комнат. Именно по этой причине множество игр с процедурной генерацией смещают внимание игрока с игровых уровней в сторону каких-либо других геймплейных элементов. Фактически, левелдизайн как таковой остаётся весьма нехитрым - есть уровни, и ладно. 

Но, всё же, я знаю одну игру, которая взяла принципиально другой подход: Unexplored. На мой взгляд, она пересмотрела устоявшийся стереотип об ограничениях левелдизайна в рогаликах. Всё, что для этого понадобилось - циклическая генерация подземелий (Cyclic dungeon generation).

К сожалению, описаний этого алгоритма в Сети мною было найдено всего лишь два, одно из которых - статья автора игры Джориса Дорманса (Joris Dormans) [1], а другая - углублённый пересказ этой статьи с прискорбно малым количеством технических деталей, которые ещё и ссылаются на отсутствующие в свободном доступе инструменты [2]. Обе из статей весьма поверхностны и описывают алгоритм слишком высокоуровнево для адекватного воспроизведения. Открытых инструментов, использующих циклическую генерацию, нет вообще, как нет и сколько-нибудь большого количества игр, использующих её.

Но мне слишком уж хотелось уметь так же, и я так и не успокоился, пока окончательно не развеял для себя завесу магии перед внутренним механизмом работы лучшего, на мой взгляд, генератора подземелий.

Предлагаю и вам рассмотреть нюансы реализации генератора процедурных зельдо-подобных уровней, и обратить внимание на встающие на этом пути технические сложности, коих оказывается не так уж мало.

Что в этом подходе такого особенного для игрока?

Здесь и далее для простоты нашего подхода будем иметь в виду такие игры, вне зависимости от их жанра (но с уклоном в поджанр adventure), где игрок управляет только одним персонажем. Также мы не рассматриваем игры с открытым миром и “песочницы”, так как их левелдизайн преследует иные цели и находится за рамками нашего обсуждения.

Будем рассматривать уровень сферической игры в вакууме как абстрактный граф, вершинами которого являются комнаты или, в общем смысле, моменты с, собственно, геймплеем игры; а рёбрами - принципиальная возможность попасть из комнаты в комнату, как то: коридор, дверь, или даже перемещающая игрока катсцена. Рассмотрение уровня как графа не только преследует цель абстракции, но и несёт чисто технический смысл в работе самого генератора.

Абстрактный пример предельно линейного левелдизайна, которому соответствует большинство “рельсовых шутеров” наподобие Call of Duty.  Приведён для наглядности и не более того.
Абстрактный пример предельно линейного левелдизайна, которому соответствует большинство “рельсовых шутеров” наподобие Call of Duty.  Приведён для наглядности и не более того.

Отличный пример уровней в общей канве подавляющего большинства рогаликов (включая классические) можно видеть в Binding of Isaac.

Уровень из Binding of Isaac и его упрощённый граф.
Уровень из Binding of Isaac и его упрощённый граф.

Это прямо-таки энциклопедический пример древовидного уровня. Для своей игры он отлично справляется с задачей, да и в реализации такой подход наиболее прост, но назвать это полноценным образцом левелдизайна adventure-игры, как по мне, нельзя. 

Какие проблемы можно увидеть, если рассматривать уровни с подобной структурой без привязки к конкретной игре?

  1. Бэктрекинг, то есть необходимость несколько раз проходить через одни и те же комнаты. Для уровней, построенных по древовидной модели, особенно характерны ситуации, когда игроку приходится плестись через половину карты - уже зачищенную, пустую и скучную - по нескольку раз, после чего для завершения уровня игрок должен тащиться обратно к боссу, теперь уже через весь уровень, ещё более пустой и скучный. Хотя сам по себе бэктрекинг не обязательно является недостатком, в древовидных уровнях трудно дозировать его количество.

  2. Трудности с логичным размещением интерактивных элементов на уровне. Если бы мы хотели запереть некоторые двери ключами, перед нами встанет проблема осмысленного расположения этих самых ключей - их сбор не должен заставить игрока слишком долго бродить по карте, чтобы не усугубить предыдущую проблему. В идеале игрок должен находить ключи просто “по пути” к финишу уровня, а не тратить лишнее время на поиск того, куда генератор карт этот ключ поставил.

    Отсюда же проистекает схожая проблема - трудности с организацией “межкомнатного взаимодействия” на уровне. Если мы хотим, чтобы в какой-либо комнате был некий триггер, изменяющий совсем другую комнату, то как-либо интересно разместить этот триггер случайным образом непросто.

  3. Крайне малая концентрация player agency в прохождении карты. Отсутствие возможности сделать геймплейно-ориентированные разветвления на картах, когда выбор “идти налево или направо” означает выбор того, как именно мы играем. Фактически, большинство игроков постараются проходить карту, посетив по возможности все комнаты (если речь не про спидран, конечно). Даже если игра даёт возможность проходить уровень несколькими разными маршрутами, смысла в них как таковых мало. За вычетом геймплейных нюансов игроку выгоднее всего проходить уровень в порядке сокровище - магазин - особая комната - испытание - босс, и чаще всего к такому порядку вдумчивый игрок и будет стремиться. Иными словами, даже если вариантов прохождения много, правильным из них является только один. В этом смысле уровень мог бы с тем же успехом быть линейным, чем тратил бы намного меньше времени игрока на брождение по уже исследованным частям.

  4. Недостаток элементов риска-награды в самом левелдизайне, когда какое-либо испытание открывает впоследствии новую часть карты. Здесь же можно упомянуть недостаток “нарративных” элементов, наподобие демонстрации игроку награды в самом начале уровня, но невозможности эту награду сразу взять. В рамках процедурной генерации трудно сделать такие моменты одновременно осмысленными и сбалансированными, но в то же время непредсказуемыми.

Чаще всего рогалики попросту игнорируют эти проблемы. Конечно, разработчики игр лучше меня в курсе этой ситуации, и иногда пытаются их если не решить, то хотя бы обойти. Я набросал сравнительную таблицу решения каждого из вышеуказанных пунктов в случае левелдизайна рогалика и левелдизайна рукотворной игры.

Номер проблемы

Типичное решение в рогаликах

Почему в играх с рукотворными картами эта проблема отсутствует

1

Добавление системы быстрого перемещения в (почти) любую уже посещённую комнату

Частичный или полный отказ от древовидности, наличие сокращающих бэктрекинг возвратов/шорткатов

2

Использование неспецифических ключей (ключ открывает любую запертую комнату, а не какую-то конкретную)

Расположение шорткатов к нужной двери неподалёку от самого ключа; выдача ключа прямо рядом с дверью при выполнении какого-то условия

3

Нерешаемая в рамках наиболее популярных алгоритмов проблема. Иногда встречается жестко заданная структура с ветвлениями (что делает уровень в меньшей степени случайным).

Уровни могут быть спроектированы с расчётом на множество осмысленных возможных путей прохождения (налево пойдёшь - лавовое озеро проходить будешь, направо пойдёшь - страшного врага встретишь).

4

Испытания в тупиковых ответвлениях (чтобы игрок мог их пропустить).

Вставки целиком рукотворного контента.

Другой вариант - да и пусть потенциально непроходимое испытание отсекает от игрока часть контента, это же рогалик, игроку и должно быть сложно.

Проблема предотвращена ещё на этапе решения пункта 3. 

Так вот, особенность циклической процедурной генерации подземелий в том, что она во всех пунктах тяготеет к правому столбцу таблицы, чем подчас создаёт ощущение рукотворности уровня.

Пример генерации уровней из Unexplored (взято из [1])
Пример генерации уровней из Unexplored (взято из [1])

Не правда ли, такой подход звучит интереснее?

Идея генератора

Генератор, бесспорно, хорош, и, пожалуй, действительно наиболее близок к тому, как подземелья проектирует человек. Но как именно это реализовано? 

Итак, на самом базовом уровне речь про генератор карт, расставляющий комнаты в соответствии с неким графом связности. Мы могли бы сделать генератор, случайно размещающий заранее заданный человеком граф-паттерн в виде связанных комнат и остановиться на этом, но тогда мы создадим проблему количества контента - сколько бы в реиграбельной игре ни было создано рукотворного контента, его всегда недостаточно. Для создания сколько-нибудь большого разнообразия в игре нам понадобились бы как минимум многие десятки рукотворных паттернов, а это сотни человеко-часов работы и, в конечном итоге, отсутствие настоящей уникальности в результатах. Небольшая рандомизация результатов (например, добавление случайных комнат) лишь разбавляет хороший рукотворный контент и снижает, если можно так выразиться, “концентрацию левелдизайна”, и не может являться ключом к истинному процедурному разнообразию.

На самом деле, циклический генератор подземелий работает ещё интереснее. Он не включает в себя готовые рукотворные графы уровней. Совсем! Сами эти графы как таковые генерируются на лету по определённым наборам правил.

По сути, этот генератор является частным случаем техники подстановки для графа (graph replacement), а название метода отсылает к тому факту, что правила построения уровней в нём акцентируются на создании в графе циклов (последовательности вершин, начинающихся и заканчивающихся в той же самой вершине, причём каждые две последовательные вершины в последовательности смежны). Польза от циклов с точки зрения левелдизайна заключается в том, что всякий цикл предоставляет игроку два варианта прохождения одного и того же участка карты, и к тому же предоставляет возможность вернуться в уже посещённое место по маршруту, который игрок ещё не исследовал.

Итак, мы начинаем с некоторого простого графа, содержащего в себе цикл, и случайно применяем к нему некоторые правила, потенциально добавляющие новые циклы.

Начальное состояние графа
Начальное состояние графа
Простые примеры правил подстановки
Простые примеры правил подстановки

Эти правила работают с любыми двумя смежными вершинами А и Б, заменяя их подграф на другой, более сложный подграф.

Пример применения обоих правил к рёбрам изначального графа.
Пример применения обоих правил к рёбрам изначального графа.

Пример, конечно, очень простой, но при наличии достаточного количества простых правил можно получить весьма сложные по организации уровни с большим количеством вариантов прохождения и минимальным бэктрекингом. Правила можно применять неоднократно, в том числе к результатам предыдущих шагов, порождая таким образом некий, если можно так выразиться, эмерджентный левелдизайн. Нетрудно видеть, что с повышением количества рукотворных правил потенциальное разнообразие генерируемых карт растёт экспоненциально.

Сложности реализации

Звучит достаточно просто, верно? Но, как говорится, дьявол кроется в деталях. Давайте посмотрим на список требований ко внутренней работе такого алгоритма.

Итак:

  1. Граф уровня случайно генерируется (не задаётся дизайнером заранее);

  2. Граф уровня должен быть планарным (то есть, его должно быть возможно изобразить на плоскости без пересечения рёбер);

  3. Граф уровня должен помещаться в квадратной сетке - в нашем случае ведь используется тайловая карта;

  4. Граф уровня должен помещаться в достаточно маленьком пространстве - если рассматривать сетку комнат, а не тайлов (а генератор работает с комнатами), то нам не нужны сотни комнат на уровень - пары десятков вполне достаточно (к примеру, типичный размер уровня в той же Unexplored - 5x5 комнат);

  5. Сетка, на которой расположены вершины, подразумевает, что смежные вершины графа являются смежными ещё и в геометрическом смысле слова;

  6. Граф уровня должен, в общем случае, достаточно хорошо использовать это пространство, не оставляя половину уровня пустой;

  7. Алгоритм генерации должен работать за адекватное время.

Все эти пункты выглядят трудно сочетаемыми, и вдвойне сложно это сделать из-за того, что оригинальное описание алгоритма вообще никак не заостряется на математике, лежащей между графом и картой самой игры. Алгоритм, если судить по описанию, будто бы работает только с абстрактным графом, ведь только такие и приведены на картинках, а между этим графом и будущим layout-ом подземелья лежит какая-то магия. Попробуем заполнить этот пробел.

Главные проблемы, с которыми мы сталкиваемся, следующие: проблема планарности графа и проблема размещения графа на плоскости. Существует множество алгоритмов проверки планарности заданного графа. Также существует множество алгоритмов рисования графа на плоскости. Но мне не удалось найти ни одного примера алгоритма, рисующего граф не просто на плоскости, но и в заранее заданной сетке определённых, причём достаточно малых, размеров. Проблема в том, что не всякий планарный граф как таковой вообще возможно разместить именно на сетке в силу требования геометрической близости смежных вершин. Даже если такой алгоритм существует, он не решил бы ещё одну фундаментальную проблему, о которой ниже.

Наивный подход даст нам приблизительно следующий алгоритм.

  1. Начинаем с неким базовым графом;

  2. Делаем один шаг подстановки на этом графе;

  3. Проверяем, планарен ли результат;

  4. Проверяем, разместим ли он на сетке;

  5. Если уровень дорос до необходимых размеров, завершаем генерацию, иначе - GOTO 2.

Подход по ряду причин нельзя назвать приемлемым. Что мы будем делать, если пункты 2 или 3 не выполняются? Мы можем:

  • Начать генерацию заново, либо откатиться на один или несколько шагов назад, получив проблему времени работы алгоритма;

  • Завершить работу, как только граф более нельзя наращивать без нарушения планарности - чем, скорее всего, получим куцый уровень без нормальной плотности контента.

В идеале, нам требуется такой генератор, который вообще в принципе способен выдавать только планарные графы. И если эта проблема, скорее всего, не так уж и велика (у меня есть эмпирическое наблюдение, гласящее, что для любого планарного графа замена любого подграфа планарным подграфом даст в результате планарный же граф, но поля статьи слишком малы для его доказательства), то остальные нюансы всё ещё требуют рассмотрения.

Что же всё-таки сработает?

Итак, наша самая большая проблема - геометрическая (расположение вершин на сетке). Возможно, стоит от этого и отталкиваться? 

В сердце алгоритма лежит метод подстановки. Если правила подстановки будут включать в себя геометрические данные, то ещё на этапе применения правил станет понятно, когда граф в эту сетку всё ещё помещается. Что, если мы не будем начинать с абстрактного графа, а начнём сразу же с этой сетки? Другими словами, проще начать с предельно большого (в рамках заданной сетки) графа с максимально возможной связностью и последующей его модификации. 

Итак, начнём с такого графа. Для простоты (и чтобы результат был читаем) рассмотрим граф-сетку размером в 4х4 вершины:

Исходный граф, состояние до начала генерации. Для представления такой структуры подходит обычный двухмерный массив
Исходный граф, состояние до начала генерации. Для представления такой структуры подходит обычный двухмерный массив

Показанные рёбра графа - все, возможные на такой структуре при соблюдении правила геометрического соседства смежных узлов.

В реальном случае ни вершин, ни стрелок нет, они все на начальном этапе выключены.

Также, для решения проблемы размещения графа, мы жертвуем толикой абстракции и наделяем граф пространственной информацией. Каждой вершине взаимно однозначно соответствует конкретная координата в пространстве комнат результирующего уровня. Каждая вершина в нашем случае должна обладать двумя свойствами - активность вершины (то есть, будет ли в этом месте размещена комната) и тэгом (какие-то метаданные, важные для левелдизайна, а не математики. Пример тэга - “тут лежит ключ”). Рёбра должны обладать аналогичными свойствами - плюс, они должны быть направленными (это позволит нам в рамках любого подграфа понять направление, по которому игрок идёт со стороны старта, и не поставить тем самым ключ позади запертой этим же ключом двери). В начале генерации наш граф “пуст” - все вершины и рёбра выключены. 

В итоге, эту структуру можно рассмотреть ещё и так: со старта мы имеем пустую сетку 4х4 клетки, в каждой из которых могут быть вершины графа. Везде, где ниже упоминается термин “граф”, имеется в виду именно такая изоморфная настоящему графу структура.

Для применения метода подстановки все наши имеющиеся правила должны отталкиваться от такой структуры, и включать в себя относительное положение узлов. Получается, метод постановки для графа превращается в метод подстановки для двухмерного массива, где каждый элемент - структура, включающая в себя вершину графа и, возможно, два из её ребер с их метаданными.

Пример такого правила:

Это правило добавит испытание для игрока и в то же время удлинит маршрут, заполняя место на карте
Это правило добавит испытание для игрока и в то же время удлинит маршрут, заполняя место на карте

Обратите внимание на “квадратность” расположения вершин. Правила, в данном контексте, учитывают сетку, ведь относительное расположение вершин для нас важно. 

Сформулируем его человекочитаемым образом:

  1. Берём две смежные активные вершины А и Б, причём такие, что их ребро направлено из А в Б, и что под ними есть две неактивные вершины.

  2. Убираем ребро А - Б.

  3. Активируем две вершины ниже А и Б, на одну ставим тэг “тревога”, на другую - “спящий монстр”.

  4. Активируем рёбра А - Тревога, Тревога - Спящий монстр, Спящий монстр - Б.

Теперь попробуем применить алгоритм подстановки на графе к такой структуре данных.

Начнём с пустой структуры (в этот раз все неактивные рёбра не будут показаны), заполненной по некому рукотворному начальному правилу. Пусть для примера это будет “два пути от старта до финиша”. Применим к этому состоянию указанное выше правило подстановки.

Единственное место, куда это правило “влезет” - вершина слева от финиша и сам финиш. Возьмём их как А и Б соответственно.

Изначальное состояние уровня. Первый цикл со стартом и финишем задан вручную как один из вариантов изначального состояния
Изначальное состояние уровня. Первый цикл со стартом и финишем задан вручную как один из вариантов изначального состояния
Применяем вышеописанное правило
Применяем вышеописанное правило

Обратите внимание: в этом примере хорошо видно, почему мы используем именно направленный граф. Так мы можем быть уверены, что генератор расположит комнаты в верном порядке, и что игрок сначала попадёт в комнату с тревожной кнопкой, а не к спящему монстру.

Что ж, в целом, вроде бы, работает. Но что, если мы применим это же правило ещё раз?..

Ой. А ведь некуда. Единственные подходящие на роль А и Б вершины - вновь добавленные “тревога” и “спящий монстр”, но ниже них заканчивается карта. А ведь хотелось бы, чтобы наши правила также могли “поворачиваться” и, например, растить уровень “вверх”, если там есть место.

Получается, задаваемые нами правила не должны нести в себе никакой абсолютной пространственной информации - то есть, быть аффинно индифферентными. Это означает, что, в идеале, единожды описанное правило должно сразу же допускать любую комбинацию поворотов и отражений, и не ломать при этом направления стрелок внутри своего подграфа. С учётом того, что в общем случае правило может быть применимо к любому количеству узлов, задачка оказалась непростая, и её решению можно было бы посвятить отдельную статью. Я же, в итоге, решил её через довольно сложный рекурсивный алгоритм перебора вершин с явно экспоненциальной сложностью и остался недоволен. Но хотя бы работает.

Примеры работы генератора, или немного ASCII

Мой proof-of-concept написан на Go, и рисует результаты генерации в консоль, поэтому примеры работы будет проще для восприятия перерисовать как граф.

Пример работы генератора. Холст, масло, ASCII, Konsole.
Пример работы генератора. Холст, масло, ASCII, Konsole.

Ниже - в виде графа для наглядности. Имейте в виду, что направления стрелок не принуждают игрока к определённому направлению движения, а лишь указывают на ожидаемое (в рамках генерации) направление движения игрока. Пустые комнаты не обязательно будут пусты в самой игре - это лишь контейнер для более простого случайного контента.

Тот же пример, в более читаемом виде
Тот же пример, в более читаемом виде

Для прохождения обязателен ключ 2 и любой из ключей 1 или 3, поэтому игрок может пройти уровень множеством способов. Игрок может отправиться за ключом 1, но попадёт в ловушку сразу после взятия ключа. С другой стороны, можно пройти по большому циклу и собрать ключ 3 из двух половин, что сразу откроет дорогу к боссу и финишу уровня. Также немаловажно, что у игрока есть возможность пройти уровень, почти не возвращаясь в уже посещённые комнаты, тем самым количество обязательного бэктрекинга минимально.

Вот другой пример.

Игрок может взять сокровище, взяв ключ 1 и пройдя через босса. Также можно обойти босса двумя способами - либо найдя секретный проход у старта (нужен ключ 1), либо найдя секретный проход внизу карты, причём там ключ не нужен. Как секреты, так и боссы опциональны.

Нерешённые проблемы

На данный момент у меня есть базовая минимальная система аффинно-индифферентной подстановки для графов и базовая грамматика для нее. Дальнейшее разнообразие этой реализации теперь зависит только от набора правил (на данный момент существует всего около 15 правил, не считая их незначительных вариаций). Подбор хороших правил, сочетаемых в интересных комбинациях, это самая увлекательная часть разработки циклического генератора карт.

Однако, моя реализация алгоритма циклической генерации не лишена серьёзных недостатков, которые мешают просто взять и использовать её в какой-либо игре.

Во-первых, в генераторе пока ещё не реализована процедура преобразования графа комнат в тайловую карту. Хотя такая процедура не является частью циклической генерации как таковой, а играет роль скорее завершающего шага, в общем случае генератор для игр пока всё ещё не применим. Его пока можно использовать «как есть» только для игр с “покомнатным” геймплеем, а-ля тот же Binding of Isaac.

Во-вторых, генератор плохо кастомизируем, то есть использует слишком много hardcoded параметров, и ещё только предстоит понять, как именно стоит регулировать генерацию через параметры, помимо очевидного размера карты и желаемого количества применённых правил. Есть смысл также ограничивать количество циклов в итоговой карте, так как слишком большое их количество будет восприниматься игроком как хаос, а не как выверенный дизайн.

В-третьих, и это наиболее сложный момент, мне еще предстоит найти способ сделать правила подстановки определяемыми вне кода генератора. В идеале это должен быть какой-либо достаточно простой описательный язык, и реализация такого вполне может оказаться сложнее, чем написание самого циклического генератора. Возможно, у хабровчан есть идеи по этому поводу?

Заключение

Конечно, не всякая игра со случайными картами нуждается в по-настоящему интересных уровнях. Нередко геймплей строится вокруг элементов, не имеющих непосредственного отношения к окружению игрока - например, в серии Diablo не так важны карты, как важны шмотки, а в той же Binding of Isaac игра строится вокруг безумной комбинаторики подбираемых предметов, а не вокруг зельдо-подобного геймплея с уровнями-головоломками. 

Я не знаю, почему циклическая генерация не столь популярна, как она того заслуживает. Если дело в малом количестве подробных описаний работы метода, то, надеюсь, данная статья внесёт свой, пусть и небольшой, но всё же вклад в пользу широты арсенала доступных разработчикам методов процедурной генерации.

Как бы то ни было, циклическая генерация вполне способна добавить глубины и разнообразия в любую игру, претендующую на жанр приключенческой, и помочь ей сделать ещё один шаг в сторону разрушения стереотипа “рогалики это однообразно”. Давайте использовать её почаще?

Источники

  1. https://habr.com/ru/articles/468957/

  2. https://www.boristhebrave.com/2021/04/10/dungeon-generation-in-unexplored/

Комментарии (10)


  1. Mingun
    05.05.2024 19:35
    +2

    Вообще, изначальное требование планарности графа ненужное, и даже вредное, так как уменьшает количество неожиданных решений в лабиринте. Если ребра пересекаются -- значит у вас должны быть проходы в 3-м измерении, либо это проходы-телепортации.

    Бектрекинг тоже странным образом записан в недостатки, когда наоборот -- обоснованное возвращение в уже исследованные комнаты это очень хорошо (естественно, в них должен появляться новый контент или становится доступным то, что не было доступным ранее, то с точки зрения теории, в общем-то, одно и то же). Собственно, такой возврат и есть исследование, а не "рельсы".

    Ну и странное пренебрежение к генераторам, которые наращивают контент вокруг вручную сгенерированных кусочков. На мой взгляд, это то, что обязательно должно быть в генераторе, так как какой-бы генератор не был классный, идеи геймдизайнера он никогда не сгенерирует, а скорее всего, даже мешать будет. И сколько бы правил у вас не было, если эти правила не могут пересекаться друг с другом, то мне кажется, шаблон быстро станет заметен. Например, в вашем примере, игрок один раз увидит, что после тревоги спящий монстр, второй раз, а на третий он уже в курсе.


    1. sidav Автор
      05.05.2024 19:35
      +1

      Благодарю за критику, но позвольте не согласиться. Требование планарности графа выглядит скорее нужным, чем нет. Без этого как мы узнаем, влезет ли наш уровень на сетку? Если не обращать на это внимания, проблема появится на этапе расположения комнат. При отсутствии планарности (с учётом телепортов) у нас появится очень уж много вариантов такого расположения, и таким образом нам потребуется вдобавок какая-либо эвристика, которая поможет размещать комнаты так, чтобы телепортов было не слишком много (ведь разные варианты расположения одного и того же графа потребуют разного их количества). Если, просто для наглядности, доведём ситуацию до крайности, то получим просто пачку геометрически несвязанных участков с кучей телепортов. Это не будет выглядеть рукотворно, это не будет выглядеть особенно логично, и, более того, по уровню с таким количеством телепортов будет трудно перемещаться игроку. Планарность помогает избежать этого заранее.

      На самом деле она не строго обязательна - телепорты можно добавить в рамках отдельных правил (то есть, преднамеренно, ещё при проектировании правил, заранее зная, где она понадобится - например, в конце длинных коридоров, или как шорткат обратно - и по совместительству с дозированием их количества), но такой нюанс остался за рамками статьи, хотя и попал в примеры генерации. К тому же, для сеттинга самой игры наличие телепортов может быть нежелательно. С упомянутой вами трёхмерностью полностью согласен, там это более чем уместно (да и требования к графу в трёх измерениях не такие строгие), но это уже следующий этап развития метода.

      По поводу бэктрекинга: возможно, это моя проблема с формулировкой. Недостатком является трудность с "дозированием" бэктрекинга в рамках древовидных генераторов, а не его наличие как таковое. Плюс, бэктрекинг бэктрекингу рознь - ведь есть разница между шорткатом а-ля Dark Souls и банальным хождением из тупика по в точности тому же маршруту, по которому мы пришли в этот тупик. "Обоснованное возвращение" в рамках циклической генерации более обоснованно, чем в древовидных уровнях - ведь у вас появляется возможность вернуться в уже посещённую комнату более чем одним маршрутом.

      ...если эти правила не могут пересекаться друг с другом, то мне кажется,
      шаблон быстро станет заметен. Например, в вашем примере, игрок один раз
      увидит, что после тревоги спящий монстр, второй раз, а на третий он уже в курсе.
      - в том и суть метода подстановки, что правила многократно применимы, в том числе к уже изменённым частям карты. Другими словами, подстановка на графах используется именно ради того, чтобы множество как можно более простых (в терминах левелдизайна) правил могли пересекаться друг с другом. В вашем примере, после комнаты-тревоги, но до спящего монстра, вполне может быть вставлен подграф с поиском ключа или вообще какая-нибудь развилка. Если правил достаточно много, вариантов их сочетаний будет столько, что вряд ли игрок привыкнет к ним всем. Примерно это я назвал "эмерджентным левелдизайном" за неимением лучшего термина.


      1. sunnybear
        05.05.2024 19:35
        +1

        Помню, в Kyrandia 3 были карты с кое-как связанными локациями. Но найти правила перехода из локации А в локацию Б было особенно интересно


    1. Fen1kz
      05.05.2024 19:35
      +2

      Например, в вашем примере, игрок один раз увидит, что после тревоги спящий монстр, второй раз, а на третий он уже в курсе.

      Это же вроде как считается эталоном хорошего геймдизайна? Та же ситуация с тем, когда кладут гору аптечек и патронов перед боссом - игрок видит эту гору и уже готов сражаться.
      И одна из проблем процедурной генерации - такое там очень трудно сделать. (окей, конкретно навалить патронов в предбанник с боссом просто :) но я про общую ситуцию "решение => проблема") Особенно если чутка разнести их в простанстве - всё, сейчас большинство игр с процедурной генерацией такое не тянут.


  1. Fen1kz
    05.05.2024 19:35
    +1

    Ха, вот это поворот, у меня точно такие же идеи и я занимаюсь этим же)

    Пока что умею вот так:

    граф в draw.io

    ST - старт

    FN - финиш

    X1, X2 - специальные ноды для перекрестков

    1,2,3,4,51,52,53 - простые корридоры

    процесс раскладки

    Несколько отличий от вашей версии:

    • Я все же хочу иметь "графы игровых ситуаций" не привязанными к плоскости

    • Расположение графа на плоскости я делаю через сильно модифицированный WFC

    • Для решения проблемы планарности графа я придумал "опциональные ноды" (ноды 51, 52 и 53) - это пустые ноды послабления для алгоритма раскладки, он их может выкладывать не все. Например граф из дроу.ио разложить на сетку геометрически невозможно, поэтому алгоритму разрешается выкинуть х оранжевых нод

    Из штук которые хотел бы обсудить:

    • Как решите закосы под метроидванию, когда один ключ открывает несколько дверей в разных, не связанных частях карты? Очень хочется иметь такую генерацию.

    • Не думали, что такая модель противоречит типичной игровой ситуации как "монстр в комнате с ключом"? Потому что по духу генерации и духу статьи, мы должны сделать такой граф: (монстр)-(ключ), что как бы подразумевает что монстр и ключ в разных комнатах, а это нетипично? Я почему-то представляю что после победы над монстром, игрок заходит в комнату с ключом и там посереди пустого места на пьедестале лежит ключ, и это мне видится... глупо!? (это не критика, это скорее вопрос который я задаю себе и своему генератору тоже)

    • Больной для анексплоред вопрос с тем, что делать если игрок очутился за запертой дверью. Считаю в оригинальной игре это решено довольно топорно

    • Нет ли идеи отказаться от планарности совсем? Сделать комнаты по типу Биндинг Оф Исаак, но все равно выигрывать в интересе за счет продвинутой генерации игровых ситуаций?

    Ну то есть если мы делаем граф (тревога)-(спящий монстр), который создает "объективно интересную" игровую ситуацию - в целом нам не нужна планарность, монстру без проблем придет из другой комнаты даже если между ними нет непрерывного пространства. Игрокам, конечно, без евклидова пространства будет тяжело, но, в целом, возможно это и к лучшему.

    Я вот пока придерживаюсь планарности, но считаю, что чтобы оправдать планарность и непрерывность уровня, необходимо её использовать в игровом процессе больше чем "ну просто чел идет и между комнатами нет черного экрана".

    ---

    Если интересно, постараюсь в относительно скором времени реализовать ваш граф


    1. sidav Автор
      05.05.2024 19:35

      Благодарю за интерес к статье!

      Подскажите, а в вашем варианте сам граф (до размещения) случайный, или же задан целиком рукотворно?

      Расположение графа на плоскости я делаю через сильно модифицированный WFC

      Вот это очень интересно! Такого подхода я ещё не встречал.

      Как решите закосы под метроидванию, когда один ключ открывает несколько дверей в разных, не связанных частях карты? Очень хочется иметь такую генерацию.

      В целом, это сейчас уже решено. Есть отдельные правила, запирающие дверь - и для дверей с общим ключом правила такие же, как для обычной запертой двери, с одним только нюансом: такой многоразовый "мастер-ключ" ставится только тогда, когда его ещё нет на карте. Если вы хотите иметь несколько, но разных мастер-ключей на одном уровне, есть правила "добавить запертую дверь для ключа, который на карте уже есть".

      Не думали, что такая модель противоречит типичной игровой ситуации как "монстр в комнате с ключом"?

      Думаю, не противоречит. Разрешения ситуации тут два, можно выбрать любой или оба сразу:

      1. У комнаты (узла) не обязательно только один тэг, их может быть сколько угодно. Правила подстановки могут поставить сразу же не только ключ, но и какого-нибудь охраняющего его босса в ту же комнату - достаточно выставить два тэга в рамках одного правила;

      2. Тэги ведь будет использовать другой генератор - генератор самих комнат (следующий этап после создания графа). На этом этапе можно подобавлять монстров или ловушек в любые комнаты, где есть ключ или сокровище, или наоборот, пропускать этап их добавления на отдельных тэгах вроде спящего монстра.

      Пункт 1 уже есть в моём варианте, но это осталось за рамками статьи, на примере одного тэга на комнату демонстрация получается нагляднее.

      Больной для анексплоред вопрос с тем, что делать если игрок очутился за запертой дверью. Считаю в оригинальной игре это решено довольно топорно

      На данном этапе заметил, что наличие подобной проблемы связано с набором правил, а не с самим алгоритмом. Такие ситуации порождают либо одноразовые проходы, либо "самозапирающиеся" двери. Текущий набор правил, какой есть у меня, таких ситуаций не порождает - что ж, возможно, всё ещё впереди.

      Мне кажется, можно избежать такой проблемы, если запретить правилам взаимодействовать с узлами графа, где есть подобные "временны́е" тэги.

      Нет ли идеи отказаться от планарности совсем? Сделать комнаты по типу Биндинг Оф Исаак, но все равно выигрывать в интересе за счет продвинутой генерации игровых ситуаций?

      Возможно, у нас с вами разное понимание слова "планарность" в этом контексте. В терминах графов, в айзеке все уровни без исключения (ну... по крайней мере все те, которые я сам видел) планарны - сам факт того, что у вас в игре рисуется полная мини-карта в 2Д, это подразумевает. В рамках статьи рёбра графа обозначают не комнаты, а переходы между ними. В терминах айзека стрелка на графе - дверь, а не комната. А двери там нигде и никогда ведь не пересекаются.

      Если добавить телепорты в определённые правила, можно сделать ещё более связный (и уже не факт, что планарный) уровень, но как я писал к комментарии выше, слишком большое их количество затрудняет навигацию по уровню. Парочка (или больше, в зависимости от размера желаемого уровня) телепортов, конечно, ничего не испортит.

      Если же вы имеете в виду не планарность, а непрерывность уровня (то есть, все комнаты существуют как бы одновременно, без перехода наподобие загрузки между ними), то да, она в общем-то не обязательна. Моё субъективное мнение в том, что с непрерывными уровнями интереснее (те же Unexplored или Dead Cells выдают из-за этого более интересные для меня уровни, чем айзек).

      чтобы оправдать <...> непрерывность уровня, необходимо её
      использовать в игровом процессе больше чем "ну просто чел идет и между
      комнатами нет черного экрана".

      Тут полностью согласен. Именно алгоритму циклической генерации всё равно, непрерывен уровень или нет, поэтому стрелки можно рассматривать как двери в айзеке, и хоть сейчас генерировать для него уровни этим методом.


      1. Fen1kz
        05.05.2024 19:35

        граф задан руками, но по факту это чутка разросшееся "правило" обозначающее пустой цикл. Т.к. снаружи он представляет из себя коридор, то его можно подставить в один из внутренних коридоров


  1. Fen1kz
    05.05.2024 19:35

    Кстати отлично раскладывает ваш граф в фиксированной области 4х4:

    Hidden text

    Правда я пока не умею в теги и в целом это дебаг-отрисовка раскладчика, но радует малое количество итераций.
    Теперь осталось только реализовать подстановку правил так, чтобы они создавали "возможные для раскладки" правила. Думаю, если не получится, то уйду в третье измерение и буду делать "ноды" друг под другом


    1. sidav Автор
      05.05.2024 19:35

      Интересно, благодарю!
      Да, самой сложной частью при таком способе раскладки будет узнать, когда граф всё ещё этой самой раскладке поддаётся.


  1. sidav Автор
    05.05.2024 19:35

    (комментарий удалён, случайно ответил не в ту ветку)