[Client Beta] In-experience Mesh & Image APIs now available in published experiences

EDIT: EditableImage documentation is incomplete. I was tipped off that only 1 EditableImage can be updated per frame. I had to create a wrapper that copies buffers into appropriate locations for a single EditableImage.

Got a bug regarding EditableImages.

Each EditableImage is created separately, assigned to its own content object, and assigned to its own ImageLabel.

However, despite using asserts to check if an EditableImage is created or not, the engine does not want to cooperate with updating more than 1 EditableImage.

In the video you can see me enabling/disabling each actor’s script.

Both EditableImages are created, assigned to a Content Object, and set to their respective ImageLabels in serial, of which rendering is then done in Parallel before calling task.synchronize() to use WritePixelsBuffer for each EditableImage.

I am looking to avoid using any Parallel Luau cross-thread communication for performance reasons, so I need this to be fixed otherwise my project is dead in the water.

2 Likes

Hello, thanks for the feedback – it’s on our roadmap!

1 Like

Disappointing that such a promising feature has such a fixed limit; 8 editable meshes on the client is a joke- makes it borderline impossible to get anything done

One way to fix this would be to allow editable meshes to be locked- stopping edits, and fixing the memory, and in turn the triangle and vertex information

At the moment, creating editable meshes of fixed sized from existing meshes does not work- as removing the original editable mesh so that it does not count towards the editable mesh memory budget will also wipe the content

Such a promising feature- cant wait for it to be fine tuned, but right now restrictions on memory and mesh count are making this unviable for any real use

3 Likes

At the moment, creating editable meshes of fixed sized from existing meshes does not work- as removing the original editable mesh so that it does not count towards the editable mesh memory budget will also wipe the content

Hi, did you apply/replace the new FixedSize emesh to the original meshPart? Happy to look at your case where the content get wiped out. local newEmesh = CreateEditableMeshAsync(Content.fromObject(editableMesh)) basically cloned a FixedSize version of unbounded emesh, and you would then follow createMeshPartAsync()... to apply it to workspace, for example:

local emesh2 = game.AssetService:CreateEditableMeshAsync(Content.fromObject(emesh))
mp:ApplyMesh(game:GetService("AssetService"):CreateMeshPartAsync(Content.fromObject(emesh2)))
emesh:Destroy()

One way to fix this would be to allow editable meshes to be locked- stopping edits, and fixing the memory, and in turn the triangle and vertex information

Thanks for the feedback and this is on our roadmap!

1 Like

Do you remember your use case for using FindClosestVertex() to find points that were not part of a face?

1 Like

image
I’ve been experimenting with editable meshes for terrain. And I have a few questions about the visuals for them. Most importantly, the very obvious shading difference between two different mesh parts, and if there’s any way to fix that. And secondly, the smooth shading itself.


Preferably I would like the triangles to be distinct like so. Is there any way to do this? (This second image uses wedge parts to generate the exact same terrain mesh, however, it takes 28x longer to render every frame, very big lag issue)

2 Likes

Hi, happy to answer your questions - this might get a bit detailed, so I’ll put a summary at the end.

Yes, the seams between tiles can be fixed, but will require some work. Or if you want to get the bottom image’s faceted style, you can actually skip all of that work.

A bit of background first.

The different types of shading behavior you’re seeing here are due to different normals. A normal is a vector that points away from the surface, that’s used by rendering when shading the faces. In EditableMesh, at each vertex, you can have a normal vector that is automatically computed, or a normal vector that is manually set. The default is automatically computed.

EditableMesh’s faces store 4 ids per corner - a vertex id, a normal id, a uv id, and a color id. When you call :AddTriangle, you only specify the vertex ids. AddTriangle will either create new ids for normal, uv, and color, or if the vertex is already part of a face, it will reuse ids from that vertex on that face.

So, if you were to run some code like:

local fid1 = emesh:AddTriangle(vid1, vid2, vid3)
local fid2 = emesh:AddTriangle(vid2, vid1, vid4)

You’d get something that looks like this:

You can see that the faces f1 and f2 share vertices, v1 and v2. Also, on those vertices, they share the same normal, uv, and color ids.

When a normal id is an automatically generated normal, EditableMesh looks at each face corner that’s using that normal id, computes the normal at that corner (using a cross product), and then averages together all of the normals on all of the face corners.

If you combine those two behaviors, the default behavior is that you get a single averaged normal vector at each vertex, which gives you the smooth look of most of the terrain in your first image.

However, between the tiles, the different EditableMeshes don’t know that they’re supposed to be connected. When computing the automatic normals, there’s not a good way for them to know that the normal vectors should be averaged out between the two EditableMeshes.

You can do this manually if you know which vertex ids on each mesh correspond with each other, something like:

local normal1 = emesh1:GetNormal(nid_on_mesh1)
local normal2 = emesh2:GetNormal(nid_on_mesh2)
local avg = (normal1 + normal2).Unit
emesh1:SetNormal(nid_on_mesh1, avg)
emesh2:SetNormal(nid_on_mesh2, avg)

Please note that this will change the normal id on both meshes from automatically computed to manually set, so if the mesh vertices near the border change local position, you’ll probably want to redo this step.

However, I have some good news about the second image. Not only is this possible, but if you go for this style, you don’t have to bother with making the normals between the terrain tiles match.

The trick to getting that faceted appearance is that you want a single normal per face, instead of averaging the normals on a vertex.

If you run something like:

local fid1 = emesh:AddTriangle(vid1, vid2, vid3)
local fid2 = emesh:AddTriangle(vid2, vid1, vid4)

local nid5 = emesh:AddNormal()
emesh:SetFaceNormals(fid1, {nid5, nid5, nid5})

local nid6 = emesh:AddNormal()
emesh:SetFaceNormals(fid2, {nid6, nid6, nid6})

emesh:RemoveUnused()

You’ll get something like this:

This will give you faceted rendering like in your second image.

So, TL;DR, if what you want is rendering like the bottom image, the easiest way is to use something like:

local function addSharpTriangle(emesh, vid1, vid2, vid3)
    local fid = emesh:AddTriangle(vid1, vid2, vid3)
    local nid = emesh:AddNormal()
    emesh:SetFaceNormals(fid, {nid, nid, nid})
    return fid
end

and then after you’re done making the mesh, call emesh:RemoveUnused(), to clean up unused ids and save a bit of memory.

4 Likes

Although this trick works, there is some sort of invisible hidden memory requirement, for what seems to be budgeting triangles

Im only able to add around 100k triangles worth of meshes, even with the locked mesh trick, which is abyssmal for any use case

Itd be a lot more useful if the memory constraints weren’t based off devices from the early 2000s :sweat_smile: :sweat_smile: :sweat_smile:

Please give us more control over these limits, doesn’t even have to be in the client, but in studio…

I believe there to be a bug with mobile. I’m unsure how to replicate it but occasionally whenever joining on mobile EditableMeshes are not rendering and are invisible.

Is there any way to surpass 8 editable meshes? I have very few vertices in each.

Hi, you can make them fixedSize by using CreateEditableMeshAsync(Content.fromObject(editableMesh)), and then destroy the un-fixed one to get more budget! hope this is helpful:

My meshes are fixed size. My question was how to have more than 8. I need 9. I dont want to make a clone of another i just need 9 unique fixed size meshes

You should be able to generate more than 8 meshes if you are using CreateEditableMeshAsync, it is fixedSize by default, and if you are using CreateEditableMesh, it is not fixed size and is currently expensive for memory estimate which is why you can easily reach the memory limit (~8). To make it fixedSize you can call CreateEditableMeshAsync with your unbounded emesh and then destroy the unbounded one created by CreateEditableMesh to free some budget to create the next. Happy to take a look at your current script to see where might be going wrong!

Why is there some sort of hidden triangle requirement, even doing this method, we reach a limit at around 2-3GB memory usage

1 Like

Hello, thanks for the feedback! What’s the specific use case driving your need for over 100k triangles or 2-3GB memory usage?

Please consider a “read-only” version of the method that has heightened permissions :pray:

(i.e. When you just want to fetch something from the mesh and not alter it)

1 Like

This is something that we considered, but unfortunately there’s not much advantage, because it is easy to copy the contents of an EditableMesh into another EditableMesh. The permissions issues you run into are the same. We’re investigating other ways of fixing the permissions issues to enable more use cases.

1 Like

@monsterjunjun @L3Norm

Not sure if this is intentional, or if I should create a bug report for it. However after deleting a face, it seems to break the entirety of RaycastLocal and results in it only returning nil?

Not intentional, a bug report is appreciated. Thanks for the heads up!