Good day, this question is a bit "technical", but how blueprints data structure looks like? I mean their storage format. I'm interested in this as I want to experiment with blueprint editing using external tools (I mean making tools for blueprint editing/creation)
Blueprints Format (Unity)
- paulevs
- Thread is marked as Resolved.
A new update is now available, introducing seasons and more!
Latest hotfix: 0.8.0.2 (2024-12-30)
Latest hotfix: 0.8.0.2 (2024-12-30)
-
-
Basically the blueprint files contain various meta data, followed by the preview image (thumbnail and screenshot, both in JPG format), followed by the actual building data (which is compressed) The first bytes of the blueprint (basically the header) contains information about where the data is located in the buffer.
Blueprint data endianness in the new version is always Little Endian.
The very first byte contains the blueprint version, currently (as of 0.4.5) it is version 6. We use the version for compatibility checks in future updates (possibly the format may slightly change in a future update, so that would be indicated by the version number).
Java blueprints use a different format btw. Since they use the same file ending (".blueprint"), you can check if it's a Java blueprint by having a look at the first three bytes: In the Java version, the whole blueprint data was GZIP compressed, so if the first byte is always "31", the second byte is "139", and the third byte is less or equal to "8", it's most likely a Java blueprint. Java blueprints use Big Endian btw, but that's a different story. Once a Java blueprint is loaded in the new version, it gets converted to the new version format automatically.
The whole data structure of a blueprint in the new version looks like this:
HEADER
Number of bytes Data 1 blueprint version 4 total number of bytes in blueprint data 4 start position of meta data in buffer 4 length of meta data (in bytes) 4 start position of thumbnail image in buffer 4 length of thumbnail data (in bytes) 4 start position of screenshot data in buffer 4 length of screenshot data (in bytes) 4 start position of actual blueprint data in buffer 4 length of actual blueprint data (in bytes) META DATA
Number of bytes Data 1 is legacy blueprint (from Java version)? 8 blueprint creation date (unix timestamp) 4 relative x coordinate of the blueprint center (float) 4 relative y coordinate of the blueprint center(float) 4 relative z coordinate of the blueprint center(float) 4 extent/size of the blueprint along x axis(float) 4 extent/size of the blueprint along y axis(float) 4 extent/size of the blueprint along z axis(float) 2 + xx blueprint name (string*) 2 + xx creator name (string*) 2 + xx creator uid (string*) 2 + xx name of world where blueprint was created (string*) THUMBNAIL DATA
Number of bytes Data 2 thumbnail width (pixels) 2 thumbnail height (pixels) 2 + xx thumnail format (string*), currently always "JPG" 4 number of bytes of thumbnail image data xx thumbnail image data SCREENSHOT DATA
Number of bytes Data 2 screenshot width (pixels) 2 screenshot height (pixels) 2 + xx screenshot format (string*), currently always "JPG" 4 number of bytes of screenshot image data xx screenshot image data BLUEPRINT CONTENT
Number of bytes Data 4 compressed blueprint data length (+ 4 bytes for uncompressed len) 4 uncompressed data length xx LZ4 compressed blueprint data UNCOMPRESSED BLUEPRINT DATA (LZ4)
Number of bytes Data 4 number of construction elements in blueprint xx serialized construction data (see below) 4 number of object elements in blueprint xx serialized object data (see below) 4 number of plants in blueprint xx serialized plant data (see below) 4 terrain array length in blueprint xx serialized terrain data SERIALIZED CONSTRUCTION DATA (per element)
Number of bytes Data 4 relative x position (float) 4 relative y position (float) 4 relative z position (float) 4 rotation x value (quaternion) 4 rotation y value (quaternion) 4 rotation z value (quaternion) 4 rotation w value (quaternion) 4 size along x (float) 4 size along y (float) 4 size along z (float) 1 construction type id 4 texture 4 RGBA color (int) - currently unused 4 RGBA color (int) - currently unused 4 RGBA color (int) - currently unused 4 RGBA color (int) - currently unused 4 RGBA color (int) - currently unused 4 RGBA color (int) 4 surface x offset (float) 4 surface y offset (float) 4 surface z offset (float) 4 surface x scale (float) 4 surface y scale (float) 4 surface z scale (float) 4 optional thickness (float) 4 texture scale factor (float) 4 optional additional flags (int bitmask) SERIALIZED OBJECT DATA (per element)
Number of bytes Data 4 relative x position (float) 4 relative y position (float) 4 relative z position (float) 4 rotation x value (quaternion) 4 rotation y value (quaternion) 4 rotation z value (quaternion) 4 rotation w value (quaternion) 4 size along x (float) 4 size along y (float) 4 size along z (float) 2 object type id 4 variation 4 RGBA color (int) 1 status 8 additional object info (long) 4 optional additional flags (int bitmask) SERIALIZED PLANT DATA (per element)
Number of bytes Data 4 relative x position (float) 4 relative y position (float) 4 relative z position (float) 4 rotation x value (quaternion) 4 rotation y value (quaternion) 4 rotation z value (quaternion) 4 rotation w value (quaternion) 4 size along x (float) 4 size along y (float) 4 size along z (float) 2 plant type id 1 is plant cut? 4 optional additional flags (int bitmask) SERIALIZED TERRAIN DATA
Number of bytes Data 4 terrain array size x 4 terrain array size y 4 terrain array size z 2 * size x * size y * size z terrain data array (short[]) Terrain data is a flattened 3d array consisting of shorts, where the first 8 bits contain the optional terrain "strength" or density between 0 and 100 (used to apply additional smoothing to the terrain) and the last 8 bits contain the actual terrain ID (0 is air).
To get the index in the terrain array from an x, y and z value, you can use this code: x + y * sizeX + z * sizeX * sizeY
I hope I didn't miss anything
* when it comes to strings, the buffer first stores the number of characters as 16 bit int (short), then the individual characters (2 bytes per char). If the string is null, a "-1" is stored instead (again as 16 bit int)
-
Thank you for detailed explanation
-
Looks like I can read files in that format, but there is something that looks wrong. Count of compressed bytes stored in blueprint (LZ4 section) is larger than amount of remaining bytes in loaded data (it is smaller on 4 bytes). I'm not sure why, I don't know, probably this is mistake on my side, but count of loaded bytes is equal to stored count of bytes, so looks like it should be fine... . I decided just to lower my buffer array on 4 bytes (if I'm not reading no-element data this should work fine), and I'm not sure if this will have effect on saving files
Edit: I tested that on 10 blueprints, the difference always exists and is always 4 bytes
-
Sorry for the confusion, the int which holds the compressed buffer length includes the 4 next bytes (the uncompressed data length). The reason is that we need the uncompressed length to decompress the data, so this always gets stored in the compressed data buffer (more precisely, we consider the 4 bytes indicating the uncompressed length being part of the compressed data). I've updated my previous post to make this a bit clearer
So if the blueprint data is 512 bytes, for example, and it would be 128 bytes compressed, the "uncompressed data length" would be "512" (4 bytes), and the "compressed blueprint data length" would be "132" (128 compressed bytes + 4 bytes for the uncompressed length int).
-
FYI: I had made this one a while back ( Blueprint Texture Editor version 0.2 ) but unfortunately haven't got around to updating it as other projects have taken priority for me in the past few years.
It is written in Java like the old version of the game and the full code is available on my github here: https://github.com/Minotorious/BlueprintTextureEditor (GPL 3.0 Licenced)
-
Looks like it works
This is test of fully procedural blueprint with simple primitives.
Addition: looks like "unused byte" section in elements is filled with RGBA value of element, if this section will be not filled - emissive elements will not have color or glow. Filling it with repeating RGBA values fix issue
-
This looks awesome, well done!
Addition: looks like "unused byte" section in elements is filled with RGBA value of element, if this section will be not filled - emissive elements will not have color or glow. Filling it with repeating RGBA values fix issue
The "unused" bytes are basically placeholders for individual colors per face (which might get implemented in the future), so right now they're all filled with the main element color value. However, after taking another look at it, the game currently uses the last of these color values - I've updated my previous post accordingly (sorry again for the confusion)
-
Simple mesh import. I want to calculate correct triangle position and offsets to make solid meshes, but this is a complex task, so here are simple grids with points
-
Love it
-
After dozens experiments and some math - polygon fill. Something is wrong with normals and UVs on my models, I think this can be a result of my "hacky" triangle fill (I set element size to zero on two cardinal axis and manipulate offsets)
-
Small test with free large 3D model
-
Wow. I really miss models in the unity-version.
-
Great work, this is quite impressive!
-
-
-
I've now added the terrain data information to my original post
-
Thank you very much
I have a small note - terrain data stores array capacity between 3 array dimensions and actual terrain data, so most my attempts to write data looked like this before I found the reason:
So, the terrain data looks like this:
4 bytes Int size X 4 bytes Int size Y 4 bytes Int size Z 4 bytes Array Capacity 2 * X * Y * Z bytes Data Also looks like terrain data is stored with [id-strength] order instead of [strength-id] order (which I noticed during test output), probably this is related to stream order
Now the last thing that I need to configure is array index order
-
-
Participate now!
Don’t have an account yet? Create a new account now and be part of our community!