Notifications
Clear all

[Closed] Help please: Want to get name from self reference

Thanks.

So the working code is:

obj = $Ball
timeres = 1f
fn getrot t =
(
 if t<=0f then return quat 0 0 0 1 -- t=0 => no rotation
 t0 = t-timeres -- previous frame time
 t1 = t -- current time
 rot0 = getrot(t0) -- previous rotation:
 p0 = at time t0 obj.position-- previous position
 p1 = at time t1 obj.position-- current position
 if(p0==p1) then return rot0 -- no distance is traveled
 dif = p1-p0 -- difference in positions
 len = Length(dif) -- distance that's traveled
 vec = dif / len -- normalized movement vector.
 r0 = at time t0 obj.radius -- previous radius
 r1 = at time t1 obj.radius -- current radius
 rotax = cross vec [0, 0, 1] -- rotation axis
 angle = 360*len/((r0+r1)*pi)-- rotation amount (in degs)
 rotdif = quat angle rotax -- rotation from t0 to t1
 rot1 = rot0 + rotdif -- total rotation
)
getrot(currentTime)

fn getWheelRotation obj t offset:1f =
(
	rot = quat 1
	if t > 0f do 
	(
		t0 = t - offset -- previous frame time
		t1 = t
		rot = getWheelRotation obj t0 offset:offset -- previous rotation:
		p0 = at time t0 obj.position-- previous position
		p1 = at time t1 obj.position-- current position
		if (p0 != p1) do 
		(
			r0 = at time t0 obj.radius -- previous radius
			r1 = at time t1 obj.radius -- current radius
			vec = normalize (p1 - p0) -- difference in positions
			len = length (p1 - p0) -- difference in positions
			axis = cross vec z_axis -- rotation axis
			angle = radtodeg (len * 2 /(r0 + r1)) -- rotation amount (in degs)
			turn = quat angle axis -- rotation from t0 to t1
			rot += turn -- total rotation
		)
	)
	rot
)

fn doWheelSetup obj = 
(
	obj.objectoffsetrot = eulerangles 90 90 0
	
	c = obj.rotation.controller = rotation_script() 
	c.addnode "obj" obj
	c.addconstant "offset" 1f
	c.setexpression "getWheelRotation obj F offset:offset"	
)


delete objects

w0 = cylinder radius:20 height:4 sides:64 slice:on slicefrom:350 pos:[0,0,20] wirecolor:green
doWheelSetup w0	
w1 = cylinder radius:20 height:4 sides:64 slice:on slicefrom:350 pos:[0,0,-30] wirecolor:orange
doWheelSetup w1

animate on at time 50 
(
	w0.pos = [200,0,40] 
	w0.radius = 40 

	w1.pos = [300,0,-40] 
	w1.radius = 10 
)

this is better (at least it is faster):

delete objects
gc()

w = cylinder radius:20 height:4 sides:64 slice:on slicefrom:350 pos:[0,0,20] wirecolor:green

fn setupWheelControl obj = 
(
	obj.objectoffsetrot = eulerangles 90 90 0
	w.radius.controller = bezier_float()

	cc = obj.rotation.controller = rotation_list()
	c = cc.available.controller = rotation_script()

	c.addconstant "step" 1f
	c.addobject "radius" obj.baseobject[#radius]
	c.addobject "pos" obj.controller[#position]

	ss = @"
	(
		rot = quat 1
		at time 0f
		(
			p0 = pos.value
			r0 = radius.value
		)
		for t = 1f to F by step do at time t
		(
			p1 = pos.value
			r1 = radius.value
			vec = normalize (p1 - p0)
			len = length (p1 - p0)
			axis = cross vec z_axis 
			angle = radtodeg (len * 2 /(r0 + r1))
			turn = quat angle axis
			rot += turn

			p0 = p1
			r0 = r1
		)
		rot
	)
	"

	c.setexpression ss
	c
)

setupWheelControl w

animate on at time 100 
(
	w.pos = [400,0,40] 
	w.radius = 40 
)

Hi, thanks for your input and help, but these scripts create a rolling cylinder… I’m working with a sphere that rolls in all directions on a plane.

Just looking for a way to run my existing script (which does exactly what I need) but I want to be able to drop it into the rotation controller for multiple balls and it work without me having to edit and retype the object name. Gonna be working with hundreds of balls.

my script does exactly what you need. just try to understand how it works

fn setupBallControl obj = 
(
	obj.radius.controller = bezier_float()

	cc = obj.rotation.controller = rotation_list()
	c = cc.available.controller = rotation_script()

	c.addconstant "step" 1f
	c.addobject "radius" obj.baseobject[#radius]
	c.addobject "pos" obj.controller[#position]

	ss = @"
	(
		rot = quat 1
		at time 0f
		(
			p0 = pos.value
			r0 = radius.value
		)
		for t = 1f to F by step do at time t
		(
			p1 = pos.value
			r1 = radius.value
			vec = normalize (p1 - p0)
			len = length (p1 - p0)
			axis = cross vec z_axis 
			angle = radtodeg (len * 2 /(r0 + r1))
			turn = quat angle axis
			rot += turn

			p0 = p1
			r0 = r1
		)
		rot
	)
	"

	c.setexpression ss
	c
)


delete objects

redraw off
(
	plane name:#ground width:400 length:400 wirecolor:green 
	ch = Checker()
	ch.coords.U_Tiling = 2
		
	mat = Standard diffusemap:ch showinviewport:on 

	for k=5 to 20 do 
	(
		at time 0 b = geosphere radius:k pos:(random [-190,-190,k] [190,190,k]) 
		animate on at time (random 25 75) b.pos = random [-190,-190,k] [190,190,k] 
		animate on at time 100 b.pos = random [-190,-190,k] [190,190,k] 
			
		b.mat = mat
		setupBallControl b		
	)
)

Thanks again, but still not working
Using Max 2021 if that makes any difference.

If I put it in the script controller for rotation then I get:

Type error: Call needs function or class, got: undefined.

If I put it in in the rotation scripts of an xform in the stack (useful for later using a point cache), it states illegal self reference and actually hangs max.

Kinda out of ideas now, except the brute force rename in scripts many times.

Any last ditch ideas or suggestions?

Thanks though for all who tried.

Page 2 / 2