Hi all,
This seems to be an issue lots of people have asked about online but I've yet to find a solution.
I have a bunch of objects that I've dynameshed and decimated in zbrush, and now I need to optimise based on a camera position. i.e. select all faces visible to the camera and delete the rest.
Ignore backfacing isn't sufficient as this doesn't excluse non-backfacing faces that are occluded by the rest of the mesh.
I found this script which apparently works but my knowledge of maxscipt is nil -
http://www.illusioncatalyst.com/mxs_files/getFrontFaces.html
I've tried running it and get errors, and I expect it's to do with the line on that site which says
Required functions: getPolyFromMeshFace() found under: "Editable Poly data from underlying Mesh"
If anyone can shed any light on this for me I'd appreciate it. I'm installing Maya right now as I've read it can do what I'm after out of the box, but it would be incredibly useful to be able to do this in the package I'm familiar with.
Thanks.
Replies
On the surface a lot of the max modeling tools seem similar but they actually call different commands.
Editable poly, is an object.
Edit Poly, is a modifier.
Edit Poly, is a modifier that is instanced across multiple objects.
Editable Mesh, is an object.
Edit Mesh, is a modifier.
Edit Mesh, is a modifier that is instanced across multiple objects.
All of these seem similar but commands like "cut" are called in different ways in each of them, if cut even exists in the tool at all.
For script authors this is a clusterf*ck because one method or command might only be accessible under certain circumstances and not available in other methods, or it might but you have to code 4-5-6 different ways to handle it, then come up with a bunch of checks to find out what kind of object you have and convert it over to what you need and then back to what they had.
Most script authors will just say "use editable poly" since its the newest and most robust and has the least chance of encountering totally random, uncontrollable modifier stack results.
Ran the script and then put "getFrontFaces $ steps:5" (or variations of the steps integer) into the maxscript listener.
It does a reasonable job but is still missing random faces that are obviously visible (ie not just small faces on the objects outline that whatever magic this script uses could possibly miss)
Had to comment out the line with the getPolyFromMeshFace function in it to get it working, so maybe that's why it's failing on some faces.
As I say, I know nothing about maxscript sadly but could really do with getting this to work before the end of the week.
http://docs.autodesk.com/3DSMAX/15/ENU/3ds-Max-Help/index.html?url=files/GUID-694580CD-2CEA-49B6-A5EE-278C50390616.htm,topicNumber=d30e128633
It operates sort of like a slice plane moving through your scene from the viewport. Anything that is behind the plane is selected and the farther you push it into your scene the more it selects.
Personally I like syncviews IC script is a lot better.
the function has gone missing in a website update somewhere.
I've rewritten it on the fly and it should do the job. It is required only if the object is an Editable Poly, the Editable Mesh doesn't call it.
[html]function getPolyFromMeshFace theEditablePoly iMeshFace =
(
local iResultPoly = 0
if ( (Filters.Is_EditPoly()) and (iMeshFace > 0) ) do
(
local theMesh = theEditablePoly.mesh
local baMeshVerts = meshOp.getVertsUsingFace theMesh iMeshFace
local abaPolysFromVert = for iVert in baMeshVerts collect (polyOp.getFacesUsingVert theEditablePoly iVert)
local baPolyFace = abaPolysFromVert[1] * abaPolysFromVert[2] * abaPolysFromVert[3]
if (baPolyFace.numberSet == 1) then
(
iResultPoly = (baPolyFace as Array)[1]
)
else
(
local baMeshFaceInPoly = meshOp.getPolysUsingFace theMesh iMeshFace ignoreVisEdges:false threshhold:179.5
local baMeshFaceVert = meshOp.getVertsUsingFace theMesh baMeshFaceInPoly
local baPolyFaceVert = #{}
for iPoly in baPolyFace while (iResultPoly == 0) do
(
baPolyFaceVert = polyOp.getVertsUsingFace theEditablePoly iPoly
if ( ((baPolyFaceVert - baMeshFaceVert).isEmpty) and ((baMeshFaceVert - baPolyFaceVert).isEmpty) ) do
(
iResultPoly = iPoly
)
)
)
)
return iResultPoly
)[/html]
I've gone quickly through the code to select front faces and I saw it is not 100% accurate mainly for two reasons: it uses a sampling raycast system, which means it can miss faces in between rays, it uses the intersectRayEx function which is known to be not very reliable (should use the RayMeshGridIntersect), plus it is very old code.
You can give it a go with the function I provided, but it might not make a huge difference. Keep in mind it is going to take longer with Editable Polys, because of the additional calculations.
Thanks
I don't suppose there's a way to increase the number of rays that are cast?
Speed hasn't been a problem so far as long as I keep my meshes fairly low poly.
Have you fiddled with these variable at the beginning of the script?
local fDotThresh = 1e-3
local fMatchThresh = 1e-3
local fOffsetThresh = 1e-2
They are meant to set the accuracy of some tests, the lower the better, but don't take it lower than 1e-6.
Can you post a snapshot of the model and the result of the selection? Just to see if I can spot anything clearly wrong.
Thanks
I can show you the basic noisy mesh test I'm doing which would be very easy to fix by hand, but the project I need this for would take a loooong time to sort by hand (sorry, can't show you the meshes I need it for but they're effectively mountain ranges that have been dynameshed/decimated and then broken down into a lot of more manageable objects poly count wise)
test:
EDIT: I should point out that this isn't the best example of what I've been seeing. I've had faces right in the center of the nearest part of the mesh that are being missed, rather than just edge cases.
The faces that appear to be not selected in the farther end of the bent tube are actually occluded by the faces of the part of the tube closer to the camera.
If a ray, cast from a face closer to the camera, hits a face farther from the camera, the latter is not selected. I suspect what you're looking for is a different script, which would select everything that can be seen from the camera and leave unselected everything that is not visible at all from the camera. Is it right?
Really appreciate the help by the way. I wish autodesk would just implement Mayas camera based selection
here is the script you're looking for (hopefully
It is not super tested (it's late). It works on Editable Polys and select all the faces visible from camera in the active viewport. It is a function and a call on the selected node, I think you should be able to run it.
You sir, have absolutely made my day! Thank you so so much.
And for the benefit of anyone who stumbles across this thread, a quick example image:
There is a bug in the code. I assumed the center of the polygons is within the polygon face (as happens for meshes) but it is incorrect. I commented the test of the center of the face for now. I am going to fix it later. The script is still working anyway, just a little less accurate.
Cheers