- Генератор ландшафта для minecraft
- Генератор ландшафта в Minecraft. Часть 1
- Воссоздаем Minecraft-подобную генерацию мира на Python
- Определения и ограничения
- Генерация миров
- Границы биомов
- Диаграмма Вороного
- Алгоритм релаксации Ллойда
- Шум Перлина/симплексный шум: зачем он нужен?
- «Правильность» ячеек — размываем границы
- Выбор биомов
- График температуры-осадков
- Карты температуры и осадков
- Выравнивание гистограмм
- Усреднение значений внутри ячеек
- Квантование
- Карта биомов
- Карта высот
- Детализированная карта высот
- Фильтрация карты высот
- Итоговые результаты в 3D
- Реки и озера
- Границы
- Деревья и растительность
- Исходный код
- Заключение
- Источники вдохновения
Генератор ландшафта для minecraft
WorldPainter is an interactive map generator for Minecraft. It allows you to «paint» landscapes using similar tools as a regular paint program. Sculpt and mould the terrain, paint materials, trees, snow and ice, etc. onto it, and much more.
WorldPainter is implemented in Java. If you do not have Java installed, install it first from the links below. It needs at least version 8, or 11 for running JavaScript scripts.
Download the program here:
WorldPainter | Java | ||
---|---|---|---|
OS | Architecture | Download the proper version of Java here, if required. You may already have it, so try it without first. | |
| 64-bit | Download installer (.exe) | 64-bit Java 17 LTS |
32-bit | Download installer (.exe) | 32-bit Java 8 | |
Mac OS X | Intel | Download installer (.dmg) (this file is not «damaged»; if Mac OS X complains, follow these instructions to temporarily disable Gatekeeper) | Intel Java 17 LTS |
Arm (M1/M2) | Arm Java 17 LTS | ||
| Debian (or Ubuntu, etc.) | Download DEB package | |
Red Hat (or Fedora, etc.) | Download RPM package | ||
Other (all versions of Linux or UNIX; can be installed as root or as a regular user) | Download installer (.sh) |
If you have trouble installing the program using one of the installers above, you can download installerless/portable archives here: 64-bit Windows, 32-bit Windows, Mac OS X and UNIX/Linux. Note that these are not recommended and unsupported!
You can check the change log here. For checking the integrity of the above files you can find MD5 checksums here and SHA256 checksums here.
WorldPainter does not contain any virus or malware! If your virus scanner says it does, it is a false positive. Please report it as such to the makers, and use the «unquarantine» or equivalent function of your virus scanner, or disable it temporarily, to install and run WorldPainter. See this page for more information. VirusTotal results for these download links can be found here.
There is not much documentation yet, but the program should be pretty self-explanatory. To play the map, choose Export in the File menu. Just experiment and try everything out. Don’t forget to check the menus, and to see what happens if you right-click instead of left-clicking.
Having said that, the documentation, such as it is (including a FAQ), can be found here.
For official support, go to the official WorldPainter subreddit. There is also an unofficial, fan-run Discord server.
WorldPainter has a merch store! Support the development and be the envy of your friends: www.worldpainter.store
Follow us on !
These installers were created with . Docking framework kindly provided by JIDE Software. For spam filtering we use the free services of MX Guarddog. The source code for WorldPainter is hosted on GitHub.
Генератор ландшафта в Minecraft. Часть 1
В последнем посте в своем блоге Notch рассказал краткую историю создания и основные принципы работы генератора ландшафта в Minecraft. Предлагаю вам перевод данного материала (Осторожно, много терминологии!) .
1) Насколько бесконечен мир?
Во-первых, позвольте разъяснить некоторые подробности насчет «бесконечности» миров. Миры не бесконечны, но и не имеют никаких жестких границ. Просто в игре появляется больше багов, когда вы заходите очень далеко. Ландшафт генерируется, хранится и загружается с диска и отрисовывается кусками по 16*16*128 блоков. У каждого куска имеется значение смещения, которое хранится в виде 32-битного целого числа и может находиться в диапазоне примерно от минус двух миллиардов до плюс двух миллиардов. Если вы зайдете за пределы этого диапазона (а это примерно четверть расстояния от Земли до Солнца), то новые куски начнут перекрывать собой старые. А после того, как вы преодолеете шестнадцатую часть этого расстояния, функции, использующие целые числа для работы с позициями блоков, такие как использование инструментов и поиск путей, начнут странно себя вести.
Это два действительно жестких ограничения.
Большинство других вещей, вроде исходных чисел для генератора ландшафта и положения объектов, используют 64-битные дробные числа и вызывают более хитрые проблемы. Например, на огромных расстояниях скорость передвижения игрока может быть медленнее, чем возле центра мира, из-за ошибок округления (значение позиции огромно, а значение смещения при движении крохотно, поэтому оно может обрезаться). Генератор ландшафта может начать создавать странные конструкции вроде огромных блоков одного материала, но такие вещи давно не замечались, и точно узнать причины их появления не удалось. Еще одна серьезная проблема на больших расстояниях появляется в физическом движке — игрок может случайно провалиться в землю или застрять в стене.
Большинство этих проблем может быть решено сменой системы координат для математических расчетов на локальную, центром которой будет игрок. Тогда все рабочие значения не будут сильно отличаться друг от друга. Нечто подобное уже реализовано для прорисовки — Minecraft использует локальные координаты игрока в пределах блока и смещает блоки относительно игрока, чтобы создать видимость движения. Такой принцип применяется в основном из-за того, что OpenGL использует 32-битные дробные числа для позиций, но, кроме этого, еще и потому, что ошибки округления очень заметны при выводе на экран.
Скорее всего, эти баги не будут исправляться, если только они не станут слишком часто возникать у игроков при игре без читов, что вряд ли случалось у кого-либо и вряд ли когда-нибудь случится. Более того, баги добавляют Дальним Землям загадочности и привлекательности.
2) Ну разве эти ландшафты не восхитительны?
В очень ранней версии Minecraft для придания миру формы была использована карта высот на основе двумерного шума Перлина. Или, если быть точнее, несколько карт высот. Одна для общей высоты, одна для шероховатости ландшафта и одна для мелких деталей. Для каждого столба блоков высота равнялась (общ_высота + (шероховатость*детали))*64+64. Карты общей высоты и шероховатости были гладкими, сильно масштабированными шумами, а детали были более мелкими. У этого метода было замечательное преимущество в скорости, так как нужно было провести всего 16*16*(количество_шумов) расчетов на кусок карты, но его недостатком был скучный ландшафт. В частности из-за невозможности генерировать нависающие над землей выступы.
Так что пришлось перейти на похожую систему, использующую трехмерный шум Перлина. Теперь уже не генерировалась «высота земли». Значение шума было рассмотрено как «плотность», и все блоки с плотностью меньше 0 становились воздухом, а блоки с плотностью больше или равной 0 становились землей. Чтобы нижний слой был твердый, а верхний — нет, к полученному результату прибавлялась высота (смещение относительно уровня моря).
К сожалению, мгновенно появились проблемы производительности и играбельности. Первые — из-за большого количества требуемых расчетов, вторые — из-за отсутствия плоских местностей и гладких холмов. Решение обеих проблем заключалось в понижении разрешения при расчетах (8x масштабирование по горизонталям и 4x по вертикали) и достраивании ландшафта с помощью линейной интерполяции. В игре появились плоскости и холмы, а заодно исчезло большинство парящих в воздухе блоков.
Окончательная формула, которая используется сейчас, немного улучшена (и секретна!). Она медленно развивалась в течение разработки игры, и, кстати, до сих пор использует двумерные карты высот и шумности.
В СЛЕДУЮЩИХ ЧАСТЯХ:
- Биомы!
- Пещеры и крупные детали рельефа!
- Деревья, озера и мелкие детали рельефа!
- Мир Nether!
Воссоздаем Minecraft-подобную генерацию мира на Python
. используя диаграммы Вороного и много шумов Перлина/симплексных шумов
Прим. переводчика: стоит отметить, что непосредственно в Minecraft используются отличные от описанных ниже подходов — игра не использует диаграммы Вороного, а кроме двумерных шумов применяет также и трёхмерные (что, в частности, позволяет генерировать не только карту высот, но и пещеры)
Minecraft, самая продаваемая игра в мире, наиболее известная своими пикселизированными блоками и бесконечными мирами, содержит потрясающий процедурный генератор ландшафта — с пещерами, водоёмами, и даже различными биомами.
Процедурная генерация является важной частью компьютерной графики — она используется в основном в играх и в фильмах. Она помогает создавать случайные структуры, не вызывающие ощущения «машинного» стиля.
Также процедурная генерация играет важную роль в машинном обучении. Она позволяет генерировать такие данные, которые сложно собрать. Обучение моделей машинного обучения требует огромных датасетов, которые может быть затруднительно собирать и подготавливать. Генерацию данных процедурным образом можно легко адаптировать к требуемому типу данных.
В детстве мне нравилось играть в Minecraft, и мне всегда было интересно, как эта игра генерирует бесконечные миры. В данной статье я попытаюсь воссоздать это на Python.
Определения и ограничения
Для начала, определимся с тем, как будет генерироваться наш мир.
Мир является трёхмерным, дискретным (состоящим из блоков единичного размера), ограниченным по оси z в диапазоне от 0 до 255 и неограниченным по осям x и y.
Мир содержит биомы, каждый из которых охватывает большую площадь по горизонтали, и которые определяют характер местности в занимаемом биомом пространстве.
Мир содержит реки, озёра и океаны.
Каждый мир определяется зерном (англ. seed). Одно и то же значение зерна всегда приводит к генерации одного и того же мира.
Генерация миров
Чтобы упростить процесс генерации, мы разделим наш мира на чанки (англ. chunk — ячейка, кусок). Каждый чанк занимает пространство размером 1024×1024×256 блоков.
Каждый чанк генерируется отдельно. Это поможет нам сохранять и загружать мир, а также упростит задачу генерации мира по частям.
Прим. переводчика. В Minecraft также используется разбиение пространства на чанки, однако они имеют значительно меньшие размеры — 16x16x256 блоков (16x16x384 — начиная с версии 1.18)
Границы биомов
Первое, что нам нужно сделать — разделить мир на ячейки в плоскости xy, каждая из которых будет принадлежать определённому биому. Каждой ячейке мы назначим точку, обозначающую её центр.
Диаграмма Вороного
Диаграмма Вороного поможет нам разбить мир на ячейки по заданному множеству точек. Основная идея диаграмм Вороного заключается в том, что каждая точка плоскости принадлежит той ячейке, центр которой находится к ней ближе всего.
Движущаяся точка окрашивается в цвет ближайшей к ней неподвижной точки. Изображение автора.
Мы можем проделать это для каждой точки плоскости xy, чтобы построить диаграмму Вороного для этих трёх точек.
Анимированная диаграмма Вороного для 3 точек. Изображение автора.
Хотя такой подход и работает, он очень медленный, особенно когда количество точек велико.
Анимированная диаграмма для 20 точек. Изображение автора.
В Python модуль scipy.spatial содержит класс Voronoi , который строит диаграммы Вороного более эффективно и предоставляет больше информации о диаграмме.
Диаграмма Вороного, построенная с помощью scipy.spatial. Изображение автора.
Класс Voronoi в scipy.spatial возвращает список вершин, регионов и рёбер, что будет полезно на следующих этапах.
Регион и ребро в диаграмме Вороного. Изображение автора.
Эти дополнительные точки помогают сформировать так называемую тесселяцию Делоне (прим. переводчика: также её часто называют триангуляцией Делоне).
Тесселяция Делоне поверх тесселяции Вороного. Изображение автора.
Алгоритм релаксации Ллойда
Теперь нам необходимо сгенерировать случайные точки, которые станут центрами ячеек.
Если мы используем функцию наподобие random из модуля numpy.random , чтобы сформировать множество точек, и построим по ним диаграмму Вороного, то получим следующий результат:
Диаграмма Вороного для множества случайных точек. Изображение автора.
Как вы могли заметить, некоторые точки оказались слишком близко друг к другу. Это называется кластеризацией. Желательно, чтобы ячейки были распределены более равномерно.
Этот эффект становится более заметен, если мы уменьшим масштаб (или увеличим число точек):
Обратите внимание, как некоторые точки кластеризуются вместе, в то время как другие области остаются пустыми. Изображение автора.
Чтобы решить эту проблему, нам необходимо распределить точки дальше друг от друга.
Одним из способов решения этой проблемы является алгоритм релаксации Ллойда, который использует диаграмму Вороного, построенную по исходным точкам.
Идея алгоритма Ллойда заключается в том, чтобы построить диаграмму Вороного, а затем переместить каждую точку в центроид (геометрический центр) соответствующей ей ячейки. Данный процесс может быть повторён несколько раз.
Центроидом многоугольника является среднее от всех его вершин.
Вот диаграмма Вороного, в которой исходные точки отмечены синим цветом, а центроиды ячеек — красным.
Диаграмма Вороного с исходными точками (синие) и центроидами ячеек (красные). Изображение автора.
Далее мы можем заменить исходные точки (синие) центроидами (красными), и повторять этот процесс снова и снова.
Анимация алгоритма релаксации Ллойда. Изображение автора.
Это позволяет получить более привлекательное распределение случайных точек.
Шум Перлина/симплексный шум: зачем он нужен?
Чтобы построить случайный ландшафт, нам необходимо сгенерировать свойства, которые будут меняться случайным образом в пространстве — такие свойства, как высота, температура или количество осадков.
Первое, что приходит на ум — воспользоваться функцией random , и это логично.
Попробуем сгенерировать случайное число в диапазоне от 0 до 255, для каждого блока в плоскости xy в нашем мире.
Это приведёт нас к следующему результату:
Слишком случайно. Изображение автора.
Что ж, это больше похоже на QR-код, нежели на мир в Minecraft.
Проблема заключается в том, что наши случайные величины не имеют под собой связной структуры. Каждое значение генерируется независимо и не имеет ничего общего с соседними.
Чтобы решить эту проблему, применим шум Перлина.
Шум Перлина. Изображение автора.
Шум Перлина был изобретён Кеном Перлином в 1983 году. В отличие от обычного случайного шума, он обладает внутренней структурой. Он больше похож на случайные паттерны, встречающиеся в природе (облака, распределение лесов).
Симплексный шум был также создан самим Кеном Перлином. Он имеет много преимуществ по сравнению с шумом Перлина. В наши дни шум Перлина и симплексный шум повсеместно используются в процедурной генерации.
Мы будем использовать реализацию симплексного шума на Python из модуля noise .
Нам доступно 4 переменных, с которыми можно поиграть: scale (масштаб), octaves (число октав), persistence (персистентность), lacunarity (лакунарность). Я не буду объяснять, за что отвечает каждая из них, но предоставлю вам следующие гифки, которые я сделал, чтобы самому разобраться в этом.
Масштаб
Число октав
Персистентность
Лакунарность
Шум Перлина с различными параметрами. Изображения автора.
Возвращаемые значения шума лежат в диапазоне от -1 до 1.
«Правильность» ячеек — размываем границы
Несмотря на то, что сгенерированные для ячеек точки неплохо разнесены друг от друга, сами ячейки выглядят почти как правильные многоугольники.
Для устранения этого недостатка будем использовать шум Перлина. Для каждой точки мы выберем случайную точку поблизости и присвоим текущей точке вновь выбранную.
Для этого нам понадобится две карты шума, одна — для смещений по оси x, другая для оси y.
Мы можем управлять зашумлённостью границ, умножая значение шума (в диапазоне от -1 до 1) на некоторую константную длину.
Анимация изменения величины шума границ. Изображение автора.
Анимация изменения числа октав шума границ. Изображение автора.
Выбор биомов
В Minecraft представлено более 60 различных биомов, каждый со своими различными свойствами. Теперь, когда наш мир разбит на ячейки, нам необходимо присвоить биом каждой из них. Для этой цели мы будем использовать шум Перлина.
График температуры-осадков
Определим биомы, основываясь на двух параметрах: температуре и количестве осадков, используя график температуры-осадков. Так биомы обычно определяются в биологии окружающей среды.
«Влияние климата на наземные биомы», автор Navarras, Public Domain, CC0.
Вдохновляясь этой диаграммой, спроектируем наш собственный график температуры-осадков.
График температуры-осадков. Изображение автора.
Карты температуры и осадков
Теперь назначим каждой ячейке температуру и количество осадков, используя шум Перлина. Сгенерируем две карты, каждая из которых содержит значения шума для всех блоков в нашем чанке.
Карты температур и количества осадков. Изображение автора.
Выравнивание гистограмм
Если мы будем использовать приведённые выше карты температур и количества осадков, мы столкнёмся с проблемой. Значения, основанные на шуме Перлина, распределены не равномерно. Среди них больше значений, близких к 0, нежели значений, близких к -1 или 1. Это снижает число биомов, которые находятся на краях графика температур-осадков.
Чтобы лучше понять эту неравномерность, я построил одномерную гистограмму и потрясающую двумерную гистограмму карт температуры и количества осадков.
Одномерная (слева) и двумерная (справа) гистограммы карт температур и осадков. Изображение автора.
Как можно видеть, значения по краям дискриминированы. Чтобы устранить этот недостаток, выровняем наши значения.
Выравнивание гистограммы используется, чтобы скорректировать экспозицию изображений. По этой причине я использовал функцию exposure из модуля skimage .
Выровненная гистограмма становится плоской:
Выровненная одномерная (слева) и двумерная (справа) гистограммы карт температур и осадков. Изображение автора.
Поскольку мы выравнивали температуру и количество осадков независимо друг от друга, двумерная гистограмма не является полностью плоской.
Возможно, мы не хотим полностью сглаживать наши гистограммы. Чтобы иметь возможность управлять степенью сглаженности гистограмм, мы можем смешивать невыровненные значения с выровненными в некоторой пропорции.
Анимация выравнивания гистограмм. Изображение автора.
Теперь мы можем управлять тем, насколько выровнены наши значения.
Усреднение значений внутри ячеек
Теперь каждая ячейка содержит значения температуры и количества осадков в диапазоне от -1 до 1.
Квантование
Чтобы упростить работу со значениями температуры и количества осадков, преобразуем их в целые числа. Будем использовать np.uint8 в качестве типа данных для хранения этих значений.
Чтобы преобразовать значения карт, представленных выше, отобразим их на интервал [0, 255], и округлим к ближайшему целому.
Квантование не меняет то, как выглядит температура и количество осадков.
Теперь мы можем определить наш график температуры-осадков в виде изображения размером 256×256 пикселей:
График температуры-осадков. Изображение автора.
Карта биомов
Мы можем назначить каждой ячейке биом, используя график температуры-осадков, карту температур и карту осадков. Выполнив это для каждой ячейки, мы получим следующий результат:
Карта, окрашенная в цвета биомов. Изображение автора.
Карта высот
Каждая точка нашего двумерного мира имеет высоту. Чтобы построить карту высот, мы будем использовать карту шума.
Карта высот. Изображение автора.
Используя эту карту высот (со значениями от -1 до 1), мы можем построить маску суши. Значения выше 0 становятся сушей, а ниже 0 — морем.
Маска суши. Изображение автора.
Совместим эту маску с изображением выше:
Карта биомов с применённой маской суши. Изображение автора.
Чтобы визуализировать высоту, добавим затенение на карту:
Карта биомов с маской суши (слева), затенённая карта биомов с маской суши (справа). Изображение автора.
Пока что результаты выглядят многообещающе. Но сейчас высота не зависит от биома. Нам следует менять карту высот внутри каждого биома. Чтобы достичь этой цели, мы будем применять некоторую функцию к карте высот.
Детализированная карта высот
Будем теперь использовать 2 карты высот с различной степенью детализации. Для этого будем использовать различное число октав в шуме Перлина.
Вот две наших карты высот:
Резкая (слева) и размытая (справа) карты высот. Изображение автора.
Фильтрация карты высот
Мы будем работать с картой высот на суше (значения высоты в диапазоне от 0 до 1). Каждый биом будет использовать комбинацию двух карт высот (размытой и резкой) — и затем применять фильтр (функцию) к результату.
Идея применения фильтров основывается на инструменте Кривые (Curves) в Photoshop. Используем кубические кривые Безье, чтобы определить функцию, которую мы применим к карте высот.
Вот некоторые примеры фильтров:
Пустыня (прим. переводчика: синий график справа — исходный срез высот, оранжевый — после применения фильтра)
Горы (прим. переводчика: синий график справа — исходный срез высот, оранжевый — после применения фильтра)
Создадим и настроим фильтр для каждого биома.
Пустыня, саванна и тропический лес. Изображение автора.
Тундра, сезонный лес и дождевой лес. Изображение автора.
Умеренный лес, умеренный дождевой лес и бореальный лес. Изображение автора.
Чтобы применить эти фильтры к нашей карте высот, мы будем использовать маски. Маска это карта, содержащая 1 в областях, принадлежащих некоторому биому, и 0 в остальных областях.
Маска биома. Изображение автора.
Если мы будем использовать жёсткую маску, у нас будут огромные перепады высот (на границах биомов — прим. переводчика). Поэтому применим размытие к маскам перед использованием. Также удалим океан с этой карты (умножив её на маску суши), чтобы применить фильтры только к суше.
Размытая маска биома (слева), только суша (справа). Изображение автора.
Применим фильтры каждого биома к карте высот, пользуясь масками выше. Получим следующие результаты:
До и после
Итоговые результаты в 3D
Используя Blender, мы можем отрендерить эти карты в 3D. Используем карту высот в модификаторе «Displace» в Blender.
Рендер карты высот с окрашиванием в зависимости от биома. Сделано при помощи Blender. Изображение автора.
Рендер карты высот с окрашиванием в зависимости от биома. Сделано при помощи Blender. Изображение автора.
Рендер карты высот с окрашиванием в зависимости от биома. Сделано при помощи Blender. Изображение автора.
Реки и озера
Границы
Добавим реки на границах между биомами. Во-первых, нам понадобится вычислить границы между биомами.
Для этой цели мы переберём все точки на карте. Если у некоторой точки все её соседи принадлежат тому же биому, то она не лежит на границе. Если среди её соседей больше одного типа биома, то она является частью границы.
Иллюстрация пограничного пикселя. Изображение автора.
Пример границы. Изображение автора.
Применяя этот подход к нашей карте биомов, получим следующие результаты.
Карта биомов (слева) и рек (справа). Изображение автора.
Можно управлять шириной рек, изменяя размер окрестности, содержащей соседние точки.
Реки различной ширины. Изображение автора.
Будем использовать 2 различных разновидности рек: биомные реки и ячеечные реки. Биомные реки широкие и размещаются на границах биомов, в то время как ячеечные реки меньше и размещаются на границах ячеек. Затем используем маску суши, чтобы ограничить реки сушей.
Реки также будут ограничены лишь средней и низкой высотой над уровнем моря.
Реки. Изображение автора.
Прим. переводчика. Честно говоря, на мой взгляд, тут у автора получилось что-то очень странное в отношении рек.
Используем эту маску рек чтобы изменить карту высот. Размоем эту маску рек, а затем применим исходную маску рек к полученному результату. Таким образом получится карта с большими значениями в серединах рек, которые плавно уменьшаются к берегам.
Вот сравнение маски рек с размытой и маскированной маской рек.
Маска рек. Изображение автора.
Боковое сечение маски рек. Изображение автора.
Используем эту карту, чтобы «прорезать» реки в карте высот.
Карта биомов (слева) и карта биомов с реками (справа). Изображение автора.
Деревья и растительность
Чтобы добавить деревья на нашу карту, используем алгоритм релаксации Ллойда, описанный ранее. Этот метод сэмплирования помогает генерировать случайные точки, разнесённые друг от друга.
Прим. переводчика: для получения равномерно разнесённых точек также неплохо подходит алгоритм сэмплирования диском Пуассона (Poisson Disc Sampling)
Будем создавать множества деревьев различной плотности в зависимости от биома.
Различные уровни плотности деревьев. Изображение автора.
Скомбинируем множество деревьев с масками биомов и маской суши, чтобы заполнить биомы деревьями. Каждый биом имеет различную плотность и, разумеется, различные типы деревьев.
Карта биомов с деревьями. Изображение автора.
Мои навыки владения Блендером не позволяют мне визуализировать карту с деревьями в 3D 🙁
Исходный код
Вот ссылка на Jupyter-ноутбук, содержащий все описанные в статье шаги в виде кода.
Предупреждение: Код очень запутанный и незадокументированный.
Заключение
Процедурная генерация является мощным инструментом в компьютерной графике. Она позволяет сгенерированному контенту выглядеть случайным, но в то же время художественным и структурированным. Как было сказано ранее, это можно использовать в машинном обучении для создания датасетов, покрывающих те области, которые дорого или затруднительно покрыть с помощью данных, собранных в реальном мире.
Эта статья была лишь забавным проектом, над которым я хотел поработать уже больше года. В нём ещё многого не хватает. Например, мне нужно создавать пещеры под землёй, деревни, а также придумать алгоритм, способный бесшовно соединять соседние чанки.
Источники вдохновения
Я вдохновлялся многими статьями, когда писал мою. Если вам понравилась эта статья, то вам определённо захочется прочитать и эти тоже: