Hi, at first I tried adding the custom attributes using the Parameter Editor, which felt like the most userfriendly way to add custom attributes. However, a parameter added this way didn't trigger an update when changed. This is what was happening:

You can tell that this goes against the interactivity everyone is used to with modifiers in max. So I made my own custom attributes where I added a call to notifyDependents whenever the parameter changes:
MXS_CA = attributes "MXS_Params" attribID:#(0x79177be, 0x53ba62b7)
(
parameters main rollout:params
(
numIters type:#integer ui:spnIters
on numIters set val do notifyDependents this partIDmsg:#select
)
rollout params "MAXScript Parameters"
(
spinner spnIters "Iterations: " type:#integer range:[0,100,1]
)
)
custAttributes.add $.modifiers[Data_Channel] MXS_CA
It looks the same way custom attributes generated by the Parameter Editor would look but there's the on numIters set line in addition. What it does is whenever the value of the iterations changes, it triggers an update of the selection of the dependents of the attributes (and since they are added to the Data Channel modifier, it gets that notification).
As for the script itself, here we go:
on process theNode theMesh elementType outputType outputArray do
(
-- the function to get expanded selection, given the polyMesh, current selection and number of iterations
fn growVertSel obj sel iterations =
(
for i = 1 to iterations do
sel = polyop.getVertsUsingFace obj (polyop.getFacesUsingVert obj sel)
return sel
)
if theMesh == undefined then return 0 -- no polyMesh to work on, nothing to do here
local numVerts = polyop.getNumVerts theMesh
if numVerts == 0 then return 0 -- no vertices to work on, nothing to do here
-- we want the DataChannel, since the attributes are extending it, let's get the dependents of the PolyMesh and find one of them of the right type
local dcm = for ref in refs.dependents theMesh where isKindOf ref DataChannelModifier do exit with ref
-- I've saved the vertex selection to VData channel 1 with the 1st Vertex Input-Output pair, let's retrieve it (for some reason, you have to add one so we're getting VData channel 2 here)
-- it's just ones and zeroes, so I'm testing each value if it's bigger than one and if it is, collecting that vertex as selected
local vertSel = for v = 1 to numVerts where polyop.getVDataValue theMesh 2 v > 0 collect v
-- now that we have the selection, get the expanded selection bitarray (basically a list of true/false values for each vertex in the mesh)
local grownSel = growVertSel theMesh (vertSel as bitArray) dcm.numIters
outputArray.count = numVerts -- these will be the returned vertex values, initialize it to their count
-- for each vertex number, check if its value in the bitarray is set to true and collect either 1 or 0 selection value based on that
for i = 1 to numVerts do outputArray[i] = if grownSel[i] then 1. else 0.
)